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
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);
|
|
}
|