blob: 040b99371cb3404d236de56ed923e75ade53e350 [file] [log] [blame]
Radek Krejci1e880032018-09-20 12:17:12 +02001/*
2 * @file set.c
3 * @author: Radek Krejci <rkrejci@cesnet.cz>
4 * @brief unit tests for functions from context.c
5 *
6 * Copyright (c) 2018 CESNET, z.s.p.o.
7 *
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 */
14
15#include "tests/config.h"
16#include "../../src/context.c"
Radek Krejcib7db73a2018-10-24 14:18:40 +020017#include "../../src/tree_schema.c"
Radek Krejci1e880032018-09-20 12:17:12 +020018
19#include <stdarg.h>
20#include <stddef.h>
21#include <setjmp.h>
22#include <cmocka.h>
23
24#include <string.h>
25#include <stdio.h>
26
27#include "libyang.h"
28
29#define BUFSIZE 1024
30char logbuf[BUFSIZE] = {0};
Radek Krejcifaa1eac2018-10-30 14:34:55 +010031int store = -1; /* negative for infinite logging, positive for limited logging */
Radek Krejci1e880032018-09-20 12:17:12 +020032
33/* set to 0 to printing error messages to stderr instead of checking them in code */
34#define ENABLE_LOGGER_CHECKING 1
35
36static void
37logger(LY_LOG_LEVEL level, const char *msg, const char *path)
38{
39 (void) level; /* unused */
Radek Krejcifaa1eac2018-10-30 14:34:55 +010040 if (store) {
Radek Krejci9ed7a192018-10-31 16:23:51 +010041 if (path && path[0]) {
Radek Krejcifaa1eac2018-10-30 14:34:55 +010042 snprintf(logbuf, BUFSIZE - 1, "%s %s", msg, path);
43 } else {
44 strncpy(logbuf, msg, BUFSIZE - 1);
45 }
46 if (store > 0) {
47 --store;
48 }
49 }
Radek Krejci1e880032018-09-20 12:17:12 +020050}
51
52static int
53logger_setup(void **state)
54{
55 (void) state; /* unused */
56#if ENABLE_LOGGER_CHECKING
Radek Krejcifaa1eac2018-10-30 14:34:55 +010057 ly_set_log_clb(logger, 1);
Radek Krejci1e880032018-09-20 12:17:12 +020058#endif
59 return 0;
60}
61
62#if ENABLE_LOGGER_CHECKING
63# define logbuf_assert(str) assert_string_equal(logbuf, str)
64#else
65# define logbuf_assert(str)
66#endif
67
68int __real_ly_set_add(struct ly_set *set, void *object, int options);
69int __wrap_ly_set_add(struct ly_set *set, void *object, int options)
70{
71 int wrap = mock_type(int);
72
73 if (wrap) {
74 /* error */
75 return -1;
76 } else {
77 return __real_ly_set_add(set, object, options);
78 }
79}
80
81static void
82test_searchdirs(void **state)
83{
84 (void) state; /* unused */
85
86 struct ly_ctx *ctx;
87 const char * const *list;
88
Radek Krejcib7db73a2018-10-24 14:18:40 +020089 will_return_count(__wrap_ly_set_add, 0, 6);
Radek Krejci1e880032018-09-20 12:17:12 +020090 assert_int_equal(LY_SUCCESS, ly_ctx_new(NULL, 0, &ctx));
91
92 /* invalid arguments */
93 assert_int_equal(LY_EINVAL, ly_ctx_set_searchdir(NULL, NULL));
94 logbuf_assert("Invalid argument ctx (ly_ctx_set_searchdir()).");
95 assert_null(ly_ctx_get_searchdirs(NULL));
96 logbuf_assert("Invalid argument ctx (ly_ctx_get_searchdirs()).");
Radek Krejci0759b792018-09-20 13:53:15 +020097 assert_int_equal(LY_EINVAL, ly_ctx_unset_searchdirs(NULL, NULL));
Radek Krejci1e880032018-09-20 12:17:12 +020098 logbuf_assert("Invalid argument ctx (ly_ctx_unset_searchdirs()).");
Radek Krejci1e880032018-09-20 12:17:12 +020099
100 /* readable and executable, but not a directory */
101 assert_int_equal(LY_EINVAL, ly_ctx_set_searchdir(ctx, TESTS_BIN"/src_context"));
102 logbuf_assert("Given search directory \""TESTS_BIN"/src_context\" is not a directory.");
103 /* not executable */
Radek Krejci9efa87a2018-09-26 15:14:03 +0200104 assert_int_equal(LY_EINVAL, ly_ctx_set_searchdir(ctx, __FILE__));
105 logbuf_assert("Unable to use search directory \""__FILE__"\" (Permission denied)");
Radek Krejci1e880032018-09-20 12:17:12 +0200106 /* not existing */
107 assert_int_equal(LY_EINVAL, ly_ctx_set_searchdir(ctx, "/nonexistingfile"));
108 logbuf_assert("Unable to use search directory \"/nonexistingfile\" (No such file or directory)");
109
110 /* ly_set_add() fails */
111 will_return(__wrap_ly_set_add, 1);
112 assert_int_equal(LY_EMEM, ly_ctx_set_searchdir(ctx, TESTS_BIN"/src"));
113
Radek Krejcia77e0e12018-09-20 12:39:15 +0200114 /* no change */
115 assert_int_equal(LY_SUCCESS, ly_ctx_set_searchdir(ctx, NULL));
116
Radek Krejci1e880032018-09-20 12:17:12 +0200117 /* correct path */
118 will_return_always(__wrap_ly_set_add, 0);
119 assert_int_equal(LY_SUCCESS, ly_ctx_set_searchdir(ctx, TESTS_BIN"/src"));
120 assert_int_equal(1, ctx->search_paths.count);
121 assert_string_equal(TESTS_BIN"/src", ctx->search_paths.objs[0]);
122
Radek Krejci14946ab2018-09-20 13:42:06 +0200123 /* duplicated paths */
124 assert_int_equal(LY_EEXIST, ly_ctx_set_searchdir(ctx, TESTS_BIN"/src"));
Radek Krejci1e880032018-09-20 12:17:12 +0200125 assert_int_equal(1, ctx->search_paths.count);
126 assert_string_equal(TESTS_BIN"/src", ctx->search_paths.objs[0]);
127
128 /* another paths - add 8 to fill the initial buffer of the searchpaths list */
129 assert_int_equal(LY_SUCCESS, ly_ctx_set_searchdir(ctx, TESTS_BIN"/CMakeFiles"));
130 assert_int_equal(LY_SUCCESS, ly_ctx_set_searchdir(ctx, TESTS_SRC"/../src"));
131 assert_int_equal(LY_SUCCESS, ly_ctx_set_searchdir(ctx, TESTS_SRC"/../CMakeModules"));
132 assert_int_equal(LY_SUCCESS, ly_ctx_set_searchdir(ctx, TESTS_SRC"/../doc"));
133 assert_int_equal(LY_SUCCESS, ly_ctx_set_searchdir(ctx, TESTS_SRC));
134 assert_int_equal(LY_SUCCESS, ly_ctx_set_searchdir(ctx, TESTS_BIN));
135 assert_int_equal(LY_SUCCESS, ly_ctx_set_searchdir(ctx, "/tmp"));
136 assert_int_equal(8, ctx->search_paths.count);
137
138 /* get searchpaths */
139 list = ly_ctx_get_searchdirs(ctx);
140 assert_non_null(list);
141 assert_string_equal(TESTS_BIN"/src", list[0]);
142 assert_string_equal(TESTS_BIN"/CMakeFiles", list[1]);
143 assert_string_equal(TESTS_SRC, list[5]);
144 assert_string_equal(TESTS_BIN, list[6]);
145 assert_string_equal("/tmp", list[7]);
146 assert_null(list[8]);
147
148 /* removing searchpaths */
Radek Krejci0759b792018-09-20 13:53:15 +0200149 /* nonexisting */
150 assert_int_equal(LY_EINVAL, ly_ctx_unset_searchdirs(ctx, "/nonexistingfile"));
151 logbuf_assert("Invalid argument value (ly_ctx_unset_searchdirs()).");
Radek Krejci1e880032018-09-20 12:17:12 +0200152 /* first */
Radek Krejci0759b792018-09-20 13:53:15 +0200153 assert_int_equal(LY_SUCCESS, ly_ctx_unset_searchdirs(ctx, TESTS_BIN"/src"));
Radek Krejci1e880032018-09-20 12:17:12 +0200154 assert_string_not_equal(TESTS_BIN"/src", list[0]);
155 assert_int_equal(7, ctx->search_paths.count);
156 /* middle */
Radek Krejci0759b792018-09-20 13:53:15 +0200157 assert_int_equal(LY_SUCCESS, ly_ctx_unset_searchdirs(ctx, TESTS_SRC));
Radek Krejci1e880032018-09-20 12:17:12 +0200158 assert_int_equal(6, ctx->search_paths.count);
159 /* last */
Radek Krejci0759b792018-09-20 13:53:15 +0200160 assert_int_equal(LY_SUCCESS, ly_ctx_unset_searchdirs(ctx, "/tmp"));
Radek Krejci1e880032018-09-20 12:17:12 +0200161 assert_int_equal(5, ctx->search_paths.count);
162 /* all */
Radek Krejci0759b792018-09-20 13:53:15 +0200163 assert_int_equal(LY_SUCCESS, ly_ctx_unset_searchdirs(ctx, NULL));
Radek Krejci1e880032018-09-20 12:17:12 +0200164 assert_int_equal(0, ctx->search_paths.count);
165
Radek Krejci555c9bb2018-09-20 12:45:13 +0200166 /* again - no change */
Radek Krejci0759b792018-09-20 13:53:15 +0200167 assert_int_equal(LY_SUCCESS, ly_ctx_unset_searchdirs(ctx, NULL));
Radek Krejci555c9bb2018-09-20 12:45:13 +0200168
169 /* cleanup */
170 ly_ctx_destroy(ctx, NULL);
171
172 /* test searchdir list in ly_ctx_new() */
173 assert_int_equal(LY_EINVAL, ly_ctx_new("/nonexistingfile", 0, &ctx));
174 logbuf_assert("Unable to use search directory \"/nonexistingfile\" (No such file or directory)");
Radek Krejci51d341d2018-09-20 14:26:41 +0200175 assert_int_equal(LY_SUCCESS, ly_ctx_new(TESTS_SRC":/tmp:/tmp:"TESTS_SRC, 0, &ctx));
Radek Krejci555c9bb2018-09-20 12:45:13 +0200176 assert_int_equal(2, ctx->search_paths.count);
177 assert_string_equal(TESTS_SRC, ctx->search_paths.objs[0]);
178 assert_string_equal("/tmp", ctx->search_paths.objs[1]);
179
180 /* cleanup */
Radek Krejci1e880032018-09-20 12:17:12 +0200181 ly_ctx_destroy(ctx, NULL);
182}
183
Radek Krejci0c4b22b2018-09-20 12:55:46 +0200184static void
185test_options(void **state)
186{
187 (void) state; /* unused */
188
189 struct ly_ctx *ctx;
Radek Krejcib7db73a2018-10-24 14:18:40 +0200190
191 will_return_always(__wrap_ly_set_add, 0);
Radek Krejci0c4b22b2018-09-20 12:55:46 +0200192 assert_int_equal(LY_SUCCESS, ly_ctx_new(NULL, 0xffffffff, &ctx));
193
Radek Krejci962a8102018-09-20 14:21:06 +0200194 /* invalid arguments */
195 assert_int_equal(0, ly_ctx_get_options(NULL));
196 logbuf_assert("Invalid argument ctx (ly_ctx_get_options()).");
197
198 assert_int_equal(LY_EINVAL, ly_ctx_set_option(NULL, 0));
199 logbuf_assert("Invalid argument ctx (ly_ctx_set_option()).");
200 assert_int_equal(LY_EINVAL, ly_ctx_unset_option(NULL, 0));
201 logbuf_assert("Invalid argument ctx (ly_ctx_unset_option()).");
202
203 /* option not allowed to be changed */
204 assert_int_equal(LY_EINVAL, ly_ctx_set_option(ctx, LY_CTX_NOYANGLIBRARY));
205 logbuf_assert("Invalid argument option (ly_ctx_set_option()).");
206 assert_int_equal(LY_EINVAL, ly_ctx_set_option(ctx, LY_CTX_NOYANGLIBRARY));
207 logbuf_assert("Invalid argument option (ly_ctx_set_option()).");
208
209
Radek Krejci0c4b22b2018-09-20 12:55:46 +0200210 /* unset */
211 /* LY_CTX_ALLIMPLEMENTED */
212 assert_int_not_equal(0, ctx->flags & LY_CTX_ALLIMPLEMENTED);
Radek Krejci962a8102018-09-20 14:21:06 +0200213 assert_int_equal(LY_SUCCESS, ly_ctx_unset_option(ctx, LY_CTX_ALLIMPLEMENTED));
Radek Krejci0c4b22b2018-09-20 12:55:46 +0200214 assert_int_equal(0, ctx->flags & LY_CTX_ALLIMPLEMENTED);
215
216 /* LY_CTX_DISABLE_SEARCHDIRS */
217 assert_int_not_equal(0, ctx->flags & LY_CTX_DISABLE_SEARCHDIRS);
Radek Krejci962a8102018-09-20 14:21:06 +0200218 assert_int_equal(LY_SUCCESS, ly_ctx_unset_option(ctx, LY_CTX_DISABLE_SEARCHDIRS));
Radek Krejci0c4b22b2018-09-20 12:55:46 +0200219 assert_int_equal(0, ctx->flags & LY_CTX_DISABLE_SEARCHDIRS);
220
221 /* LY_CTX_DISABLE_SEARCHDIR_CWD */
222 assert_int_not_equal(0, ctx->flags & LY_CTX_DISABLE_SEARCHDIR_CWD);
Radek Krejci962a8102018-09-20 14:21:06 +0200223 assert_int_equal(LY_SUCCESS, ly_ctx_unset_option(ctx, LY_CTX_DISABLE_SEARCHDIR_CWD));
Radek Krejci0c4b22b2018-09-20 12:55:46 +0200224 assert_int_equal(0, ctx->flags & LY_CTX_DISABLE_SEARCHDIR_CWD);
225
Radek Krejci0c4b22b2018-09-20 12:55:46 +0200226 /* LY_CTX_PREFER_SEARCHDIRS */
227 assert_int_not_equal(0, ctx->flags & LY_CTX_PREFER_SEARCHDIRS);
Radek Krejci962a8102018-09-20 14:21:06 +0200228 assert_int_equal(LY_SUCCESS, ly_ctx_unset_option(ctx, LY_CTX_PREFER_SEARCHDIRS));
Radek Krejci0c4b22b2018-09-20 12:55:46 +0200229 assert_int_equal(0, ctx->flags & LY_CTX_PREFER_SEARCHDIRS);
230
231 /* LY_CTX_TRUSTED */
232 assert_int_not_equal(0, ctx->flags & LY_CTX_TRUSTED);
Radek Krejci962a8102018-09-20 14:21:06 +0200233 assert_int_equal(LY_SUCCESS, ly_ctx_unset_option(ctx, LY_CTX_TRUSTED));
Radek Krejci0c4b22b2018-09-20 12:55:46 +0200234 assert_int_equal(0, ctx->flags & LY_CTX_TRUSTED);
235
Radek Krejci962a8102018-09-20 14:21:06 +0200236 assert_int_equal(ctx->flags, ly_ctx_get_options(ctx));
237
Radek Krejci0c4b22b2018-09-20 12:55:46 +0200238 /* set back */
239 /* LY_CTX_ALLIMPLEMENTED */
Radek Krejci962a8102018-09-20 14:21:06 +0200240 assert_int_equal(LY_SUCCESS, ly_ctx_set_option(ctx, LY_CTX_ALLIMPLEMENTED));
Radek Krejci0c4b22b2018-09-20 12:55:46 +0200241 assert_int_not_equal(0, ctx->flags & LY_CTX_ALLIMPLEMENTED);
242
243 /* LY_CTX_DISABLE_SEARCHDIRS */
Radek Krejci962a8102018-09-20 14:21:06 +0200244 assert_int_equal(LY_SUCCESS, ly_ctx_set_option(ctx, LY_CTX_DISABLE_SEARCHDIRS));
Radek Krejci0c4b22b2018-09-20 12:55:46 +0200245 assert_int_not_equal(0, ctx->flags & LY_CTX_DISABLE_SEARCHDIRS);
246
247 /* LY_CTX_DISABLE_SEARCHDIR_CWD */
Radek Krejci962a8102018-09-20 14:21:06 +0200248 assert_int_equal(LY_SUCCESS, ly_ctx_set_option(ctx, LY_CTX_DISABLE_SEARCHDIR_CWD));
Radek Krejci0c4b22b2018-09-20 12:55:46 +0200249 assert_int_not_equal(0, ctx->flags & LY_CTX_DISABLE_SEARCHDIR_CWD);
250
251 /* LY_CTX_PREFER_SEARCHDIRS */
Radek Krejci962a8102018-09-20 14:21:06 +0200252 assert_int_equal(LY_SUCCESS, ly_ctx_set_option(ctx, LY_CTX_PREFER_SEARCHDIRS));
Radek Krejci0c4b22b2018-09-20 12:55:46 +0200253 assert_int_not_equal(0, ctx->flags & LY_CTX_PREFER_SEARCHDIRS);
254
255 /* LY_CTX_TRUSTED */
Radek Krejci962a8102018-09-20 14:21:06 +0200256 assert_int_equal(LY_SUCCESS, ly_ctx_set_option(ctx, LY_CTX_TRUSTED));
Radek Krejci0c4b22b2018-09-20 12:55:46 +0200257 assert_int_not_equal(0, ctx->flags & LY_CTX_TRUSTED);
258
Radek Krejci962a8102018-09-20 14:21:06 +0200259 assert_int_equal(ctx->flags, ly_ctx_get_options(ctx));
260
261 /* cleanup */
262 ly_ctx_destroy(ctx, NULL);
263}
264
Radek Krejci2d31ea72018-10-25 15:46:42 +0200265static LY_ERR test_imp_clb(const char *UNUSED(mod_name), const char *UNUSED(mod_rev), const char *UNUSED(submod_name),
266 const char *UNUSED(sub_rev), void *user_data, LYS_INFORMAT *format,
267 const char **module_data, void (**free_module_data)(void *model_data, void *user_data))
268{
269 *module_data = user_data;
270 *format = LYS_IN_YANG;
271 *free_module_data = NULL;
272 return LY_SUCCESS;
273}
274
Radek Krejci962a8102018-09-20 14:21:06 +0200275static void
276test_models(void **state)
277{
278 (void) state; /* unused */
279
Radek Krejcib7db73a2018-10-24 14:18:40 +0200280 struct ly_ctx *ctx;
Radek Krejci2d31ea72018-10-25 15:46:42 +0200281 const char *str;
282 const struct lys_module *mod1, *mod2;
Radek Krejcib7db73a2018-10-24 14:18:40 +0200283
Radek Krejci962a8102018-09-20 14:21:06 +0200284 /* invalid arguments */
285 assert_int_equal(0, ly_ctx_get_module_set_id(NULL));
286 logbuf_assert("Invalid argument ctx (ly_ctx_get_module_set_id()).");
287
Radek Krejcib7db73a2018-10-24 14:18:40 +0200288 will_return_always(__wrap_ly_set_add, 0);
Radek Krejcifaa1eac2018-10-30 14:34:55 +0100289 assert_int_equal(LY_SUCCESS, ly_ctx_new(NULL, LY_CTX_DISABLE_SEARCHDIRS, &ctx));
Radek Krejci962a8102018-09-20 14:21:06 +0200290 assert_int_equal(ctx->module_set_id, ly_ctx_get_module_set_id(ctx));
291
Radek Krejci9ed7a192018-10-31 16:23:51 +0100292 assert_null(lys_parse_mem(ctx, "module x {namespace urn:x;prefix x;}", 3));
293 logbuf_assert("Invalid schema input format.");
294
Radek Krejci2d31ea72018-10-25 15:46:42 +0200295 /* import callback */
296 ly_ctx_set_module_imp_clb(ctx, test_imp_clb, (void*)(str = "test"));
297 assert_ptr_equal(test_imp_clb, ctx->imp_clb);
298 assert_ptr_equal(str, ctx->imp_clb_data);
299 assert_ptr_equal(test_imp_clb, ly_ctx_get_module_imp_clb(ctx, (void**)&str));
300 assert_string_equal("test", str);
301
302 ly_ctx_set_module_imp_clb(ctx, NULL, NULL);
303 assert_null(ctx->imp_clb);
304 assert_null(ctx->imp_clb_data);
305
306 /* submodule in multiple modules */
307 ly_ctx_set_module_imp_clb(ctx, test_imp_clb, "submodule xx {belongs-to x;}");
308 mod1 = lys_parse_mem(ctx, "module x {namespace urn:x;prefix x;include xx;revision 2018-10-24;}", LYS_IN_YANG);
309 assert_non_null(mod1);
Radek Krejci9ed7a192018-10-31 16:23:51 +0100310 mod2 = lys_parse_mem_(ctx, "module x {namespace urn:x;prefix x;include xx;revision 2018-10-25;}", LYS_IN_YANG, 0, NULL, NULL);
Radek Krejci2d31ea72018-10-25 15:46:42 +0200311 assert_non_null(mod2);
312 assert_ptr_equal(mod1->parsed->includes[0].submodule, mod2->parsed->includes[0].submodule);
313
Radek Krejcifaa1eac2018-10-30 14:34:55 +0100314 /* name collision of module and submodule */
Radek Krejcie9e987e2018-10-31 12:50:27 +0100315 ly_ctx_set_module_imp_clb(ctx, test_imp_clb, "submodule y {belongs-to a; revision 2018-10-30;}");
Radek Krejcifaa1eac2018-10-30 14:34:55 +0100316 assert_null(lys_parse_mem(ctx, "module y {namespace urn:y;prefix y;include y;}", LYS_IN_YANG));
317 assert_int_equal(LY_EVALID, ly_errcode(ctx));
318 logbuf_assert("Name collision between module and submodule of name \"y\". Line number 1.");
319
Radek Krejcie9e987e2018-10-31 12:50:27 +0100320 assert_non_null(lys_parse_mem(ctx, "module a {namespace urn:a;prefix a;include y;revision 2018-10-30; }", LYS_IN_YANG));
Radek Krejcifaa1eac2018-10-30 14:34:55 +0100321 assert_null(lys_parse_mem(ctx, "module y {namespace urn:y;prefix y;}", LYS_IN_YANG));
322 assert_int_equal(LY_EVALID, ly_errcode(ctx));
323 logbuf_assert("Name collision between module and submodule of name \"y\". Line number 1.");
324
325 store = 1;
326 ly_ctx_set_module_imp_clb(ctx, test_imp_clb, "submodule y {belongs-to b;}");
327 assert_null(lys_parse_mem(ctx, "module b {namespace urn:b;prefix b;include y;}", LYS_IN_YANG));
328 assert_int_equal(LY_EVALID, ly_errcode(ctx));
329 logbuf_assert("Name collision between submodules of name \"y\". Line number 1.");
330 store = -1;
331
Radek Krejcie9e987e2018-10-31 12:50:27 +0100332 /* selecting correct revision of the submodules */
333 ly_ctx_reset_latests(ctx);
334 ly_ctx_set_module_imp_clb(ctx, test_imp_clb, "submodule y {belongs-to a; revision 2018-10-31;}");
Radek Krejci9ed7a192018-10-31 16:23:51 +0100335 mod2 = lys_parse_mem_(ctx, "module a {namespace urn:a;prefix a;include y; revision 2018-10-31;}", LYS_IN_YANG, 0, NULL, NULL);
Radek Krejcie9e987e2018-10-31 12:50:27 +0100336 assert_non_null(mod2);
337 assert_string_equal("2018-10-31", mod2->parsed->includes[0].submodule->revs[0].date);
338
Radek Krejci9c536962018-10-31 14:52:13 +0100339 /* reloading module in case only the compiled module resists in the context */
Radek Krejci9c536962018-10-31 14:52:13 +0100340 mod1 = lys_parse_mem(ctx, "module w {namespace urn:w;prefix w;revision 2018-10-24;}", LYS_IN_YANG);
341 assert_non_null(mod1);
342 assert_int_equal(LY_SUCCESS, lys_compile(mod1, LYSC_OPT_FREE_SP));
343 assert_non_null(mod1->compiled);
344 assert_null(mod1->parsed);
345 mod2 = lys_parse_mem(ctx, "module z {namespace urn:z;prefix z;import w {prefix w;revision-date 2018-10-24;}}", LYS_IN_YANG);
346 assert_non_null(mod2);
347 /* mod1->parsed is necessary to compile mod2 because of possible groupings, typedefs, ... */
Radek Krejci9ed7a192018-10-31 16:23:51 +0100348 ly_ctx_set_module_imp_clb(ctx, NULL, NULL);
349 assert_int_equal(LY_ENOTFOUND, lys_compile(mod2, 0));
350 logbuf_assert("Unable to reload \"w\" module to import it into \"z\", source data not found.");
351 ly_ctx_set_module_imp_clb(ctx, test_imp_clb, "module w {namespace urn:w;prefix w;revision 2018-10-24;}");
Radek Krejci9c536962018-10-31 14:52:13 +0100352 assert_int_equal(LY_SUCCESS, lys_compile(mod2, 0));
353 assert_non_null(mod1->parsed);
354 assert_string_equal("w", mod1->parsed->name);
Radek Krejcie9e987e2018-10-31 12:50:27 +0100355
Radek Krejci0c4b22b2018-09-20 12:55:46 +0200356 /* cleanup */
357 ly_ctx_destroy(ctx, NULL);
358}
359
Radek Krejcib7db73a2018-10-24 14:18:40 +0200360static void
361test_get_models(void **state)
362{
363 (void) state; /* unused */
364
365 struct ly_ctx *ctx;
366 const struct lys_module *mod, *mod2;
Radek Krejci5e163df2018-10-24 14:42:15 +0200367 const char *str0 = "module a {namespace urn:a;prefix a;}";
Radek Krejcib7db73a2018-10-24 14:18:40 +0200368 const char *str1 = "module a {namespace urn:a;prefix a;revision 2018-10-23;}";
369 const char *str2 = "module a {namespace urn:a;prefix a;revision 2018-10-23;revision 2018-10-24;}";
370
371 will_return_always(__wrap_ly_set_add, 0);
372 assert_int_equal(LY_SUCCESS, ly_ctx_new(NULL, 0, &ctx));
373
374 /* invalid arguments */
375 assert_ptr_equal(NULL, ly_ctx_get_module(NULL, NULL, NULL));
376 logbuf_assert("Invalid argument ctx (ly_ctx_get_module()).");
377 assert_ptr_equal(NULL, ly_ctx_get_module(ctx, NULL, NULL));
378 logbuf_assert("Invalid argument name (ly_ctx_get_module()).");
379 assert_ptr_equal(NULL, ly_ctx_get_module_ns(NULL, NULL, NULL));
380 logbuf_assert("Invalid argument ctx (ly_ctx_get_module_ns()).");
381 assert_ptr_equal(NULL, ly_ctx_get_module_ns(ctx, NULL, NULL));
382 logbuf_assert("Invalid argument ns (ly_ctx_get_module_ns()).");
383 assert_null(ly_ctx_get_module(ctx, "nonsence", NULL));
384
385 /* internal modules */
386 assert_null(ly_ctx_get_module_implemented(ctx, "ietf-yang-types"));
387 mod = ly_ctx_get_module_implemented(ctx, "yang");
388 assert_non_null(mod);
389 assert_non_null(mod->parsed);
390 assert_string_equal("yang", mod->parsed->name);
391 mod2 = ly_ctx_get_module_implemented_ns(ctx, mod->parsed->ns);
392 assert_ptr_equal(mod, mod2);
393 assert_non_null(ly_ctx_get_module(ctx, "ietf-yang-metadata", "2016-08-05"));
394 assert_non_null(ly_ctx_get_module(ctx, "ietf-yang-types", "2013-07-15"));
395 assert_non_null(ly_ctx_get_module(ctx, "ietf-inet-types", "2013-07-15"));
Radek Krejci5e163df2018-10-24 14:42:15 +0200396 assert_non_null(ly_ctx_get_module_ns(ctx, "urn:ietf:params:xml:ns:yang:ietf-datastores", "2017-08-17"));
Radek Krejcib7db73a2018-10-24 14:18:40 +0200397
398 /* select module by revision */
399 mod = lys_parse_mem(ctx, str1, LYS_IN_YANG);
400 /* invalid attempts - implementing module of the same name and inserting the same module */
401 assert_null(lys_parse_mem(ctx, str2, LYS_IN_YANG));
402 logbuf_assert("Module \"a\" is already implemented in the context.");
Radek Krejci9ed7a192018-10-31 16:23:51 +0100403 assert_null(lys_parse_mem_(ctx, str1, LYS_IN_YANG, 0, NULL, NULL));
Radek Krejcib7db73a2018-10-24 14:18:40 +0200404 logbuf_assert("Module \"a\" of revision \"2018-10-23\" is already present in the context.");
405 /* insert the second module only as imported, not implemented */
Radek Krejci9ed7a192018-10-31 16:23:51 +0100406 mod2 = lys_parse_mem_(ctx, str2, LYS_IN_YANG, 0, NULL, NULL);
Radek Krejcib7db73a2018-10-24 14:18:40 +0200407 assert_non_null(mod);
408 assert_non_null(mod2);
409 assert_ptr_not_equal(mod, mod2);
410 mod = ly_ctx_get_module_latest(ctx, "a");
411 assert_ptr_equal(mod, mod2);
412 mod2 = ly_ctx_get_module_latest_ns(ctx, mod->parsed->ns);
413 assert_ptr_equal(mod, mod2);
Radek Krejci5e163df2018-10-24 14:42:15 +0200414 /* work with module with no revision */
Radek Krejci9ed7a192018-10-31 16:23:51 +0100415 mod = lys_parse_mem_(ctx, str0, LYS_IN_YANG, 0, NULL, NULL);
Radek Krejci5e163df2018-10-24 14:42:15 +0200416 assert_non_null(mod);
417 assert_ptr_equal(mod, ly_ctx_get_module(ctx, "a", NULL));
418 assert_ptr_not_equal(mod, ly_ctx_get_module_latest(ctx, "a"));
Radek Krejcib7db73a2018-10-24 14:18:40 +0200419
Radek Krejcid33273d2018-10-25 14:55:52 +0200420 str1 = "submodule b {belongs-to a;}";
421 assert_null(lys_parse_mem(ctx, str1, LYS_IN_YANG));
422 logbuf_assert("Input data contains submodule \"b\" which cannot be parsed directly without its main module.");
423
Radek Krejcib7db73a2018-10-24 14:18:40 +0200424 /* cleanup */
425 ly_ctx_destroy(ctx, NULL);
426}
427
Radek Krejci1e880032018-09-20 12:17:12 +0200428int main(void)
429{
430 const struct CMUnitTest tests[] = {
431 cmocka_unit_test_setup(test_searchdirs, logger_setup),
Radek Krejci0c4b22b2018-09-20 12:55:46 +0200432 cmocka_unit_test_setup(test_options, logger_setup),
Radek Krejci962a8102018-09-20 14:21:06 +0200433 cmocka_unit_test_setup(test_models, logger_setup),
Radek Krejcib7db73a2018-10-24 14:18:40 +0200434 cmocka_unit_test_setup(test_get_models, logger_setup),
Radek Krejci1e880032018-09-20 12:17:12 +0200435 };
436
437 return cmocka_run_group_tests(tests, NULL, NULL);
438}