blob: 1e74bb9ffd43b852319ec2453f690d2bb84bb402 [file] [log] [blame]
Radek Krejci509e2592019-05-15 16:30:48 +02001/*
2 * @file test_parser_xml.c
3 * @author: Radek Krejci <rkrejci@cesnet.cz>
4 * @brief unit tests for functions from parser_xml.c
5 *
Radek Iša56ca9e42020-09-08 18:42:00 +02006 * Copyright (c) 2019-2020 CESNET, z.s.p.o.
Radek Krejci509e2592019-05-15 16:30:48 +02007 *
8 * This source code is licensed under BSD 3-Clause License (the "License").
9 * You may not use this file except in compliance with the License.
10 * You may obtain a copy of the License at
11 *
12 * https://opensource.org/licenses/BSD-3-Clause
13 */
Radek Iša56ca9e42020-09-08 18:42:00 +020014#define _UTEST_MAIN_
15#include "utests.h"
Radek Krejci509e2592019-05-15 16:30:48 +020016
Radek Krejci70593c12020-06-13 20:48:09 +020017#include "context.h"
Michal Vaskoafac7822020-10-20 14:22:26 +020018#include "in.h"
Michal Vaskoafac7822020-10-20 14:22:26 +020019#include "out.h"
Radek Krejcib4ac5a92020-11-23 17:54:33 +010020#include "parser_data.h"
Radek Krejci70593c12020-06-13 20:48:09 +020021#include "printer_data.h"
Radek Krejci70593c12020-06-13 20:48:09 +020022#include "tree_data_internal.h"
23#include "tree_schema.h"
Radek Krejci509e2592019-05-15 16:30:48 +020024
25static int
26setup(void **state)
27{
Radek Iša56ca9e42020-09-08 18:42:00 +020028 const char *schema =
Radek Krejcib4ac5a92020-11-23 17:54:33 +010029 "module a {\n"
30 " namespace urn:tests:a;\n"
31 " prefix a;\n"
32 " yang-version 1.1;\n"
33 " list l1 {\n"
34 " key \"a b c\";\n"
35 " leaf a {type string;}\n"
36 " leaf b {type string;}\n"
37 " leaf c {type int16;}\n"
38 " leaf d {type string;}}\n"
39 " leaf foo { type string;}\n"
40 " container c {\n"
41 " leaf x {type string;}\n"
42 " action act { input { leaf al {type string;} } output { leaf al {type uint8;} } }\n"
43 " notification n1 { leaf nl {type string;}}}\n"
44 " container cp {presence \"container switch\"; leaf y {type string;} leaf z {type int8;}}\n"
45 " anydata any {config false;}\n"
46 " leaf foo2 { type string; default \"default-val\"; }\n"
47 " leaf foo3 { type uint32; }\n"
48 " notification n2;}";
Radek Krejci509e2592019-05-15 16:30:48 +020049
Radek Iša56ca9e42020-09-08 18:42:00 +020050 UTEST_SETUP;
Radek Krejci509e2592019-05-15 16:30:48 +020051
Radek Iša56ca9e42020-09-08 18:42:00 +020052 UTEST_ADD_MODULE(schema, LYS_IN_YANG, NULL, NULL);
53 assert_int_equal(LY_SUCCESS, ly_ctx_set_searchdir(UTEST_LYCTX, TESTS_DIR_MODULES_YANG));
Radek Krejci509e2592019-05-15 16:30:48 +020054
55 return 0;
56}
57
Radek Iša56ca9e42020-09-08 18:42:00 +020058#define CHECK_PARSE_LYD(INPUT, PARSE_OPTION, VALIDATE_OPTION, TREE) \
59 CHECK_PARSE_LYD_PARAM(INPUT, LYD_XML, PARSE_OPTION, VALIDATE_OPTION, LY_SUCCESS, TREE)
Radek Krejci509e2592019-05-15 16:30:48 +020060
Radek Iša56ca9e42020-09-08 18:42:00 +020061#define PARSER_CHECK_ERROR(INPUT, PARSE_OPTION, VALIDATE_OPTION, MODEL, RET_VAL, ERR_MESSAGE, ERR_PATH) \
62 assert_int_equal(RET_VAL, lyd_parse_data_mem(UTEST_LYCTX, INPUT, LYD_XML, PARSE_OPTION, VALIDATE_OPTION, &MODEL));\
63 CHECK_LOG_CTX(ERR_MESSAGE, ERR_PATH);\
64 assert_null(MODEL)
Radek Krejci509e2592019-05-15 16:30:48 +020065
Radek Iša56ca9e42020-09-08 18:42:00 +020066#define CHECK_LYD_STRING(IN_MODEL, PRINT_OPTION, TEXT) \
67 CHECK_LYD_STRING_PARAM(IN_MODEL, TEXT, LYD_XML, PRINT_OPTION)
Radek Krejci509e2592019-05-15 16:30:48 +020068
Radek Krejci509e2592019-05-15 16:30:48 +020069static void
70test_leaf(void **state)
71{
Radek Krejci509e2592019-05-15 16:30:48 +020072 const char *data = "<foo xmlns=\"urn:tests:a\">foo value</foo>";
73 struct lyd_node *tree;
74 struct lyd_node_term *leaf;
75
Radek Iša56ca9e42020-09-08 18:42:00 +020076 assert_non_null(ly_ctx_load_module(UTEST_LYCTX, "ietf-netconf-with-defaults", "2011-06-01", NULL));
Radek Krejci509e2592019-05-15 16:30:48 +020077
Radek Iša56ca9e42020-09-08 18:42:00 +020078 CHECK_PARSE_LYD(data, 0, LYD_VALIDATE_PRESENT, tree);
79 CHECK_LYSC_NODE(tree->schema, NULL, 0, LYS_CONFIG_W | LYS_STATUS_CURR, 1, "foo", 1, LYS_LEAF, 0, 0, NULL, 0);
80 leaf = (struct lyd_node_term *)tree;
81 CHECK_LYD_VALUE(leaf->value, STRING, "foo value");
82
83 CHECK_LYSC_NODE(tree->next->next->schema, NULL, 0, LYS_CONFIG_W | LYS_STATUS_CURR | LYS_SET_DFLT, 1, "foo2",
84 1, LYS_LEAF, 0, 0, NULL, 0);
Radek Krejcib4ac5a92020-11-23 17:54:33 +010085 leaf = (struct lyd_node_term *)tree->next->next;
Radek Iša56ca9e42020-09-08 18:42:00 +020086 CHECK_LYD_VALUE(leaf->value, STRING, "default-val");
Michal Vasko8d544252020-03-02 10:19:52 +010087 assert_true(leaf->flags & LYD_DEFAULT);
88
Radek Krejci509e2592019-05-15 16:30:48 +020089 lyd_free_all(tree);
Michal Vasko8d544252020-03-02 10:19:52 +010090
91 /* make foo2 explicit */
92 data = "<foo2 xmlns=\"urn:tests:a\">default-val</foo2>";
Radek Iša56ca9e42020-09-08 18:42:00 +020093 CHECK_PARSE_LYD(data, 0, LYD_VALIDATE_PRESENT, tree);
Michal Vasko8d544252020-03-02 10:19:52 +010094 assert_non_null(tree);
Michal Vasko26123192020-11-09 21:02:34 +010095 tree = tree->next;
Radek Iša56ca9e42020-09-08 18:42:00 +020096 CHECK_LYSC_NODE(tree->schema, NULL, 0, LYS_CONFIG_W | LYS_STATUS_CURR | LYS_SET_DFLT, 1, "foo2",
97 1, LYS_LEAF, 0, 0, NULL, 0);
Radek Krejcib4ac5a92020-11-23 17:54:33 +010098 leaf = (struct lyd_node_term *)tree;
Radek Iša56ca9e42020-09-08 18:42:00 +020099 CHECK_LYD_VALUE(leaf->value, STRING, "default-val");
Michal Vasko8d544252020-03-02 10:19:52 +0100100 assert_false(leaf->flags & LYD_DEFAULT);
101
102 lyd_free_all(tree);
103
Radek Krejci1798aae2020-07-14 13:26:06 +0200104 /* parse foo2 but make it implicit, skip metadata xxx from missing schema */
Radek Iša56ca9e42020-09-08 18:42:00 +0200105 data = "<foo2 xmlns=\"urn:tests:a\" xmlns:wd=\"urn:ietf:params:xml:ns:yang:ietf-netconf-with-defaults\" "
106 "wd:default=\"true\" xmlns:x=\"urn:x\" x:xxx=\"false\">default-val</foo2>";
107 CHECK_PARSE_LYD(data, 0, LYD_VALIDATE_PRESENT, tree);
Michal Vasko8d544252020-03-02 10:19:52 +0100108 assert_non_null(tree);
Michal Vasko26123192020-11-09 21:02:34 +0100109 tree = tree->next;
Radek Iša56ca9e42020-09-08 18:42:00 +0200110 CHECK_LYSC_NODE(tree->schema, NULL, 0, LYS_CONFIG_W | LYS_STATUS_CURR | LYS_SET_DFLT, 1, "foo2",
111 1, LYS_LEAF, 0, 0, NULL, 0);
Radek Krejcib4ac5a92020-11-23 17:54:33 +0100112 leaf = (struct lyd_node_term *)tree;
Radek Iša56ca9e42020-09-08 18:42:00 +0200113 CHECK_LYD_VALUE(leaf->value, STRING, "default-val");
Michal Vasko8d544252020-03-02 10:19:52 +0100114 assert_true(leaf->flags & LYD_DEFAULT);
115
116 lyd_free_all(tree);
Radek Krejci509e2592019-05-15 16:30:48 +0200117}
118
Radek Krejciee4cab22019-07-17 17:07:47 +0200119static void
120test_anydata(void **state)
121{
Michal Vasko52927e22020-03-16 17:26:14 +0100122 const char *data;
Radek Krejciee4cab22019-07-17 17:07:47 +0200123 struct lyd_node *tree;
Radek Krejciee4cab22019-07-17 17:07:47 +0200124
Radek Iša56ca9e42020-09-08 18:42:00 +0200125 data = "<any xmlns=\"urn:tests:a\">\n"
Radek Krejcib4ac5a92020-11-23 17:54:33 +0100126 " <element1>\n"
127 " <x:element2 x:attr2=\"test\" xmlns:a=\"urn:tests:a\" xmlns:x=\"urn:x\">a:data</x:element2>\n"
128 " </element1>\n"
129 " <element1a/>\n"
130 "</any>\n";
Radek Iša56ca9e42020-09-08 18:42:00 +0200131 CHECK_PARSE_LYD(data, 0, LYD_VALIDATE_PRESENT, tree);
Radek Krejciee4cab22019-07-17 17:07:47 +0200132 assert_non_null(tree);
Michal Vasko26123192020-11-09 21:02:34 +0100133 tree = tree->next;
Radek Iša56ca9e42020-09-08 18:42:00 +0200134 CHECK_LYSC_NODE(tree->schema, NULL, 0, LYS_CONFIG_R | LYS_STATUS_CURR | LYS_SET_CONFIG, 1, "any",
135 1, LYS_ANYDATA, 0, 0, NULL, 0);
136 const char *data_expected =
Radek Krejcib4ac5a92020-11-23 17:54:33 +0100137 "<any xmlns=\"urn:tests:a\">\n"
138 " <element1>\n"
139 " <element2 xmlns=\"urn:x\" xmlns:x=\"urn:x\" x:attr2=\"test\" xmlns:a=\"urn:tests:a\">a:data</element2>\n"
140 " </element1>\n"
141 " <element1a/>\n"
Radek Iša56ca9e42020-09-08 18:42:00 +0200142 "</any>\n";
143
144 CHECK_LYD_STRING(tree, LYD_PRINT_WITHSIBLINGS, data_expected);
Radek Krejciee4cab22019-07-17 17:07:47 +0200145
146 lyd_free_all(tree);
Radek Krejciee4cab22019-07-17 17:07:47 +0200147}
148
Radek Krejci1f05b6a2019-07-18 16:15:06 +0200149static void
150test_list(void **state)
151{
Radek Iša56ca9e42020-09-08 18:42:00 +0200152 const char *data;
Radek Krejci1f05b6a2019-07-18 16:15:06 +0200153 struct lyd_node *tree, *iter;
154 struct lyd_node_inner *list;
Radek Krejci710226d2019-07-24 17:24:59 +0200155 struct lyd_node_term *leaf;
Radek Krejci1f05b6a2019-07-18 16:15:06 +0200156
Radek Krejci710226d2019-07-24 17:24:59 +0200157 /* check hashes */
Radek Iša56ca9e42020-09-08 18:42:00 +0200158 data = "<l1 xmlns=\"urn:tests:a\"><a>one</a><b>one</b><c>1</c></l1>";
159 CHECK_PARSE_LYD(data, 0, LYD_VALIDATE_PRESENT, tree);
160 CHECK_LYSC_NODE(tree->schema, NULL, 0, LYS_CONFIG_W | LYS_STATUS_CURR | LYS_ORDBY_SYSTEM, 1, "l1",
161 1, LYS_LIST, 0, 0, NULL, 0);
Radek Krejcib4ac5a92020-11-23 17:54:33 +0100162 list = (struct lyd_node_inner *)tree;
Radek Krejci1f05b6a2019-07-18 16:15:06 +0200163 LY_LIST_FOR(list->child, iter) {
164 assert_int_not_equal(0, iter->hash);
165 }
Radek Krejci1f05b6a2019-07-18 16:15:06 +0200166 lyd_free_all(tree);
Radek Krejci710226d2019-07-24 17:24:59 +0200167
Michal Vasko9f96a052020-03-10 09:41:45 +0100168 /* missing keys */
Radek Iša56ca9e42020-09-08 18:42:00 +0200169 PARSER_CHECK_ERROR("<l1 xmlns=\"urn:tests:a\"><c>1</c><b>b</b></l1>", 0, LYD_VALIDATE_PRESENT, tree, LY_EVALID,
Radek Krejci2efc45b2020-12-22 16:25:44 +0100170 "List instance is missing its key \"a\".", "Schema location /a:l1, data location /a:l1[b='b'][c='1'], line number 1.");
Michal Vasko9f96a052020-03-10 09:41:45 +0100171
Radek Iša56ca9e42020-09-08 18:42:00 +0200172 PARSER_CHECK_ERROR("<l1 xmlns=\"urn:tests:a\"><a>a</a></l1>", 0, LYD_VALIDATE_PRESENT, tree, LY_EVALID,
Radek Krejci2efc45b2020-12-22 16:25:44 +0100173 "List instance is missing its key \"b\".", "Schema location /a:l1, data location /a:l1[a='a'], line number 1.");
Michal Vasko9f96a052020-03-10 09:41:45 +0100174
Radek Iša56ca9e42020-09-08 18:42:00 +0200175 PARSER_CHECK_ERROR("<l1 xmlns=\"urn:tests:a\"><b>b</b><a>a</a></l1>", 0, LYD_VALIDATE_PRESENT, tree, LY_EVALID,
Radek Krejci2efc45b2020-12-22 16:25:44 +0100176 "List instance is missing its key \"c\".", "Schema location /a:l1, data location /a:l1[a='a'][b='b'], line number 1.");
Michal Vasko9f96a052020-03-10 09:41:45 +0100177
178 /* key duplicate */
Radek Iša56ca9e42020-09-08 18:42:00 +0200179 PARSER_CHECK_ERROR("<l1 xmlns=\"urn:tests:a\"><c>1</c><b>b</b><a>a</a><c>1</c></l1>", 0, LYD_VALIDATE_PRESENT, tree, LY_EVALID,
Radek Krejci2efc45b2020-12-22 16:25:44 +0100180 "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.");
Michal Vasko9f96a052020-03-10 09:41:45 +0100181
Radek Krejci710226d2019-07-24 17:24:59 +0200182 /* keys order */
Radek Iša56ca9e42020-09-08 18:42:00 +0200183 CHECK_PARSE_LYD("<l1 xmlns=\"urn:tests:a\"><d>d</d><a>a</a><c>1</c><b>b</b></l1>", 0, LYD_VALIDATE_PRESENT, tree);
184 CHECK_LYSC_NODE(tree->schema, NULL, 0, LYS_CONFIG_W | LYS_STATUS_CURR | LYS_ORDBY_SYSTEM, 1, "l1",
185 1, LYS_LIST, 0, 0, NULL, 0);
Radek Krejcib4ac5a92020-11-23 17:54:33 +0100186 list = (struct lyd_node_inner *)tree;
187 assert_non_null(leaf = (struct lyd_node_term *)list->child);
Radek Iša56ca9e42020-09-08 18:42:00 +0200188 CHECK_LYSC_NODE(leaf->schema, NULL, 0, LYS_CONFIG_W | LYS_STATUS_CURR | LYS_KEY, 1, "a", 1, LYS_LEAF, 1, 0, NULL, 0);
Radek Krejcib4ac5a92020-11-23 17:54:33 +0100189 assert_non_null(leaf = (struct lyd_node_term *)leaf->next);
Radek Iša56ca9e42020-09-08 18:42:00 +0200190 CHECK_LYSC_NODE(leaf->schema, NULL, 0, LYS_CONFIG_W | LYS_STATUS_CURR | LYS_KEY, 1, "b", 1, LYS_LEAF, 1, 0, NULL, 0);
Radek Krejcib4ac5a92020-11-23 17:54:33 +0100191 assert_non_null(leaf = (struct lyd_node_term *)leaf->next);
Radek Iša56ca9e42020-09-08 18:42:00 +0200192 CHECK_LYSC_NODE(leaf->schema, NULL, 0, LYS_CONFIG_W | LYS_STATUS_CURR | LYS_KEY, 1, "c", 1, LYS_LEAF, 1, 0, NULL, 0);
Radek Krejcib4ac5a92020-11-23 17:54:33 +0100193 assert_non_null(leaf = (struct lyd_node_term *)leaf->next);
Radek Iša56ca9e42020-09-08 18:42:00 +0200194 CHECK_LYSC_NODE(leaf->schema, NULL, 0, LYS_CONFIG_W | LYS_STATUS_CURR, 1, "d", 0, LYS_LEAF, 1, 0, NULL, 0);
195 CHECK_LOG_CTX("Invalid position of the key \"b\" in a list.", NULL);
Radek Krejci710226d2019-07-24 17:24:59 +0200196 lyd_free_all(tree);
197
Michal Vasko44685da2020-03-17 15:38:06 +0100198 data = "<l1 xmlns=\"urn:tests:a\"><c>1</c><b>b</b><a>a</a></l1>";
Radek Iša56ca9e42020-09-08 18:42:00 +0200199 CHECK_PARSE_LYD(data, 0, LYD_VALIDATE_PRESENT, tree);
200 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);
Radek Krejcib4ac5a92020-11-23 17:54:33 +0100201 list = (struct lyd_node_inner *)tree;
202 assert_non_null(leaf = (struct lyd_node_term *)list->child);
Radek Iša56ca9e42020-09-08 18:42:00 +0200203 CHECK_LYSC_NODE(leaf->schema, NULL, 0, LYS_CONFIG_W | LYS_STATUS_CURR | LYS_KEY, 1, "a", 1, LYS_LEAF, 1, 0, NULL, 0);
Radek Krejcib4ac5a92020-11-23 17:54:33 +0100204 assert_non_null(leaf = (struct lyd_node_term *)leaf->next);
Radek Iša56ca9e42020-09-08 18:42:00 +0200205 CHECK_LYSC_NODE(leaf->schema, NULL, 0, LYS_CONFIG_W | LYS_STATUS_CURR | LYS_KEY, 1, "b", 1, LYS_LEAF, 1, 0, NULL, 0);
Radek Krejcib4ac5a92020-11-23 17:54:33 +0100206 assert_non_null(leaf = (struct lyd_node_term *)leaf->next);
Radek Iša56ca9e42020-09-08 18:42:00 +0200207 CHECK_LYSC_NODE(leaf->schema, NULL, 0, LYS_CONFIG_W | LYS_STATUS_CURR | LYS_KEY, 1, "c", 1, LYS_LEAF, 1, 0, NULL, 0);
208 CHECK_LOG_CTX("Invalid position of the key \"a\" in a list.", NULL);
Radek Krejci710226d2019-07-24 17:24:59 +0200209 lyd_free_all(tree);
210
Radek Iša56ca9e42020-09-08 18:42:00 +0200211 PARSER_CHECK_ERROR(data, LYD_PARSE_STRICT, LYD_VALIDATE_PRESENT, tree, LY_EVALID,
Radek Krejci2efc45b2020-12-22 16:25:44 +0100212 "Invalid position of the key \"b\" in a list.", "Schema location /a:l1/b, data location /a:b, line number 1.");
Radek Krejci1f05b6a2019-07-18 16:15:06 +0200213}
214
Radek Krejcib6f7ae52019-07-19 10:31:42 +0200215static void
216test_container(void **state)
217{
Radek Krejcib6f7ae52019-07-19 10:31:42 +0200218 struct lyd_node *tree;
219 struct lyd_node_inner *cont;
220
Radek Iša56ca9e42020-09-08 18:42:00 +0200221 CHECK_PARSE_LYD("<c xmlns=\"urn:tests:a\"/>", 0, LYD_VALIDATE_PRESENT, tree);
222 CHECK_LYSC_NODE(tree->schema, NULL, 0, LYS_CONFIG_W | LYS_STATUS_CURR, 1, "c", 1, LYS_CONTAINER, 0, 0, NULL, 0);
Radek Krejcib4ac5a92020-11-23 17:54:33 +0100223 cont = (struct lyd_node_inner *)tree;
Radek Krejcib6f7ae52019-07-19 10:31:42 +0200224 assert_true(cont->flags & LYD_DEFAULT);
225 lyd_free_all(tree);
226
Radek Iša56ca9e42020-09-08 18:42:00 +0200227 CHECK_PARSE_LYD("<cp xmlns=\"urn:tests:a\"/>", 0, LYD_VALIDATE_PRESENT, tree);
Radek Krejcib6f7ae52019-07-19 10:31:42 +0200228 assert_non_null(tree);
Michal Vasko26123192020-11-09 21:02:34 +0100229 tree = tree->next;
Michal Vaskoe16c7b72021-02-26 10:39:06 +0100230 CHECK_LYSC_NODE(tree->schema, NULL, 0, LYS_CONFIG_W | LYS_STATUS_CURR | LYS_PRESENCE, 1, "cp",
Radek Iša56ca9e42020-09-08 18:42:00 +0200231 1, LYS_CONTAINER, 0, 0, NULL, 0);
Radek Krejcib4ac5a92020-11-23 17:54:33 +0100232 cont = (struct lyd_node_inner *)tree;
Radek Krejcib6f7ae52019-07-19 10:31:42 +0200233 assert_false(cont->flags & LYD_DEFAULT);
234 lyd_free_all(tree);
Radek Krejcib6f7ae52019-07-19 10:31:42 +0200235}
236
Michal Vasko44685da2020-03-17 15:38:06 +0100237static void
238test_opaq(void **state)
239{
Michal Vasko44685da2020-03-17 15:38:06 +0100240 const char *data;
Michal Vasko44685da2020-03-17 15:38:06 +0100241 struct lyd_node *tree;
242
243 /* invalid value, no flags */
244 data = "<foo3 xmlns=\"urn:tests:a\"/>";
Radek Iša56ca9e42020-09-08 18:42:00 +0200245 PARSER_CHECK_ERROR(data, 0, LYD_VALIDATE_PRESENT, tree, LY_EVALID,
Radek Krejci2efc45b2020-12-22 16:25:44 +0100246 "Invalid empty uint32 value.", "Schema location /a:foo3, line number 1.");
Michal Vasko44685da2020-03-17 15:38:06 +0100247
248 /* opaq flag */
Radek Iša56ca9e42020-09-08 18:42:00 +0200249 CHECK_PARSE_LYD(data, LYD_PARSE_OPAQ | LYD_PARSE_ONLY, 0, tree);
Radek Krejci8df109d2021-04-23 12:19:08 +0200250 CHECK_LYD_NODE_OPAQ((struct lyd_node_opaq *)tree, 0, 0, LY_VALUE_XML, "foo3", 0, 0, NULL, 0, "");
Radek Iša56ca9e42020-09-08 18:42:00 +0200251 CHECK_LYD_STRING(tree, LYD_PRINT_WITHSIBLINGS, "<foo3 xmlns=\"urn:tests:a\"/>\n");
Michal Vasko44685da2020-03-17 15:38:06 +0100252 lyd_free_all(tree);
253
Michal Vaskoda8fbbf2021-06-16 11:44:44 +0200254 /* list, opaq flag */
255 data = "<l1 xmlns=\"urn:tests:a\"/>";
256 CHECK_PARSE_LYD(data, LYD_PARSE_OPAQ | LYD_PARSE_ONLY, 0, tree);
257 CHECK_LYD_NODE_OPAQ((struct lyd_node_opaq *)tree, 0, 0, LY_VALUE_XML, "l1", 0, 0, NULL, 0, "");
258 CHECK_LYD_STRING(tree, LYD_PRINT_WITHSIBLINGS, "<l1 xmlns=\"urn:tests:a\"/>\n");
259 lyd_free_all(tree);
260
Michal Vasko44685da2020-03-17 15:38:06 +0100261 /* missing key, no flags */
Radek Iša56ca9e42020-09-08 18:42:00 +0200262 data = "<l1 xmlns=\"urn:tests:a\">\n"
263 " <a>val_a</a>\n"
264 " <b>val_b</b>\n"
265 " <d>val_d</d>\n"
266 "</l1>\n";
267 PARSER_CHECK_ERROR(data, 0, LYD_VALIDATE_PRESENT, tree, LY_EVALID,
Radek Krejci2efc45b2020-12-22 16:25:44 +0100268 "List instance is missing its key \"c\".", "Schema location /a:l1, data location /a:l1[a='val_a'][b='val_b'], line number 5.");
Michal Vasko44685da2020-03-17 15:38:06 +0100269
270 /* opaq flag */
Radek Iša56ca9e42020-09-08 18:42:00 +0200271 CHECK_PARSE_LYD(data, LYD_PARSE_OPAQ | LYD_PARSE_ONLY, 0, tree);
Radek Krejci8df109d2021-04-23 12:19:08 +0200272 CHECK_LYD_NODE_OPAQ((struct lyd_node_opaq *)tree, 0, 0x1, LY_VALUE_XML, "l1", 0, 0, NULL, 0, "");
Radek Iša56ca9e42020-09-08 18:42:00 +0200273 CHECK_LYD_STRING(tree, LYD_PRINT_WITHSIBLINGS, data);
Michal Vasko44685da2020-03-17 15:38:06 +0100274 lyd_free_all(tree);
275
276 /* invalid key, no flags */
Radek Iša56ca9e42020-09-08 18:42:00 +0200277 data = "<l1 xmlns=\"urn:tests:a\">\n"
278 " <a>val_a</a>\n"
279 " <b>val_b</b>\n"
280 " <c>val_c</c>\n"
281 "</l1>\n";
282 PARSER_CHECK_ERROR(data, 0, LYD_VALIDATE_PRESENT, tree, LY_EVALID,
Radek Krejci2efc45b2020-12-22 16:25:44 +0100283 "Invalid int16 value \"val_c\".", "Schema location /a:l1/c, data location /a:l1[a='val_a'][b='val_b'], line number 4.");
Michal Vasko44685da2020-03-17 15:38:06 +0100284
285 /* opaq flag */
Radek Iša56ca9e42020-09-08 18:42:00 +0200286 CHECK_PARSE_LYD(data, LYD_PARSE_OPAQ | LYD_PARSE_ONLY, 0, tree);
Radek Krejci8df109d2021-04-23 12:19:08 +0200287 CHECK_LYD_NODE_OPAQ((struct lyd_node_opaq *)tree, 0, 0x1, LY_VALUE_XML, "l1", 0, 0, NULL, 0, "");
Radek Iša56ca9e42020-09-08 18:42:00 +0200288 CHECK_LYD_STRING(tree, LYD_PRINT_WITHSIBLINGS, data);
Michal Vasko44685da2020-03-17 15:38:06 +0100289 lyd_free_all(tree);
290
Michal Vasko413c7f22020-05-05 12:34:06 +0200291 /* opaq flag and fail */
Radek Iša56ca9e42020-09-08 18:42:00 +0200292 assert_int_equal(LY_EVALID, lyd_parse_data_mem(UTEST_LYCTX,
293 "<a xmlns=\"ns\">\n"
294 " <b>x</b>\n"
295 " <c xml:id=\"D\">1</c>\n"
296 "</a>\n",
297 LYD_XML, LYD_PARSE_OPAQ, LYD_VALIDATE_PRESENT, &tree));
298 CHECK_LOG_CTX("Unknown XML prefix \"xml\".", "Line number 3.");
Michal Vasko44685da2020-03-17 15:38:06 +0100299}
300
Michal Vaskob36053d2020-03-26 15:49:30 +0100301static void
302test_rpc(void **state)
303{
Michal Vaskob36053d2020-03-26 15:49:30 +0100304 const char *data;
Michal Vasko63f3d842020-07-08 10:10:14 +0200305 struct ly_in *in;
Michal Vaskob36053d2020-03-26 15:49:30 +0100306 struct lyd_node *tree, *op;
Michal Vasko1bf09392020-03-27 12:38:10 +0100307 const struct lyd_node *node;
Radek Iša56ca9e42020-09-08 18:42:00 +0200308 const char *dsc = "The <edit-config> operation loads all or part of a specified\n"
309 "configuration to the specified target configuration.";
310 const char *ref = "RFC 6241, Section 7.2";
311 const char *feats[] = {"writable-running", NULL};
Michal Vaskob36053d2020-03-26 15:49:30 +0100312
Radek Iša56ca9e42020-09-08 18:42:00 +0200313 assert_non_null((ly_ctx_load_module(UTEST_LYCTX, "ietf-netconf", "2011-06-01", feats)));
Radek Krejcib4ac5a92020-11-23 17:54:33 +0100314
Michal Vasko2552ea32020-12-08 15:32:34 +0100315 data = "<edit-config xmlns=\"urn:ietf:params:xml:ns:netconf:base:1.0\">\n"
316 " <target>\n"
317 " <running/>\n"
318 " </target>\n"
319 " <config xmlns:nc=\"urn:ietf:params:xml:ns:netconf:base:1.0\">\n"
320 " <l1 xmlns=\"urn:tests:a\" nc:operation=\"replace\">\n"
321 " <a>val_a</a>\n"
322 " <b>val_b</b>\n"
323 " <c>val_c</c>\n"
324 " </l1>\n"
325 " <cp xmlns=\"urn:tests:a\">\n"
326 " <z nc:operation=\"delete\"/>\n"
327 " </cp>\n"
328 " </config>\n"
329 "</edit-config>\n";
Michal Vasko63f3d842020-07-08 10:10:14 +0200330 assert_int_equal(LY_SUCCESS, ly_in_new_memory(data, &in));
Michal Vasko1e4c68e2021-02-18 15:03:01 +0100331 assert_int_equal(LY_SUCCESS, lyd_parse_op(UTEST_LYCTX, NULL, in, LYD_XML, LYD_TYPE_RPC_YANG, &tree, &op));
Michal Vasko63f3d842020-07-08 10:10:14 +0200332 ly_in_free(in, 0);
Michal Vasko1bf09392020-03-27 12:38:10 +0100333
334 assert_non_null(op);
Radek Iša56ca9e42020-09-08 18:42:00 +0200335
Radek Krejci2a9fc652021-01-22 17:44:34 +0100336 CHECK_LYSC_ACTION((struct lysc_node_action *)op->schema, dsc, 0, LYS_STATUS_CURR,
Radek Iša56ca9e42020-09-08 18:42:00 +0200337 1, 0, 0, 1, "edit-config", LYS_RPC,
338 0, 0, 0, 0, 0, ref, 0);
Michal Vasko1bf09392020-03-27 12:38:10 +0100339
Michal Vaskob36053d2020-03-26 15:49:30 +0100340 assert_non_null(tree);
Radek Iša56ca9e42020-09-08 18:42:00 +0200341
Michal Vasko2552ea32020-12-08 15:32:34 +0100342 node = tree;
Radek Krejci2a9fc652021-01-22 17:44:34 +0100343 CHECK_LYSC_ACTION((struct lysc_node_action *)node->schema, dsc, 0, LYS_STATUS_CURR,
Radek Iša56ca9e42020-09-08 18:42:00 +0200344 1, 0, 0, 1, "edit-config", LYS_RPC,
345 0, 0, 0, 0, 0, ref, 0);
Radek Krejcia1c1e542020-09-29 16:06:52 +0200346 node = lyd_child(node)->next;
Radek Iša56ca9e42020-09-08 18:42:00 +0200347 dsc = "Inline Config content.";
Michal Vaskod1e53b92021-01-28 13:11:06 +0100348 CHECK_LYSC_NODE(node->schema, dsc, 0, LYS_STATUS_CURR | LYS_IS_INPUT, 1, "config", 0, LYS_ANYXML, 1, 0, NULL, 0);
Michal Vaskob104f112020-07-17 09:54:54 +0200349
Michal Vasko1bf09392020-03-27 12:38:10 +0100350 node = ((struct lyd_node_any *)node)->value.tree;
Michal Vaskoe16c7b72021-02-26 10:39:06 +0100351 CHECK_LYSC_NODE(node->schema, NULL, 0, LYS_CONFIG_W | LYS_STATUS_CURR | LYS_PRESENCE, 1, "cp",
Radek Iša56ca9e42020-09-08 18:42:00 +0200352 1, LYS_CONTAINER, 0, 0, NULL, 0);
353
Radek Krejcia1c1e542020-09-29 16:06:52 +0200354 node = lyd_child(node);
Michal Vasko1bf09392020-03-27 12:38:10 +0100355 /* z has no value */
Radek Krejci8df109d2021-04-23 12:19:08 +0200356 CHECK_LYD_NODE_OPAQ((struct lyd_node_opaq *)node, 0x1, 0, LY_VALUE_XML, "z", 0, 0, NULL, 0, "");
Michal Vaskob104f112020-07-17 09:54:54 +0200357 node = node->parent->next;
358 /* l1 key c has invalid value so it is at the end */
Radek Krejci8df109d2021-04-23 12:19:08 +0200359 CHECK_LYD_NODE_OPAQ((struct lyd_node_opaq *)node, 0x1, 0x1, LY_VALUE_XML, "l1", 0, 0, NULL, 0, "");
Michal Vaskob36053d2020-03-26 15:49:30 +0100360
Radek Iša56ca9e42020-09-08 18:42:00 +0200361 CHECK_LYD_STRING(tree, LYD_PRINT_WITHSIBLINGS,
Michal Vasko2552ea32020-12-08 15:32:34 +0100362 "<edit-config xmlns=\"urn:ietf:params:xml:ns:netconf:base:1.0\">\n"
363 " <target>\n"
364 " <running/>\n"
365 " </target>\n"
366 " <config>\n"
367 " <cp xmlns=\"urn:tests:a\">\n"
368 " <z xmlns:nc=\"urn:ietf:params:xml:ns:netconf:base:1.0\" nc:operation=\"delete\"/>\n"
369 " </cp>\n"
370 " <l1 xmlns=\"urn:tests:a\" xmlns:nc=\"urn:ietf:params:xml:ns:netconf:base:1.0\" nc:operation=\"replace\">\n"
371 " <a>val_a</a>\n"
372 " <b>val_b</b>\n"
373 " <c>val_c</c>\n"
374 " </l1>\n"
375 " </config>\n"
376 "</edit-config>\n");
Radek Iša56ca9e42020-09-08 18:42:00 +0200377
Michal Vaskob36053d2020-03-26 15:49:30 +0100378 lyd_free_all(tree);
379
380 /* wrong namespace, element name, whatever... */
Michal Vaskoa8edff02020-03-27 14:47:01 +0100381 /* TODO */
Michal Vaskoa8edff02020-03-27 14:47:01 +0100382}
383
384static void
385test_action(void **state)
386{
Michal Vaskoa8edff02020-03-27 14:47:01 +0100387 const char *data;
Michal Vasko63f3d842020-07-08 10:10:14 +0200388 struct ly_in *in;
Michal Vaskoa8edff02020-03-27 14:47:01 +0100389 struct lyd_node *tree, *op;
Michal Vaskoa8edff02020-03-27 14:47:01 +0100390
Michal Vasko2552ea32020-12-08 15:32:34 +0100391 data = "<c xmlns=\"urn:tests:a\">\n"
392 " <act>\n"
393 " <al>value</al>\n"
394 " </act>\n"
395 "</c>\n";
Michal Vasko63f3d842020-07-08 10:10:14 +0200396 assert_int_equal(LY_SUCCESS, ly_in_new_memory(data, &in));
Michal Vasko1e4c68e2021-02-18 15:03:01 +0100397 assert_int_equal(LY_SUCCESS, lyd_parse_op(UTEST_LYCTX, NULL, in, LYD_XML, LYD_TYPE_RPC_YANG, &tree, &op));
Michal Vasko63f3d842020-07-08 10:10:14 +0200398 ly_in_free(in, 0);
Michal Vaskoa8edff02020-03-27 14:47:01 +0100399
400 assert_non_null(op);
Radek Krejci2a9fc652021-01-22 17:44:34 +0100401 CHECK_LYSC_ACTION((struct lysc_node_action *)op->schema, NULL, 0, LYS_STATUS_CURR,
Radek Iša56ca9e42020-09-08 18:42:00 +0200402 1, 0, 0, 1, "act", LYS_ACTION,
403 1, 0, 0, 1, 0, NULL, 0);
Michal Vaskoa8edff02020-03-27 14:47:01 +0100404
Radek Iša56ca9e42020-09-08 18:42:00 +0200405 CHECK_LYD_STRING(tree, LYD_PRINT_WITHSIBLINGS,
Michal Vasko2552ea32020-12-08 15:32:34 +0100406 "<c xmlns=\"urn:tests:a\">\n"
407 " <act>\n"
408 " <al>value</al>\n"
409 " </act>\n"
410 "</c>\n");
Radek Iša56ca9e42020-09-08 18:42:00 +0200411
Michal Vaskoa8edff02020-03-27 14:47:01 +0100412 lyd_free_all(tree);
413
414 /* wrong namespace, element name, whatever... */
415 /* TODO */
Michal Vaskoa8edff02020-03-27 14:47:01 +0100416}
417
418static void
419test_notification(void **state)
420{
Michal Vaskoa8edff02020-03-27 14:47:01 +0100421 const char *data;
Michal Vasko63f3d842020-07-08 10:10:14 +0200422 struct ly_in *in;
Michal Vaskoa8edff02020-03-27 14:47:01 +0100423 struct lyd_node *tree, *ntf;
Michal Vaskoa8edff02020-03-27 14:47:01 +0100424
Michal Vasko2552ea32020-12-08 15:32:34 +0100425 data = "<c xmlns=\"urn:tests:a\">\n"
426 " <n1>\n"
427 " <nl>value</nl>\n"
428 " </n1>\n"
429 "</c>\n";
Michal Vasko63f3d842020-07-08 10:10:14 +0200430 assert_int_equal(LY_SUCCESS, ly_in_new_memory(data, &in));
Michal Vasko1e4c68e2021-02-18 15:03:01 +0100431 assert_int_equal(LY_SUCCESS, lyd_parse_op(UTEST_LYCTX, NULL, in, LYD_XML, LYD_TYPE_NOTIF_YANG, &tree, &ntf));
Michal Vasko63f3d842020-07-08 10:10:14 +0200432 ly_in_free(in, 0);
Michal Vaskoa8edff02020-03-27 14:47:01 +0100433
434 assert_non_null(ntf);
Radek Krejci2a9fc652021-01-22 17:44:34 +0100435 CHECK_LYSC_NOTIF((struct lysc_node_notif *)ntf->schema, 1, NULL, 0, 0x4, 1, 0, "n1", 1, 0, NULL, 0);
Michal Vaskoa8edff02020-03-27 14:47:01 +0100436
Michal Vasko2552ea32020-12-08 15:32:34 +0100437 CHECK_LYSC_NODE(tree->schema, NULL, 0, LYS_CONFIG_W | LYS_STATUS_CURR, 1, "c", 1, LYS_CONTAINER, 0, 0, NULL, 0);
Michal Vaskoa8edff02020-03-27 14:47:01 +0100438
Radek Iša56ca9e42020-09-08 18:42:00 +0200439 CHECK_LYD_STRING(tree, LYD_PRINT_WITHSIBLINGS, data);
Michal Vaskoa8edff02020-03-27 14:47:01 +0100440 lyd_free_all(tree);
441
442 /* top-level notif without envelope */
Radek Iša56ca9e42020-09-08 18:42:00 +0200443 data = "<n2 xmlns=\"urn:tests:a\"/>\n";
Michal Vasko63f3d842020-07-08 10:10:14 +0200444 assert_int_equal(LY_SUCCESS, ly_in_new_memory(data, &in));
Michal Vasko1e4c68e2021-02-18 15:03:01 +0100445 assert_int_equal(LY_SUCCESS, lyd_parse_op(UTEST_LYCTX, NULL, in, LYD_XML, LYD_TYPE_NOTIF_YANG, &tree, &ntf));
Michal Vasko63f3d842020-07-08 10:10:14 +0200446 ly_in_free(in, 0);
Michal Vaskoa8edff02020-03-27 14:47:01 +0100447
448 assert_non_null(ntf);
Radek Krejci2a9fc652021-01-22 17:44:34 +0100449 CHECK_LYSC_NOTIF((struct lysc_node_notif *)ntf->schema, 0, NULL, 0, 0x4, 1, 0, "n2", 0, 0, NULL, 0);
Michal Vaskoa8edff02020-03-27 14:47:01 +0100450
451 assert_non_null(tree);
452 assert_ptr_equal(ntf, tree);
453
Radek Iša56ca9e42020-09-08 18:42:00 +0200454 CHECK_LYD_STRING(tree, LYD_PRINT_WITHSIBLINGS, data);
Michal Vaskoa8edff02020-03-27 14:47:01 +0100455 lyd_free_all(tree);
456
457 /* wrong namespace, element name, whatever... */
458 /* TODO */
Michal Vaskob36053d2020-03-26 15:49:30 +0100459}
460
Michal Vasko1ce933a2020-03-30 12:38:22 +0200461static void
462test_reply(void **state)
463{
Michal Vasko1ce933a2020-03-30 12:38:22 +0200464 const char *data;
Michal Vasko63f3d842020-07-08 10:10:14 +0200465 struct ly_in *in;
Michal Vasko2552ea32020-12-08 15:32:34 +0100466 struct lyd_node *tree, *op;
Michal Vasko1ce933a2020-03-30 12:38:22 +0200467 const struct lyd_node *node;
468
Michal Vasko79135ae2020-12-16 10:08:35 +0100469 data = "<c xmlns=\"urn:tests:a\">\n"
Radek Krejcib4ac5a92020-11-23 17:54:33 +0100470 " <act>\n"
Michal Vasko2552ea32020-12-08 15:32:34 +0100471 " <al>25</al>\n"
Radek Krejcib4ac5a92020-11-23 17:54:33 +0100472 " </act>\n"
473 "</c>\n";
Michal Vasko63f3d842020-07-08 10:10:14 +0200474 assert_int_equal(LY_SUCCESS, ly_in_new_memory(data, &in));
Michal Vasko1e4c68e2021-02-18 15:03:01 +0100475 assert_int_equal(LY_SUCCESS, lyd_parse_op(UTEST_LYCTX, NULL, in, LYD_XML, LYD_TYPE_REPLY_YANG, &tree, &op));
Michal Vasko63f3d842020-07-08 10:10:14 +0200476 ly_in_free(in, 0);
477
Michal Vasko1ce933a2020-03-30 12:38:22 +0200478 assert_non_null(op);
Radek Iša56ca9e42020-09-08 18:42:00 +0200479
Radek Krejci2a9fc652021-01-22 17:44:34 +0100480 CHECK_LYSC_ACTION((struct lysc_node_action *)op->schema, NULL, 0, LYS_STATUS_CURR,
Radek Iša56ca9e42020-09-08 18:42:00 +0200481 1, 0, 0, 1, "act", LYS_ACTION,
482 1, 0, 0, 1, 0, NULL, 0);
Radek Krejcia1c1e542020-09-29 16:06:52 +0200483 node = lyd_child(op);
Michal Vaskod1e53b92021-01-28 13:11:06 +0100484 CHECK_LYSC_NODE(node->schema, NULL, 0, LYS_STATUS_CURR | LYS_IS_OUTPUT, 1, "al", 0, LYS_LEAF, 1, 0, NULL, 0);
Michal Vasko1ce933a2020-03-30 12:38:22 +0200485
Michal Vasko2552ea32020-12-08 15:32:34 +0100486 CHECK_LYSC_NODE(tree->schema, NULL, 0, LYS_CONFIG_W | LYS_STATUS_CURR, 1, "c", 1, LYS_CONTAINER, 0, 0, NULL, 0);
Michal Vasko1ce933a2020-03-30 12:38:22 +0200487
488 /* TODO print only rpc-reply node and then output subtree */
Radek Iša56ca9e42020-09-08 18:42:00 +0200489 CHECK_LYD_STRING(lyd_child(op), LYD_PRINT_WITHSIBLINGS, "<al xmlns=\"urn:tests:a\">25</al>\n");
Michal Vasko1ce933a2020-03-30 12:38:22 +0200490 lyd_free_all(tree);
491
492 /* wrong namespace, element name, whatever... */
493 /* TODO */
Michal Vasko1ce933a2020-03-30 12:38:22 +0200494}
495
Michal Vaskoe0665742021-02-11 11:08:44 +0100496static void
497test_netconf_rpc(void **state)
498{
499 const char *data;
500 struct ly_in *in;
501 struct lyd_node *tree, *op;
502 const struct lyd_node *node;
503 const char *dsc = "The <edit-config> operation loads all or part of a specified\n"
504 "configuration to the specified target configuration.";
505 const char *ref = "RFC 6241, Section 7.2";
506 const char *feats[] = {"writable-running", NULL};
507
508 assert_non_null((ly_ctx_load_module(UTEST_LYCTX, "ietf-netconf", "2011-06-01", feats)));
509
510 data = "<rpc message-id=\"25\" xmlns=\"urn:ietf:params:xml:ns:netconf:base:1.0\">"
511 "<edit-config>\n"
512 " <target>\n"
513 " <running/>\n"
514 " </target>\n"
515 " <config xmlns:nc=\"urn:ietf:params:xml:ns:netconf:base:1.0\">\n"
516 " <l1 xmlns=\"urn:tests:a\" nc:operation=\"replace\">\n"
517 " <a>val_a</a>\n"
518 " <b>val_b</b>\n"
519 " <c>val_c</c>\n"
520 " </l1>\n"
521 " <cp xmlns=\"urn:tests:a\">\n"
522 " <z nc:operation=\"delete\"/>\n"
523 " </cp>\n"
524 " </config>\n"
525 "</edit-config>\n"
526 "</rpc>\n";
527 assert_int_equal(LY_SUCCESS, ly_in_new_memory(data, &in));
Michal Vasko1e4c68e2021-02-18 15:03:01 +0100528 assert_int_equal(LY_SUCCESS, lyd_parse_op(UTEST_LYCTX, NULL, in, LYD_XML, LYD_TYPE_RPC_NETCONF, &tree, &op));
Michal Vaskoe0665742021-02-11 11:08:44 +0100529 ly_in_free(in, 0);
530
531 assert_non_null(op);
532
533 node = tree;
Radek Krejci8df109d2021-04-23 12:19:08 +0200534 CHECK_LYD_NODE_OPAQ((struct lyd_node_opaq *)node, 1, 0, LY_VALUE_XML, "rpc", 0, 0, 0, 0, "");
Michal Vaskoe0665742021-02-11 11:08:44 +0100535
536 assert_non_null(tree);
537
538 node = op;
539 CHECK_LYSC_ACTION((struct lysc_node_action *)node->schema, dsc, 0, LYS_STATUS_CURR,
540 1, 0, 0, 1, "edit-config", LYS_RPC,
541 0, 0, 0, 0, 0, ref, 0);
542 node = lyd_child(node)->next;
543 dsc = "Inline Config content.";
544 CHECK_LYSC_NODE(node->schema, dsc, 0, LYS_STATUS_CURR | LYS_IS_INPUT, 1, "config", 0, LYS_ANYXML, 1, 0, NULL, 0);
545
546 node = ((struct lyd_node_any *)node)->value.tree;
Michal Vaskoe16c7b72021-02-26 10:39:06 +0100547 CHECK_LYSC_NODE(node->schema, NULL, 0, LYS_CONFIG_W | LYS_STATUS_CURR | LYS_PRESENCE, 1, "cp",
Michal Vaskoe0665742021-02-11 11:08:44 +0100548 1, LYS_CONTAINER, 0, 0, NULL, 0);
549
550 node = lyd_child(node);
551 /* z has no value */
Radek Krejci8df109d2021-04-23 12:19:08 +0200552 CHECK_LYD_NODE_OPAQ((struct lyd_node_opaq *)node, 0x1, 0, LY_VALUE_XML, "z", 0, 0, NULL, 0, "");
Michal Vaskoe0665742021-02-11 11:08:44 +0100553 node = node->parent->next;
554 /* l1 key c has invalid value so it is at the end */
Radek Krejci8df109d2021-04-23 12:19:08 +0200555 CHECK_LYD_NODE_OPAQ((struct lyd_node_opaq *)node, 0x1, 0x1, LY_VALUE_XML, "l1", 0, 0, NULL, 0, "");
Michal Vaskoe0665742021-02-11 11:08:44 +0100556
557 CHECK_LYD_STRING(tree, LYD_PRINT_WITHSIBLINGS,
558 "<rpc xmlns=\"urn:ietf:params:xml:ns:netconf:base:1.0\" message-id=\"25\"/>\n");
559 CHECK_LYD_STRING(op, LYD_PRINT_WITHSIBLINGS,
560 "<edit-config xmlns=\"urn:ietf:params:xml:ns:netconf:base:1.0\">\n"
561 " <target>\n"
562 " <running/>\n"
563 " </target>\n"
564 " <config>\n"
565 " <cp xmlns=\"urn:tests:a\">\n"
566 " <z xmlns:nc=\"urn:ietf:params:xml:ns:netconf:base:1.0\" nc:operation=\"delete\"/>\n"
567 " </cp>\n"
568 " <l1 xmlns=\"urn:tests:a\" xmlns:nc=\"urn:ietf:params:xml:ns:netconf:base:1.0\" nc:operation=\"replace\">\n"
569 " <a>val_a</a>\n"
570 " <b>val_b</b>\n"
571 " <c>val_c</c>\n"
572 " </l1>\n"
573 " </config>\n"
574 "</edit-config>\n");
575
576 lyd_free_all(tree);
577 lyd_free_all(op);
578
579 /* wrong namespace, element name, whatever... */
580 /* TODO */
581}
582
583static void
584test_netconf_action(void **state)
585{
586 const char *data;
587 struct ly_in *in;
588 struct lyd_node *tree, *op;
589
590 data = "<rpc message-id=\"25\" xmlns=\"urn:ietf:params:xml:ns:netconf:base:1.0\">"
591 "<action xmlns=\"urn:ietf:params:xml:ns:yang:1\">"
592 "<c xmlns=\"urn:tests:a\">\n"
593 " <act>\n"
594 " <al>value</al>\n"
595 " </act>\n"
596 "</c>\n"
597 "</action>\n"
598 "</rpc>\n";
599 assert_int_equal(LY_SUCCESS, ly_in_new_memory(data, &in));
Michal Vasko1e4c68e2021-02-18 15:03:01 +0100600 assert_int_equal(LY_SUCCESS, lyd_parse_op(UTEST_LYCTX, NULL, in, LYD_XML, LYD_TYPE_RPC_NETCONF, &tree, &op));
Michal Vaskoe0665742021-02-11 11:08:44 +0100601 ly_in_free(in, 0);
602
Radek Krejci8df109d2021-04-23 12:19:08 +0200603 CHECK_LYD_NODE_OPAQ((struct lyd_node_opaq *)tree, 1, 1, LY_VALUE_XML, "rpc", 0, 0, 0, 0, "");
604 CHECK_LYD_NODE_OPAQ((struct lyd_node_opaq *)lyd_child(tree), 0, 0, LY_VALUE_XML, "action", 0, 0, 0, 0, "");
Michal Vaskoe0665742021-02-11 11:08:44 +0100605
606 assert_non_null(op);
607 CHECK_LYSC_ACTION((struct lysc_node_action *)op->schema, NULL, 0, LYS_STATUS_CURR,
608 1, 0, 0, 1, "act", LYS_ACTION,
609 1, 0, 0, 1, 0, NULL, 0);
610
611 CHECK_LYD_STRING(tree, LYD_PRINT_WITHSIBLINGS,
612 "<rpc xmlns=\"urn:ietf:params:xml:ns:netconf:base:1.0\" message-id=\"25\">\n"
613 " <action xmlns=\"urn:ietf:params:xml:ns:yang:1\"/>\n"
614 "</rpc>\n");
615 CHECK_LYD_STRING(op, LYD_PRINT_WITHSIBLINGS,
616 "<act xmlns=\"urn:tests:a\">\n"
617 " <al>value</al>\n"
618 "</act>\n");
619
620 lyd_free_all(tree);
621 lyd_free_all(op);
622
623 /* wrong namespace, element name, whatever... */
624 /* TODO */
625}
626
627static void
628test_netconf_reply_or_notification(void **state)
629{
630 const char *data;
631 struct ly_in *in;
632 struct lyd_node *action, *tree, *op, *op2;
633
634 /* parse the action */
635 data = "<c xmlns=\"urn:tests:a\">\n"
636 " <act>\n"
637 " <al>value</al>\n"
638 " </act>\n"
639 "</c>\n";
640 assert_int_equal(LY_SUCCESS, ly_in_new_memory(data, &in));
Michal Vasko1e4c68e2021-02-18 15:03:01 +0100641 assert_int_equal(LY_SUCCESS, lyd_parse_op(UTEST_LYCTX, NULL, in, LYD_XML, LYD_TYPE_RPC_YANG, &action, &op));
Michal Vaskoe0665742021-02-11 11:08:44 +0100642 ly_in_free(in, 0);
643
644 /* parse notification first */
645 data = "<notification xmlns=\"urn:ietf:params:xml:ns:netconf:notification:1.0\">\n"
646 "<eventTime>2010-12-06T08:00:01Z</eventTime>\n"
647 "<c xmlns=\"urn:tests:a\">\n"
648 " <n1>\n"
649 " <nl>value</nl>\n"
650 " </n1>\n"
651 "</c>\n"
652 "</notification>\n";
653 assert_int_equal(LY_SUCCESS, ly_in_new_memory(data, &in));
Michal Vasko1e4c68e2021-02-18 15:03:01 +0100654 assert_int_equal(LY_SUCCESS, lyd_parse_op(UTEST_LYCTX, NULL, in, LYD_XML, LYD_TYPE_NOTIF_NETCONF, &tree, &op2));
Michal Vaskoe0665742021-02-11 11:08:44 +0100655 ly_in_free(in, 0);
656
Radek Krejci8df109d2021-04-23 12:19:08 +0200657 CHECK_LYD_NODE_OPAQ((struct lyd_node_opaq *)tree, 0, 1, LY_VALUE_XML, "notification", 0, 0, 0, 0, "");
658 CHECK_LYD_NODE_OPAQ((struct lyd_node_opaq *)lyd_child(tree), 0, 0, LY_VALUE_XML, "eventTime", 0, 0, 0, 0,
Michal Vaskoe0665742021-02-11 11:08:44 +0100659 "2010-12-06T08:00:01Z");
660
661 assert_non_null(op2);
662 CHECK_LYSC_NOTIF((struct lysc_node_notif *)op2->schema, 1, NULL, 0, 0x4, 1, 0, "n1", 1, 0, NULL, 0);
663
664 CHECK_LYD_STRING(tree, LYD_PRINT_WITHSIBLINGS,
665 "<notification xmlns=\"urn:ietf:params:xml:ns:netconf:notification:1.0\">\n"
666 " <eventTime>2010-12-06T08:00:01Z</eventTime>\n"
667 "</notification>\n");
668 CHECK_LYD_STRING(op2, LYD_PRINT_WITHSIBLINGS,
669 "<n1 xmlns=\"urn:tests:a\">\n"
670 " <nl>value</nl>\n"
671 "</n1>\n");
672
673 lyd_free_all(tree);
674 lyd_free_all(op2);
675
676 /* parse a data reply */
677 data = "<rpc-reply message-id=\"55\" xmlns=\"urn:ietf:params:xml:ns:netconf:base:1.0\">\n"
678 " <al xmlns=\"urn:tests:a\">25</al>\n"
679 "</rpc-reply>\n";
680 assert_int_equal(LY_SUCCESS, ly_in_new_memory(data, &in));
Michal Vasko1e4c68e2021-02-18 15:03:01 +0100681 assert_int_equal(LY_SUCCESS, lyd_parse_op(UTEST_LYCTX, op, in, LYD_XML, LYD_TYPE_REPLY_NETCONF, &tree, NULL));
Michal Vaskoe0665742021-02-11 11:08:44 +0100682 ly_in_free(in, 0);
683
Radek Krejci8df109d2021-04-23 12:19:08 +0200684 CHECK_LYD_NODE_OPAQ((struct lyd_node_opaq *)tree, 1, 0, LY_VALUE_XML, "rpc-reply", 0, 0, 0, 0, "");
Michal Vaskoe0665742021-02-11 11:08:44 +0100685
Michal Vaskoe0665742021-02-11 11:08:44 +0100686 CHECK_LYD_STRING(tree, LYD_PRINT_WITHSIBLINGS,
687 "<rpc-reply xmlns=\"urn:ietf:params:xml:ns:netconf:base:1.0\" message-id=\"55\"/>\n");
Michal Vaskoe0665742021-02-11 11:08:44 +0100688
689 lyd_free_all(tree);
690 /* it was connected to the action, do not free */
691
692 /* parse an ok reply */
693 data = "<rpc-reply xmlns=\"urn:ietf:params:xml:ns:netconf:base:1.0\" message-id=\"55\">\n"
694 " <ok/>\n"
695 "</rpc-reply>\n";
696 assert_int_equal(LY_SUCCESS, ly_in_new_memory(data, &in));
Michal Vasko1e4c68e2021-02-18 15:03:01 +0100697 assert_int_equal(LY_SUCCESS, lyd_parse_op(UTEST_LYCTX, op, in, LYD_XML, LYD_TYPE_REPLY_NETCONF, &tree, NULL));
Michal Vaskoe0665742021-02-11 11:08:44 +0100698 ly_in_free(in, 0);
699
Radek Krejci8df109d2021-04-23 12:19:08 +0200700 CHECK_LYD_NODE_OPAQ((struct lyd_node_opaq *)tree, 1, 1, LY_VALUE_XML, "rpc-reply", 0, 0, 0, 0, "");
701 CHECK_LYD_NODE_OPAQ((struct lyd_node_opaq *)lyd_child(tree), 0, 0, LY_VALUE_XML, "ok", 0, 0, 0, 0, "");
Michal Vaskoe0665742021-02-11 11:08:44 +0100702
Michal Vaskoe0665742021-02-11 11:08:44 +0100703 CHECK_LYD_STRING(tree, LYD_PRINT_WITHSIBLINGS, data);
704
705 lyd_free_all(tree);
706
707 /* parse an error reply */
708 data = "<rpc-reply xmlns=\"urn:ietf:params:xml:ns:netconf:base:1.0\" message-id=\"55\">\n"
709 " <rpc-error>\n"
710 " <error-type>rpc</error-type>\n"
711 " <error-tag>missing-attribute</error-tag>\n"
712 " <error-severity>error</error-severity>\n"
713 " <error-info>\n"
714 " <bad-attribute>message-id</bad-attribute>\n"
715 " <bad-element>rpc</bad-element>\n"
716 " </error-info>\n"
717 " </rpc-error>\n"
718 "</rpc-reply>\n";
719 assert_int_equal(LY_SUCCESS, ly_in_new_memory(data, &in));
Michal Vasko1e4c68e2021-02-18 15:03:01 +0100720 assert_int_equal(LY_SUCCESS, lyd_parse_op(UTEST_LYCTX, op, in, LYD_XML, LYD_TYPE_REPLY_NETCONF, &tree, NULL));
Michal Vaskoe0665742021-02-11 11:08:44 +0100721 ly_in_free(in, 0);
722
Radek Krejci8df109d2021-04-23 12:19:08 +0200723 CHECK_LYD_NODE_OPAQ((struct lyd_node_opaq *)tree, 1, 1, LY_VALUE_XML, "rpc-reply", 0, 0, 0, 0, "");
724 CHECK_LYD_NODE_OPAQ((struct lyd_node_opaq *)lyd_child(tree), 0, 1, LY_VALUE_XML, "rpc-error", 0, 0, 0, 0, "");
Michal Vaskoe0665742021-02-11 11:08:44 +0100725
Michal Vaskoe0665742021-02-11 11:08:44 +0100726 CHECK_LYD_STRING(tree, LYD_PRINT_WITHSIBLINGS, data);
727
728 lyd_free_all(tree);
729
730 lyd_free_all(action);
731
732 /* wrong namespace, element name, whatever... */
733 /* TODO */
734}
735
aPiecek9cdb9e62021-05-18 09:46:20 +0200736static void
737test_data_skip(void **state)
738{
739 const char *data;
740 struct lyd_node *tree;
741 struct lyd_node_term *leaf;
742
743 /* add invalid data to a module that is not implemented */
744 data = "<foo xmlns=\"urn:ietf:params:xml:ns:yang:ietf-yang-metadata\"><u/></foo>";
745 assert_int_equal(LY_SUCCESS, lyd_parse_data_mem(_UC->ctx, data, LYD_XML, 0, LYD_VALIDATE_PRESENT, &tree));
746 assert_null(tree);
747
748 /* add invalid data to a module that is implemented */
749 data = "<fooX xmlns=\"urn:tests:a\"><u/><list><value/></list></fooX>";
750 assert_int_equal(LY_SUCCESS, lyd_parse_data_mem(_UC->ctx, data, LYD_XML, 0, LYD_VALIDATE_PRESENT, &tree));
751 assert_null(tree);
752
753 /* first invalid, next valid */
754 data = "<fooX xmlns=\"urn:tests:a\"><u/></fooX> <foo xmlns=\"urn:tests:a\">foo value</foo>";
755 CHECK_PARSE_LYD(data, 0, LYD_VALIDATE_PRESENT, tree);
756 CHECK_LYSC_NODE(tree->schema, NULL, 0, LYS_CONFIG_W | LYS_STATUS_CURR, 1, "foo", 1, LYS_LEAF, 0, 0, NULL, 0);
757 leaf = (struct lyd_node_term *)tree;
758 CHECK_LYD_VALUE(leaf->value, STRING, "foo value");
759 lyd_free_all(tree);
760}
761
Radek Krejcib4ac5a92020-11-23 17:54:33 +0100762int
763main(void)
Radek Krejci509e2592019-05-15 16:30:48 +0200764{
765 const struct CMUnitTest tests[] = {
Radek Iša56ca9e42020-09-08 18:42:00 +0200766 UTEST(test_leaf, setup),
767 UTEST(test_anydata, setup),
768 UTEST(test_list, setup),
769 UTEST(test_container, setup),
770 UTEST(test_opaq, setup),
771 UTEST(test_rpc, setup),
772 UTEST(test_action, setup),
773 UTEST(test_notification, setup),
774 UTEST(test_reply, setup),
Michal Vaskoe0665742021-02-11 11:08:44 +0100775 UTEST(test_netconf_rpc, setup),
776 UTEST(test_netconf_action, setup),
777 UTEST(test_netconf_reply_or_notification, setup),
aPiecek9cdb9e62021-05-18 09:46:20 +0200778 UTEST(test_data_skip, setup),
Radek Krejci509e2592019-05-15 16:30:48 +0200779 };
780
781 return cmocka_run_group_tests(tests, NULL, NULL);
782}