| /* |
| * @file test_parser_json.c |
| * @author: Radek Krejci <rkrejci@cesnet.cz> |
| * @brief unit tests for functions from parser_xml.c |
| * |
| * Copyright (c) 2019 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 "context.h" |
| #include "in.h" |
| #include "out.h" |
| #include "parser_data.h" |
| #include "printer_data.h" |
| #include "tests_config.h" |
| #include "tree_data_internal.h" |
| #include "tree_schema.h" |
| |
| static int |
| setup(void **state) |
| { |
| const char *schema = "module a {namespace urn:tests:a;prefix a;yang-version 1.1; import ietf-yang-metadata {prefix md;}" |
| "md:annotation hint { type int8;}" |
| "list l1 { key \"a b c\"; leaf a {type string;} leaf b {type string;} leaf c {type int16;} leaf d {type string;}}" |
| "leaf foo { type string;}" |
| "container c {" |
| " leaf x {type string;}" |
| " action act { input { leaf al {type string;} } output { leaf al {type uint8;} } }" |
| " notification n1 { leaf nl {type string;} }" |
| "}" |
| "container cp {presence \"container switch\"; leaf y {type string;} leaf z {type int8;}}" |
| "anydata any {config false;}" |
| "leaf-list ll1 { type uint8; }" |
| "leaf foo2 { type string; default \"default-val\"; }" |
| "leaf foo3 { type uint32; }" |
| "notification n2;}"; |
| |
| UTEST_SETUP; |
| |
| UTEST_ADD_MODULE(schema, LYS_IN_YANG, NULL, NULL); |
| assert_int_equal(LY_SUCCESS, ly_ctx_set_searchdir(UTEST_LYCTX, TESTS_DIR_MODULES_YANG)); |
| |
| return 0; |
| } |
| |
| #define CHECK_PARSE_LYD(INPUT, PARSE_OPTION, VALIDATE_OPTION, TREE) \ |
| CHECK_PARSE_LYD_PARAM(INPUT, LYD_JSON, PARSE_OPTION, VALIDATE_OPTION, LY_SUCCESS, TREE) |
| |
| #define PARSER_CHECK_ERROR(INPUT, PARSE_OPTION, VALIDATE_OPTION, MODEL, RET_VAL, ERR_MESSAGE, ERR_PATH) \ |
| assert_int_equal(RET_VAL, lyd_parse_data_mem(UTEST_LYCTX, INPUT, LYD_JSON, PARSE_OPTION, VALIDATE_OPTION, &MODEL));\ |
| CHECK_LOG_CTX(ERR_MESSAGE, ERR_PATH);\ |
| assert_null(MODEL) |
| |
| #define CHECK_LYD_STRING(IN_MODEL, PRINT_OPTION, TEXT) \ |
| CHECK_LYD_STRING_PARAM(IN_MODEL, TEXT, LYD_JSON, PRINT_OPTION) |
| |
| static void |
| test_leaf(void **state) |
| { |
| struct lyd_node *tree; |
| struct lyd_node_term *leaf; |
| const char *data; |
| |
| assert_non_null(ly_ctx_load_module(UTEST_LYCTX, "ietf-netconf-with-defaults", "2011-06-01", NULL)); |
| |
| data = "{\"a:foo\":\"foo value\"}"; |
| CHECK_PARSE_LYD(data, 0, LYD_VALIDATE_PRESENT, tree); |
| CHECK_LYSC_NODE(tree->schema, NULL, 0, LYS_CONFIG_W | LYS_STATUS_CURR, 1, "foo", 1, LYS_LEAF, 0, 0, NULL, 0); |
| leaf = (struct lyd_node_term *)tree; |
| CHECK_LYD_VALUE(leaf->value, STRING, "foo value"); |
| |
| CHECK_LYSC_NODE(tree->next->next->schema, NULL, 0, LYS_CONFIG_W | LYS_STATUS_CURR | LYS_SET_DFLT, 1, "foo2", |
| 1, LYS_LEAF, 0, 0, NULL, 0); |
| leaf = (struct lyd_node_term *)tree->next->next; |
| |
| CHECK_LYD_VALUE(leaf->value, STRING, "default-val"); |
| assert_true(leaf->flags & LYD_DEFAULT); |
| |
| CHECK_LYD_STRING(tree, LYD_PRINT_SHRINK | LYD_PRINT_WITHSIBLINGS, data); |
| lyd_free_all(tree); |
| |
| /* make foo2 explicit */ |
| data = "{\"a:foo2\":\"default-val\"}"; |
| CHECK_PARSE_LYD(data, 0, LYD_VALIDATE_PRESENT, tree); |
| assert_non_null(tree); |
| tree = tree->next; |
| CHECK_LYSC_NODE(tree->schema, NULL, 0, LYS_CONFIG_W | LYS_STATUS_CURR | LYS_SET_DFLT, 1, "foo2", |
| 1, LYS_LEAF, 0, 0, NULL, 0); |
| leaf = (struct lyd_node_term *)tree; |
| CHECK_LYD_VALUE(leaf->value, STRING, "default-val"); |
| assert_false(leaf->flags & LYD_DEFAULT); |
| |
| CHECK_LYD_STRING(tree, LYD_PRINT_SHRINK | LYD_PRINT_WITHSIBLINGS, data); |
| lyd_free_all(tree); |
| |
| /* parse foo2 but make it implicit */ |
| data = "{\"a:foo2\":\"default-val\",\"@a:foo2\":{\"ietf-netconf-with-defaults:default\":true}}"; |
| CHECK_PARSE_LYD(data, 0, LYD_VALIDATE_PRESENT, tree); |
| assert_non_null(tree); |
| tree = tree->next; |
| CHECK_LYSC_NODE(tree->schema, NULL, 0, LYS_CONFIG_W | LYS_STATUS_CURR | LYS_SET_DFLT, 1, "foo2", |
| 1, LYS_LEAF, 0, 0, NULL, 0); |
| leaf = (struct lyd_node_term *)tree; |
| CHECK_LYD_VALUE(leaf->value, STRING, "default-val"); |
| assert_true(leaf->flags & LYD_DEFAULT); |
| |
| /* TODO default values |
| lyd_print_tree(out, tree, LYD_JSON, LYD_PRINT_SHRINK); |
| assert_string_equal(printed, data); |
| ly_out_reset(out); |
| */ |
| lyd_free_all(tree); |
| |
| /* multiple meatadata hint and unknown metadata xxx supposed to be skipped since it is from missing schema */ |
| data = "{\"@a:foo\":{\"a:hint\":1,\"a:hint\":2,\"x:xxx\":{\"value\":\"/x:no/x:yes\"}},\"a:foo\":\"xxx\"}"; |
| CHECK_PARSE_LYD(data, 0, LYD_VALIDATE_PRESENT, tree); |
| CHECK_LYSC_NODE(tree->schema, NULL, 0, LYS_CONFIG_W | LYS_STATUS_CURR, 1, "foo", 1, LYS_LEAF, 0, 0, NULL, 0); |
| CHECK_LYD_META(tree->meta, 1, "hint", 1, 1, INT8, "1", 1); |
| CHECK_LYD_META(tree->meta->next, 1, "hint", 0, 1, INT8, "2", 2); |
| assert_null(tree->meta->next->next); |
| |
| CHECK_LYD_STRING(tree, LYD_PRINT_SHRINK | LYD_PRINT_WITHSIBLINGS, |
| "{\"a:foo\":\"xxx\",\"@a:foo\":{\"a:hint\":1,\"a:hint\":2}}"); |
| lyd_free_all(tree); |
| |
| PARSER_CHECK_ERROR(data, LYD_PARSE_STRICT, LYD_VALIDATE_PRESENT, tree, LY_EVALID, |
| "Unknown (or not implemented) YANG module \"x\" of metadata \"x:xxx\".", "Data location /@a:foo, line number 1."); |
| |
| /* missing referenced metadata node */ |
| PARSER_CHECK_ERROR("{\"@a:foo\" : { \"a:hint\" : 1 }}", 0, LYD_VALIDATE_PRESENT, tree, LY_EVALID, |
| "Missing JSON data instance to be coupled with @a:foo metadata.", "Data location /@a:foo, line number 1."); |
| |
| /* missing namespace for meatadata*/ |
| PARSER_CHECK_ERROR("{\"a:foo\" : \"value\", \"@a:foo\" : { \"hint\" : 1 }}", 0, LYD_VALIDATE_PRESENT, tree, LY_EVALID, |
| "Metadata in JSON must be namespace-qualified, missing prefix for \"hint\".", "Schema location /a:foo, line number 1."); |
| |
| /* reverse solidus in JSON object member name */ |
| data = "{\"@a:foo\":{\"a:hi\\nt\":1},\"a:foo\":\"xxx\"}"; |
| assert_int_equal(LY_EINVAL, lyd_parse_data_mem(UTEST_LYCTX, data, LYD_JSON, 0, LYD_VALIDATE_PRESENT, &tree)); |
| } |
| |
| static void |
| test_leaflist(void **state) |
| { |
| const char *data; |
| struct lyd_node *tree; |
| struct lyd_node_term *ll; |
| |
| data = "{\"a:ll1\":[10,11]}"; |
| CHECK_PARSE_LYD(data, 0, LYD_VALIDATE_PRESENT, tree); |
| assert_non_null(tree); |
| tree = tree->next; |
| CHECK_LYSC_NODE(tree->schema, NULL, 0, LYS_CONFIG_W | LYS_STATUS_CURR | LYS_ORDBY_SYSTEM, 1, "ll1", |
| 1, LYS_LEAFLIST, 0, 0, NULL, 0); |
| ll = (struct lyd_node_term *)tree; |
| CHECK_LYD_VALUE(ll->value, UINT8, "10", 10); |
| |
| assert_non_null(tree->next); |
| CHECK_LYSC_NODE(tree->next->schema, NULL, 0, LYS_CONFIG_W | LYS_STATUS_CURR | LYS_ORDBY_SYSTEM, 1, "ll1", |
| 1, LYS_LEAFLIST, 0, 0, NULL, 0); |
| ll = (struct lyd_node_term *)tree->next; |
| CHECK_LYD_VALUE(ll->value, UINT8, "11", 11); |
| |
| CHECK_LYD_STRING(tree, LYD_PRINT_SHRINK | LYD_PRINT_WITHSIBLINGS, data); |
| lyd_free_all(tree); |
| |
| /* simple metadata */ |
| data = "{\"a:ll1\":[10,11],\"@a:ll1\":[null,{\"a:hint\":2}]}"; |
| CHECK_PARSE_LYD(data, 0, LYD_VALIDATE_PRESENT, tree); |
| assert_non_null(tree); |
| tree = tree->next; |
| CHECK_LYSC_NODE(tree->schema, NULL, 0, LYS_CONFIG_W | LYS_STATUS_CURR | LYS_ORDBY_SYSTEM, 1, "ll1", |
| 1, LYS_LEAFLIST, 0, 0, NULL, 0); |
| ll = (struct lyd_node_term *)tree; |
| CHECK_LYD_VALUE(ll->value, UINT8, "10", 10); |
| assert_null(ll->meta); |
| |
| assert_non_null(tree->next); |
| CHECK_LYSC_NODE(tree->next->schema, NULL, 0, LYS_CONFIG_W | LYS_STATUS_CURR | LYS_ORDBY_SYSTEM, 1, "ll1", |
| 1, LYS_LEAFLIST, 0, 0, NULL, 0); |
| ll = (struct lyd_node_term *)tree->next; |
| CHECK_LYD_VALUE(ll->value, UINT8, "11", 11); |
| CHECK_LYD_META(ll->meta, 1, "hint", 0, 1, INT8, "2", 2); |
| assert_null(ll->meta->next); |
| |
| CHECK_LYD_STRING(tree, LYD_PRINT_SHRINK | LYD_PRINT_WITHSIBLINGS, data); |
| lyd_free_all(tree); |
| |
| /* multiple meatadata hint and unknown metadata xxx supposed to be skipped since it is from missing schema */ |
| data = "{\"@a:ll1\" : [{\"a:hint\" : 1, \"x:xxx\" : { \"value\" : \"/x:no/x:yes\" }, " |
| "\"a:hint\" : 10},null,{\"a:hint\" : 3}], \"a:ll1\" : [1,2,3]}"; |
| CHECK_PARSE_LYD(data, 0, LYD_VALIDATE_PRESENT, tree); |
| assert_non_null(tree); |
| tree = tree->next; |
| CHECK_LYSC_NODE(tree->schema, NULL, 0, LYS_CONFIG_W | LYS_STATUS_CURR | LYS_ORDBY_SYSTEM, 1, "ll1", |
| 1, LYS_LEAFLIST, 0, 0, NULL, 0); |
| ll = (struct lyd_node_term *)tree; |
| CHECK_LYD_VALUE(ll->value, UINT8, "1", 1); |
| CHECK_LYD_META(ll->meta, 1, "hint", 1, 1, INT8, "1", 1); |
| CHECK_LYD_META(ll->meta->next, 1, "hint", 0, 1, INT8, "10", 10); |
| |
| assert_non_null(tree->next); |
| CHECK_LYSC_NODE(tree->next->schema, NULL, 0, LYS_CONFIG_W | LYS_STATUS_CURR | LYS_ORDBY_SYSTEM, 1, "ll1", |
| 1, LYS_LEAFLIST, 0, 0, NULL, 0); |
| ll = (struct lyd_node_term *)tree->next; |
| CHECK_LYD_VALUE(ll->value, UINT8, "2", 2); |
| assert_null(ll->meta); |
| |
| assert_non_null(tree->next->next); |
| CHECK_LYSC_NODE(tree->next->next->schema, NULL, 0, LYS_CONFIG_W | LYS_STATUS_CURR | LYS_ORDBY_SYSTEM, 1, "ll1", |
| 1, LYS_LEAFLIST, 0, 0, NULL, 0); |
| ll = (struct lyd_node_term *)tree->next->next; |
| CHECK_LYD_VALUE(ll->value, UINT8, "3", 3); |
| CHECK_LYD_META(ll->meta, 1, "hint", 0, 1, INT8, "3", 3); |
| assert_null(ll->meta->next); |
| |
| CHECK_LYD_STRING(tree, LYD_PRINT_SHRINK | LYD_PRINT_WITHSIBLINGS, |
| "{\"a:ll1\":[1,2,3],\"@a:ll1\":[{\"a:hint\":1,\"a:hint\":10},null,{\"a:hint\":3}]}"); |
| lyd_free_all(tree); |
| |
| /* missing referenced metadata node */ |
| PARSER_CHECK_ERROR("{\"@a:ll1\":[{\"a:hint\":1}]}", 0, LYD_VALIDATE_PRESENT, tree, LY_EVALID, |
| "Missing JSON data instance to be coupled with @a:ll1 metadata.", "Data location /@a:ll1, line number 1."); |
| |
| PARSER_CHECK_ERROR("{\"a:ll1\":[1],\"@a:ll1\":[{\"a:hint\":1},{\"a:hint\":2}]}", 0, LYD_VALIDATE_PRESENT, tree, LY_EVALID, |
| "Missing JSON data instance #2 of a:ll1 to be coupled with metadata.", "Schema location /a:ll1, line number 1."); |
| |
| PARSER_CHECK_ERROR("{\"@a:ll1\":[{\"a:hint\":1},{\"a:hint\":2},{\"a:hint\":3}],\"a:ll1\" : [1, 2]}", 0, LYD_VALIDATE_PRESENT, |
| tree, LY_EVALID, "Missing JSON data instance #3 to be coupled with @a:ll1 metadata.", "Data location /@a:ll1, line number 1."); |
| } |
| |
| static void |
| test_anydata(void **state) |
| { |
| const char *data; |
| struct lyd_node *tree; |
| |
| data = "{\"a:any\":{\"x:element1\":{\"element2\":\"/a:some/a:path\",\"list\":[{},{\"key\":\"a\"}]}}}"; |
| CHECK_PARSE_LYD(data, 0, LYD_VALIDATE_PRESENT, tree); |
| assert_non_null(tree); |
| tree = tree->next; |
| CHECK_LYSC_NODE(tree->schema, NULL, 0, LYS_SET_ENUM | LYS_CONFIG_R | LYS_YIN_ARGUMENT, 1, "any", |
| 1, LYS_ANYDATA, 0, 0, NULL, 0); |
| CHECK_LYD_STRING(tree, LYD_PRINT_SHRINK | LYD_PRINT_WITHSIBLINGS, data); |
| lyd_free_all(tree); |
| } |
| |
| static void |
| test_list(void **state) |
| { |
| const char *data; |
| struct lyd_node *tree, *iter; |
| struct lyd_node_inner *list; |
| struct lyd_node_term *leaf; |
| |
| /* check hashes */ |
| data = "{\"a:l1\":[{\"a\":\"one\",\"b\":\"one\",\"c\":1}]}"; |
| CHECK_PARSE_LYD(data, 0, LYD_VALIDATE_PRESENT, tree); |
| CHECK_LYSC_NODE(tree->schema, NULL, 0, LYS_CONFIG_W | LYS_STATUS_CURR | LYS_ORDBY_SYSTEM, 1, "l1", |
| 1, LYS_LIST, 0, 0, NULL, 0); |
| list = (struct lyd_node_inner *)tree; |
| LY_LIST_FOR(list->child, iter) { |
| assert_int_not_equal(0, iter->hash); |
| } |
| |
| CHECK_LYD_STRING(tree, LYD_PRINT_SHRINK | LYD_PRINT_WITHSIBLINGS, data); |
| lyd_free_all(tree); |
| |
| /* missing keys */ |
| PARSER_CHECK_ERROR("{ \"a:l1\": [ {\"c\" : 1, \"b\" : \"b\"}]}", 0, LYD_VALIDATE_PRESENT, tree, LY_EVALID, |
| "List instance is missing its key \"a\".", "Schema location /a:l1, data location /a:l1[b='b'][c='1'], line number 1."); |
| |
| PARSER_CHECK_ERROR("{ \"a:l1\": [ {\"a\" : \"a\"}]}", 0, LYD_VALIDATE_PRESENT, tree, LY_EVALID, |
| "List instance is missing its key \"b\".", "Schema location /a:l1, data location /a:l1[a='a'], line number 1."); |
| |
| PARSER_CHECK_ERROR("{ \"a:l1\": [ {\"b\" : \"b\", \"a\" : \"a\"}]}", 0, LYD_VALIDATE_PRESENT, tree, LY_EVALID, |
| "List instance is missing its key \"c\".", "Schema location /a:l1, data location /a:l1[a='a'][b='b'], line number 1."); |
| |
| /* key duplicate */ |
| PARSER_CHECK_ERROR("{ \"a:l1\": [ {\"c\" : 1, \"b\" : \"b\", \"a\" : \"a\", \"c\" : 1}]}", 0, LYD_VALIDATE_PRESENT, |
| tree, LY_EVALID, "Duplicate instance of \"c\".", "Schema location /a:l1/c, data location /a:l1[a='a'][b='b'][c='1'][c='1']/c, line number 1."); |
| |
| /* keys order, in contrast to XML, JSON accepts keys in any order even in strict mode */ |
| CHECK_PARSE_LYD("{ \"a:l1\": [ {\"d\" : \"d\", \"a\" : \"a\", \"c\" : 1, \"b\" : \"b\"}]}", 0, LYD_VALIDATE_PRESENT, tree); |
| CHECK_LYSC_NODE(tree->schema, NULL, 0, LYS_CONFIG_W | LYS_STATUS_CURR | LYS_ORDBY_SYSTEM, 1, "l1", |
| 1, LYS_LIST, 0, 0, NULL, 0); |
| list = (struct lyd_node_inner *)tree; |
| assert_non_null(leaf = (struct lyd_node_term *)list->child); |
| CHECK_LYSC_NODE(leaf->schema, NULL, 0, LYS_CONFIG_W | LYS_STATUS_CURR | LYS_KEY, 1, "a", 1, LYS_LEAF, 1, 0, NULL, 0); |
| assert_non_null(leaf = (struct lyd_node_term *)leaf->next); |
| CHECK_LYSC_NODE(leaf->schema, NULL, 0, LYS_CONFIG_W | LYS_STATUS_CURR | LYS_KEY, 1, "b", 1, LYS_LEAF, 1, 0, NULL, 0); |
| assert_non_null(leaf = (struct lyd_node_term *)leaf->next); |
| CHECK_LYSC_NODE(leaf->schema, NULL, 0, LYS_CONFIG_W | LYS_STATUS_CURR | LYS_KEY, 1, "c", 1, LYS_LEAF, 1, 0, NULL, 0); |
| assert_non_null(leaf = (struct lyd_node_term *)leaf->next); |
| CHECK_LYSC_NODE(leaf->schema, NULL, 0, LYS_CONFIG_W | LYS_STATUS_CURR, 1, "d", 0, LYS_LEAF, 1, 0, NULL, 0); |
| CHECK_LOG_CTX(NULL, NULL); |
| |
| CHECK_LYD_STRING(tree, LYD_PRINT_SHRINK | LYD_PRINT_WITHSIBLINGS, |
| "{\"a:l1\":[{\"a\":\"a\",\"b\":\"b\",\"c\":1,\"d\":\"d\"}]}"); |
| lyd_free_all(tree); |
| |
| /* */ |
| CHECK_PARSE_LYD("{\"a:l1\":[{\"c\":1,\"b\":\"b\",\"a\":\"a\"}]}", LYD_PARSE_STRICT, LYD_VALIDATE_PRESENT, tree); |
| CHECK_LYSC_NODE(tree->schema, NULL, 0, LYS_CONFIG_W | LYS_STATUS_CURR | LYS_ORDBY_SYSTEM, 1, "l1", |
| 1, LYS_LIST, 0, 0, NULL, 0); |
| list = (struct lyd_node_inner *)tree; |
| assert_non_null(leaf = (struct lyd_node_term *)list->child); |
| CHECK_LYSC_NODE(leaf->schema, NULL, 0, LYS_CONFIG_W | LYS_STATUS_CURR | LYS_KEY, 1, "a", |
| 1, LYS_LEAF, 1, 0, NULL, 0); |
| assert_non_null(leaf = (struct lyd_node_term *)leaf->next); |
| CHECK_LYSC_NODE(leaf->schema, NULL, 0, LYS_CONFIG_W | LYS_STATUS_CURR | LYS_KEY, 1, "b", |
| 1, LYS_LEAF, 1, 0, NULL, 0); |
| assert_non_null(leaf = (struct lyd_node_term *)leaf->next); |
| CHECK_LYSC_NODE(leaf->schema, NULL, 0, LYS_CONFIG_W | LYS_STATUS_CURR | LYS_KEY, 1, "c", |
| 1, LYS_LEAF, 1, 0, NULL, 0); |
| CHECK_LOG_CTX(NULL, NULL); |
| |
| CHECK_LYD_STRING(tree, LYD_PRINT_SHRINK | LYD_PRINT_WITHSIBLINGS, |
| "{\"a:l1\":[{\"a\":\"a\",\"b\":\"b\",\"c\":1}]}"); |
| lyd_free_all(tree); |
| |
| data = "{\"a:cp\":{\"@\":{\"a:hint\":1}}}"; |
| CHECK_PARSE_LYD(data, 0, LYD_VALIDATE_PRESENT, tree); |
| assert_non_null(tree); |
| tree = tree->next; |
| CHECK_LYSC_NODE(tree->schema, NULL, 0, LYS_CONFIG_W | LYS_STATUS_CURR | LYS_PRESENCE, 1, "cp", |
| 1, LYS_CONTAINER, 0, 0, NULL, 0); |
| CHECK_LYD_META(tree->meta, 1, "hint", 0, 1, INT8, "1", 1); |
| assert_null(tree->meta->next); |
| |
| CHECK_LYD_STRING(tree, LYD_PRINT_SHRINK | LYD_PRINT_WITHSIBLINGS, data); |
| lyd_free_all(tree); |
| } |
| |
| static void |
| test_container(void **state) |
| { |
| const char *data; |
| struct lyd_node *tree; |
| struct lyd_node_inner *cont; |
| |
| CHECK_PARSE_LYD("{\"a:c\":{}}", 0, LYD_VALIDATE_PRESENT, tree); |
| CHECK_LYSC_NODE(tree->schema, NULL, 0, LYS_CONFIG_W | LYS_STATUS_CURR, 1, "c", |
| 1, LYS_CONTAINER, 0, 0, NULL, 0); |
| cont = (struct lyd_node_inner *)tree; |
| assert_true(cont->flags & LYD_DEFAULT); |
| |
| CHECK_LYD_STRING(tree, LYD_PRINT_SHRINK | LYD_PRINT_WITHSIBLINGS, "{}"); |
| lyd_free_all(tree); |
| |
| data = "{\"a:cp\":{}}"; |
| CHECK_PARSE_LYD(data, 0, LYD_VALIDATE_PRESENT, tree); |
| assert_non_null(tree); |
| tree = tree->next; |
| CHECK_LYSC_NODE(tree->schema, NULL, 0, LYS_CONFIG_W | LYS_STATUS_CURR | LYS_PRESENCE, 1, "cp", |
| 1, LYS_CONTAINER, 0, 0, NULL, 0); |
| cont = (struct lyd_node_inner *)tree; |
| assert_false(cont->flags & LYD_DEFAULT); |
| |
| CHECK_LYD_STRING(tree, LYD_PRINT_SHRINK | LYD_PRINT_WITHSIBLINGS, data); |
| lyd_free_all(tree); |
| } |
| |
| static void |
| test_opaq(void **state) |
| { |
| const char *data; |
| struct lyd_node *tree; |
| |
| /* invalid value, no flags */ |
| data = "{\"a:foo3\":[null]}"; |
| PARSER_CHECK_ERROR(data, 0, LYD_VALIDATE_PRESENT, tree, LY_EVALID, |
| "Invalid non-number-encoded uint32 value \"\".", "Schema location /a:foo3, line number 1."); |
| |
| /* opaq flag */ |
| CHECK_PARSE_LYD(data, LYD_PARSE_OPAQ | LYD_PARSE_ONLY, LYD_VALIDATE_PRESENT, tree); |
| CHECK_LYD_NODE_OPAQ((struct lyd_node_opaq *)tree, 0, 0, LY_VALUE_JSON, "foo3", 0, 0, NULL, 0, ""); |
| CHECK_LYD_STRING(tree, LYD_PRINT_SHRINK | LYD_PRINT_WITHSIBLINGS, data); |
| lyd_free_all(tree); |
| |
| /* missing key, no flags */ |
| data = "{\"a:l1\":[{\"a\":\"val_a\",\"b\":\"val_b\",\"d\":\"val_d\"}]}"; |
| PARSER_CHECK_ERROR(data, 0, LYD_VALIDATE_PRESENT, tree, LY_EVALID, |
| "List instance is missing its key \"c\".", "Schema location /a:l1, data location /a:l1[a='val_a'][b='val_b'], line number 1."); |
| |
| /* opaq flag */ |
| CHECK_PARSE_LYD(data, LYD_PARSE_OPAQ | LYD_PARSE_ONLY, LYD_VALIDATE_PRESENT, tree); |
| CHECK_LYD_NODE_OPAQ((struct lyd_node_opaq *)tree, 0, 0x1, LY_VALUE_JSON, "l1", 0, 0, NULL, 0, ""); |
| CHECK_LYD_STRING(tree, LYD_PRINT_SHRINK | LYD_PRINT_WITHSIBLINGS, data); |
| lyd_free_all(tree); |
| |
| /* invalid key, no flags */ |
| data = "{\"a:l1\":[{\"a\":\"val_a\",\"b\":\"val_b\",\"c\":\"val_c\"}]}"; |
| PARSER_CHECK_ERROR(data, 0, LYD_VALIDATE_PRESENT, tree, LY_EVALID, |
| "Invalid non-number-encoded int16 value \"val_c\".", "Schema location /a:l1/c, data location /a:l1[a='val_a'][b='val_b'], line number 1."); |
| |
| /* opaq flag */ |
| CHECK_PARSE_LYD(data, LYD_PARSE_OPAQ | LYD_PARSE_ONLY, LYD_VALIDATE_PRESENT, tree); |
| CHECK_LYD_NODE_OPAQ((struct lyd_node_opaq *)tree, 0, 0x1, LY_VALUE_JSON, "l1", 0, 0, NULL, 0, ""); |
| CHECK_LYD_STRING(tree, LYD_PRINT_SHRINK | LYD_PRINT_WITHSIBLINGS, data); |
| lyd_free_all(tree); |
| |
| data = "{\"a:l1\":[{\"a\":\"val_a\",\"b\":\"val_b\",\"c\":{\"val\":\"val_c\"}}]}"; |
| CHECK_PARSE_LYD(data, LYD_PARSE_OPAQ | LYD_PARSE_ONLY, LYD_VALIDATE_PRESENT, tree); |
| CHECK_LYD_NODE_OPAQ((struct lyd_node_opaq *)tree, 0, 0x1, LY_VALUE_JSON, "l1", 0, 0, NULL, 0, ""); |
| CHECK_LYD_STRING(tree, LYD_PRINT_SHRINK | LYD_PRINT_WITHSIBLINGS, data); |
| lyd_free_all(tree); |
| |
| data = "{\"a:l1\":[{\"a\":\"val_a\",\"b\":\"val_b\"}]}"; |
| CHECK_PARSE_LYD(data, LYD_PARSE_OPAQ | LYD_PARSE_ONLY, LYD_VALIDATE_PRESENT, tree); |
| CHECK_LYD_NODE_OPAQ((struct lyd_node_opaq *)tree, 0, 0x1, LY_VALUE_JSON, "l1", 0, 0, NULL, 0, ""); |
| CHECK_LYD_STRING(tree, LYD_PRINT_SHRINK | LYD_PRINT_WITHSIBLINGS, data); |
| lyd_free_all(tree); |
| |
| /* mixing with metadata that is ignored */ |
| data = "{\"@a:foo\":\"str\",\"@a:foo3\":1,\"a:foo3\":2}"; |
| CHECK_PARSE_LYD(data, 0, LYD_VALIDATE_PRESENT, tree); |
| lyd_free_all(tree); |
| |
| /* empty name */ |
| PARSER_CHECK_ERROR("{\"@a:foo\":{\"\":0}}", 0, LYD_VALIDATE_PRESENT, tree, LY_EVALID, |
| "A JSON object member name cannot be a zero-length string.", "Line number 1."); |
| } |
| |
| static void |
| test_rpc(void **state) |
| { |
| const char *data; |
| struct ly_in *in; |
| struct lyd_node *tree, *op; |
| const struct lyd_node *node; |
| const char *dsc = "The <edit-config> operation loads all or part of a specified\n" |
| "configuration to the specified target configuration."; |
| const char *ref = "RFC 6241, Section 7.2"; |
| const char *feats[] = {"writable-running", NULL}; |
| |
| assert_non_null((ly_ctx_load_module(UTEST_LYCTX, "ietf-netconf", "2011-06-01", feats))); |
| |
| data = "{\"ietf-netconf:edit-config\":{" |
| "\"target\":{\"running\":[null]}," |
| "\"config\":{\"a:cp\":{\"z\":[null],\"@z\":{\"ietf-netconf:operation\":\"replace\"}}," |
| "\"a:l1\":[{\"@\":{\"ietf-netconf:operation\":\"replace\"},\"a\":\"val_a\",\"b\":\"val_b\",\"c\":\"val_c\"}]}" |
| "}}"; |
| assert_int_equal(LY_SUCCESS, ly_in_new_memory(data, &in)); |
| assert_int_equal(LY_SUCCESS, lyd_parse_op(UTEST_LYCTX, NULL, in, LYD_JSON, LYD_TYPE_RPC_YANG, &tree, &op)); |
| ly_in_free(in, 0); |
| |
| assert_non_null(op); |
| |
| CHECK_LYSC_ACTION((struct lysc_node_action *)op->schema, dsc, 0, LYS_STATUS_CURR, |
| 1, 0, 0, 1, "edit-config", LYS_RPC, |
| 0, 0, 0, 0, 0, ref, 0); |
| |
| node = tree; |
| CHECK_LYSC_ACTION((struct lysc_node_action *)node->schema, dsc, 0, LYS_STATUS_CURR, |
| 1, 0, 0, 1, "edit-config", LYS_RPC, |
| 0, 0, 0, 0, 0, ref, 0); |
| node = lyd_child(node)->next; |
| CHECK_LYSC_NODE(node->schema, "Inline Config content.", 0, LYS_STATUS_CURR | LYS_IS_INPUT, 1, "config", |
| 0, LYS_ANYXML, 1, 0, NULL, 0); |
| |
| node = ((struct lyd_node_any *)node)->value.tree; |
| CHECK_LYSC_NODE(node->schema, NULL, 0, LYS_CONFIG_W | LYS_STATUS_CURR | LYS_PRESENCE, 1, "cp", |
| 1, LYS_CONTAINER, 0, 0, NULL, 0); |
| node = lyd_child(node); |
| /* z has no value */ |
| CHECK_LYD_NODE_OPAQ((struct lyd_node_opaq *)node, 0x1, 0, LY_VALUE_JSON, "z", 0, 0, NULL, 0, ""); |
| node = node->parent->next; |
| /* l1 key c has invalid value so it is at the end */ |
| CHECK_LYD_NODE_OPAQ((struct lyd_node_opaq *)node, 0x1, 0x1, LY_VALUE_JSON, "l1", 0, 0, NULL, 0, ""); |
| |
| CHECK_LYD_STRING(tree, LYD_PRINT_SHRINK | LYD_PRINT_WITHSIBLINGS, data); |
| lyd_free_all(tree); |
| |
| /* wrong namespace, element name, whatever... */ |
| /* TODO */ |
| } |
| |
| static void |
| test_action(void **state) |
| { |
| const char *data; |
| struct ly_in *in; |
| struct lyd_node *tree, *op; |
| |
| data = "{\"a:c\":{\"act\":{\"al\":\"value\"}}}"; |
| assert_int_equal(LY_SUCCESS, ly_in_new_memory(data, &in)); |
| assert_int_equal(LY_SUCCESS, lyd_parse_op(UTEST_LYCTX, NULL, in, LYD_JSON, LYD_TYPE_RPC_YANG, &tree, &op)); |
| ly_in_free(in, 0); |
| |
| assert_non_null(op); |
| CHECK_LYSC_ACTION((struct lysc_node_action *)op->schema, NULL, 0, LYS_STATUS_CURR, |
| 1, 0, 0, 1, "act", LYS_ACTION, |
| 1, 0, 0, 1, 0, NULL, 0); |
| |
| CHECK_LYD_STRING(tree, LYD_PRINT_SHRINK | LYD_PRINT_WITHSIBLINGS, data); |
| lyd_free_all(tree); |
| |
| /* wrong namespace, element name, whatever... */ |
| /* TODO */ |
| } |
| |
| static void |
| test_notification(void **state) |
| { |
| const char *data; |
| struct ly_in *in; |
| struct lyd_node *tree, *ntf; |
| |
| data = "{\"a:c\":{\"n1\":{\"nl\":\"value\"}}}"; |
| assert_int_equal(LY_SUCCESS, ly_in_new_memory(data, &in)); |
| assert_int_equal(LY_SUCCESS, lyd_parse_op(UTEST_LYCTX, NULL, in, LYD_JSON, LYD_TYPE_NOTIF_YANG, &tree, &ntf)); |
| ly_in_free(in, 0); |
| |
| assert_non_null(ntf); |
| CHECK_LYSC_NOTIF((struct lysc_node_notif *)ntf->schema, 1, NULL, 0, 0x4, 1, 0, "n1", 1, 0, NULL, 0); |
| |
| CHECK_LYSC_NODE(tree->schema, NULL, 0, LYS_CONFIG_W | LYS_STATUS_CURR, 1, "c", 1, LYS_CONTAINER, 0, 0, NULL, 0); |
| |
| CHECK_LYD_STRING(tree, LYD_PRINT_SHRINK | LYD_PRINT_WITHSIBLINGS, data); |
| lyd_free_all(tree); |
| |
| data = "{\"a:n2\":{}}"; |
| assert_int_equal(LY_SUCCESS, ly_in_new_memory(data, &in)); |
| assert_int_equal(LY_SUCCESS, lyd_parse_op(UTEST_LYCTX, NULL, in, LYD_JSON, LYD_TYPE_NOTIF_YANG, &tree, &ntf)); |
| ly_in_free(in, 0); |
| |
| assert_non_null(ntf); |
| CHECK_LYSC_NOTIF((struct lysc_node_notif *)ntf->schema, 0, NULL, 0, 0x4, 1, 0, "n2", 0, 0, NULL, 0); |
| |
| assert_non_null(tree); |
| assert_ptr_equal(ntf, tree); |
| |
| CHECK_LYD_STRING(tree, LYD_PRINT_SHRINK | LYD_PRINT_WITHSIBLINGS, data); |
| lyd_free_all(tree); |
| |
| /* wrong namespace, element name, whatever... */ |
| /* TODO */ |
| } |
| |
| static void |
| test_reply(void **state) |
| { |
| const char *data; |
| struct ly_in *in; |
| struct lyd_node *tree, *op; |
| const struct lyd_node *node; |
| |
| data = "{\"a:c\":{\"act\":{\"al\":25}}}"; |
| assert_int_equal(LY_SUCCESS, ly_in_new_memory(data, &in)); |
| assert_int_equal(LY_SUCCESS, lyd_parse_op(UTEST_LYCTX, NULL, in, LYD_JSON, LYD_TYPE_REPLY_YANG, &tree, &op)); |
| ly_in_free(in, 0); |
| |
| assert_non_null(op); |
| CHECK_LYSC_ACTION((struct lysc_node_action *)op->schema, NULL, 0, LYS_STATUS_CURR, |
| 1, 0, 0, 1, "act", LYS_ACTION, |
| 1, 0, 0, 1, 0, NULL, 0); |
| node = lyd_child(op); |
| CHECK_LYSC_NODE(node->schema, NULL, 0, LYS_STATUS_CURR | LYS_IS_OUTPUT, 1, "al", 0, LYS_LEAF, 1, 0, NULL, 0); |
| |
| CHECK_LYSC_NODE(tree->schema, NULL, 0, LYS_CONFIG_W | LYS_STATUS_CURR, 1, "c", 1, LYS_CONTAINER, 0, 0, NULL, 0); |
| |
| /* TODO print only rpc-reply node and then output subtree */ |
| CHECK_LYD_STRING(lyd_child(op), LYD_PRINT_SHRINK | LYD_PRINT_WITHSIBLINGS, "{\"a:al\":25}"); |
| CHECK_LYD_STRING(tree, LYD_PRINT_SHRINK | LYD_PRINT_WITHSIBLINGS, "{\"a:c\":{\"act\":{\"al\":25}}}"); |
| lyd_free_all(tree); |
| |
| /* wrong namespace, element name, whatever... */ |
| /* TODO */ |
| } |
| |
| int |
| main(void) |
| { |
| const struct CMUnitTest tests[] = { |
| UTEST(test_leaf, setup), |
| UTEST(test_leaflist, setup), |
| UTEST(test_anydata, setup), |
| UTEST(test_list, setup), |
| UTEST(test_container, setup), |
| UTEST(test_opaq, setup), |
| UTEST(test_rpc, setup), |
| UTEST(test_action, setup), |
| UTEST(test_notification, setup), |
| UTEST(test_reply, setup), |
| }; |
| |
| return cmocka_run_group_tests(tests, NULL, NULL); |
| } |