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.

199 lines
5.2 KiB

/*
** 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 "zbxcommon.h"
#include "zbxeval.h"
#include "mock_eval.h"
typedef struct
{
const char *name;
int args_num;
zbx_variant_t retval;
const char *error;
}
zbx_mock_callback_t;
zbx_vector_ptr_t callbacks;
static void mock_callback_free(zbx_mock_callback_t *cb)
{
zbx_variant_clear(&cb->retval);
zbx_free(cb);
}
static void mock_read_callbacks(const char *path)
{
zbx_mock_handle_t hcbs, hcb, hdata, hvalue;
zbx_mock_error_t err;
if (ZBX_MOCK_SUCCESS != zbx_mock_parameter(path, &hcbs))
return;
while (ZBX_MOCK_END_OF_VECTOR != (err = (zbx_mock_vector_element(hcbs, &hcb))))
{
zbx_mock_callback_t *cb;
const char *value;
if (ZBX_MOCK_SUCCESS != err)
fail_msg("cannot read callback contents");
cb = (zbx_mock_callback_t *)zbx_malloc(NULL, sizeof(zbx_mock_callback_t));
memset(cb, 0, sizeof(zbx_mock_callback_t));
cb->name = zbx_mock_get_object_member_string(hcb, "name");
cb->args_num = zbx_mock_get_object_member_uint64(hcb, "args_num");
if (ZBX_MOCK_SUCCESS == zbx_mock_object_member(hcb, "retval", &hdata))
{
if (ZBX_MOCK_SUCCESS == zbx_mock_string(hdata, &value))
{
zbx_variant_set_dbl(&cb->retval, atof(value));
}
else
{
zbx_vector_var_t *values;
values = (zbx_vector_var_t *)zbx_malloc(NULL, sizeof(zbx_vector_var_t));
zbx_vector_var_create(values);
while (ZBX_MOCK_END_OF_VECTOR != (err = (zbx_mock_vector_element(hdata, &hvalue))))
{
zbx_variant_t tmp;
if (ZBX_MOCK_SUCCESS != err)
fail_msg("cannot read callback retval contents");
if (ZBX_MOCK_SUCCESS != zbx_mock_string(hvalue, &value))
fail_msg("cannot read callback retval");
zbx_variant_set_str(&tmp, zbx_strdup(NULL, value));
zbx_variant_convert(&tmp, ZBX_VARIANT_DBL);
zbx_vector_var_append(values, tmp);
}
zbx_variant_set_vector(&cb->retval, values);
}
}
else if (ZBX_MOCK_SUCCESS == zbx_mock_object_member(hcb, "error", &hdata))
{
if (ZBX_MOCK_SUCCESS != zbx_mock_string(hdata, &cb->error))
fail_msg("invalid token error");
}
else if (ZBX_MOCK_SUCCESS == zbx_mock_object_member(hcb, "reterr", &hdata))
{
const char *errmsg;
if (ZBX_MOCK_SUCCESS != zbx_mock_string(hdata, &errmsg))
fail_msg("invalid token error");
zbx_variant_set_error(&cb->retval, zbx_strdup(NULL, errmsg));
}
else
fail_msg("invalid token contents");
zbx_vector_ptr_append(&callbacks, cb);
}
}
static int callback_cb(const char *name, size_t len, int args_num, zbx_variant_t *args, void *data,
const zbx_timespec_t *ts, zbx_variant_t *value, char **error)
{
int i;
ZBX_UNUSED(args);
ZBX_UNUSED(data);
ZBX_UNUSED(ts);
for (i = 0; i < callbacks.values_num; i++)
{
zbx_mock_callback_t *cb = (zbx_mock_callback_t *)callbacks.values[i];
if (len == strlen(cb->name) && 0 == memcmp(name, cb->name, len))
{
zbx_mock_assert_int_eq("callback argument count", cb->args_num, args_num);
if (NULL != cb->error)
{
*error = zbx_strdup(*error, cb->error);
return FAIL;
}
zbx_variant_copy(value, &cb->retval);
return SUCCEED;
}
}
*error = zbx_strdup(*error, "unknown callback");
return FAIL;
}
void zbx_mock_test_entry(void **state)
{
zbx_eval_context_t ctx;
char *error = NULL;
zbx_uint64_t rules;
int expected_ret, returned_ret;
zbx_variant_t value;
ZBX_UNUSED(state);
zbx_vector_ptr_create(&callbacks);
rules = mock_eval_read_rules("in.rules");
expected_ret = zbx_mock_str_to_return_code(zbx_mock_get_parameter_string("out.result"));
if (SUCCEED != zbx_eval_parse_expression(&ctx, zbx_mock_get_parameter_string("in.expression"), rules, &error))
{
if (SUCCEED != expected_ret)
goto out;
fail_msg("failed to parse expression: %s", error);
}
mock_eval_read_values(&ctx, "in.replace");
mock_read_callbacks("in.callbacks");
returned_ret = zbx_eval_execute_ext(&ctx, NULL, callback_cb, callback_cb, NULL, &value, &error);
if (SUCCEED != returned_ret)
printf("ERROR: %s\n", error);
zbx_mock_assert_result_eq("return value", expected_ret, returned_ret);
if (SUCCEED == expected_ret)
{
zbx_mock_assert_str_eq("output value", zbx_mock_get_parameter_string("out.value"),
zbx_variant_value_desc(&value));
zbx_variant_clear(&value);
}
else
zbx_free(error);
out:
zbx_free(error);
zbx_vector_ptr_clear_ext(&callbacks, (zbx_clean_func_t)mock_callback_free);
zbx_vector_ptr_destroy(&callbacks);
zbx_eval_clear(&ctx);
}