blob: ebb835ce4b5344c38bc33eea3f798e2031a6b443 [file] [log] [blame]
/*
* @file set.c
* @author: Radek Krejci <rkrejci@cesnet.cz>
* @brief unit tests for functions from set.c
*
* Copyright (c) 2018 CESNET, z.s.p.o.
*
* This source code is licensed under BSD 3-Clause License (the "License").
* You may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* https://opensource.org/licenses/BSD-3-Clause
*/
#define _UTEST_MAIN_
#include "utests.h"
#include <stdlib.h>
#include <string.h>
#include "config.h"
#include "plugins.h"
#include "plugins_internal.h"
const char *simple = "module libyang-plugins-simple {"
" namespace urn:libyang:tests:plugins:simple;"
" prefix s;"
" typedef note { type string; }"
" extension hint { argument value; }"
" leaf test {"
" type s:note {length 255;}"
" s:hint \"some hint here\";"
" }"
"}";
static void
test_add_invalid(void **state)
{
assert_int_equal(LY_ESYS, lyplg_add(TESTS_BIN "/plugins/plugin_does_not_exist" LYPLG_SUFFIX));
#ifdef __APPLE__
CHECK_LOG("Loading \""TESTS_BIN "/plugins/plugin_does_not_exist" LYPLG_SUFFIX "\" as a plugin failed "
"(dlopen("TESTS_BIN "/plugins/plugin_does_not_exist" LYPLG_SUFFIX ", 2): image not found).", NULL);
#else
CHECK_LOG("Loading \""TESTS_BIN "/plugins/plugin_does_not_exist" LYPLG_SUFFIX "\" as a plugin failed "
"("TESTS_BIN "/plugins/plugin_does_not_exist" LYPLG_SUFFIX ": cannot open shared object file: "
"No such file or directory).", NULL);
#endif
assert_int_equal(LY_EINVAL, lyplg_add(TESTS_BIN "/plugins/plugin_invalid" LYPLG_SUFFIX));
#ifndef __APPLE__
/* OS X prints address of the symbol being searched and cmocka doesn't support wildcards in string checking assert */
CHECK_LOG("Processing user type plugin \""TESTS_BIN "/plugins/plugin_invalid"LYPLG_SUFFIX "\" failed, "
"missing type plugins information ("TESTS_BIN "/plugins/plugin_invalid"LYPLG_SUFFIX ": "
"undefined symbol: plugins_types__).", NULL);
#endif
}
static void
test_add_simple(void **state)
{
const struct lys_module *mod;
struct lysc_node_leaf *leaf;
struct lyplg_ext *plugin_e;
struct lyplg_type *plugin_t;
assert_int_equal(LY_SUCCESS, lyplg_add(TESTS_BIN "/plugins/plugin_simple" LYPLG_SUFFIX));
UTEST_ADD_MODULE(simple, LYS_IN_YANG, NULL, &mod);
leaf = (struct lysc_node_leaf *)mod->compiled->data;
assert_int_equal(LYS_LEAF, leaf->nodetype);
assert_non_null(plugin_t = lyplg_find(LYPLG_TYPE, "libyang-plugins-simple", NULL, "note"));
assert_string_equal("libyang 2 - simple test, version 1", plugin_t->id);
assert_ptr_equal(leaf->type->plugin, plugin_t);
assert_int_equal(1, LY_ARRAY_COUNT(leaf->exts));
assert_non_null(plugin_e = lyplg_find(LYPLG_EXTENSION, "libyang-plugins-simple", NULL, "hint"));
assert_string_equal("libyang 2 - simple test, version 1", plugin_e->id);
assert_ptr_equal(leaf->exts[0].def->plugin, plugin_e);
/* the second loading of the same plugin - still success */
assert_int_equal(LY_SUCCESS, lyplg_add(TESTS_BIN "/plugins/plugin_simple" LYPLG_SUFFIX));
}
static void
test_validation(void **state)
{
const struct lys_module *mod;
struct lyd_node *tree;
const char *data;
const char *schema = "module libyang-plugins-validate {"
" namespace urn:libyang:tests:plugins:validate;"
" prefix v;"
" extension extra-validation;"
" typedef note { type string { v:extra-validation;}}"
" leaf test1 {"
" type v:note;"
" }"
" leaf test2 {"
" type string;"
" v:extra-validation;"
" }"
" leaf test3 {"
" type string {v:extra-validation;}"
" }"
" leaf test4 {"
" type string;"
" }"
"}";
assert_int_equal(LY_SUCCESS, lyplg_add(TESTS_BIN "/plugins/plugin_validate" LYPLG_SUFFIX));
UTEST_ADD_MODULE(schema, LYS_IN_YANG, NULL, &mod);
/* test1 - extra-validation done based on typedef's extension */
data = "<test1 xmlns=\"urn:libyang:tests:plugins:validate\">xxx</test1>";
assert_int_equal(LY_SUCCESS, lyd_parse_data_mem(UTEST_LYCTX, data, LYD_XML, 0, LYD_VALIDATE_PRESENT, &tree));
CHECK_LOG_CTX("Extension plugin \"libyang 2 - validation test, version 1\": extra validation callback invoked on test1", NULL);
lyd_free_all(tree);
/* test2 - extra-validation done based on node's extension */
data = "<test2 xmlns=\"urn:libyang:tests:plugins:validate\">xxx</test2>";
assert_int_equal(LY_SUCCESS, lyd_parse_data_mem(UTEST_LYCTX, data, LYD_XML, 0, LYD_VALIDATE_PRESENT, &tree));
CHECK_LOG_CTX("Extension plugin \"libyang 2 - validation test, version 1\": extra validation callback invoked on test2", NULL);
lyd_free_all(tree);
/* test3 - extra-validation done based on node type's extension */
data = "<test3 xmlns=\"urn:libyang:tests:plugins:validate\">xxx</test3>";
assert_int_equal(LY_SUCCESS, lyd_parse_data_mem(UTEST_LYCTX, data, LYD_XML, 0, LYD_VALIDATE_PRESENT, &tree));
CHECK_LOG_CTX("Extension plugin \"libyang 2 - validation test, version 1\": extra validation callback invoked on test3", NULL);
lyd_free_all(tree);
/* test4 - extra-validation not done */
data = "<test4 xmlns=\"urn:libyang:tests:plugins:validate\">xxx</test4>";
assert_int_equal(LY_SUCCESS, lyd_parse_data_mem(UTEST_LYCTX, data, LYD_XML, 0, LYD_VALIDATE_PRESENT, &tree));
CHECK_LOG_CTX(NULL, NULL);
lyd_free_all(tree);
}
static void
test_not_implemented(void **state)
{
const struct lys_module *mod;
struct lyd_node *tree;
const char *schema = "module libyang-plugins-unknown {"
" namespace urn:libyang:tests:plugins:unknown;"
" prefix u;"
" extension myext;"
" typedef mytype { type string;}"
" leaf test {"
" u:myext;"
" type mytype;"
" }"
"}";
const char *data = "<test xmlns=\"urn:libyang:tests:plugins:unknown\">xxx</test>";
char *printed = NULL;
UTEST_ADD_MODULE(schema, LYS_IN_YANG, NULL, &mod);
assert_int_equal(LY_SUCCESS, lys_print_mem(&printed, mod, LYS_OUT_YANG_COMPILED, 0));
free(printed);
assert_int_equal(LY_SUCCESS, lyd_parse_data_mem(UTEST_LYCTX, data, LYD_XML, 0, LYD_VALIDATE_PRESENT, &tree));
CHECK_LOG_CTX(NULL, NULL);
lyd_free_all(tree);
}
int
main(void)
{
const struct CMUnitTest tests[] = {
UTEST(test_add_invalid),
UTEST(test_add_simple),
UTEST(test_validation),
UTEST(test_not_implemented),
};
return cmocka_run_group_tests(tests, NULL, NULL);
}