You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
221 lines
8.1 KiB
221 lines
8.1 KiB
1 year ago
|
/*
|
||
|
** Zabbix
|
||
|
** Copyright (C) 2001-2023 Zabbix SIA
|
||
|
**
|
||
|
** This program is free software; you can redistribute it and/or modify
|
||
|
** it under the terms of the GNU General Public License as published by
|
||
|
** the Free Software Foundation; either version 2 of the License, or
|
||
|
** (at your option) any later version.
|
||
|
**
|
||
|
** This program is distributed in the hope that it will be useful,
|
||
|
** but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||
|
** MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||
|
** GNU General Public License for more details.
|
||
|
**
|
||
|
** You should have received a copy of the GNU General Public License
|
||
|
** along with this program; if not, write to the Free Software
|
||
|
** Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
||
|
**/
|
||
|
|
||
|
#include "zbxmocktest.h"
|
||
|
#include "zbxmockdata.h"
|
||
|
#include "zbxmockassert.h"
|
||
|
#include "zbxmockutil.h"
|
||
|
|
||
|
#include "zbxexpression.h"
|
||
|
#include "zbxcachevalue.h"
|
||
|
#include "mocks/valuecache/valuecache_mock.h"
|
||
|
#include "../../../src/libs/zbxexpression/anomalystl.h"
|
||
|
|
||
|
int __wrap_substitute_simple_macros(zbx_uint64_t *actionid, const zbx_db_event *event, const zbx_db_event *r_event,
|
||
|
zbx_uint64_t *userid, const zbx_uint64_t *hostid, const zbx_dc_host_t *dc_host,
|
||
|
const zbx_dc_item_t *dc_item, zbx_db_alert *alert, const zbx_db_acknowledge *ack,
|
||
|
const zbx_service_alarm_t *service_alarm, const zbx_db_service *service, const char *tz, char **data,
|
||
|
int macro_type, char *error, int maxerrlen);
|
||
|
|
||
|
int __wrap_zbx_dc_get_data_expected_from(zbx_uint64_t itemid, int *seconds);
|
||
|
|
||
|
int __wrap_substitute_simple_macros(zbx_uint64_t *actionid, const zbx_db_event *event, const zbx_db_event *r_event,
|
||
|
zbx_uint64_t *userid, const zbx_uint64_t *hostid, const zbx_dc_host_t *dc_host,
|
||
|
const zbx_dc_item_t *dc_item, zbx_db_alert *alert, const zbx_db_acknowledge *ack,
|
||
|
const zbx_service_alarm_t *service_alarm, const zbx_db_service *service, const char *tz, char **data,
|
||
|
int macro_type, char *error, int maxerrlen)
|
||
|
{
|
||
|
ZBX_UNUSED(actionid);
|
||
|
ZBX_UNUSED(event);
|
||
|
ZBX_UNUSED(r_event);
|
||
|
ZBX_UNUSED(userid);
|
||
|
ZBX_UNUSED(hostid);
|
||
|
ZBX_UNUSED(dc_host);
|
||
|
ZBX_UNUSED(dc_item);
|
||
|
ZBX_UNUSED(alert);
|
||
|
ZBX_UNUSED(ack);
|
||
|
ZBX_UNUSED(tz);
|
||
|
ZBX_UNUSED(data);
|
||
|
ZBX_UNUSED(macro_type);
|
||
|
ZBX_UNUSED(error);
|
||
|
ZBX_UNUSED(maxerrlen);
|
||
|
ZBX_UNUSED(service_alarm);
|
||
|
ZBX_UNUSED(service);
|
||
|
|
||
|
return SUCCEED;
|
||
|
}
|
||
|
|
||
|
int __wrap_zbx_dc_get_data_expected_from(zbx_uint64_t itemid, int *seconds)
|
||
|
{
|
||
|
ZBX_UNUSED(itemid);
|
||
|
*seconds = zbx_vcmock_get_ts().sec - 600;
|
||
|
|
||
|
return SUCCEED;
|
||
|
}
|
||
|
|
||
|
static void zbx_vcmock_history_dump(unsigned char value_type, const zbx_vector_history_record_t *values)
|
||
|
{
|
||
|
int i;
|
||
|
char buffer[256];
|
||
|
|
||
|
for (i = 0; i < values->values_num; i++)
|
||
|
{
|
||
|
const zbx_history_record_t *rec = &values->values[i];
|
||
|
|
||
|
zbx_timespec_to_strtime(&rec->timestamp, buffer, sizeof(buffer));
|
||
|
printf(" - %s\n", buffer);
|
||
|
zbx_history_value2str(buffer, sizeof(buffer), &rec->value, value_type);
|
||
|
printf(" %s\n", buffer);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
static void read_values_stl(zbx_mock_handle_t hdata, unsigned char value_type, zbx_vector_history_record_t *values)
|
||
|
{
|
||
|
zbx_mock_error_t err;
|
||
|
zbx_mock_handle_t hvalue;
|
||
|
zbx_history_record_t rec;
|
||
|
const char *data;
|
||
|
|
||
|
ZBX_UNUSED(value_type);
|
||
|
|
||
|
while (ZBX_MOCK_END_OF_VECTOR != (err = (zbx_mock_vector_element(hdata, &hvalue))))
|
||
|
{
|
||
|
rec.value.dbl = atof(zbx_mock_get_object_member_string(hvalue, "value"));
|
||
|
data = zbx_mock_get_object_member_string(hvalue, "ts");
|
||
|
|
||
|
if (ZBX_MOCK_SUCCESS != (err = zbx_strtime_to_timespec(data, &rec.timestamp)))
|
||
|
fail_msg("Invalid value timestamp \"%s\": %s", data, zbx_mock_error_string(err));
|
||
|
|
||
|
zbx_vector_history_record_append_ptr(values, &rec);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
static void check_records_stl(const char *prefix, unsigned char value_type,
|
||
|
const zbx_vector_history_record_t *expected_values, const zbx_vector_history_record_t *returned_values)
|
||
|
{
|
||
|
int i;
|
||
|
const zbx_history_record_t *expected, *returned;
|
||
|
|
||
|
printf("Expected %s:\n", prefix);
|
||
|
zbx_vcmock_history_dump(value_type, expected_values);
|
||
|
|
||
|
printf("Returned %s:\n", prefix);
|
||
|
zbx_vcmock_history_dump(value_type, returned_values);
|
||
|
|
||
|
zbx_mock_assert_int_eq(prefix, expected_values->values_num, returned_values->values_num);
|
||
|
|
||
|
for (i = 0; i < expected_values->values_num; i++)
|
||
|
{
|
||
|
expected = &expected_values->values[i];
|
||
|
returned = &returned_values->values[i];
|
||
|
|
||
|
zbx_mock_assert_timespec_eq(prefix, &expected->timestamp, &returned->timestamp);
|
||
|
zbx_mock_assert_double_eq(prefix, expected->value.dbl, returned->value.dbl);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void zbx_mock_test_entry(void **state)
|
||
|
{
|
||
|
int err, expected_ret, returned_ret, nvalues = 0;
|
||
|
zbx_uint64_t s_window, season, seconds = 0;
|
||
|
char *error = NULL;
|
||
|
zbx_dc_item_t item;
|
||
|
zbx_vcmock_ds_item_t *ds_item;
|
||
|
zbx_timespec_t ts;
|
||
|
zbx_mock_handle_t handle;
|
||
|
zbx_vector_history_record_t values_in, trend_values_received, trend_values_expected,
|
||
|
seasonal_values_received, seasonal_values_expected, remainder_values_received,
|
||
|
remainder_values_expected;
|
||
|
|
||
|
/*ZBX_DOUBLE_EPSILON = 0.000001; results into output that is different from python test case output */
|
||
|
zbx_update_epsilon_to_python_compatible_precision();
|
||
|
|
||
|
zbx_history_record_vector_create(&values_in);
|
||
|
zbx_history_record_vector_create(&trend_values_received);
|
||
|
zbx_history_record_vector_create(&trend_values_expected);
|
||
|
zbx_history_record_vector_create(&seasonal_values_received);
|
||
|
zbx_history_record_vector_create(&seasonal_values_expected);
|
||
|
zbx_history_record_vector_create(&remainder_values_received);
|
||
|
zbx_history_record_vector_create(&remainder_values_expected);
|
||
|
|
||
|
err = zbx_vc_init(get_zbx_config_value_cache_size(), &error);
|
||
|
zbx_mock_assert_result_eq("Value cache initialization failed", SUCCEED, err);
|
||
|
zbx_vc_enable();
|
||
|
zbx_vcmock_ds_init();
|
||
|
memset(&item, 0, sizeof(zbx_dc_item_t));
|
||
|
ds_item = zbx_vcmock_ds_first_item();
|
||
|
item.itemid = ds_item->itemid;
|
||
|
item.value_type = ds_item->value_type;
|
||
|
|
||
|
season = zbx_mock_get_parameter_uint64("in.season");
|
||
|
s_window = zbx_mock_get_parameter_uint64("in.s_window");
|
||
|
seconds = zbx_mock_get_parameter_uint64("in.seconds");
|
||
|
handle = zbx_mock_get_parameter_handle("in");
|
||
|
zbx_vcmock_set_time(handle, "time");
|
||
|
ts = zbx_vcmock_get_ts();
|
||
|
|
||
|
if (FAIL == zbx_vc_get_values(item.itemid, item.value_type, &values_in, (int)seconds, nvalues, &ts))
|
||
|
{
|
||
|
error = zbx_strdup(error, "cannot get values from value cache");
|
||
|
printf("%s\n", error);
|
||
|
zbx_free(error);
|
||
|
goto out;
|
||
|
}
|
||
|
|
||
|
if (SUCCEED != (returned_ret = zbx_STL(&values_in, (int)season, ROBUST_DEF, (int)s_window, S_DEGREE_DEF,
|
||
|
T_WINDOW_DEF, T_DEGREE_DEF, L_WINDOW_DEF, L_DEGREE_DEF, S_JUMP_DEF, T_JUMP_DEF, L_JUMP_DEF,
|
||
|
INNER_DEF, OUTER_DEF, &trend_values_received, &seasonal_values_received,
|
||
|
&remainder_values_received, &error)))
|
||
|
{
|
||
|
printf("zbx_STL returned error: %s\n", error);
|
||
|
zbx_free(error);
|
||
|
}
|
||
|
|
||
|
expected_ret = zbx_mock_str_to_return_code(zbx_mock_get_parameter_string("out.return"));
|
||
|
zbx_mock_assert_result_eq("return value", expected_ret, returned_ret);
|
||
|
|
||
|
zbx_vc_flush_stats();
|
||
|
|
||
|
if (SUCCEED == expected_ret)
|
||
|
{
|
||
|
read_values_stl(zbx_mock_get_parameter_handle("out.trend"), item.value_type, &trend_values_expected);
|
||
|
read_values_stl(zbx_mock_get_parameter_handle("out.seasonal"), item.value_type,
|
||
|
&seasonal_values_expected);
|
||
|
read_values_stl(zbx_mock_get_parameter_handle("out.remainder"), item.value_type,
|
||
|
&remainder_values_expected);
|
||
|
check_records_stl("trend values", item.value_type, &trend_values_expected, &trend_values_received);
|
||
|
check_records_stl("seasonal values", item.value_type, &seasonal_values_expected,
|
||
|
&seasonal_values_received);
|
||
|
check_records_stl("remainder values", item.value_type, &remainder_values_expected,
|
||
|
&remainder_values_received);
|
||
|
}
|
||
|
out:
|
||
|
zbx_history_record_vector_destroy(&trend_values_received, ITEM_VALUE_TYPE_FLOAT);
|
||
|
zbx_history_record_vector_destroy(&trend_values_expected, ITEM_VALUE_TYPE_FLOAT);
|
||
|
zbx_history_record_vector_destroy(&seasonal_values_received, ITEM_VALUE_TYPE_FLOAT);
|
||
|
zbx_history_record_vector_destroy(&seasonal_values_expected, ITEM_VALUE_TYPE_FLOAT);
|
||
|
zbx_history_record_vector_destroy(&remainder_values_received, ITEM_VALUE_TYPE_FLOAT);
|
||
|
zbx_history_record_vector_destroy(&remainder_values_expected, ITEM_VALUE_TYPE_FLOAT);
|
||
|
zbx_history_record_vector_destroy(&values_in, item.value_type);
|
||
|
|
||
|
zbx_vcmock_ds_destroy();
|
||
|
|
||
|
ZBX_UNUSED(state);
|
||
|
}
|