| /** |
| * @file test_autodel.c |
| * @author Radek Krejci <rkrejci@cesnet.cz> |
| * @brief Cmocka tests for auto-delete of choice's case when a node from another case comes. |
| * |
| * Copyright (c) 2016 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 |
| */ |
| |
| #include <stdio.h> |
| #include <stdlib.h> |
| #include <setjmp.h> |
| #include <stdarg.h> |
| #include <cmocka.h> |
| |
| #include "tests/config.h" |
| #include "libyang.h" |
| |
| struct state { |
| struct ly_ctx *ctx; |
| struct lyd_node *dt; |
| struct lyd_node *aux; |
| char *xml; |
| }; |
| static int |
| setup_f(void **state) |
| { |
| struct state *st; |
| const struct lys_module *mod; |
| const char *schemafile = TESTS_DIR"/data/files/autodel.yin"; |
| |
| (*state) = st = calloc(1, sizeof *st); |
| if (!st) { |
| fprintf(stderr, "Memory allocation error"); |
| return -1; |
| } |
| |
| /* libyang context */ |
| st->ctx = ly_ctx_new(NULL, 0); |
| if (!st->ctx) { |
| fprintf(stderr, "Failed to create context.\n"); |
| goto error; |
| } |
| |
| /* schema */ |
| mod = lys_parse_path(st->ctx, schemafile, LYS_IN_YIN); |
| if (!mod) { |
| fprintf(stderr, "Failed to load data model \"%s\".\n", schemafile); |
| goto error; |
| } |
| |
| /* root */ |
| st->dt = lyd_new(NULL, mod, "c"); |
| if (!st->dt) { |
| fprintf(stderr, "Failed to create data root.\n"); |
| goto error; |
| } |
| |
| return 0; |
| |
| error: |
| lyd_free_withsiblings(st->dt); |
| ly_ctx_destroy(st->ctx, NULL); |
| free(st); |
| (*state) = NULL; |
| |
| return -1; |
| } |
| |
| static int |
| teardown_f(void **state) |
| { |
| struct state *st = (*state); |
| |
| lyd_free_withsiblings(st->dt); |
| lyd_free_withsiblings(st->aux); |
| ly_ctx_destroy(st->ctx, NULL); |
| free(st->xml); |
| free(st); |
| (*state) = NULL; |
| |
| return 0; |
| } |
| |
| static void |
| test_atob(void **state) |
| { |
| struct state *st = (*state); |
| struct lyd_node *node; |
| |
| node = lyd_new_leaf(st->dt, NULL, "a", "x"); |
| assert_ptr_not_equal(node, NULL); |
| assert_ptr_equal(st->dt->child, node); |
| assert_int_equal(lyd_validate(&(st->dt), LYD_OPT_CONFIG, NULL), 0); |
| |
| lyd_print_mem(&(st->xml), st->dt, LYD_XML, 0); |
| assert_string_equal(st->xml, "<c xmlns=\"urn:autodel\"><a>x</a></c>"); |
| |
| node = lyd_new(st->dt, NULL, "b1"); |
| assert_ptr_not_equal(node, NULL); |
| assert_int_equal(lyd_validate(&(st->dt), LYD_OPT_CONFIG, NULL), 0); |
| |
| free(st->xml); |
| lyd_print_mem(&(st->xml), st->dt, LYD_XML, 0); |
| assert_string_equal(st->xml, "<c xmlns=\"urn:autodel\"><b1/></c>"); |
| } |
| |
| static void |
| test_btoa(void **state) |
| { |
| struct state *st = (*state); |
| struct lyd_node *node; |
| |
| node = lyd_new(st->dt, NULL, "b1"); |
| assert_ptr_not_equal(node, NULL); |
| node = lyd_new_leaf(st->dt, NULL, "b2", "z"); |
| assert_ptr_not_equal(node, NULL); |
| assert_int_equal(lyd_validate(&(st->dt), LYD_OPT_CONFIG, NULL), 0); |
| |
| lyd_print_mem(&(st->xml), st->dt, LYD_XML, 0); |
| assert_string_equal(st->xml, "<c xmlns=\"urn:autodel\"><b1/><b2>z</b2></c>"); |
| |
| node = lyd_new_leaf(st->dt, NULL, "a", "x"); |
| assert_ptr_not_equal(node, NULL); |
| assert_int_equal(lyd_validate(&(st->dt), LYD_OPT_CONFIG, NULL), 0); |
| |
| free(st->xml); |
| lyd_print_mem(&(st->xml), st->dt, LYD_XML, 0); |
| assert_string_equal(st->xml, "<c xmlns=\"urn:autodel\"><a>x</a></c>"); |
| } |
| |
| static void |
| test_insert_fail(void **state) |
| { |
| struct state *st = (*state); |
| struct lyd_node *node; |
| |
| node = lyd_new_leaf(st->dt, NULL, "a", "x"); |
| assert_ptr_not_equal(node, NULL); |
| assert_ptr_equal(st->dt->child, node); |
| |
| st->aux = lyd_new(NULL, st->dt->schema->module, "c"); |
| assert_ptr_not_equal(st->aux, NULL); |
| node = lyd_new(st->aux, NULL, "b1"); |
| assert_ptr_not_equal(node, NULL); |
| assert_int_equal(lyd_validate(&(st->aux), LYD_OPT_CONFIG, NULL), 0); |
| |
| assert_int_not_equal(lyd_insert_after(st->dt->child, node), 0); |
| assert_string_equal(ly_errmsg(st->ctx), "Insert request refers node (/autodel:c/a) that is going to be auto-deleted."); |
| } |
| |
| static void |
| test_insert_correct(void **state) |
| { |
| struct state *st = (*state); |
| struct lyd_node *node, *node2; |
| |
| node = lyd_new_leaf(st->dt, NULL, "a", "x"); |
| assert_ptr_not_equal(node, NULL); |
| assert_ptr_equal(st->dt->child, node); |
| node2 = lyd_new_leaf(st->dt, NULL, "x", "node"); |
| assert_ptr_not_equal(node, NULL); |
| |
| st->aux = lyd_new(NULL, st->dt->schema->module, "c"); |
| assert_ptr_not_equal(st->aux, NULL); |
| node = lyd_new(st->aux, NULL, "b1"); |
| assert_ptr_not_equal(node, NULL); |
| assert_int_equal(lyd_validate(&(st->aux), LYD_OPT_CONFIG, NULL), 0); |
| |
| assert_int_equal(lyd_insert_after(node2, node), 0); |
| lyd_print_mem(&(st->xml), st->dt, LYD_XML, 0); |
| assert_string_equal(st->xml, "<c xmlns=\"urn:autodel\"><x>node</x><b1/></c>"); |
| assert_int_equal(lyd_validate(&(st->dt), LYD_OPT_CONFIG, NULL), 0); |
| } |
| |
| |
| int main(void) |
| { |
| const struct CMUnitTest tests[] = { |
| cmocka_unit_test_setup_teardown(test_atob, setup_f, teardown_f), |
| cmocka_unit_test_setup_teardown(test_btoa, setup_f, teardown_f), |
| cmocka_unit_test_setup_teardown(test_insert_fail, setup_f, teardown_f), |
| cmocka_unit_test_setup_teardown(test_insert_correct, setup_f, teardown_f), }; |
| |
| return cmocka_run_group_tests(tests, NULL, NULL); |
| } |