blob: 369e83cc48b9fffe347bd6e1858011d888410f76 [file] [log] [blame]
Radek Krejcib1646a92018-11-02 16:08:26 +01001/**
2 * @file xpath.c
3 * @author Michal Vasko <mvasko@cesnet.cz>
4 * @brief YANG XPath evaluation functions
5 *
Michal Vasko49fec8e2022-05-24 10:28:33 +02006 * Copyright (c) 2015 - 2022 CESNET, z.s.p.o.
Radek Krejcib1646a92018-11-02 16:08:26 +01007 *
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 */
Christian Hopps32874e12021-05-01 09:43:54 -040014#define _GNU_SOURCE /* asprintf, strdup */
Radek Krejcib1646a92018-11-02 16:08:26 +010015
Radek Krejci535ea9f2020-05-29 16:01:05 +020016#include "xpath.h"
Radek Krejcib1646a92018-11-02 16:08:26 +010017
Radek Krejci535ea9f2020-05-29 16:01:05 +020018#include <assert.h>
Radek Krejcib1646a92018-11-02 16:08:26 +010019#include <ctype.h>
Radek Krejci535ea9f2020-05-29 16:01:05 +020020#include <errno.h>
Radek Krejci535ea9f2020-05-29 16:01:05 +020021#include <math.h>
Radek Krejcie7b95092019-05-15 11:03:07 +020022#include <stdint.h>
Radek Krejcib1646a92018-11-02 16:08:26 +010023#include <stdio.h>
24#include <stdlib.h>
Radek Krejcib1646a92018-11-02 16:08:26 +010025#include <string.h>
Radek Krejcib1646a92018-11-02 16:08:26 +010026
Radek Krejci535ea9f2020-05-29 16:01:05 +020027#include "common.h"
Michal Vasko5aa44c02020-06-29 11:47:02 +020028#include "compat.h"
Radek Krejci535ea9f2020-05-29 16:01:05 +020029#include "context.h"
Radek Krejcie7b95092019-05-15 11:03:07 +020030#include "dict.h"
Radek Krejci535ea9f2020-05-29 16:01:05 +020031#include "hash_table.h"
Radek Krejci47fab892020-11-05 17:02:41 +010032#include "out.h"
Radek Krejci7931b192020-06-25 17:05:03 +020033#include "parser_data.h"
Michal Vasko004d3152020-06-11 19:59:22 +020034#include "path.h"
Michal Vasko03ff5a72019-09-11 13:49:33 +020035#include "plugins_types.h"
Radek Krejci535ea9f2020-05-29 16:01:05 +020036#include "printer_data.h"
Michal Vasko1a7a7bd2020-10-16 14:39:15 +020037#include "schema_compile_node.h"
Radek Krejci535ea9f2020-05-29 16:01:05 +020038#include "tree.h"
Radek Krejci77114102021-03-10 15:21:57 +010039#include "tree_data.h"
Radek Krejci535ea9f2020-05-29 16:01:05 +020040#include "tree_data_internal.h"
Radek Krejci859a15a2021-03-05 20:56:59 +010041#include "tree_edit.h"
Radek Krejci535ea9f2020-05-29 16:01:05 +020042#include "tree_schema_internal.h"
43#include "xml.h"
Michal Vasko03ff5a72019-09-11 13:49:33 +020044
aPiecekbf968d92021-05-27 14:35:05 +020045static LY_ERR reparse_or_expr(const struct ly_ctx *ctx, struct lyxp_expr *exp, uint16_t *tok_idx, uint16_t depth);
Michal Vasko40308e72020-10-20 16:38:40 +020046static LY_ERR eval_expr_select(const struct lyxp_expr *exp, uint16_t *tok_idx, enum lyxp_expr_type etype,
47 struct lyxp_set *set, uint32_t options);
Michal Vasko93923692021-05-07 15:28:02 +020048static LY_ERR moveto_resolve_model(const char **qname, uint16_t *qname_len, const struct lyxp_set *set,
49 const struct lysc_node *ctx_scnode, const struct lys_module **moveto_mod);
Michal Vasko49fec8e2022-05-24 10:28:33 +020050static LY_ERR moveto_node(struct lyxp_set *set, const struct lys_module *moveto_mod, const char *ncname,
51 enum lyxp_axis axis, uint32_t options);
52static LY_ERR moveto_scnode(struct lyxp_set *set, const struct lys_module *moveto_mod, const char *ncname,
53 enum lyxp_axis axis, uint32_t options);
Michal Vasko03ff5a72019-09-11 13:49:33 +020054
aPiecek96dc1e32021-10-08 15:45:59 +020055/* Functions are divided into the following basic classes:
56 *
57 * (re)parse functions:
58 * Parse functions parse the expression into
59 * tokens (syntactic analysis).
60 * Reparse functions perform semantic analysis
61 * (do not save the result, just a check) of
62 * the expression and fill repeat indices.
63 *
64 * warn functions:
65 * Warn functions check specific reasonable conditions for schema XPath
66 * and print a warning if they are not satisfied.
67 *
68 * moveto functions:
69 * They and only they actually change the context (set).
70 *
71 * eval functions:
72 * They execute a parsed XPath expression on some data subtree.
73 */
74
Michal Vasko03ff5a72019-09-11 13:49:33 +020075/**
76 * @brief Print the type of an XPath \p set.
77 *
78 * @param[in] set Set to use.
79 * @return Set type string.
80 */
81static const char *
82print_set_type(struct lyxp_set *set)
83{
84 switch (set->type) {
Michal Vasko03ff5a72019-09-11 13:49:33 +020085 case LYXP_SET_NODE_SET:
86 return "node set";
87 case LYXP_SET_SCNODE_SET:
88 return "schema node set";
89 case LYXP_SET_BOOLEAN:
90 return "boolean";
91 case LYXP_SET_NUMBER:
92 return "number";
93 case LYXP_SET_STRING:
94 return "string";
95 }
96
97 return NULL;
98}
99
Michal Vasko24cddf82020-06-01 08:17:01 +0200100const char *
Michal Vasko49fec8e2022-05-24 10:28:33 +0200101lyxp_token2str(enum lyxp_token tok)
Michal Vasko03ff5a72019-09-11 13:49:33 +0200102{
103 switch (tok) {
104 case LYXP_TOKEN_PAR1:
105 return "(";
106 case LYXP_TOKEN_PAR2:
107 return ")";
108 case LYXP_TOKEN_BRACK1:
109 return "[";
110 case LYXP_TOKEN_BRACK2:
111 return "]";
112 case LYXP_TOKEN_DOT:
113 return ".";
114 case LYXP_TOKEN_DDOT:
115 return "..";
116 case LYXP_TOKEN_AT:
117 return "@";
118 case LYXP_TOKEN_COMMA:
119 return ",";
120 case LYXP_TOKEN_NAMETEST:
121 return "NameTest";
122 case LYXP_TOKEN_NODETYPE:
123 return "NodeType";
aPiecekfba75362021-10-07 12:39:48 +0200124 case LYXP_TOKEN_VARREF:
125 return "VariableReference";
Michal Vasko03ff5a72019-09-11 13:49:33 +0200126 case LYXP_TOKEN_FUNCNAME:
127 return "FunctionName";
Michal Vasko3e48bf32020-06-01 08:39:07 +0200128 case LYXP_TOKEN_OPER_LOG:
Michal Vasko03ff5a72019-09-11 13:49:33 +0200129 return "Operator(Logic)";
Michal Vasko3e48bf32020-06-01 08:39:07 +0200130 case LYXP_TOKEN_OPER_EQUAL:
131 return "Operator(Equal)";
132 case LYXP_TOKEN_OPER_NEQUAL:
133 return "Operator(Non-equal)";
134 case LYXP_TOKEN_OPER_COMP:
Michal Vasko03ff5a72019-09-11 13:49:33 +0200135 return "Operator(Comparison)";
Michal Vasko3e48bf32020-06-01 08:39:07 +0200136 case LYXP_TOKEN_OPER_MATH:
Michal Vasko03ff5a72019-09-11 13:49:33 +0200137 return "Operator(Math)";
Michal Vasko3e48bf32020-06-01 08:39:07 +0200138 case LYXP_TOKEN_OPER_UNI:
Michal Vasko03ff5a72019-09-11 13:49:33 +0200139 return "Operator(Union)";
Michal Vasko3e48bf32020-06-01 08:39:07 +0200140 case LYXP_TOKEN_OPER_PATH:
Michal Vasko03ff5a72019-09-11 13:49:33 +0200141 return "Operator(Path)";
Michal Vasko3e48bf32020-06-01 08:39:07 +0200142 case LYXP_TOKEN_OPER_RPATH:
Michal Vasko14676352020-05-29 11:35:55 +0200143 return "Operator(Recursive Path)";
Michal Vasko03ff5a72019-09-11 13:49:33 +0200144 case LYXP_TOKEN_LITERAL:
145 return "Literal";
146 case LYXP_TOKEN_NUMBER:
147 return "Number";
148 default:
149 LOGINT(NULL);
150 return "";
151 }
152}
153
154/**
Michal Vasko49fec8e2022-05-24 10:28:33 +0200155 * @brief Transform string into an axis.
156 *
157 * @param[in] str String to transform.
158 * @param[in] str_len Length of @p str.
159 * @return Transformed axis.
160 */
161static enum lyxp_axis
162str2axis(const char *str, uint16_t str_len)
163{
164 switch (str_len) {
165 case 4:
166 assert(!strncmp("self", str, str_len));
167 return LYXP_AXIS_SELF;
168 case 5:
169 assert(!strncmp("child", str, str_len));
170 return LYXP_AXIS_CHILD;
171 case 6:
172 assert(!strncmp("parent", str, str_len));
173 return LYXP_AXIS_PARENT;
174 case 8:
175 assert(!strncmp("ancestor", str, str_len));
176 return LYXP_AXIS_ANCESTOR;
177 case 9:
178 if (str[0] == 'a') {
179 assert(!strncmp("attribute", str, str_len));
180 return LYXP_AXIS_ATTRIBUTE;
181 } else if (str[0] == 'f') {
182 assert(!strncmp("following", str, str_len));
183 return LYXP_AXIS_FOLLOWING;
184 } else {
185 assert(!strncmp("preceding", str, str_len));
186 return LYXP_AXIS_PRECEDING;
187 }
188 break;
189 case 10:
190 assert(!strncmp("descendant", str, str_len));
191 return LYXP_AXIS_DESCENDANT;
192 case 16:
193 assert(!strncmp("ancestor-or-self", str, str_len));
194 return LYXP_AXIS_ANCESTOR_OR_SELF;
195 case 17:
196 if (str[0] == 'f') {
197 assert(!strncmp("following-sibling", str, str_len));
198 return LYXP_AXIS_FOLLOWING_SIBLING;
199 } else {
200 assert(!strncmp("preceding-sibling", str, str_len));
201 return LYXP_AXIS_PRECEDING_SIBLING;
202 }
203 break;
204 case 18:
205 assert(!strncmp("descendant-or-self", str, str_len));
206 return LYXP_AXIS_DESCENDANT_OR_SELF;
207 }
208
209 LOGINT(NULL);
210 return 0;
211}
212
213/**
Michal Vasko03ff5a72019-09-11 13:49:33 +0200214 * @brief Print the whole expression \p exp to debug output.
215 *
216 * @param[in] exp Expression to use.
217 */
218static void
Michal Vasko40308e72020-10-20 16:38:40 +0200219print_expr_struct_debug(const struct lyxp_expr *exp)
Michal Vasko03ff5a72019-09-11 13:49:33 +0200220{
Radek Krejcif13b87b2020-12-01 22:02:17 +0100221#define MSG_BUFFER_SIZE 128
222 char tmp[MSG_BUFFER_SIZE];
Michal Vasko1fdd8fa2021-01-08 09:21:45 +0100223 uint32_t i, j;
Michal Vasko03ff5a72019-09-11 13:49:33 +0200224
Radek Krejci52b6d512020-10-12 12:33:17 +0200225 if (!exp || (ly_ll < LY_LLDBG)) {
Michal Vasko03ff5a72019-09-11 13:49:33 +0200226 return;
227 }
228
229 LOGDBG(LY_LDGXPATH, "expression \"%s\":", exp->expr);
230 for (i = 0; i < exp->used; ++i) {
Michal Vasko49fec8e2022-05-24 10:28:33 +0200231 sprintf(tmp, "\ttoken %s, in expression \"%.*s\"", lyxp_token2str(exp->tokens[i]), exp->tok_len[i],
Michal Vasko69730152020-10-09 16:30:07 +0200232 &exp->expr[exp->tok_pos[i]]);
Michal Vasko23049552021-03-04 15:57:44 +0100233 if (exp->repeat && exp->repeat[i]) {
Michal Vasko03ff5a72019-09-11 13:49:33 +0200234 sprintf(tmp + strlen(tmp), " (repeat %d", exp->repeat[i][0]);
235 for (j = 1; exp->repeat[i][j]; ++j) {
236 sprintf(tmp + strlen(tmp), ", %d", exp->repeat[i][j]);
237 }
238 strcat(tmp, ")");
239 }
240 LOGDBG(LY_LDGXPATH, tmp);
241 }
Radek Krejcif13b87b2020-12-01 22:02:17 +0100242#undef MSG_BUFFER_SIZE
Michal Vasko03ff5a72019-09-11 13:49:33 +0200243}
244
245#ifndef NDEBUG
246
247/**
248 * @brief Print XPath set content to debug output.
249 *
250 * @param[in] set Set to print.
251 */
252static void
253print_set_debug(struct lyxp_set *set)
254{
255 uint32_t i;
256 char *str;
Michal Vasko03ff5a72019-09-11 13:49:33 +0200257 struct lyxp_set_node *item;
258 struct lyxp_set_scnode *sitem;
259
Radek Krejci52b6d512020-10-12 12:33:17 +0200260 if (ly_ll < LY_LLDBG) {
Michal Vasko03ff5a72019-09-11 13:49:33 +0200261 return;
262 }
263
264 switch (set->type) {
265 case LYXP_SET_NODE_SET:
266 LOGDBG(LY_LDGXPATH, "set NODE SET:");
267 for (i = 0; i < set->used; ++i) {
268 item = &set->val.nodes[i];
269
270 switch (item->type) {
Michal Vasko2caefc12019-11-14 16:07:56 +0100271 case LYXP_NODE_NONE:
272 LOGDBG(LY_LDGXPATH, "\t%d (pos %u): NONE", i + 1, item->pos);
273 break;
Michal Vasko03ff5a72019-09-11 13:49:33 +0200274 case LYXP_NODE_ROOT:
275 LOGDBG(LY_LDGXPATH, "\t%d (pos %u): ROOT", i + 1, item->pos);
276 break;
277 case LYXP_NODE_ROOT_CONFIG:
278 LOGDBG(LY_LDGXPATH, "\t%d (pos %u): ROOT CONFIG", i + 1, item->pos);
279 break;
280 case LYXP_NODE_ELEM:
Michal Vasko9e685082021-01-29 14:49:09 +0100281 if ((item->node->schema->nodetype == LYS_LIST) && (lyd_child(item->node)->schema->nodetype == LYS_LEAF)) {
Michal Vasko03ff5a72019-09-11 13:49:33 +0200282 LOGDBG(LY_LDGXPATH, "\t%d (pos %u): ELEM %s (1st child val: %s)", i + 1, item->pos,
Radek Krejci6d5ba0c2021-04-26 07:49:59 +0200283 item->node->schema->name, lyd_get_value(lyd_child(item->node)));
Michal Vasko9e685082021-01-29 14:49:09 +0100284 } else if (item->node->schema->nodetype == LYS_LEAFLIST) {
Michal Vasko03ff5a72019-09-11 13:49:33 +0200285 LOGDBG(LY_LDGXPATH, "\t%d (pos %u): ELEM %s (val: %s)", i + 1, item->pos,
Radek Krejci6d5ba0c2021-04-26 07:49:59 +0200286 item->node->schema->name, lyd_get_value(item->node));
Michal Vasko03ff5a72019-09-11 13:49:33 +0200287 } else {
288 LOGDBG(LY_LDGXPATH, "\t%d (pos %u): ELEM %s", i + 1, item->pos, item->node->schema->name);
289 }
290 break;
291 case LYXP_NODE_TEXT:
292 if (item->node->schema->nodetype & LYS_ANYDATA) {
293 LOGDBG(LY_LDGXPATH, "\t%d (pos %u): TEXT <%s>", i + 1, item->pos,
Michal Vasko69730152020-10-09 16:30:07 +0200294 item->node->schema->nodetype == LYS_ANYXML ? "anyxml" : "anydata");
Michal Vasko03ff5a72019-09-11 13:49:33 +0200295 } else {
Radek Krejci6d5ba0c2021-04-26 07:49:59 +0200296 LOGDBG(LY_LDGXPATH, "\t%d (pos %u): TEXT %s", i + 1, item->pos, lyd_get_value(item->node));
Michal Vasko03ff5a72019-09-11 13:49:33 +0200297 }
298 break;
Michal Vasko9f96a052020-03-10 09:41:45 +0100299 case LYXP_NODE_META:
300 LOGDBG(LY_LDGXPATH, "\t%d (pos %u): META %s = %s", i + 1, item->pos, set->val.meta[i].meta->name,
Michal Vasko69730152020-10-09 16:30:07 +0200301 set->val.meta[i].meta->value);
Michal Vasko03ff5a72019-09-11 13:49:33 +0200302 break;
303 }
304 }
305 break;
306
307 case LYXP_SET_SCNODE_SET:
308 LOGDBG(LY_LDGXPATH, "set SCNODE SET:");
309 for (i = 0; i < set->used; ++i) {
310 sitem = &set->val.scnodes[i];
311
312 switch (sitem->type) {
313 case LYXP_NODE_ROOT:
314 LOGDBG(LY_LDGXPATH, "\t%d (%u): ROOT", i + 1, sitem->in_ctx);
315 break;
316 case LYXP_NODE_ROOT_CONFIG:
317 LOGDBG(LY_LDGXPATH, "\t%d (%u): ROOT CONFIG", i + 1, sitem->in_ctx);
318 break;
319 case LYXP_NODE_ELEM:
320 LOGDBG(LY_LDGXPATH, "\t%d (%u): ELEM %s", i + 1, sitem->in_ctx, sitem->scnode->name);
321 break;
322 default:
323 LOGINT(NULL);
324 break;
325 }
326 }
327 break;
328
Michal Vasko03ff5a72019-09-11 13:49:33 +0200329 case LYXP_SET_BOOLEAN:
330 LOGDBG(LY_LDGXPATH, "set BOOLEAN");
Michal Vasko004d3152020-06-11 19:59:22 +0200331 LOGDBG(LY_LDGXPATH, "\t%s", (set->val.bln ? "true" : "false"));
Michal Vasko03ff5a72019-09-11 13:49:33 +0200332 break;
333
334 case LYXP_SET_STRING:
335 LOGDBG(LY_LDGXPATH, "set STRING");
336 LOGDBG(LY_LDGXPATH, "\t%s", set->val.str);
337 break;
338
339 case LYXP_SET_NUMBER:
340 LOGDBG(LY_LDGXPATH, "set NUMBER");
341
342 if (isnan(set->val.num)) {
343 str = strdup("NaN");
344 } else if ((set->val.num == 0) || (set->val.num == -0.0f)) {
345 str = strdup("0");
346 } else if (isinf(set->val.num) && !signbit(set->val.num)) {
347 str = strdup("Infinity");
348 } else if (isinf(set->val.num) && signbit(set->val.num)) {
349 str = strdup("-Infinity");
350 } else if ((long long)set->val.num == set->val.num) {
351 if (asprintf(&str, "%lld", (long long)set->val.num) == -1) {
352 str = NULL;
353 }
354 } else {
355 if (asprintf(&str, "%03.1Lf", set->val.num) == -1) {
356 str = NULL;
357 }
358 }
359 LY_CHECK_ERR_RET(!str, LOGMEM(NULL), );
360
361 LOGDBG(LY_LDGXPATH, "\t%s", str);
362 free(str);
363 }
364}
365
366#endif
367
368/**
369 * @brief Realloc the string \p str.
370 *
371 * @param[in] ctx libyang context for logging.
372 * @param[in] needed How much free space is required.
373 * @param[in,out] str Pointer to the string to use.
374 * @param[in,out] used Used bytes in \p str.
375 * @param[in,out] size Allocated bytes in \p str.
376 * @return LY_ERR
377 */
378static LY_ERR
Michal Vasko52927e22020-03-16 17:26:14 +0100379cast_string_realloc(const struct ly_ctx *ctx, uint16_t needed, char **str, uint16_t *used, uint16_t *size)
Michal Vasko03ff5a72019-09-11 13:49:33 +0200380{
381 if (*size - *used < needed) {
382 do {
383 if ((UINT16_MAX - *size) < LYXP_STRING_CAST_SIZE_STEP) {
384 LOGERR(ctx, LY_EINVAL, "XPath string length limit (%u) reached.", UINT16_MAX);
385 return LY_EINVAL;
386 }
387 *size += LYXP_STRING_CAST_SIZE_STEP;
388 } while (*size - *used < needed);
389 *str = ly_realloc(*str, *size * sizeof(char));
390 LY_CHECK_ERR_RET(!(*str), LOGMEM(ctx), LY_EMEM);
391 }
392
393 return LY_SUCCESS;
394}
395
396/**
397 * @brief Cast nodes recursively to one string @p str.
398 *
399 * @param[in] node Node to cast.
400 * @param[in] fake_cont Whether to put the data into a "fake" container.
401 * @param[in] root_type Type of the XPath root.
402 * @param[in] indent Current indent.
403 * @param[in,out] str Resulting string.
404 * @param[in,out] used Used bytes in @p str.
405 * @param[in,out] size Allocated bytes in @p str.
406 * @return LY_ERR
407 */
408static LY_ERR
Michal Vaskoddd76592022-01-17 13:34:48 +0100409cast_string_recursive(const struct lyd_node *node, ly_bool fake_cont, enum lyxp_node_type root_type, uint16_t indent,
410 char **str, uint16_t *used, uint16_t *size)
Michal Vasko03ff5a72019-09-11 13:49:33 +0200411{
Radek Krejci7f769d72020-07-11 23:13:56 +0200412 char *buf, *line, *ptr = NULL;
Michal Vasko03ff5a72019-09-11 13:49:33 +0200413 const char *value_str;
Michal Vasko03ff5a72019-09-11 13:49:33 +0200414 const struct lyd_node *child;
Michal Vasko60ea6352020-06-29 13:39:39 +0200415 struct lyd_node *tree;
Michal Vasko03ff5a72019-09-11 13:49:33 +0200416 struct lyd_node_any *any;
417 LY_ERR rc;
418
419 if ((root_type == LYXP_NODE_ROOT_CONFIG) && (node->schema->flags & LYS_CONFIG_R)) {
420 return LY_SUCCESS;
421 }
422
423 if (fake_cont) {
Michal Vaskob7be7a82020-08-20 09:09:04 +0200424 rc = cast_string_realloc(LYD_CTX(node), 1, str, used, size);
Michal Vasko03ff5a72019-09-11 13:49:33 +0200425 LY_CHECK_RET(rc);
426 strcpy(*str + (*used - 1), "\n");
427 ++(*used);
428
429 ++indent;
430 }
431
432 switch (node->schema->nodetype) {
433 case LYS_CONTAINER:
434 case LYS_LIST:
435 case LYS_RPC:
436 case LYS_NOTIF:
Michal Vaskob7be7a82020-08-20 09:09:04 +0200437 rc = cast_string_realloc(LYD_CTX(node), 1, str, used, size);
Michal Vasko03ff5a72019-09-11 13:49:33 +0200438 LY_CHECK_RET(rc);
439 strcpy(*str + (*used - 1), "\n");
440 ++(*used);
441
Radek Krejcia1c1e542020-09-29 16:06:52 +0200442 for (child = lyd_child(node); child; child = child->next) {
Michal Vasko03ff5a72019-09-11 13:49:33 +0200443 rc = cast_string_recursive(child, 0, root_type, indent + 1, str, used, size);
444 LY_CHECK_RET(rc);
445 }
446
447 break;
448
449 case LYS_LEAF:
450 case LYS_LEAFLIST:
Radek Krejci6d5ba0c2021-04-26 07:49:59 +0200451 value_str = lyd_get_value(node);
Michal Vasko03ff5a72019-09-11 13:49:33 +0200452
453 /* print indent */
Michal Vaskob7be7a82020-08-20 09:09:04 +0200454 LY_CHECK_RET(cast_string_realloc(LYD_CTX(node), indent * 2 + strlen(value_str) + 1, str, used, size));
Michal Vasko03ff5a72019-09-11 13:49:33 +0200455 memset(*str + (*used - 1), ' ', indent * 2);
456 *used += indent * 2;
457
458 /* print value */
459 if (*used == 1) {
460 sprintf(*str + (*used - 1), "%s", value_str);
461 *used += strlen(value_str);
462 } else {
463 sprintf(*str + (*used - 1), "%s\n", value_str);
464 *used += strlen(value_str) + 1;
465 }
Michal Vasko03ff5a72019-09-11 13:49:33 +0200466
467 break;
468
469 case LYS_ANYXML:
470 case LYS_ANYDATA:
471 any = (struct lyd_node_any *)node;
472 if (!(void *)any->value.tree) {
473 /* no content */
474 buf = strdup("");
Michal Vaskob7be7a82020-08-20 09:09:04 +0200475 LY_CHECK_ERR_RET(!buf, LOGMEM(LYD_CTX(node)), LY_EMEM);
Michal Vasko03ff5a72019-09-11 13:49:33 +0200476 } else {
Radek Krejci241f6b52020-05-21 18:13:49 +0200477 struct ly_out *out;
Radek Krejcia5bba312020-01-09 15:41:20 +0100478
Michal Vasko60ea6352020-06-29 13:39:39 +0200479 if (any->value_type == LYD_ANYDATA_LYB) {
480 /* try to parse it into a data tree */
Michal Vaskoddd76592022-01-17 13:34:48 +0100481 if (lyd_parse_data_mem((struct ly_ctx *)LYD_CTX(node), any->value.mem, LYD_LYB,
482 LYD_PARSE_ONLY | LYD_PARSE_STRICT, 0, &tree) == LY_SUCCESS) {
Michal Vasko60ea6352020-06-29 13:39:39 +0200483 /* successfully parsed */
484 free(any->value.mem);
485 any->value.tree = tree;
486 any->value_type = LYD_ANYDATA_DATATREE;
487 }
Radek Krejci7931b192020-06-25 17:05:03 +0200488 /* error is covered by the following switch where LYD_ANYDATA_LYB causes failure */
Michal Vasko60ea6352020-06-29 13:39:39 +0200489 }
490
Michal Vasko03ff5a72019-09-11 13:49:33 +0200491 switch (any->value_type) {
492 case LYD_ANYDATA_STRING:
493 case LYD_ANYDATA_XML:
494 case LYD_ANYDATA_JSON:
495 buf = strdup(any->value.json);
Michal Vaskob7be7a82020-08-20 09:09:04 +0200496 LY_CHECK_ERR_RET(!buf, LOGMEM(LYD_CTX(node)), LY_EMEM);
Michal Vasko03ff5a72019-09-11 13:49:33 +0200497 break;
498 case LYD_ANYDATA_DATATREE:
Radek Krejci84ce7b12020-06-11 17:28:25 +0200499 LY_CHECK_RET(ly_out_new_memory(&buf, 0, &out));
Michal Vasko3a41dff2020-07-15 14:30:28 +0200500 rc = lyd_print_all(out, any->value.tree, LYD_XML, 0);
Radek Krejci241f6b52020-05-21 18:13:49 +0200501 ly_out_free(out, NULL, 0);
Radek Krejcia5bba312020-01-09 15:41:20 +0100502 LY_CHECK_RET(rc < 0, -rc);
Michal Vasko03ff5a72019-09-11 13:49:33 +0200503 break;
Michal Vasko60ea6352020-06-29 13:39:39 +0200504 case LYD_ANYDATA_LYB:
Michal Vaskob7be7a82020-08-20 09:09:04 +0200505 LOGERR(LYD_CTX(node), LY_EINVAL, "Cannot convert LYB anydata into string.");
Michal Vasko60ea6352020-06-29 13:39:39 +0200506 return LY_EINVAL;
Michal Vasko03ff5a72019-09-11 13:49:33 +0200507 }
508 }
509
510 line = strtok_r(buf, "\n", &ptr);
511 do {
Michal Vaskob7be7a82020-08-20 09:09:04 +0200512 rc = cast_string_realloc(LYD_CTX(node), indent * 2 + strlen(line) + 1, str, used, size);
Michal Vasko03ff5a72019-09-11 13:49:33 +0200513 if (rc != LY_SUCCESS) {
514 free(buf);
515 return rc;
516 }
517 memset(*str + (*used - 1), ' ', indent * 2);
518 *used += indent * 2;
519
520 strcpy(*str + (*used - 1), line);
521 *used += strlen(line);
522
523 strcpy(*str + (*used - 1), "\n");
524 *used += 1;
525 } while ((line = strtok_r(NULL, "\n", &ptr)));
526
527 free(buf);
528 break;
529
530 default:
Michal Vaskob7be7a82020-08-20 09:09:04 +0200531 LOGINT_RET(LYD_CTX(node));
Michal Vasko03ff5a72019-09-11 13:49:33 +0200532 }
533
534 if (fake_cont) {
Michal Vaskob7be7a82020-08-20 09:09:04 +0200535 rc = cast_string_realloc(LYD_CTX(node), 1, str, used, size);
Michal Vasko03ff5a72019-09-11 13:49:33 +0200536 LY_CHECK_RET(rc);
537 strcpy(*str + (*used - 1), "\n");
538 ++(*used);
539
540 --indent;
541 }
542
543 return LY_SUCCESS;
544}
545
546/**
547 * @brief Cast an element into a string.
548 *
549 * @param[in] node Node to cast.
550 * @param[in] fake_cont Whether to put the data into a "fake" container.
551 * @param[in] root_type Type of the XPath root.
552 * @param[out] str Element cast to dynamically-allocated string.
553 * @return LY_ERR
554 */
555static LY_ERR
Radek Krejci857189e2020-09-01 13:26:36 +0200556cast_string_elem(struct lyd_node *node, ly_bool fake_cont, enum lyxp_node_type root_type, char **str)
Michal Vasko03ff5a72019-09-11 13:49:33 +0200557{
558 uint16_t used, size;
559 LY_ERR rc;
560
561 *str = malloc(LYXP_STRING_CAST_SIZE_START * sizeof(char));
Michal Vaskob7be7a82020-08-20 09:09:04 +0200562 LY_CHECK_ERR_RET(!*str, LOGMEM(LYD_CTX(node)), LY_EMEM);
Michal Vasko03ff5a72019-09-11 13:49:33 +0200563 (*str)[0] = '\0';
564 used = 1;
565 size = LYXP_STRING_CAST_SIZE_START;
566
567 rc = cast_string_recursive(node, fake_cont, root_type, 0, str, &used, &size);
568 if (rc != LY_SUCCESS) {
569 free(*str);
570 return rc;
571 }
572
573 if (size > used) {
574 *str = ly_realloc(*str, used * sizeof(char));
Michal Vaskob7be7a82020-08-20 09:09:04 +0200575 LY_CHECK_ERR_RET(!*str, LOGMEM(LYD_CTX(node)), LY_EMEM);
Michal Vasko03ff5a72019-09-11 13:49:33 +0200576 }
577 return LY_SUCCESS;
578}
579
580/**
581 * @brief Cast a LYXP_SET_NODE_SET set into a string.
582 * Context position aware.
583 *
584 * @param[in] set Set to cast.
Michal Vasko03ff5a72019-09-11 13:49:33 +0200585 * @param[out] str Cast dynamically-allocated string.
586 * @return LY_ERR
587 */
588static LY_ERR
Michal Vasko5e0e6eb2019-11-06 15:47:50 +0100589cast_node_set_to_string(struct lyxp_set *set, char **str)
Michal Vasko03ff5a72019-09-11 13:49:33 +0200590{
Michal Vaskoaa677062021-03-09 13:52:53 +0100591 if (!set->used) {
592 *str = strdup("");
593 if (!*str) {
594 LOGMEM_RET(set->ctx);
595 }
596 return LY_SUCCESS;
597 }
598
Michal Vasko03ff5a72019-09-11 13:49:33 +0200599 switch (set->val.nodes[0].type) {
Michal Vasko2caefc12019-11-14 16:07:56 +0100600 case LYXP_NODE_NONE:
601 /* invalid */
602 LOGINT_RET(set->ctx);
Michal Vasko03ff5a72019-09-11 13:49:33 +0200603 case LYXP_NODE_ROOT:
604 case LYXP_NODE_ROOT_CONFIG:
Michal Vasko5e0e6eb2019-11-06 15:47:50 +0100605 return cast_string_elem(set->val.nodes[0].node, 1, set->root_type, str);
Michal Vasko03ff5a72019-09-11 13:49:33 +0200606 case LYXP_NODE_ELEM:
607 case LYXP_NODE_TEXT:
Michal Vasko5e0e6eb2019-11-06 15:47:50 +0100608 return cast_string_elem(set->val.nodes[0].node, 0, set->root_type, str);
Michal Vasko9f96a052020-03-10 09:41:45 +0100609 case LYXP_NODE_META:
Radek Krejci6d5ba0c2021-04-26 07:49:59 +0200610 *str = strdup(lyd_get_meta_value(set->val.meta[0].meta));
Michal Vaskoba99a3e2020-08-18 15:50:05 +0200611 if (!*str) {
612 LOGMEM_RET(set->ctx);
Michal Vasko03ff5a72019-09-11 13:49:33 +0200613 }
614 return LY_SUCCESS;
615 }
616
617 LOGINT_RET(set->ctx);
618}
619
620/**
621 * @brief Cast a string into an XPath number.
622 *
623 * @param[in] str String to use.
624 * @return Cast number.
625 */
626static long double
627cast_string_to_number(const char *str)
628{
629 long double num;
630 char *ptr;
631
632 errno = 0;
633 num = strtold(str, &ptr);
634 if (errno || *ptr) {
635 num = NAN;
636 }
637 return num;
638}
639
640/**
641 * @brief Callback for checking value equality.
642 *
Michal Vasko62524a92021-02-26 10:08:50 +0100643 * Implementation of ::lyht_value_equal_cb.
Radek Krejci857189e2020-09-01 13:26:36 +0200644 *
Michal Vasko03ff5a72019-09-11 13:49:33 +0200645 * @param[in] val1_p First value.
646 * @param[in] val2_p Second value.
647 * @param[in] mod Whether hash table is being modified.
648 * @param[in] cb_data Callback data.
Radek Krejci857189e2020-09-01 13:26:36 +0200649 * @return Boolean value whether values are equal or not.
Michal Vasko03ff5a72019-09-11 13:49:33 +0200650 */
Radek Krejci857189e2020-09-01 13:26:36 +0200651static ly_bool
652set_values_equal_cb(void *val1_p, void *val2_p, ly_bool UNUSED(mod), void *UNUSED(cb_data))
Michal Vasko03ff5a72019-09-11 13:49:33 +0200653{
654 struct lyxp_set_hash_node *val1, *val2;
655
656 val1 = (struct lyxp_set_hash_node *)val1_p;
657 val2 = (struct lyxp_set_hash_node *)val2_p;
658
659 if ((val1->node == val2->node) && (val1->type == val2->type)) {
660 return 1;
661 }
662
663 return 0;
664}
665
666/**
667 * @brief Insert node and its hash into set.
668 *
669 * @param[in] set et to insert to.
670 * @param[in] node Node with hash.
671 * @param[in] type Node type.
672 */
673static void
674set_insert_node_hash(struct lyxp_set *set, struct lyd_node *node, enum lyxp_node_type type)
675{
Radek Krejci1deb5be2020-08-26 16:43:36 +0200676 LY_ERR r;
Michal Vasko03ff5a72019-09-11 13:49:33 +0200677 uint32_t i, hash;
678 struct lyxp_set_hash_node hnode;
679
680 if (!set->ht && (set->used >= LYD_HT_MIN_ITEMS)) {
681 /* create hash table and add all the nodes */
682 set->ht = lyht_new(1, sizeof(struct lyxp_set_hash_node), set_values_equal_cb, NULL, 1);
683 for (i = 0; i < set->used; ++i) {
684 hnode.node = set->val.nodes[i].node;
685 hnode.type = set->val.nodes[i].type;
686
687 hash = dict_hash_multi(0, (const char *)&hnode.node, sizeof hnode.node);
688 hash = dict_hash_multi(hash, (const char *)&hnode.type, sizeof hnode.type);
689 hash = dict_hash_multi(hash, NULL, 0);
690
691 r = lyht_insert(set->ht, &hnode, hash, NULL);
692 assert(!r);
693 (void)r;
Michal Vasko03ff5a72019-09-11 13:49:33 +0200694
Michal Vasko9d6befd2019-12-11 14:56:56 +0100695 if (hnode.node == node) {
696 /* it was just added, do not add it twice */
697 node = NULL;
698 }
699 }
700 }
701
702 if (set->ht && node) {
Michal Vasko03ff5a72019-09-11 13:49:33 +0200703 /* add the new node into hash table */
704 hnode.node = node;
705 hnode.type = type;
706
707 hash = dict_hash_multi(0, (const char *)&hnode.node, sizeof hnode.node);
708 hash = dict_hash_multi(hash, (const char *)&hnode.type, sizeof hnode.type);
709 hash = dict_hash_multi(hash, NULL, 0);
710
711 r = lyht_insert(set->ht, &hnode, hash, NULL);
712 assert(!r);
713 (void)r;
714 }
715}
716
717/**
718 * @brief Remove node and its hash from set.
719 *
720 * @param[in] set Set to remove from.
721 * @param[in] node Node to remove.
722 * @param[in] type Node type.
723 */
724static void
725set_remove_node_hash(struct lyxp_set *set, struct lyd_node *node, enum lyxp_node_type type)
726{
Radek Krejci1deb5be2020-08-26 16:43:36 +0200727 LY_ERR r;
Michal Vasko03ff5a72019-09-11 13:49:33 +0200728 struct lyxp_set_hash_node hnode;
729 uint32_t hash;
730
731 if (set->ht) {
732 hnode.node = node;
733 hnode.type = type;
734
735 hash = dict_hash_multi(0, (const char *)&hnode.node, sizeof hnode.node);
736 hash = dict_hash_multi(hash, (const char *)&hnode.type, sizeof hnode.type);
737 hash = dict_hash_multi(hash, NULL, 0);
738
739 r = lyht_remove(set->ht, &hnode, hash);
740 assert(!r);
741 (void)r;
742
743 if (!set->ht->used) {
744 lyht_free(set->ht);
745 set->ht = NULL;
746 }
747 }
748}
749
750/**
751 * @brief Check whether node is in set based on its hash.
752 *
753 * @param[in] set Set to search in.
754 * @param[in] node Node to search for.
755 * @param[in] type Node type.
756 * @param[in] skip_idx Index in @p set to skip.
757 * @return LY_ERR
758 */
759static LY_ERR
760set_dup_node_hash_check(const struct lyxp_set *set, struct lyd_node *node, enum lyxp_node_type type, int skip_idx)
761{
762 struct lyxp_set_hash_node hnode, *match_p;
763 uint32_t hash;
764
765 hnode.node = node;
766 hnode.type = type;
767
768 hash = dict_hash_multi(0, (const char *)&hnode.node, sizeof hnode.node);
769 hash = dict_hash_multi(hash, (const char *)&hnode.type, sizeof hnode.type);
770 hash = dict_hash_multi(hash, NULL, 0);
771
772 if (!lyht_find(set->ht, &hnode, hash, (void **)&match_p)) {
773 if ((skip_idx > -1) && (set->val.nodes[skip_idx].node == match_p->node) && (set->val.nodes[skip_idx].type == match_p->type)) {
774 /* we found it on the index that should be skipped, find another */
775 hnode = *match_p;
776 if (lyht_find_next(set->ht, &hnode, hash, (void **)&match_p)) {
777 /* none other found */
778 return LY_SUCCESS;
779 }
780 }
781
782 return LY_EEXIST;
783 }
784
785 /* not found */
786 return LY_SUCCESS;
787}
788
Michal Vaskod3678892020-05-21 10:06:58 +0200789void
790lyxp_set_free_content(struct lyxp_set *set)
Michal Vasko03ff5a72019-09-11 13:49:33 +0200791{
792 if (!set) {
793 return;
794 }
795
796 if (set->type == LYXP_SET_NODE_SET) {
797 free(set->val.nodes);
798 lyht_free(set->ht);
Michal Vasko03ff5a72019-09-11 13:49:33 +0200799 } else if (set->type == LYXP_SET_SCNODE_SET) {
800 free(set->val.scnodes);
Michal Vaskod3678892020-05-21 10:06:58 +0200801 lyht_free(set->ht);
802 } else {
803 if (set->type == LYXP_SET_STRING) {
804 free(set->val.str);
805 }
806 set->type = LYXP_SET_NODE_SET;
Michal Vasko03ff5a72019-09-11 13:49:33 +0200807 }
Michal Vaskod3678892020-05-21 10:06:58 +0200808
809 set->val.nodes = NULL;
810 set->used = 0;
811 set->size = 0;
812 set->ht = NULL;
813 set->ctx_pos = 0;
aPiecek748da732021-06-01 09:56:43 +0200814 set->ctx_size = 0;
Michal Vasko03ff5a72019-09-11 13:49:33 +0200815}
816
Michal Vasko5e0e6eb2019-11-06 15:47:50 +0100817/**
818 * @brief Free dynamically-allocated set.
819 *
820 * @param[in] set Set to free.
821 */
822static void
Michal Vasko03ff5a72019-09-11 13:49:33 +0200823lyxp_set_free(struct lyxp_set *set)
824{
825 if (!set) {
826 return;
827 }
828
Michal Vaskod3678892020-05-21 10:06:58 +0200829 lyxp_set_free_content(set);
Michal Vasko03ff5a72019-09-11 13:49:33 +0200830 free(set);
831}
832
833/**
834 * @brief Initialize set context.
835 *
836 * @param[in] new Set to initialize.
837 * @param[in] set Arbitrary initialized set.
838 */
839static void
Michal Vasko4c7763f2020-07-27 17:40:37 +0200840set_init(struct lyxp_set *new, const struct lyxp_set *set)
Michal Vasko03ff5a72019-09-11 13:49:33 +0200841{
842 memset(new, 0, sizeof *new);
Michal Vasko02a77382019-09-12 11:47:35 +0200843 if (set) {
844 new->ctx = set->ctx;
Michal Vasko5d24f6c2020-10-13 13:49:06 +0200845 new->cur_node = set->cur_node;
Michal Vasko588112f2019-12-10 14:51:53 +0100846 new->root_type = set->root_type;
Michal Vasko6b26e742020-07-17 15:02:10 +0200847 new->context_op = set->context_op;
Michal Vaskof03ed032020-03-04 13:31:44 +0100848 new->tree = set->tree;
Michal Vasko5d24f6c2020-10-13 13:49:06 +0200849 new->cur_mod = set->cur_mod;
Michal Vasko02a77382019-09-12 11:47:35 +0200850 new->format = set->format;
Michal Vasko5d24f6c2020-10-13 13:49:06 +0200851 new->prefix_data = set->prefix_data;
aPiecekfba75362021-10-07 12:39:48 +0200852 new->vars = set->vars;
Michal Vasko02a77382019-09-12 11:47:35 +0200853 }
Michal Vasko03ff5a72019-09-11 13:49:33 +0200854}
855
856/**
857 * @brief Create a deep copy of a set.
858 *
859 * @param[in] set Set to copy.
860 * @return Copy of @p set.
861 */
862static struct lyxp_set *
863set_copy(struct lyxp_set *set)
864{
865 struct lyxp_set *ret;
Michal Vasko1fdd8fa2021-01-08 09:21:45 +0100866 uint32_t i;
Michal Vasko03ff5a72019-09-11 13:49:33 +0200867
868 if (!set) {
869 return NULL;
870 }
871
872 ret = malloc(sizeof *ret);
873 LY_CHECK_ERR_RET(!ret, LOGMEM(set->ctx), NULL);
874 set_init(ret, set);
875
876 if (set->type == LYXP_SET_SCNODE_SET) {
877 ret->type = set->type;
878
879 for (i = 0; i < set->used; ++i) {
Radek Krejcif13b87b2020-12-01 22:02:17 +0100880 if ((set->val.scnodes[i].in_ctx == LYXP_SET_SCNODE_ATOM_CTX) ||
881 (set->val.scnodes[i].in_ctx == LYXP_SET_SCNODE_START)) {
Radek Krejciaa6b53f2020-08-27 15:19:03 +0200882 uint32_t idx;
883 LY_CHECK_ERR_RET(lyxp_set_scnode_insert_node(ret, set->val.scnodes[i].scnode, set->val.scnodes[i].type, &idx),
884 lyxp_set_free(ret), NULL);
Michal Vasko3f27c522020-01-06 08:37:49 +0100885 /* coverity seems to think scnodes can be NULL */
Radek Krejciaa6b53f2020-08-27 15:19:03 +0200886 if (!ret->val.scnodes) {
Michal Vasko03ff5a72019-09-11 13:49:33 +0200887 lyxp_set_free(ret);
888 return NULL;
889 }
Michal Vaskoba716542019-12-16 10:01:58 +0100890 ret->val.scnodes[idx].in_ctx = set->val.scnodes[i].in_ctx;
Michal Vasko03ff5a72019-09-11 13:49:33 +0200891 }
892 }
893 } else if (set->type == LYXP_SET_NODE_SET) {
894 ret->type = set->type;
Michal Vasko08e9b112021-06-11 15:41:17 +0200895 if (set->used) {
896 ret->val.nodes = malloc(set->used * sizeof *ret->val.nodes);
897 LY_CHECK_ERR_RET(!ret->val.nodes, LOGMEM(set->ctx); free(ret), NULL);
898 memcpy(ret->val.nodes, set->val.nodes, set->used * sizeof *ret->val.nodes);
899 } else {
900 ret->val.nodes = NULL;
901 }
Michal Vasko03ff5a72019-09-11 13:49:33 +0200902
903 ret->used = ret->size = set->used;
904 ret->ctx_pos = set->ctx_pos;
905 ret->ctx_size = set->ctx_size;
Michal Vasko4a04e542021-02-01 08:58:30 +0100906 if (set->ht) {
907 ret->ht = lyht_dup(set->ht);
908 }
Michal Vasko03ff5a72019-09-11 13:49:33 +0200909 } else {
Radek Krejci0f969882020-08-21 16:56:47 +0200910 memcpy(ret, set, sizeof *ret);
911 if (set->type == LYXP_SET_STRING) {
912 ret->val.str = strdup(set->val.str);
913 LY_CHECK_ERR_RET(!ret->val.str, LOGMEM(set->ctx); free(ret), NULL);
914 }
Michal Vasko03ff5a72019-09-11 13:49:33 +0200915 }
916
917 return ret;
918}
919
920/**
921 * @brief Fill XPath set with a string. Any current data are disposed of.
922 *
923 * @param[in] set Set to fill.
924 * @param[in] string String to fill into \p set.
925 * @param[in] str_len Length of \p string. 0 is a valid value!
926 */
927static void
928set_fill_string(struct lyxp_set *set, const char *string, uint16_t str_len)
929{
Michal Vaskod3678892020-05-21 10:06:58 +0200930 lyxp_set_free_content(set);
Michal Vasko03ff5a72019-09-11 13:49:33 +0200931
932 set->type = LYXP_SET_STRING;
933 if ((str_len == 0) && (string[0] != '\0')) {
934 string = "";
935 }
936 set->val.str = strndup(string, str_len);
937}
938
939/**
940 * @brief Fill XPath set with a number. Any current data are disposed of.
941 *
942 * @param[in] set Set to fill.
943 * @param[in] number Number to fill into \p set.
944 */
945static void
946set_fill_number(struct lyxp_set *set, long double number)
947{
Michal Vaskod3678892020-05-21 10:06:58 +0200948 lyxp_set_free_content(set);
Michal Vasko03ff5a72019-09-11 13:49:33 +0200949
950 set->type = LYXP_SET_NUMBER;
951 set->val.num = number;
952}
953
954/**
955 * @brief Fill XPath set with a boolean. Any current data are disposed of.
956 *
957 * @param[in] set Set to fill.
958 * @param[in] boolean Boolean to fill into \p set.
959 */
960static void
Radek Krejci857189e2020-09-01 13:26:36 +0200961set_fill_boolean(struct lyxp_set *set, ly_bool boolean)
Michal Vasko03ff5a72019-09-11 13:49:33 +0200962{
Michal Vaskod3678892020-05-21 10:06:58 +0200963 lyxp_set_free_content(set);
Michal Vasko03ff5a72019-09-11 13:49:33 +0200964
965 set->type = LYXP_SET_BOOLEAN;
Michal Vasko004d3152020-06-11 19:59:22 +0200966 set->val.bln = boolean;
Michal Vasko03ff5a72019-09-11 13:49:33 +0200967}
968
969/**
970 * @brief Fill XPath set with the value from another set (deep assign).
971 * Any current data are disposed of.
972 *
973 * @param[in] trg Set to fill.
974 * @param[in] src Source set to copy into \p trg.
975 */
976static void
Michal Vasko4c7763f2020-07-27 17:40:37 +0200977set_fill_set(struct lyxp_set *trg, const struct lyxp_set *src)
Michal Vasko03ff5a72019-09-11 13:49:33 +0200978{
979 if (!trg || !src) {
980 return;
981 }
982
983 if (trg->type == LYXP_SET_NODE_SET) {
984 free(trg->val.nodes);
985 } else if (trg->type == LYXP_SET_STRING) {
986 free(trg->val.str);
987 }
988 set_init(trg, src);
989
990 if (src->type == LYXP_SET_SCNODE_SET) {
991 trg->type = LYXP_SET_SCNODE_SET;
992 trg->used = src->used;
993 trg->size = src->used;
994
Michal Vasko08e9b112021-06-11 15:41:17 +0200995 if (trg->size) {
996 trg->val.scnodes = ly_realloc(trg->val.scnodes, trg->size * sizeof *trg->val.scnodes);
997 LY_CHECK_ERR_RET(!trg->val.scnodes, LOGMEM(src->ctx); memset(trg, 0, sizeof *trg), );
998 memcpy(trg->val.scnodes, src->val.scnodes, src->used * sizeof *src->val.scnodes);
999 } else {
1000 trg->val.scnodes = NULL;
1001 }
Michal Vasko03ff5a72019-09-11 13:49:33 +02001002 } else if (src->type == LYXP_SET_BOOLEAN) {
Michal Vasko004d3152020-06-11 19:59:22 +02001003 set_fill_boolean(trg, src->val.bln);
Michal Vasko44f3d2c2020-08-24 09:49:38 +02001004 } else if (src->type == LYXP_SET_NUMBER) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02001005 set_fill_number(trg, src->val.num);
1006 } else if (src->type == LYXP_SET_STRING) {
1007 set_fill_string(trg, src->val.str, strlen(src->val.str));
1008 } else {
1009 if (trg->type == LYXP_SET_NODE_SET) {
1010 free(trg->val.nodes);
1011 } else if (trg->type == LYXP_SET_STRING) {
1012 free(trg->val.str);
1013 }
1014
Michal Vaskod3678892020-05-21 10:06:58 +02001015 assert(src->type == LYXP_SET_NODE_SET);
1016
1017 trg->type = LYXP_SET_NODE_SET;
1018 trg->used = src->used;
1019 trg->size = src->used;
1020 trg->ctx_pos = src->ctx_pos;
1021 trg->ctx_size = src->ctx_size;
1022
Michal Vasko08e9b112021-06-11 15:41:17 +02001023 if (trg->size) {
1024 trg->val.nodes = malloc(trg->size * sizeof *trg->val.nodes);
1025 LY_CHECK_ERR_RET(!trg->val.nodes, LOGMEM(src->ctx); memset(trg, 0, sizeof *trg), );
1026 memcpy(trg->val.nodes, src->val.nodes, src->used * sizeof *src->val.nodes);
1027 } else {
1028 trg->val.nodes = NULL;
1029 }
Michal Vaskod3678892020-05-21 10:06:58 +02001030 if (src->ht) {
1031 trg->ht = lyht_dup(src->ht);
Michal Vasko03ff5a72019-09-11 13:49:33 +02001032 } else {
Michal Vaskod3678892020-05-21 10:06:58 +02001033 trg->ht = NULL;
Michal Vasko03ff5a72019-09-11 13:49:33 +02001034 }
1035 }
1036}
1037
1038/**
1039 * @brief Clear context of all schema nodes.
1040 *
1041 * @param[in] set Set to clear.
Michal Vasko1a09b212021-05-06 13:00:10 +02001042 * @param[in] new_ctx New context state for all the nodes currently in the context.
Michal Vasko03ff5a72019-09-11 13:49:33 +02001043 */
1044static void
Michal Vasko1a09b212021-05-06 13:00:10 +02001045set_scnode_clear_ctx(struct lyxp_set *set, int32_t new_ctx)
Michal Vasko03ff5a72019-09-11 13:49:33 +02001046{
1047 uint32_t i;
1048
1049 for (i = 0; i < set->used; ++i) {
Radek Krejcif13b87b2020-12-01 22:02:17 +01001050 if (set->val.scnodes[i].in_ctx == LYXP_SET_SCNODE_ATOM_CTX) {
Michal Vasko1a09b212021-05-06 13:00:10 +02001051 set->val.scnodes[i].in_ctx = new_ctx;
Radek Krejcif13b87b2020-12-01 22:02:17 +01001052 } else if (set->val.scnodes[i].in_ctx == LYXP_SET_SCNODE_START) {
1053 set->val.scnodes[i].in_ctx = LYXP_SET_SCNODE_START_USED;
Michal Vasko03ff5a72019-09-11 13:49:33 +02001054 }
1055 }
1056}
1057
1058/**
1059 * @brief Remove a node from a set. Removing last node changes
1060 * set into LYXP_SET_EMPTY. Context position aware.
1061 *
1062 * @param[in] set Set to use.
1063 * @param[in] idx Index from @p set of the node to be removed.
1064 */
1065static void
1066set_remove_node(struct lyxp_set *set, uint32_t idx)
1067{
1068 assert(set && (set->type == LYXP_SET_NODE_SET));
1069 assert(idx < set->used);
1070
1071 set_remove_node_hash(set, set->val.nodes[idx].node, set->val.nodes[idx].type);
1072
1073 --set->used;
Michal Vasko08e9b112021-06-11 15:41:17 +02001074 if (idx < set->used) {
1075 memmove(&set->val.nodes[idx], &set->val.nodes[idx + 1], (set->used - idx) * sizeof *set->val.nodes);
1076 } else if (!set->used) {
Michal Vaskod3678892020-05-21 10:06:58 +02001077 lyxp_set_free_content(set);
Michal Vasko03ff5a72019-09-11 13:49:33 +02001078 }
1079}
1080
1081/**
Michal Vasko2caefc12019-11-14 16:07:56 +01001082 * @brief Remove a node from a set by setting its type to LYXP_NODE_NONE.
Michal Vasko57eab132019-09-24 11:46:26 +02001083 *
1084 * @param[in] set Set to use.
1085 * @param[in] idx Index from @p set of the node to be removed.
1086 */
1087static void
Michal Vasko2caefc12019-11-14 16:07:56 +01001088set_remove_node_none(struct lyxp_set *set, uint32_t idx)
Michal Vasko57eab132019-09-24 11:46:26 +02001089{
1090 assert(set && (set->type == LYXP_SET_NODE_SET));
1091 assert(idx < set->used);
1092
Michal Vasko2caefc12019-11-14 16:07:56 +01001093 if (set->val.nodes[idx].type == LYXP_NODE_ELEM) {
1094 set_remove_node_hash(set, set->val.nodes[idx].node, set->val.nodes[idx].type);
1095 }
1096 set->val.nodes[idx].type = LYXP_NODE_NONE;
Michal Vasko57eab132019-09-24 11:46:26 +02001097}
1098
1099/**
Michal Vasko2caefc12019-11-14 16:07:56 +01001100 * @brief Remove all LYXP_NODE_NONE nodes from a set. Removing last node changes
Michal Vasko57eab132019-09-24 11:46:26 +02001101 * set into LYXP_SET_EMPTY. Context position aware.
1102 *
1103 * @param[in] set Set to consolidate.
1104 */
1105static void
Michal Vasko2caefc12019-11-14 16:07:56 +01001106set_remove_nodes_none(struct lyxp_set *set)
Michal Vasko57eab132019-09-24 11:46:26 +02001107{
Michal Vasko1fdd8fa2021-01-08 09:21:45 +01001108 uint32_t i, orig_used, end = 0;
1109 int64_t start;
Michal Vasko57eab132019-09-24 11:46:26 +02001110
Michal Vaskod3678892020-05-21 10:06:58 +02001111 assert(set);
Michal Vasko57eab132019-09-24 11:46:26 +02001112
1113 orig_used = set->used;
1114 set->used = 0;
Michal Vaskod989ba02020-08-24 10:59:24 +02001115 for (i = 0; i < orig_used; ) {
Michal Vasko57eab132019-09-24 11:46:26 +02001116 start = -1;
1117 do {
Michal Vasko2caefc12019-11-14 16:07:56 +01001118 if ((set->val.nodes[i].type != LYXP_NODE_NONE) && (start == -1)) {
Michal Vasko57eab132019-09-24 11:46:26 +02001119 start = i;
Michal Vasko2caefc12019-11-14 16:07:56 +01001120 } else if ((start > -1) && (set->val.nodes[i].type == LYXP_NODE_NONE)) {
Michal Vasko57eab132019-09-24 11:46:26 +02001121 end = i;
1122 ++i;
1123 break;
1124 }
1125
1126 ++i;
1127 if (i == orig_used) {
1128 end = i;
1129 }
1130 } while (i < orig_used);
1131
1132 if (start > -1) {
1133 /* move the whole chunk of valid nodes together */
1134 if (set->used != (unsigned)start) {
1135 memmove(&set->val.nodes[set->used], &set->val.nodes[start], (end - start) * sizeof *set->val.nodes);
1136 }
1137 set->used += end - start;
1138 }
1139 }
Michal Vasko57eab132019-09-24 11:46:26 +02001140}
1141
1142/**
Michal Vasko03ff5a72019-09-11 13:49:33 +02001143 * @brief Check for duplicates in a node set.
1144 *
1145 * @param[in] set Set to check.
1146 * @param[in] node Node to look for in @p set.
1147 * @param[in] node_type Type of @p node.
1148 * @param[in] skip_idx Index from @p set to skip.
1149 * @return LY_ERR
1150 */
1151static LY_ERR
1152set_dup_node_check(const struct lyxp_set *set, const struct lyd_node *node, enum lyxp_node_type node_type, int skip_idx)
1153{
1154 uint32_t i;
1155
Michal Vasko2caefc12019-11-14 16:07:56 +01001156 if (set->ht && node) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02001157 return set_dup_node_hash_check(set, (struct lyd_node *)node, node_type, skip_idx);
1158 }
1159
1160 for (i = 0; i < set->used; ++i) {
1161 if ((skip_idx > -1) && (i == (unsigned)skip_idx)) {
1162 continue;
1163 }
1164
1165 if ((set->val.nodes[i].node == node) && (set->val.nodes[i].type == node_type)) {
1166 return LY_EEXIST;
1167 }
1168 }
1169
1170 return LY_SUCCESS;
1171}
1172
Radek Krejci857189e2020-09-01 13:26:36 +02001173ly_bool
Radek Krejciaa6b53f2020-08-27 15:19:03 +02001174lyxp_set_scnode_contains(struct lyxp_set *set, const struct lysc_node *node, enum lyxp_node_type node_type, int skip_idx,
1175 uint32_t *index_p)
Michal Vasko03ff5a72019-09-11 13:49:33 +02001176{
1177 uint32_t i;
1178
1179 for (i = 0; i < set->used; ++i) {
1180 if ((skip_idx > -1) && (i == (unsigned)skip_idx)) {
1181 continue;
1182 }
1183
1184 if ((set->val.scnodes[i].scnode == node) && (set->val.scnodes[i].type == node_type)) {
Radek Krejciaa6b53f2020-08-27 15:19:03 +02001185 if (index_p) {
1186 *index_p = i;
1187 }
1188 return 1;
Michal Vasko03ff5a72019-09-11 13:49:33 +02001189 }
1190 }
1191
Radek Krejciaa6b53f2020-08-27 15:19:03 +02001192 return 0;
Michal Vasko03ff5a72019-09-11 13:49:33 +02001193}
1194
Michal Vaskoecd62de2019-11-13 12:35:11 +01001195void
1196lyxp_set_scnode_merge(struct lyxp_set *set1, struct lyxp_set *set2)
Michal Vasko03ff5a72019-09-11 13:49:33 +02001197{
1198 uint32_t orig_used, i, j;
1199
Michal Vaskod3678892020-05-21 10:06:58 +02001200 assert((set1->type == LYXP_SET_SCNODE_SET) && (set2->type == LYXP_SET_SCNODE_SET));
Michal Vasko03ff5a72019-09-11 13:49:33 +02001201
Michal Vaskod3678892020-05-21 10:06:58 +02001202 if (!set2->used) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02001203 return;
1204 }
1205
Michal Vaskod3678892020-05-21 10:06:58 +02001206 if (!set1->used) {
aPiecekadc1e4f2021-10-07 11:15:12 +02001207 /* release hidden allocated data (lyxp_set.size) */
1208 lyxp_set_free_content(set1);
1209 /* direct copying of the entire structure */
Michal Vasko03ff5a72019-09-11 13:49:33 +02001210 memcpy(set1, set2, sizeof *set1);
1211 return;
1212 }
1213
1214 if (set1->used + set2->used > set1->size) {
1215 set1->size = set1->used + set2->used;
1216 set1->val.scnodes = ly_realloc(set1->val.scnodes, set1->size * sizeof *set1->val.scnodes);
1217 LY_CHECK_ERR_RET(!set1->val.scnodes, LOGMEM(set1->ctx), );
1218 }
1219
1220 orig_used = set1->used;
1221
1222 for (i = 0; i < set2->used; ++i) {
1223 for (j = 0; j < orig_used; ++j) {
1224 /* detect duplicities */
1225 if (set1->val.scnodes[j].scnode == set2->val.scnodes[i].scnode) {
1226 break;
1227 }
1228 }
1229
Michal Vasko0587bce2020-12-10 12:19:21 +01001230 if (j < orig_used) {
1231 /* node is there, but update its status if needed */
1232 if (set1->val.scnodes[j].in_ctx == LYXP_SET_SCNODE_START_USED) {
1233 set1->val.scnodes[j].in_ctx = set2->val.scnodes[i].in_ctx;
Michal Vasko1a09b212021-05-06 13:00:10 +02001234 } else if ((set1->val.scnodes[j].in_ctx == LYXP_SET_SCNODE_ATOM_NODE) &&
1235 (set2->val.scnodes[i].in_ctx == LYXP_SET_SCNODE_ATOM_VAL)) {
1236 set1->val.scnodes[j].in_ctx = set2->val.scnodes[i].in_ctx;
Michal Vasko0587bce2020-12-10 12:19:21 +01001237 }
1238 } else {
Michal Vasko03ff5a72019-09-11 13:49:33 +02001239 memcpy(&set1->val.scnodes[set1->used], &set2->val.scnodes[i], sizeof *set2->val.scnodes);
1240 ++set1->used;
1241 }
1242 }
1243
Michal Vaskod3678892020-05-21 10:06:58 +02001244 lyxp_set_free_content(set2);
1245 set2->type = LYXP_SET_SCNODE_SET;
Michal Vasko03ff5a72019-09-11 13:49:33 +02001246}
1247
1248/**
1249 * @brief Insert a node into a set. Context position aware.
1250 *
1251 * @param[in] set Set to use.
1252 * @param[in] node Node to insert to @p set.
1253 * @param[in] pos Sort position of @p node. If left 0, it is filled just before sorting.
1254 * @param[in] node_type Node type of @p node.
1255 * @param[in] idx Index in @p set to insert into.
1256 */
1257static void
1258set_insert_node(struct lyxp_set *set, const struct lyd_node *node, uint32_t pos, enum lyxp_node_type node_type, uint32_t idx)
1259{
Michal Vaskod3678892020-05-21 10:06:58 +02001260 assert(set && (set->type == LYXP_SET_NODE_SET));
Michal Vasko03ff5a72019-09-11 13:49:33 +02001261
Michal Vaskod3678892020-05-21 10:06:58 +02001262 if (!set->size) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02001263 /* first item */
1264 if (idx) {
1265 /* no real harm done, but it is a bug */
1266 LOGINT(set->ctx);
1267 idx = 0;
1268 }
1269 set->val.nodes = malloc(LYXP_SET_SIZE_START * sizeof *set->val.nodes);
1270 LY_CHECK_ERR_RET(!set->val.nodes, LOGMEM(set->ctx), );
1271 set->type = LYXP_SET_NODE_SET;
1272 set->used = 0;
1273 set->size = LYXP_SET_SIZE_START;
1274 set->ctx_pos = 1;
1275 set->ctx_size = 1;
1276 set->ht = NULL;
1277 } else {
1278 /* not an empty set */
1279 if (set->used == set->size) {
1280
1281 /* set is full */
Michal Vasko871df522022-04-06 12:14:41 +02001282 set->val.nodes = ly_realloc(set->val.nodes, (set->size * LYXP_SET_SIZE_MUL_STEP) * sizeof *set->val.nodes);
Michal Vasko03ff5a72019-09-11 13:49:33 +02001283 LY_CHECK_ERR_RET(!set->val.nodes, LOGMEM(set->ctx), );
Michal Vasko871df522022-04-06 12:14:41 +02001284 set->size *= LYXP_SET_SIZE_MUL_STEP;
Michal Vasko03ff5a72019-09-11 13:49:33 +02001285 }
1286
1287 if (idx > set->used) {
1288 LOGINT(set->ctx);
1289 idx = set->used;
1290 }
1291
1292 /* make space for the new node */
1293 if (idx < set->used) {
1294 memmove(&set->val.nodes[idx + 1], &set->val.nodes[idx], (set->used - idx) * sizeof *set->val.nodes);
1295 }
1296 }
1297
1298 /* finally assign the value */
1299 set->val.nodes[idx].node = (struct lyd_node *)node;
1300 set->val.nodes[idx].type = node_type;
1301 set->val.nodes[idx].pos = pos;
1302 ++set->used;
1303
Michal Vasko2416fdd2021-10-01 11:07:10 +02001304 /* add into hash table */
1305 set_insert_node_hash(set, (struct lyd_node *)node, node_type);
Michal Vasko03ff5a72019-09-11 13:49:33 +02001306}
1307
Radek Krejciaa6b53f2020-08-27 15:19:03 +02001308LY_ERR
Michal Vaskoddd76592022-01-17 13:34:48 +01001309lyxp_set_scnode_insert_node(struct lyxp_set *set, const struct lysc_node *node, enum lyxp_node_type node_type,
1310 uint32_t *index_p)
Michal Vasko03ff5a72019-09-11 13:49:33 +02001311{
Radek Krejciaa6b53f2020-08-27 15:19:03 +02001312 uint32_t index;
Michal Vasko03ff5a72019-09-11 13:49:33 +02001313
1314 assert(set->type == LYXP_SET_SCNODE_SET);
1315
Michal Vasko871df522022-04-06 12:14:41 +02001316 if (!set->size) {
1317 /* first item */
1318 set->val.scnodes = malloc(LYXP_SET_SIZE_START * sizeof *set->val.scnodes);
1319 LY_CHECK_ERR_RET(!set->val.scnodes, LOGMEM(set->ctx), LY_EMEM);
1320 set->type = LYXP_SET_SCNODE_SET;
1321 set->used = 0;
1322 set->size = LYXP_SET_SIZE_START;
1323 set->ctx_pos = 1;
1324 set->ctx_size = 1;
1325 set->ht = NULL;
1326 }
1327
Radek Krejciaa6b53f2020-08-27 15:19:03 +02001328 if (lyxp_set_scnode_contains(set, node, node_type, -1, &index)) {
Radek Krejcif13b87b2020-12-01 22:02:17 +01001329 set->val.scnodes[index].in_ctx = LYXP_SET_SCNODE_ATOM_CTX;
Michal Vasko03ff5a72019-09-11 13:49:33 +02001330 } else {
1331 if (set->used == set->size) {
Michal Vasko871df522022-04-06 12:14:41 +02001332 set->val.scnodes = ly_realloc(set->val.scnodes, (set->size * LYXP_SET_SIZE_MUL_STEP) * sizeof *set->val.scnodes);
Radek Krejciaa6b53f2020-08-27 15:19:03 +02001333 LY_CHECK_ERR_RET(!set->val.scnodes, LOGMEM(set->ctx), LY_EMEM);
Michal Vasko871df522022-04-06 12:14:41 +02001334 set->size *= LYXP_SET_SIZE_MUL_STEP;
Michal Vasko03ff5a72019-09-11 13:49:33 +02001335 }
1336
Radek Krejciaa6b53f2020-08-27 15:19:03 +02001337 index = set->used;
1338 set->val.scnodes[index].scnode = (struct lysc_node *)node;
1339 set->val.scnodes[index].type = node_type;
Radek Krejcif13b87b2020-12-01 22:02:17 +01001340 set->val.scnodes[index].in_ctx = LYXP_SET_SCNODE_ATOM_CTX;
Michal Vasko03ff5a72019-09-11 13:49:33 +02001341 ++set->used;
1342 }
1343
Radek Krejciaa6b53f2020-08-27 15:19:03 +02001344 if (index_p) {
1345 *index_p = index;
1346 }
1347
1348 return LY_SUCCESS;
Michal Vasko03ff5a72019-09-11 13:49:33 +02001349}
1350
1351/**
Michal Vasko03ff5a72019-09-11 13:49:33 +02001352 * @brief Set all nodes with ctx 1 to a new unique context value.
1353 *
1354 * @param[in] set Set to modify.
1355 * @return New context value.
1356 */
Michal Vasko5c4e5892019-11-14 12:31:38 +01001357static int32_t
Michal Vasko03ff5a72019-09-11 13:49:33 +02001358set_scnode_new_in_ctx(struct lyxp_set *set)
1359{
Michal Vasko5c4e5892019-11-14 12:31:38 +01001360 uint32_t i;
1361 int32_t ret_ctx;
Michal Vasko03ff5a72019-09-11 13:49:33 +02001362
1363 assert(set->type == LYXP_SET_SCNODE_SET);
1364
Radek Krejcif13b87b2020-12-01 22:02:17 +01001365 ret_ctx = LYXP_SET_SCNODE_ATOM_PRED_CTX;
Michal Vasko03ff5a72019-09-11 13:49:33 +02001366retry:
1367 for (i = 0; i < set->used; ++i) {
1368 if (set->val.scnodes[i].in_ctx >= ret_ctx) {
1369 ret_ctx = set->val.scnodes[i].in_ctx + 1;
1370 goto retry;
1371 }
1372 }
1373 for (i = 0; i < set->used; ++i) {
Radek Krejcif13b87b2020-12-01 22:02:17 +01001374 if (set->val.scnodes[i].in_ctx == LYXP_SET_SCNODE_ATOM_CTX) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02001375 set->val.scnodes[i].in_ctx = ret_ctx;
1376 }
1377 }
1378
1379 return ret_ctx;
1380}
1381
1382/**
1383 * @brief Get unique @p node position in the data.
1384 *
1385 * @param[in] node Node to find.
1386 * @param[in] node_type Node type of @p node.
1387 * @param[in] root Root node.
1388 * @param[in] root_type Type of the XPath @p root node.
1389 * @param[in] prev Node that we think is before @p node in DFS from @p root. Can optionally
1390 * be used to increase efficiency and start the DFS from this node.
1391 * @param[in] prev_pos Node @p prev position. Optional, but must be set if @p prev is set.
1392 * @return Node position.
1393 */
1394static uint32_t
1395get_node_pos(const struct lyd_node *node, enum lyxp_node_type node_type, const struct lyd_node *root,
Radek Krejci0f969882020-08-21 16:56:47 +02001396 enum lyxp_node_type root_type, const struct lyd_node **prev, uint32_t *prev_pos)
Michal Vasko03ff5a72019-09-11 13:49:33 +02001397{
Michal Vasko56daf732020-08-10 10:57:18 +02001398 const struct lyd_node *elem = NULL, *top_sibling;
Michal Vasko03ff5a72019-09-11 13:49:33 +02001399 uint32_t pos = 1;
Michal Vaskofb6c9dd2020-11-18 18:18:47 +01001400 ly_bool found = 0;
Michal Vasko03ff5a72019-09-11 13:49:33 +02001401
1402 assert(prev && prev_pos && !root->prev->next);
1403
1404 if ((node_type == LYXP_NODE_ROOT) || (node_type == LYXP_NODE_ROOT_CONFIG)) {
1405 return 0;
1406 }
1407
1408 if (*prev) {
1409 /* start from the previous element instead from the root */
Michal Vasko03ff5a72019-09-11 13:49:33 +02001410 pos = *prev_pos;
Michal Vaskofb6c9dd2020-11-18 18:18:47 +01001411 for (top_sibling = *prev; top_sibling->parent; top_sibling = lyd_parent(top_sibling)) {}
Michal Vasko03ff5a72019-09-11 13:49:33 +02001412 goto dfs_search;
1413 }
1414
Michal Vaskofb6c9dd2020-11-18 18:18:47 +01001415 LY_LIST_FOR(root, top_sibling) {
Michal Vasko56daf732020-08-10 10:57:18 +02001416 LYD_TREE_DFS_BEGIN(top_sibling, elem) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02001417dfs_search:
Michal Vaskoa9309bb2021-07-09 09:31:55 +02001418 LYD_TREE_DFS_continue = 0;
1419
Michal Vasko56daf732020-08-10 10:57:18 +02001420 if (*prev && !elem) {
1421 /* resume previous DFS */
1422 elem = LYD_TREE_DFS_next = (struct lyd_node *)*prev;
1423 LYD_TREE_DFS_continue = 0;
1424 }
1425
Michal Vasko482a6822022-05-06 08:56:12 +02001426 if (!elem->schema || ((root_type == LYXP_NODE_ROOT_CONFIG) && (elem->schema->flags & LYS_CONFIG_R))) {
Michal Vasko56daf732020-08-10 10:57:18 +02001427 /* skip */
1428 LYD_TREE_DFS_continue = 1;
Michal Vasko03ff5a72019-09-11 13:49:33 +02001429 } else {
Michal Vasko56daf732020-08-10 10:57:18 +02001430 if (elem == node) {
Michal Vaskofb6c9dd2020-11-18 18:18:47 +01001431 found = 1;
Michal Vasko03ff5a72019-09-11 13:49:33 +02001432 break;
1433 }
Michal Vasko56daf732020-08-10 10:57:18 +02001434 ++pos;
Michal Vasko03ff5a72019-09-11 13:49:33 +02001435 }
Michal Vasko56daf732020-08-10 10:57:18 +02001436
1437 LYD_TREE_DFS_END(top_sibling, elem);
Michal Vasko03ff5a72019-09-11 13:49:33 +02001438 }
1439
1440 /* node found */
Michal Vaskofb6c9dd2020-11-18 18:18:47 +01001441 if (found) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02001442 break;
1443 }
1444 }
1445
Michal Vaskofb6c9dd2020-11-18 18:18:47 +01001446 if (!found) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02001447 if (!(*prev)) {
1448 /* we went from root and failed to find it, cannot be */
Michal Vaskob7be7a82020-08-20 09:09:04 +02001449 LOGINT(LYD_CTX(node));
Michal Vasko03ff5a72019-09-11 13:49:33 +02001450 return 0;
1451 } else {
Michal Vasko56daf732020-08-10 10:57:18 +02001452 /* start the search again from the beginning */
1453 *prev = root;
Michal Vasko03ff5a72019-09-11 13:49:33 +02001454
Michal Vasko56daf732020-08-10 10:57:18 +02001455 top_sibling = root;
Michal Vasko03ff5a72019-09-11 13:49:33 +02001456 pos = 1;
1457 goto dfs_search;
1458 }
1459 }
1460
1461 /* remember the last found node for next time */
1462 *prev = node;
1463 *prev_pos = pos;
1464
1465 return pos;
1466}
1467
1468/**
1469 * @brief Assign (fill) missing node positions.
1470 *
1471 * @param[in] set Set to fill positions in.
1472 * @param[in] root Context root node.
1473 * @param[in] root_type Context root type.
1474 * @return LY_ERR
1475 */
1476static LY_ERR
1477set_assign_pos(struct lyxp_set *set, const struct lyd_node *root, enum lyxp_node_type root_type)
1478{
1479 const struct lyd_node *prev = NULL, *tmp_node;
1480 uint32_t i, tmp_pos = 0;
1481
1482 for (i = 0; i < set->used; ++i) {
1483 if (!set->val.nodes[i].pos) {
1484 tmp_node = NULL;
1485 switch (set->val.nodes[i].type) {
Michal Vasko9f96a052020-03-10 09:41:45 +01001486 case LYXP_NODE_META:
1487 tmp_node = set->val.meta[i].meta->parent;
Michal Vasko03ff5a72019-09-11 13:49:33 +02001488 if (!tmp_node) {
1489 LOGINT_RET(root->schema->module->ctx);
1490 }
Radek Krejcif13b87b2020-12-01 22:02:17 +01001491 /* fall through */
Michal Vasko03ff5a72019-09-11 13:49:33 +02001492 case LYXP_NODE_ELEM:
1493 case LYXP_NODE_TEXT:
1494 if (!tmp_node) {
1495 tmp_node = set->val.nodes[i].node;
1496 }
1497 set->val.nodes[i].pos = get_node_pos(tmp_node, set->val.nodes[i].type, root, root_type, &prev, &tmp_pos);
1498 break;
1499 default:
1500 /* all roots have position 0 */
1501 break;
1502 }
1503 }
1504 }
1505
1506 return LY_SUCCESS;
1507}
1508
1509/**
Michal Vasko9f96a052020-03-10 09:41:45 +01001510 * @brief Get unique @p meta position in the parent metadata.
Michal Vasko03ff5a72019-09-11 13:49:33 +02001511 *
Michal Vasko9f96a052020-03-10 09:41:45 +01001512 * @param[in] meta Metadata to use.
1513 * @return Metadata position.
Michal Vasko03ff5a72019-09-11 13:49:33 +02001514 */
1515static uint16_t
Michal Vasko9f96a052020-03-10 09:41:45 +01001516get_meta_pos(struct lyd_meta *meta)
Michal Vasko03ff5a72019-09-11 13:49:33 +02001517{
1518 uint16_t pos = 0;
Michal Vasko9f96a052020-03-10 09:41:45 +01001519 struct lyd_meta *meta2;
Michal Vasko03ff5a72019-09-11 13:49:33 +02001520
Michal Vasko9f96a052020-03-10 09:41:45 +01001521 for (meta2 = meta->parent->meta; meta2 && (meta2 != meta); meta2 = meta2->next) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02001522 ++pos;
1523 }
1524
Michal Vasko9f96a052020-03-10 09:41:45 +01001525 assert(meta2);
Michal Vasko03ff5a72019-09-11 13:49:33 +02001526 return pos;
1527}
1528
1529/**
1530 * @brief Compare 2 nodes in respect to XPath document order.
1531 *
1532 * @param[in] item1 1st node.
1533 * @param[in] item2 2nd node.
1534 * @return If 1st > 2nd returns 1, 1st == 2nd returns 0, and 1st < 2nd returns -1.
1535 */
1536static int
1537set_sort_compare(struct lyxp_set_node *item1, struct lyxp_set_node *item2)
1538{
Michal Vasko9f96a052020-03-10 09:41:45 +01001539 uint32_t meta_pos1 = 0, meta_pos2 = 0;
Michal Vasko03ff5a72019-09-11 13:49:33 +02001540
1541 if (item1->pos < item2->pos) {
1542 return -1;
1543 }
1544
1545 if (item1->pos > item2->pos) {
1546 return 1;
1547 }
1548
1549 /* node positions are equal, the fun case */
1550
1551 /* 1st ELEM - == - 2nd TEXT, 1st TEXT - == - 2nd ELEM */
1552 /* special case since text nodes are actually saved as their parents */
1553 if ((item1->node == item2->node) && (item1->type != item2->type)) {
1554 if (item1->type == LYXP_NODE_ELEM) {
1555 assert(item2->type == LYXP_NODE_TEXT);
1556 return -1;
1557 } else {
1558 assert((item1->type == LYXP_NODE_TEXT) && (item2->type == LYXP_NODE_ELEM));
1559 return 1;
1560 }
1561 }
1562
Michal Vasko9f96a052020-03-10 09:41:45 +01001563 /* we need meta positions now */
1564 if (item1->type == LYXP_NODE_META) {
1565 meta_pos1 = get_meta_pos((struct lyd_meta *)item1->node);
Michal Vasko03ff5a72019-09-11 13:49:33 +02001566 }
Michal Vasko9f96a052020-03-10 09:41:45 +01001567 if (item2->type == LYXP_NODE_META) {
1568 meta_pos2 = get_meta_pos((struct lyd_meta *)item2->node);
Michal Vasko03ff5a72019-09-11 13:49:33 +02001569 }
1570
Michal Vasko9f96a052020-03-10 09:41:45 +01001571 /* 1st ROOT - 2nd ROOT, 1st ELEM - 2nd ELEM, 1st TEXT - 2nd TEXT, 1st META - =pos= - 2nd META */
Michal Vasko03ff5a72019-09-11 13:49:33 +02001572 /* check for duplicates */
1573 if (item1->node == item2->node) {
Michal Vasko9f96a052020-03-10 09:41:45 +01001574 assert((item1->type == item2->type) && ((item1->type != LYXP_NODE_META) || (meta_pos1 == meta_pos2)));
Michal Vasko03ff5a72019-09-11 13:49:33 +02001575 return 0;
1576 }
1577
Michal Vasko9f96a052020-03-10 09:41:45 +01001578 /* 1st ELEM - 2nd TEXT, 1st ELEM - any pos - 2nd META */
Michal Vasko03ff5a72019-09-11 13:49:33 +02001579 /* elem is always first, 2nd node is after it */
1580 if (item1->type == LYXP_NODE_ELEM) {
1581 assert(item2->type != LYXP_NODE_ELEM);
1582 return -1;
1583 }
1584
Michal Vasko9f96a052020-03-10 09:41:45 +01001585 /* 1st TEXT - 2nd ELEM, 1st TEXT - any pos - 2nd META, 1st META - any pos - 2nd ELEM, 1st META - >pos> - 2nd META */
Michal Vasko03ff5a72019-09-11 13:49:33 +02001586 /* 2nd is before 1st */
Michal Vasko69730152020-10-09 16:30:07 +02001587 if (((item1->type == LYXP_NODE_TEXT) &&
1588 ((item2->type == LYXP_NODE_ELEM) || (item2->type == LYXP_NODE_META))) ||
1589 ((item1->type == LYXP_NODE_META) && (item2->type == LYXP_NODE_ELEM)) ||
1590 (((item1->type == LYXP_NODE_META) && (item2->type == LYXP_NODE_META)) &&
1591 (meta_pos1 > meta_pos2))) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02001592 return 1;
1593 }
1594
Michal Vasko9f96a052020-03-10 09:41:45 +01001595 /* 1st META - any pos - 2nd TEXT, 1st META <pos< - 2nd META */
Michal Vasko03ff5a72019-09-11 13:49:33 +02001596 /* 2nd is after 1st */
1597 return -1;
1598}
1599
1600/**
1601 * @brief Set cast for comparisons.
1602 *
1603 * @param[in] trg Target set to cast source into.
1604 * @param[in] src Source set.
1605 * @param[in] type Target set type.
1606 * @param[in] src_idx Source set node index.
Michal Vasko03ff5a72019-09-11 13:49:33 +02001607 * @return LY_ERR
1608 */
1609static LY_ERR
Michal Vasko5e0e6eb2019-11-06 15:47:50 +01001610set_comp_cast(struct lyxp_set *trg, struct lyxp_set *src, enum lyxp_set_type type, uint32_t src_idx)
Michal Vasko03ff5a72019-09-11 13:49:33 +02001611{
1612 assert(src->type == LYXP_SET_NODE_SET);
1613
1614 set_init(trg, src);
1615
1616 /* insert node into target set */
1617 set_insert_node(trg, src->val.nodes[src_idx].node, src->val.nodes[src_idx].pos, src->val.nodes[src_idx].type, 0);
1618
1619 /* cast target set appropriately */
Michal Vasko5e0e6eb2019-11-06 15:47:50 +01001620 return lyxp_set_cast(trg, type);
Michal Vasko03ff5a72019-09-11 13:49:33 +02001621}
1622
Michal Vasko4c7763f2020-07-27 17:40:37 +02001623/**
1624 * @brief Set content canonization for comparisons.
1625 *
1626 * @param[in] trg Target set to put the canononized source into.
1627 * @param[in] src Source set.
1628 * @param[in] xp_node Source XPath node/meta to use for canonization.
1629 * @return LY_ERR
1630 */
1631static LY_ERR
1632set_comp_canonize(struct lyxp_set *trg, const struct lyxp_set *src, const struct lyxp_set_node *xp_node)
1633{
Michal Vaskofeca4fb2020-10-05 08:58:40 +02001634 const struct lysc_type *type = NULL;
Michal Vasko4c7763f2020-07-27 17:40:37 +02001635 struct lyd_value val;
1636 struct ly_err_item *err = NULL;
1637 char *str, *ptr;
Michal Vasko4c7763f2020-07-27 17:40:37 +02001638 LY_ERR rc;
1639
1640 /* is there anything to canonize even? */
1641 if ((src->type == LYXP_SET_NUMBER) || (src->type == LYXP_SET_STRING)) {
1642 /* do we have a type to use for canonization? */
1643 if ((xp_node->type == LYXP_NODE_ELEM) && (xp_node->node->schema->nodetype & LYD_NODE_TERM)) {
1644 type = ((struct lyd_node_term *)xp_node->node)->value.realtype;
1645 } else if (xp_node->type == LYXP_NODE_META) {
1646 type = ((struct lyd_meta *)xp_node->node)->value.realtype;
1647 }
1648 }
1649 if (!type) {
1650 goto fill;
1651 }
1652
1653 if (src->type == LYXP_SET_NUMBER) {
1654 /* canonize number */
1655 if (asprintf(&str, "%Lf", src->val.num) == -1) {
1656 LOGMEM(src->ctx);
1657 return LY_EMEM;
1658 }
1659 } else {
1660 /* canonize string */
1661 str = strdup(src->val.str);
1662 }
1663
1664 /* ignore errors, the value may not satisfy schema constraints */
Radek Krejci0b013302021-03-29 15:22:32 +02001665 rc = type->plugin->store(src->ctx, type, str, strlen(str), LYPLG_TYPE_STORE_DYNAMIC, src->format, src->prefix_data,
Michal Vasko405cc9e2020-12-01 12:01:27 +01001666 LYD_HINT_DATA, xp_node->node->schema, &val, NULL, &err);
Michal Vasko4c7763f2020-07-27 17:40:37 +02001667 ly_err_free(err);
1668 if (rc) {
1669 /* invalid value */
Radek Iša48460222021-03-02 15:18:32 +01001670 /* function store automaticaly dealloc value when fail */
Michal Vasko4c7763f2020-07-27 17:40:37 +02001671 goto fill;
1672 }
1673
Michal Vasko4c7763f2020-07-27 17:40:37 +02001674 /* use the canonized value */
1675 set_init(trg, src);
1676 trg->type = src->type;
1677 if (src->type == LYXP_SET_NUMBER) {
Radek Krejci6d5ba0c2021-04-26 07:49:59 +02001678 trg->val.num = strtold(type->plugin->print(src->ctx, &val, LY_VALUE_CANON, NULL, NULL, NULL), &ptr);
Michal Vasko4c7763f2020-07-27 17:40:37 +02001679 LY_CHECK_ERR_RET(ptr[0], LOGINT(src->ctx), LY_EINT);
1680 } else {
Radek Krejci6d5ba0c2021-04-26 07:49:59 +02001681 trg->val.str = strdup(type->plugin->print(src->ctx, &val, LY_VALUE_CANON, NULL, NULL, NULL));
Michal Vasko4c7763f2020-07-27 17:40:37 +02001682 }
1683 type->plugin->free(src->ctx, &val);
1684 return LY_SUCCESS;
1685
1686fill:
1687 /* no canonization needed/possible */
1688 set_fill_set(trg, src);
1689 return LY_SUCCESS;
1690}
1691
Michal Vasko03ff5a72019-09-11 13:49:33 +02001692/**
1693 * @brief Bubble sort @p set into XPath document order.
Michal Vasko49fec8e2022-05-24 10:28:33 +02001694 * Context position aware.
Michal Vasko03ff5a72019-09-11 13:49:33 +02001695 *
1696 * @param[in] set Set to sort.
Michal Vasko03ff5a72019-09-11 13:49:33 +02001697 * @return How many times the whole set was traversed - 1 (if set was sorted, returns 0).
1698 */
1699static int
Michal Vasko5e0e6eb2019-11-06 15:47:50 +01001700set_sort(struct lyxp_set *set)
Michal Vasko03ff5a72019-09-11 13:49:33 +02001701{
1702 uint32_t i, j;
Radek Krejci1deb5be2020-08-26 16:43:36 +02001703 int ret = 0, cmp;
Radek Krejci857189e2020-09-01 13:26:36 +02001704 ly_bool inverted, change;
Michal Vasko03ff5a72019-09-11 13:49:33 +02001705 const struct lyd_node *root;
Michal Vasko03ff5a72019-09-11 13:49:33 +02001706 struct lyxp_set_node item;
1707 struct lyxp_set_hash_node hnode;
1708 uint64_t hash;
1709
Michal Vasko3cf8fbf2020-05-27 15:21:21 +02001710 if ((set->type != LYXP_SET_NODE_SET) || (set->used < 2)) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02001711 return 0;
1712 }
1713
Michal Vasko5e0e6eb2019-11-06 15:47:50 +01001714 /* find first top-level node to be used as anchor for positions */
Michal Vasko4a7d4d62021-12-13 17:05:06 +01001715 for (root = set->tree; root->parent; root = lyd_parent(root)) {}
Michal Vaskod989ba02020-08-24 10:59:24 +02001716 for ( ; root->prev->next; root = root->prev) {}
Michal Vasko03ff5a72019-09-11 13:49:33 +02001717
1718 /* fill positions */
Michal Vasko5e0e6eb2019-11-06 15:47:50 +01001719 if (set_assign_pos(set, root, set->root_type)) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02001720 return -1;
1721 }
1722
Michal Vasko88a9e802022-05-24 10:50:28 +02001723#ifndef NDEBUG
Michal Vasko03ff5a72019-09-11 13:49:33 +02001724 LOGDBG(LY_LDGXPATH, "SORT BEGIN");
1725 print_set_debug(set);
Michal Vasko88a9e802022-05-24 10:50:28 +02001726#endif
Michal Vasko03ff5a72019-09-11 13:49:33 +02001727
1728 for (i = 0; i < set->used; ++i) {
1729 inverted = 0;
1730 change = 0;
1731
1732 for (j = 1; j < set->used - i; ++j) {
1733 /* compare node positions */
1734 if (inverted) {
1735 cmp = set_sort_compare(&set->val.nodes[j], &set->val.nodes[j - 1]);
1736 } else {
1737 cmp = set_sort_compare(&set->val.nodes[j - 1], &set->val.nodes[j]);
1738 }
1739
1740 /* swap if needed */
1741 if ((inverted && (cmp < 0)) || (!inverted && (cmp > 0))) {
1742 change = 1;
1743
1744 item = set->val.nodes[j - 1];
1745 set->val.nodes[j - 1] = set->val.nodes[j];
1746 set->val.nodes[j] = item;
1747 } else {
1748 /* whether node_pos1 should be smaller than node_pos2 or the other way around */
1749 inverted = !inverted;
1750 }
1751 }
1752
1753 ++ret;
1754
1755 if (!change) {
1756 break;
1757 }
1758 }
1759
Michal Vasko88a9e802022-05-24 10:50:28 +02001760#ifndef NDEBUG
Michal Vasko03ff5a72019-09-11 13:49:33 +02001761 LOGDBG(LY_LDGXPATH, "SORT END %d", ret);
1762 print_set_debug(set);
Michal Vasko88a9e802022-05-24 10:50:28 +02001763#endif
Michal Vasko03ff5a72019-09-11 13:49:33 +02001764
1765 /* check node hashes */
1766 if (set->used >= LYD_HT_MIN_ITEMS) {
1767 assert(set->ht);
1768 for (i = 0; i < set->used; ++i) {
1769 hnode.node = set->val.nodes[i].node;
1770 hnode.type = set->val.nodes[i].type;
1771
1772 hash = dict_hash_multi(0, (const char *)&hnode.node, sizeof hnode.node);
1773 hash = dict_hash_multi(hash, (const char *)&hnode.type, sizeof hnode.type);
1774 hash = dict_hash_multi(hash, NULL, 0);
1775
1776 assert(!lyht_find(set->ht, &hnode, hash, NULL));
1777 }
1778 }
1779
1780 return ret - 1;
1781}
1782
Michal Vasko03ff5a72019-09-11 13:49:33 +02001783/**
1784 * @brief Merge 2 sorted sets into one.
1785 *
1786 * @param[in,out] trg Set to merge into. Duplicates are removed.
1787 * @param[in] src Set to be merged into @p trg. It is cast to #LYXP_SET_EMPTY on success.
Michal Vasko03ff5a72019-09-11 13:49:33 +02001788 * @return LY_ERR
1789 */
1790static LY_ERR
Michal Vasko5e0e6eb2019-11-06 15:47:50 +01001791set_sorted_merge(struct lyxp_set *trg, struct lyxp_set *src)
Michal Vasko03ff5a72019-09-11 13:49:33 +02001792{
1793 uint32_t i, j, k, count, dup_count;
1794 int cmp;
1795 const struct lyd_node *root;
Michal Vasko03ff5a72019-09-11 13:49:33 +02001796
Michal Vaskod3678892020-05-21 10:06:58 +02001797 if ((trg->type != LYXP_SET_NODE_SET) || (src->type != LYXP_SET_NODE_SET)) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02001798 return LY_EINVAL;
1799 }
1800
Michal Vaskod3678892020-05-21 10:06:58 +02001801 if (!src->used) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02001802 return LY_SUCCESS;
Michal Vaskod3678892020-05-21 10:06:58 +02001803 } else if (!trg->used) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02001804 set_fill_set(trg, src);
Michal Vaskod3678892020-05-21 10:06:58 +02001805 lyxp_set_free_content(src);
Michal Vasko03ff5a72019-09-11 13:49:33 +02001806 return LY_SUCCESS;
1807 }
1808
Michal Vasko5e0e6eb2019-11-06 15:47:50 +01001809 /* find first top-level node to be used as anchor for positions */
Michal Vasko0143b682021-12-13 17:13:09 +01001810 for (root = trg->tree; root->parent; root = lyd_parent(root)) {}
Michal Vaskod989ba02020-08-24 10:59:24 +02001811 for ( ; root->prev->next; root = root->prev) {}
Michal Vasko03ff5a72019-09-11 13:49:33 +02001812
1813 /* fill positions */
Michal Vasko5e0e6eb2019-11-06 15:47:50 +01001814 if (set_assign_pos(trg, root, trg->root_type) || set_assign_pos(src, root, src->root_type)) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02001815 return LY_EINT;
1816 }
1817
1818#ifndef NDEBUG
1819 LOGDBG(LY_LDGXPATH, "MERGE target");
1820 print_set_debug(trg);
1821 LOGDBG(LY_LDGXPATH, "MERGE source");
1822 print_set_debug(src);
1823#endif
1824
1825 /* make memory for the merge (duplicates are not detected yet, so space
1826 * will likely be wasted on them, too bad) */
1827 if (trg->size - trg->used < src->used) {
1828 trg->size = trg->used + src->used;
1829
1830 trg->val.nodes = ly_realloc(trg->val.nodes, trg->size * sizeof *trg->val.nodes);
1831 LY_CHECK_ERR_RET(!trg->val.nodes, LOGMEM(src->ctx), LY_EMEM);
1832 }
1833
1834 i = 0;
1835 j = 0;
1836 count = 0;
1837 dup_count = 0;
1838 do {
1839 cmp = set_sort_compare(&src->val.nodes[i], &trg->val.nodes[j]);
1840 if (!cmp) {
1841 if (!count) {
1842 /* duplicate, just skip it */
1843 ++i;
1844 ++j;
1845 } else {
1846 /* we are copying something already, so let's copy the duplicate too,
1847 * we are hoping that afterwards there are some more nodes to
1848 * copy and this way we can copy them all together */
1849 ++count;
1850 ++dup_count;
1851 ++i;
1852 ++j;
1853 }
1854 } else if (cmp < 0) {
1855 /* inserting src node into trg, just remember it for now */
1856 ++count;
1857 ++i;
1858
1859 /* insert the hash now */
1860 set_insert_node_hash(trg, src->val.nodes[i - 1].node, src->val.nodes[i - 1].type);
1861 } else if (count) {
1862copy_nodes:
1863 /* time to actually copy the nodes, we have found the largest block of nodes */
1864 memmove(&trg->val.nodes[j + (count - dup_count)],
1865 &trg->val.nodes[j],
1866 (trg->used - j) * sizeof *trg->val.nodes);
1867 memcpy(&trg->val.nodes[j - dup_count], &src->val.nodes[i - count], count * sizeof *src->val.nodes);
1868
1869 trg->used += count - dup_count;
1870 /* do not change i, except the copying above, we are basically doing exactly what is in the else branch below */
1871 j += count - dup_count;
1872
1873 count = 0;
1874 dup_count = 0;
1875 } else {
1876 ++j;
1877 }
1878 } while ((i < src->used) && (j < trg->used));
1879
1880 if ((i < src->used) || count) {
1881 /* insert all the hashes first */
1882 for (k = i; k < src->used; ++k) {
1883 set_insert_node_hash(trg, src->val.nodes[k].node, src->val.nodes[k].type);
1884 }
1885
1886 /* loop ended, but we need to copy something at trg end */
1887 count += src->used - i;
1888 i = src->used;
1889 goto copy_nodes;
1890 }
1891
1892 /* we are inserting hashes before the actual node insert, which causes
1893 * situations when there were initially not enough items for a hash table,
1894 * but even after some were inserted, hash table was not created (during
1895 * insertion the number of items is not updated yet) */
1896 if (!trg->ht && (trg->used >= LYD_HT_MIN_ITEMS)) {
1897 set_insert_node_hash(trg, NULL, 0);
1898 }
1899
1900#ifndef NDEBUG
1901 LOGDBG(LY_LDGXPATH, "MERGE result");
1902 print_set_debug(trg);
1903#endif
1904
Michal Vaskod3678892020-05-21 10:06:58 +02001905 lyxp_set_free_content(src);
Michal Vasko03ff5a72019-09-11 13:49:33 +02001906 return LY_SUCCESS;
1907}
1908
Michal Vasko14676352020-05-29 11:35:55 +02001909LY_ERR
Michal Vasko004d3152020-06-11 19:59:22 +02001910lyxp_check_token(const struct ly_ctx *ctx, const struct lyxp_expr *exp, uint16_t tok_idx, enum lyxp_token want_tok)
Michal Vasko03ff5a72019-09-11 13:49:33 +02001911{
Michal Vasko004d3152020-06-11 19:59:22 +02001912 if (exp->used == tok_idx) {
Michal Vasko14676352020-05-29 11:35:55 +02001913 if (ctx) {
Radek Krejci2efc45b2020-12-22 16:25:44 +01001914 LOGVAL(ctx, LY_VCODE_XP_EOF);
Michal Vasko03ff5a72019-09-11 13:49:33 +02001915 }
Michal Vasko14676352020-05-29 11:35:55 +02001916 return LY_EINCOMPLETE;
Michal Vasko03ff5a72019-09-11 13:49:33 +02001917 }
1918
Michal Vasko004d3152020-06-11 19:59:22 +02001919 if (want_tok && (exp->tokens[tok_idx] != want_tok)) {
Michal Vasko14676352020-05-29 11:35:55 +02001920 if (ctx) {
Michal Vasko49fec8e2022-05-24 10:28:33 +02001921 LOGVAL(ctx, LY_VCODE_XP_INTOK2, lyxp_token2str(exp->tokens[tok_idx]),
1922 &exp->expr[exp->tok_pos[tok_idx]], lyxp_token2str(want_tok));
Michal Vasko03ff5a72019-09-11 13:49:33 +02001923 }
Michal Vasko14676352020-05-29 11:35:55 +02001924 return LY_ENOT;
1925 }
1926
1927 return LY_SUCCESS;
1928}
1929
Michal Vasko004d3152020-06-11 19:59:22 +02001930LY_ERR
1931lyxp_next_token(const struct ly_ctx *ctx, const struct lyxp_expr *exp, uint16_t *tok_idx, enum lyxp_token want_tok)
1932{
1933 LY_CHECK_RET(lyxp_check_token(ctx, exp, *tok_idx, want_tok));
1934
1935 /* skip the token */
1936 ++(*tok_idx);
1937
1938 return LY_SUCCESS;
1939}
1940
Michal Vasko14676352020-05-29 11:35:55 +02001941/* just like lyxp_check_token() but tests for 2 tokens */
1942static LY_ERR
Michal Vasko40308e72020-10-20 16:38:40 +02001943exp_check_token2(const struct ly_ctx *ctx, const struct lyxp_expr *exp, uint16_t tok_idx, enum lyxp_token want_tok1,
Radek Krejci0f969882020-08-21 16:56:47 +02001944 enum lyxp_token want_tok2)
Michal Vasko14676352020-05-29 11:35:55 +02001945{
Michal Vasko004d3152020-06-11 19:59:22 +02001946 if (exp->used == tok_idx) {
Michal Vasko14676352020-05-29 11:35:55 +02001947 if (ctx) {
Radek Krejci2efc45b2020-12-22 16:25:44 +01001948 LOGVAL(ctx, LY_VCODE_XP_EOF);
Michal Vasko14676352020-05-29 11:35:55 +02001949 }
1950 return LY_EINCOMPLETE;
1951 }
1952
Michal Vasko004d3152020-06-11 19:59:22 +02001953 if ((exp->tokens[tok_idx] != want_tok1) && (exp->tokens[tok_idx] != want_tok2)) {
Michal Vasko14676352020-05-29 11:35:55 +02001954 if (ctx) {
Michal Vasko49fec8e2022-05-24 10:28:33 +02001955 LOGVAL(ctx, LY_VCODE_XP_INTOK, lyxp_token2str(exp->tokens[tok_idx]),
Michal Vasko0b468e62020-10-19 09:33:04 +02001956 &exp->expr[exp->tok_pos[tok_idx]]);
Michal Vasko14676352020-05-29 11:35:55 +02001957 }
1958 return LY_ENOT;
Michal Vasko03ff5a72019-09-11 13:49:33 +02001959 }
1960
1961 return LY_SUCCESS;
1962}
1963
Michal Vasko4911eeb2021-06-28 11:23:05 +02001964LY_ERR
1965lyxp_next_token2(const struct ly_ctx *ctx, const struct lyxp_expr *exp, uint16_t *tok_idx, enum lyxp_token want_tok1,
1966 enum lyxp_token want_tok2)
1967{
1968 LY_CHECK_RET(exp_check_token2(ctx, exp, *tok_idx, want_tok1, want_tok2));
1969
1970 /* skip the token */
1971 ++(*tok_idx);
1972
1973 return LY_SUCCESS;
1974}
1975
Michal Vasko03ff5a72019-09-11 13:49:33 +02001976/**
1977 * @brief Stack operation push on the repeat array.
1978 *
1979 * @param[in] exp Expression to use.
Michal Vasko004d3152020-06-11 19:59:22 +02001980 * @param[in] tok_idx Position in the expresion \p exp.
Michal Vasko03ff5a72019-09-11 13:49:33 +02001981 * @param[in] repeat_op_idx Index from \p exp of the operator token. This value is pushed.
1982 */
1983static void
Michal Vasko004d3152020-06-11 19:59:22 +02001984exp_repeat_push(struct lyxp_expr *exp, uint16_t tok_idx, uint16_t repeat_op_idx)
Michal Vasko03ff5a72019-09-11 13:49:33 +02001985{
1986 uint16_t i;
1987
Michal Vasko004d3152020-06-11 19:59:22 +02001988 if (exp->repeat[tok_idx]) {
Radek Krejci1e008d22020-08-17 11:37:37 +02001989 for (i = 0; exp->repeat[tok_idx][i]; ++i) {}
Michal Vasko004d3152020-06-11 19:59:22 +02001990 exp->repeat[tok_idx] = realloc(exp->repeat[tok_idx], (i + 2) * sizeof *exp->repeat[tok_idx]);
1991 LY_CHECK_ERR_RET(!exp->repeat[tok_idx], LOGMEM(NULL), );
1992 exp->repeat[tok_idx][i] = repeat_op_idx;
1993 exp->repeat[tok_idx][i + 1] = 0;
Michal Vasko03ff5a72019-09-11 13:49:33 +02001994 } else {
Michal Vasko004d3152020-06-11 19:59:22 +02001995 exp->repeat[tok_idx] = calloc(2, sizeof *exp->repeat[tok_idx]);
1996 LY_CHECK_ERR_RET(!exp->repeat[tok_idx], LOGMEM(NULL), );
1997 exp->repeat[tok_idx][0] = repeat_op_idx;
Michal Vasko03ff5a72019-09-11 13:49:33 +02001998 }
1999}
2000
2001/**
2002 * @brief Reparse Predicate. Logs directly on error.
2003 *
2004 * [7] Predicate ::= '[' Expr ']'
2005 *
2006 * @param[in] ctx Context for logging.
2007 * @param[in] exp Parsed XPath expression.
Michal Vasko004d3152020-06-11 19:59:22 +02002008 * @param[in] tok_idx Position in the expression @p exp.
aPiecekbf968d92021-05-27 14:35:05 +02002009 * @param[in] depth Current number of nested expressions.
Michal Vasko03ff5a72019-09-11 13:49:33 +02002010 * @return LY_ERR
2011 */
2012static LY_ERR
aPiecekbf968d92021-05-27 14:35:05 +02002013reparse_predicate(const struct ly_ctx *ctx, struct lyxp_expr *exp, uint16_t *tok_idx, uint16_t depth)
Michal Vasko03ff5a72019-09-11 13:49:33 +02002014{
2015 LY_ERR rc;
2016
Michal Vasko004d3152020-06-11 19:59:22 +02002017 rc = lyxp_check_token(ctx, exp, *tok_idx, LYXP_TOKEN_BRACK1);
Michal Vasko03ff5a72019-09-11 13:49:33 +02002018 LY_CHECK_RET(rc);
Michal Vasko004d3152020-06-11 19:59:22 +02002019 ++(*tok_idx);
Michal Vasko03ff5a72019-09-11 13:49:33 +02002020
aPiecekbf968d92021-05-27 14:35:05 +02002021 rc = reparse_or_expr(ctx, exp, tok_idx, depth);
Michal Vasko03ff5a72019-09-11 13:49:33 +02002022 LY_CHECK_RET(rc);
2023
Michal Vasko004d3152020-06-11 19:59:22 +02002024 rc = lyxp_check_token(ctx, exp, *tok_idx, LYXP_TOKEN_BRACK2);
Michal Vasko03ff5a72019-09-11 13:49:33 +02002025 LY_CHECK_RET(rc);
Michal Vasko004d3152020-06-11 19:59:22 +02002026 ++(*tok_idx);
Michal Vasko03ff5a72019-09-11 13:49:33 +02002027
2028 return LY_SUCCESS;
2029}
2030
2031/**
2032 * @brief Reparse RelativeLocationPath. Logs directly on error.
2033 *
2034 * [4] RelativeLocationPath ::= Step | RelativeLocationPath '/' Step | RelativeLocationPath '//' Step
2035 * [5] Step ::= '@'? NodeTest Predicate* | '.' | '..'
2036 * [6] NodeTest ::= NameTest | NodeType '(' ')'
2037 *
2038 * @param[in] ctx Context for logging.
2039 * @param[in] exp Parsed XPath expression.
Michal Vasko004d3152020-06-11 19:59:22 +02002040 * @param[in] tok_idx Position in the expression \p exp.
aPiecekbf968d92021-05-27 14:35:05 +02002041 * @param[in] depth Current number of nested expressions.
Michal Vasko03ff5a72019-09-11 13:49:33 +02002042 * @return LY_ERR (LY_EINCOMPLETE on forward reference)
2043 */
2044static LY_ERR
aPiecekbf968d92021-05-27 14:35:05 +02002045reparse_relative_location_path(const struct ly_ctx *ctx, struct lyxp_expr *exp, uint16_t *tok_idx, uint16_t depth)
Michal Vasko03ff5a72019-09-11 13:49:33 +02002046{
2047 LY_ERR rc;
2048
Michal Vasko004d3152020-06-11 19:59:22 +02002049 rc = lyxp_check_token(ctx, exp, *tok_idx, LYXP_TOKEN_NONE);
Michal Vasko03ff5a72019-09-11 13:49:33 +02002050 LY_CHECK_RET(rc);
2051
2052 goto step;
2053 do {
2054 /* '/' or '//' */
Michal Vasko004d3152020-06-11 19:59:22 +02002055 ++(*tok_idx);
Michal Vasko03ff5a72019-09-11 13:49:33 +02002056
Michal Vasko004d3152020-06-11 19:59:22 +02002057 rc = lyxp_check_token(ctx, exp, *tok_idx, LYXP_TOKEN_NONE);
Michal Vasko03ff5a72019-09-11 13:49:33 +02002058 LY_CHECK_RET(rc);
2059step:
2060 /* Step */
Michal Vasko004d3152020-06-11 19:59:22 +02002061 switch (exp->tokens[*tok_idx]) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02002062 case LYXP_TOKEN_DOT:
Michal Vasko004d3152020-06-11 19:59:22 +02002063 ++(*tok_idx);
Michal Vasko03ff5a72019-09-11 13:49:33 +02002064 break;
2065
2066 case LYXP_TOKEN_DDOT:
Michal Vasko004d3152020-06-11 19:59:22 +02002067 ++(*tok_idx);
Michal Vasko03ff5a72019-09-11 13:49:33 +02002068 break;
2069
Michal Vasko49fec8e2022-05-24 10:28:33 +02002070 case LYXP_TOKEN_AXISNAME:
2071 ++(*tok_idx);
2072
2073 rc = lyxp_check_token(ctx, exp, *tok_idx, LYXP_TOKEN_DCOLON);
2074 LY_CHECK_RET(rc);
2075
2076 /* fall through */
Michal Vasko03ff5a72019-09-11 13:49:33 +02002077 case LYXP_TOKEN_AT:
Michal Vasko004d3152020-06-11 19:59:22 +02002078 ++(*tok_idx);
Michal Vasko03ff5a72019-09-11 13:49:33 +02002079
Michal Vasko004d3152020-06-11 19:59:22 +02002080 rc = lyxp_check_token(ctx, exp, *tok_idx, LYXP_TOKEN_NONE);
Michal Vasko03ff5a72019-09-11 13:49:33 +02002081 LY_CHECK_RET(rc);
Michal Vasko004d3152020-06-11 19:59:22 +02002082 if ((exp->tokens[*tok_idx] != LYXP_TOKEN_NAMETEST) && (exp->tokens[*tok_idx] != LYXP_TOKEN_NODETYPE)) {
Michal Vasko49fec8e2022-05-24 10:28:33 +02002083 LOGVAL(ctx, LY_VCODE_XP_INTOK, lyxp_token2str(exp->tokens[*tok_idx]), &exp->expr[exp->tok_pos[*tok_idx]]);
Michal Vasko03ff5a72019-09-11 13:49:33 +02002084 return LY_EVALID;
2085 }
Michal Vasko49fec8e2022-05-24 10:28:33 +02002086 if (exp->tokens[*tok_idx] == LYXP_TOKEN_NODETYPE) {
2087 goto reparse_nodetype;
2088 }
Radek Krejci0f969882020-08-21 16:56:47 +02002089 /* fall through */
Michal Vasko03ff5a72019-09-11 13:49:33 +02002090 case LYXP_TOKEN_NAMETEST:
Michal Vasko004d3152020-06-11 19:59:22 +02002091 ++(*tok_idx);
Michal Vasko03ff5a72019-09-11 13:49:33 +02002092 goto reparse_predicate;
Michal Vasko03ff5a72019-09-11 13:49:33 +02002093
2094 case LYXP_TOKEN_NODETYPE:
Michal Vasko49fec8e2022-05-24 10:28:33 +02002095reparse_nodetype:
Michal Vasko004d3152020-06-11 19:59:22 +02002096 ++(*tok_idx);
Michal Vasko03ff5a72019-09-11 13:49:33 +02002097
2098 /* '(' */
Michal Vasko004d3152020-06-11 19:59:22 +02002099 rc = lyxp_check_token(ctx, exp, *tok_idx, LYXP_TOKEN_PAR1);
Michal Vasko03ff5a72019-09-11 13:49:33 +02002100 LY_CHECK_RET(rc);
Michal Vasko004d3152020-06-11 19:59:22 +02002101 ++(*tok_idx);
Michal Vasko03ff5a72019-09-11 13:49:33 +02002102
2103 /* ')' */
Michal Vasko004d3152020-06-11 19:59:22 +02002104 rc = lyxp_check_token(ctx, exp, *tok_idx, LYXP_TOKEN_PAR2);
Michal Vasko03ff5a72019-09-11 13:49:33 +02002105 LY_CHECK_RET(rc);
Michal Vasko004d3152020-06-11 19:59:22 +02002106 ++(*tok_idx);
Michal Vasko03ff5a72019-09-11 13:49:33 +02002107
2108reparse_predicate:
2109 /* Predicate* */
Michal Vasko004d3152020-06-11 19:59:22 +02002110 while (!lyxp_check_token(NULL, exp, *tok_idx, LYXP_TOKEN_BRACK1)) {
aPiecekbf968d92021-05-27 14:35:05 +02002111 rc = reparse_predicate(ctx, exp, tok_idx, depth);
Michal Vasko03ff5a72019-09-11 13:49:33 +02002112 LY_CHECK_RET(rc);
2113 }
2114 break;
2115 default:
Michal Vasko49fec8e2022-05-24 10:28:33 +02002116 LOGVAL(ctx, LY_VCODE_XP_INTOK, lyxp_token2str(exp->tokens[*tok_idx]), &exp->expr[exp->tok_pos[*tok_idx]]);
Michal Vasko03ff5a72019-09-11 13:49:33 +02002117 return LY_EVALID;
2118 }
Michal Vasko004d3152020-06-11 19:59:22 +02002119 } while (!exp_check_token2(NULL, exp, *tok_idx, LYXP_TOKEN_OPER_PATH, LYXP_TOKEN_OPER_RPATH));
Michal Vasko03ff5a72019-09-11 13:49:33 +02002120
2121 return LY_SUCCESS;
2122}
2123
2124/**
2125 * @brief Reparse AbsoluteLocationPath. Logs directly on error.
2126 *
2127 * [3] AbsoluteLocationPath ::= '/' RelativeLocationPath? | '//' RelativeLocationPath
2128 *
2129 * @param[in] ctx Context for logging.
2130 * @param[in] exp Parsed XPath expression.
Michal Vasko004d3152020-06-11 19:59:22 +02002131 * @param[in] tok_idx Position in the expression \p exp.
aPiecekbf968d92021-05-27 14:35:05 +02002132 * @param[in] depth Current number of nested expressions.
Michal Vasko03ff5a72019-09-11 13:49:33 +02002133 * @return LY_ERR
2134 */
2135static LY_ERR
aPiecekbf968d92021-05-27 14:35:05 +02002136reparse_absolute_location_path(const struct ly_ctx *ctx, struct lyxp_expr *exp, uint16_t *tok_idx, uint16_t depth)
Michal Vasko03ff5a72019-09-11 13:49:33 +02002137{
2138 LY_ERR rc;
2139
Michal Vasko004d3152020-06-11 19:59:22 +02002140 LY_CHECK_RET(exp_check_token2(ctx, exp, *tok_idx, LYXP_TOKEN_OPER_PATH, LYXP_TOKEN_OPER_RPATH));
Michal Vasko03ff5a72019-09-11 13:49:33 +02002141
2142 /* '/' RelativeLocationPath? */
Michal Vasko004d3152020-06-11 19:59:22 +02002143 if (exp->tokens[*tok_idx] == LYXP_TOKEN_OPER_PATH) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02002144 /* '/' */
Michal Vasko004d3152020-06-11 19:59:22 +02002145 ++(*tok_idx);
Michal Vasko03ff5a72019-09-11 13:49:33 +02002146
Michal Vasko004d3152020-06-11 19:59:22 +02002147 if (lyxp_check_token(NULL, exp, *tok_idx, LYXP_TOKEN_NONE)) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02002148 return LY_SUCCESS;
2149 }
Michal Vasko004d3152020-06-11 19:59:22 +02002150 switch (exp->tokens[*tok_idx]) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02002151 case LYXP_TOKEN_DOT:
2152 case LYXP_TOKEN_DDOT:
Michal Vasko49fec8e2022-05-24 10:28:33 +02002153 case LYXP_TOKEN_AXISNAME:
Michal Vasko03ff5a72019-09-11 13:49:33 +02002154 case LYXP_TOKEN_AT:
2155 case LYXP_TOKEN_NAMETEST:
2156 case LYXP_TOKEN_NODETYPE:
aPiecekbf968d92021-05-27 14:35:05 +02002157 rc = reparse_relative_location_path(ctx, exp, tok_idx, depth);
Michal Vasko03ff5a72019-09-11 13:49:33 +02002158 LY_CHECK_RET(rc);
Radek Krejci0f969882020-08-21 16:56:47 +02002159 /* fall through */
Michal Vasko03ff5a72019-09-11 13:49:33 +02002160 default:
2161 break;
2162 }
2163
Michal Vasko03ff5a72019-09-11 13:49:33 +02002164 } else {
Radek Krejcif6a11002020-08-21 13:29:07 +02002165 /* '//' RelativeLocationPath */
Michal Vasko004d3152020-06-11 19:59:22 +02002166 ++(*tok_idx);
Michal Vasko03ff5a72019-09-11 13:49:33 +02002167
aPiecekbf968d92021-05-27 14:35:05 +02002168 rc = reparse_relative_location_path(ctx, exp, tok_idx, depth);
Michal Vasko03ff5a72019-09-11 13:49:33 +02002169 LY_CHECK_RET(rc);
2170 }
2171
2172 return LY_SUCCESS;
2173}
2174
2175/**
2176 * @brief Reparse FunctionCall. Logs directly on error.
2177 *
2178 * [9] FunctionCall ::= FunctionName '(' ( Expr ( ',' Expr )* )? ')'
2179 *
2180 * @param[in] ctx Context for logging.
2181 * @param[in] exp Parsed XPath expression.
Michal Vasko004d3152020-06-11 19:59:22 +02002182 * @param[in] tok_idx Position in the expression @p exp.
aPiecekbf968d92021-05-27 14:35:05 +02002183 * @param[in] depth Current number of nested expressions.
Michal Vasko03ff5a72019-09-11 13:49:33 +02002184 * @return LY_ERR
2185 */
2186static LY_ERR
aPiecekbf968d92021-05-27 14:35:05 +02002187reparse_function_call(const struct ly_ctx *ctx, struct lyxp_expr *exp, uint16_t *tok_idx, uint16_t depth)
Michal Vasko03ff5a72019-09-11 13:49:33 +02002188{
Radek Krejci1deb5be2020-08-26 16:43:36 +02002189 int8_t min_arg_count = -1;
2190 uint32_t arg_count, max_arg_count = 0;
Michal Vasko004d3152020-06-11 19:59:22 +02002191 uint16_t func_tok_idx;
Michal Vasko03ff5a72019-09-11 13:49:33 +02002192 LY_ERR rc;
2193
Michal Vasko004d3152020-06-11 19:59:22 +02002194 rc = lyxp_check_token(ctx, exp, *tok_idx, LYXP_TOKEN_FUNCNAME);
Michal Vasko03ff5a72019-09-11 13:49:33 +02002195 LY_CHECK_RET(rc);
Michal Vasko004d3152020-06-11 19:59:22 +02002196 func_tok_idx = *tok_idx;
2197 switch (exp->tok_len[*tok_idx]) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02002198 case 3:
Michal Vasko004d3152020-06-11 19:59:22 +02002199 if (!strncmp(&exp->expr[exp->tok_pos[*tok_idx]], "not", 3)) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02002200 min_arg_count = 1;
2201 max_arg_count = 1;
Michal Vasko004d3152020-06-11 19:59:22 +02002202 } else if (!strncmp(&exp->expr[exp->tok_pos[*tok_idx]], "sum", 3)) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02002203 min_arg_count = 1;
2204 max_arg_count = 1;
2205 }
2206 break;
2207 case 4:
Michal Vasko004d3152020-06-11 19:59:22 +02002208 if (!strncmp(&exp->expr[exp->tok_pos[*tok_idx]], "lang", 4)) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02002209 min_arg_count = 1;
2210 max_arg_count = 1;
Michal Vasko004d3152020-06-11 19:59:22 +02002211 } else if (!strncmp(&exp->expr[exp->tok_pos[*tok_idx]], "last", 4)) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02002212 min_arg_count = 0;
2213 max_arg_count = 0;
Michal Vasko004d3152020-06-11 19:59:22 +02002214 } else if (!strncmp(&exp->expr[exp->tok_pos[*tok_idx]], "name", 4)) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02002215 min_arg_count = 0;
2216 max_arg_count = 1;
Michal Vasko004d3152020-06-11 19:59:22 +02002217 } else if (!strncmp(&exp->expr[exp->tok_pos[*tok_idx]], "true", 4)) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02002218 min_arg_count = 0;
2219 max_arg_count = 0;
2220 }
2221 break;
2222 case 5:
Michal Vasko004d3152020-06-11 19:59:22 +02002223 if (!strncmp(&exp->expr[exp->tok_pos[*tok_idx]], "count", 5)) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02002224 min_arg_count = 1;
2225 max_arg_count = 1;
Michal Vasko004d3152020-06-11 19:59:22 +02002226 } else if (!strncmp(&exp->expr[exp->tok_pos[*tok_idx]], "false", 5)) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02002227 min_arg_count = 0;
2228 max_arg_count = 0;
Michal Vasko004d3152020-06-11 19:59:22 +02002229 } else if (!strncmp(&exp->expr[exp->tok_pos[*tok_idx]], "floor", 5)) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02002230 min_arg_count = 1;
2231 max_arg_count = 1;
Michal Vasko004d3152020-06-11 19:59:22 +02002232 } else if (!strncmp(&exp->expr[exp->tok_pos[*tok_idx]], "round", 5)) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02002233 min_arg_count = 1;
2234 max_arg_count = 1;
Michal Vasko004d3152020-06-11 19:59:22 +02002235 } else if (!strncmp(&exp->expr[exp->tok_pos[*tok_idx]], "deref", 5)) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02002236 min_arg_count = 1;
2237 max_arg_count = 1;
2238 }
2239 break;
2240 case 6:
Michal Vasko004d3152020-06-11 19:59:22 +02002241 if (!strncmp(&exp->expr[exp->tok_pos[*tok_idx]], "concat", 6)) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02002242 min_arg_count = 2;
Radek Krejci1deb5be2020-08-26 16:43:36 +02002243 max_arg_count = UINT32_MAX;
Michal Vasko004d3152020-06-11 19:59:22 +02002244 } else if (!strncmp(&exp->expr[exp->tok_pos[*tok_idx]], "number", 6)) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02002245 min_arg_count = 0;
2246 max_arg_count = 1;
Michal Vasko004d3152020-06-11 19:59:22 +02002247 } else if (!strncmp(&exp->expr[exp->tok_pos[*tok_idx]], "string", 6)) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02002248 min_arg_count = 0;
2249 max_arg_count = 1;
2250 }
2251 break;
2252 case 7:
Michal Vasko004d3152020-06-11 19:59:22 +02002253 if (!strncmp(&exp->expr[exp->tok_pos[*tok_idx]], "boolean", 7)) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02002254 min_arg_count = 1;
2255 max_arg_count = 1;
Michal Vasko004d3152020-06-11 19:59:22 +02002256 } else if (!strncmp(&exp->expr[exp->tok_pos[*tok_idx]], "ceiling", 7)) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02002257 min_arg_count = 1;
2258 max_arg_count = 1;
Michal Vasko004d3152020-06-11 19:59:22 +02002259 } else if (!strncmp(&exp->expr[exp->tok_pos[*tok_idx]], "current", 7)) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02002260 min_arg_count = 0;
2261 max_arg_count = 0;
2262 }
2263 break;
2264 case 8:
Michal Vasko004d3152020-06-11 19:59:22 +02002265 if (!strncmp(&exp->expr[exp->tok_pos[*tok_idx]], "contains", 8)) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02002266 min_arg_count = 2;
2267 max_arg_count = 2;
Michal Vasko004d3152020-06-11 19:59:22 +02002268 } else if (!strncmp(&exp->expr[exp->tok_pos[*tok_idx]], "position", 8)) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02002269 min_arg_count = 0;
2270 max_arg_count = 0;
Michal Vasko004d3152020-06-11 19:59:22 +02002271 } else if (!strncmp(&exp->expr[exp->tok_pos[*tok_idx]], "re-match", 8)) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02002272 min_arg_count = 2;
2273 max_arg_count = 2;
2274 }
2275 break;
2276 case 9:
Michal Vasko004d3152020-06-11 19:59:22 +02002277 if (!strncmp(&exp->expr[exp->tok_pos[*tok_idx]], "substring", 9)) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02002278 min_arg_count = 2;
2279 max_arg_count = 3;
Michal Vasko004d3152020-06-11 19:59:22 +02002280 } else if (!strncmp(&exp->expr[exp->tok_pos[*tok_idx]], "translate", 9)) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02002281 min_arg_count = 3;
2282 max_arg_count = 3;
2283 }
2284 break;
2285 case 10:
Michal Vasko004d3152020-06-11 19:59:22 +02002286 if (!strncmp(&exp->expr[exp->tok_pos[*tok_idx]], "local-name", 10)) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02002287 min_arg_count = 0;
2288 max_arg_count = 1;
Michal Vasko004d3152020-06-11 19:59:22 +02002289 } else if (!strncmp(&exp->expr[exp->tok_pos[*tok_idx]], "enum-value", 10)) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02002290 min_arg_count = 1;
2291 max_arg_count = 1;
Michal Vasko004d3152020-06-11 19:59:22 +02002292 } else if (!strncmp(&exp->expr[exp->tok_pos[*tok_idx]], "bit-is-set", 10)) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02002293 min_arg_count = 2;
2294 max_arg_count = 2;
2295 }
2296 break;
2297 case 11:
Michal Vasko004d3152020-06-11 19:59:22 +02002298 if (!strncmp(&exp->expr[exp->tok_pos[*tok_idx]], "starts-with", 11)) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02002299 min_arg_count = 2;
2300 max_arg_count = 2;
2301 }
2302 break;
2303 case 12:
Michal Vasko004d3152020-06-11 19:59:22 +02002304 if (!strncmp(&exp->expr[exp->tok_pos[*tok_idx]], "derived-from", 12)) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02002305 min_arg_count = 2;
2306 max_arg_count = 2;
2307 }
2308 break;
2309 case 13:
Michal Vasko004d3152020-06-11 19:59:22 +02002310 if (!strncmp(&exp->expr[exp->tok_pos[*tok_idx]], "namespace-uri", 13)) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02002311 min_arg_count = 0;
2312 max_arg_count = 1;
Michal Vasko004d3152020-06-11 19:59:22 +02002313 } else if (!strncmp(&exp->expr[exp->tok_pos[*tok_idx]], "string-length", 13)) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02002314 min_arg_count = 0;
2315 max_arg_count = 1;
2316 }
2317 break;
2318 case 15:
Michal Vasko004d3152020-06-11 19:59:22 +02002319 if (!strncmp(&exp->expr[exp->tok_pos[*tok_idx]], "normalize-space", 15)) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02002320 min_arg_count = 0;
2321 max_arg_count = 1;
Michal Vasko004d3152020-06-11 19:59:22 +02002322 } else if (!strncmp(&exp->expr[exp->tok_pos[*tok_idx]], "substring-after", 15)) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02002323 min_arg_count = 2;
2324 max_arg_count = 2;
2325 }
2326 break;
2327 case 16:
Michal Vasko004d3152020-06-11 19:59:22 +02002328 if (!strncmp(&exp->expr[exp->tok_pos[*tok_idx]], "substring-before", 16)) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02002329 min_arg_count = 2;
2330 max_arg_count = 2;
2331 }
2332 break;
2333 case 20:
Michal Vasko004d3152020-06-11 19:59:22 +02002334 if (!strncmp(&exp->expr[exp->tok_pos[*tok_idx]], "derived-from-or-self", 20)) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02002335 min_arg_count = 2;
2336 max_arg_count = 2;
2337 }
2338 break;
2339 }
2340 if (min_arg_count == -1) {
Radek Krejci2efc45b2020-12-22 16:25:44 +01002341 LOGVAL(ctx, LY_VCODE_XP_INFUNC, exp->tok_len[*tok_idx], &exp->expr[exp->tok_pos[*tok_idx]]);
Michal Vasko03ff5a72019-09-11 13:49:33 +02002342 return LY_EINVAL;
2343 }
Michal Vasko004d3152020-06-11 19:59:22 +02002344 ++(*tok_idx);
Michal Vasko03ff5a72019-09-11 13:49:33 +02002345
2346 /* '(' */
Michal Vasko004d3152020-06-11 19:59:22 +02002347 rc = lyxp_check_token(ctx, exp, *tok_idx, LYXP_TOKEN_PAR1);
Michal Vasko03ff5a72019-09-11 13:49:33 +02002348 LY_CHECK_RET(rc);
Michal Vasko004d3152020-06-11 19:59:22 +02002349 ++(*tok_idx);
Michal Vasko03ff5a72019-09-11 13:49:33 +02002350
2351 /* ( Expr ( ',' Expr )* )? */
2352 arg_count = 0;
Michal Vasko004d3152020-06-11 19:59:22 +02002353 rc = lyxp_check_token(ctx, exp, *tok_idx, LYXP_TOKEN_NONE);
Michal Vasko03ff5a72019-09-11 13:49:33 +02002354 LY_CHECK_RET(rc);
Michal Vasko004d3152020-06-11 19:59:22 +02002355 if (exp->tokens[*tok_idx] != LYXP_TOKEN_PAR2) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02002356 ++arg_count;
aPiecekbf968d92021-05-27 14:35:05 +02002357 rc = reparse_or_expr(ctx, exp, tok_idx, depth);
Michal Vasko03ff5a72019-09-11 13:49:33 +02002358 LY_CHECK_RET(rc);
2359 }
Michal Vasko004d3152020-06-11 19:59:22 +02002360 while (!lyxp_check_token(NULL, exp, *tok_idx, LYXP_TOKEN_COMMA)) {
2361 ++(*tok_idx);
Michal Vasko03ff5a72019-09-11 13:49:33 +02002362
2363 ++arg_count;
aPiecekbf968d92021-05-27 14:35:05 +02002364 rc = reparse_or_expr(ctx, exp, tok_idx, depth);
Michal Vasko03ff5a72019-09-11 13:49:33 +02002365 LY_CHECK_RET(rc);
2366 }
2367
2368 /* ')' */
Michal Vasko004d3152020-06-11 19:59:22 +02002369 rc = lyxp_check_token(ctx, exp, *tok_idx, LYXP_TOKEN_PAR2);
Michal Vasko03ff5a72019-09-11 13:49:33 +02002370 LY_CHECK_RET(rc);
Michal Vasko004d3152020-06-11 19:59:22 +02002371 ++(*tok_idx);
Michal Vasko03ff5a72019-09-11 13:49:33 +02002372
Radek Krejci857189e2020-09-01 13:26:36 +02002373 if ((arg_count < (uint32_t)min_arg_count) || (arg_count > max_arg_count)) {
Radek Krejci2efc45b2020-12-22 16:25:44 +01002374 LOGVAL(ctx, LY_VCODE_XP_INARGCOUNT, arg_count, exp->tok_len[func_tok_idx], &exp->expr[exp->tok_pos[func_tok_idx]]);
Michal Vasko03ff5a72019-09-11 13:49:33 +02002375 return LY_EVALID;
2376 }
2377
2378 return LY_SUCCESS;
2379}
2380
2381/**
2382 * @brief Reparse PathExpr. Logs directly on error.
2383 *
2384 * [10] PathExpr ::= LocationPath | PrimaryExpr Predicate*
2385 * | PrimaryExpr Predicate* '/' RelativeLocationPath
2386 * | PrimaryExpr Predicate* '//' RelativeLocationPath
2387 * [2] LocationPath ::= RelativeLocationPath | AbsoluteLocationPath
aPiecekfba75362021-10-07 12:39:48 +02002388 * [8] PrimaryExpr ::= VariableReference | '(' Expr ')' | Literal | Number | FunctionCall
Michal Vasko03ff5a72019-09-11 13:49:33 +02002389 *
2390 * @param[in] ctx Context for logging.
2391 * @param[in] exp Parsed XPath expression.
Michal Vasko004d3152020-06-11 19:59:22 +02002392 * @param[in] tok_idx Position in the expression @p exp.
aPiecekbf968d92021-05-27 14:35:05 +02002393 * @param[in] depth Current number of nested expressions.
Michal Vasko03ff5a72019-09-11 13:49:33 +02002394 * @return LY_ERR
2395 */
2396static LY_ERR
aPiecekbf968d92021-05-27 14:35:05 +02002397reparse_path_expr(const struct ly_ctx *ctx, struct lyxp_expr *exp, uint16_t *tok_idx, uint16_t depth)
Michal Vasko03ff5a72019-09-11 13:49:33 +02002398{
2399 LY_ERR rc;
2400
Michal Vasko004d3152020-06-11 19:59:22 +02002401 if (lyxp_check_token(ctx, exp, *tok_idx, LYXP_TOKEN_NONE)) {
Michal Vasko14676352020-05-29 11:35:55 +02002402 return LY_EVALID;
Michal Vasko03ff5a72019-09-11 13:49:33 +02002403 }
2404
Michal Vasko004d3152020-06-11 19:59:22 +02002405 switch (exp->tokens[*tok_idx]) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02002406 case LYXP_TOKEN_PAR1:
2407 /* '(' Expr ')' Predicate* */
Michal Vasko004d3152020-06-11 19:59:22 +02002408 ++(*tok_idx);
Michal Vasko03ff5a72019-09-11 13:49:33 +02002409
aPiecekbf968d92021-05-27 14:35:05 +02002410 rc = reparse_or_expr(ctx, exp, tok_idx, depth);
Michal Vasko03ff5a72019-09-11 13:49:33 +02002411 LY_CHECK_RET(rc);
2412
Michal Vasko004d3152020-06-11 19:59:22 +02002413 rc = lyxp_check_token(ctx, exp, *tok_idx, LYXP_TOKEN_PAR2);
Michal Vasko03ff5a72019-09-11 13:49:33 +02002414 LY_CHECK_RET(rc);
Michal Vasko004d3152020-06-11 19:59:22 +02002415 ++(*tok_idx);
Michal Vasko03ff5a72019-09-11 13:49:33 +02002416 goto predicate;
Michal Vasko03ff5a72019-09-11 13:49:33 +02002417 case LYXP_TOKEN_DOT:
2418 case LYXP_TOKEN_DDOT:
Michal Vasko49fec8e2022-05-24 10:28:33 +02002419 case LYXP_TOKEN_AXISNAME:
Michal Vasko03ff5a72019-09-11 13:49:33 +02002420 case LYXP_TOKEN_AT:
2421 case LYXP_TOKEN_NAMETEST:
2422 case LYXP_TOKEN_NODETYPE:
2423 /* RelativeLocationPath */
aPiecekbf968d92021-05-27 14:35:05 +02002424 rc = reparse_relative_location_path(ctx, exp, tok_idx, depth);
Michal Vasko03ff5a72019-09-11 13:49:33 +02002425 LY_CHECK_RET(rc);
2426 break;
aPiecekfba75362021-10-07 12:39:48 +02002427 case LYXP_TOKEN_VARREF:
2428 /* VariableReference */
2429 ++(*tok_idx);
2430 goto predicate;
Michal Vasko03ff5a72019-09-11 13:49:33 +02002431 case LYXP_TOKEN_FUNCNAME:
2432 /* FunctionCall */
aPiecekbf968d92021-05-27 14:35:05 +02002433 rc = reparse_function_call(ctx, exp, tok_idx, depth);
Michal Vasko03ff5a72019-09-11 13:49:33 +02002434 LY_CHECK_RET(rc);
2435 goto predicate;
Michal Vasko3e48bf32020-06-01 08:39:07 +02002436 case LYXP_TOKEN_OPER_PATH:
2437 case LYXP_TOKEN_OPER_RPATH:
Michal Vasko03ff5a72019-09-11 13:49:33 +02002438 /* AbsoluteLocationPath */
aPiecekbf968d92021-05-27 14:35:05 +02002439 rc = reparse_absolute_location_path(ctx, exp, tok_idx, depth);
Michal Vasko03ff5a72019-09-11 13:49:33 +02002440 LY_CHECK_RET(rc);
2441 break;
2442 case LYXP_TOKEN_LITERAL:
2443 /* Literal */
Michal Vasko004d3152020-06-11 19:59:22 +02002444 ++(*tok_idx);
Michal Vasko03ff5a72019-09-11 13:49:33 +02002445 goto predicate;
Michal Vasko03ff5a72019-09-11 13:49:33 +02002446 case LYXP_TOKEN_NUMBER:
2447 /* Number */
Michal Vasko004d3152020-06-11 19:59:22 +02002448 ++(*tok_idx);
Michal Vasko03ff5a72019-09-11 13:49:33 +02002449 goto predicate;
Michal Vasko03ff5a72019-09-11 13:49:33 +02002450 default:
Michal Vasko49fec8e2022-05-24 10:28:33 +02002451 LOGVAL(ctx, LY_VCODE_XP_INTOK, lyxp_token2str(exp->tokens[*tok_idx]), &exp->expr[exp->tok_pos[*tok_idx]]);
Michal Vasko03ff5a72019-09-11 13:49:33 +02002452 return LY_EVALID;
2453 }
2454
2455 return LY_SUCCESS;
2456
2457predicate:
2458 /* Predicate* */
Michal Vasko004d3152020-06-11 19:59:22 +02002459 while (!lyxp_check_token(NULL, exp, *tok_idx, LYXP_TOKEN_BRACK1)) {
aPiecekbf968d92021-05-27 14:35:05 +02002460 rc = reparse_predicate(ctx, exp, tok_idx, depth);
Michal Vasko03ff5a72019-09-11 13:49:33 +02002461 LY_CHECK_RET(rc);
2462 }
2463
2464 /* ('/' or '//') RelativeLocationPath */
Michal Vasko004d3152020-06-11 19:59:22 +02002465 if (!exp_check_token2(NULL, exp, *tok_idx, LYXP_TOKEN_OPER_PATH, LYXP_TOKEN_OPER_RPATH)) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02002466
2467 /* '/' or '//' */
Michal Vasko004d3152020-06-11 19:59:22 +02002468 ++(*tok_idx);
Michal Vasko03ff5a72019-09-11 13:49:33 +02002469
aPiecekbf968d92021-05-27 14:35:05 +02002470 rc = reparse_relative_location_path(ctx, exp, tok_idx, depth);
Michal Vasko03ff5a72019-09-11 13:49:33 +02002471 LY_CHECK_RET(rc);
2472 }
2473
2474 return LY_SUCCESS;
2475}
2476
2477/**
2478 * @brief Reparse UnaryExpr. Logs directly on error.
2479 *
2480 * [17] UnaryExpr ::= UnionExpr | '-' UnaryExpr
2481 * [18] UnionExpr ::= PathExpr | UnionExpr '|' PathExpr
2482 *
2483 * @param[in] ctx Context for logging.
2484 * @param[in] exp Parsed XPath expression.
Michal Vasko004d3152020-06-11 19:59:22 +02002485 * @param[in] tok_idx Position in the expression @p exp.
aPiecekbf968d92021-05-27 14:35:05 +02002486 * @param[in] depth Current number of nested expressions.
Michal Vasko03ff5a72019-09-11 13:49:33 +02002487 * @return LY_ERR
2488 */
2489static LY_ERR
aPiecekbf968d92021-05-27 14:35:05 +02002490reparse_unary_expr(const struct ly_ctx *ctx, struct lyxp_expr *exp, uint16_t *tok_idx, uint16_t depth)
Michal Vasko03ff5a72019-09-11 13:49:33 +02002491{
2492 uint16_t prev_exp;
2493 LY_ERR rc;
2494
2495 /* ('-')* */
Michal Vasko004d3152020-06-11 19:59:22 +02002496 prev_exp = *tok_idx;
Michal Vasko69730152020-10-09 16:30:07 +02002497 while (!lyxp_check_token(NULL, exp, *tok_idx, LYXP_TOKEN_OPER_MATH) &&
2498 (exp->expr[exp->tok_pos[*tok_idx]] == '-')) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02002499 exp_repeat_push(exp, prev_exp, LYXP_EXPR_UNARY);
Michal Vasko004d3152020-06-11 19:59:22 +02002500 ++(*tok_idx);
Michal Vasko03ff5a72019-09-11 13:49:33 +02002501 }
2502
2503 /* PathExpr */
Michal Vasko004d3152020-06-11 19:59:22 +02002504 prev_exp = *tok_idx;
aPiecekbf968d92021-05-27 14:35:05 +02002505 rc = reparse_path_expr(ctx, exp, tok_idx, depth);
Michal Vasko03ff5a72019-09-11 13:49:33 +02002506 LY_CHECK_RET(rc);
2507
2508 /* ('|' PathExpr)* */
Michal Vasko004d3152020-06-11 19:59:22 +02002509 while (!lyxp_check_token(NULL, exp, *tok_idx, LYXP_TOKEN_OPER_UNI)) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02002510 exp_repeat_push(exp, prev_exp, LYXP_EXPR_UNION);
Michal Vasko004d3152020-06-11 19:59:22 +02002511 ++(*tok_idx);
Michal Vasko03ff5a72019-09-11 13:49:33 +02002512
aPiecekbf968d92021-05-27 14:35:05 +02002513 rc = reparse_path_expr(ctx, exp, tok_idx, depth);
Michal Vasko03ff5a72019-09-11 13:49:33 +02002514 LY_CHECK_RET(rc);
2515 }
2516
2517 return LY_SUCCESS;
2518}
2519
2520/**
2521 * @brief Reparse AdditiveExpr. Logs directly on error.
2522 *
2523 * [15] AdditiveExpr ::= MultiplicativeExpr
2524 * | AdditiveExpr '+' MultiplicativeExpr
2525 * | AdditiveExpr '-' MultiplicativeExpr
2526 * [16] MultiplicativeExpr ::= UnaryExpr
2527 * | MultiplicativeExpr '*' UnaryExpr
2528 * | MultiplicativeExpr 'div' UnaryExpr
2529 * | MultiplicativeExpr 'mod' UnaryExpr
2530 *
2531 * @param[in] ctx Context for logging.
2532 * @param[in] exp Parsed XPath expression.
Michal Vasko004d3152020-06-11 19:59:22 +02002533 * @param[in] tok_idx Position in the expression @p exp.
aPiecekbf968d92021-05-27 14:35:05 +02002534 * @param[in] depth Current number of nested expressions.
Michal Vasko03ff5a72019-09-11 13:49:33 +02002535 * @return LY_ERR
2536 */
2537static LY_ERR
aPiecekbf968d92021-05-27 14:35:05 +02002538reparse_additive_expr(const struct ly_ctx *ctx, struct lyxp_expr *exp, uint16_t *tok_idx, uint16_t depth)
Michal Vasko03ff5a72019-09-11 13:49:33 +02002539{
2540 uint16_t prev_add_exp, prev_mul_exp;
2541 LY_ERR rc;
2542
Michal Vasko004d3152020-06-11 19:59:22 +02002543 prev_add_exp = *tok_idx;
Michal Vasko03ff5a72019-09-11 13:49:33 +02002544 goto reparse_multiplicative_expr;
2545
2546 /* ('+' / '-' MultiplicativeExpr)* */
Michal Vasko69730152020-10-09 16:30:07 +02002547 while (!lyxp_check_token(NULL, exp, *tok_idx, LYXP_TOKEN_OPER_MATH) &&
2548 ((exp->expr[exp->tok_pos[*tok_idx]] == '+') || (exp->expr[exp->tok_pos[*tok_idx]] == '-'))) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02002549 exp_repeat_push(exp, prev_add_exp, LYXP_EXPR_ADDITIVE);
Michal Vasko004d3152020-06-11 19:59:22 +02002550 ++(*tok_idx);
Michal Vasko03ff5a72019-09-11 13:49:33 +02002551
2552reparse_multiplicative_expr:
2553 /* UnaryExpr */
Michal Vasko004d3152020-06-11 19:59:22 +02002554 prev_mul_exp = *tok_idx;
aPiecekbf968d92021-05-27 14:35:05 +02002555 rc = reparse_unary_expr(ctx, exp, tok_idx, depth);
Michal Vasko03ff5a72019-09-11 13:49:33 +02002556 LY_CHECK_RET(rc);
2557
2558 /* ('*' / 'div' / 'mod' UnaryExpr)* */
Michal Vasko69730152020-10-09 16:30:07 +02002559 while (!lyxp_check_token(NULL, exp, *tok_idx, LYXP_TOKEN_OPER_MATH) &&
2560 ((exp->expr[exp->tok_pos[*tok_idx]] == '*') || (exp->tok_len[*tok_idx] == 3))) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02002561 exp_repeat_push(exp, prev_mul_exp, LYXP_EXPR_MULTIPLICATIVE);
Michal Vasko004d3152020-06-11 19:59:22 +02002562 ++(*tok_idx);
Michal Vasko03ff5a72019-09-11 13:49:33 +02002563
aPiecekbf968d92021-05-27 14:35:05 +02002564 rc = reparse_unary_expr(ctx, exp, tok_idx, depth);
Michal Vasko03ff5a72019-09-11 13:49:33 +02002565 LY_CHECK_RET(rc);
2566 }
2567 }
2568
2569 return LY_SUCCESS;
2570}
2571
2572/**
2573 * @brief Reparse EqualityExpr. Logs directly on error.
2574 *
2575 * [13] EqualityExpr ::= RelationalExpr | EqualityExpr '=' RelationalExpr
2576 * | EqualityExpr '!=' RelationalExpr
2577 * [14] RelationalExpr ::= AdditiveExpr
2578 * | RelationalExpr '<' AdditiveExpr
2579 * | RelationalExpr '>' AdditiveExpr
2580 * | RelationalExpr '<=' AdditiveExpr
2581 * | RelationalExpr '>=' AdditiveExpr
2582 *
2583 * @param[in] ctx Context for logging.
2584 * @param[in] exp Parsed XPath expression.
Michal Vasko004d3152020-06-11 19:59:22 +02002585 * @param[in] tok_idx Position in the expression @p exp.
aPiecekbf968d92021-05-27 14:35:05 +02002586 * @param[in] depth Current number of nested expressions.
Michal Vasko03ff5a72019-09-11 13:49:33 +02002587 * @return LY_ERR
2588 */
2589static LY_ERR
aPiecekbf968d92021-05-27 14:35:05 +02002590reparse_equality_expr(const struct ly_ctx *ctx, struct lyxp_expr *exp, uint16_t *tok_idx, uint16_t depth)
Michal Vasko03ff5a72019-09-11 13:49:33 +02002591{
2592 uint16_t prev_eq_exp, prev_rel_exp;
2593 LY_ERR rc;
2594
Michal Vasko004d3152020-06-11 19:59:22 +02002595 prev_eq_exp = *tok_idx;
Michal Vasko03ff5a72019-09-11 13:49:33 +02002596 goto reparse_additive_expr;
2597
2598 /* ('=' / '!=' RelationalExpr)* */
Michal Vasko004d3152020-06-11 19:59:22 +02002599 while (!exp_check_token2(NULL, exp, *tok_idx, LYXP_TOKEN_OPER_EQUAL, LYXP_TOKEN_OPER_NEQUAL)) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02002600 exp_repeat_push(exp, prev_eq_exp, LYXP_EXPR_EQUALITY);
Michal Vasko004d3152020-06-11 19:59:22 +02002601 ++(*tok_idx);
Michal Vasko03ff5a72019-09-11 13:49:33 +02002602
2603reparse_additive_expr:
2604 /* AdditiveExpr */
Michal Vasko004d3152020-06-11 19:59:22 +02002605 prev_rel_exp = *tok_idx;
aPiecekbf968d92021-05-27 14:35:05 +02002606 rc = reparse_additive_expr(ctx, exp, tok_idx, depth);
Michal Vasko03ff5a72019-09-11 13:49:33 +02002607 LY_CHECK_RET(rc);
2608
2609 /* ('<' / '>' / '<=' / '>=' AdditiveExpr)* */
Michal Vasko004d3152020-06-11 19:59:22 +02002610 while (!lyxp_check_token(NULL, exp, *tok_idx, LYXP_TOKEN_OPER_COMP)) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02002611 exp_repeat_push(exp, prev_rel_exp, LYXP_EXPR_RELATIONAL);
Michal Vasko004d3152020-06-11 19:59:22 +02002612 ++(*tok_idx);
Michal Vasko03ff5a72019-09-11 13:49:33 +02002613
aPiecekbf968d92021-05-27 14:35:05 +02002614 rc = reparse_additive_expr(ctx, exp, tok_idx, depth);
Michal Vasko03ff5a72019-09-11 13:49:33 +02002615 LY_CHECK_RET(rc);
2616 }
2617 }
2618
2619 return LY_SUCCESS;
2620}
2621
2622/**
2623 * @brief Reparse OrExpr. Logs directly on error.
2624 *
2625 * [11] OrExpr ::= AndExpr | OrExpr 'or' AndExpr
2626 * [12] AndExpr ::= EqualityExpr | AndExpr 'and' EqualityExpr
2627 *
2628 * @param[in] ctx Context for logging.
2629 * @param[in] exp Parsed XPath expression.
Michal Vasko004d3152020-06-11 19:59:22 +02002630 * @param[in] tok_idx Position in the expression @p exp.
aPiecekbf968d92021-05-27 14:35:05 +02002631 * @param[in] depth Current number of nested expressions.
Michal Vasko03ff5a72019-09-11 13:49:33 +02002632 * @return LY_ERR
2633 */
2634static LY_ERR
aPiecekbf968d92021-05-27 14:35:05 +02002635reparse_or_expr(const struct ly_ctx *ctx, struct lyxp_expr *exp, uint16_t *tok_idx, uint16_t depth)
Michal Vasko03ff5a72019-09-11 13:49:33 +02002636{
2637 uint16_t prev_or_exp, prev_and_exp;
2638 LY_ERR rc;
2639
aPiecekbf968d92021-05-27 14:35:05 +02002640 ++depth;
2641 LY_CHECK_ERR_RET(depth > LYXP_MAX_BLOCK_DEPTH, LOGVAL(ctx, LY_VCODE_XP_DEPTH), LY_EINVAL);
2642
Michal Vasko004d3152020-06-11 19:59:22 +02002643 prev_or_exp = *tok_idx;
Michal Vasko03ff5a72019-09-11 13:49:33 +02002644 goto reparse_equality_expr;
2645
2646 /* ('or' AndExpr)* */
Michal Vasko004d3152020-06-11 19:59:22 +02002647 while (!lyxp_check_token(NULL, exp, *tok_idx, LYXP_TOKEN_OPER_LOG) && (exp->tok_len[*tok_idx] == 2)) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02002648 exp_repeat_push(exp, prev_or_exp, LYXP_EXPR_OR);
Michal Vasko004d3152020-06-11 19:59:22 +02002649 ++(*tok_idx);
Michal Vasko03ff5a72019-09-11 13:49:33 +02002650
2651reparse_equality_expr:
2652 /* EqualityExpr */
Michal Vasko004d3152020-06-11 19:59:22 +02002653 prev_and_exp = *tok_idx;
aPiecekbf968d92021-05-27 14:35:05 +02002654 rc = reparse_equality_expr(ctx, exp, tok_idx, depth);
Michal Vasko03ff5a72019-09-11 13:49:33 +02002655 LY_CHECK_RET(rc);
2656
2657 /* ('and' EqualityExpr)* */
Michal Vasko004d3152020-06-11 19:59:22 +02002658 while (!lyxp_check_token(NULL, exp, *tok_idx, LYXP_TOKEN_OPER_LOG) && (exp->tok_len[*tok_idx] == 3)) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02002659 exp_repeat_push(exp, prev_and_exp, LYXP_EXPR_AND);
Michal Vasko004d3152020-06-11 19:59:22 +02002660 ++(*tok_idx);
Michal Vasko03ff5a72019-09-11 13:49:33 +02002661
aPiecekbf968d92021-05-27 14:35:05 +02002662 rc = reparse_equality_expr(ctx, exp, tok_idx, depth);
Michal Vasko03ff5a72019-09-11 13:49:33 +02002663 LY_CHECK_RET(rc);
2664 }
2665 }
2666
2667 return LY_SUCCESS;
2668}
Radek Krejcib1646a92018-11-02 16:08:26 +01002669
2670/**
2671 * @brief Parse NCName.
2672 *
2673 * @param[in] ncname Name to parse.
Michal Vasko03ff5a72019-09-11 13:49:33 +02002674 * @return Length of @p ncname valid bytes.
Radek Krejcib1646a92018-11-02 16:08:26 +01002675 */
Michal Vasko49fec8e2022-05-24 10:28:33 +02002676static ssize_t
Radek Krejcib1646a92018-11-02 16:08:26 +01002677parse_ncname(const char *ncname)
2678{
Radek Krejci1deb5be2020-08-26 16:43:36 +02002679 uint32_t uc;
Radek Krejcid4270262019-01-07 15:07:25 +01002680 size_t size;
Michal Vasko49fec8e2022-05-24 10:28:33 +02002681 ssize_t len = 0;
Radek Krejcib1646a92018-11-02 16:08:26 +01002682
2683 LY_CHECK_RET(ly_getutf8(&ncname, &uc, &size), 0);
2684 if (!is_xmlqnamestartchar(uc) || (uc == ':')) {
2685 return len;
2686 }
2687
2688 do {
2689 len += size;
Radek Krejci9a564c92019-01-07 14:53:57 +01002690 if (!*ncname) {
2691 break;
2692 }
Radek Krejcid4270262019-01-07 15:07:25 +01002693 LY_CHECK_RET(ly_getutf8(&ncname, &uc, &size), -len);
Radek Krejcib1646a92018-11-02 16:08:26 +01002694 } while (is_xmlqnamechar(uc) && (uc != ':'));
2695
2696 return len;
2697}
2698
2699/**
Michal Vasko03ff5a72019-09-11 13:49:33 +02002700 * @brief Add @p token into the expression @p exp.
Radek Krejcib1646a92018-11-02 16:08:26 +01002701 *
Michal Vasko03ff5a72019-09-11 13:49:33 +02002702 * @param[in] ctx Context for logging.
Radek Krejcib1646a92018-11-02 16:08:26 +01002703 * @param[in] exp Expression to use.
2704 * @param[in] token Token to add.
Michal Vasko03ff5a72019-09-11 13:49:33 +02002705 * @param[in] tok_pos Token position in the XPath expression.
Radek Krejcib1646a92018-11-02 16:08:26 +01002706 * @param[in] tok_len Token length in the XPath expression.
Michal Vasko03ff5a72019-09-11 13:49:33 +02002707 * @return LY_ERR
Radek Krejcib1646a92018-11-02 16:08:26 +01002708 */
2709static LY_ERR
Michal Vasko14676352020-05-29 11:35:55 +02002710exp_add_token(const struct ly_ctx *ctx, struct lyxp_expr *exp, enum lyxp_token token, uint16_t tok_pos, uint16_t tok_len)
Radek Krejcib1646a92018-11-02 16:08:26 +01002711{
2712 uint32_t prev;
2713
2714 if (exp->used == exp->size) {
2715 prev = exp->size;
2716 exp->size += LYXP_EXPR_SIZE_STEP;
2717 if (prev > exp->size) {
2718 LOGINT(ctx);
2719 return LY_EINT;
2720 }
2721
2722 exp->tokens = ly_realloc(exp->tokens, exp->size * sizeof *exp->tokens);
2723 LY_CHECK_ERR_RET(!exp->tokens, LOGMEM(ctx), LY_EMEM);
2724 exp->tok_pos = ly_realloc(exp->tok_pos, exp->size * sizeof *exp->tok_pos);
2725 LY_CHECK_ERR_RET(!exp->tok_pos, LOGMEM(ctx), LY_EMEM);
2726 exp->tok_len = ly_realloc(exp->tok_len, exp->size * sizeof *exp->tok_len);
2727 LY_CHECK_ERR_RET(!exp->tok_len, LOGMEM(ctx), LY_EMEM);
2728 }
2729
2730 exp->tokens[exp->used] = token;
Michal Vasko03ff5a72019-09-11 13:49:33 +02002731 exp->tok_pos[exp->used] = tok_pos;
Radek Krejcib1646a92018-11-02 16:08:26 +01002732 exp->tok_len[exp->used] = tok_len;
2733 ++exp->used;
2734 return LY_SUCCESS;
2735}
2736
2737void
Michal Vasko14676352020-05-29 11:35:55 +02002738lyxp_expr_free(const struct ly_ctx *ctx, struct lyxp_expr *expr)
Radek Krejcib1646a92018-11-02 16:08:26 +01002739{
2740 uint16_t i;
2741
2742 if (!expr) {
2743 return;
2744 }
2745
2746 lydict_remove(ctx, expr->expr);
2747 free(expr->tokens);
2748 free(expr->tok_pos);
2749 free(expr->tok_len);
2750 if (expr->repeat) {
2751 for (i = 0; i < expr->used; ++i) {
2752 free(expr->repeat[i]);
2753 }
2754 }
2755 free(expr->repeat);
2756 free(expr);
2757}
2758
Michal Vasko49fec8e2022-05-24 10:28:33 +02002759/**
2760 * @brief Parse Axis name.
2761 *
2762 * @param[in] str String to parse.
2763 * @param[in] str_len Length of @p str.
2764 * @return LY_SUCCESS if an axis.
2765 * @return LY_ENOT otherwise.
2766 */
2767static LY_ERR
2768expr_parse_axis(const char *str, size_t str_len)
2769{
2770 switch (str_len) {
2771 case 4:
2772 if (!strncmp("self", str, str_len)) {
2773 return LY_SUCCESS;
2774 }
2775 break;
2776 case 5:
2777 if (!strncmp("child", str, str_len)) {
2778 return LY_SUCCESS;
2779 }
2780 break;
2781 case 6:
2782 if (!strncmp("parent", str, str_len)) {
2783 return LY_SUCCESS;
2784 }
2785 break;
2786 case 8:
2787 if (!strncmp("ancestor", str, str_len)) {
2788 return LY_SUCCESS;
2789 }
2790 break;
2791 case 9:
2792 if (!strncmp("attribute", str, str_len)) {
2793 return LY_SUCCESS;
2794 } else if (!strncmp("following", str, str_len)) {
2795 return LY_SUCCESS;
2796 } else if (!strncmp("namespace", str, str_len)) {
2797 LOGERR(NULL, LY_EINVAL, "Axis \"namespace\" not supported.");
2798 return LY_ENOT;
2799 } else if (!strncmp("preceding", str, str_len)) {
2800 return LY_SUCCESS;
2801 }
2802 break;
2803 case 10:
2804 if (!strncmp("descendant", str, str_len)) {
2805 return LY_SUCCESS;
2806 }
2807 break;
2808 case 16:
2809 if (!strncmp("ancestor-or-self", str, str_len)) {
2810 return LY_SUCCESS;
2811 }
2812 break;
2813 case 17:
2814 if (!strncmp("following-sibling", str, str_len)) {
2815 return LY_SUCCESS;
2816 } else if (!strncmp("preceding-sibling", str, str_len)) {
2817 return LY_SUCCESS;
2818 }
2819 break;
2820 case 18:
2821 if (!strncmp("descendant-or-self", str, str_len)) {
2822 return LY_SUCCESS;
2823 }
2824 break;
2825 }
2826
2827 return LY_ENOT;
2828}
2829
Radek Krejcif03a9e22020-09-18 20:09:31 +02002830LY_ERR
2831lyxp_expr_parse(const struct ly_ctx *ctx, const char *expr_str, size_t expr_len, ly_bool reparse, struct lyxp_expr **expr_p)
Radek Krejcib1646a92018-11-02 16:08:26 +01002832{
Radek Krejcif03a9e22020-09-18 20:09:31 +02002833 LY_ERR ret = LY_SUCCESS;
2834 struct lyxp_expr *expr;
Radek Krejcid4270262019-01-07 15:07:25 +01002835 size_t parsed = 0, tok_len;
Radek Krejcib1646a92018-11-02 16:08:26 +01002836 enum lyxp_token tok_type;
Michal Vasko49fec8e2022-05-24 10:28:33 +02002837 ly_bool prev_func_check = 0, prev_ntype_check = 0, has_axis;
Michal Vasko004d3152020-06-11 19:59:22 +02002838 uint16_t tok_idx = 0;
Michal Vasko49fec8e2022-05-24 10:28:33 +02002839 ssize_t ncname_len;
Radek Krejcib1646a92018-11-02 16:08:26 +01002840
Radek Krejcif03a9e22020-09-18 20:09:31 +02002841 assert(expr_p);
2842
2843 if (!expr_str[0]) {
Radek Krejci2efc45b2020-12-22 16:25:44 +01002844 LOGVAL(ctx, LY_VCODE_XP_EOF);
Radek Krejcif03a9e22020-09-18 20:09:31 +02002845 return LY_EVALID;
Michal Vasko004d3152020-06-11 19:59:22 +02002846 }
2847
2848 if (!expr_len) {
Radek Krejcif03a9e22020-09-18 20:09:31 +02002849 expr_len = strlen(expr_str);
Michal Vasko004d3152020-06-11 19:59:22 +02002850 }
2851 if (expr_len > UINT16_MAX) {
Michal Vasko623ac7b2021-04-09 12:44:23 +02002852 LOGVAL(ctx, LYVE_XPATH, "XPath expression cannot be longer than %u characters.", UINT16_MAX);
Radek Krejcif03a9e22020-09-18 20:09:31 +02002853 return LY_EVALID;
Radek Krejcib1646a92018-11-02 16:08:26 +01002854 }
2855
2856 /* init lyxp_expr structure */
Radek Krejcif03a9e22020-09-18 20:09:31 +02002857 expr = calloc(1, sizeof *expr);
2858 LY_CHECK_ERR_GOTO(!expr, LOGMEM(ctx); ret = LY_EMEM, error);
2859 LY_CHECK_GOTO(ret = lydict_insert(ctx, expr_str, expr_len, &expr->expr), error);
2860 expr->used = 0;
2861 expr->size = LYXP_EXPR_SIZE_START;
2862 expr->tokens = malloc(expr->size * sizeof *expr->tokens);
2863 LY_CHECK_ERR_GOTO(!expr->tokens, LOGMEM(ctx); ret = LY_EMEM, error);
Radek Krejcib1646a92018-11-02 16:08:26 +01002864
Radek Krejcif03a9e22020-09-18 20:09:31 +02002865 expr->tok_pos = malloc(expr->size * sizeof *expr->tok_pos);
2866 LY_CHECK_ERR_GOTO(!expr->tok_pos, LOGMEM(ctx); ret = LY_EMEM, error);
Radek Krejcib1646a92018-11-02 16:08:26 +01002867
Radek Krejcif03a9e22020-09-18 20:09:31 +02002868 expr->tok_len = malloc(expr->size * sizeof *expr->tok_len);
2869 LY_CHECK_ERR_GOTO(!expr->tok_len, LOGMEM(ctx); ret = LY_EMEM, error);
Radek Krejcib1646a92018-11-02 16:08:26 +01002870
Michal Vasko004d3152020-06-11 19:59:22 +02002871 /* make expr 0-terminated */
Radek Krejcif03a9e22020-09-18 20:09:31 +02002872 expr_str = expr->expr;
Michal Vasko004d3152020-06-11 19:59:22 +02002873
Radek Krejcif03a9e22020-09-18 20:09:31 +02002874 while (is_xmlws(expr_str[parsed])) {
Radek Krejcib1646a92018-11-02 16:08:26 +01002875 ++parsed;
2876 }
2877
2878 do {
Radek Krejcif03a9e22020-09-18 20:09:31 +02002879 if (expr_str[parsed] == '(') {
Radek Krejcib1646a92018-11-02 16:08:26 +01002880
2881 /* '(' */
2882 tok_len = 1;
2883 tok_type = LYXP_TOKEN_PAR1;
2884
Michal Vasko49fec8e2022-05-24 10:28:33 +02002885 if (prev_ntype_check && expr->used && (expr->tokens[expr->used - 1] == LYXP_TOKEN_NAMETEST) &&
2886 (((expr->tok_len[expr->used - 1] == 4) &&
2887 (!strncmp(&expr_str[expr->tok_pos[expr->used - 1]], "node", 4) ||
2888 !strncmp(&expr_str[expr->tok_pos[expr->used - 1]], "text", 4))) ||
2889 ((expr->tok_len[expr->used - 1] == 7) &&
2890 !strncmp(&expr_str[expr->tok_pos[expr->used - 1]], "comment", 7)))) {
2891 /* it is NodeType after all */
2892 expr->tokens[expr->used - 1] = LYXP_TOKEN_NODETYPE;
2893
2894 prev_ntype_check = 0;
2895 prev_func_check = 0;
2896 } else if (prev_func_check && expr->used && (expr->tokens[expr->used - 1] == LYXP_TOKEN_NAMETEST)) {
2897 /* it is FunctionName after all */
2898 expr->tokens[expr->used - 1] = LYXP_TOKEN_FUNCNAME;
2899
2900 prev_ntype_check = 0;
2901 prev_func_check = 0;
Radek Krejcib1646a92018-11-02 16:08:26 +01002902 }
2903
Radek Krejcif03a9e22020-09-18 20:09:31 +02002904 } else if (expr_str[parsed] == ')') {
Radek Krejcib1646a92018-11-02 16:08:26 +01002905
2906 /* ')' */
2907 tok_len = 1;
2908 tok_type = LYXP_TOKEN_PAR2;
2909
Radek Krejcif03a9e22020-09-18 20:09:31 +02002910 } else if (expr_str[parsed] == '[') {
Radek Krejcib1646a92018-11-02 16:08:26 +01002911
2912 /* '[' */
2913 tok_len = 1;
2914 tok_type = LYXP_TOKEN_BRACK1;
2915
Radek Krejcif03a9e22020-09-18 20:09:31 +02002916 } else if (expr_str[parsed] == ']') {
Radek Krejcib1646a92018-11-02 16:08:26 +01002917
2918 /* ']' */
2919 tok_len = 1;
2920 tok_type = LYXP_TOKEN_BRACK2;
2921
Radek Krejcif03a9e22020-09-18 20:09:31 +02002922 } else if (!strncmp(&expr_str[parsed], "..", 2)) {
Radek Krejcib1646a92018-11-02 16:08:26 +01002923
2924 /* '..' */
2925 tok_len = 2;
2926 tok_type = LYXP_TOKEN_DDOT;
2927
Radek Krejcif03a9e22020-09-18 20:09:31 +02002928 } else if ((expr_str[parsed] == '.') && (!isdigit(expr_str[parsed + 1]))) {
Radek Krejcib1646a92018-11-02 16:08:26 +01002929
2930 /* '.' */
2931 tok_len = 1;
2932 tok_type = LYXP_TOKEN_DOT;
2933
Radek Krejcif03a9e22020-09-18 20:09:31 +02002934 } else if (expr_str[parsed] == '@') {
Radek Krejcib1646a92018-11-02 16:08:26 +01002935
2936 /* '@' */
2937 tok_len = 1;
2938 tok_type = LYXP_TOKEN_AT;
2939
Radek Krejcif03a9e22020-09-18 20:09:31 +02002940 } else if (expr_str[parsed] == ',') {
Radek Krejcib1646a92018-11-02 16:08:26 +01002941
2942 /* ',' */
2943 tok_len = 1;
2944 tok_type = LYXP_TOKEN_COMMA;
2945
Radek Krejcif03a9e22020-09-18 20:09:31 +02002946 } else if (expr_str[parsed] == '\'') {
Radek Krejcib1646a92018-11-02 16:08:26 +01002947
2948 /* Literal with ' */
Radek Krejcif03a9e22020-09-18 20:09:31 +02002949 for (tok_len = 1; (expr_str[parsed + tok_len] != '\0') && (expr_str[parsed + tok_len] != '\''); ++tok_len) {}
2950 LY_CHECK_ERR_GOTO(expr_str[parsed + tok_len] == '\0',
Radek Krejci2efc45b2020-12-22 16:25:44 +01002951 LOGVAL(ctx, LY_VCODE_XP_EOE, expr_str[parsed], &expr_str[parsed]); ret = LY_EVALID,
Michal Vasko69730152020-10-09 16:30:07 +02002952 error);
Radek Krejcib1646a92018-11-02 16:08:26 +01002953 ++tok_len;
2954 tok_type = LYXP_TOKEN_LITERAL;
2955
Radek Krejcif03a9e22020-09-18 20:09:31 +02002956 } else if (expr_str[parsed] == '\"') {
Radek Krejcib1646a92018-11-02 16:08:26 +01002957
2958 /* Literal with " */
Radek Krejcif03a9e22020-09-18 20:09:31 +02002959 for (tok_len = 1; (expr_str[parsed + tok_len] != '\0') && (expr_str[parsed + tok_len] != '\"'); ++tok_len) {}
2960 LY_CHECK_ERR_GOTO(expr_str[parsed + tok_len] == '\0',
Radek Krejci2efc45b2020-12-22 16:25:44 +01002961 LOGVAL(ctx, LY_VCODE_XP_EOE, expr_str[parsed], &expr_str[parsed]); ret = LY_EVALID,
Michal Vasko69730152020-10-09 16:30:07 +02002962 error);
Radek Krejcib1646a92018-11-02 16:08:26 +01002963 ++tok_len;
2964 tok_type = LYXP_TOKEN_LITERAL;
2965
Radek Krejcif03a9e22020-09-18 20:09:31 +02002966 } else if ((expr_str[parsed] == '.') || (isdigit(expr_str[parsed]))) {
Radek Krejcib1646a92018-11-02 16:08:26 +01002967
2968 /* Number */
Radek Krejcif03a9e22020-09-18 20:09:31 +02002969 for (tok_len = 0; isdigit(expr_str[parsed + tok_len]); ++tok_len) {}
2970 if (expr_str[parsed + tok_len] == '.') {
Radek Krejcib1646a92018-11-02 16:08:26 +01002971 ++tok_len;
Radek Krejcif03a9e22020-09-18 20:09:31 +02002972 for ( ; isdigit(expr_str[parsed + tok_len]); ++tok_len) {}
Radek Krejcib1646a92018-11-02 16:08:26 +01002973 }
2974 tok_type = LYXP_TOKEN_NUMBER;
2975
aPiecekfba75362021-10-07 12:39:48 +02002976 } else if (expr_str[parsed] == '$') {
2977
2978 /* VariableReference */
2979 parsed++;
Michal Vasko49fec8e2022-05-24 10:28:33 +02002980 ncname_len = parse_ncname(&expr_str[parsed]);
aPiecekfba75362021-10-07 12:39:48 +02002981 LY_CHECK_ERR_GOTO(ncname_len < 1, LOGVAL(ctx, LY_VCODE_XP_INEXPR, expr_str[parsed - ncname_len],
2982 parsed - ncname_len + 1, expr_str); ret = LY_EVALID, error);
2983 tok_len = ncname_len;
2984 LY_CHECK_ERR_GOTO(expr_str[parsed + tok_len] == ':',
2985 LOGVAL(ctx, LYVE_XPATH, "Variable with prefix is not supported."); ret = LY_EVALID,
2986 error);
2987 tok_type = LYXP_TOKEN_VARREF;
2988
Radek Krejcif03a9e22020-09-18 20:09:31 +02002989 } else if (expr_str[parsed] == '/') {
Radek Krejcib1646a92018-11-02 16:08:26 +01002990
2991 /* Operator '/', '//' */
Radek Krejcif03a9e22020-09-18 20:09:31 +02002992 if (!strncmp(&expr_str[parsed], "//", 2)) {
Radek Krejcib1646a92018-11-02 16:08:26 +01002993 tok_len = 2;
Michal Vasko3e48bf32020-06-01 08:39:07 +02002994 tok_type = LYXP_TOKEN_OPER_RPATH;
Radek Krejcib1646a92018-11-02 16:08:26 +01002995 } else {
2996 tok_len = 1;
Michal Vasko3e48bf32020-06-01 08:39:07 +02002997 tok_type = LYXP_TOKEN_OPER_PATH;
Radek Krejcib1646a92018-11-02 16:08:26 +01002998 }
Radek Krejcib1646a92018-11-02 16:08:26 +01002999
Radek Krejcif03a9e22020-09-18 20:09:31 +02003000 } else if (!strncmp(&expr_str[parsed], "!=", 2)) {
Radek Krejcib1646a92018-11-02 16:08:26 +01003001
Michal Vasko3e48bf32020-06-01 08:39:07 +02003002 /* Operator '!=' */
Radek Krejcib1646a92018-11-02 16:08:26 +01003003 tok_len = 2;
Michal Vasko3e48bf32020-06-01 08:39:07 +02003004 tok_type = LYXP_TOKEN_OPER_NEQUAL;
3005
Radek Krejcif03a9e22020-09-18 20:09:31 +02003006 } else if (!strncmp(&expr_str[parsed], "<=", 2) || !strncmp(&expr_str[parsed], ">=", 2)) {
Michal Vasko3e48bf32020-06-01 08:39:07 +02003007
3008 /* Operator '<=', '>=' */
3009 tok_len = 2;
3010 tok_type = LYXP_TOKEN_OPER_COMP;
Radek Krejcib1646a92018-11-02 16:08:26 +01003011
Radek Krejcif03a9e22020-09-18 20:09:31 +02003012 } else if (expr_str[parsed] == '|') {
Radek Krejcib1646a92018-11-02 16:08:26 +01003013
3014 /* Operator '|' */
3015 tok_len = 1;
Michal Vasko3e48bf32020-06-01 08:39:07 +02003016 tok_type = LYXP_TOKEN_OPER_UNI;
Radek Krejcib1646a92018-11-02 16:08:26 +01003017
Radek Krejcif03a9e22020-09-18 20:09:31 +02003018 } else if ((expr_str[parsed] == '+') || (expr_str[parsed] == '-')) {
Radek Krejcib1646a92018-11-02 16:08:26 +01003019
3020 /* Operator '+', '-' */
3021 tok_len = 1;
Michal Vasko3e48bf32020-06-01 08:39:07 +02003022 tok_type = LYXP_TOKEN_OPER_MATH;
Radek Krejcib1646a92018-11-02 16:08:26 +01003023
Radek Krejcif03a9e22020-09-18 20:09:31 +02003024 } else if (expr_str[parsed] == '=') {
Radek Krejcib1646a92018-11-02 16:08:26 +01003025
Michal Vasko3e48bf32020-06-01 08:39:07 +02003026 /* Operator '=' */
Radek Krejcib1646a92018-11-02 16:08:26 +01003027 tok_len = 1;
Michal Vasko3e48bf32020-06-01 08:39:07 +02003028 tok_type = LYXP_TOKEN_OPER_EQUAL;
3029
Radek Krejcif03a9e22020-09-18 20:09:31 +02003030 } else if ((expr_str[parsed] == '<') || (expr_str[parsed] == '>')) {
Michal Vasko3e48bf32020-06-01 08:39:07 +02003031
3032 /* Operator '<', '>' */
3033 tok_len = 1;
3034 tok_type = LYXP_TOKEN_OPER_COMP;
Radek Krejcib1646a92018-11-02 16:08:26 +01003035
Michal Vasko69730152020-10-09 16:30:07 +02003036 } else if (expr->used && (expr->tokens[expr->used - 1] != LYXP_TOKEN_AT) &&
3037 (expr->tokens[expr->used - 1] != LYXP_TOKEN_PAR1) &&
3038 (expr->tokens[expr->used - 1] != LYXP_TOKEN_BRACK1) &&
3039 (expr->tokens[expr->used - 1] != LYXP_TOKEN_COMMA) &&
3040 (expr->tokens[expr->used - 1] != LYXP_TOKEN_OPER_LOG) &&
3041 (expr->tokens[expr->used - 1] != LYXP_TOKEN_OPER_EQUAL) &&
3042 (expr->tokens[expr->used - 1] != LYXP_TOKEN_OPER_NEQUAL) &&
3043 (expr->tokens[expr->used - 1] != LYXP_TOKEN_OPER_COMP) &&
3044 (expr->tokens[expr->used - 1] != LYXP_TOKEN_OPER_MATH) &&
3045 (expr->tokens[expr->used - 1] != LYXP_TOKEN_OPER_UNI) &&
3046 (expr->tokens[expr->used - 1] != LYXP_TOKEN_OPER_PATH) &&
3047 (expr->tokens[expr->used - 1] != LYXP_TOKEN_OPER_RPATH)) {
Radek Krejcib1646a92018-11-02 16:08:26 +01003048
3049 /* Operator '*', 'or', 'and', 'mod', or 'div' */
Radek Krejcif03a9e22020-09-18 20:09:31 +02003050 if (expr_str[parsed] == '*') {
Radek Krejcib1646a92018-11-02 16:08:26 +01003051 tok_len = 1;
Michal Vasko3e48bf32020-06-01 08:39:07 +02003052 tok_type = LYXP_TOKEN_OPER_MATH;
Radek Krejcib1646a92018-11-02 16:08:26 +01003053
Radek Krejcif03a9e22020-09-18 20:09:31 +02003054 } else if (!strncmp(&expr_str[parsed], "or", 2)) {
Radek Krejcib1646a92018-11-02 16:08:26 +01003055 tok_len = 2;
Michal Vasko3e48bf32020-06-01 08:39:07 +02003056 tok_type = LYXP_TOKEN_OPER_LOG;
Radek Krejcib1646a92018-11-02 16:08:26 +01003057
Radek Krejcif03a9e22020-09-18 20:09:31 +02003058 } else if (!strncmp(&expr_str[parsed], "and", 3)) {
Radek Krejcib1646a92018-11-02 16:08:26 +01003059 tok_len = 3;
Michal Vasko3e48bf32020-06-01 08:39:07 +02003060 tok_type = LYXP_TOKEN_OPER_LOG;
Radek Krejcib1646a92018-11-02 16:08:26 +01003061
Radek Krejcif03a9e22020-09-18 20:09:31 +02003062 } else if (!strncmp(&expr_str[parsed], "mod", 3) || !strncmp(&expr_str[parsed], "div", 3)) {
Radek Krejcib1646a92018-11-02 16:08:26 +01003063 tok_len = 3;
Michal Vasko3e48bf32020-06-01 08:39:07 +02003064 tok_type = LYXP_TOKEN_OPER_MATH;
Radek Krejcib1646a92018-11-02 16:08:26 +01003065
Michal Vasko49fec8e2022-05-24 10:28:33 +02003066 } else if (prev_ntype_check || prev_func_check) {
Radek Krejci2efc45b2020-12-22 16:25:44 +01003067 LOGVAL(ctx, LYVE_XPATH, "Invalid character 0x%x ('%c'), perhaps \"%.*s\" is supposed to be a function call.",
Michal Vasko69730152020-10-09 16:30:07 +02003068 expr_str[parsed], expr_str[parsed], expr->tok_len[expr->used - 1], &expr->expr[expr->tok_pos[expr->used - 1]]);
Radek Krejcif03a9e22020-09-18 20:09:31 +02003069 ret = LY_EVALID;
Radek Krejcib1646a92018-11-02 16:08:26 +01003070 goto error;
3071 } else {
Michal Vasko774ce402021-04-14 15:35:06 +02003072 LOGVAL(ctx, LY_VCODE_XP_INEXPR, expr_str[parsed], parsed + 1, expr_str);
Radek Krejcif03a9e22020-09-18 20:09:31 +02003073 ret = LY_EVALID;
Radek Krejcib1646a92018-11-02 16:08:26 +01003074 goto error;
3075 }
Radek Krejcib1646a92018-11-02 16:08:26 +01003076 } else {
3077
Michal Vasko49fec8e2022-05-24 10:28:33 +02003078 /* (AxisName '::')? ((NCName ':')? '*' | QName) or NodeType/FunctionName */
3079 if (expr_str[parsed] == '*') {
3080 ncname_len = 1;
3081 } else {
3082 ncname_len = parse_ncname(&expr_str[parsed]);
3083 LY_CHECK_ERR_GOTO(ncname_len < 1, LOGVAL(ctx, LY_VCODE_XP_INEXPR, expr_str[parsed - ncname_len],
3084 parsed - ncname_len + 1, expr_str); ret = LY_EVALID, error);
3085 }
Radek Krejcib1646a92018-11-02 16:08:26 +01003086 tok_len = ncname_len;
3087
Michal Vasko49fec8e2022-05-24 10:28:33 +02003088 has_axis = 0;
3089 if (!strncmp(&expr_str[parsed + tok_len], "::", 2)) {
3090 /* axis */
3091 LY_CHECK_ERR_GOTO(expr_parse_axis(&expr_str[parsed], ncname_len),
3092 LOGVAL(ctx, LY_VCODE_XP_INEXPR, expr_str[parsed], parsed + 1, expr_str); ret = LY_EVALID, error);
3093 tok_type = LYXP_TOKEN_AXISNAME;
3094
3095 LY_CHECK_GOTO(ret = exp_add_token(ctx, expr, tok_type, parsed, tok_len), error);
3096 parsed += tok_len;
3097
3098 /* '::' */
3099 tok_len = 2;
3100 tok_type = LYXP_TOKEN_DCOLON;
3101
3102 LY_CHECK_GOTO(ret = exp_add_token(ctx, expr, tok_type, parsed, tok_len), error);
3103 parsed += tok_len;
3104
3105 if (expr_str[parsed] == '*') {
3106 ncname_len = 1;
3107 } else {
3108 ncname_len = parse_ncname(&expr_str[parsed]);
3109 LY_CHECK_ERR_GOTO(ncname_len < 1, LOGVAL(ctx, LY_VCODE_XP_INEXPR, expr_str[parsed - ncname_len],
3110 parsed - ncname_len + 1, expr_str); ret = LY_EVALID, error);
3111 }
3112 tok_len = ncname_len;
3113
3114 has_axis = 1;
3115 }
3116
Radek Krejcif03a9e22020-09-18 20:09:31 +02003117 if (expr_str[parsed + tok_len] == ':') {
Radek Krejcib1646a92018-11-02 16:08:26 +01003118 ++tok_len;
Radek Krejcif03a9e22020-09-18 20:09:31 +02003119 if (expr_str[parsed + tok_len] == '*') {
Radek Krejcib1646a92018-11-02 16:08:26 +01003120 ++tok_len;
3121 } else {
Radek Krejcif03a9e22020-09-18 20:09:31 +02003122 ncname_len = parse_ncname(&expr_str[parsed + tok_len]);
Michal Vaskoe2be5462021-08-04 10:49:42 +02003123 LY_CHECK_ERR_GOTO(ncname_len < 1, LOGVAL(ctx, LY_VCODE_XP_INEXPR, expr_str[parsed - ncname_len],
Michal Vasko774ce402021-04-14 15:35:06 +02003124 parsed - ncname_len + 1, expr_str); ret = LY_EVALID, error);
Radek Krejcib1646a92018-11-02 16:08:26 +01003125 tok_len += ncname_len;
3126 }
Michal Vasko49fec8e2022-05-24 10:28:33 +02003127 /* remove old flags to prevent ambiguities */
3128 prev_ntype_check = 0;
3129 prev_func_check = 0;
Radek Krejcib1646a92018-11-02 16:08:26 +01003130 tok_type = LYXP_TOKEN_NAMETEST;
3131 } else {
Michal Vasko49fec8e2022-05-24 10:28:33 +02003132 /* if not '*', there is no prefix so it can still be NodeType/FunctionName, we can't finally decide now */
3133 prev_ntype_check = (expr_str[parsed] == '*') ? 0 : 1;
3134 prev_func_check = (prev_ntype_check && !has_axis) ? 1 : 0;
Radek Krejcib1646a92018-11-02 16:08:26 +01003135 tok_type = LYXP_TOKEN_NAMETEST;
3136 }
3137 }
3138
3139 /* store the token, move on to the next one */
Radek Krejcif03a9e22020-09-18 20:09:31 +02003140 LY_CHECK_GOTO(ret = exp_add_token(ctx, expr, tok_type, parsed, tok_len), error);
Radek Krejcib1646a92018-11-02 16:08:26 +01003141 parsed += tok_len;
Radek Krejcif03a9e22020-09-18 20:09:31 +02003142 while (is_xmlws(expr_str[parsed])) {
Radek Krejcib1646a92018-11-02 16:08:26 +01003143 ++parsed;
3144 }
3145
Radek Krejcif03a9e22020-09-18 20:09:31 +02003146 } while (expr_str[parsed]);
Radek Krejcib1646a92018-11-02 16:08:26 +01003147
Michal Vasko004d3152020-06-11 19:59:22 +02003148 if (reparse) {
3149 /* prealloc repeat */
Radek Krejcif03a9e22020-09-18 20:09:31 +02003150 expr->repeat = calloc(expr->size, sizeof *expr->repeat);
3151 LY_CHECK_ERR_GOTO(!expr->repeat, LOGMEM(ctx); ret = LY_EMEM, error);
Radek Krejcib1646a92018-11-02 16:08:26 +01003152
Michal Vasko004d3152020-06-11 19:59:22 +02003153 /* fill repeat */
aPiecekbf968d92021-05-27 14:35:05 +02003154 LY_CHECK_ERR_GOTO(reparse_or_expr(ctx, expr, &tok_idx, 0), ret = LY_EVALID, error);
Radek Krejcif03a9e22020-09-18 20:09:31 +02003155 if (expr->used > tok_idx) {
Radek Krejci2efc45b2020-12-22 16:25:44 +01003156 LOGVAL(ctx, LYVE_XPATH, "Unparsed characters \"%s\" left at the end of an XPath expression.",
Michal Vasko69730152020-10-09 16:30:07 +02003157 &expr->expr[expr->tok_pos[tok_idx]]);
Radek Krejcif03a9e22020-09-18 20:09:31 +02003158 ret = LY_EVALID;
Michal Vasko004d3152020-06-11 19:59:22 +02003159 goto error;
3160 }
Michal Vasko03ff5a72019-09-11 13:49:33 +02003161 }
3162
Radek Krejcif03a9e22020-09-18 20:09:31 +02003163 print_expr_struct_debug(expr);
3164 *expr_p = expr;
3165 return LY_SUCCESS;
Radek Krejcib1646a92018-11-02 16:08:26 +01003166
3167error:
Radek Krejcif03a9e22020-09-18 20:09:31 +02003168 lyxp_expr_free(ctx, expr);
3169 return ret;
Radek Krejcib1646a92018-11-02 16:08:26 +01003170}
3171
Michal Vasko1734be92020-09-22 08:55:10 +02003172LY_ERR
3173lyxp_expr_dup(const struct ly_ctx *ctx, const struct lyxp_expr *exp, struct lyxp_expr **dup_p)
Michal Vasko004d3152020-06-11 19:59:22 +02003174{
Michal Vasko1734be92020-09-22 08:55:10 +02003175 LY_ERR ret = LY_SUCCESS;
3176 struct lyxp_expr *dup = NULL;
Michal Vasko004d3152020-06-11 19:59:22 +02003177 uint32_t i, j;
3178
Michal Vasko7f45cf22020-10-01 12:49:44 +02003179 if (!exp) {
3180 goto cleanup;
3181 }
3182
Michal Vasko004d3152020-06-11 19:59:22 +02003183 dup = calloc(1, sizeof *dup);
Michal Vasko1734be92020-09-22 08:55:10 +02003184 LY_CHECK_ERR_GOTO(!dup, LOGMEM(ctx); ret = LY_EMEM, cleanup);
Michal Vasko004d3152020-06-11 19:59:22 +02003185
Michal Vasko08e9b112021-06-11 15:41:17 +02003186 if (exp->used) {
3187 dup->tokens = malloc(exp->used * sizeof *dup->tokens);
3188 LY_CHECK_ERR_GOTO(!dup->tokens, LOGMEM(ctx); ret = LY_EMEM, cleanup);
3189 memcpy(dup->tokens, exp->tokens, exp->used * sizeof *dup->tokens);
Michal Vasko004d3152020-06-11 19:59:22 +02003190
Michal Vasko08e9b112021-06-11 15:41:17 +02003191 dup->tok_pos = malloc(exp->used * sizeof *dup->tok_pos);
3192 LY_CHECK_ERR_GOTO(!dup->tok_pos, LOGMEM(ctx); ret = LY_EMEM, cleanup);
3193 memcpy(dup->tok_pos, exp->tok_pos, exp->used * sizeof *dup->tok_pos);
Michal Vasko004d3152020-06-11 19:59:22 +02003194
Michal Vasko08e9b112021-06-11 15:41:17 +02003195 dup->tok_len = malloc(exp->used * sizeof *dup->tok_len);
3196 LY_CHECK_ERR_GOTO(!dup->tok_len, LOGMEM(ctx); ret = LY_EMEM, cleanup);
3197 memcpy(dup->tok_len, exp->tok_len, exp->used * sizeof *dup->tok_len);
Michal Vasko004d3152020-06-11 19:59:22 +02003198
Michal Vasko79a7a872022-06-17 09:00:48 +02003199 if (exp->repeat) {
3200 dup->repeat = malloc(exp->used * sizeof *dup->repeat);
3201 LY_CHECK_ERR_GOTO(!dup->repeat, LOGMEM(ctx); ret = LY_EMEM, cleanup);
3202 for (i = 0; i < exp->used; ++i) {
3203 if (!exp->repeat[i]) {
3204 dup->repeat[i] = NULL;
3205 } else {
3206 for (j = 0; exp->repeat[i][j]; ++j) {}
3207 /* the ending 0 as well */
3208 ++j;
Michal Vasko004d3152020-06-11 19:59:22 +02003209
Michal Vasko79a7a872022-06-17 09:00:48 +02003210 dup->repeat[i] = malloc(j * sizeof **dup->repeat);
3211 LY_CHECK_ERR_GOTO(!dup->repeat[i], LOGMEM(ctx); ret = LY_EMEM, cleanup);
3212 memcpy(dup->repeat[i], exp->repeat[i], j * sizeof **dup->repeat);
3213 dup->repeat[i][j - 1] = 0;
3214 }
Michal Vasko08e9b112021-06-11 15:41:17 +02003215 }
Michal Vasko004d3152020-06-11 19:59:22 +02003216 }
3217 }
3218
3219 dup->used = exp->used;
3220 dup->size = exp->used;
Michal Vasko1734be92020-09-22 08:55:10 +02003221 LY_CHECK_GOTO(ret = lydict_insert(ctx, exp->expr, 0, &dup->expr), cleanup);
Michal Vasko004d3152020-06-11 19:59:22 +02003222
Michal Vasko1734be92020-09-22 08:55:10 +02003223cleanup:
3224 if (ret) {
3225 lyxp_expr_free(ctx, dup);
3226 } else {
3227 *dup_p = dup;
3228 }
3229 return ret;
Michal Vasko004d3152020-06-11 19:59:22 +02003230}
3231
Michal Vasko03ff5a72019-09-11 13:49:33 +02003232/**
3233 * @brief Get the last-added schema node that is currently in the context.
3234 *
3235 * @param[in] set Set to search in.
3236 * @return Last-added schema context node, NULL if no node is in context.
3237 */
3238static struct lysc_node *
3239warn_get_scnode_in_ctx(struct lyxp_set *set)
3240{
3241 uint32_t i;
3242
3243 if (!set || (set->type != LYXP_SET_SCNODE_SET)) {
3244 return NULL;
3245 }
3246
3247 i = set->used;
3248 do {
3249 --i;
Radek Krejcif13b87b2020-12-01 22:02:17 +01003250 if (set->val.scnodes[i].in_ctx == LYXP_SET_SCNODE_ATOM_CTX) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02003251 /* if there are more, simply return the first found (last added) */
3252 return set->val.scnodes[i].scnode;
3253 }
3254 } while (i);
3255
3256 return NULL;
3257}
3258
3259/**
3260 * @brief Test whether a type is numeric - integer type or decimal64.
3261 *
3262 * @param[in] type Type to test.
Radek Krejci857189e2020-09-01 13:26:36 +02003263 * @return Boolean value whether @p type is numeric type or not.
Michal Vasko03ff5a72019-09-11 13:49:33 +02003264 */
Radek Krejci857189e2020-09-01 13:26:36 +02003265static ly_bool
Michal Vasko03ff5a72019-09-11 13:49:33 +02003266warn_is_numeric_type(struct lysc_type *type)
3267{
3268 struct lysc_type_union *uni;
Radek Krejci857189e2020-09-01 13:26:36 +02003269 ly_bool ret;
Michal Vaskofd69e1d2020-07-03 11:57:17 +02003270 LY_ARRAY_COUNT_TYPE u;
Michal Vasko03ff5a72019-09-11 13:49:33 +02003271
3272 switch (type->basetype) {
3273 case LY_TYPE_DEC64:
3274 case LY_TYPE_INT8:
3275 case LY_TYPE_UINT8:
3276 case LY_TYPE_INT16:
3277 case LY_TYPE_UINT16:
3278 case LY_TYPE_INT32:
3279 case LY_TYPE_UINT32:
3280 case LY_TYPE_INT64:
3281 case LY_TYPE_UINT64:
3282 return 1;
3283 case LY_TYPE_UNION:
3284 uni = (struct lysc_type_union *)type;
Radek Krejci7eb54ba2020-05-18 16:30:04 +02003285 LY_ARRAY_FOR(uni->types, u) {
3286 ret = warn_is_numeric_type(uni->types[u]);
Michal Vasko03ff5a72019-09-11 13:49:33 +02003287 if (ret) {
3288 /* found a suitable type */
Radek Krejci857189e2020-09-01 13:26:36 +02003289 return ret;
Michal Vasko03ff5a72019-09-11 13:49:33 +02003290 }
3291 }
3292 /* did not find any suitable type */
3293 return 0;
3294 case LY_TYPE_LEAFREF:
3295 return warn_is_numeric_type(((struct lysc_type_leafref *)type)->realtype);
3296 default:
3297 return 0;
3298 }
3299}
3300
3301/**
3302 * @brief Test whether a type is string-like - no integers, decimal64 or binary.
3303 *
3304 * @param[in] type Type to test.
Radek Krejci857189e2020-09-01 13:26:36 +02003305 * @return Boolean value whether @p type's basetype is string type or not.
Michal Vasko03ff5a72019-09-11 13:49:33 +02003306 */
Radek Krejci857189e2020-09-01 13:26:36 +02003307static ly_bool
Michal Vasko03ff5a72019-09-11 13:49:33 +02003308warn_is_string_type(struct lysc_type *type)
3309{
3310 struct lysc_type_union *uni;
Radek Krejci857189e2020-09-01 13:26:36 +02003311 ly_bool ret;
Michal Vaskofd69e1d2020-07-03 11:57:17 +02003312 LY_ARRAY_COUNT_TYPE u;
Michal Vasko03ff5a72019-09-11 13:49:33 +02003313
3314 switch (type->basetype) {
3315 case LY_TYPE_BITS:
3316 case LY_TYPE_ENUM:
3317 case LY_TYPE_IDENT:
3318 case LY_TYPE_INST:
3319 case LY_TYPE_STRING:
3320 return 1;
3321 case LY_TYPE_UNION:
3322 uni = (struct lysc_type_union *)type;
Radek Krejci7eb54ba2020-05-18 16:30:04 +02003323 LY_ARRAY_FOR(uni->types, u) {
3324 ret = warn_is_string_type(uni->types[u]);
Michal Vasko03ff5a72019-09-11 13:49:33 +02003325 if (ret) {
3326 /* found a suitable type */
Radek Krejci857189e2020-09-01 13:26:36 +02003327 return ret;
Michal Vasko03ff5a72019-09-11 13:49:33 +02003328 }
3329 }
3330 /* did not find any suitable type */
3331 return 0;
3332 case LY_TYPE_LEAFREF:
3333 return warn_is_string_type(((struct lysc_type_leafref *)type)->realtype);
3334 default:
3335 return 0;
3336 }
3337}
3338
3339/**
3340 * @brief Test whether a type is one specific type.
3341 *
3342 * @param[in] type Type to test.
3343 * @param[in] base Expected type.
Radek Krejci857189e2020-09-01 13:26:36 +02003344 * @return Boolean value whether the given @p type is of the specific basetype @p base.
Michal Vasko03ff5a72019-09-11 13:49:33 +02003345 */
Radek Krejci857189e2020-09-01 13:26:36 +02003346static ly_bool
Michal Vasko03ff5a72019-09-11 13:49:33 +02003347warn_is_specific_type(struct lysc_type *type, LY_DATA_TYPE base)
3348{
3349 struct lysc_type_union *uni;
Radek Krejci857189e2020-09-01 13:26:36 +02003350 ly_bool ret;
Michal Vaskofd69e1d2020-07-03 11:57:17 +02003351 LY_ARRAY_COUNT_TYPE u;
Michal Vasko03ff5a72019-09-11 13:49:33 +02003352
3353 if (type->basetype == base) {
3354 return 1;
3355 } else if (type->basetype == LY_TYPE_UNION) {
3356 uni = (struct lysc_type_union *)type;
Radek Krejci7eb54ba2020-05-18 16:30:04 +02003357 LY_ARRAY_FOR(uni->types, u) {
3358 ret = warn_is_specific_type(uni->types[u], base);
Michal Vasko03ff5a72019-09-11 13:49:33 +02003359 if (ret) {
3360 /* found a suitable type */
Radek Krejci857189e2020-09-01 13:26:36 +02003361 return ret;
Michal Vasko03ff5a72019-09-11 13:49:33 +02003362 }
3363 }
3364 /* did not find any suitable type */
3365 return 0;
3366 } else if (type->basetype == LY_TYPE_LEAFREF) {
3367 return warn_is_specific_type(((struct lysc_type_leafref *)type)->realtype, base);
3368 }
3369
3370 return 0;
3371}
3372
3373/**
3374 * @brief Get next type of a (union) type.
3375 *
3376 * @param[in] type Base type.
3377 * @param[in] prev_type Previously returned type.
3378 * @return Next type or NULL.
3379 */
3380static struct lysc_type *
3381warn_is_equal_type_next_type(struct lysc_type *type, struct lysc_type *prev_type)
3382{
3383 struct lysc_type_union *uni;
Radek Krejci857189e2020-09-01 13:26:36 +02003384 ly_bool found = 0;
Michal Vaskofd69e1d2020-07-03 11:57:17 +02003385 LY_ARRAY_COUNT_TYPE u;
Michal Vasko03ff5a72019-09-11 13:49:33 +02003386
3387 switch (type->basetype) {
3388 case LY_TYPE_UNION:
3389 uni = (struct lysc_type_union *)type;
3390 if (!prev_type) {
3391 return uni->types[0];
3392 }
Radek Krejci7eb54ba2020-05-18 16:30:04 +02003393 LY_ARRAY_FOR(uni->types, u) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02003394 if (found) {
Radek Krejci7eb54ba2020-05-18 16:30:04 +02003395 return uni->types[u];
Michal Vasko03ff5a72019-09-11 13:49:33 +02003396 }
Radek Krejci7eb54ba2020-05-18 16:30:04 +02003397 if (prev_type == uni->types[u]) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02003398 found = 1;
3399 }
3400 }
3401 return NULL;
3402 default:
3403 if (prev_type) {
3404 assert(type == prev_type);
3405 return NULL;
3406 } else {
3407 return type;
3408 }
3409 }
3410}
3411
3412/**
3413 * @brief Test whether 2 types have a common type.
3414 *
3415 * @param[in] type1 First type.
3416 * @param[in] type2 Second type.
3417 * @return 1 if they do, 0 otherwise.
3418 */
3419static int
3420warn_is_equal_type(struct lysc_type *type1, struct lysc_type *type2)
3421{
3422 struct lysc_type *t1, *rt1, *t2, *rt2;
3423
3424 t1 = NULL;
3425 while ((t1 = warn_is_equal_type_next_type(type1, t1))) {
3426 if (t1->basetype == LY_TYPE_LEAFREF) {
3427 rt1 = ((struct lysc_type_leafref *)t1)->realtype;
3428 } else {
3429 rt1 = t1;
3430 }
3431
3432 t2 = NULL;
3433 while ((t2 = warn_is_equal_type_next_type(type2, t2))) {
3434 if (t2->basetype == LY_TYPE_LEAFREF) {
3435 rt2 = ((struct lysc_type_leafref *)t2)->realtype;
3436 } else {
3437 rt2 = t2;
3438 }
3439
3440 if (rt2->basetype == rt1->basetype) {
3441 /* match found */
3442 return 1;
3443 }
3444 }
3445 }
3446
3447 return 0;
3448}
3449
3450/**
Michal Vaskoaa956522021-11-11 10:45:34 +01003451 * @brief Print warning with information about the XPath subexpression that caused previous warning.
3452 *
3453 * @param[in] ctx Context for logging.
3454 * @param[in] tok_pos Index of the subexpression in the whole expression.
3455 * @param[in] subexpr Subexpression start.
3456 * @param[in] subexpr_len Length of @p subexpr to print.
3457 * @param[in] cur_scnode Expression context node.
3458 */
3459static void
3460warn_subexpr_log(const struct ly_ctx *ctx, uint16_t tok_pos, const char *subexpr, int subexpr_len,
3461 const struct lysc_node *cur_scnode)
3462{
3463 char *path;
3464
3465 path = lysc_path(cur_scnode, LYSC_PATH_LOG, NULL, 0);
3466 LOGWRN(ctx, "Previous warning generated by XPath subexpression[%" PRIu16 "] \"%.*s\" with context node \"%s\".",
3467 tok_pos, subexpr_len, subexpr, path);
3468 free(path);
3469}
3470
3471/**
Michal Vasko03ff5a72019-09-11 13:49:33 +02003472 * @brief Check both operands of comparison operators.
3473 *
3474 * @param[in] ctx Context for errors.
3475 * @param[in] set1 First operand set.
3476 * @param[in] set2 Second operand set.
3477 * @param[in] numbers_only Whether accept only numbers or other types are fine too (for '=' and '!=').
3478 * @param[in] expr Start of the expression to print with the warning.
3479 * @param[in] tok_pos Token position.
3480 */
3481static void
Michal Vaskoaa956522021-11-11 10:45:34 +01003482warn_operands(struct ly_ctx *ctx, struct lyxp_set *set1, struct lyxp_set *set2, ly_bool numbers_only, const char *expr,
3483 uint16_t tok_pos)
Michal Vasko03ff5a72019-09-11 13:49:33 +02003484{
3485 struct lysc_node_leaf *node1, *node2;
Radek Krejci857189e2020-09-01 13:26:36 +02003486 ly_bool leaves = 1, warning = 0;
Michal Vasko03ff5a72019-09-11 13:49:33 +02003487
3488 node1 = (struct lysc_node_leaf *)warn_get_scnode_in_ctx(set1);
3489 node2 = (struct lysc_node_leaf *)warn_get_scnode_in_ctx(set2);
3490
3491 if (!node1 && !node2) {
3492 /* no node-sets involved, nothing to do */
3493 return;
3494 }
3495
3496 if (node1) {
3497 if (!(node1->nodetype & (LYS_LEAF | LYS_LEAFLIST))) {
3498 LOGWRN(ctx, "Node type %s \"%s\" used as operand.", lys_nodetype2str(node1->nodetype), node1->name);
3499 warning = 1;
3500 leaves = 0;
3501 } else if (numbers_only && !warn_is_numeric_type(node1->type)) {
3502 LOGWRN(ctx, "Node \"%s\" is not of a numeric type, but used where it was expected.", node1->name);
3503 warning = 1;
3504 }
3505 }
3506
3507 if (node2) {
3508 if (!(node2->nodetype & (LYS_LEAF | LYS_LEAFLIST))) {
3509 LOGWRN(ctx, "Node type %s \"%s\" used as operand.", lys_nodetype2str(node2->nodetype), node2->name);
3510 warning = 1;
3511 leaves = 0;
3512 } else if (numbers_only && !warn_is_numeric_type(node2->type)) {
3513 LOGWRN(ctx, "Node \"%s\" is not of a numeric type, but used where it was expected.", node2->name);
3514 warning = 1;
3515 }
3516 }
3517
3518 if (node1 && node2 && leaves && !numbers_only) {
Michal Vasko69730152020-10-09 16:30:07 +02003519 if ((warn_is_numeric_type(node1->type) && !warn_is_numeric_type(node2->type)) ||
3520 (!warn_is_numeric_type(node1->type) && warn_is_numeric_type(node2->type)) ||
3521 (!warn_is_numeric_type(node1->type) && !warn_is_numeric_type(node2->type) &&
3522 !warn_is_equal_type(node1->type, node2->type))) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02003523 LOGWRN(ctx, "Incompatible types of operands \"%s\" and \"%s\" for comparison.", node1->name, node2->name);
3524 warning = 1;
3525 }
3526 }
3527
3528 if (warning) {
Michal Vaskoaa956522021-11-11 10:45:34 +01003529 warn_subexpr_log(ctx, tok_pos, expr + tok_pos, 20, set1->cur_scnode);
Michal Vasko03ff5a72019-09-11 13:49:33 +02003530 }
3531}
3532
3533/**
3534 * @brief Check that a value is valid for a leaf. If not applicable, does nothing.
3535 *
3536 * @param[in] exp Parsed XPath expression.
3537 * @param[in] set Set with the leaf/leaf-list.
3538 * @param[in] val_exp Index of the value (literal/number) in @p exp.
3539 * @param[in] equal_exp Index of the start of the equality expression in @p exp.
3540 * @param[in] last_equal_exp Index of the end of the equality expression in @p exp.
3541 */
3542static void
Michal Vasko40308e72020-10-20 16:38:40 +02003543warn_equality_value(const struct lyxp_expr *exp, struct lyxp_set *set, uint16_t val_exp, uint16_t equal_exp,
3544 uint16_t last_equal_exp)
Michal Vasko03ff5a72019-09-11 13:49:33 +02003545{
3546 struct lysc_node *scnode;
3547 struct lysc_type *type;
3548 char *value;
Michal Vaskoba99a3e2020-08-18 15:50:05 +02003549 struct lyd_value storage;
Michal Vasko03ff5a72019-09-11 13:49:33 +02003550 LY_ERR rc;
3551 struct ly_err_item *err = NULL;
3552
Michal Vasko69730152020-10-09 16:30:07 +02003553 if ((scnode = warn_get_scnode_in_ctx(set)) && (scnode->nodetype & (LYS_LEAF | LYS_LEAFLIST)) &&
3554 ((exp->tokens[val_exp] == LYXP_TOKEN_LITERAL) || (exp->tokens[val_exp] == LYXP_TOKEN_NUMBER))) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02003555 /* check that the node can have the specified value */
3556 if (exp->tokens[val_exp] == LYXP_TOKEN_LITERAL) {
3557 value = strndup(exp->expr + exp->tok_pos[val_exp] + 1, exp->tok_len[val_exp] - 2);
3558 } else {
3559 value = strndup(exp->expr + exp->tok_pos[val_exp], exp->tok_len[val_exp]);
3560 }
3561 if (!value) {
3562 LOGMEM(set->ctx);
3563 return;
3564 }
3565
3566 if ((((struct lysc_node_leaf *)scnode)->type->basetype == LY_TYPE_IDENT) && !strchr(value, ':')) {
3567 LOGWRN(set->ctx, "Identityref \"%s\" comparison with identity \"%s\" without prefix, consider adding"
Michal Vasko69730152020-10-09 16:30:07 +02003568 " a prefix or best using \"derived-from(-or-self)()\" functions.", scnode->name, value);
Michal Vaskoaa956522021-11-11 10:45:34 +01003569 warn_subexpr_log(set->ctx, exp->tok_pos[equal_exp], exp->expr + exp->tok_pos[equal_exp],
Radek Krejci0f969882020-08-21 16:56:47 +02003570 (exp->tok_pos[last_equal_exp] - exp->tok_pos[equal_exp]) + exp->tok_len[last_equal_exp],
Michal Vaskoaa956522021-11-11 10:45:34 +01003571 set->cur_scnode);
Michal Vasko03ff5a72019-09-11 13:49:33 +02003572 }
3573
3574 type = ((struct lysc_node_leaf *)scnode)->type;
3575 if (type->basetype != LY_TYPE_IDENT) {
Michal Vasko5d24f6c2020-10-13 13:49:06 +02003576 rc = type->plugin->store(set->ctx, type, value, strlen(value), 0, set->format, set->prefix_data,
Michal Vasko405cc9e2020-12-01 12:01:27 +01003577 LYD_HINT_DATA, scnode, &storage, NULL, &err);
Michal Vaskobf42e832020-11-23 16:59:42 +01003578 if (rc == LY_EINCOMPLETE) {
3579 rc = LY_SUCCESS;
3580 }
Michal Vasko03ff5a72019-09-11 13:49:33 +02003581
3582 if (err) {
3583 LOGWRN(set->ctx, "Invalid value \"%s\" which does not fit the type (%s).", value, err->msg);
3584 ly_err_free(err);
3585 } else if (rc != LY_SUCCESS) {
3586 LOGWRN(set->ctx, "Invalid value \"%s\" which does not fit the type.", value);
3587 }
3588 if (rc != LY_SUCCESS) {
Michal Vaskoaa956522021-11-11 10:45:34 +01003589 warn_subexpr_log(set->ctx, exp->tok_pos[equal_exp], exp->expr + exp->tok_pos[equal_exp],
Radek Krejci0f969882020-08-21 16:56:47 +02003590 (exp->tok_pos[last_equal_exp] - exp->tok_pos[equal_exp]) + exp->tok_len[last_equal_exp],
Michal Vaskoaa956522021-11-11 10:45:34 +01003591 set->cur_scnode);
Michal Vaskoba99a3e2020-08-18 15:50:05 +02003592 } else {
3593 type->plugin->free(set->ctx, &storage);
Michal Vasko03ff5a72019-09-11 13:49:33 +02003594 }
3595 }
3596 free(value);
3597 }
3598}
3599
3600/*
3601 * XPath functions
3602 */
3603
3604/**
3605 * @brief Execute the YANG 1.1 bit-is-set(node-set, string) function. Returns LYXP_SET_BOOLEAN
3606 * depending on whether the first node bit value from the second argument is set.
3607 *
3608 * @param[in] args Array of arguments.
3609 * @param[in] arg_count Count of elements in @p args.
3610 * @param[in,out] set Context and result set at the same time.
3611 * @param[in] options XPath options.
Michal Vasko0cbf54f2019-12-16 10:01:06 +01003612 * @return LY_ERR
Michal Vasko03ff5a72019-09-11 13:49:33 +02003613 */
3614static LY_ERR
Radek Krejci1deb5be2020-08-26 16:43:36 +02003615xpath_bit_is_set(struct lyxp_set **args, uint16_t UNUSED(arg_count), struct lyxp_set *set, uint32_t options)
Michal Vasko03ff5a72019-09-11 13:49:33 +02003616{
3617 struct lyd_node_term *leaf;
3618 struct lysc_node_leaf *sleaf;
Michal Vasko2588b952021-07-29 07:43:26 +02003619 struct lyd_value_bits *bits;
Michal Vasko03ff5a72019-09-11 13:49:33 +02003620 LY_ERR rc = LY_SUCCESS;
Michal Vaskofd69e1d2020-07-03 11:57:17 +02003621 LY_ARRAY_COUNT_TYPE u;
Michal Vasko03ff5a72019-09-11 13:49:33 +02003622
3623 if (options & LYXP_SCNODE_ALL) {
Michal Vasko5676f4e2021-04-06 17:14:45 +02003624 if (args[0]->type != LYXP_SET_SCNODE_SET) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02003625 LOGWRN(set->ctx, "Argument #1 of %s not a node-set as expected.", __func__);
Michal Vasko5676f4e2021-04-06 17:14:45 +02003626 } else if ((sleaf = (struct lysc_node_leaf *)warn_get_scnode_in_ctx(args[0]))) {
3627 if (!(sleaf->nodetype & (LYS_LEAF | LYS_LEAFLIST))) {
3628 LOGWRN(set->ctx, "Argument #1 of %s is a %s node \"%s\".", __func__, lys_nodetype2str(sleaf->nodetype),
3629 sleaf->name);
3630 } else if (!warn_is_specific_type(sleaf->type, LY_TYPE_BITS)) {
3631 LOGWRN(set->ctx, "Argument #1 of %s is node \"%s\", not of type \"bits\".", __func__, sleaf->name);
3632 }
Michal Vasko03ff5a72019-09-11 13:49:33 +02003633 }
3634
3635 if ((args[1]->type == LYXP_SET_SCNODE_SET) && (sleaf = (struct lysc_node_leaf *)warn_get_scnode_in_ctx(args[1]))) {
3636 if (!(sleaf->nodetype & (LYS_LEAF | LYS_LEAFLIST))) {
Michal Vasko5676f4e2021-04-06 17:14:45 +02003637 LOGWRN(set->ctx, "Argument #2 of %s is a %s node \"%s\".", __func__, lys_nodetype2str(sleaf->nodetype),
3638 sleaf->name);
Michal Vasko03ff5a72019-09-11 13:49:33 +02003639 } else if (!warn_is_string_type(sleaf->type)) {
3640 LOGWRN(set->ctx, "Argument #2 of %s is node \"%s\", not of string-type.", __func__, sleaf->name);
Michal Vasko03ff5a72019-09-11 13:49:33 +02003641 }
3642 }
Michal Vasko1a09b212021-05-06 13:00:10 +02003643 set_scnode_clear_ctx(set, LYXP_SET_SCNODE_ATOM_VAL);
Michal Vasko03ff5a72019-09-11 13:49:33 +02003644 return rc;
3645 }
3646
Michal Vaskod3678892020-05-21 10:06:58 +02003647 if (args[0]->type != LYXP_SET_NODE_SET) {
Radek Krejci2efc45b2020-12-22 16:25:44 +01003648 LOGVAL(set->ctx, LY_VCODE_XP_INARGTYPE, 1, print_set_type(args[0]), "bit-is-set(node-set, string)");
Michal Vasko03ff5a72019-09-11 13:49:33 +02003649 return LY_EVALID;
3650 }
Michal Vasko5e0e6eb2019-11-06 15:47:50 +01003651 rc = lyxp_set_cast(args[1], LYXP_SET_STRING);
Michal Vasko03ff5a72019-09-11 13:49:33 +02003652 LY_CHECK_RET(rc);
3653
3654 set_fill_boolean(set, 0);
Michal Vaskod3678892020-05-21 10:06:58 +02003655 if (args[0]->used) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02003656 leaf = (struct lyd_node_term *)args[0]->val.nodes[0].node;
Michal Vasko2588b952021-07-29 07:43:26 +02003657 if ((leaf->schema->nodetype & (LYS_LEAF | LYS_LEAFLIST)) && (leaf->value.realtype->basetype == LY_TYPE_BITS)) {
3658 LYD_VALUE_GET(&leaf->value, bits);
3659 LY_ARRAY_FOR(bits->items, u) {
3660 if (!strcmp(bits->items[u]->name, args[1]->val.str)) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02003661 set_fill_boolean(set, 1);
3662 break;
3663 }
3664 }
3665 }
3666 }
3667
3668 return LY_SUCCESS;
3669}
3670
3671/**
3672 * @brief Execute the XPath boolean(object) function. Returns LYXP_SET_BOOLEAN
3673 * with the argument converted to boolean.
3674 *
3675 * @param[in] args Array of arguments.
3676 * @param[in] arg_count Count of elements in @p args.
3677 * @param[in,out] set Context and result set at the same time.
3678 * @param[in] options XPath options.
Michal Vasko0cbf54f2019-12-16 10:01:06 +01003679 * @return LY_ERR
Michal Vasko03ff5a72019-09-11 13:49:33 +02003680 */
3681static LY_ERR
Radek Krejci1deb5be2020-08-26 16:43:36 +02003682xpath_boolean(struct lyxp_set **args, uint16_t UNUSED(arg_count), struct lyxp_set *set, uint32_t options)
Michal Vasko03ff5a72019-09-11 13:49:33 +02003683{
3684 LY_ERR rc;
3685
3686 if (options & LYXP_SCNODE_ALL) {
Michal Vasko1a09b212021-05-06 13:00:10 +02003687 set_scnode_clear_ctx(set, LYXP_SET_SCNODE_ATOM_NODE);
Michal Vasko03ff5a72019-09-11 13:49:33 +02003688 return LY_SUCCESS;
3689 }
3690
Michal Vasko5e0e6eb2019-11-06 15:47:50 +01003691 rc = lyxp_set_cast(args[0], LYXP_SET_BOOLEAN);
Michal Vasko03ff5a72019-09-11 13:49:33 +02003692 LY_CHECK_RET(rc);
3693 set_fill_set(set, args[0]);
3694
3695 return LY_SUCCESS;
3696}
3697
3698/**
3699 * @brief Execute the XPath ceiling(number) function. Returns LYXP_SET_NUMBER
3700 * with the first argument rounded up to the nearest integer.
3701 *
3702 * @param[in] args Array of arguments.
3703 * @param[in] arg_count Count of elements in @p args.
3704 * @param[in,out] set Context and result set at the same time.
3705 * @param[in] options XPath options.
Michal Vasko0cbf54f2019-12-16 10:01:06 +01003706 * @return LY_ERR
Michal Vasko03ff5a72019-09-11 13:49:33 +02003707 */
3708static LY_ERR
Radek Krejci1deb5be2020-08-26 16:43:36 +02003709xpath_ceiling(struct lyxp_set **args, uint16_t UNUSED(arg_count), struct lyxp_set *set, uint32_t options)
Michal Vasko03ff5a72019-09-11 13:49:33 +02003710{
3711 struct lysc_node_leaf *sleaf;
3712 LY_ERR rc = LY_SUCCESS;
3713
3714 if (options & LYXP_SCNODE_ALL) {
Michal Vasko5676f4e2021-04-06 17:14:45 +02003715 if (args[0]->type != LYXP_SET_SCNODE_SET) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02003716 LOGWRN(set->ctx, "Argument #1 of %s not a node-set as expected.", __func__);
Michal Vasko5676f4e2021-04-06 17:14:45 +02003717 } else if ((sleaf = (struct lysc_node_leaf *)warn_get_scnode_in_ctx(args[0]))) {
3718 if (!(sleaf->nodetype & (LYS_LEAF | LYS_LEAFLIST))) {
3719 LOGWRN(set->ctx, "Argument #1 of %s is a %s node \"%s\".", __func__, lys_nodetype2str(sleaf->nodetype),
3720 sleaf->name);
3721 } else if (!warn_is_specific_type(sleaf->type, LY_TYPE_DEC64)) {
3722 LOGWRN(set->ctx, "Argument #1 of %s is node \"%s\", not of type \"decimal64\".", __func__, sleaf->name);
3723 }
Michal Vasko03ff5a72019-09-11 13:49:33 +02003724 }
Michal Vasko1a09b212021-05-06 13:00:10 +02003725 set_scnode_clear_ctx(set, LYXP_SET_SCNODE_ATOM_VAL);
Michal Vasko03ff5a72019-09-11 13:49:33 +02003726 return rc;
3727 }
3728
Michal Vasko5e0e6eb2019-11-06 15:47:50 +01003729 rc = lyxp_set_cast(args[0], LYXP_SET_NUMBER);
Michal Vasko03ff5a72019-09-11 13:49:33 +02003730 LY_CHECK_RET(rc);
3731 if ((long long)args[0]->val.num != args[0]->val.num) {
3732 set_fill_number(set, ((long long)args[0]->val.num) + 1);
3733 } else {
3734 set_fill_number(set, args[0]->val.num);
3735 }
3736
3737 return LY_SUCCESS;
3738}
3739
3740/**
3741 * @brief Execute the XPath concat(string, string, string*) function.
3742 * Returns LYXP_SET_STRING with the concatenation of all the arguments.
3743 *
3744 * @param[in] args Array of arguments.
3745 * @param[in] arg_count Count of elements in @p args.
3746 * @param[in,out] set Context and result set at the same time.
3747 * @param[in] options XPath options.
Michal Vasko0cbf54f2019-12-16 10:01:06 +01003748 * @return LY_ERR
Michal Vasko03ff5a72019-09-11 13:49:33 +02003749 */
3750static LY_ERR
Radek Krejci1deb5be2020-08-26 16:43:36 +02003751xpath_concat(struct lyxp_set **args, uint16_t arg_count, struct lyxp_set *set, uint32_t options)
Michal Vasko03ff5a72019-09-11 13:49:33 +02003752{
3753 uint16_t i;
3754 char *str = NULL;
3755 size_t used = 1;
3756 LY_ERR rc = LY_SUCCESS;
3757 struct lysc_node_leaf *sleaf;
3758
3759 if (options & LYXP_SCNODE_ALL) {
3760 for (i = 0; i < arg_count; ++i) {
3761 if ((args[i]->type == LYXP_SET_SCNODE_SET) && (sleaf = (struct lysc_node_leaf *)warn_get_scnode_in_ctx(args[i]))) {
3762 if (!(sleaf->nodetype & (LYS_LEAF | LYS_LEAFLIST))) {
3763 LOGWRN(set->ctx, "Argument #%u of %s is a %s node \"%s\".",
Michal Vasko69730152020-10-09 16:30:07 +02003764 i + 1, __func__, lys_nodetype2str(sleaf->nodetype), sleaf->name);
Michal Vasko03ff5a72019-09-11 13:49:33 +02003765 } else if (!warn_is_string_type(sleaf->type)) {
Radek Krejci70124c82020-08-14 22:17:03 +02003766 LOGWRN(set->ctx, "Argument #%u of %s is node \"%s\", not of string-type.", i + 1, __func__, sleaf->name);
Michal Vasko03ff5a72019-09-11 13:49:33 +02003767 }
3768 }
3769 }
Michal Vasko1a09b212021-05-06 13:00:10 +02003770 set_scnode_clear_ctx(set, LYXP_SET_SCNODE_ATOM_VAL);
Michal Vasko03ff5a72019-09-11 13:49:33 +02003771 return rc;
3772 }
3773
3774 for (i = 0; i < arg_count; ++i) {
Michal Vasko5e0e6eb2019-11-06 15:47:50 +01003775 rc = lyxp_set_cast(args[i], LYXP_SET_STRING);
Michal Vasko03ff5a72019-09-11 13:49:33 +02003776 if (rc != LY_SUCCESS) {
3777 free(str);
3778 return rc;
3779 }
3780
3781 str = ly_realloc(str, (used + strlen(args[i]->val.str)) * sizeof(char));
3782 LY_CHECK_ERR_RET(!str, LOGMEM(set->ctx), LY_EMEM);
3783 strcpy(str + used - 1, args[i]->val.str);
3784 used += strlen(args[i]->val.str);
3785 }
3786
3787 /* free, kind of */
Michal Vaskod3678892020-05-21 10:06:58 +02003788 lyxp_set_free_content(set);
Michal Vasko03ff5a72019-09-11 13:49:33 +02003789 set->type = LYXP_SET_STRING;
3790 set->val.str = str;
3791
3792 return LY_SUCCESS;
3793}
3794
3795/**
3796 * @brief Execute the XPath contains(string, string) function.
3797 * Returns LYXP_SET_BOOLEAN whether the second argument can
3798 * be found in the first or not.
3799 *
3800 * @param[in] args Array of arguments.
3801 * @param[in] arg_count Count of elements in @p args.
3802 * @param[in,out] set Context and result set at the same time.
3803 * @param[in] options XPath options.
Michal Vasko0cbf54f2019-12-16 10:01:06 +01003804 * @return LY_ERR
Michal Vasko03ff5a72019-09-11 13:49:33 +02003805 */
3806static LY_ERR
Radek Krejci1deb5be2020-08-26 16:43:36 +02003807xpath_contains(struct lyxp_set **args, uint16_t UNUSED(arg_count), struct lyxp_set *set, uint32_t options)
Michal Vasko03ff5a72019-09-11 13:49:33 +02003808{
3809 struct lysc_node_leaf *sleaf;
3810 LY_ERR rc = LY_SUCCESS;
3811
3812 if (options & LYXP_SCNODE_ALL) {
3813 if ((args[0]->type == LYXP_SET_SCNODE_SET) && (sleaf = (struct lysc_node_leaf *)warn_get_scnode_in_ctx(args[0]))) {
3814 if (!(sleaf->nodetype & (LYS_LEAF | LYS_LEAFLIST))) {
Michal Vasko5676f4e2021-04-06 17:14:45 +02003815 LOGWRN(set->ctx, "Argument #1 of %s is a %s node \"%s\".", __func__, lys_nodetype2str(sleaf->nodetype),
3816 sleaf->name);
Michal Vasko03ff5a72019-09-11 13:49:33 +02003817 } else if (!warn_is_string_type(sleaf->type)) {
3818 LOGWRN(set->ctx, "Argument #1 of %s is node \"%s\", not of string-type.", __func__, sleaf->name);
Michal Vasko03ff5a72019-09-11 13:49:33 +02003819 }
3820 }
3821
3822 if ((args[1]->type == LYXP_SET_SCNODE_SET) && (sleaf = (struct lysc_node_leaf *)warn_get_scnode_in_ctx(args[1]))) {
3823 if (!(sleaf->nodetype & (LYS_LEAF | LYS_LEAFLIST))) {
Michal Vasko5676f4e2021-04-06 17:14:45 +02003824 LOGWRN(set->ctx, "Argument #2 of %s is a %s node \"%s\".", __func__, lys_nodetype2str(sleaf->nodetype),
3825 sleaf->name);
Michal Vasko03ff5a72019-09-11 13:49:33 +02003826 } else if (!warn_is_string_type(sleaf->type)) {
3827 LOGWRN(set->ctx, "Argument #2 of %s is node \"%s\", not of string-type.", __func__, sleaf->name);
Michal Vasko03ff5a72019-09-11 13:49:33 +02003828 }
3829 }
Michal Vasko1a09b212021-05-06 13:00:10 +02003830 set_scnode_clear_ctx(set, LYXP_SET_SCNODE_ATOM_VAL);
Michal Vasko03ff5a72019-09-11 13:49:33 +02003831 return rc;
3832 }
3833
Michal Vasko5e0e6eb2019-11-06 15:47:50 +01003834 rc = lyxp_set_cast(args[0], LYXP_SET_STRING);
Michal Vasko03ff5a72019-09-11 13:49:33 +02003835 LY_CHECK_RET(rc);
Michal Vasko5e0e6eb2019-11-06 15:47:50 +01003836 rc = lyxp_set_cast(args[1], LYXP_SET_STRING);
Michal Vasko03ff5a72019-09-11 13:49:33 +02003837 LY_CHECK_RET(rc);
3838
3839 if (strstr(args[0]->val.str, args[1]->val.str)) {
3840 set_fill_boolean(set, 1);
3841 } else {
3842 set_fill_boolean(set, 0);
3843 }
3844
3845 return LY_SUCCESS;
3846}
3847
3848/**
3849 * @brief Execute the XPath count(node-set) function. Returns LYXP_SET_NUMBER
3850 * with the size of the node-set from the argument.
3851 *
3852 * @param[in] args Array of arguments.
3853 * @param[in] arg_count Count of elements in @p args.
3854 * @param[in,out] set Context and result set at the same time.
3855 * @param[in] options XPath options.
Michal Vasko0cbf54f2019-12-16 10:01:06 +01003856 * @return LY_ERR
Michal Vasko03ff5a72019-09-11 13:49:33 +02003857 */
3858static LY_ERR
Radek Krejci1deb5be2020-08-26 16:43:36 +02003859xpath_count(struct lyxp_set **args, uint16_t UNUSED(arg_count), struct lyxp_set *set, uint32_t options)
Michal Vasko03ff5a72019-09-11 13:49:33 +02003860{
Michal Vasko03ff5a72019-09-11 13:49:33 +02003861 LY_ERR rc = LY_SUCCESS;
3862
3863 if (options & LYXP_SCNODE_ALL) {
Michal Vasko5676f4e2021-04-06 17:14:45 +02003864 if (args[0]->type != LYXP_SET_SCNODE_SET) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02003865 LOGWRN(set->ctx, "Argument #1 of %s not a node-set as expected.", __func__);
Michal Vasko03ff5a72019-09-11 13:49:33 +02003866 }
Michal Vasko1a09b212021-05-06 13:00:10 +02003867 set_scnode_clear_ctx(set, LYXP_SET_SCNODE_ATOM_NODE);
Michal Vasko03ff5a72019-09-11 13:49:33 +02003868 return rc;
3869 }
3870
Michal Vasko03ff5a72019-09-11 13:49:33 +02003871 if (args[0]->type != LYXP_SET_NODE_SET) {
Radek Krejci2efc45b2020-12-22 16:25:44 +01003872 LOGVAL(set->ctx, LY_VCODE_XP_INARGTYPE, 1, print_set_type(args[0]), "count(node-set)");
Michal Vasko03ff5a72019-09-11 13:49:33 +02003873 return LY_EVALID;
3874 }
3875
3876 set_fill_number(set, args[0]->used);
3877 return LY_SUCCESS;
3878}
3879
3880/**
3881 * @brief Execute the XPath current() function. Returns LYXP_SET_NODE_SET
3882 * with the context with the intial node.
3883 *
3884 * @param[in] args Array of arguments.
3885 * @param[in] arg_count Count of elements in @p args.
3886 * @param[in,out] set Context and result set at the same time.
3887 * @param[in] options XPath options.
Michal Vasko0cbf54f2019-12-16 10:01:06 +01003888 * @return LY_ERR
Michal Vasko03ff5a72019-09-11 13:49:33 +02003889 */
3890static LY_ERR
Radek Krejci1deb5be2020-08-26 16:43:36 +02003891xpath_current(struct lyxp_set **args, uint16_t arg_count, struct lyxp_set *set, uint32_t options)
Michal Vasko03ff5a72019-09-11 13:49:33 +02003892{
3893 if (arg_count || args) {
Radek Krejcie87c7dc2021-06-02 21:25:42 +02003894 LOGVAL(set->ctx, LY_VCODE_XP_INARGCOUNT, arg_count, LY_PRI_LENSTR("current()"));
Michal Vasko03ff5a72019-09-11 13:49:33 +02003895 return LY_EVALID;
3896 }
3897
3898 if (options & LYXP_SCNODE_ALL) {
Michal Vasko1a09b212021-05-06 13:00:10 +02003899 set_scnode_clear_ctx(set, LYXP_SET_SCNODE_ATOM_NODE);
Michal Vasko03ff5a72019-09-11 13:49:33 +02003900
Michal Vasko296dfaf2021-12-13 16:57:42 +01003901 if (set->cur_scnode) {
3902 LY_CHECK_RET(lyxp_set_scnode_insert_node(set, set->cur_scnode, LYXP_NODE_ELEM, NULL));
3903 } else {
3904 /* root node */
3905 LY_CHECK_RET(lyxp_set_scnode_insert_node(set, NULL, set->root_type, NULL));
3906 }
Michal Vasko03ff5a72019-09-11 13:49:33 +02003907 } else {
Michal Vaskod3678892020-05-21 10:06:58 +02003908 lyxp_set_free_content(set);
Michal Vasko03ff5a72019-09-11 13:49:33 +02003909
Michal Vasko296dfaf2021-12-13 16:57:42 +01003910 if (set->cur_node) {
3911 /* position is filled later */
3912 set_insert_node(set, set->cur_node, 0, LYXP_NODE_ELEM, 0);
3913 } else {
3914 /* root node */
3915 set_insert_node(set, NULL, 0, set->root_type, 0);
3916 }
Michal Vasko03ff5a72019-09-11 13:49:33 +02003917 }
3918
3919 return LY_SUCCESS;
3920}
3921
3922/**
3923 * @brief Execute the YANG 1.1 deref(node-set) function. Returns LYXP_SET_NODE_SET with either
3924 * leafref or instance-identifier target node(s).
3925 *
3926 * @param[in] args Array of arguments.
3927 * @param[in] arg_count Count of elements in @p args.
3928 * @param[in,out] set Context and result set at the same time.
3929 * @param[in] options XPath options.
Michal Vasko0cbf54f2019-12-16 10:01:06 +01003930 * @return LY_ERR
Michal Vasko03ff5a72019-09-11 13:49:33 +02003931 */
3932static LY_ERR
Radek Krejci1deb5be2020-08-26 16:43:36 +02003933xpath_deref(struct lyxp_set **args, uint16_t UNUSED(arg_count), struct lyxp_set *set, uint32_t options)
Michal Vasko03ff5a72019-09-11 13:49:33 +02003934{
3935 struct lyd_node_term *leaf;
Michal Vasko42e497c2020-01-06 08:38:25 +01003936 struct lysc_node_leaf *sleaf = NULL;
Michal Vasko004d3152020-06-11 19:59:22 +02003937 struct lysc_type_leafref *lref;
Michal Vaskoae9e4cb2019-09-25 08:43:05 +02003938 const struct lysc_node *target;
Michal Vasko004d3152020-06-11 19:59:22 +02003939 struct ly_path *p;
3940 struct lyd_node *node;
Michal Vasko03ff5a72019-09-11 13:49:33 +02003941 char *errmsg = NULL;
Michal Vasko00cbf532020-06-15 13:58:47 +02003942 uint8_t oper;
Michal Vasko741bb562021-06-24 11:59:50 +02003943 LY_ERR r;
Michal Vasko03ff5a72019-09-11 13:49:33 +02003944
3945 if (options & LYXP_SCNODE_ALL) {
Michal Vasko5676f4e2021-04-06 17:14:45 +02003946 if (args[0]->type != LYXP_SET_SCNODE_SET) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02003947 LOGWRN(set->ctx, "Argument #1 of %s not a node-set as expected.", __func__);
Michal Vasko5676f4e2021-04-06 17:14:45 +02003948 } else if ((sleaf = (struct lysc_node_leaf *)warn_get_scnode_in_ctx(args[0]))) {
Michal Vasko423ba3f2021-07-19 13:08:50 +02003949 if (!(sleaf->nodetype & LYD_NODE_TERM)) {
Michal Vasko5676f4e2021-04-06 17:14:45 +02003950 LOGWRN(set->ctx, "Argument #1 of %s is a %s node \"%s\".", __func__, lys_nodetype2str(sleaf->nodetype),
3951 sleaf->name);
Michal Vaskoed725d72021-06-23 12:03:45 +02003952 } else if (!warn_is_specific_type(sleaf->type, LY_TYPE_LEAFREF) &&
3953 !warn_is_specific_type(sleaf->type, LY_TYPE_INST)) {
Michal Vasko5676f4e2021-04-06 17:14:45 +02003954 LOGWRN(set->ctx, "Argument #1 of %s is node \"%s\", not of type \"leafref\" nor \"instance-identifier\".",
3955 __func__, sleaf->name);
3956 }
Michal Vasko03ff5a72019-09-11 13:49:33 +02003957 }
Michal Vasko1a09b212021-05-06 13:00:10 +02003958 set_scnode_clear_ctx(set, LYXP_SET_SCNODE_ATOM_VAL);
Michal Vasko423ba3f2021-07-19 13:08:50 +02003959 if (sleaf && (sleaf->nodetype & LYD_NODE_TERM) && (sleaf->type->basetype == LY_TYPE_LEAFREF)) {
Michal Vasko004d3152020-06-11 19:59:22 +02003960 lref = (struct lysc_type_leafref *)sleaf->type;
Michal Vaskod1e53b92021-01-28 13:11:06 +01003961 oper = (sleaf->flags & LYS_IS_OUTPUT) ? LY_PATH_OPER_OUTPUT : LY_PATH_OPER_INPUT;
Michal Vasko004d3152020-06-11 19:59:22 +02003962
3963 /* it was already evaluated on schema, it must succeed */
Michal Vasko741bb562021-06-24 11:59:50 +02003964 r = ly_path_compile_leafref(set->ctx, &sleaf->node, NULL, lref->path, oper, LY_PATH_TARGET_MANY,
Michal Vasko24fc4d12021-07-12 14:41:20 +02003965 LY_VALUE_SCHEMA_RESOLVED, lref->prefixes, &p);
Michal Vasko741bb562021-06-24 11:59:50 +02003966 if (!r) {
3967 /* get the target node */
3968 target = p[LY_ARRAY_COUNT(p) - 1].node;
3969 ly_path_free(set->ctx, p);
Michal Vasko004d3152020-06-11 19:59:22 +02003970
Michal Vasko741bb562021-06-24 11:59:50 +02003971 LY_CHECK_RET(lyxp_set_scnode_insert_node(set, target, LYXP_NODE_ELEM, NULL));
3972 } /* else the target was found before but is disabled so it was removed */
Michal Vaskoae9e4cb2019-09-25 08:43:05 +02003973 }
3974
Michal Vasko741bb562021-06-24 11:59:50 +02003975 return LY_SUCCESS;
Michal Vasko03ff5a72019-09-11 13:49:33 +02003976 }
3977
Michal Vaskod3678892020-05-21 10:06:58 +02003978 if (args[0]->type != LYXP_SET_NODE_SET) {
Radek Krejci2efc45b2020-12-22 16:25:44 +01003979 LOGVAL(set->ctx, LY_VCODE_XP_INARGTYPE, 1, print_set_type(args[0]), "deref(node-set)");
Michal Vasko03ff5a72019-09-11 13:49:33 +02003980 return LY_EVALID;
3981 }
3982
Michal Vaskod3678892020-05-21 10:06:58 +02003983 lyxp_set_free_content(set);
3984 if (args[0]->used) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02003985 leaf = (struct lyd_node_term *)args[0]->val.nodes[0].node;
3986 sleaf = (struct lysc_node_leaf *)leaf->schema;
3987 if (sleaf->nodetype & (LYS_LEAF | LYS_LEAFLIST)) {
3988 if (sleaf->type->basetype == LY_TYPE_LEAFREF) {
3989 /* find leafref target */
Radek Krejci0b013302021-03-29 15:22:32 +02003990 if (lyplg_type_resolve_leafref((struct lysc_type_leafref *)sleaf->type, &leaf->node, &leaf->value, set->tree,
Michal Vasko9e685082021-01-29 14:49:09 +01003991 &node, &errmsg)) {
Michal Vasko004d3152020-06-11 19:59:22 +02003992 LOGERR(set->ctx, LY_EVALID, errmsg);
Michal Vasko03ff5a72019-09-11 13:49:33 +02003993 free(errmsg);
Michal Vasko004d3152020-06-11 19:59:22 +02003994 return LY_EVALID;
Michal Vasko03ff5a72019-09-11 13:49:33 +02003995 }
Michal Vasko03ff5a72019-09-11 13:49:33 +02003996 } else {
3997 assert(sleaf->type->basetype == LY_TYPE_INST);
Michal Vasko004d3152020-06-11 19:59:22 +02003998 if (ly_path_eval(leaf->value.target, set->tree, &node)) {
Michal Vaskoba99a3e2020-08-18 15:50:05 +02003999 LOGERR(set->ctx, LY_EVALID, "Invalid instance-identifier \"%s\" value - required instance not found.",
Radek Krejci6d5ba0c2021-04-26 07:49:59 +02004000 lyd_get_value(&leaf->node));
Michal Vasko03ff5a72019-09-11 13:49:33 +02004001 return LY_EVALID;
4002 }
4003 }
Michal Vasko004d3152020-06-11 19:59:22 +02004004
4005 /* insert it */
4006 set_insert_node(set, node, 0, LYXP_NODE_ELEM, 0);
Michal Vasko03ff5a72019-09-11 13:49:33 +02004007 }
4008 }
4009
4010 return LY_SUCCESS;
4011}
4012
Michal Vaskoe0dd59d2020-07-17 16:10:23 +02004013static LY_ERR
Radek Krejci857189e2020-09-01 13:26:36 +02004014xpath_derived_(struct lyxp_set **args, struct lyxp_set *set, uint32_t options, ly_bool self_match, const char *func)
Michal Vaskoe0dd59d2020-07-17 16:10:23 +02004015{
Michal Vasko1fdd8fa2021-01-08 09:21:45 +01004016 uint32_t i;
Michal Vaskoe0dd59d2020-07-17 16:10:23 +02004017 LY_ARRAY_COUNT_TYPE u;
4018 struct lyd_node_term *leaf;
4019 struct lysc_node_leaf *sleaf;
4020 struct lyd_meta *meta;
Michal Vasko93923692021-05-07 15:28:02 +02004021 struct lyd_value *val;
4022 const struct lys_module *mod;
4023 const char *id_name;
4024 uint16_t id_len;
4025 struct lysc_ident *id;
Michal Vaskoe0dd59d2020-07-17 16:10:23 +02004026 LY_ERR rc = LY_SUCCESS;
Radek Krejci857189e2020-09-01 13:26:36 +02004027 ly_bool found;
Michal Vaskoe0dd59d2020-07-17 16:10:23 +02004028
4029 if (options & LYXP_SCNODE_ALL) {
Michal Vasko5676f4e2021-04-06 17:14:45 +02004030 if (args[0]->type != LYXP_SET_SCNODE_SET) {
Michal Vaskoe0dd59d2020-07-17 16:10:23 +02004031 LOGWRN(set->ctx, "Argument #1 of %s not a node-set as expected.", func);
Michal Vasko5676f4e2021-04-06 17:14:45 +02004032 } else if ((sleaf = (struct lysc_node_leaf *)warn_get_scnode_in_ctx(args[0]))) {
4033 if (!(sleaf->nodetype & (LYS_LEAF | LYS_LEAFLIST))) {
4034 LOGWRN(set->ctx, "Argument #1 of %s is a %s node \"%s\".", func, lys_nodetype2str(sleaf->nodetype),
4035 sleaf->name);
4036 } else if (!warn_is_specific_type(sleaf->type, LY_TYPE_IDENT)) {
4037 LOGWRN(set->ctx, "Argument #1 of %s is node \"%s\", not of type \"identityref\".", func, sleaf->name);
4038 }
Michal Vaskoe0dd59d2020-07-17 16:10:23 +02004039 }
4040
4041 if ((args[1]->type == LYXP_SET_SCNODE_SET) && (sleaf = (struct lysc_node_leaf *)warn_get_scnode_in_ctx(args[1]))) {
4042 if (!(sleaf->nodetype & (LYS_LEAF | LYS_LEAFLIST))) {
4043 LOGWRN(set->ctx, "Argument #2 of %s is a %s node \"%s\".", func, lys_nodetype2str(sleaf->nodetype),
Michal Vasko69730152020-10-09 16:30:07 +02004044 sleaf->name);
Michal Vaskoe0dd59d2020-07-17 16:10:23 +02004045 } else if (!warn_is_string_type(sleaf->type)) {
4046 LOGWRN(set->ctx, "Argument #2 of %s is node \"%s\", not of string-type.", func, sleaf->name);
4047 }
4048 }
Michal Vasko1a09b212021-05-06 13:00:10 +02004049 set_scnode_clear_ctx(set, LYXP_SET_SCNODE_ATOM_VAL);
Michal Vaskoe0dd59d2020-07-17 16:10:23 +02004050 return rc;
4051 }
4052
4053 if (args[0]->type != LYXP_SET_NODE_SET) {
Radek Krejci2efc45b2020-12-22 16:25:44 +01004054 LOGVAL(set->ctx, LY_VCODE_XP_INARGTYPE, 1, print_set_type(args[0]), "derived-from(-or-self)(node-set, string)");
Michal Vaskoe0dd59d2020-07-17 16:10:23 +02004055 return LY_EVALID;
4056 }
4057 rc = lyxp_set_cast(args[1], LYXP_SET_STRING);
4058 LY_CHECK_RET(rc);
4059
Michal Vasko93923692021-05-07 15:28:02 +02004060 /* parse the identity */
4061 id_name = args[1]->val.str;
4062 id_len = strlen(id_name);
4063 rc = moveto_resolve_model(&id_name, &id_len, set, set->cur_node ? set->cur_node->schema : NULL, &mod);
4064 LY_CHECK_RET(rc);
4065 if (!mod) {
4066 LOGVAL(set->ctx, LYVE_XPATH, "Identity \"%.*s\" without a prefix.", (int)id_len, id_name);
4067 return LY_EVALID;
4068 }
4069
4070 /* find the identity */
4071 found = 0;
4072 LY_ARRAY_FOR(mod->identities, u) {
4073 if (!ly_strncmp(mod->identities[u].name, id_name, id_len)) {
4074 /* we have match */
4075 found = 1;
4076 break;
4077 }
4078 }
4079 if (!found) {
4080 LOGVAL(set->ctx, LYVE_XPATH, "Identity \"%.*s\" not found in module \"%s\".", (int)id_len, id_name, mod->name);
4081 return LY_EVALID;
4082 }
4083 id = &mod->identities[u];
4084
Michal Vaskoe0dd59d2020-07-17 16:10:23 +02004085 set_fill_boolean(set, 0);
4086 found = 0;
4087 for (i = 0; i < args[0]->used; ++i) {
4088 if ((args[0]->val.nodes[i].type != LYXP_NODE_ELEM) && (args[0]->val.nodes[i].type != LYXP_NODE_META)) {
4089 continue;
4090 }
4091
4092 if (args[0]->val.nodes[i].type == LYXP_NODE_ELEM) {
4093 leaf = (struct lyd_node_term *)args[0]->val.nodes[i].node;
4094 sleaf = (struct lysc_node_leaf *)leaf->schema;
4095 val = &leaf->value;
4096 if (!(sleaf->nodetype & LYD_NODE_TERM) || (leaf->value.realtype->basetype != LY_TYPE_IDENT)) {
4097 /* uninteresting */
4098 continue;
4099 }
Michal Vaskoe0dd59d2020-07-17 16:10:23 +02004100 } else {
4101 meta = args[0]->val.meta[i].meta;
4102 val = &meta->value;
4103 if (val->realtype->basetype != LY_TYPE_IDENT) {
4104 /* uninteresting */
4105 continue;
4106 }
Michal Vaskoe0dd59d2020-07-17 16:10:23 +02004107 }
4108
Michal Vasko93923692021-05-07 15:28:02 +02004109 /* check the identity itself */
4110 if (self_match && (id == val->ident)) {
Michal Vaskoe0dd59d2020-07-17 16:10:23 +02004111 set_fill_boolean(set, 1);
4112 found = 1;
4113 }
Michal Vasko93923692021-05-07 15:28:02 +02004114 if (!found && !lyplg_type_identity_isderived(id, val->ident)) {
4115 set_fill_boolean(set, 1);
4116 found = 1;
Michal Vaskoe0dd59d2020-07-17 16:10:23 +02004117 }
4118
Michal Vaskoe0dd59d2020-07-17 16:10:23 +02004119 if (found) {
4120 break;
4121 }
4122 }
4123
4124 return LY_SUCCESS;
4125}
4126
Michal Vasko03ff5a72019-09-11 13:49:33 +02004127/**
4128 * @brief Execute the YANG 1.1 derived-from(node-set, string) function. Returns LYXP_SET_BOOLEAN depending
4129 * on whether the first argument nodes contain a node of an identity derived from the second
4130 * argument identity.
4131 *
4132 * @param[in] args Array of arguments.
4133 * @param[in] arg_count Count of elements in @p args.
4134 * @param[in,out] set Context and result set at the same time.
4135 * @param[in] options XPath options.
Michal Vasko0cbf54f2019-12-16 10:01:06 +01004136 * @return LY_ERR
Michal Vasko03ff5a72019-09-11 13:49:33 +02004137 */
4138static LY_ERR
Radek Krejci1deb5be2020-08-26 16:43:36 +02004139xpath_derived_from(struct lyxp_set **args, uint16_t UNUSED(arg_count), struct lyxp_set *set, uint32_t options)
Michal Vasko03ff5a72019-09-11 13:49:33 +02004140{
Michal Vaskoe0dd59d2020-07-17 16:10:23 +02004141 return xpath_derived_(args, set, options, 0, __func__);
Michal Vasko03ff5a72019-09-11 13:49:33 +02004142}
4143
4144/**
4145 * @brief Execute the YANG 1.1 derived-from-or-self(node-set, string) function. Returns LYXP_SET_BOOLEAN depending
4146 * on whether the first argument nodes contain a node of an identity that either is or is derived from
4147 * the second argument identity.
4148 *
4149 * @param[in] args Array of arguments.
4150 * @param[in] arg_count Count of elements in @p args.
4151 * @param[in,out] set Context and result set at the same time.
4152 * @param[in] options XPath options.
Michal Vasko0cbf54f2019-12-16 10:01:06 +01004153 * @return LY_ERR
Michal Vasko03ff5a72019-09-11 13:49:33 +02004154 */
4155static LY_ERR
Radek Krejci1deb5be2020-08-26 16:43:36 +02004156xpath_derived_from_or_self(struct lyxp_set **args, uint16_t UNUSED(arg_count), struct lyxp_set *set, uint32_t options)
Michal Vasko03ff5a72019-09-11 13:49:33 +02004157{
Michal Vaskoe0dd59d2020-07-17 16:10:23 +02004158 return xpath_derived_(args, set, options, 1, __func__);
Michal Vasko03ff5a72019-09-11 13:49:33 +02004159}
4160
4161/**
4162 * @brief Execute the YANG 1.1 enum-value(node-set) function. Returns LYXP_SET_NUMBER
4163 * with the integer value of the first node's enum value, otherwise NaN.
4164 *
4165 * @param[in] args Array of arguments.
4166 * @param[in] arg_count Count of elements in @p args.
4167 * @param[in,out] set Context and result set at the same time.
4168 * @param[in] options XPath options.
Michal Vasko0cbf54f2019-12-16 10:01:06 +01004169 * @return LY_ERR
Michal Vasko03ff5a72019-09-11 13:49:33 +02004170 */
4171static LY_ERR
Radek Krejci1deb5be2020-08-26 16:43:36 +02004172xpath_enum_value(struct lyxp_set **args, uint16_t UNUSED(arg_count), struct lyxp_set *set, uint32_t options)
Michal Vasko03ff5a72019-09-11 13:49:33 +02004173{
4174 struct lyd_node_term *leaf;
4175 struct lysc_node_leaf *sleaf;
4176 LY_ERR rc = LY_SUCCESS;
4177
4178 if (options & LYXP_SCNODE_ALL) {
Michal Vasko5676f4e2021-04-06 17:14:45 +02004179 if (args[0]->type != LYXP_SET_SCNODE_SET) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02004180 LOGWRN(set->ctx, "Argument #1 of %s not a node-set as expected.", __func__);
Michal Vasko5676f4e2021-04-06 17:14:45 +02004181 } else if ((sleaf = (struct lysc_node_leaf *)warn_get_scnode_in_ctx(args[0]))) {
4182 if (!(sleaf->nodetype & (LYS_LEAF | LYS_LEAFLIST))) {
4183 LOGWRN(set->ctx, "Argument #1 of %s is a %s node \"%s\".", __func__, lys_nodetype2str(sleaf->nodetype),
4184 sleaf->name);
4185 } else if (!warn_is_specific_type(sleaf->type, LY_TYPE_ENUM)) {
4186 LOGWRN(set->ctx, "Argument #1 of %s is node \"%s\", not of type \"enumeration\".", __func__, sleaf->name);
4187 }
Michal Vasko03ff5a72019-09-11 13:49:33 +02004188 }
Michal Vasko1a09b212021-05-06 13:00:10 +02004189 set_scnode_clear_ctx(set, LYXP_SET_SCNODE_ATOM_VAL);
Michal Vasko03ff5a72019-09-11 13:49:33 +02004190 return rc;
4191 }
4192
Michal Vaskod3678892020-05-21 10:06:58 +02004193 if (args[0]->type != LYXP_SET_NODE_SET) {
Radek Krejci2efc45b2020-12-22 16:25:44 +01004194 LOGVAL(set->ctx, LY_VCODE_XP_INARGTYPE, 1, print_set_type(args[0]), "enum-value(node-set)");
Michal Vasko03ff5a72019-09-11 13:49:33 +02004195 return LY_EVALID;
4196 }
4197
4198 set_fill_number(set, NAN);
Michal Vaskod3678892020-05-21 10:06:58 +02004199 if (args[0]->used) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02004200 leaf = (struct lyd_node_term *)args[0]->val.nodes[0].node;
4201 sleaf = (struct lysc_node_leaf *)leaf->schema;
4202 if ((sleaf->nodetype & (LYS_LEAF | LYS_LEAFLIST)) && (sleaf->type->basetype == LY_TYPE_ENUM)) {
4203 set_fill_number(set, leaf->value.enum_item->value);
4204 }
4205 }
4206
4207 return LY_SUCCESS;
4208}
4209
4210/**
4211 * @brief Execute the XPath false() function. Returns LYXP_SET_BOOLEAN
4212 * with false value.
4213 *
4214 * @param[in] args Array of arguments.
4215 * @param[in] arg_count Count of elements in @p args.
4216 * @param[in,out] set Context and result set at the same time.
4217 * @param[in] options XPath options.
Michal Vasko0cbf54f2019-12-16 10:01:06 +01004218 * @return LY_ERR
Michal Vasko03ff5a72019-09-11 13:49:33 +02004219 */
4220static LY_ERR
Radek Krejci1deb5be2020-08-26 16:43:36 +02004221xpath_false(struct lyxp_set **UNUSED(args), uint16_t UNUSED(arg_count), struct lyxp_set *set, uint32_t options)
Michal Vasko03ff5a72019-09-11 13:49:33 +02004222{
4223 if (options & LYXP_SCNODE_ALL) {
Michal Vasko1a09b212021-05-06 13:00:10 +02004224 set_scnode_clear_ctx(set, LYXP_SET_SCNODE_ATOM_NODE);
Michal Vasko03ff5a72019-09-11 13:49:33 +02004225 return LY_SUCCESS;
4226 }
4227
4228 set_fill_boolean(set, 0);
4229 return LY_SUCCESS;
4230}
4231
4232/**
4233 * @brief Execute the XPath floor(number) function. Returns LYXP_SET_NUMBER
4234 * with the first argument floored (truncated).
4235 *
4236 * @param[in] args Array of arguments.
4237 * @param[in] arg_count Count of elements in @p args.
4238 * @param[in,out] set Context and result set at the same time.
4239 * @param[in] options XPath options.
Michal Vasko0cbf54f2019-12-16 10:01:06 +01004240 * @return LY_ERR
Michal Vasko03ff5a72019-09-11 13:49:33 +02004241 */
4242static LY_ERR
Radek Krejci1deb5be2020-08-26 16:43:36 +02004243xpath_floor(struct lyxp_set **args, uint16_t UNUSED(arg_count), struct lyxp_set *set, uint32_t UNUSED(options))
Michal Vasko03ff5a72019-09-11 13:49:33 +02004244{
4245 LY_ERR rc;
4246
Michal Vasko5e0e6eb2019-11-06 15:47:50 +01004247 rc = lyxp_set_cast(args[0], LYXP_SET_NUMBER);
Michal Vasko03ff5a72019-09-11 13:49:33 +02004248 LY_CHECK_RET(rc);
4249 if (isfinite(args[0]->val.num)) {
4250 set_fill_number(set, (long long)args[0]->val.num);
4251 }
4252
4253 return LY_SUCCESS;
4254}
4255
4256/**
4257 * @brief Execute the XPath lang(string) function. Returns LYXP_SET_BOOLEAN
4258 * whether the language of the text matches the one from the argument.
4259 *
4260 * @param[in] args Array of arguments.
4261 * @param[in] arg_count Count of elements in @p args.
4262 * @param[in,out] set Context and result set at the same time.
4263 * @param[in] options XPath options.
Michal Vasko0cbf54f2019-12-16 10:01:06 +01004264 * @return LY_ERR
Michal Vasko03ff5a72019-09-11 13:49:33 +02004265 */
4266static LY_ERR
Radek Krejci1deb5be2020-08-26 16:43:36 +02004267xpath_lang(struct lyxp_set **args, uint16_t UNUSED(arg_count), struct lyxp_set *set, uint32_t options)
Michal Vasko03ff5a72019-09-11 13:49:33 +02004268{
4269 const struct lyd_node *node;
4270 struct lysc_node_leaf *sleaf;
Michal Vasko9f96a052020-03-10 09:41:45 +01004271 struct lyd_meta *meta = NULL;
Michal Vasko03ff5a72019-09-11 13:49:33 +02004272 const char *val;
Michal Vasko03ff5a72019-09-11 13:49:33 +02004273 LY_ERR rc = LY_SUCCESS;
4274
4275 if (options & LYXP_SCNODE_ALL) {
4276 if ((args[0]->type == LYXP_SET_SCNODE_SET) && (sleaf = (struct lysc_node_leaf *)warn_get_scnode_in_ctx(args[0]))) {
4277 if (!(sleaf->nodetype & (LYS_LEAF | LYS_LEAFLIST))) {
Michal Vasko1a09b212021-05-06 13:00:10 +02004278 LOGWRN(set->ctx, "Argument #1 of %s is a %s node \"%s\".", __func__, lys_nodetype2str(sleaf->nodetype),
4279 sleaf->name);
Michal Vasko03ff5a72019-09-11 13:49:33 +02004280 } else if (!warn_is_string_type(sleaf->type)) {
4281 LOGWRN(set->ctx, "Argument #1 of %s is node \"%s\", not of string-type.", __func__, sleaf->name);
Michal Vasko03ff5a72019-09-11 13:49:33 +02004282 }
4283 }
Michal Vasko1a09b212021-05-06 13:00:10 +02004284 set_scnode_clear_ctx(set, LYXP_SET_SCNODE_ATOM_VAL);
Michal Vasko03ff5a72019-09-11 13:49:33 +02004285 return rc;
4286 }
4287
Michal Vasko5e0e6eb2019-11-06 15:47:50 +01004288 rc = lyxp_set_cast(args[0], LYXP_SET_STRING);
Michal Vasko03ff5a72019-09-11 13:49:33 +02004289 LY_CHECK_RET(rc);
4290
Michal Vasko03ff5a72019-09-11 13:49:33 +02004291 if (set->type != LYXP_SET_NODE_SET) {
Radek Krejci2efc45b2020-12-22 16:25:44 +01004292 LOGVAL(set->ctx, LY_VCODE_XP_INCTX, print_set_type(set), "lang(string)");
Michal Vasko03ff5a72019-09-11 13:49:33 +02004293 return LY_EVALID;
Michal Vaskod3678892020-05-21 10:06:58 +02004294 } else if (!set->used) {
4295 set_fill_boolean(set, 0);
4296 return LY_SUCCESS;
Michal Vasko03ff5a72019-09-11 13:49:33 +02004297 }
4298
4299 switch (set->val.nodes[0].type) {
4300 case LYXP_NODE_ELEM:
4301 case LYXP_NODE_TEXT:
4302 node = set->val.nodes[0].node;
4303 break;
Michal Vasko9f96a052020-03-10 09:41:45 +01004304 case LYXP_NODE_META:
4305 node = set->val.meta[0].meta->parent;
Michal Vasko03ff5a72019-09-11 13:49:33 +02004306 break;
4307 default:
4308 /* nothing to do with roots */
4309 set_fill_boolean(set, 0);
4310 return LY_SUCCESS;
4311 }
4312
Michal Vasko9f96a052020-03-10 09:41:45 +01004313 /* find lang metadata */
Michal Vasko9e685082021-01-29 14:49:09 +01004314 for ( ; node; node = lyd_parent(node)) {
Michal Vasko9f96a052020-03-10 09:41:45 +01004315 for (meta = node->meta; meta; meta = meta->next) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02004316 /* annotations */
Michal Vasko9f96a052020-03-10 09:41:45 +01004317 if (meta->name && !strcmp(meta->name, "lang") && !strcmp(meta->annotation->module->name, "xml")) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02004318 break;
4319 }
4320 }
4321
Michal Vasko9f96a052020-03-10 09:41:45 +01004322 if (meta) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02004323 break;
4324 }
4325 }
4326
4327 /* compare languages */
Michal Vasko9f96a052020-03-10 09:41:45 +01004328 if (!meta) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02004329 set_fill_boolean(set, 0);
4330 } else {
Radek Krejci1deb5be2020-08-26 16:43:36 +02004331 uint64_t i;
4332
Radek Krejci6d5ba0c2021-04-26 07:49:59 +02004333 val = lyd_get_meta_value(meta);
Michal Vasko03ff5a72019-09-11 13:49:33 +02004334 for (i = 0; args[0]->val.str[i]; ++i) {
4335 if (tolower(args[0]->val.str[i]) != tolower(val[i])) {
4336 set_fill_boolean(set, 0);
4337 break;
4338 }
4339 }
4340 if (!args[0]->val.str[i]) {
4341 if (!val[i] || (val[i] == '-')) {
4342 set_fill_boolean(set, 1);
4343 } else {
4344 set_fill_boolean(set, 0);
4345 }
4346 }
Michal Vasko03ff5a72019-09-11 13:49:33 +02004347 }
4348
4349 return LY_SUCCESS;
4350}
4351
4352/**
4353 * @brief Execute the XPath last() function. Returns LYXP_SET_NUMBER
4354 * with the context size.
4355 *
4356 * @param[in] args Array of arguments.
4357 * @param[in] arg_count Count of elements in @p args.
4358 * @param[in,out] set Context and result set at the same time.
4359 * @param[in] options XPath options.
Michal Vasko0cbf54f2019-12-16 10:01:06 +01004360 * @return LY_ERR
Michal Vasko03ff5a72019-09-11 13:49:33 +02004361 */
4362static LY_ERR
Radek Krejci1deb5be2020-08-26 16:43:36 +02004363xpath_last(struct lyxp_set **UNUSED(args), uint16_t UNUSED(arg_count), struct lyxp_set *set, uint32_t options)
Michal Vasko03ff5a72019-09-11 13:49:33 +02004364{
4365 if (options & LYXP_SCNODE_ALL) {
Michal Vasko1a09b212021-05-06 13:00:10 +02004366 set_scnode_clear_ctx(set, LYXP_SET_SCNODE_ATOM_NODE);
Michal Vasko03ff5a72019-09-11 13:49:33 +02004367 return LY_SUCCESS;
4368 }
4369
Michal Vasko03ff5a72019-09-11 13:49:33 +02004370 if (set->type != LYXP_SET_NODE_SET) {
Radek Krejci2efc45b2020-12-22 16:25:44 +01004371 LOGVAL(set->ctx, LY_VCODE_XP_INCTX, print_set_type(set), "last()");
Michal Vasko03ff5a72019-09-11 13:49:33 +02004372 return LY_EVALID;
Michal Vaskod3678892020-05-21 10:06:58 +02004373 } else if (!set->used) {
4374 set_fill_number(set, 0);
4375 return LY_SUCCESS;
Michal Vasko03ff5a72019-09-11 13:49:33 +02004376 }
4377
4378 set_fill_number(set, set->ctx_size);
4379 return LY_SUCCESS;
4380}
4381
4382/**
4383 * @brief Execute the XPath local-name(node-set?) function. Returns LYXP_SET_STRING
4384 * with the node name without namespace from the argument or the context.
4385 *
4386 * @param[in] args Array of arguments.
4387 * @param[in] arg_count Count of elements in @p args.
4388 * @param[in,out] set Context and result set at the same time.
4389 * @param[in] options XPath options.
Michal Vasko0cbf54f2019-12-16 10:01:06 +01004390 * @return LY_ERR
Michal Vasko03ff5a72019-09-11 13:49:33 +02004391 */
4392static LY_ERR
Radek Krejci1deb5be2020-08-26 16:43:36 +02004393xpath_local_name(struct lyxp_set **args, uint16_t arg_count, struct lyxp_set *set, uint32_t options)
Michal Vasko03ff5a72019-09-11 13:49:33 +02004394{
4395 struct lyxp_set_node *item;
Michal Vasko69730152020-10-09 16:30:07 +02004396
Michal Vasko03ff5a72019-09-11 13:49:33 +02004397 /* suppress unused variable warning */
4398 (void)options;
4399
4400 if (options & LYXP_SCNODE_ALL) {
Michal Vasko1a09b212021-05-06 13:00:10 +02004401 set_scnode_clear_ctx(set, LYXP_SET_SCNODE_ATOM_NODE);
Michal Vasko03ff5a72019-09-11 13:49:33 +02004402 return LY_SUCCESS;
4403 }
4404
4405 if (arg_count) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02004406 if (args[0]->type != LYXP_SET_NODE_SET) {
Radek Krejci2efc45b2020-12-22 16:25:44 +01004407 LOGVAL(set->ctx, LY_VCODE_XP_INARGTYPE, 1, print_set_type(args[0]),
Michal Vasko5d24f6c2020-10-13 13:49:06 +02004408 "local-name(node-set?)");
Michal Vasko03ff5a72019-09-11 13:49:33 +02004409 return LY_EVALID;
Michal Vaskod3678892020-05-21 10:06:58 +02004410 } else if (!args[0]->used) {
4411 set_fill_string(set, "", 0);
4412 return LY_SUCCESS;
Michal Vasko03ff5a72019-09-11 13:49:33 +02004413 }
4414
4415 /* we need the set sorted, it affects the result */
Michal Vasko5e0e6eb2019-11-06 15:47:50 +01004416 assert(!set_sort(args[0]));
Michal Vasko03ff5a72019-09-11 13:49:33 +02004417
4418 item = &args[0]->val.nodes[0];
4419 } else {
Michal Vasko03ff5a72019-09-11 13:49:33 +02004420 if (set->type != LYXP_SET_NODE_SET) {
Radek Krejci2efc45b2020-12-22 16:25:44 +01004421 LOGVAL(set->ctx, LY_VCODE_XP_INCTX, print_set_type(set), "local-name(node-set?)");
Michal Vasko03ff5a72019-09-11 13:49:33 +02004422 return LY_EVALID;
Michal Vaskod3678892020-05-21 10:06:58 +02004423 } else if (!set->used) {
4424 set_fill_string(set, "", 0);
4425 return LY_SUCCESS;
Michal Vasko03ff5a72019-09-11 13:49:33 +02004426 }
4427
4428 /* we need the set sorted, it affects the result */
Michal Vasko5e0e6eb2019-11-06 15:47:50 +01004429 assert(!set_sort(set));
Michal Vasko03ff5a72019-09-11 13:49:33 +02004430
4431 item = &set->val.nodes[0];
4432 }
4433
4434 switch (item->type) {
Michal Vasko2caefc12019-11-14 16:07:56 +01004435 case LYXP_NODE_NONE:
4436 LOGINT_RET(set->ctx);
Michal Vasko03ff5a72019-09-11 13:49:33 +02004437 case LYXP_NODE_ROOT:
4438 case LYXP_NODE_ROOT_CONFIG:
4439 case LYXP_NODE_TEXT:
4440 set_fill_string(set, "", 0);
4441 break;
4442 case LYXP_NODE_ELEM:
4443 set_fill_string(set, item->node->schema->name, strlen(item->node->schema->name));
4444 break;
Michal Vasko9f96a052020-03-10 09:41:45 +01004445 case LYXP_NODE_META:
4446 set_fill_string(set, ((struct lyd_meta *)item->node)->name, strlen(((struct lyd_meta *)item->node)->name));
Michal Vasko03ff5a72019-09-11 13:49:33 +02004447 break;
4448 }
4449
4450 return LY_SUCCESS;
4451}
4452
4453/**
4454 * @brief Execute the XPath name(node-set?) function. Returns LYXP_SET_STRING
4455 * with the node name fully qualified (with namespace) from the argument or the context.
4456 *
4457 * @param[in] args Array of arguments.
4458 * @param[in] arg_count Count of elements in @p args.
4459 * @param[in,out] set Context and result set at the same time.
4460 * @param[in] options XPath options.
Michal Vasko0cbf54f2019-12-16 10:01:06 +01004461 * @return LY_ERR
Michal Vasko03ff5a72019-09-11 13:49:33 +02004462 */
4463static LY_ERR
Radek Krejci1deb5be2020-08-26 16:43:36 +02004464xpath_name(struct lyxp_set **args, uint16_t arg_count, struct lyxp_set *set, uint32_t options)
Michal Vasko03ff5a72019-09-11 13:49:33 +02004465{
4466 struct lyxp_set_node *item;
Michal Vaskoed4fcfe2020-07-08 10:38:56 +02004467 struct lys_module *mod = NULL;
Michal Vasko03ff5a72019-09-11 13:49:33 +02004468 char *str;
Michal Vaskoed4fcfe2020-07-08 10:38:56 +02004469 const char *name = NULL;
Michal Vasko03ff5a72019-09-11 13:49:33 +02004470
4471 if (options & LYXP_SCNODE_ALL) {
Michal Vasko1a09b212021-05-06 13:00:10 +02004472 set_scnode_clear_ctx(set, LYXP_SET_SCNODE_ATOM_NODE);
Michal Vasko03ff5a72019-09-11 13:49:33 +02004473 return LY_SUCCESS;
4474 }
4475
4476 if (arg_count) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02004477 if (args[0]->type != LYXP_SET_NODE_SET) {
Radek Krejci2efc45b2020-12-22 16:25:44 +01004478 LOGVAL(set->ctx, LY_VCODE_XP_INARGTYPE, 1, print_set_type(args[0]), "name(node-set?)");
Michal Vasko03ff5a72019-09-11 13:49:33 +02004479 return LY_EVALID;
Michal Vaskod3678892020-05-21 10:06:58 +02004480 } else if (!args[0]->used) {
4481 set_fill_string(set, "", 0);
4482 return LY_SUCCESS;
Michal Vasko03ff5a72019-09-11 13:49:33 +02004483 }
4484
4485 /* we need the set sorted, it affects the result */
Michal Vasko5e0e6eb2019-11-06 15:47:50 +01004486 assert(!set_sort(args[0]));
Michal Vasko03ff5a72019-09-11 13:49:33 +02004487
4488 item = &args[0]->val.nodes[0];
4489 } else {
Michal Vasko03ff5a72019-09-11 13:49:33 +02004490 if (set->type != LYXP_SET_NODE_SET) {
Radek Krejci2efc45b2020-12-22 16:25:44 +01004491 LOGVAL(set->ctx, LY_VCODE_XP_INCTX, print_set_type(set), "name(node-set?)");
Michal Vasko03ff5a72019-09-11 13:49:33 +02004492 return LY_EVALID;
Michal Vaskod3678892020-05-21 10:06:58 +02004493 } else if (!set->used) {
4494 set_fill_string(set, "", 0);
4495 return LY_SUCCESS;
Michal Vasko03ff5a72019-09-11 13:49:33 +02004496 }
4497
4498 /* we need the set sorted, it affects the result */
Michal Vasko5e0e6eb2019-11-06 15:47:50 +01004499 assert(!set_sort(set));
Michal Vasko03ff5a72019-09-11 13:49:33 +02004500
4501 item = &set->val.nodes[0];
4502 }
4503
4504 switch (item->type) {
Michal Vasko2caefc12019-11-14 16:07:56 +01004505 case LYXP_NODE_NONE:
4506 LOGINT_RET(set->ctx);
Michal Vasko03ff5a72019-09-11 13:49:33 +02004507 case LYXP_NODE_ROOT:
4508 case LYXP_NODE_ROOT_CONFIG:
4509 case LYXP_NODE_TEXT:
Michal Vaskoed4fcfe2020-07-08 10:38:56 +02004510 /* keep NULL */
Michal Vasko03ff5a72019-09-11 13:49:33 +02004511 break;
4512 case LYXP_NODE_ELEM:
4513 mod = item->node->schema->module;
4514 name = item->node->schema->name;
4515 break;
Michal Vasko9f96a052020-03-10 09:41:45 +01004516 case LYXP_NODE_META:
4517 mod = ((struct lyd_meta *)item->node)->annotation->module;
4518 name = ((struct lyd_meta *)item->node)->name;
Michal Vasko03ff5a72019-09-11 13:49:33 +02004519 break;
4520 }
4521
4522 if (mod && name) {
Michal Vasko5d24f6c2020-10-13 13:49:06 +02004523 int rc = asprintf(&str, "%s:%s", ly_get_prefix(mod, set->format, set->prefix_data), name);
Michal Vasko03ff5a72019-09-11 13:49:33 +02004524 LY_CHECK_ERR_RET(rc == -1, LOGMEM(set->ctx), LY_EMEM);
4525 set_fill_string(set, str, strlen(str));
4526 free(str);
4527 } else {
4528 set_fill_string(set, "", 0);
4529 }
4530
4531 return LY_SUCCESS;
4532}
4533
4534/**
4535 * @brief Execute the XPath namespace-uri(node-set?) function. Returns LYXP_SET_STRING
4536 * with the namespace of the node from the argument or the context.
4537 *
4538 * @param[in] args Array of arguments.
4539 * @param[in] arg_count Count of elements in @p args.
4540 * @param[in,out] set Context and result set at the same time.
4541 * @param[in] options XPath options.
4542 * @return LY_ERR (LY_EINVAL for wrong arguments on schema)
4543 */
4544static LY_ERR
Radek Krejci1deb5be2020-08-26 16:43:36 +02004545xpath_namespace_uri(struct lyxp_set **args, uint16_t arg_count, struct lyxp_set *set, uint32_t options)
Michal Vasko03ff5a72019-09-11 13:49:33 +02004546{
4547 struct lyxp_set_node *item;
4548 struct lys_module *mod;
Michal Vasko69730152020-10-09 16:30:07 +02004549
Michal Vasko03ff5a72019-09-11 13:49:33 +02004550 /* suppress unused variable warning */
4551 (void)options;
4552
4553 if (options & LYXP_SCNODE_ALL) {
Michal Vasko1a09b212021-05-06 13:00:10 +02004554 set_scnode_clear_ctx(set, LYXP_SET_SCNODE_ATOM_VAL);
Michal Vasko03ff5a72019-09-11 13:49:33 +02004555 return LY_SUCCESS;
4556 }
4557
4558 if (arg_count) {
Michal Vaskod3678892020-05-21 10:06:58 +02004559 if (args[0]->type != LYXP_SET_NODE_SET) {
Radek Krejci2efc45b2020-12-22 16:25:44 +01004560 LOGVAL(set->ctx, LY_VCODE_XP_INARGTYPE, 1, print_set_type(args[0]),
Michal Vasko69730152020-10-09 16:30:07 +02004561 "namespace-uri(node-set?)");
Michal Vaskod3678892020-05-21 10:06:58 +02004562 return LY_EVALID;
4563 } else if (!args[0]->used) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02004564 set_fill_string(set, "", 0);
4565 return LY_SUCCESS;
4566 }
Michal Vasko03ff5a72019-09-11 13:49:33 +02004567
4568 /* we need the set sorted, it affects the result */
Michal Vasko5e0e6eb2019-11-06 15:47:50 +01004569 assert(!set_sort(args[0]));
Michal Vasko03ff5a72019-09-11 13:49:33 +02004570
4571 item = &args[0]->val.nodes[0];
4572 } else {
Michal Vasko03ff5a72019-09-11 13:49:33 +02004573 if (set->type != LYXP_SET_NODE_SET) {
Radek Krejci2efc45b2020-12-22 16:25:44 +01004574 LOGVAL(set->ctx, LY_VCODE_XP_INCTX, print_set_type(set), "namespace-uri(node-set?)");
Michal Vasko03ff5a72019-09-11 13:49:33 +02004575 return LY_EVALID;
Michal Vaskod3678892020-05-21 10:06:58 +02004576 } else if (!set->used) {
4577 set_fill_string(set, "", 0);
4578 return LY_SUCCESS;
Michal Vasko03ff5a72019-09-11 13:49:33 +02004579 }
4580
4581 /* we need the set sorted, it affects the result */
Michal Vasko5e0e6eb2019-11-06 15:47:50 +01004582 assert(!set_sort(set));
Michal Vasko03ff5a72019-09-11 13:49:33 +02004583
4584 item = &set->val.nodes[0];
4585 }
4586
4587 switch (item->type) {
Michal Vasko2caefc12019-11-14 16:07:56 +01004588 case LYXP_NODE_NONE:
4589 LOGINT_RET(set->ctx);
Michal Vasko03ff5a72019-09-11 13:49:33 +02004590 case LYXP_NODE_ROOT:
4591 case LYXP_NODE_ROOT_CONFIG:
4592 case LYXP_NODE_TEXT:
4593 set_fill_string(set, "", 0);
4594 break;
4595 case LYXP_NODE_ELEM:
Michal Vasko9f96a052020-03-10 09:41:45 +01004596 case LYXP_NODE_META:
Michal Vasko03ff5a72019-09-11 13:49:33 +02004597 if (item->type == LYXP_NODE_ELEM) {
4598 mod = item->node->schema->module;
Michal Vasko9f96a052020-03-10 09:41:45 +01004599 } else { /* LYXP_NODE_META */
Michal Vasko03ff5a72019-09-11 13:49:33 +02004600 /* annotations */
Michal Vasko9f96a052020-03-10 09:41:45 +01004601 mod = ((struct lyd_meta *)item->node)->annotation->module;
Michal Vasko03ff5a72019-09-11 13:49:33 +02004602 }
4603
4604 set_fill_string(set, mod->ns, strlen(mod->ns));
4605 break;
4606 }
4607
4608 return LY_SUCCESS;
4609}
4610
4611/**
Michal Vasko03ff5a72019-09-11 13:49:33 +02004612 * @brief Execute the XPath normalize-space(string?) function. Returns LYXP_SET_STRING
4613 * with normalized value (no leading, trailing, double white spaces) of the node
4614 * from the argument or the context.
4615 *
4616 * @param[in] args Array of arguments.
4617 * @param[in] arg_count Count of elements in @p args.
4618 * @param[in,out] set Context and result set at the same time.
4619 * @param[in] options XPath options.
Michal Vasko0cbf54f2019-12-16 10:01:06 +01004620 * @return LY_ERR
Michal Vasko03ff5a72019-09-11 13:49:33 +02004621 */
4622static LY_ERR
Radek Krejci1deb5be2020-08-26 16:43:36 +02004623xpath_normalize_space(struct lyxp_set **args, uint16_t arg_count, struct lyxp_set *set, uint32_t options)
Michal Vasko03ff5a72019-09-11 13:49:33 +02004624{
4625 uint16_t i, new_used;
4626 char *new;
Radek Krejci857189e2020-09-01 13:26:36 +02004627 ly_bool have_spaces = 0, space_before = 0;
Michal Vasko03ff5a72019-09-11 13:49:33 +02004628 struct lysc_node_leaf *sleaf;
4629 LY_ERR rc = LY_SUCCESS;
4630
4631 if (options & LYXP_SCNODE_ALL) {
Michal Vasko1a09b212021-05-06 13:00:10 +02004632 if (arg_count && (args[0]->type == LYXP_SET_SCNODE_SET) &&
4633 (sleaf = (struct lysc_node_leaf *)warn_get_scnode_in_ctx(args[0]))) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02004634 if (!(sleaf->nodetype & (LYS_LEAF | LYS_LEAFLIST))) {
Michal Vasko1a09b212021-05-06 13:00:10 +02004635 LOGWRN(set->ctx, "Argument #1 of %s is a %s node \"%s\".", __func__, lys_nodetype2str(sleaf->nodetype),
4636 sleaf->name);
Michal Vasko03ff5a72019-09-11 13:49:33 +02004637 } else if (!warn_is_string_type(sleaf->type)) {
4638 LOGWRN(set->ctx, "Argument #1 of %s is node \"%s\", not of string-type.", __func__, sleaf->name);
Michal Vasko03ff5a72019-09-11 13:49:33 +02004639 }
4640 }
Michal Vasko1a09b212021-05-06 13:00:10 +02004641 set_scnode_clear_ctx(set, LYXP_SET_SCNODE_ATOM_VAL);
Michal Vasko03ff5a72019-09-11 13:49:33 +02004642 return rc;
4643 }
4644
4645 if (arg_count) {
4646 set_fill_set(set, args[0]);
4647 }
Michal Vasko5e0e6eb2019-11-06 15:47:50 +01004648 rc = lyxp_set_cast(set, LYXP_SET_STRING);
Michal Vasko03ff5a72019-09-11 13:49:33 +02004649 LY_CHECK_RET(rc);
4650
4651 /* is there any normalization necessary? */
4652 for (i = 0; set->val.str[i]; ++i) {
4653 if (is_xmlws(set->val.str[i])) {
4654 if ((i == 0) || space_before || (!set->val.str[i + 1])) {
4655 have_spaces = 1;
4656 break;
4657 }
4658 space_before = 1;
4659 } else {
4660 space_before = 0;
4661 }
4662 }
4663
4664 /* yep, there is */
4665 if (have_spaces) {
4666 /* it's enough, at least one character will go, makes space for ending '\0' */
4667 new = malloc(strlen(set->val.str) * sizeof(char));
4668 LY_CHECK_ERR_RET(!new, LOGMEM(set->ctx), LY_EMEM);
4669 new_used = 0;
4670
4671 space_before = 0;
4672 for (i = 0; set->val.str[i]; ++i) {
4673 if (is_xmlws(set->val.str[i])) {
4674 if ((i == 0) || space_before) {
4675 space_before = 1;
4676 continue;
4677 } else {
4678 space_before = 1;
4679 }
4680 } else {
4681 space_before = 0;
4682 }
4683
4684 new[new_used] = (space_before ? ' ' : set->val.str[i]);
4685 ++new_used;
4686 }
4687
4688 /* at worst there is one trailing space now */
4689 if (new_used && is_xmlws(new[new_used - 1])) {
4690 --new_used;
4691 }
4692
4693 new = ly_realloc(new, (new_used + 1) * sizeof(char));
4694 LY_CHECK_ERR_RET(!new, LOGMEM(set->ctx), LY_EMEM);
4695 new[new_used] = '\0';
4696
4697 free(set->val.str);
4698 set->val.str = new;
4699 }
4700
4701 return LY_SUCCESS;
4702}
4703
4704/**
4705 * @brief Execute the XPath not(boolean) function. Returns LYXP_SET_BOOLEAN
4706 * with the argument converted to boolean and logically inverted.
4707 *
4708 * @param[in] args Array of arguments.
4709 * @param[in] arg_count Count of elements in @p args.
4710 * @param[in,out] set Context and result set at the same time.
4711 * @param[in] options XPath options.
Michal Vasko0cbf54f2019-12-16 10:01:06 +01004712 * @return LY_ERR
Michal Vasko03ff5a72019-09-11 13:49:33 +02004713 */
4714static LY_ERR
Radek Krejci1deb5be2020-08-26 16:43:36 +02004715xpath_not(struct lyxp_set **args, uint16_t UNUSED(arg_count), struct lyxp_set *set, uint32_t options)
Michal Vasko03ff5a72019-09-11 13:49:33 +02004716{
4717 if (options & LYXP_SCNODE_ALL) {
Michal Vasko1a09b212021-05-06 13:00:10 +02004718 set_scnode_clear_ctx(set, LYXP_SET_SCNODE_ATOM_NODE);
Michal Vasko03ff5a72019-09-11 13:49:33 +02004719 return LY_SUCCESS;
4720 }
4721
Michal Vasko5e0e6eb2019-11-06 15:47:50 +01004722 lyxp_set_cast(args[0], LYXP_SET_BOOLEAN);
Michal Vasko004d3152020-06-11 19:59:22 +02004723 if (args[0]->val.bln) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02004724 set_fill_boolean(set, 0);
4725 } else {
4726 set_fill_boolean(set, 1);
4727 }
4728
4729 return LY_SUCCESS;
4730}
4731
4732/**
4733 * @brief Execute the XPath number(object?) function. Returns LYXP_SET_NUMBER
4734 * with the number representation of either the argument or the context.
4735 *
4736 * @param[in] args Array of arguments.
4737 * @param[in] arg_count Count of elements in @p args.
4738 * @param[in,out] set Context and result set at the same time.
4739 * @param[in] options XPath options.
Michal Vasko0cbf54f2019-12-16 10:01:06 +01004740 * @return LY_ERR
Michal Vasko03ff5a72019-09-11 13:49:33 +02004741 */
4742static LY_ERR
Radek Krejci1deb5be2020-08-26 16:43:36 +02004743xpath_number(struct lyxp_set **args, uint16_t arg_count, struct lyxp_set *set, uint32_t options)
Michal Vasko03ff5a72019-09-11 13:49:33 +02004744{
4745 LY_ERR rc;
4746
4747 if (options & LYXP_SCNODE_ALL) {
Michal Vasko1a09b212021-05-06 13:00:10 +02004748 set_scnode_clear_ctx(set, LYXP_SET_SCNODE_ATOM_VAL);
Michal Vasko03ff5a72019-09-11 13:49:33 +02004749 return LY_SUCCESS;
4750 }
4751
4752 if (arg_count) {
Michal Vasko5e0e6eb2019-11-06 15:47:50 +01004753 rc = lyxp_set_cast(args[0], LYXP_SET_NUMBER);
Michal Vasko03ff5a72019-09-11 13:49:33 +02004754 LY_CHECK_RET(rc);
4755 set_fill_set(set, args[0]);
4756 } else {
Michal Vasko5e0e6eb2019-11-06 15:47:50 +01004757 rc = lyxp_set_cast(set, LYXP_SET_NUMBER);
Michal Vasko03ff5a72019-09-11 13:49:33 +02004758 LY_CHECK_RET(rc);
4759 }
4760
4761 return LY_SUCCESS;
4762}
4763
4764/**
4765 * @brief Execute the XPath position() function. Returns LYXP_SET_NUMBER
4766 * with the context position.
4767 *
4768 * @param[in] args Array of arguments.
4769 * @param[in] arg_count Count of elements in @p args.
4770 * @param[in,out] set Context and result set at the same time.
4771 * @param[in] options XPath options.
Michal Vasko0cbf54f2019-12-16 10:01:06 +01004772 * @return LY_ERR
Michal Vasko03ff5a72019-09-11 13:49:33 +02004773 */
4774static LY_ERR
Radek Krejci1deb5be2020-08-26 16:43:36 +02004775xpath_position(struct lyxp_set **UNUSED(args), uint16_t UNUSED(arg_count), struct lyxp_set *set, uint32_t options)
Michal Vasko03ff5a72019-09-11 13:49:33 +02004776{
4777 if (options & LYXP_SCNODE_ALL) {
Michal Vasko1a09b212021-05-06 13:00:10 +02004778 set_scnode_clear_ctx(set, LYXP_SET_SCNODE_ATOM_NODE);
Michal Vasko03ff5a72019-09-11 13:49:33 +02004779 return LY_SUCCESS;
4780 }
4781
Michal Vasko03ff5a72019-09-11 13:49:33 +02004782 if (set->type != LYXP_SET_NODE_SET) {
Radek Krejci2efc45b2020-12-22 16:25:44 +01004783 LOGVAL(set->ctx, LY_VCODE_XP_INCTX, print_set_type(set), "position()");
Michal Vasko03ff5a72019-09-11 13:49:33 +02004784 return LY_EVALID;
Michal Vaskod3678892020-05-21 10:06:58 +02004785 } else if (!set->used) {
4786 set_fill_number(set, 0);
4787 return LY_SUCCESS;
Michal Vasko03ff5a72019-09-11 13:49:33 +02004788 }
4789
4790 set_fill_number(set, set->ctx_pos);
4791
4792 /* UNUSED in 'Release' build type */
4793 (void)options;
4794 return LY_SUCCESS;
4795}
4796
4797/**
4798 * @brief Execute the YANG 1.1 re-match(string, string) function. Returns LYXP_SET_BOOLEAN
4799 * depending on whether the second argument regex matches the first argument string. For details refer to
4800 * YANG 1.1 RFC section 10.2.1.
4801 *
4802 * @param[in] args Array of arguments.
4803 * @param[in] arg_count Count of elements in @p args.
4804 * @param[in,out] set Context and result set at the same time.
4805 * @param[in] options XPath options.
Michal Vasko0cbf54f2019-12-16 10:01:06 +01004806 * @return LY_ERR
Michal Vasko03ff5a72019-09-11 13:49:33 +02004807 */
4808static LY_ERR
Radek Krejci1deb5be2020-08-26 16:43:36 +02004809xpath_re_match(struct lyxp_set **args, uint16_t UNUSED(arg_count), struct lyxp_set *set, uint32_t options)
Michal Vasko03ff5a72019-09-11 13:49:33 +02004810{
4811 struct lysc_pattern **patterns = NULL, **pattern;
4812 struct lysc_node_leaf *sleaf;
Michal Vasko03ff5a72019-09-11 13:49:33 +02004813 LY_ERR rc = LY_SUCCESS;
4814 struct ly_err_item *err;
4815
4816 if (options & LYXP_SCNODE_ALL) {
4817 if ((args[0]->type == LYXP_SET_SCNODE_SET) && (sleaf = (struct lysc_node_leaf *)warn_get_scnode_in_ctx(args[0]))) {
4818 if (!(sleaf->nodetype & (LYS_LEAF | LYS_LEAFLIST))) {
4819 LOGWRN(set->ctx, "Argument #1 of %s is a %s node \"%s\".", __func__, lys_nodetype2str(sleaf->nodetype), sleaf->name);
Michal Vasko03ff5a72019-09-11 13:49:33 +02004820 } else if (!warn_is_string_type(sleaf->type)) {
4821 LOGWRN(set->ctx, "Argument #1 of %s is node \"%s\", not of string-type.", __func__, sleaf->name);
Michal Vasko03ff5a72019-09-11 13:49:33 +02004822 }
4823 }
4824
4825 if ((args[1]->type == LYXP_SET_SCNODE_SET) && (sleaf = (struct lysc_node_leaf *)warn_get_scnode_in_ctx(args[1]))) {
4826 if (!(sleaf->nodetype & (LYS_LEAF | LYS_LEAFLIST))) {
4827 LOGWRN(set->ctx, "Argument #2 of %s is a %s node \"%s\".", __func__, lys_nodetype2str(sleaf->nodetype), sleaf->name);
Michal Vasko03ff5a72019-09-11 13:49:33 +02004828 } else if (!warn_is_string_type(sleaf->type)) {
4829 LOGWRN(set->ctx, "Argument #2 of %s is node \"%s\", not of string-type.", __func__, sleaf->name);
Michal Vasko03ff5a72019-09-11 13:49:33 +02004830 }
4831 }
Michal Vasko1a09b212021-05-06 13:00:10 +02004832 set_scnode_clear_ctx(set, LYXP_SET_SCNODE_ATOM_VAL);
Michal Vasko03ff5a72019-09-11 13:49:33 +02004833 return rc;
4834 }
4835
Michal Vasko5e0e6eb2019-11-06 15:47:50 +01004836 rc = lyxp_set_cast(args[0], LYXP_SET_STRING);
Michal Vasko03ff5a72019-09-11 13:49:33 +02004837 LY_CHECK_RET(rc);
Michal Vasko5e0e6eb2019-11-06 15:47:50 +01004838 rc = lyxp_set_cast(args[1], LYXP_SET_STRING);
Michal Vasko03ff5a72019-09-11 13:49:33 +02004839 LY_CHECK_RET(rc);
4840
4841 LY_ARRAY_NEW_RET(set->ctx, patterns, pattern, LY_EMEM);
Radek Iša45802b52021-02-09 09:21:58 +01004842 *pattern = calloc(1, sizeof **pattern);
Radek Krejciddace2c2021-01-08 11:30:56 +01004843 LOG_LOCSET(NULL, set->cur_node, NULL, NULL);
Radek Krejci2efc45b2020-12-22 16:25:44 +01004844 rc = lys_compile_type_pattern_check(set->ctx, args[1]->val.str, &(*pattern)->code);
Michal Vasko4a7d4d62021-12-13 17:05:06 +01004845 if (set->cur_node) {
4846 LOG_LOCBACK(0, 1, 0, 0);
4847 }
Michal Vasko03ff5a72019-09-11 13:49:33 +02004848 if (rc != LY_SUCCESS) {
4849 LY_ARRAY_FREE(patterns);
4850 return rc;
4851 }
4852
Radek Krejci0b013302021-03-29 15:22:32 +02004853 rc = lyplg_type_validate_patterns(patterns, args[0]->val.str, strlen(args[0]->val.str), &err);
Michal Vasko03ff5a72019-09-11 13:49:33 +02004854 pcre2_code_free((*pattern)->code);
4855 free(*pattern);
4856 LY_ARRAY_FREE(patterns);
4857 if (rc && (rc != LY_EVALID)) {
Michal Vasko177d0ed2020-11-23 16:43:03 +01004858 ly_err_print(set->ctx, err);
Michal Vasko03ff5a72019-09-11 13:49:33 +02004859 ly_err_free(err);
4860 return rc;
4861 }
4862
4863 if (rc == LY_EVALID) {
4864 ly_err_free(err);
4865 set_fill_boolean(set, 0);
4866 } else {
4867 set_fill_boolean(set, 1);
4868 }
4869
4870 return LY_SUCCESS;
4871}
4872
4873/**
4874 * @brief Execute the XPath round(number) function. Returns LYXP_SET_NUMBER
4875 * with the rounded first argument. For details refer to
4876 * http://www.w3.org/TR/1999/REC-xpath-19991116/#function-round.
4877 *
4878 * @param[in] args Array of arguments.
4879 * @param[in] arg_count Count of elements in @p args.
4880 * @param[in,out] set Context and result set at the same time.
4881 * @param[in] options XPath options.
Michal Vasko0cbf54f2019-12-16 10:01:06 +01004882 * @return LY_ERR
Michal Vasko03ff5a72019-09-11 13:49:33 +02004883 */
4884static LY_ERR
Radek Krejci1deb5be2020-08-26 16:43:36 +02004885xpath_round(struct lyxp_set **args, uint16_t UNUSED(arg_count), struct lyxp_set *set, uint32_t options)
Michal Vasko03ff5a72019-09-11 13:49:33 +02004886{
4887 struct lysc_node_leaf *sleaf;
4888 LY_ERR rc = LY_SUCCESS;
4889
4890 if (options & LYXP_SCNODE_ALL) {
Michal Vasko5676f4e2021-04-06 17:14:45 +02004891 if (args[0]->type != LYXP_SET_SCNODE_SET) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02004892 LOGWRN(set->ctx, "Argument #1 of %s not a node-set as expected.", __func__);
Michal Vasko5676f4e2021-04-06 17:14:45 +02004893 } else if ((sleaf = (struct lysc_node_leaf *)warn_get_scnode_in_ctx(args[0]))) {
4894 if (!(sleaf->nodetype & (LYS_LEAF | LYS_LEAFLIST))) {
4895 LOGWRN(set->ctx, "Argument #1 of %s is a %s node \"%s\".", __func__, lys_nodetype2str(sleaf->nodetype),
4896 sleaf->name);
4897 } else if (!warn_is_specific_type(sleaf->type, LY_TYPE_DEC64)) {
4898 LOGWRN(set->ctx, "Argument #1 of %s is node \"%s\", not of type \"decimal64\".", __func__, sleaf->name);
4899 }
Michal Vasko03ff5a72019-09-11 13:49:33 +02004900 }
Michal Vasko1a09b212021-05-06 13:00:10 +02004901 set_scnode_clear_ctx(set, LYXP_SET_SCNODE_ATOM_VAL);
Michal Vasko03ff5a72019-09-11 13:49:33 +02004902 return rc;
4903 }
4904
Michal Vasko5e0e6eb2019-11-06 15:47:50 +01004905 rc = lyxp_set_cast(args[0], LYXP_SET_NUMBER);
Michal Vasko03ff5a72019-09-11 13:49:33 +02004906 LY_CHECK_RET(rc);
4907
4908 /* cover only the cases where floor can't be used */
4909 if ((args[0]->val.num == -0.0f) || ((args[0]->val.num < 0) && (args[0]->val.num >= -0.5))) {
4910 set_fill_number(set, -0.0f);
4911 } else {
4912 args[0]->val.num += 0.5;
4913 rc = xpath_floor(args, 1, args[0], options);
4914 LY_CHECK_RET(rc);
4915 set_fill_number(set, args[0]->val.num);
4916 }
4917
4918 return LY_SUCCESS;
4919}
4920
4921/**
4922 * @brief Execute the XPath starts-with(string, string) function.
4923 * Returns LYXP_SET_BOOLEAN whether the second argument is
4924 * the prefix of the first or not.
4925 *
4926 * @param[in] args Array of arguments.
4927 * @param[in] arg_count Count of elements in @p args.
4928 * @param[in,out] set Context and result set at the same time.
4929 * @param[in] options XPath options.
Michal Vasko0cbf54f2019-12-16 10:01:06 +01004930 * @return LY_ERR
Michal Vasko03ff5a72019-09-11 13:49:33 +02004931 */
4932static LY_ERR
Radek Krejci1deb5be2020-08-26 16:43:36 +02004933xpath_starts_with(struct lyxp_set **args, uint16_t UNUSED(arg_count), struct lyxp_set *set, uint32_t options)
Michal Vasko03ff5a72019-09-11 13:49:33 +02004934{
4935 struct lysc_node_leaf *sleaf;
4936 LY_ERR rc = LY_SUCCESS;
4937
4938 if (options & LYXP_SCNODE_ALL) {
4939 if ((args[0]->type == LYXP_SET_SCNODE_SET) && (sleaf = (struct lysc_node_leaf *)warn_get_scnode_in_ctx(args[0]))) {
4940 if (!(sleaf->nodetype & (LYS_LEAF | LYS_LEAFLIST))) {
4941 LOGWRN(set->ctx, "Argument #1 of %s is a %s node \"%s\".", __func__, lys_nodetype2str(sleaf->nodetype), sleaf->name);
Michal Vasko03ff5a72019-09-11 13:49:33 +02004942 } else if (!warn_is_string_type(sleaf->type)) {
4943 LOGWRN(set->ctx, "Argument #1 of %s is node \"%s\", not of string-type.", __func__, sleaf->name);
Michal Vasko03ff5a72019-09-11 13:49:33 +02004944 }
4945 }
4946
4947 if ((args[1]->type == LYXP_SET_SCNODE_SET) && (sleaf = (struct lysc_node_leaf *)warn_get_scnode_in_ctx(args[1]))) {
4948 if (!(sleaf->nodetype & (LYS_LEAF | LYS_LEAFLIST))) {
4949 LOGWRN(set->ctx, "Argument #2 of %s is a %s node \"%s\".", __func__, lys_nodetype2str(sleaf->nodetype), sleaf->name);
Michal Vasko03ff5a72019-09-11 13:49:33 +02004950 } else if (!warn_is_string_type(sleaf->type)) {
4951 LOGWRN(set->ctx, "Argument #2 of %s is node \"%s\", not of string-type.", __func__, sleaf->name);
Michal Vasko03ff5a72019-09-11 13:49:33 +02004952 }
4953 }
Michal Vasko1a09b212021-05-06 13:00:10 +02004954 set_scnode_clear_ctx(set, LYXP_SET_SCNODE_ATOM_VAL);
Michal Vasko03ff5a72019-09-11 13:49:33 +02004955 return rc;
4956 }
4957
Michal Vasko5e0e6eb2019-11-06 15:47:50 +01004958 rc = lyxp_set_cast(args[0], LYXP_SET_STRING);
Michal Vasko03ff5a72019-09-11 13:49:33 +02004959 LY_CHECK_RET(rc);
Michal Vasko5e0e6eb2019-11-06 15:47:50 +01004960 rc = lyxp_set_cast(args[1], LYXP_SET_STRING);
Michal Vasko03ff5a72019-09-11 13:49:33 +02004961 LY_CHECK_RET(rc);
4962
4963 if (strncmp(args[0]->val.str, args[1]->val.str, strlen(args[1]->val.str))) {
4964 set_fill_boolean(set, 0);
4965 } else {
4966 set_fill_boolean(set, 1);
4967 }
4968
4969 return LY_SUCCESS;
4970}
4971
4972/**
4973 * @brief Execute the XPath string(object?) function. Returns LYXP_SET_STRING
4974 * with the string representation of either the argument or the context.
4975 *
4976 * @param[in] args Array of arguments.
4977 * @param[in] arg_count Count of elements in @p args.
4978 * @param[in,out] set Context and result set at the same time.
4979 * @param[in] options XPath options.
Michal Vasko0cbf54f2019-12-16 10:01:06 +01004980 * @return LY_ERR
Michal Vasko03ff5a72019-09-11 13:49:33 +02004981 */
4982static LY_ERR
Radek Krejci1deb5be2020-08-26 16:43:36 +02004983xpath_string(struct lyxp_set **args, uint16_t arg_count, struct lyxp_set *set, uint32_t options)
Michal Vasko03ff5a72019-09-11 13:49:33 +02004984{
4985 LY_ERR rc;
4986
4987 if (options & LYXP_SCNODE_ALL) {
Michal Vasko1a09b212021-05-06 13:00:10 +02004988 set_scnode_clear_ctx(set, LYXP_SET_SCNODE_ATOM_VAL);
Michal Vasko03ff5a72019-09-11 13:49:33 +02004989 return LY_SUCCESS;
4990 }
4991
4992 if (arg_count) {
Michal Vasko5e0e6eb2019-11-06 15:47:50 +01004993 rc = lyxp_set_cast(args[0], LYXP_SET_STRING);
Michal Vasko03ff5a72019-09-11 13:49:33 +02004994 LY_CHECK_RET(rc);
4995 set_fill_set(set, args[0]);
4996 } else {
Michal Vasko5e0e6eb2019-11-06 15:47:50 +01004997 rc = lyxp_set_cast(set, LYXP_SET_STRING);
Michal Vasko03ff5a72019-09-11 13:49:33 +02004998 LY_CHECK_RET(rc);
4999 }
5000
5001 return LY_SUCCESS;
5002}
5003
5004/**
5005 * @brief Execute the XPath string-length(string?) function. Returns LYXP_SET_NUMBER
5006 * with the length of the string in either the argument or the context.
5007 *
5008 * @param[in] args Array of arguments.
5009 * @param[in] arg_count Count of elements in @p args.
5010 * @param[in,out] set Context and result set at the same time.
5011 * @param[in] options XPath options.
Michal Vasko0cbf54f2019-12-16 10:01:06 +01005012 * @return LY_ERR
Michal Vasko03ff5a72019-09-11 13:49:33 +02005013 */
5014static LY_ERR
Radek Krejci1deb5be2020-08-26 16:43:36 +02005015xpath_string_length(struct lyxp_set **args, uint16_t arg_count, struct lyxp_set *set, uint32_t options)
Michal Vasko03ff5a72019-09-11 13:49:33 +02005016{
5017 struct lysc_node_leaf *sleaf;
5018 LY_ERR rc = LY_SUCCESS;
5019
5020 if (options & LYXP_SCNODE_ALL) {
5021 if (arg_count && (args[0]->type == LYXP_SET_SCNODE_SET) && (sleaf = (struct lysc_node_leaf *)warn_get_scnode_in_ctx(args[0]))) {
5022 if (!(sleaf->nodetype & (LYS_LEAF | LYS_LEAFLIST))) {
5023 LOGWRN(set->ctx, "Argument #1 of %s is a %s node \"%s\".", __func__, lys_nodetype2str(sleaf->nodetype), sleaf->name);
Michal Vasko03ff5a72019-09-11 13:49:33 +02005024 } else if (!warn_is_string_type(sleaf->type)) {
5025 LOGWRN(set->ctx, "Argument #1 of %s is node \"%s\", not of string-type.", __func__, sleaf->name);
Michal Vasko03ff5a72019-09-11 13:49:33 +02005026 }
5027 }
5028 if (!arg_count && (set->type == LYXP_SET_SCNODE_SET) && (sleaf = (struct lysc_node_leaf *)warn_get_scnode_in_ctx(set))) {
5029 if (!(sleaf->nodetype & (LYS_LEAF | LYS_LEAFLIST))) {
5030 LOGWRN(set->ctx, "Argument #0 of %s is a %s node \"%s\".", __func__, lys_nodetype2str(sleaf->nodetype), sleaf->name);
Michal Vasko03ff5a72019-09-11 13:49:33 +02005031 } else if (!warn_is_string_type(sleaf->type)) {
5032 LOGWRN(set->ctx, "Argument #0 of %s is node \"%s\", not of string-type.", __func__, sleaf->name);
Michal Vasko03ff5a72019-09-11 13:49:33 +02005033 }
5034 }
Michal Vasko1a09b212021-05-06 13:00:10 +02005035 set_scnode_clear_ctx(set, LYXP_SET_SCNODE_ATOM_VAL);
Michal Vasko03ff5a72019-09-11 13:49:33 +02005036 return rc;
5037 }
5038
5039 if (arg_count) {
Michal Vasko5e0e6eb2019-11-06 15:47:50 +01005040 rc = lyxp_set_cast(args[0], LYXP_SET_STRING);
Michal Vasko03ff5a72019-09-11 13:49:33 +02005041 LY_CHECK_RET(rc);
5042 set_fill_number(set, strlen(args[0]->val.str));
5043 } else {
Michal Vasko5e0e6eb2019-11-06 15:47:50 +01005044 rc = lyxp_set_cast(set, LYXP_SET_STRING);
Michal Vasko03ff5a72019-09-11 13:49:33 +02005045 LY_CHECK_RET(rc);
5046 set_fill_number(set, strlen(set->val.str));
5047 }
5048
5049 return LY_SUCCESS;
5050}
5051
5052/**
5053 * @brief Execute the XPath substring(string, number, number?) function.
5054 * Returns LYXP_SET_STRING substring of the first argument starting
5055 * on the second argument index ending on the third argument index,
5056 * indexed from 1. For exact definition refer to
5057 * http://www.w3.org/TR/1999/REC-xpath-19991116/#function-substring.
5058 *
5059 * @param[in] args Array of arguments.
5060 * @param[in] arg_count Count of elements in @p args.
5061 * @param[in,out] set Context and result set at the same time.
5062 * @param[in] options XPath options.
Michal Vasko0cbf54f2019-12-16 10:01:06 +01005063 * @return LY_ERR
Michal Vasko03ff5a72019-09-11 13:49:33 +02005064 */
5065static LY_ERR
Radek Krejci1deb5be2020-08-26 16:43:36 +02005066xpath_substring(struct lyxp_set **args, uint16_t arg_count, struct lyxp_set *set, uint32_t options)
Michal Vasko03ff5a72019-09-11 13:49:33 +02005067{
Radek Krejci1deb5be2020-08-26 16:43:36 +02005068 int32_t start, len;
Michal Vasko03ff5a72019-09-11 13:49:33 +02005069 uint16_t str_start, str_len, pos;
5070 struct lysc_node_leaf *sleaf;
5071 LY_ERR rc = LY_SUCCESS;
5072
5073 if (options & LYXP_SCNODE_ALL) {
5074 if ((args[0]->type == LYXP_SET_SCNODE_SET) && (sleaf = (struct lysc_node_leaf *)warn_get_scnode_in_ctx(args[0]))) {
5075 if (!(sleaf->nodetype & (LYS_LEAF | LYS_LEAFLIST))) {
5076 LOGWRN(set->ctx, "Argument #1 of %s is a %s node \"%s\".", __func__, lys_nodetype2str(sleaf->nodetype), sleaf->name);
Michal Vasko03ff5a72019-09-11 13:49:33 +02005077 } else if (!warn_is_string_type(sleaf->type)) {
5078 LOGWRN(set->ctx, "Argument #1 of %s is node \"%s\", not of string-type.", __func__, sleaf->name);
Michal Vasko03ff5a72019-09-11 13:49:33 +02005079 }
5080 }
5081
5082 if ((args[1]->type == LYXP_SET_SCNODE_SET) && (sleaf = (struct lysc_node_leaf *)warn_get_scnode_in_ctx(args[1]))) {
5083 if (!(sleaf->nodetype & (LYS_LEAF | LYS_LEAFLIST))) {
5084 LOGWRN(set->ctx, "Argument #2 of %s is a %s node \"%s\".", __func__, lys_nodetype2str(sleaf->nodetype), sleaf->name);
Michal Vasko03ff5a72019-09-11 13:49:33 +02005085 } else if (!warn_is_numeric_type(sleaf->type)) {
5086 LOGWRN(set->ctx, "Argument #2 of %s is node \"%s\", not of numeric type.", __func__, sleaf->name);
Michal Vasko03ff5a72019-09-11 13:49:33 +02005087 }
5088 }
5089
Michal Vasko69730152020-10-09 16:30:07 +02005090 if ((arg_count == 3) && (args[2]->type == LYXP_SET_SCNODE_SET) &&
5091 (sleaf = (struct lysc_node_leaf *)warn_get_scnode_in_ctx(args[2]))) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02005092 if (!(sleaf->nodetype & (LYS_LEAF | LYS_LEAFLIST))) {
5093 LOGWRN(set->ctx, "Argument #3 of %s is a %s node \"%s\".", __func__, lys_nodetype2str(sleaf->nodetype), sleaf->name);
Michal Vasko03ff5a72019-09-11 13:49:33 +02005094 } else if (!warn_is_numeric_type(sleaf->type)) {
5095 LOGWRN(set->ctx, "Argument #3 of %s is node \"%s\", not of numeric type.", __func__, sleaf->name);
Michal Vasko03ff5a72019-09-11 13:49:33 +02005096 }
5097 }
Michal Vasko1a09b212021-05-06 13:00:10 +02005098 set_scnode_clear_ctx(set, LYXP_SET_SCNODE_ATOM_VAL);
Michal Vasko03ff5a72019-09-11 13:49:33 +02005099 return rc;
5100 }
5101
Michal Vasko5e0e6eb2019-11-06 15:47:50 +01005102 rc = lyxp_set_cast(args[0], LYXP_SET_STRING);
Michal Vasko03ff5a72019-09-11 13:49:33 +02005103 LY_CHECK_RET(rc);
5104
5105 /* start */
5106 if (xpath_round(&args[1], 1, args[1], options)) {
5107 return -1;
5108 }
5109 if (isfinite(args[1]->val.num)) {
5110 start = args[1]->val.num - 1;
5111 } else if (isinf(args[1]->val.num) && signbit(args[1]->val.num)) {
Radek Krejci1deb5be2020-08-26 16:43:36 +02005112 start = INT32_MIN;
Michal Vasko03ff5a72019-09-11 13:49:33 +02005113 } else {
Radek Krejci1deb5be2020-08-26 16:43:36 +02005114 start = INT32_MAX;
Michal Vasko03ff5a72019-09-11 13:49:33 +02005115 }
5116
5117 /* len */
5118 if (arg_count == 3) {
5119 rc = xpath_round(&args[2], 1, args[2], options);
5120 LY_CHECK_RET(rc);
Radek Krejci1deb5be2020-08-26 16:43:36 +02005121 if (isnan(args[2]->val.num) || signbit(args[2]->val.num)) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02005122 len = 0;
Radek Krejci1deb5be2020-08-26 16:43:36 +02005123 } else if (isfinite(args[2]->val.num)) {
5124 len = args[2]->val.num;
Michal Vasko03ff5a72019-09-11 13:49:33 +02005125 } else {
Radek Krejci1deb5be2020-08-26 16:43:36 +02005126 len = INT32_MAX;
Michal Vasko03ff5a72019-09-11 13:49:33 +02005127 }
5128 } else {
Radek Krejci1deb5be2020-08-26 16:43:36 +02005129 len = INT32_MAX;
Michal Vasko03ff5a72019-09-11 13:49:33 +02005130 }
5131
5132 /* find matching character positions */
5133 str_start = 0;
5134 str_len = 0;
5135 for (pos = 0; args[0]->val.str[pos]; ++pos) {
5136 if (pos < start) {
5137 ++str_start;
5138 } else if (pos < start + len) {
5139 ++str_len;
5140 } else {
5141 break;
5142 }
5143 }
5144
5145 set_fill_string(set, args[0]->val.str + str_start, str_len);
5146 return LY_SUCCESS;
5147}
5148
5149/**
5150 * @brief Execute the XPath substring-after(string, string) function.
5151 * Returns LYXP_SET_STRING with the string succeeding the occurance
5152 * of the second argument in the first or an empty string.
5153 *
5154 * @param[in] args Array of arguments.
5155 * @param[in] arg_count Count of elements in @p args.
5156 * @param[in,out] set Context and result set at the same time.
5157 * @param[in] options XPath options.
Michal Vasko0cbf54f2019-12-16 10:01:06 +01005158 * @return LY_ERR
Michal Vasko03ff5a72019-09-11 13:49:33 +02005159 */
5160static LY_ERR
Radek Krejci1deb5be2020-08-26 16:43:36 +02005161xpath_substring_after(struct lyxp_set **args, uint16_t UNUSED(arg_count), struct lyxp_set *set, uint32_t options)
Michal Vasko03ff5a72019-09-11 13:49:33 +02005162{
5163 char *ptr;
5164 struct lysc_node_leaf *sleaf;
5165 LY_ERR rc = LY_SUCCESS;
5166
5167 if (options & LYXP_SCNODE_ALL) {
5168 if ((args[0]->type == LYXP_SET_SCNODE_SET) && (sleaf = (struct lysc_node_leaf *)warn_get_scnode_in_ctx(args[0]))) {
5169 if (!(sleaf->nodetype & (LYS_LEAF | LYS_LEAFLIST))) {
5170 LOGWRN(set->ctx, "Argument #1 of %s is a %s node \"%s\".", __func__, lys_nodetype2str(sleaf->nodetype), sleaf->name);
Michal Vasko03ff5a72019-09-11 13:49:33 +02005171 } else if (!warn_is_string_type(sleaf->type)) {
5172 LOGWRN(set->ctx, "Argument #1 of %s is node \"%s\", not of string-type.", __func__, sleaf->name);
Michal Vasko03ff5a72019-09-11 13:49:33 +02005173 }
5174 }
5175
5176 if ((args[1]->type == LYXP_SET_SCNODE_SET) && (sleaf = (struct lysc_node_leaf *)warn_get_scnode_in_ctx(args[1]))) {
5177 if (!(sleaf->nodetype & (LYS_LEAF | LYS_LEAFLIST))) {
5178 LOGWRN(set->ctx, "Argument #2 of %s is a %s node \"%s\".", __func__, lys_nodetype2str(sleaf->nodetype), sleaf->name);
Michal Vasko03ff5a72019-09-11 13:49:33 +02005179 } else if (!warn_is_string_type(sleaf->type)) {
5180 LOGWRN(set->ctx, "Argument #2 of %s is node \"%s\", not of string-type.", __func__, sleaf->name);
Michal Vasko03ff5a72019-09-11 13:49:33 +02005181 }
5182 }
Michal Vasko1a09b212021-05-06 13:00:10 +02005183 set_scnode_clear_ctx(set, LYXP_SET_SCNODE_ATOM_VAL);
Michal Vasko03ff5a72019-09-11 13:49:33 +02005184 return rc;
5185 }
5186
Michal Vasko5e0e6eb2019-11-06 15:47:50 +01005187 rc = lyxp_set_cast(args[0], LYXP_SET_STRING);
Michal Vasko03ff5a72019-09-11 13:49:33 +02005188 LY_CHECK_RET(rc);
Michal Vasko5e0e6eb2019-11-06 15:47:50 +01005189 rc = lyxp_set_cast(args[1], LYXP_SET_STRING);
Michal Vasko03ff5a72019-09-11 13:49:33 +02005190 LY_CHECK_RET(rc);
5191
5192 ptr = strstr(args[0]->val.str, args[1]->val.str);
5193 if (ptr) {
5194 set_fill_string(set, ptr + strlen(args[1]->val.str), strlen(ptr + strlen(args[1]->val.str)));
5195 } else {
5196 set_fill_string(set, "", 0);
5197 }
5198
5199 return LY_SUCCESS;
5200}
5201
5202/**
5203 * @brief Execute the XPath substring-before(string, string) function.
5204 * Returns LYXP_SET_STRING with the string preceding the occurance
5205 * of the second argument in the first or an empty string.
5206 *
5207 * @param[in] args Array of arguments.
5208 * @param[in] arg_count Count of elements in @p args.
5209 * @param[in,out] set Context and result set at the same time.
5210 * @param[in] options XPath options.
Michal Vasko0cbf54f2019-12-16 10:01:06 +01005211 * @return LY_ERR
Michal Vasko03ff5a72019-09-11 13:49:33 +02005212 */
5213static LY_ERR
Radek Krejci1deb5be2020-08-26 16:43:36 +02005214xpath_substring_before(struct lyxp_set **args, uint16_t UNUSED(arg_count), struct lyxp_set *set, uint32_t options)
Michal Vasko03ff5a72019-09-11 13:49:33 +02005215{
5216 char *ptr;
5217 struct lysc_node_leaf *sleaf;
5218 LY_ERR rc = LY_SUCCESS;
5219
5220 if (options & LYXP_SCNODE_ALL) {
5221 if ((args[0]->type == LYXP_SET_SCNODE_SET) && (sleaf = (struct lysc_node_leaf *)warn_get_scnode_in_ctx(args[0]))) {
5222 if (!(sleaf->nodetype & (LYS_LEAF | LYS_LEAFLIST))) {
5223 LOGWRN(set->ctx, "Argument #1 of %s is a %s node \"%s\".", __func__, lys_nodetype2str(sleaf->nodetype), sleaf->name);
Michal Vasko03ff5a72019-09-11 13:49:33 +02005224 } else if (!warn_is_string_type(sleaf->type)) {
5225 LOGWRN(set->ctx, "Argument #1 of %s is node \"%s\", not of string-type.", __func__, sleaf->name);
Michal Vasko03ff5a72019-09-11 13:49:33 +02005226 }
5227 }
5228
5229 if ((args[1]->type == LYXP_SET_SCNODE_SET) && (sleaf = (struct lysc_node_leaf *)warn_get_scnode_in_ctx(args[1]))) {
5230 if (!(sleaf->nodetype & (LYS_LEAF | LYS_LEAFLIST))) {
5231 LOGWRN(set->ctx, "Argument #2 of %s is a %s node \"%s\".", __func__, lys_nodetype2str(sleaf->nodetype), sleaf->name);
Michal Vasko03ff5a72019-09-11 13:49:33 +02005232 } else if (!warn_is_string_type(sleaf->type)) {
5233 LOGWRN(set->ctx, "Argument #2 of %s is node \"%s\", not of string-type.", __func__, sleaf->name);
Michal Vasko03ff5a72019-09-11 13:49:33 +02005234 }
5235 }
Michal Vasko1a09b212021-05-06 13:00:10 +02005236 set_scnode_clear_ctx(set, LYXP_SET_SCNODE_ATOM_VAL);
Michal Vasko03ff5a72019-09-11 13:49:33 +02005237 return rc;
5238 }
5239
Michal Vasko5e0e6eb2019-11-06 15:47:50 +01005240 rc = lyxp_set_cast(args[0], LYXP_SET_STRING);
Michal Vasko03ff5a72019-09-11 13:49:33 +02005241 LY_CHECK_RET(rc);
Michal Vasko5e0e6eb2019-11-06 15:47:50 +01005242 rc = lyxp_set_cast(args[1], LYXP_SET_STRING);
Michal Vasko03ff5a72019-09-11 13:49:33 +02005243 LY_CHECK_RET(rc);
5244
5245 ptr = strstr(args[0]->val.str, args[1]->val.str);
5246 if (ptr) {
5247 set_fill_string(set, args[0]->val.str, ptr - args[0]->val.str);
5248 } else {
5249 set_fill_string(set, "", 0);
5250 }
5251
5252 return LY_SUCCESS;
5253}
5254
5255/**
5256 * @brief Execute the XPath sum(node-set) function. Returns LYXP_SET_NUMBER
5257 * with the sum of all the nodes in the context.
5258 *
5259 * @param[in] args Array of arguments.
5260 * @param[in] arg_count Count of elements in @p args.
5261 * @param[in,out] set Context and result set at the same time.
5262 * @param[in] options XPath options.
Michal Vasko0cbf54f2019-12-16 10:01:06 +01005263 * @return LY_ERR
Michal Vasko03ff5a72019-09-11 13:49:33 +02005264 */
5265static LY_ERR
Radek Krejci1deb5be2020-08-26 16:43:36 +02005266xpath_sum(struct lyxp_set **args, uint16_t UNUSED(arg_count), struct lyxp_set *set, uint32_t options)
Michal Vasko03ff5a72019-09-11 13:49:33 +02005267{
5268 long double num;
5269 char *str;
Michal Vasko1fdd8fa2021-01-08 09:21:45 +01005270 uint32_t i;
Michal Vasko03ff5a72019-09-11 13:49:33 +02005271 struct lyxp_set set_item;
5272 struct lysc_node_leaf *sleaf;
5273 LY_ERR rc = LY_SUCCESS;
5274
5275 if (options & LYXP_SCNODE_ALL) {
5276 if (args[0]->type == LYXP_SET_SCNODE_SET) {
5277 for (i = 0; i < args[0]->used; ++i) {
Radek Krejcif13b87b2020-12-01 22:02:17 +01005278 if (args[0]->val.scnodes[i].in_ctx == LYXP_SET_SCNODE_ATOM_CTX) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02005279 sleaf = (struct lysc_node_leaf *)args[0]->val.scnodes[i].scnode;
5280 if (!(sleaf->nodetype & (LYS_LEAF | LYS_LEAFLIST))) {
5281 LOGWRN(set->ctx, "Argument #1 of %s is a %s node \"%s\".", __func__,
Michal Vasko69730152020-10-09 16:30:07 +02005282 lys_nodetype2str(sleaf->nodetype), sleaf->name);
Michal Vasko03ff5a72019-09-11 13:49:33 +02005283 } else if (!warn_is_numeric_type(sleaf->type)) {
5284 LOGWRN(set->ctx, "Argument #1 of %s is node \"%s\", not of numeric type.", __func__, sleaf->name);
Michal Vasko03ff5a72019-09-11 13:49:33 +02005285 }
5286 }
5287 }
5288 }
Michal Vasko1a09b212021-05-06 13:00:10 +02005289 set_scnode_clear_ctx(set, LYXP_SET_SCNODE_ATOM_VAL);
Michal Vasko03ff5a72019-09-11 13:49:33 +02005290 return rc;
5291 }
5292
5293 set_fill_number(set, 0);
Michal Vasko03ff5a72019-09-11 13:49:33 +02005294
5295 if (args[0]->type != LYXP_SET_NODE_SET) {
Radek Krejci2efc45b2020-12-22 16:25:44 +01005296 LOGVAL(set->ctx, LY_VCODE_XP_INARGTYPE, 1, print_set_type(args[0]), "sum(node-set)");
Michal Vasko03ff5a72019-09-11 13:49:33 +02005297 return LY_EVALID;
Michal Vaskod3678892020-05-21 10:06:58 +02005298 } else if (!args[0]->used) {
5299 return LY_SUCCESS;
Michal Vasko03ff5a72019-09-11 13:49:33 +02005300 }
5301
Michal Vasko5c4e5892019-11-14 12:31:38 +01005302 set_init(&set_item, set);
5303
Michal Vasko03ff5a72019-09-11 13:49:33 +02005304 set_item.type = LYXP_SET_NODE_SET;
Michal Vasko41decbf2021-11-02 11:50:21 +01005305 set_item.val.nodes = calloc(1, sizeof *set_item.val.nodes);
Michal Vasko03ff5a72019-09-11 13:49:33 +02005306 LY_CHECK_ERR_RET(!set_item.val.nodes, LOGMEM(set->ctx), LY_EMEM);
5307
5308 set_item.used = 1;
5309 set_item.size = 1;
5310
5311 for (i = 0; i < args[0]->used; ++i) {
5312 set_item.val.nodes[0] = args[0]->val.nodes[i];
5313
Michal Vasko5e0e6eb2019-11-06 15:47:50 +01005314 rc = cast_node_set_to_string(&set_item, &str);
Michal Vasko03ff5a72019-09-11 13:49:33 +02005315 LY_CHECK_RET(rc);
5316 num = cast_string_to_number(str);
5317 free(str);
5318 set->val.num += num;
5319 }
5320
5321 free(set_item.val.nodes);
5322
5323 return LY_SUCCESS;
5324}
5325
5326/**
Michal Vasko03ff5a72019-09-11 13:49:33 +02005327 * @brief Execute the XPath translate(string, string, string) function.
5328 * Returns LYXP_SET_STRING with the first argument with the characters
5329 * from the second argument replaced by those on the corresponding
5330 * positions in the third argument.
5331 *
5332 * @param[in] args Array of arguments.
5333 * @param[in] arg_count Count of elements in @p args.
5334 * @param[in,out] set Context and result set at the same time.
5335 * @param[in] options XPath options.
Michal Vasko0cbf54f2019-12-16 10:01:06 +01005336 * @return LY_ERR
Michal Vasko03ff5a72019-09-11 13:49:33 +02005337 */
5338static LY_ERR
Radek Krejci1deb5be2020-08-26 16:43:36 +02005339xpath_translate(struct lyxp_set **args, uint16_t UNUSED(arg_count), struct lyxp_set *set, uint32_t options)
Michal Vasko03ff5a72019-09-11 13:49:33 +02005340{
5341 uint16_t i, j, new_used;
5342 char *new;
Radek Krejci857189e2020-09-01 13:26:36 +02005343 ly_bool have_removed;
Michal Vasko03ff5a72019-09-11 13:49:33 +02005344 struct lysc_node_leaf *sleaf;
5345 LY_ERR rc = LY_SUCCESS;
5346
5347 if (options & LYXP_SCNODE_ALL) {
5348 if ((args[0]->type == LYXP_SET_SCNODE_SET) && (sleaf = (struct lysc_node_leaf *)warn_get_scnode_in_ctx(args[0]))) {
5349 if (!(sleaf->nodetype & (LYS_LEAF | LYS_LEAFLIST))) {
5350 LOGWRN(set->ctx, "Argument #1 of %s is a %s node \"%s\".", __func__, lys_nodetype2str(sleaf->nodetype), sleaf->name);
Michal Vasko03ff5a72019-09-11 13:49:33 +02005351 } else if (!warn_is_string_type(sleaf->type)) {
5352 LOGWRN(set->ctx, "Argument #1 of %s is node \"%s\", not of string-type.", __func__, sleaf->name);
Michal Vasko03ff5a72019-09-11 13:49:33 +02005353 }
5354 }
5355
5356 if ((args[1]->type == LYXP_SET_SCNODE_SET) && (sleaf = (struct lysc_node_leaf *)warn_get_scnode_in_ctx(args[1]))) {
5357 if (!(sleaf->nodetype & (LYS_LEAF | LYS_LEAFLIST))) {
5358 LOGWRN(set->ctx, "Argument #2 of %s is a %s node \"%s\".", __func__, lys_nodetype2str(sleaf->nodetype), sleaf->name);
Michal Vasko03ff5a72019-09-11 13:49:33 +02005359 } else if (!warn_is_string_type(sleaf->type)) {
5360 LOGWRN(set->ctx, "Argument #2 of %s is node \"%s\", not of string-type.", __func__, sleaf->name);
Michal Vasko03ff5a72019-09-11 13:49:33 +02005361 }
5362 }
5363
5364 if ((args[2]->type == LYXP_SET_SCNODE_SET) && (sleaf = (struct lysc_node_leaf *)warn_get_scnode_in_ctx(args[2]))) {
5365 if (!(sleaf->nodetype & (LYS_LEAF | LYS_LEAFLIST))) {
5366 LOGWRN(set->ctx, "Argument #3 of %s is a %s node \"%s\".", __func__, lys_nodetype2str(sleaf->nodetype), sleaf->name);
Michal Vasko03ff5a72019-09-11 13:49:33 +02005367 } else if (!warn_is_string_type(sleaf->type)) {
5368 LOGWRN(set->ctx, "Argument #3 of %s is node \"%s\", not of string-type.", __func__, sleaf->name);
Michal Vasko03ff5a72019-09-11 13:49:33 +02005369 }
5370 }
Michal Vasko1a09b212021-05-06 13:00:10 +02005371 set_scnode_clear_ctx(set, LYXP_SET_SCNODE_ATOM_VAL);
Michal Vasko03ff5a72019-09-11 13:49:33 +02005372 return rc;
5373 }
5374
Michal Vasko5e0e6eb2019-11-06 15:47:50 +01005375 rc = lyxp_set_cast(args[0], LYXP_SET_STRING);
Michal Vasko03ff5a72019-09-11 13:49:33 +02005376 LY_CHECK_RET(rc);
Michal Vasko5e0e6eb2019-11-06 15:47:50 +01005377 rc = lyxp_set_cast(args[1], LYXP_SET_STRING);
Michal Vasko03ff5a72019-09-11 13:49:33 +02005378 LY_CHECK_RET(rc);
Michal Vasko5e0e6eb2019-11-06 15:47:50 +01005379 rc = lyxp_set_cast(args[2], LYXP_SET_STRING);
Michal Vasko03ff5a72019-09-11 13:49:33 +02005380 LY_CHECK_RET(rc);
5381
5382 new = malloc((strlen(args[0]->val.str) + 1) * sizeof(char));
5383 LY_CHECK_ERR_RET(!new, LOGMEM(set->ctx), LY_EMEM);
5384 new_used = 0;
5385
5386 have_removed = 0;
5387 for (i = 0; args[0]->val.str[i]; ++i) {
Radek Krejci857189e2020-09-01 13:26:36 +02005388 ly_bool found = 0;
Michal Vasko03ff5a72019-09-11 13:49:33 +02005389
5390 for (j = 0; args[1]->val.str[j]; ++j) {
5391 if (args[0]->val.str[i] == args[1]->val.str[j]) {
5392 /* removing this char */
5393 if (j >= strlen(args[2]->val.str)) {
5394 have_removed = 1;
5395 found = 1;
5396 break;
5397 }
5398 /* replacing this char */
5399 new[new_used] = args[2]->val.str[j];
5400 ++new_used;
5401 found = 1;
5402 break;
5403 }
5404 }
5405
5406 /* copying this char */
5407 if (!found) {
5408 new[new_used] = args[0]->val.str[i];
5409 ++new_used;
5410 }
5411 }
5412
5413 if (have_removed) {
5414 new = ly_realloc(new, (new_used + 1) * sizeof(char));
5415 LY_CHECK_ERR_RET(!new, LOGMEM(set->ctx), LY_EMEM);
5416 }
5417 new[new_used] = '\0';
5418
Michal Vaskod3678892020-05-21 10:06:58 +02005419 lyxp_set_free_content(set);
Michal Vasko03ff5a72019-09-11 13:49:33 +02005420 set->type = LYXP_SET_STRING;
5421 set->val.str = new;
5422
5423 return LY_SUCCESS;
5424}
5425
5426/**
5427 * @brief Execute the XPath true() function. Returns LYXP_SET_BOOLEAN
5428 * with true value.
5429 *
5430 * @param[in] args Array of arguments.
5431 * @param[in] arg_count Count of elements in @p args.
5432 * @param[in,out] set Context and result set at the same time.
5433 * @param[in] options XPath options.
Michal Vasko0cbf54f2019-12-16 10:01:06 +01005434 * @return LY_ERR
Michal Vasko03ff5a72019-09-11 13:49:33 +02005435 */
5436static LY_ERR
Radek Krejci1deb5be2020-08-26 16:43:36 +02005437xpath_true(struct lyxp_set **UNUSED(args), uint16_t UNUSED(arg_count), struct lyxp_set *set, uint32_t options)
Michal Vasko03ff5a72019-09-11 13:49:33 +02005438{
5439 if (options & LYXP_SCNODE_ALL) {
Michal Vasko1a09b212021-05-06 13:00:10 +02005440 set_scnode_clear_ctx(set, LYXP_SET_SCNODE_ATOM_NODE);
Michal Vasko03ff5a72019-09-11 13:49:33 +02005441 return LY_SUCCESS;
5442 }
5443
5444 set_fill_boolean(set, 1);
5445 return LY_SUCCESS;
5446}
5447
Michal Vasko03ff5a72019-09-11 13:49:33 +02005448/**
Michal Vasko49fec8e2022-05-24 10:28:33 +02005449 * @brief Execute the XPath node() processing instruction (node type). Returns LYXP_SET_NODE_SET
5450 * with only nodes from the context.
5451 *
5452 * @param[in,out] set Context and result set at the same time.
5453 * @param[in] axis Axis to search on.
5454 * @param[in] options XPath options.
5455 * @return LY_ERR
5456 */
5457static LY_ERR
5458xpath_pi_node(struct lyxp_set *set, enum lyxp_axis axis, uint32_t options)
5459{
5460 if (options & LYXP_SCNODE_ALL) {
5461 return moveto_scnode(set, NULL, NULL, axis, options);
5462 }
5463
5464 if (set->type != LYXP_SET_NODE_SET) {
5465 lyxp_set_free_content(set);
5466 return LY_SUCCESS;
5467 }
5468
5469 /* just like moving to a node with no restrictions */
5470 return moveto_node(set, NULL, NULL, axis, options);
5471}
5472
5473/**
5474 * @brief Execute the XPath text() processing instruction (node type). Returns LYXP_SET_NODE_SET
5475 * with the text content of the nodes in the context.
5476 *
5477 * @param[in,out] set Context and result set at the same time.
5478 * @param[in] axis Axis to search on.
5479 * @param[in] options XPath options.
5480 * @return LY_ERR
5481 */
5482static LY_ERR
5483xpath_pi_text(struct lyxp_set *set, enum lyxp_axis axis, uint32_t options)
5484{
5485 uint32_t i;
5486
5487 if (options & LYXP_SCNODE_ALL) {
5488 set_scnode_clear_ctx(set, LYXP_SET_SCNODE_ATOM_VAL);
5489 return LY_SUCCESS;
5490 }
5491
5492 if (set->type != LYXP_SET_NODE_SET) {
5493 LOGVAL(set->ctx, LY_VCODE_XP_INCTX, print_set_type(set), "text()");
5494 return LY_EVALID;
5495 }
5496
5497 if (axis != LYXP_AXIS_CHILD) {
5498 /* even following and preceding axescan return text nodes, but whatever */
5499 lyxp_set_free_content(set);
5500 return LY_SUCCESS;
5501 }
5502
5503 for (i = 0; i < set->used; ++i) {
5504 switch (set->val.nodes[i].type) {
5505 case LYXP_NODE_NONE:
5506 LOGINT_RET(set->ctx);
5507 case LYXP_NODE_ELEM:
5508 if (set->val.nodes[i].node->schema->nodetype & (LYS_LEAF | LYS_LEAFLIST)) {
5509 set->val.nodes[i].type = LYXP_NODE_TEXT;
5510 break;
5511 }
5512 /* fall through */
5513 case LYXP_NODE_ROOT:
5514 case LYXP_NODE_ROOT_CONFIG:
5515 case LYXP_NODE_TEXT:
5516 case LYXP_NODE_META:
5517 set_remove_node_none(set, i);
5518 break;
5519 }
5520 }
5521 set_remove_nodes_none(set);
5522
5523 return LY_SUCCESS;
5524}
5525
5526/**
Michal Vasko6346ece2019-09-24 13:12:53 +02005527 * @brief Skip prefix and return corresponding model if there is a prefix. Logs directly.
Michal Vasko03ff5a72019-09-11 13:49:33 +02005528 *
Michal Vasko2104e9f2020-03-06 08:23:25 +01005529 * XPath @p set is expected to be a (sc)node set!
5530 *
Michal Vasko6346ece2019-09-24 13:12:53 +02005531 * @param[in,out] qname Qualified node name. If includes prefix, it is skipped.
5532 * @param[in,out] qname_len Length of @p qname, is updated accordingly.
Michal Vasko5d24f6c2020-10-13 13:49:06 +02005533 * @param[in] set Set with general XPath context.
5534 * @param[in] ctx_scnode Context node to inherit module for unprefixed node for ::LY_PREF_JSON.
Michal Vasko6346ece2019-09-24 13:12:53 +02005535 * @param[out] moveto_mod Expected module of a matching node.
5536 * @return LY_ERR
Michal Vasko03ff5a72019-09-11 13:49:33 +02005537 */
Michal Vasko6346ece2019-09-24 13:12:53 +02005538static LY_ERR
Michal Vasko5d24f6c2020-10-13 13:49:06 +02005539moveto_resolve_model(const char **qname, uint16_t *qname_len, const struct lyxp_set *set,
5540 const struct lysc_node *ctx_scnode, const struct lys_module **moveto_mod)
Michal Vasko03ff5a72019-09-11 13:49:33 +02005541{
Michal Vaskoed4fcfe2020-07-08 10:38:56 +02005542 const struct lys_module *mod = NULL;
Michal Vasko6346ece2019-09-24 13:12:53 +02005543 const char *ptr;
Radek Krejci1deb5be2020-08-26 16:43:36 +02005544 size_t pref_len;
Michal Vasko03ff5a72019-09-11 13:49:33 +02005545
Michal Vasko2104e9f2020-03-06 08:23:25 +01005546 assert((set->type == LYXP_SET_NODE_SET) || (set->type == LYXP_SET_SCNODE_SET));
5547
Michal Vasko6346ece2019-09-24 13:12:53 +02005548 if ((ptr = ly_strnchr(*qname, ':', *qname_len))) {
5549 /* specific module */
5550 pref_len = ptr - *qname;
Michal Vasko5d24f6c2020-10-13 13:49:06 +02005551 mod = ly_resolve_prefix(set->ctx, *qname, pref_len, set->format, set->prefix_data);
Michal Vasko6346ece2019-09-24 13:12:53 +02005552
Michal Vasko004d3152020-06-11 19:59:22 +02005553 /* check for errors and non-implemented modules, as they are not valid */
Juraj Vijtiukd75faa62019-11-26 14:10:10 +01005554 if (!mod || !mod->implemented) {
Radek Krejci2efc45b2020-12-22 16:25:44 +01005555 LOGVAL(set->ctx, LY_VCODE_XP_INMOD, pref_len, *qname);
Michal Vasko6346ece2019-09-24 13:12:53 +02005556 return LY_EVALID;
5557 }
Juraj Vijtiukd75faa62019-11-26 14:10:10 +01005558
Michal Vasko6346ece2019-09-24 13:12:53 +02005559 *qname += pref_len + 1;
5560 *qname_len -= pref_len + 1;
5561 } else if (((*qname)[0] == '*') && (*qname_len == 1)) {
5562 /* all modules - special case */
Michal Vasko03ff5a72019-09-11 13:49:33 +02005563 mod = NULL;
Michal Vasko6346ece2019-09-24 13:12:53 +02005564 } else {
Michal Vasko5d24f6c2020-10-13 13:49:06 +02005565 switch (set->format) {
Radek Krejci8df109d2021-04-23 12:19:08 +02005566 case LY_VALUE_SCHEMA:
5567 case LY_VALUE_SCHEMA_RESOLVED:
Michal Vasko5d24f6c2020-10-13 13:49:06 +02005568 /* current module */
5569 mod = set->cur_mod;
5570 break;
Radek Krejci224d4b42021-04-23 13:54:59 +02005571 case LY_VALUE_CANON:
Radek Krejci8df109d2021-04-23 12:19:08 +02005572 case LY_VALUE_JSON:
Radek Krejcif9943642021-04-26 10:18:21 +02005573 case LY_VALUE_LYB:
Michal Vaskoddd76592022-01-17 13:34:48 +01005574 case LY_VALUE_STR_NS:
Michal Vasko5d24f6c2020-10-13 13:49:06 +02005575 /* inherit parent (context node) module */
5576 if (ctx_scnode) {
5577 mod = ctx_scnode->module;
5578 } else {
5579 mod = NULL;
5580 }
5581 break;
Radek Krejci8df109d2021-04-23 12:19:08 +02005582 case LY_VALUE_XML:
Michal Vasko52143d12021-04-14 15:36:39 +02005583 /* all nodes need to be prefixed */
5584 LOGVAL(set->ctx, LYVE_DATA, "Non-prefixed node \"%.*s\" in XML xpath found.", *qname_len, *qname);
5585 return LY_EVALID;
Michal Vasko5d24f6c2020-10-13 13:49:06 +02005586 }
Michal Vasko03ff5a72019-09-11 13:49:33 +02005587 }
5588
Michal Vasko6346ece2019-09-24 13:12:53 +02005589 *moveto_mod = mod;
5590 return LY_SUCCESS;
Michal Vasko03ff5a72019-09-11 13:49:33 +02005591}
5592
5593/**
Michal Vasko03ff5a72019-09-11 13:49:33 +02005594 * @brief Move context @p set to the root. Handles absolute path.
5595 * Result is LYXP_SET_NODE_SET.
5596 *
5597 * @param[in,out] set Set to use.
5598 * @param[in] options Xpath options.
Michal Vaskob0099a92020-08-31 14:55:23 +02005599 * @return LY_ERR value.
Michal Vasko03ff5a72019-09-11 13:49:33 +02005600 */
Michal Vaskob0099a92020-08-31 14:55:23 +02005601static LY_ERR
Radek Krejci1deb5be2020-08-26 16:43:36 +02005602moveto_root(struct lyxp_set *set, uint32_t options)
Michal Vasko03ff5a72019-09-11 13:49:33 +02005603{
aPiecek8b0cc152021-05-31 16:40:31 +02005604 assert(!(options & LYXP_SKIP_EXPR));
Michal Vasko03ff5a72019-09-11 13:49:33 +02005605
5606 if (options & LYXP_SCNODE_ALL) {
Michal Vasko1a09b212021-05-06 13:00:10 +02005607 set_scnode_clear_ctx(set, LYXP_SET_SCNODE_ATOM_NODE);
Michal Vaskob0099a92020-08-31 14:55:23 +02005608 LY_CHECK_RET(lyxp_set_scnode_insert_node(set, NULL, set->root_type, NULL));
Michal Vasko03ff5a72019-09-11 13:49:33 +02005609 } else {
Michal Vaskod3678892020-05-21 10:06:58 +02005610 set->type = LYXP_SET_NODE_SET;
5611 set->used = 0;
Michal Vasko5e0e6eb2019-11-06 15:47:50 +01005612 set_insert_node(set, NULL, 0, set->root_type, 0);
Michal Vasko03ff5a72019-09-11 13:49:33 +02005613 }
Michal Vaskob0099a92020-08-31 14:55:23 +02005614
5615 return LY_SUCCESS;
Michal Vasko03ff5a72019-09-11 13:49:33 +02005616}
5617
5618/**
5619 * @brief Check @p node as a part of NameTest processing.
5620 *
5621 * @param[in] node Node to check.
Michal Vasko49fec8e2022-05-24 10:28:33 +02005622 * @param[in] node_type Node type of @p node.
Michal Vasko5d24f6c2020-10-13 13:49:06 +02005623 * @param[in] set Set to read general context from.
Michal Vaskod3678892020-05-21 10:06:58 +02005624 * @param[in] node_name Node name in the dictionary to move to, NULL for any node.
Michal Vasko5d24f6c2020-10-13 13:49:06 +02005625 * @param[in] moveto_mod Expected module of the node, NULL for no prefix.
Michal Vaskocdad7122020-11-09 21:04:44 +01005626 * @param[in] options XPath options.
Michal Vasko6346ece2019-09-24 13:12:53 +02005627 * @return LY_ERR (LY_ENOT if node does not match, LY_EINCOMPLETE on unresolved when,
Michal Vaskocd2c88a2022-06-07 10:54:34 +02005628 * LY_EINVAL if neither node nor any children match)
Michal Vasko03ff5a72019-09-11 13:49:33 +02005629 */
5630static LY_ERR
Michal Vasko49fec8e2022-05-24 10:28:33 +02005631moveto_node_check(const struct lyd_node *node, enum lyxp_node_type node_type, const struct lyxp_set *set,
5632 const char *node_name, const struct lys_module *moveto_mod, uint32_t options)
Michal Vasko03ff5a72019-09-11 13:49:33 +02005633{
Michal Vasko49fec8e2022-05-24 10:28:33 +02005634 if ((node_type == LYXP_NODE_ROOT_CONFIG) || (node_type == LYXP_NODE_ROOT)) {
5635 assert(node_type == set->root_type);
5636
5637 if (node_name || moveto_mod) {
5638 /* root will not match a specific node */
5639 return LY_ENOT;
5640 }
5641 return LY_SUCCESS;
5642 } else if (node_type != LYXP_NODE_ELEM) {
5643 /* other types will not match */
5644 return LY_ENOT;
5645 }
5646
Michal Vaskodca9f122021-07-16 13:56:22 +02005647 if (!node->schema) {
5648 /* opaque node never matches */
5649 return LY_ENOT;
5650 }
5651
Michal Vasko03ff5a72019-09-11 13:49:33 +02005652 /* module check */
Michal Vasko19089f02022-06-07 11:02:11 +02005653 if (moveto_mod) {
5654 if (!(node->flags & LYD_EXT) && (node->schema->module != moveto_mod)) {
5655 return LY_ENOT;
5656 } else if ((node->flags & LYD_EXT) && strcmp(node->schema->module->name, moveto_mod->name)) {
5657 return LY_ENOT;
5658 }
Michal Vasko03ff5a72019-09-11 13:49:33 +02005659 }
5660
Michal Vasko5c4e5892019-11-14 12:31:38 +01005661 /* context check */
Michal Vasko5d24f6c2020-10-13 13:49:06 +02005662 if ((set->root_type == LYXP_NODE_ROOT_CONFIG) && (node->schema->flags & LYS_CONFIG_R)) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02005663 return LY_EINVAL;
Michal Vasko5d24f6c2020-10-13 13:49:06 +02005664 } else if (set->context_op && (node->schema->nodetype & (LYS_RPC | LYS_ACTION | LYS_NOTIF)) &&
5665 (node->schema != set->context_op)) {
Michal Vasko6b26e742020-07-17 15:02:10 +02005666 return LY_EINVAL;
Michal Vasko03ff5a72019-09-11 13:49:33 +02005667 }
5668
5669 /* name check */
Michal Vasko19089f02022-06-07 11:02:11 +02005670 if (node_name) {
5671 if (!(node->flags & LYD_EXT) && (node->schema->name != node_name)) {
5672 return LY_ENOT;
5673 } else if ((node->flags & LYD_EXT) && strcmp(node->schema->name, node_name)) {
5674 return LY_ENOT;
5675 }
Michal Vasko03ff5a72019-09-11 13:49:33 +02005676 }
5677
Michal Vaskoa1424542019-11-14 16:08:52 +01005678 /* when check */
Michal Vaskod5cfa6e2020-11-23 16:56:08 +01005679 if (!(options & LYXP_IGNORE_WHEN) && lysc_has_when(node->schema) && !(node->flags & LYD_WHEN_TRUE)) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02005680 return LY_EINCOMPLETE;
Michal Vaskoa1424542019-11-14 16:08:52 +01005681 }
Michal Vasko03ff5a72019-09-11 13:49:33 +02005682
5683 /* match */
5684 return LY_SUCCESS;
5685}
5686
5687/**
Michal Vasko49fec8e2022-05-24 10:28:33 +02005688 * @brief Get the next node in a forward DFS.
Michal Vasko03ff5a72019-09-11 13:49:33 +02005689 *
Michal Vasko49fec8e2022-05-24 10:28:33 +02005690 * @param[in] iter Last returned node.
5691 * @param[in] stop Node to stop the search on and not return.
5692 * @return Next node, NULL if there are no more.
Michal Vasko03ff5a72019-09-11 13:49:33 +02005693 */
Michal Vasko49fec8e2022-05-24 10:28:33 +02005694static const struct lyd_node *
5695moveto_axis_node_next_dfs_forward(const struct lyd_node *iter, const struct lyd_node *stop)
Michal Vasko03ff5a72019-09-11 13:49:33 +02005696{
Michal Vasko49fec8e2022-05-24 10:28:33 +02005697 const struct lyd_node *next = NULL;
5698
5699 /* 1) child */
5700 next = lyd_child(iter);
5701 if (!next) {
5702 if (iter == stop) {
5703 /* reached stop, no more descendants */
5704 return NULL;
Michal Vasko5d24f6c2020-10-13 13:49:06 +02005705 }
Michal Vasko49fec8e2022-05-24 10:28:33 +02005706 /* 2) child next sibling */
5707 next = iter->next;
5708 }
5709 while (!next) {
5710 iter = lyd_parent(iter);
5711 if ((!stop && !iter) || (stop && (lyd_parent(iter) == lyd_parent(stop)))) {
5712 return NULL;
5713 }
5714 next = iter->next;
Michal Vasko5d24f6c2020-10-13 13:49:06 +02005715 }
5716
Michal Vasko49fec8e2022-05-24 10:28:33 +02005717 return next;
Michal Vasko03ff5a72019-09-11 13:49:33 +02005718}
5719
5720/**
Michal Vasko49fec8e2022-05-24 10:28:33 +02005721 * @brief Get the next node in a backward DFS.
5722 *
5723 * @param[in] iter Last returned node.
5724 * @param[in] stop Node to stop the search on and not return.
5725 * @return Next node, NULL if there are no more.
5726 */
5727static const struct lyd_node *
5728moveto_axis_node_next_dfs_backward(const struct lyd_node *iter, const struct lyd_node *stop)
5729{
5730 const struct lyd_node *next = NULL;
5731
5732 /* 1) previous sibling innermost last child */
5733 next = iter->prev->next ? iter->prev : NULL;
5734 while (next && lyd_child(next)) {
5735 next = lyd_child(next);
5736 next = next->prev;
5737 }
5738
5739 if (!next) {
5740 /* 2) parent */
5741 iter = lyd_parent(iter);
5742 if ((!stop && !iter) || (stop && (lyd_parent(iter) == lyd_parent(stop)))) {
5743 return NULL;
5744 }
5745 next = iter;
5746 }
5747
5748 return next;
5749}
5750
5751/**
5752 * @brief Get the first node on an axis for a context node.
5753 *
5754 * @param[in,out] iter NULL, updated to the next node.
5755 * @param[in,out] iter_type Node type 0 of @p iter, updated to the node type of the next node.
5756 * @param[in] node Context node.
5757 * @param[in] node_type Type of @p node.
5758 * @param[in] axis Axis to use.
5759 * @param[in] set XPath set with the general context.
5760 * @return LY_SUCCESS on success.
5761 * @return LY_ENOTFOUND if no next node found.
5762 */
5763static LY_ERR
5764moveto_axis_node_next_first(const struct lyd_node **iter, enum lyxp_node_type *iter_type, const struct lyd_node *node,
5765 enum lyxp_node_type node_type, enum lyxp_axis axis, struct lyxp_set *set)
5766{
5767 const struct lyd_node *next = NULL;
5768 enum lyxp_node_type next_type = 0;
5769
5770 assert(!*iter);
5771 assert(!*iter_type);
5772
5773 switch (axis) {
5774 case LYXP_AXIS_ANCESTOR_OR_SELF:
5775 case LYXP_AXIS_DESCENDANT_OR_SELF:
5776 case LYXP_AXIS_SELF:
5777 /* return the context node */
5778 next = node;
5779 next_type = node_type;
5780 break;
5781
5782 case LYXP_AXIS_ANCESTOR:
5783 case LYXP_AXIS_PARENT:
5784 if (node_type == LYXP_NODE_ELEM) {
5785 next = lyd_parent(node);
5786 next_type = next ? LYXP_NODE_ELEM : set->root_type;
5787 } else if (node_type == LYXP_NODE_TEXT) {
5788 next = node;
5789 next_type = LYXP_NODE_ELEM;
5790 } else if (node_type == LYXP_NODE_META) {
5791 next = ((struct lyd_meta *)node)->parent;
5792 next_type = LYXP_NODE_ELEM;
5793 } /* else root does not have a parent */
5794 break;
5795
5796 case LYXP_AXIS_CHILD:
5797 if ((node_type == LYXP_NODE_ROOT_CONFIG) || (node_type == LYXP_NODE_ROOT)) {
5798 assert(!node);
5799
5800 /* search in all the trees */
5801 next = set->tree;
5802 next_type = next ? LYXP_NODE_ELEM : 0;
5803 } else {
5804 /* search in children */
5805 next = lyd_child(node);
5806 next_type = next ? LYXP_NODE_ELEM : 0;
5807 }
5808 break;
5809
5810 case LYXP_AXIS_DESCENDANT:
5811 if ((node_type == LYXP_NODE_ROOT_CONFIG) || (node_type == LYXP_NODE_ROOT)) {
5812 /* top-level nodes */
5813 next = set->tree;
5814 next_type = LYXP_NODE_ELEM;
5815 } else if (node_type == LYXP_NODE_ELEM) {
5816 /* start from the context node */
5817 next = moveto_axis_node_next_dfs_forward(node, node);
5818 next_type = next ? LYXP_NODE_ELEM : 0;
5819 } /* else no children */
5820 break;
5821
5822 case LYXP_AXIS_FOLLOWING:
5823 case LYXP_AXIS_FOLLOWING_SIBLING:
5824 if (node_type == LYXP_NODE_ELEM) {
5825 /* first next sibling */
5826 next = node->next;
5827 next_type = next ? LYXP_NODE_ELEM : 0;
5828 } /* else no sibling */
5829 break;
5830
5831 case LYXP_AXIS_PRECEDING:
5832 if ((node_type == LYXP_NODE_ELEM) && node->prev->next) {
5833 /* skip ancestors */
5834 next = moveto_axis_node_next_dfs_backward(node, NULL);
5835 assert(next);
5836 next_type = LYXP_NODE_ELEM;
5837 } /* else no sibling */
5838 break;
5839
5840 case LYXP_AXIS_PRECEDING_SIBLING:
5841 if (node_type == LYXP_NODE_ELEM) {
5842 /* first previous sibling */
5843 next = node->prev->next ? node->prev : NULL;
5844 next_type = next ? LYXP_NODE_ELEM : 0;
5845 } /* else no sibling */
5846 break;
5847
5848 case LYXP_AXIS_ATTRIBUTE:
5849 /* handled specially */
5850 assert(0);
5851 LOGINT(set->ctx);
5852 break;
5853 }
5854
5855 *iter = next;
5856 *iter_type = next_type;
5857 return next_type ? LY_SUCCESS : LY_ENOTFOUND;
5858}
5859
5860/**
5861 * @brief Iterate over all nodes on an axis for a context node.
5862 *
5863 * @param[in,out] iter Last returned node, start with NULL, updated to the next node.
5864 * @param[in,out] iter_type Node type of @p iter, start with 0, updated to the node type of the next node.
5865 * @param[in] node Context node.
5866 * @param[in] node_type Type of @p node.
5867 * @param[in] axis Axis to use.
5868 * @param[in] set XPath set with the general context.
5869 * @return LY_SUCCESS on success.
5870 * @return LY_ENOTFOUND if no next node found.
5871 */
5872static LY_ERR
5873moveto_axis_node_next(const struct lyd_node **iter, enum lyxp_node_type *iter_type, const struct lyd_node *node,
5874 enum lyxp_node_type node_type, enum lyxp_axis axis, struct lyxp_set *set)
5875{
5876 const struct lyd_node *next = NULL;
5877 enum lyxp_node_type next_type = 0;
5878
5879 if (!*iter_type) {
5880 /* first returned node */
5881 return moveto_axis_node_next_first(iter, iter_type, node, node_type, axis, set);
5882 }
5883
5884 switch (axis) {
5885 case LYXP_AXIS_ANCESTOR_OR_SELF:
5886 if ((*iter == node) && (*iter_type == node_type)) {
5887 /* fake first ancestor, we returned self before */
5888 *iter = NULL;
5889 *iter_type = 0;
5890 return moveto_axis_node_next_first(iter, iter_type, node, node_type, LYXP_AXIS_ANCESTOR, set);
5891 } /* else continue ancestor */
5892
5893 /* fallthrough */
5894 case LYXP_AXIS_ANCESTOR:
5895 if (*iter_type == LYXP_NODE_ELEM) {
5896 /* iter parent */
5897 next = lyd_parent(*iter);
5898 next_type = next ? LYXP_NODE_ELEM : set->root_type;
5899 } /* else root, no ancestors */
5900 break;
5901
5902 case LYXP_AXIS_CHILD:
5903 assert(*iter_type == LYXP_NODE_ELEM);
5904
5905 /* next sibling (child) */
5906 next = (*iter)->next;
5907 next_type = next ? LYXP_NODE_ELEM : 0;
5908 break;
5909
5910 case LYXP_AXIS_DESCENDANT_OR_SELF:
5911 if ((*iter == node) && (*iter_type == node_type)) {
5912 /* fake first descendant, we returned self before */
5913 *iter = NULL;
5914 *iter_type = 0;
5915 return moveto_axis_node_next_first(iter, iter_type, node, node_type, LYXP_AXIS_DESCENDANT, set);
5916 } /* else continue descendant */
5917
5918 /* fallthrough */
5919 case LYXP_AXIS_DESCENDANT:
5920 assert(*iter_type == LYXP_NODE_ELEM);
5921 next = moveto_axis_node_next_dfs_forward(*iter, node);
5922 next_type = next ? LYXP_NODE_ELEM : 0;
5923 break;
5924
5925 case LYXP_AXIS_FOLLOWING:
5926 assert(*iter_type == LYXP_NODE_ELEM);
5927 next = moveto_axis_node_next_dfs_forward(*iter, NULL);
5928 next_type = next ? LYXP_NODE_ELEM : 0;
5929 break;
5930
5931 case LYXP_AXIS_FOLLOWING_SIBLING:
5932 assert(*iter_type == LYXP_NODE_ELEM);
5933
5934 /* next sibling */
5935 next = (*iter)->next;
5936 next_type = next ? LYXP_NODE_ELEM : 0;
5937 break;
5938
5939 case LYXP_AXIS_PARENT:
5940 case LYXP_AXIS_SELF:
5941 /* parent/self was returned before */
5942 break;
5943
5944 case LYXP_AXIS_PRECEDING:
5945 assert(*iter_type == LYXP_NODE_ELEM);
5946 next = moveto_axis_node_next_dfs_backward(*iter, NULL);
5947 next_type = next ? LYXP_NODE_ELEM : 0;
5948 break;
5949
5950 case LYXP_AXIS_PRECEDING_SIBLING:
5951 assert(*iter_type == LYXP_NODE_ELEM);
5952
5953 /* previous sibling */
5954 next = (*iter)->prev->next ? (*iter)->prev : NULL;
5955 next_type = next ? LYXP_NODE_ELEM : 0;
5956 break;
5957
5958 case LYXP_AXIS_ATTRIBUTE:
5959 /* handled specially */
5960 assert(0);
5961 LOGINT(set->ctx);
5962 break;
5963 }
5964
5965 *iter = next;
5966 *iter_type = next_type;
5967 return next_type ? LY_SUCCESS : LY_ENOTFOUND;
5968}
5969
5970/**
5971 * @brief Move context @p set to a node. Result is LYXP_SET_NODE_SET. Context position aware.
Michal Vasko03ff5a72019-09-11 13:49:33 +02005972 *
5973 * @param[in,out] set Set to use.
Michal Vasko5d24f6c2020-10-13 13:49:06 +02005974 * @param[in] moveto_mod Matching node module, NULL for no prefix.
Michal Vaskod3678892020-05-21 10:06:58 +02005975 * @param[in] ncname Matching node name in the dictionary, NULL for any.
Michal Vasko49fec8e2022-05-24 10:28:33 +02005976 * @param[in] axis Axis to search on.
Michal Vaskocdad7122020-11-09 21:04:44 +01005977 * @param[in] options XPath options.
Michal Vasko03ff5a72019-09-11 13:49:33 +02005978 * @return LY_ERR (LY_EINCOMPLETE on unresolved when)
5979 */
5980static LY_ERR
Michal Vasko49fec8e2022-05-24 10:28:33 +02005981moveto_node(struct lyxp_set *set, const struct lys_module *moveto_mod, const char *ncname, enum lyxp_axis axis,
5982 uint32_t options)
Michal Vasko03ff5a72019-09-11 13:49:33 +02005983{
Michal Vasko230cf972021-12-02 12:31:00 +01005984 LY_ERR r, rc = LY_SUCCESS;
Michal Vasko49fec8e2022-05-24 10:28:33 +02005985 const struct lyd_node *iter;
5986 enum lyxp_node_type iter_type;
Michal Vasko230cf972021-12-02 12:31:00 +01005987 struct lyxp_set result;
Michal Vasko49fec8e2022-05-24 10:28:33 +02005988 uint32_t i;
Michal Vasko03ff5a72019-09-11 13:49:33 +02005989
aPiecek8b0cc152021-05-31 16:40:31 +02005990 if (options & LYXP_SKIP_EXPR) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02005991 return LY_SUCCESS;
5992 }
5993
5994 if (set->type != LYXP_SET_NODE_SET) {
Radek Krejci2efc45b2020-12-22 16:25:44 +01005995 LOGVAL(set->ctx, LY_VCODE_XP_INOP_1, "path operator", print_set_type(set));
Michal Vasko03ff5a72019-09-11 13:49:33 +02005996 return LY_EVALID;
5997 }
5998
Michal Vasko230cf972021-12-02 12:31:00 +01005999 /* init result set */
6000 set_init(&result, set);
Michal Vasko03ff5a72019-09-11 13:49:33 +02006001
Michal Vasko49fec8e2022-05-24 10:28:33 +02006002 for (i = 0; i < set->used; ++i) {
6003 /* iterate over all the nodes on the axis of the node */
6004 iter = NULL;
6005 iter_type = 0;
6006 while (!moveto_axis_node_next(&iter, &iter_type, set->val.nodes[i].node, set->val.nodes[i].type, axis, set)) {
6007 r = moveto_node_check(iter, iter_type, set, ncname, moveto_mod, options);
6008 if (r == LY_EINCOMPLETE) {
Michal Vasko230cf972021-12-02 12:31:00 +01006009 rc = r;
6010 goto cleanup;
Michal Vasko49fec8e2022-05-24 10:28:33 +02006011 } else if (r) {
6012 continue;
Michal Vasko03ff5a72019-09-11 13:49:33 +02006013 }
Michal Vasko49fec8e2022-05-24 10:28:33 +02006014
6015 /* check for duplicates if they are possible */
6016 switch (axis) {
6017 case LYXP_AXIS_ANCESTOR:
6018 case LYXP_AXIS_ANCESTOR_OR_SELF:
6019 case LYXP_AXIS_DESCENDANT:
6020 case LYXP_AXIS_DESCENDANT_OR_SELF:
6021 case LYXP_AXIS_FOLLOWING:
6022 case LYXP_AXIS_FOLLOWING_SIBLING:
6023 case LYXP_AXIS_PARENT:
6024 case LYXP_AXIS_PRECEDING:
6025 case LYXP_AXIS_PRECEDING_SIBLING:
6026 if (set_dup_node_check(&result, iter, iter_type, -1)) {
6027 continue;
6028 }
6029 break;
6030 case LYXP_AXIS_CHILD:
6031 case LYXP_AXIS_SELF:
6032 break;
6033 case LYXP_AXIS_ATTRIBUTE:
6034 /* handled specially */
6035 assert(0);
6036 LOGINT(set->ctx);
6037 break;
6038 }
6039
6040 /* matching node */
6041 set_insert_node(&result, iter, 0, iter_type, result.used);
Michal Vasko03ff5a72019-09-11 13:49:33 +02006042 }
Michal Vasko03ff5a72019-09-11 13:49:33 +02006043 }
Michal Vasko03ff5a72019-09-11 13:49:33 +02006044
Michal Vasko230cf972021-12-02 12:31:00 +01006045 /* move result to the set */
6046 lyxp_set_free_content(set);
6047 *set = result;
6048 result.type = LYXP_SET_NUMBER;
Michal Vasko49fec8e2022-05-24 10:28:33 +02006049
6050 /* sort the final set */
6051 set_sort(set);
Michal Vasko230cf972021-12-02 12:31:00 +01006052
6053cleanup:
6054 lyxp_set_free_content(&result);
6055 return rc;
Michal Vasko03ff5a72019-09-11 13:49:33 +02006056}
6057
6058/**
Michal Vasko49fec8e2022-05-24 10:28:33 +02006059 * @brief Move context @p set to child nodes using hashes. Result is LYXP_SET_NODE_SET. Context position aware.
Michal Vasko03ff5a72019-09-11 13:49:33 +02006060 *
6061 * @param[in,out] set Set to use.
Michal Vaskod3678892020-05-21 10:06:58 +02006062 * @param[in] scnode Matching node schema.
Michal Vasko004d3152020-06-11 19:59:22 +02006063 * @param[in] predicates If @p scnode is ::LYS_LIST or ::LYS_LEAFLIST, the predicates specifying a single instance.
Michal Vaskocdad7122020-11-09 21:04:44 +01006064 * @param[in] options XPath options.
Michal Vaskod3678892020-05-21 10:06:58 +02006065 * @return LY_ERR (LY_EINCOMPLETE on unresolved when)
6066 */
6067static LY_ERR
Michal Vasko49fec8e2022-05-24 10:28:33 +02006068moveto_node_hash_child(struct lyxp_set *set, const struct lysc_node *scnode, const struct ly_path_predicate *predicates,
Michal Vaskocdad7122020-11-09 21:04:44 +01006069 uint32_t options)
Michal Vaskod3678892020-05-21 10:06:58 +02006070{
Michal Vaskoddd76592022-01-17 13:34:48 +01006071 LY_ERR ret = LY_SUCCESS, r;
Michal Vaskod3678892020-05-21 10:06:58 +02006072 uint32_t i;
Michal Vaskod3678892020-05-21 10:06:58 +02006073 const struct lyd_node *siblings;
Michal Vasko230cf972021-12-02 12:31:00 +01006074 struct lyxp_set result;
Michal Vasko004d3152020-06-11 19:59:22 +02006075 struct lyd_node *sub, *inst = NULL;
Michal Vaskod3678892020-05-21 10:06:58 +02006076
Michal Vasko004d3152020-06-11 19:59:22 +02006077 assert(scnode && (!(scnode->nodetype & (LYS_LIST | LYS_LEAFLIST)) || predicates));
Michal Vaskod3678892020-05-21 10:06:58 +02006078
Michal Vasko50aaa072021-12-02 13:11:56 +01006079 /* init result set */
6080 set_init(&result, set);
6081
aPiecek8b0cc152021-05-31 16:40:31 +02006082 if (options & LYXP_SKIP_EXPR) {
Michal Vasko004d3152020-06-11 19:59:22 +02006083 goto cleanup;
Michal Vaskod3678892020-05-21 10:06:58 +02006084 }
6085
6086 if (set->type != LYXP_SET_NODE_SET) {
Radek Krejci2efc45b2020-12-22 16:25:44 +01006087 LOGVAL(set->ctx, LY_VCODE_XP_INOP_1, "path operator", print_set_type(set));
Michal Vasko004d3152020-06-11 19:59:22 +02006088 ret = LY_EVALID;
6089 goto cleanup;
Michal Vaskod3678892020-05-21 10:06:58 +02006090 }
6091
6092 /* context check for all the nodes since we have the schema node */
6093 if ((set->root_type == LYXP_NODE_ROOT_CONFIG) && (scnode->flags & LYS_CONFIG_R)) {
6094 lyxp_set_free_content(set);
Michal Vasko004d3152020-06-11 19:59:22 +02006095 goto cleanup;
Michal Vasko69730152020-10-09 16:30:07 +02006096 } else if (set->context_op && (scnode->nodetype & (LYS_RPC | LYS_ACTION | LYS_NOTIF)) &&
6097 (scnode != set->context_op)) {
Michal Vasko6b26e742020-07-17 15:02:10 +02006098 lyxp_set_free_content(set);
6099 goto cleanup;
Michal Vasko004d3152020-06-11 19:59:22 +02006100 }
6101
6102 /* create specific data instance if needed */
6103 if (scnode->nodetype == LYS_LIST) {
6104 LY_CHECK_GOTO(ret = lyd_create_list(scnode, predicates, &inst), cleanup);
6105 } else if (scnode->nodetype == LYS_LEAFLIST) {
6106 LY_CHECK_GOTO(ret = lyd_create_term2(scnode, &predicates[0].value, &inst), cleanup);
Michal Vaskod3678892020-05-21 10:06:58 +02006107 }
6108
Michal Vasko230cf972021-12-02 12:31:00 +01006109 for (i = 0; i < set->used; ++i) {
Michal Vaskod3678892020-05-21 10:06:58 +02006110 siblings = NULL;
6111
6112 if ((set->val.nodes[i].type == LYXP_NODE_ROOT_CONFIG) || (set->val.nodes[i].type == LYXP_NODE_ROOT)) {
6113 assert(!set->val.nodes[i].node);
6114
6115 /* search in all the trees */
6116 siblings = set->tree;
6117 } else if (set->val.nodes[i].type == LYXP_NODE_ELEM) {
6118 /* search in children */
Radek Krejcia1c1e542020-09-29 16:06:52 +02006119 siblings = lyd_child(set->val.nodes[i].node);
Michal Vaskod3678892020-05-21 10:06:58 +02006120 }
6121
6122 /* find the node using hashes */
Michal Vasko004d3152020-06-11 19:59:22 +02006123 if (inst) {
Michal Vaskoddd76592022-01-17 13:34:48 +01006124 r = lyd_find_sibling_first(siblings, inst, &sub);
Michal Vaskod3678892020-05-21 10:06:58 +02006125 } else {
Michal Vaskoddd76592022-01-17 13:34:48 +01006126 r = lyd_find_sibling_val(siblings, scnode, NULL, 0, &sub);
Michal Vaskod3678892020-05-21 10:06:58 +02006127 }
Michal Vaskoddd76592022-01-17 13:34:48 +01006128 LY_CHECK_ERR_GOTO(r && (r != LY_ENOTFOUND), ret = r, cleanup);
Michal Vaskod3678892020-05-21 10:06:58 +02006129
6130 /* when check */
Michal Vaskod5cfa6e2020-11-23 16:56:08 +01006131 if (!(options & LYXP_IGNORE_WHEN) && sub && lysc_has_when(sub->schema) && !(sub->flags & LYD_WHEN_TRUE)) {
Michal Vasko004d3152020-06-11 19:59:22 +02006132 ret = LY_EINCOMPLETE;
6133 goto cleanup;
Michal Vaskod3678892020-05-21 10:06:58 +02006134 }
6135
6136 if (sub) {
6137 /* pos filled later */
Michal Vasko230cf972021-12-02 12:31:00 +01006138 set_insert_node(&result, sub, 0, LYXP_NODE_ELEM, result.used);
Michal Vaskod3678892020-05-21 10:06:58 +02006139 }
6140 }
6141
Michal Vasko230cf972021-12-02 12:31:00 +01006142 /* move result to the set */
6143 lyxp_set_free_content(set);
6144 *set = result;
6145 result.type = LYXP_SET_NUMBER;
6146 assert(!set_sort(set));
6147
Michal Vasko004d3152020-06-11 19:59:22 +02006148cleanup:
Michal Vasko230cf972021-12-02 12:31:00 +01006149 lyxp_set_free_content(&result);
Michal Vasko004d3152020-06-11 19:59:22 +02006150 lyd_free_tree(inst);
6151 return ret;
Michal Vaskod3678892020-05-21 10:06:58 +02006152}
6153
6154/**
Michal Vasko49fec8e2022-05-24 10:28:33 +02006155 * @brief Check @p node as a part of schema NameTest processing.
6156 *
6157 * @param[in] node Schema node to check.
6158 * @param[in] ctx_scnode Context node.
6159 * @param[in] set Set to read general context from.
6160 * @param[in] node_name Node name in the dictionary to move to, NULL for any nodes.
6161 * @param[in] moveto_mod Expected module of the node, NULL for no prefix.
6162 * @return LY_ERR (LY_ENOT if node does not match, LY_EINVAL if neither node nor any children match)
6163 */
6164static LY_ERR
6165moveto_scnode_check(const struct lysc_node *node, const struct lysc_node *ctx_scnode, const struct lyxp_set *set,
6166 const char *node_name, const struct lys_module *moveto_mod)
6167{
6168 if (!moveto_mod && node_name) {
6169 switch (set->format) {
6170 case LY_VALUE_SCHEMA:
6171 case LY_VALUE_SCHEMA_RESOLVED:
6172 /* use current module */
6173 moveto_mod = set->cur_mod;
6174 break;
6175 case LY_VALUE_JSON:
6176 case LY_VALUE_LYB:
6177 case LY_VALUE_STR_NS:
6178 /* inherit module of the context node, if any */
6179 if (ctx_scnode) {
6180 moveto_mod = ctx_scnode->module;
6181 }
6182 break;
6183 case LY_VALUE_CANON:
6184 case LY_VALUE_XML:
6185 /* not defined */
6186 LOGINT(set->ctx);
6187 return LY_EINVAL;
6188 }
6189 }
6190
6191 if (!node) {
6192 /* root will not match a specific node */
6193 if (node_name || moveto_mod) {
6194 return LY_ENOT;
6195 }
6196 return LY_SUCCESS;
6197 }
6198
6199 /* module check */
6200 if (moveto_mod && (node->module != moveto_mod)) {
6201 return LY_ENOT;
6202 }
6203
6204 /* context check */
6205 if ((set->root_type == LYXP_NODE_ROOT_CONFIG) && (node->flags & LYS_CONFIG_R)) {
6206 return LY_EINVAL;
6207 } else if (set->context_op && (node->nodetype & (LYS_RPC | LYS_ACTION | LYS_NOTIF)) && (node != set->context_op)) {
6208 return LY_EINVAL;
6209 }
6210
6211 /* name check */
6212 if (node_name && (node->name != node_name)) {
6213 return LY_ENOT;
6214 }
6215
6216 /* match */
6217 return LY_SUCCESS;
6218}
6219
6220/**
6221 * @brief Get the next node in a forward schema node DFS.
6222 *
6223 * @param[in] iter Last returned node.
6224 * @param[in] stop Node to stop the search on and not return.
6225 * @param[in] getnext_opts Options for ::lys_getnext().
6226 * @return Next node, NULL if there are no more.
6227 */
6228static const struct lysc_node *
6229moveto_axis_scnode_next_dfs_forward(const struct lysc_node *iter, const struct lysc_node *stop, uint32_t getnext_opts)
6230{
6231 const struct lysc_node *next = NULL;
6232
6233 next = lysc_node_child(iter);
6234 if (!next) {
6235 /* no children, try siblings */
Michal Vasko34a22fe2022-06-15 07:58:55 +02006236 if ((iter == stop) || !lysc_data_parent(iter)) {
Michal Vasko49fec8e2022-05-24 10:28:33 +02006237 /* we are done, no next element to process */
6238 return NULL;
6239 }
6240
6241 next = lys_getnext(iter, lysc_data_parent(iter), NULL, getnext_opts);
6242 }
6243 while (!next && iter) {
6244 /* parent is already processed, go to its sibling */
6245 iter = iter->parent;
Michal Vasko34a22fe2022-06-15 07:58:55 +02006246 if ((iter == stop) || !lysc_data_parent(iter)) {
Michal Vasko49fec8e2022-05-24 10:28:33 +02006247 /* we are done, no next element to process */
6248 return NULL;
6249 }
6250 next = lys_getnext(iter, lysc_data_parent(iter), NULL, getnext_opts);
6251 }
6252
6253 return next;
6254}
6255
6256/**
6257 * @brief Consider schema node based on its in_ctx enum value.
6258 *
6259 * @param[in,out] in_ctx In_ctx enum of the schema node, may be updated.
6260 * @param[in] axis Axis to use.
6261 * @return LY_SUCCESS on success.
6262 * @return LY_ENOT if the node should not be returned.
6263 */
6264static LY_ERR
6265moveto_axis_scnode_next_in_ctx(int32_t *in_ctx, enum lyxp_axis axis)
6266{
6267 switch (axis) {
6268 case LYXP_AXIS_SELF:
6269 if ((*in_ctx == LYXP_SET_SCNODE_START) || (*in_ctx == LYXP_SET_SCNODE_ATOM_CTX)) {
6270 /* additionally put the start node into context */
6271 *in_ctx = LYXP_SET_SCNODE_ATOM_CTX;
6272 return LY_SUCCESS;
6273 }
6274 break;
6275 case LYXP_AXIS_PARENT:
6276 case LYXP_AXIS_ANCESTOR_OR_SELF:
6277 case LYXP_AXIS_ANCESTOR:
6278 case LYXP_AXIS_DESCENDANT_OR_SELF:
6279 case LYXP_AXIS_DESCENDANT:
6280 case LYXP_AXIS_FOLLOWING:
6281 case LYXP_AXIS_FOLLOWING_SIBLING:
6282 case LYXP_AXIS_PRECEDING:
6283 case LYXP_AXIS_PRECEDING_SIBLING:
6284 case LYXP_AXIS_CHILD:
6285 if (*in_ctx == LYXP_SET_SCNODE_START) {
6286 /* remember that context node was used */
6287 *in_ctx = LYXP_SET_SCNODE_START_USED;
6288 return LY_SUCCESS;
6289 } else if (*in_ctx == LYXP_SET_SCNODE_ATOM_CTX) {
6290 /* traversed */
6291 *in_ctx = LYXP_SET_SCNODE_ATOM_NODE;
6292 return LY_SUCCESS;
6293 }
6294 break;
6295 case LYXP_AXIS_ATTRIBUTE:
6296 /* unreachable */
6297 assert(0);
6298 LOGINT(NULL);
6299 break;
6300 }
6301
6302 return LY_ENOT;
6303}
6304
6305/**
6306 * @brief Get previous sibling for a schema node.
6307 *
6308 * @param[in] scnode Schema node.
6309 * @param[in] getnext_opts Options for ::lys_getnext().
6310 * @return Previous sibling, NULL if none.
6311 */
6312static const struct lysc_node *
6313moveto_axis_scnode_preceding_sibling(const struct lysc_node *scnode, uint32_t getnext_opts)
6314{
6315 const struct lysc_node *next = NULL, *prev = NULL;
6316
6317 while ((next = lys_getnext(next, lysc_data_parent(scnode), scnode->module->compiled, getnext_opts))) {
6318 if (next == scnode) {
6319 break;
6320 }
6321
6322 prev = next;
6323 }
6324
6325 return prev;
6326}
6327
6328/**
6329 * @brief Get the first schema node on an axis for a context node.
6330 *
6331 * @param[in,out] iter Last returned node, start with NULL, updated to the next node.
6332 * @param[in,out] iter_type Node type of @p iter, start with 0, updated to the node type of the next node.
6333 * @param[in,out] iter_mod Internal module iterator, do not change.
6334 * @param[in,out] iter_mod_idx Internal module index iterator, do not change.
6335 * @param[in] scnode Context node.
6336 * @param[in] node_type Type of @p scnode.
6337 * @param[in] in_ctx In_ctx enum of @p scnode.
6338 * @param[in] axis Axis to use.
6339 * @param[in] set XPath set with the general context.
6340 * @param[in] getnext_opts Options for ::lys_getnext().
6341 * @return LY_SUCCESS on success.
6342 * @return LY_ENOTFOUND if no next node found.
6343 */
6344static LY_ERR
6345moveto_axis_scnode_next_first(const struct lysc_node **iter, enum lyxp_node_type *iter_type, const struct lys_module **iter_mod,
6346 uint32_t *iter_mod_idx, const struct lysc_node *scnode, enum lyxp_node_type node_type, enum lyxp_axis axis,
6347 struct lyxp_set *set, uint32_t getnext_opts)
6348{
6349 const struct lysc_node *next = NULL;
6350 enum lyxp_node_type next_type = 0;
6351
6352 assert(!*iter);
6353 assert(!*iter_type);
6354
6355 *iter_mod = NULL;
6356 *iter_mod_idx = 0;
6357
6358 switch (axis) {
6359 case LYXP_AXIS_ANCESTOR_OR_SELF:
6360 case LYXP_AXIS_DESCENDANT_OR_SELF:
6361 case LYXP_AXIS_SELF:
6362 if ((node_type == LYXP_NODE_ROOT_CONFIG) || (node_type == LYXP_NODE_ROOT) || (node_type == LYXP_NODE_ELEM)) {
6363 /* just return the node */
6364 next = scnode;
6365 next_type = node_type;
6366 }
6367 break;
6368
6369 case LYXP_AXIS_ANCESTOR:
6370 case LYXP_AXIS_PARENT:
6371 if (node_type == LYXP_NODE_ELEM) {
6372 next = lysc_data_parent(scnode);
6373 next_type = next ? LYXP_NODE_ELEM : set->root_type;
6374 } /* else no parent */
6375 break;
6376
6377 case LYXP_AXIS_DESCENDANT:
6378 case LYXP_AXIS_CHILD:
6379 if ((node_type == LYXP_NODE_ROOT_CONFIG) || (node_type == LYXP_NODE_ROOT)) {
6380 /* it can actually be in any module, it's all <running>, and even if it's moveto_mod (if set),
6381 * it can be in a top-level augment */
6382 while ((*iter_mod = ly_ctx_get_module_iter(set->ctx, iter_mod_idx))) {
6383 /* module may not be implemented or not compiled yet */
6384 if (!(*iter_mod)->compiled) {
6385 continue;
6386 }
6387
6388 /* get next node */
6389 if ((next = lys_getnext(NULL, NULL, (*iter_mod)->compiled, getnext_opts))) {
6390 next_type = LYXP_NODE_ELEM;
6391 break;
6392 }
6393 }
6394 } else if (node_type == LYXP_NODE_ELEM) {
6395 /* get next node */
6396 next = lys_getnext(NULL, scnode, NULL, getnext_opts);
6397 next_type = next ? LYXP_NODE_ELEM : 0;
6398 }
6399 break;
6400
6401 case LYXP_AXIS_FOLLOWING:
6402 case LYXP_AXIS_FOLLOWING_SIBLING:
6403 if (node_type == LYXP_NODE_ELEM) {
6404 /* first next sibling */
6405 next = lys_getnext(scnode, lysc_data_parent(scnode), scnode->module->compiled, getnext_opts);
6406 next_type = next ? LYXP_NODE_ELEM : 0;
6407 } /* else no sibling */
6408 break;
6409
6410 case LYXP_AXIS_PRECEDING:
6411 case LYXP_AXIS_PRECEDING_SIBLING:
6412 if (node_type == LYXP_NODE_ELEM) {
6413 /* first parent sibling */
6414 next = lys_getnext(NULL, lysc_data_parent(scnode), scnode->module->compiled, getnext_opts);
6415 if (next == scnode) {
6416 /* no preceding sibling */
6417 next = NULL;
6418 }
6419 next_type = next ? LYXP_NODE_ELEM : 0;
6420 } /* else no sibling */
6421 break;
6422
6423 case LYXP_AXIS_ATTRIBUTE:
6424 /* unreachable */
6425 assert(0);
6426 LOGINT(set->ctx);
6427 break;
6428 }
6429
6430 *iter = next;
6431 *iter_type = next_type;
6432 return next_type ? LY_SUCCESS : LY_ENOTFOUND;
6433}
6434
6435/**
6436 * @brief Iterate over all schema nodes on an axis for a context node.
6437 *
6438 * @param[in,out] iter Last returned node, start with NULL, updated to the next node.
6439 * @param[in,out] iter_type Node type of @p iter, start with 0, updated to the node type of the next node.
6440 * @param[in,out] iter_mod Internal module iterator, do not change.
6441 * @param[in,out] iter_mod_idx Internal module index iterator, do not change.
6442 * @param[in] scnode Context node.
6443 * @param[in] node_type Type of @p scnode.
6444 * @param[in] axis Axis to use.
6445 * @param[in] set XPath set with the general context.
6446 * @param[in] getnext_opts Options for ::lys_getnext().
6447 * @return LY_SUCCESS on success.
6448 * @return LY_ENOTFOUND if no next node found.
6449 */
6450static LY_ERR
6451moveto_axis_scnode_next(const struct lysc_node **iter, enum lyxp_node_type *iter_type, const struct lys_module **iter_mod,
6452 uint32_t *iter_mod_idx, const struct lysc_node *scnode, enum lyxp_node_type node_type, enum lyxp_axis axis,
6453 struct lyxp_set *set, uint32_t getnext_opts)
6454{
6455 const struct lysc_node *next = NULL, *dfs_stop;
6456 enum lyxp_node_type next_type = 0;
6457
6458 if (!*iter_type) {
6459 /* first returned node */
6460 return moveto_axis_scnode_next_first(iter, iter_type, iter_mod, iter_mod_idx, scnode, node_type, axis, set,
6461 getnext_opts);
6462 }
6463
6464 switch (axis) {
6465 case LYXP_AXIS_PARENT:
6466 case LYXP_AXIS_SELF:
6467 /* parent/self was returned before */
6468 break;
6469
6470 case LYXP_AXIS_ANCESTOR_OR_SELF:
6471 if ((*iter == scnode) && (*iter_type == node_type)) {
6472 /* fake first ancestor, we returned self before */
6473 *iter = NULL;
6474 *iter_type = 0;
6475 return moveto_axis_scnode_next_first(iter, iter_type, iter_mod, iter_mod_idx, scnode, node_type,
6476 LYXP_AXIS_ANCESTOR, set, getnext_opts);
6477 } /* else continue ancestor */
6478
6479 /* fallthrough */
6480 case LYXP_AXIS_ANCESTOR:
6481 if (*iter_type == LYXP_NODE_ELEM) {
6482 next = lysc_data_parent(*iter);
6483 next_type = next ? LYXP_NODE_ELEM : set->root_type;
6484 } /* else no ancestor */
6485 break;
6486
6487 case LYXP_AXIS_DESCENDANT_OR_SELF:
6488 if ((*iter == scnode) && (*iter_type == node_type)) {
6489 /* fake first descendant, we returned self before */
6490 *iter = NULL;
6491 *iter_type = 0;
6492 return moveto_axis_scnode_next_first(iter, iter_type, iter_mod, iter_mod_idx, scnode, node_type,
6493 LYXP_AXIS_DESCENDANT, set, getnext_opts);
6494 } /* else DFS until context node */
6495 dfs_stop = scnode;
6496
6497 /* fallthrough */
6498 case LYXP_AXIS_DESCENDANT:
6499 if (axis == LYXP_AXIS_DESCENDANT) {
6500 /* DFS until the context node */
6501 dfs_stop = scnode;
6502 }
6503
6504 /* fallthrough */
6505 case LYXP_AXIS_PRECEDING:
6506 if (axis == LYXP_AXIS_PRECEDING) {
6507 /* DFS until the previous sibling */
6508 dfs_stop = moveto_axis_scnode_preceding_sibling(scnode, getnext_opts);
6509 assert(dfs_stop);
6510
6511 if (*iter == dfs_stop) {
6512 /* we are done */
6513 break;
6514 }
6515 }
6516
6517 /* fallthrough */
6518 case LYXP_AXIS_FOLLOWING:
6519 if (axis == LYXP_AXIS_FOLLOWING) {
6520 /* DFS through the whole module */
6521 dfs_stop = NULL;
6522 }
6523
6524 /* nested nodes */
6525 assert(*iter);
6526 next = moveto_axis_scnode_next_dfs_forward(*iter, dfs_stop, getnext_opts);
6527 if (next) {
6528 next_type = LYXP_NODE_ELEM;
6529 break;
6530 } /* else get next top-level node just like a child */
6531
6532 /* fallthrough */
6533 case LYXP_AXIS_CHILD:
6534 case LYXP_AXIS_FOLLOWING_SIBLING:
6535 if (!*iter_mod) {
6536 /* nodes from a single module */
6537 if ((next = lys_getnext(*iter, lysc_data_parent(*iter), (*iter)->module->compiled, getnext_opts))) {
6538 next_type = LYXP_NODE_ELEM;
6539 break;
6540 }
6541
6542 assert(scnode);
6543 if (!lysc_data_parent(scnode)) {
6544 /* iterating over top-level nodes, find next */
6545 while (lysc_data_parent(*iter)) {
6546 *iter = lysc_data_parent(*iter);
6547 }
6548 if ((next = lys_getnext(*iter, NULL, (*iter)->module->compiled, getnext_opts))) {
6549 next_type = LYXP_NODE_ELEM;
6550 break;
6551 }
6552 }
6553 }
6554
6555 while (*iter_mod) {
6556 /* module top-level nodes */
6557 if ((next = lys_getnext(*iter, NULL, (*iter_mod)->compiled, getnext_opts))) {
6558 next_type = LYXP_NODE_ELEM;
6559 break;
6560 }
6561
6562 /* get next module */
6563 while ((*iter_mod = ly_ctx_get_module_iter(set->ctx, iter_mod_idx))) {
6564 /* module may not be implemented or not compiled yet */
6565 if ((*iter_mod)->compiled) {
6566 break;
6567 }
6568 }
6569
6570 /* new module, start over */
6571 *iter = NULL;
6572 }
6573 break;
6574
6575 case LYXP_AXIS_PRECEDING_SIBLING:
6576 assert(*iter);
6577
6578 /* next parent sibling until scnode */
6579 next = lys_getnext(*iter, lysc_data_parent(*iter), (*iter)->module->compiled, getnext_opts);
6580 if (next == scnode) {
6581 /* no previous sibling */
6582 next = NULL;
6583 }
6584 next_type = next ? LYXP_NODE_ELEM : 0;
6585 break;
6586
6587 case LYXP_AXIS_ATTRIBUTE:
6588 /* unreachable */
6589 assert(0);
6590 LOGINT(set->ctx);
6591 break;
6592 }
6593
6594 *iter = next;
6595 *iter_type = next_type;
6596 return next_type ? LY_SUCCESS : LY_ENOTFOUND;
6597}
6598
6599/**
Michal Vaskod3678892020-05-21 10:06:58 +02006600 * @brief Move context @p set to a schema node. Result is LYXP_SET_SCNODE_SET (or LYXP_SET_EMPTY).
6601 *
6602 * @param[in,out] set Set to use.
Michal Vasko5d24f6c2020-10-13 13:49:06 +02006603 * @param[in] moveto_mod Matching node module, NULL for no prefix.
Michal Vaskod3678892020-05-21 10:06:58 +02006604 * @param[in] ncname Matching node name in the dictionary, NULL for any.
Michal Vasko49fec8e2022-05-24 10:28:33 +02006605 * @param[in] axis Axis to search on.
Michal Vasko03ff5a72019-09-11 13:49:33 +02006606 * @param[in] options XPath options.
6607 * @return LY_ERR
6608 */
6609static LY_ERR
Michal Vasko49fec8e2022-05-24 10:28:33 +02006610moveto_scnode(struct lyxp_set *set, const struct lys_module *moveto_mod, const char *ncname, enum lyxp_axis axis,
6611 uint32_t options)
Michal Vasko03ff5a72019-09-11 13:49:33 +02006612{
Radek Krejci857189e2020-09-01 13:26:36 +02006613 ly_bool temp_ctx = 0;
Michal Vasko49fec8e2022-05-24 10:28:33 +02006614 uint32_t getnext_opts, orig_used, i, mod_idx, idx;
Michal Vasko5d24f6c2020-10-13 13:49:06 +02006615 const struct lys_module *mod;
Michal Vasko49fec8e2022-05-24 10:28:33 +02006616 const struct lysc_node *iter;
6617 enum lyxp_node_type iter_type;
Michal Vasko03ff5a72019-09-11 13:49:33 +02006618
aPiecek8b0cc152021-05-31 16:40:31 +02006619 if (options & LYXP_SKIP_EXPR) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02006620 return LY_SUCCESS;
6621 }
6622
6623 if (set->type != LYXP_SET_SCNODE_SET) {
Radek Krejci2efc45b2020-12-22 16:25:44 +01006624 LOGVAL(set->ctx, LY_VCODE_XP_INOP_1, "path operator", print_set_type(set));
Michal Vasko03ff5a72019-09-11 13:49:33 +02006625 return LY_EVALID;
6626 }
6627
Michal Vaskocafad9d2019-11-07 15:20:03 +01006628 /* getnext opts */
Michal Vasko7b1ad1a2020-11-02 15:41:27 +01006629 getnext_opts = 0;
Michal Vaskocafad9d2019-11-07 15:20:03 +01006630 if (options & LYXP_SCNODE_OUTPUT) {
6631 getnext_opts |= LYS_GETNEXT_OUTPUT;
6632 }
6633
Michal Vasko03ff5a72019-09-11 13:49:33 +02006634 orig_used = set->used;
6635 for (i = 0; i < orig_used; ++i) {
Michal Vasko49fec8e2022-05-24 10:28:33 +02006636 /* update in_ctx first */
6637 if (moveto_axis_scnode_next_in_ctx(&set->val.scnodes[i].in_ctx, axis)) {
6638 /* not usable, skip */
6639 continue;
6640 }
Radek Krejciaa6b53f2020-08-27 15:19:03 +02006641
Michal Vasko49fec8e2022-05-24 10:28:33 +02006642 iter = NULL;
6643 iter_type = 0;
6644 while (!moveto_axis_scnode_next(&iter, &iter_type, &mod, &mod_idx, set->val.scnodes[i].scnode,
6645 set->val.scnodes[i].type, axis, set, getnext_opts)) {
6646 if (moveto_scnode_check(iter, NULL, set, ncname, moveto_mod)) {
Michal Vasko5c4e5892019-11-14 12:31:38 +01006647 continue;
6648 }
6649
Michal Vasko49fec8e2022-05-24 10:28:33 +02006650 /* insert */
6651 LY_CHECK_RET(lyxp_set_scnode_insert_node(set, iter, iter_type, &idx));
Michal Vasko03ff5a72019-09-11 13:49:33 +02006652
Michal Vasko49fec8e2022-05-24 10:28:33 +02006653 /* we need to prevent these nodes from being considered in this moveto */
6654 if ((idx < orig_used) && (idx > i)) {
6655 set->val.scnodes[idx].in_ctx = LYXP_SET_SCNODE_ATOM_NEW_CTX;
6656 temp_ctx = 1;
Michal Vasko03ff5a72019-09-11 13:49:33 +02006657 }
6658 }
6659 }
Michal Vasko03ff5a72019-09-11 13:49:33 +02006660
6661 /* correct temporary in_ctx values */
6662 if (temp_ctx) {
6663 for (i = 0; i < orig_used; ++i) {
Radek Krejcif13b87b2020-12-01 22:02:17 +01006664 if (set->val.scnodes[i].in_ctx == LYXP_SET_SCNODE_ATOM_NEW_CTX) {
6665 set->val.scnodes[i].in_ctx = LYXP_SET_SCNODE_ATOM_CTX;
Michal Vasko03ff5a72019-09-11 13:49:33 +02006666 }
6667 }
6668 }
6669
6670 return LY_SUCCESS;
6671}
6672
6673/**
Michal Vasko49fec8e2022-05-24 10:28:33 +02006674 * @brief Move context @p set to a child node and all its descendants. Result is LYXP_SET_NODE_SET.
Michal Vasko03ff5a72019-09-11 13:49:33 +02006675 * Context position aware.
6676 *
6677 * @param[in] set Set to use.
Michal Vasko5d24f6c2020-10-13 13:49:06 +02006678 * @param[in] moveto_mod Matching node module, NULL for no prefix.
Michal Vaskod3678892020-05-21 10:06:58 +02006679 * @param[in] ncname Matching node name in the dictionary, NULL for any.
Michal Vaskocdad7122020-11-09 21:04:44 +01006680 * @param[in] options XPath options.
Michal Vasko03ff5a72019-09-11 13:49:33 +02006681 * @return LY_ERR (LY_EINCOMPLETE on unresolved when)
6682 */
6683static LY_ERR
Michal Vasko49fec8e2022-05-24 10:28:33 +02006684moveto_node_alldesc_child(struct lyxp_set *set, const struct lys_module *moveto_mod, const char *ncname, uint32_t options)
Michal Vasko03ff5a72019-09-11 13:49:33 +02006685{
6686 uint32_t i;
Michal Vasko03ff5a72019-09-11 13:49:33 +02006687 const struct lyd_node *next, *elem, *start;
Michal Vasko03ff5a72019-09-11 13:49:33 +02006688 struct lyxp_set ret_set;
6689 LY_ERR rc;
6690
aPiecek8b0cc152021-05-31 16:40:31 +02006691 if (options & LYXP_SKIP_EXPR) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02006692 return LY_SUCCESS;
6693 }
6694
6695 if (set->type != LYXP_SET_NODE_SET) {
Radek Krejci2efc45b2020-12-22 16:25:44 +01006696 LOGVAL(set->ctx, LY_VCODE_XP_INOP_1, "path operator", print_set_type(set));
Michal Vasko03ff5a72019-09-11 13:49:33 +02006697 return LY_EVALID;
6698 }
6699
Michal Vasko9f96a052020-03-10 09:41:45 +01006700 /* replace the original nodes (and throws away all text and meta nodes, root is replaced by a child) */
Michal Vasko49fec8e2022-05-24 10:28:33 +02006701 rc = xpath_pi_node(set, LYXP_AXIS_CHILD, options);
Michal Vasko03ff5a72019-09-11 13:49:33 +02006702 LY_CHECK_RET(rc);
6703
Michal Vasko6346ece2019-09-24 13:12:53 +02006704 /* this loop traverses all the nodes in the set and adds/keeps only those that match qname */
Michal Vasko03ff5a72019-09-11 13:49:33 +02006705 set_init(&ret_set, set);
6706 for (i = 0; i < set->used; ++i) {
6707
6708 /* TREE DFS */
6709 start = set->val.nodes[i].node;
6710 for (elem = next = start; elem; elem = next) {
Michal Vasko49fec8e2022-05-24 10:28:33 +02006711 rc = moveto_node_check(elem, LYXP_NODE_ELEM, set, ncname, moveto_mod, options);
Michal Vasko6346ece2019-09-24 13:12:53 +02006712 if (!rc) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02006713 /* add matching node into result set */
6714 set_insert_node(&ret_set, elem, 0, LYXP_NODE_ELEM, ret_set.used);
6715 if (set_dup_node_check(set, elem, LYXP_NODE_ELEM, i)) {
6716 /* the node is a duplicate, we'll process it later in the set */
6717 goto skip_children;
6718 }
Michal Vasko6346ece2019-09-24 13:12:53 +02006719 } else if (rc == LY_EINCOMPLETE) {
Michal Vasko6346ece2019-09-24 13:12:53 +02006720 return rc;
6721 } else if (rc == LY_EINVAL) {
6722 goto skip_children;
Michal Vasko03ff5a72019-09-11 13:49:33 +02006723 }
6724
6725 /* TREE DFS NEXT ELEM */
6726 /* select element for the next run - children first */
Radek Krejcia1c1e542020-09-29 16:06:52 +02006727 next = lyd_child(elem);
Michal Vasko03ff5a72019-09-11 13:49:33 +02006728 if (!next) {
6729skip_children:
6730 /* no children, so try siblings, but only if it's not the start,
6731 * that is considered to be the root and it's siblings are not traversed */
6732 if (elem != start) {
6733 next = elem->next;
6734 } else {
6735 break;
6736 }
6737 }
6738 while (!next) {
6739 /* no siblings, go back through the parents */
Michal Vasko9e685082021-01-29 14:49:09 +01006740 if (lyd_parent(elem) == start) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02006741 /* we are done, no next element to process */
6742 break;
6743 }
6744 /* parent is already processed, go to its sibling */
Michal Vasko9e685082021-01-29 14:49:09 +01006745 elem = lyd_parent(elem);
Michal Vasko03ff5a72019-09-11 13:49:33 +02006746 next = elem->next;
6747 }
6748 }
6749 }
6750
6751 /* make the temporary set the current one */
6752 ret_set.ctx_pos = set->ctx_pos;
6753 ret_set.ctx_size = set->ctx_size;
Michal Vaskod3678892020-05-21 10:06:58 +02006754 lyxp_set_free_content(set);
Michal Vasko03ff5a72019-09-11 13:49:33 +02006755 memcpy(set, &ret_set, sizeof *set);
6756
6757 return LY_SUCCESS;
6758}
6759
6760/**
Michal Vasko49fec8e2022-05-24 10:28:33 +02006761 * @brief Move context @p set to a child schema node and all its descendants starting from a node.
6762 * Result is LYXP_SET_NODE_SET.
6763 *
6764 * @param[in] set Set to use.
6765 * @param[in] start Start node whose subtree to add.
6766 * @param[in] start_idx Index of @p start in @p set.
6767 * @param[in] moveto_mod Matching node module, NULL for no prefix.
6768 * @param[in] ncname Matching node name in the dictionary, NULL for any.
6769 * @param[in] options XPath options.
6770 * @return LY_ERR value.
6771 */
6772static LY_ERR
6773moveto_scnode_dfs(struct lyxp_set *set, const struct lysc_node *start, uint32_t start_idx,
6774 const struct lys_module *moveto_mod, const char *ncname, uint32_t options)
6775{
6776 const struct lysc_node *next, *elem;
6777 uint32_t idx;
6778 LY_ERR rc;
6779
6780 /* TREE DFS */
6781 for (elem = next = start; elem; elem = next) {
6782 if ((elem == start) || (elem->nodetype & (LYS_CHOICE | LYS_CASE))) {
6783 /* schema-only nodes, skip root */
6784 goto next_iter;
6785 }
6786
6787 rc = moveto_scnode_check(elem, start, set, ncname, moveto_mod);
6788 if (!rc) {
6789 if (lyxp_set_scnode_contains(set, elem, LYXP_NODE_ELEM, start_idx, &idx)) {
6790 set->val.scnodes[idx].in_ctx = LYXP_SET_SCNODE_ATOM_CTX;
6791 if (idx > start_idx) {
6792 /* we will process it later in the set */
6793 goto skip_children;
6794 }
6795 } else {
6796 LY_CHECK_RET(lyxp_set_scnode_insert_node(set, elem, LYXP_NODE_ELEM, NULL));
6797 }
6798 } else if (rc == LY_EINVAL) {
6799 goto skip_children;
6800 }
6801
6802next_iter:
6803 /* TREE DFS NEXT ELEM */
6804 /* select element for the next run - children first */
6805 next = lysc_node_child(elem);
6806 if (next && (next->nodetype == LYS_INPUT) && (options & LYXP_SCNODE_OUTPUT)) {
6807 next = next->next;
6808 } else if (next && (next->nodetype == LYS_OUTPUT) && !(options & LYXP_SCNODE_OUTPUT)) {
6809 next = next->next;
6810 }
6811 if (!next) {
6812skip_children:
6813 /* no children, so try siblings, but only if it's not the start,
6814 * that is considered to be the root and it's siblings are not traversed */
6815 if (elem != start) {
6816 next = elem->next;
6817 } else {
6818 break;
6819 }
6820 }
6821 while (!next) {
6822 /* no siblings, go back through the parents */
6823 if (elem->parent == start) {
6824 /* we are done, no next element to process */
6825 break;
6826 }
6827 /* parent is already processed, go to its sibling */
6828 elem = elem->parent;
6829 next = elem->next;
6830 }
6831 }
6832
6833 return LY_SUCCESS;
6834}
6835
6836/**
6837 * @brief Move context @p set to a child schema node and all its descendants. Result is LYXP_SET_NODE_SET.
Michal Vasko03ff5a72019-09-11 13:49:33 +02006838 *
6839 * @param[in] set Set to use.
Michal Vasko5d24f6c2020-10-13 13:49:06 +02006840 * @param[in] moveto_mod Matching node module, NULL for no prefix.
Michal Vaskod3678892020-05-21 10:06:58 +02006841 * @param[in] ncname Matching node name in the dictionary, NULL for any.
Michal Vasko03ff5a72019-09-11 13:49:33 +02006842 * @param[in] options XPath options.
Michal Vasko49fec8e2022-05-24 10:28:33 +02006843 * @return LY_ERR value.
Michal Vasko03ff5a72019-09-11 13:49:33 +02006844 */
6845static LY_ERR
Michal Vasko49fec8e2022-05-24 10:28:33 +02006846moveto_scnode_alldesc_child(struct lyxp_set *set, const struct lys_module *moveto_mod, const char *ncname, uint32_t options)
Michal Vasko03ff5a72019-09-11 13:49:33 +02006847{
Michal Vasko49fec8e2022-05-24 10:28:33 +02006848 uint32_t i, orig_used, mod_idx;
6849 const struct lys_module *mod;
6850 const struct lysc_node *root;
Michal Vasko03ff5a72019-09-11 13:49:33 +02006851
aPiecek8b0cc152021-05-31 16:40:31 +02006852 if (options & LYXP_SKIP_EXPR) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02006853 return LY_SUCCESS;
6854 }
6855
6856 if (set->type != LYXP_SET_SCNODE_SET) {
Radek Krejci2efc45b2020-12-22 16:25:44 +01006857 LOGVAL(set->ctx, LY_VCODE_XP_INOP_1, "path operator", print_set_type(set));
Michal Vasko03ff5a72019-09-11 13:49:33 +02006858 return LY_EVALID;
6859 }
6860
Michal Vasko03ff5a72019-09-11 13:49:33 +02006861 orig_used = set->used;
6862 for (i = 0; i < orig_used; ++i) {
Radek Krejcif13b87b2020-12-01 22:02:17 +01006863 if (set->val.scnodes[i].in_ctx != LYXP_SET_SCNODE_ATOM_CTX) {
6864 if (set->val.scnodes[i].in_ctx != LYXP_SET_SCNODE_START) {
Michal Vasko5c4e5892019-11-14 12:31:38 +01006865 continue;
6866 }
6867
6868 /* remember context node */
Radek Krejcif13b87b2020-12-01 22:02:17 +01006869 set->val.scnodes[i].in_ctx = LYXP_SET_SCNODE_START_USED;
Michal Vaskoec4df482019-12-16 10:02:18 +01006870 } else {
Michal Vasko1a09b212021-05-06 13:00:10 +02006871 set->val.scnodes[i].in_ctx = LYXP_SET_SCNODE_ATOM_NODE;
Michal Vasko03ff5a72019-09-11 13:49:33 +02006872 }
Michal Vasko03ff5a72019-09-11 13:49:33 +02006873
Michal Vasko49fec8e2022-05-24 10:28:33 +02006874 if ((set->val.scnodes[i].type == LYXP_NODE_ROOT_CONFIG) || (set->val.scnodes[i].type == LYXP_NODE_ROOT)) {
6875 /* traverse all top-level nodes in all the modules */
6876 mod_idx = 0;
6877 while ((mod = ly_ctx_get_module_iter(set->ctx, &mod_idx))) {
6878 /* module may not be implemented or not compiled yet */
6879 if (!mod->compiled) {
6880 continue;
Michal Vasko03ff5a72019-09-11 13:49:33 +02006881 }
Michal Vasko03ff5a72019-09-11 13:49:33 +02006882
Michal Vasko49fec8e2022-05-24 10:28:33 +02006883 root = NULL;
6884 /* no getnext opts needed */
6885 while ((root = lys_getnext(root, NULL, mod->compiled, 0))) {
6886 LY_CHECK_RET(moveto_scnode_dfs(set, root, i, moveto_mod, ncname, options));
Michal Vasko03ff5a72019-09-11 13:49:33 +02006887 }
6888 }
Michal Vasko49fec8e2022-05-24 10:28:33 +02006889
6890 } else if (set->val.scnodes[i].type == LYXP_NODE_ELEM) {
6891 /* add all the descendants recursively */
6892 LY_CHECK_RET(moveto_scnode_dfs(set, set->val.scnodes[i].scnode, i, moveto_mod, ncname, options));
Michal Vasko03ff5a72019-09-11 13:49:33 +02006893 }
6894 }
6895
6896 return LY_SUCCESS;
6897}
6898
6899/**
Michal Vasko61ac2f62020-05-25 12:39:51 +02006900 * @brief Move context @p set to an attribute. Result is LYXP_SET_NODE_SET.
Michal Vasko03ff5a72019-09-11 13:49:33 +02006901 * Indirectly context position aware.
6902 *
6903 * @param[in,out] set Set to use.
Michal Vaskod3678892020-05-21 10:06:58 +02006904 * @param[in] mod Matching metadata module, NULL for any.
6905 * @param[in] ncname Matching metadata name in the dictionary, NULL for any.
aPiecek8b0cc152021-05-31 16:40:31 +02006906 * @param[in] options XPath options.
Michal Vasko03ff5a72019-09-11 13:49:33 +02006907 * @return LY_ERR
6908 */
6909static LY_ERR
aPiecek8b0cc152021-05-31 16:40:31 +02006910moveto_attr(struct lyxp_set *set, const struct lys_module *mod, const char *ncname, uint32_t options)
Michal Vasko03ff5a72019-09-11 13:49:33 +02006911{
Michal Vasko9f96a052020-03-10 09:41:45 +01006912 struct lyd_meta *sub;
Michal Vasko03ff5a72019-09-11 13:49:33 +02006913
aPiecek8b0cc152021-05-31 16:40:31 +02006914 if (options & LYXP_SKIP_EXPR) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02006915 return LY_SUCCESS;
6916 }
6917
6918 if (set->type != LYXP_SET_NODE_SET) {
Radek Krejci2efc45b2020-12-22 16:25:44 +01006919 LOGVAL(set->ctx, LY_VCODE_XP_INOP_1, "path operator", print_set_type(set));
Michal Vasko03ff5a72019-09-11 13:49:33 +02006920 return LY_EVALID;
6921 }
6922
Radek Krejci1deb5be2020-08-26 16:43:36 +02006923 for (uint32_t i = 0; i < set->used; ) {
Radek Krejci857189e2020-09-01 13:26:36 +02006924 ly_bool replaced = 0;
Michal Vasko03ff5a72019-09-11 13:49:33 +02006925
6926 /* only attributes of an elem (not dummy) can be in the result, skip all the rest;
6927 * our attributes are always qualified */
Michal Vasko5c4e5892019-11-14 12:31:38 +01006928 if (set->val.nodes[i].type == LYXP_NODE_ELEM) {
Michal Vasko9f96a052020-03-10 09:41:45 +01006929 for (sub = set->val.nodes[i].node->meta; sub; sub = sub->next) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02006930
6931 /* check "namespace" */
Michal Vaskod3678892020-05-21 10:06:58 +02006932 if (mod && (sub->annotation->module != mod)) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02006933 continue;
6934 }
6935
Michal Vaskod3678892020-05-21 10:06:58 +02006936 if (!ncname || (sub->name == ncname)) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02006937 /* match */
6938 if (!replaced) {
Michal Vasko9f96a052020-03-10 09:41:45 +01006939 set->val.meta[i].meta = sub;
6940 set->val.meta[i].type = LYXP_NODE_META;
Michal Vasko03ff5a72019-09-11 13:49:33 +02006941 /* pos does not change */
6942 replaced = 1;
6943 } else {
Michal Vasko9f96a052020-03-10 09:41:45 +01006944 set_insert_node(set, (struct lyd_node *)sub, set->val.nodes[i].pos, LYXP_NODE_META, i + 1);
Michal Vasko03ff5a72019-09-11 13:49:33 +02006945 }
6946 ++i;
6947 }
6948 }
6949 }
6950
6951 if (!replaced) {
6952 /* no match */
6953 set_remove_node(set, i);
6954 }
6955 }
6956
6957 return LY_SUCCESS;
6958}
6959
6960/**
6961 * @brief Move context @p set1 to union with @p set2. @p set2 is emptied afterwards.
Michal Vasko61ac2f62020-05-25 12:39:51 +02006962 * Result is LYXP_SET_NODE_SET. Context position aware.
Michal Vasko03ff5a72019-09-11 13:49:33 +02006963 *
6964 * @param[in,out] set1 Set to use for the result.
6965 * @param[in] set2 Set that is copied to @p set1.
Michal Vasko03ff5a72019-09-11 13:49:33 +02006966 * @return LY_ERR
6967 */
6968static LY_ERR
Michal Vasko5e0e6eb2019-11-06 15:47:50 +01006969moveto_union(struct lyxp_set *set1, struct lyxp_set *set2)
Michal Vasko03ff5a72019-09-11 13:49:33 +02006970{
6971 LY_ERR rc;
6972
Michal Vaskod3678892020-05-21 10:06:58 +02006973 if ((set1->type != LYXP_SET_NODE_SET) || (set2->type != LYXP_SET_NODE_SET)) {
Radek Krejci2efc45b2020-12-22 16:25:44 +01006974 LOGVAL(set1->ctx, LY_VCODE_XP_INOP_2, "union", print_set_type(set1), print_set_type(set2));
Michal Vasko03ff5a72019-09-11 13:49:33 +02006975 return LY_EVALID;
6976 }
6977
6978 /* set2 is empty or both set1 and set2 */
Michal Vaskod3678892020-05-21 10:06:58 +02006979 if (!set2->used) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02006980 return LY_SUCCESS;
6981 }
6982
Michal Vaskod3678892020-05-21 10:06:58 +02006983 if (!set1->used) {
aPiecekadc1e4f2021-10-07 11:15:12 +02006984 /* release hidden allocated data (lyxp_set.size) */
6985 lyxp_set_free_content(set1);
6986 /* direct copying of the entire structure */
Michal Vasko03ff5a72019-09-11 13:49:33 +02006987 memcpy(set1, set2, sizeof *set1);
6988 /* dynamic memory belongs to set1 now, do not free */
Michal Vaskod3678892020-05-21 10:06:58 +02006989 memset(set2, 0, sizeof *set2);
Michal Vasko03ff5a72019-09-11 13:49:33 +02006990 return LY_SUCCESS;
6991 }
6992
6993 /* we assume sets are sorted */
Michal Vasko5e0e6eb2019-11-06 15:47:50 +01006994 assert(!set_sort(set1) && !set_sort(set2));
Michal Vasko03ff5a72019-09-11 13:49:33 +02006995
6996 /* sort, remove duplicates */
Michal Vasko5e0e6eb2019-11-06 15:47:50 +01006997 rc = set_sorted_merge(set1, set2);
Michal Vasko03ff5a72019-09-11 13:49:33 +02006998 LY_CHECK_RET(rc);
6999
7000 /* final set must be sorted */
Michal Vasko5e0e6eb2019-11-06 15:47:50 +01007001 assert(!set_sort(set1));
Michal Vasko03ff5a72019-09-11 13:49:33 +02007002
7003 return LY_SUCCESS;
7004}
7005
7006/**
Michal Vasko61ac2f62020-05-25 12:39:51 +02007007 * @brief Move context @p set to an attribute in any of the descendants. Result is LYXP_SET_NODE_SET.
Michal Vasko03ff5a72019-09-11 13:49:33 +02007008 * Context position aware.
7009 *
7010 * @param[in,out] set Set to use.
Michal Vaskod3678892020-05-21 10:06:58 +02007011 * @param[in] mod Matching metadata module, NULL for any.
7012 * @param[in] ncname Matching metadata name in the dictionary, NULL for any.
Michal Vaskocdad7122020-11-09 21:04:44 +01007013 * @param[in] options XPath options.
Michal Vasko03ff5a72019-09-11 13:49:33 +02007014 * @return LY_ERR (LY_EINCOMPLETE on unresolved when)
7015 */
7016static int
Michal Vaskocdad7122020-11-09 21:04:44 +01007017moveto_attr_alldesc(struct lyxp_set *set, const struct lys_module *mod, const char *ncname, uint32_t options)
Michal Vasko03ff5a72019-09-11 13:49:33 +02007018{
Michal Vasko9f96a052020-03-10 09:41:45 +01007019 struct lyd_meta *sub;
Michal Vasko03ff5a72019-09-11 13:49:33 +02007020 struct lyxp_set *set_all_desc = NULL;
7021 LY_ERR rc;
7022
aPiecek8b0cc152021-05-31 16:40:31 +02007023 if (options & LYXP_SKIP_EXPR) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02007024 return LY_SUCCESS;
7025 }
7026
7027 if (set->type != LYXP_SET_NODE_SET) {
Radek Krejci2efc45b2020-12-22 16:25:44 +01007028 LOGVAL(set->ctx, LY_VCODE_XP_INOP_1, "path operator", print_set_type(set));
Michal Vasko03ff5a72019-09-11 13:49:33 +02007029 return LY_EVALID;
7030 }
7031
Michal Vasko03ff5a72019-09-11 13:49:33 +02007032 /* can be optimized similarly to moveto_node_alldesc() and save considerable amount of memory,
7033 * but it likely won't be used much, so it's a waste of time */
7034 /* copy the context */
7035 set_all_desc = set_copy(set);
7036 /* get all descendant nodes (the original context nodes are removed) */
Michal Vasko49fec8e2022-05-24 10:28:33 +02007037 rc = moveto_node_alldesc_child(set_all_desc, NULL, NULL, options);
Michal Vasko03ff5a72019-09-11 13:49:33 +02007038 if (rc != LY_SUCCESS) {
7039 lyxp_set_free(set_all_desc);
7040 return rc;
7041 }
7042 /* prepend the original context nodes */
Michal Vasko5e0e6eb2019-11-06 15:47:50 +01007043 rc = moveto_union(set, set_all_desc);
Michal Vasko03ff5a72019-09-11 13:49:33 +02007044 if (rc != LY_SUCCESS) {
7045 lyxp_set_free(set_all_desc);
7046 return rc;
7047 }
7048 lyxp_set_free(set_all_desc);
7049
Radek Krejci1deb5be2020-08-26 16:43:36 +02007050 for (uint32_t i = 0; i < set->used; ) {
Radek Krejci857189e2020-09-01 13:26:36 +02007051 ly_bool replaced = 0;
Michal Vasko03ff5a72019-09-11 13:49:33 +02007052
7053 /* only attributes of an elem can be in the result, skip all the rest,
7054 * we have all attributes qualified in lyd tree */
7055 if (set->val.nodes[i].type == LYXP_NODE_ELEM) {
Michal Vasko9f96a052020-03-10 09:41:45 +01007056 for (sub = set->val.nodes[i].node->meta; sub; sub = sub->next) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02007057 /* check "namespace" */
Michal Vaskod3678892020-05-21 10:06:58 +02007058 if (mod && (sub->annotation->module != mod)) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02007059 continue;
7060 }
7061
Michal Vaskod3678892020-05-21 10:06:58 +02007062 if (!ncname || (sub->name == ncname)) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02007063 /* match */
7064 if (!replaced) {
Michal Vasko9f96a052020-03-10 09:41:45 +01007065 set->val.meta[i].meta = sub;
7066 set->val.meta[i].type = LYXP_NODE_META;
Michal Vasko03ff5a72019-09-11 13:49:33 +02007067 /* pos does not change */
7068 replaced = 1;
7069 } else {
Michal Vasko9f96a052020-03-10 09:41:45 +01007070 set_insert_node(set, (struct lyd_node *)sub, set->val.meta[i].pos, LYXP_NODE_META, i + 1);
Michal Vasko03ff5a72019-09-11 13:49:33 +02007071 }
7072 ++i;
7073 }
7074 }
7075 }
7076
7077 if (!replaced) {
7078 /* no match */
7079 set_remove_node(set, i);
7080 }
7081 }
7082
7083 return LY_SUCCESS;
7084}
7085
7086/**
Michal Vasko03ff5a72019-09-11 13:49:33 +02007087 * @brief Move context @p set to the result of a comparison. Handles '=', '!=', '<=', '<', '>=', or '>'.
7088 * Result is LYXP_SET_BOOLEAN. Indirectly context position aware.
7089 *
7090 * @param[in,out] set1 Set to use for the result.
7091 * @param[in] set2 Set acting as the second operand for @p op.
7092 * @param[in] op Comparison operator to process.
7093 * @param[in] options XPath options.
7094 * @return LY_ERR
7095 */
7096static LY_ERR
Radek Krejci1deb5be2020-08-26 16:43:36 +02007097moveto_op_comp(struct lyxp_set *set1, struct lyxp_set *set2, const char *op, uint32_t options)
Michal Vasko03ff5a72019-09-11 13:49:33 +02007098{
7099 /*
7100 * NODE SET + NODE SET = NODE SET + STRING /(1 NODE SET) 2 STRING
7101 * NODE SET + STRING = STRING + STRING /1 STRING (2 STRING)
7102 * NODE SET + NUMBER = NUMBER + NUMBER /1 NUMBER (2 NUMBER)
7103 * NODE SET + BOOLEAN = BOOLEAN + BOOLEAN /1 BOOLEAN (2 BOOLEAN)
7104 * STRING + NODE SET = STRING + STRING /(1 STRING) 2 STRING
7105 * NUMBER + NODE SET = NUMBER + NUMBER /(1 NUMBER) 2 NUMBER
7106 * BOOLEAN + NODE SET = BOOLEAN + BOOLEAN /(1 BOOLEAN) 2 BOOLEAN
7107 *
7108 * '=' or '!='
7109 * BOOLEAN + BOOLEAN
7110 * BOOLEAN + STRING = BOOLEAN + BOOLEAN /(1 BOOLEAN) 2 BOOLEAN
7111 * BOOLEAN + NUMBER = BOOLEAN + BOOLEAN /(1 BOOLEAN) 2 BOOLEAN
7112 * STRING + BOOLEAN = BOOLEAN + BOOLEAN /1 BOOLEAN (2 BOOLEAN)
7113 * NUMBER + BOOLEAN = BOOLEAN + BOOLEAN /1 BOOLEAN (2 BOOLEAN)
7114 * NUMBER + NUMBER
7115 * NUMBER + STRING = NUMBER + NUMBER /(1 NUMBER) 2 NUMBER
7116 * STRING + NUMBER = NUMBER + NUMBER /1 NUMBER (2 NUMBER)
7117 * STRING + STRING
7118 *
7119 * '<=', '<', '>=', '>'
7120 * NUMBER + NUMBER
7121 * BOOLEAN + BOOLEAN = NUMBER + NUMBER /1 NUMBER, 2 NUMBER
7122 * BOOLEAN + NUMBER = NUMBER + NUMBER /1 NUMBER (2 NUMBER)
7123 * BOOLEAN + STRING = NUMBER + NUMBER /1 NUMBER, 2 NUMBER
7124 * NUMBER + STRING = NUMBER + NUMBER /(1 NUMBER) 2 NUMBER
7125 * STRING + STRING = NUMBER + NUMBER /1 NUMBER, 2 NUMBER
7126 * STRING + NUMBER = NUMBER + NUMBER /1 NUMBER (2 NUMBER)
7127 * NUMBER + BOOLEAN = NUMBER + NUMBER /(1 NUMBER) 2 NUMBER
7128 * STRING + BOOLEAN = NUMBER + NUMBER /(1 NUMBER) 2 NUMBER
7129 */
7130 struct lyxp_set iter1, iter2;
7131 int result;
7132 int64_t i;
7133 LY_ERR rc;
7134
Michal Vasko004d3152020-06-11 19:59:22 +02007135 memset(&iter1, 0, sizeof iter1);
7136 memset(&iter2, 0, sizeof iter2);
Michal Vasko03ff5a72019-09-11 13:49:33 +02007137
7138 /* iterative evaluation with node-sets */
7139 if ((set1->type == LYXP_SET_NODE_SET) || (set2->type == LYXP_SET_NODE_SET)) {
7140 if (set1->type == LYXP_SET_NODE_SET) {
7141 for (i = 0; i < set1->used; ++i) {
Michal Vasko4c7763f2020-07-27 17:40:37 +02007142 /* cast set1 */
Michal Vasko03ff5a72019-09-11 13:49:33 +02007143 switch (set2->type) {
7144 case LYXP_SET_NUMBER:
Michal Vasko5e0e6eb2019-11-06 15:47:50 +01007145 rc = set_comp_cast(&iter1, set1, LYXP_SET_NUMBER, i);
Michal Vasko03ff5a72019-09-11 13:49:33 +02007146 break;
7147 case LYXP_SET_BOOLEAN:
Michal Vasko5e0e6eb2019-11-06 15:47:50 +01007148 rc = set_comp_cast(&iter1, set1, LYXP_SET_BOOLEAN, i);
Michal Vasko03ff5a72019-09-11 13:49:33 +02007149 break;
7150 default:
Michal Vasko5e0e6eb2019-11-06 15:47:50 +01007151 rc = set_comp_cast(&iter1, set1, LYXP_SET_STRING, i);
Michal Vasko03ff5a72019-09-11 13:49:33 +02007152 break;
7153 }
7154 LY_CHECK_RET(rc);
7155
Michal Vasko4c7763f2020-07-27 17:40:37 +02007156 /* canonize set2 */
7157 LY_CHECK_ERR_RET(rc = set_comp_canonize(&iter2, set2, &set1->val.nodes[i]), lyxp_set_free_content(&iter1), rc);
7158
7159 /* compare recursively */
7160 rc = moveto_op_comp(&iter1, &iter2, op, options);
7161 lyxp_set_free_content(&iter2);
7162 LY_CHECK_ERR_RET(rc, lyxp_set_free_content(&iter1), rc);
Michal Vasko03ff5a72019-09-11 13:49:33 +02007163
7164 /* lazy evaluation until true */
Michal Vasko004d3152020-06-11 19:59:22 +02007165 if (iter1.val.bln) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02007166 set_fill_boolean(set1, 1);
7167 return LY_SUCCESS;
7168 }
7169 }
7170 } else {
7171 for (i = 0; i < set2->used; ++i) {
Michal Vasko4c7763f2020-07-27 17:40:37 +02007172 /* set set2 */
Michal Vasko03ff5a72019-09-11 13:49:33 +02007173 switch (set1->type) {
Michal Vasko4c7763f2020-07-27 17:40:37 +02007174 case LYXP_SET_NUMBER:
7175 rc = set_comp_cast(&iter2, set2, LYXP_SET_NUMBER, i);
7176 break;
7177 case LYXP_SET_BOOLEAN:
7178 rc = set_comp_cast(&iter2, set2, LYXP_SET_BOOLEAN, i);
7179 break;
7180 default:
7181 rc = set_comp_cast(&iter2, set2, LYXP_SET_STRING, i);
7182 break;
Michal Vasko03ff5a72019-09-11 13:49:33 +02007183 }
7184 LY_CHECK_RET(rc);
7185
Michal Vasko4c7763f2020-07-27 17:40:37 +02007186 /* canonize set1 */
7187 LY_CHECK_ERR_RET(rc = set_comp_canonize(&iter1, set1, &set2->val.nodes[i]), lyxp_set_free_content(&iter2), rc);
Michal Vasko03ff5a72019-09-11 13:49:33 +02007188
Michal Vasko4c7763f2020-07-27 17:40:37 +02007189 /* compare recursively */
Michal Vasko03ff5a72019-09-11 13:49:33 +02007190 rc = moveto_op_comp(&iter1, &iter2, op, options);
Michal Vaskod3678892020-05-21 10:06:58 +02007191 lyxp_set_free_content(&iter2);
Michal Vasko4c7763f2020-07-27 17:40:37 +02007192 LY_CHECK_ERR_RET(rc, lyxp_set_free_content(&iter1), rc);
Michal Vasko03ff5a72019-09-11 13:49:33 +02007193
7194 /* lazy evaluation until true */
Michal Vasko004d3152020-06-11 19:59:22 +02007195 if (iter1.val.bln) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02007196 set_fill_boolean(set1, 1);
7197 return LY_SUCCESS;
7198 }
7199 }
7200 }
7201
7202 /* false for all nodes */
7203 set_fill_boolean(set1, 0);
7204 return LY_SUCCESS;
7205 }
7206
7207 /* first convert properly */
7208 if ((op[0] == '=') || (op[0] == '!')) {
7209 if ((set1->type == LYXP_SET_BOOLEAN) || (set2->type == LYXP_SET_BOOLEAN)) {
Michal Vasko5e0e6eb2019-11-06 15:47:50 +01007210 lyxp_set_cast(set1, LYXP_SET_BOOLEAN);
7211 lyxp_set_cast(set2, LYXP_SET_BOOLEAN);
Michal Vasko03ff5a72019-09-11 13:49:33 +02007212 } else if ((set1->type == LYXP_SET_NUMBER) || (set2->type == LYXP_SET_NUMBER)) {
Michal Vasko5e0e6eb2019-11-06 15:47:50 +01007213 rc = lyxp_set_cast(set1, LYXP_SET_NUMBER);
Michal Vasko03ff5a72019-09-11 13:49:33 +02007214 LY_CHECK_RET(rc);
Michal Vasko5e0e6eb2019-11-06 15:47:50 +01007215 rc = lyxp_set_cast(set2, LYXP_SET_NUMBER);
Michal Vasko03ff5a72019-09-11 13:49:33 +02007216 LY_CHECK_RET(rc);
7217 } /* else we have 2 strings */
7218 } else {
Michal Vasko5e0e6eb2019-11-06 15:47:50 +01007219 rc = lyxp_set_cast(set1, LYXP_SET_NUMBER);
Michal Vasko03ff5a72019-09-11 13:49:33 +02007220 LY_CHECK_RET(rc);
Michal Vasko5e0e6eb2019-11-06 15:47:50 +01007221 rc = lyxp_set_cast(set2, LYXP_SET_NUMBER);
Michal Vasko03ff5a72019-09-11 13:49:33 +02007222 LY_CHECK_RET(rc);
7223 }
7224
7225 assert(set1->type == set2->type);
7226
7227 /* compute result */
7228 if (op[0] == '=') {
7229 if (set1->type == LYXP_SET_BOOLEAN) {
Michal Vasko004d3152020-06-11 19:59:22 +02007230 result = (set1->val.bln == set2->val.bln);
Michal Vasko03ff5a72019-09-11 13:49:33 +02007231 } else if (set1->type == LYXP_SET_NUMBER) {
7232 result = (set1->val.num == set2->val.num);
7233 } else {
7234 assert(set1->type == LYXP_SET_STRING);
Michal Vaskoac6c72f2019-11-14 16:09:34 +01007235 result = !strcmp(set1->val.str, set2->val.str);
Michal Vasko03ff5a72019-09-11 13:49:33 +02007236 }
7237 } else if (op[0] == '!') {
7238 if (set1->type == LYXP_SET_BOOLEAN) {
Michal Vasko004d3152020-06-11 19:59:22 +02007239 result = (set1->val.bln != set2->val.bln);
Michal Vasko03ff5a72019-09-11 13:49:33 +02007240 } else if (set1->type == LYXP_SET_NUMBER) {
7241 result = (set1->val.num != set2->val.num);
7242 } else {
7243 assert(set1->type == LYXP_SET_STRING);
Michal Vaskoc2058432020-11-06 17:26:21 +01007244 result = strcmp(set1->val.str, set2->val.str);
Michal Vasko03ff5a72019-09-11 13:49:33 +02007245 }
7246 } else {
7247 assert(set1->type == LYXP_SET_NUMBER);
7248 if (op[0] == '<') {
7249 if (op[1] == '=') {
7250 result = (set1->val.num <= set2->val.num);
7251 } else {
7252 result = (set1->val.num < set2->val.num);
7253 }
7254 } else {
7255 if (op[1] == '=') {
7256 result = (set1->val.num >= set2->val.num);
7257 } else {
7258 result = (set1->val.num > set2->val.num);
7259 }
7260 }
7261 }
7262
7263 /* assign result */
7264 if (result) {
7265 set_fill_boolean(set1, 1);
7266 } else {
7267 set_fill_boolean(set1, 0);
7268 }
7269
7270 return LY_SUCCESS;
7271}
7272
7273/**
7274 * @brief Move context @p set to the result of a basic operation. Handles '+', '-', unary '-', '*', 'div',
7275 * or 'mod'. Result is LYXP_SET_NUMBER. Indirectly context position aware.
7276 *
7277 * @param[in,out] set1 Set to use for the result.
7278 * @param[in] set2 Set acting as the second operand for @p op.
7279 * @param[in] op Operator to process.
Michal Vasko03ff5a72019-09-11 13:49:33 +02007280 * @return LY_ERR
7281 */
7282static LY_ERR
Michal Vasko5e0e6eb2019-11-06 15:47:50 +01007283moveto_op_math(struct lyxp_set *set1, struct lyxp_set *set2, const char *op)
Michal Vasko03ff5a72019-09-11 13:49:33 +02007284{
7285 LY_ERR rc;
7286
7287 /* unary '-' */
7288 if (!set2 && (op[0] == '-')) {
Michal Vasko5e0e6eb2019-11-06 15:47:50 +01007289 rc = lyxp_set_cast(set1, LYXP_SET_NUMBER);
Michal Vasko03ff5a72019-09-11 13:49:33 +02007290 LY_CHECK_RET(rc);
7291 set1->val.num *= -1;
7292 lyxp_set_free(set2);
7293 return LY_SUCCESS;
7294 }
7295
7296 assert(set1 && set2);
7297
Michal Vasko5e0e6eb2019-11-06 15:47:50 +01007298 rc = lyxp_set_cast(set1, LYXP_SET_NUMBER);
Michal Vasko03ff5a72019-09-11 13:49:33 +02007299 LY_CHECK_RET(rc);
Michal Vasko5e0e6eb2019-11-06 15:47:50 +01007300 rc = lyxp_set_cast(set2, LYXP_SET_NUMBER);
Michal Vasko03ff5a72019-09-11 13:49:33 +02007301 LY_CHECK_RET(rc);
7302
7303 switch (op[0]) {
7304 /* '+' */
7305 case '+':
7306 set1->val.num += set2->val.num;
7307 break;
7308
7309 /* '-' */
7310 case '-':
7311 set1->val.num -= set2->val.num;
7312 break;
7313
7314 /* '*' */
7315 case '*':
7316 set1->val.num *= set2->val.num;
7317 break;
7318
7319 /* 'div' */
7320 case 'd':
7321 set1->val.num /= set2->val.num;
7322 break;
7323
7324 /* 'mod' */
7325 case 'm':
7326 set1->val.num = ((long long)set1->val.num) % ((long long)set2->val.num);
7327 break;
7328
7329 default:
7330 LOGINT_RET(set1->ctx);
7331 }
7332
7333 return LY_SUCCESS;
7334}
7335
Michal Vasko03ff5a72019-09-11 13:49:33 +02007336/**
Michal Vasko03ff5a72019-09-11 13:49:33 +02007337 * @brief Evaluate Predicate. Logs directly on error.
7338 *
Michal Vaskod3678892020-05-21 10:06:58 +02007339 * [9] Predicate ::= '[' Expr ']'
Michal Vasko03ff5a72019-09-11 13:49:33 +02007340 *
7341 * @param[in] exp Parsed XPath expression.
Michal Vasko004d3152020-06-11 19:59:22 +02007342 * @param[in] tok_idx Position in the expression @p exp.
aPiecek8b0cc152021-05-31 16:40:31 +02007343 * @param[in,out] set Context and result set.
Michal Vasko03ff5a72019-09-11 13:49:33 +02007344 * @param[in] options XPath options.
Michal Vasko49fec8e2022-05-24 10:28:33 +02007345 * @param[in] axis Axis to search on.
Michal Vasko03ff5a72019-09-11 13:49:33 +02007346 * @return LY_ERR (LY_EINCOMPLETE on unresolved when)
7347 */
7348static LY_ERR
Michal Vasko49fec8e2022-05-24 10:28:33 +02007349eval_predicate(const struct lyxp_expr *exp, uint16_t *tok_idx, struct lyxp_set *set, uint32_t options, enum lyxp_axis axis)
Michal Vasko03ff5a72019-09-11 13:49:33 +02007350{
7351 LY_ERR rc;
Michal Vasko1fdd8fa2021-01-08 09:21:45 +01007352 uint16_t orig_exp;
7353 uint32_t i, orig_pos, orig_size;
Michal Vasko5c4e5892019-11-14 12:31:38 +01007354 int32_t pred_in_ctx;
Michal Vasko88a9e802022-05-24 10:50:28 +02007355 ly_bool reverse_axis = 0;
aPiecekfff4dca2021-10-07 10:59:53 +02007356 struct lyxp_set set2 = {0};
Michal Vasko03ff5a72019-09-11 13:49:33 +02007357
7358 /* '[' */
aPiecek8b0cc152021-05-31 16:40:31 +02007359 LOGDBG(LY_LDGXPATH, "%-27s %s %s[%u]", __func__, (options & LYXP_SKIP_EXPR ? "skipped" : "parsed"),
Michal Vasko49fec8e2022-05-24 10:28:33 +02007360 lyxp_token2str(exp->tokens[*tok_idx]), exp->tok_pos[*tok_idx]);
Michal Vasko004d3152020-06-11 19:59:22 +02007361 ++(*tok_idx);
Michal Vasko03ff5a72019-09-11 13:49:33 +02007362
aPiecek8b0cc152021-05-31 16:40:31 +02007363 if (options & LYXP_SKIP_EXPR) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02007364only_parse:
aPiecek8b0cc152021-05-31 16:40:31 +02007365 rc = eval_expr_select(exp, tok_idx, 0, set, options | LYXP_SKIP_EXPR);
Michal Vasko03ff5a72019-09-11 13:49:33 +02007366 LY_CHECK_RET(rc);
7367 } else if (set->type == LYXP_SET_NODE_SET) {
7368 /* we (possibly) need the set sorted, it can affect the result (if the predicate result is a number) */
Michal Vasko5e0e6eb2019-11-06 15:47:50 +01007369 assert(!set_sort(set));
Michal Vasko03ff5a72019-09-11 13:49:33 +02007370
7371 /* empty set, nothing to evaluate */
7372 if (!set->used) {
7373 goto only_parse;
7374 }
7375
Michal Vasko49fec8e2022-05-24 10:28:33 +02007376 /* decide forward or reverse axis */
7377 switch (axis) {
7378 case LYXP_AXIS_ANCESTOR:
7379 case LYXP_AXIS_ANCESTOR_OR_SELF:
7380 case LYXP_AXIS_PRECEDING:
7381 case LYXP_AXIS_PRECEDING_SIBLING:
7382 reverse_axis = 1;
7383 break;
7384 case LYXP_AXIS_DESCENDANT:
7385 case LYXP_AXIS_DESCENDANT_OR_SELF:
7386 case LYXP_AXIS_FOLLOWING:
7387 case LYXP_AXIS_FOLLOWING_SIBLING:
7388 case LYXP_AXIS_PARENT:
7389 case LYXP_AXIS_CHILD:
7390 case LYXP_AXIS_SELF:
7391 case LYXP_AXIS_ATTRIBUTE:
7392 reverse_axis = 0;
7393 break;
7394 }
7395
Michal Vasko004d3152020-06-11 19:59:22 +02007396 orig_exp = *tok_idx;
Michal Vasko49fec8e2022-05-24 10:28:33 +02007397 orig_pos = reverse_axis ? set->used + 1 : 0;
Michal Vasko03ff5a72019-09-11 13:49:33 +02007398 orig_size = set->used;
Michal Vasko39dbf352020-05-21 10:08:59 +02007399 for (i = 0; i < set->used; ++i) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02007400 set_init(&set2, set);
7401 set_insert_node(&set2, set->val.nodes[i].node, set->val.nodes[i].pos, set->val.nodes[i].type, 0);
Michal Vasko49fec8e2022-05-24 10:28:33 +02007402
7403 /* remember the node context position for position() and context size for last() */
7404 orig_pos += reverse_axis ? -1 : 1;
Michal Vasko03ff5a72019-09-11 13:49:33 +02007405
7406 set2.ctx_pos = orig_pos;
7407 set2.ctx_size = orig_size;
Michal Vasko004d3152020-06-11 19:59:22 +02007408 *tok_idx = orig_exp;
Michal Vasko03ff5a72019-09-11 13:49:33 +02007409
Michal Vasko004d3152020-06-11 19:59:22 +02007410 rc = eval_expr_select(exp, tok_idx, 0, &set2, options);
Michal Vasko03ff5a72019-09-11 13:49:33 +02007411 if (rc != LY_SUCCESS) {
Michal Vaskod3678892020-05-21 10:06:58 +02007412 lyxp_set_free_content(&set2);
Michal Vasko03ff5a72019-09-11 13:49:33 +02007413 return rc;
7414 }
7415
Michal Vasko49fec8e2022-05-24 10:28:33 +02007416 /* number is a proximity position */
Michal Vasko03ff5a72019-09-11 13:49:33 +02007417 if (set2.type == LYXP_SET_NUMBER) {
7418 if ((long long)set2.val.num == orig_pos) {
7419 set2.val.num = 1;
7420 } else {
7421 set2.val.num = 0;
7422 }
7423 }
Michal Vasko5e0e6eb2019-11-06 15:47:50 +01007424 lyxp_set_cast(&set2, LYXP_SET_BOOLEAN);
Michal Vasko03ff5a72019-09-11 13:49:33 +02007425
7426 /* predicate satisfied or not? */
Michal Vasko004d3152020-06-11 19:59:22 +02007427 if (!set2.val.bln) {
Michal Vasko2caefc12019-11-14 16:07:56 +01007428 set_remove_node_none(set, i);
Michal Vasko03ff5a72019-09-11 13:49:33 +02007429 }
7430 }
Michal Vasko2caefc12019-11-14 16:07:56 +01007431 set_remove_nodes_none(set);
Michal Vasko03ff5a72019-09-11 13:49:33 +02007432
7433 } else if (set->type == LYXP_SET_SCNODE_SET) {
7434 for (i = 0; i < set->used; ++i) {
Radek Krejcif13b87b2020-12-01 22:02:17 +01007435 if (set->val.scnodes[i].in_ctx == LYXP_SET_SCNODE_ATOM_CTX) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02007436 /* there is a currently-valid node */
7437 break;
7438 }
7439 }
7440 /* empty set, nothing to evaluate */
7441 if (i == set->used) {
7442 goto only_parse;
7443 }
7444
Michal Vasko004d3152020-06-11 19:59:22 +02007445 orig_exp = *tok_idx;
Michal Vasko03ff5a72019-09-11 13:49:33 +02007446
Michal Vasko03ff5a72019-09-11 13:49:33 +02007447 /* set special in_ctx to all the valid snodes */
7448 pred_in_ctx = set_scnode_new_in_ctx(set);
7449
7450 /* use the valid snodes one-by-one */
7451 for (i = 0; i < set->used; ++i) {
7452 if (set->val.scnodes[i].in_ctx != pred_in_ctx) {
7453 continue;
7454 }
Radek Krejcif13b87b2020-12-01 22:02:17 +01007455 set->val.scnodes[i].in_ctx = LYXP_SET_SCNODE_ATOM_CTX;
Michal Vasko03ff5a72019-09-11 13:49:33 +02007456
Michal Vasko004d3152020-06-11 19:59:22 +02007457 *tok_idx = orig_exp;
Michal Vasko03ff5a72019-09-11 13:49:33 +02007458
Michal Vasko004d3152020-06-11 19:59:22 +02007459 rc = eval_expr_select(exp, tok_idx, 0, set, options);
Michal Vasko03ff5a72019-09-11 13:49:33 +02007460 LY_CHECK_RET(rc);
7461
7462 set->val.scnodes[i].in_ctx = pred_in_ctx;
7463 }
7464
7465 /* restore the state as it was before the predicate */
7466 for (i = 0; i < set->used; ++i) {
Radek Krejcif13b87b2020-12-01 22:02:17 +01007467 if (set->val.scnodes[i].in_ctx == LYXP_SET_SCNODE_ATOM_CTX) {
Michal Vasko1a09b212021-05-06 13:00:10 +02007468 set->val.scnodes[i].in_ctx = LYXP_SET_SCNODE_ATOM_NODE;
Michal Vasko03ff5a72019-09-11 13:49:33 +02007469 } else if (set->val.scnodes[i].in_ctx == pred_in_ctx) {
Radek Krejcif13b87b2020-12-01 22:02:17 +01007470 set->val.scnodes[i].in_ctx = LYXP_SET_SCNODE_ATOM_CTX;
Michal Vasko03ff5a72019-09-11 13:49:33 +02007471 }
7472 }
7473
7474 } else {
Michal Vaskod3678892020-05-21 10:06:58 +02007475 set2.type = LYXP_SET_NODE_SET;
Michal Vasko03ff5a72019-09-11 13:49:33 +02007476 set_fill_set(&set2, set);
7477
Michal Vasko004d3152020-06-11 19:59:22 +02007478 rc = eval_expr_select(exp, tok_idx, 0, &set2, options);
Michal Vasko03ff5a72019-09-11 13:49:33 +02007479 if (rc != LY_SUCCESS) {
Michal Vaskod3678892020-05-21 10:06:58 +02007480 lyxp_set_free_content(&set2);
Michal Vasko03ff5a72019-09-11 13:49:33 +02007481 return rc;
7482 }
7483
Michal Vasko5e0e6eb2019-11-06 15:47:50 +01007484 lyxp_set_cast(&set2, LYXP_SET_BOOLEAN);
Michal Vasko004d3152020-06-11 19:59:22 +02007485 if (!set2.val.bln) {
Michal Vaskod3678892020-05-21 10:06:58 +02007486 lyxp_set_free_content(set);
Michal Vasko03ff5a72019-09-11 13:49:33 +02007487 }
Michal Vaskod3678892020-05-21 10:06:58 +02007488 lyxp_set_free_content(&set2);
Michal Vasko03ff5a72019-09-11 13:49:33 +02007489 }
7490
7491 /* ']' */
Michal Vasko004d3152020-06-11 19:59:22 +02007492 assert(exp->tokens[*tok_idx] == LYXP_TOKEN_BRACK2);
aPiecek8b0cc152021-05-31 16:40:31 +02007493 LOGDBG(LY_LDGXPATH, "%-27s %s %s[%u]", __func__, (options & LYXP_SKIP_EXPR ? "skipped" : "parsed"),
Michal Vasko49fec8e2022-05-24 10:28:33 +02007494 lyxp_token2str(exp->tokens[*tok_idx]), exp->tok_pos[*tok_idx]);
Michal Vasko004d3152020-06-11 19:59:22 +02007495 ++(*tok_idx);
Michal Vasko03ff5a72019-09-11 13:49:33 +02007496
7497 return LY_SUCCESS;
7498}
7499
7500/**
Michal Vaskod3678892020-05-21 10:06:58 +02007501 * @brief Evaluate Literal. Logs directly on error.
7502 *
7503 * @param[in] exp Parsed XPath expression.
Michal Vasko004d3152020-06-11 19:59:22 +02007504 * @param[in] tok_idx Position in the expression @p exp.
Michal Vaskod3678892020-05-21 10:06:58 +02007505 * @param[in,out] set Context and result set. On NULL the rule is only parsed.
7506 */
7507static void
Michal Vasko40308e72020-10-20 16:38:40 +02007508eval_literal(const struct lyxp_expr *exp, uint16_t *tok_idx, struct lyxp_set *set)
Michal Vaskod3678892020-05-21 10:06:58 +02007509{
7510 if (set) {
Michal Vasko004d3152020-06-11 19:59:22 +02007511 if (exp->tok_len[*tok_idx] == 2) {
Michal Vaskod3678892020-05-21 10:06:58 +02007512 set_fill_string(set, "", 0);
7513 } else {
Michal Vasko004d3152020-06-11 19:59:22 +02007514 set_fill_string(set, &exp->expr[exp->tok_pos[*tok_idx] + 1], exp->tok_len[*tok_idx] - 2);
Michal Vaskod3678892020-05-21 10:06:58 +02007515 }
7516 }
7517 LOGDBG(LY_LDGXPATH, "%-27s %s %s[%u]", __func__, (set ? "parsed" : "skipped"),
Michal Vasko49fec8e2022-05-24 10:28:33 +02007518 lyxp_token2str(exp->tokens[*tok_idx]), exp->tok_pos[*tok_idx]);
Michal Vasko004d3152020-06-11 19:59:22 +02007519 ++(*tok_idx);
Michal Vaskod3678892020-05-21 10:06:58 +02007520}
7521
7522/**
Michal Vasko004d3152020-06-11 19:59:22 +02007523 * @brief Try to compile list or leaf-list predicate in the known format to be used for hash-based instance search.
Michal Vaskod3678892020-05-21 10:06:58 +02007524 *
Michal Vasko004d3152020-06-11 19:59:22 +02007525 * @param[in] exp Full parsed XPath expression.
7526 * @param[in,out] tok_idx Index in @p exp at the beginning of the predicate, is updated on success.
Michal Vasko5d24f6c2020-10-13 13:49:06 +02007527 * @param[in] ctx_node Found schema node as the context for the predicate.
7528 * @param[in] cur_mod Current module for the expression.
7529 * @param[in] cur_node Current (original context) node.
Michal Vasko004d3152020-06-11 19:59:22 +02007530 * @param[in] format Format of any prefixes in key names/values.
Michal Vasko5d24f6c2020-10-13 13:49:06 +02007531 * @param[in] prefix_data Format-specific prefix data (see ::ly_resolve_prefix).
Michal Vasko004d3152020-06-11 19:59:22 +02007532 * @param[out] predicates Parsed predicates.
7533 * @param[out] pred_type Type of @p predicates.
7534 * @return LY_SUCCESS on success,
7535 * @return LY_ERR on any error.
Michal Vaskod3678892020-05-21 10:06:58 +02007536 */
7537static LY_ERR
Michal Vasko40308e72020-10-20 16:38:40 +02007538eval_name_test_try_compile_predicates(const struct lyxp_expr *exp, uint16_t *tok_idx, const struct lysc_node *ctx_node,
Radek Krejci8df109d2021-04-23 12:19:08 +02007539 const struct lys_module *cur_mod, const struct lysc_node *cur_node, LY_VALUE_FORMAT format, void *prefix_data,
Michal Vasko5d24f6c2020-10-13 13:49:06 +02007540 struct ly_path_predicate **predicates, enum ly_path_pred_type *pred_type)
Michal Vaskod3678892020-05-21 10:06:58 +02007541{
Michal Vasko004d3152020-06-11 19:59:22 +02007542 LY_ERR ret = LY_SUCCESS;
7543 uint16_t key_count, e_idx, pred_idx = 0;
Michal Vaskod3678892020-05-21 10:06:58 +02007544 const struct lysc_node *key;
Michal Vasko004d3152020-06-11 19:59:22 +02007545 size_t pred_len;
Radek Krejci1deb5be2020-08-26 16:43:36 +02007546 uint32_t prev_lo;
Michal Vasko004d3152020-06-11 19:59:22 +02007547 struct lyxp_expr *exp2 = NULL;
Michal Vaskod3678892020-05-21 10:06:58 +02007548
Michal Vasko5d24f6c2020-10-13 13:49:06 +02007549 assert(ctx_node->nodetype & (LYS_LIST | LYS_LEAFLIST));
Michal Vaskod3678892020-05-21 10:06:58 +02007550
Michal Vasko5d24f6c2020-10-13 13:49:06 +02007551 if (ctx_node->nodetype == LYS_LIST) {
Michal Vasko004d3152020-06-11 19:59:22 +02007552 /* get key count */
Michal Vasko5d24f6c2020-10-13 13:49:06 +02007553 if (ctx_node->flags & LYS_KEYLESS) {
Michal Vasko004d3152020-06-11 19:59:22 +02007554 return LY_EINVAL;
7555 }
Michal Vasko544e58a2021-01-28 14:33:41 +01007556 for (key_count = 0, key = lysc_node_child(ctx_node); key && (key->flags & LYS_KEY); key = key->next, ++key_count) {}
Michal Vasko004d3152020-06-11 19:59:22 +02007557 assert(key_count);
Michal Vaskod3678892020-05-21 10:06:58 +02007558
Michal Vasko004d3152020-06-11 19:59:22 +02007559 /* learn where the predicates end */
7560 e_idx = *tok_idx;
7561 while (key_count) {
7562 /* '[' */
7563 if (lyxp_check_token(NULL, exp, e_idx, LYXP_TOKEN_BRACK1)) {
7564 return LY_EINVAL;
7565 }
7566 ++e_idx;
7567
Michal Vasko3354d272021-04-06 09:40:06 +02007568 if (lyxp_check_token(NULL, exp, e_idx, LYXP_TOKEN_NAMETEST)) {
7569 /* definitely not a key */
7570 return LY_EINVAL;
7571 }
7572
Michal Vasko004d3152020-06-11 19:59:22 +02007573 /* ']' */
7574 while (lyxp_check_token(NULL, exp, e_idx, LYXP_TOKEN_BRACK2)) {
7575 ++e_idx;
7576 }
7577 ++e_idx;
7578
7579 /* another presumably key predicate parsed */
7580 --key_count;
7581 }
Michal Vasko004d3152020-06-11 19:59:22 +02007582 } else {
7583 /* learn just where this single predicate ends */
7584 e_idx = *tok_idx;
7585
Michal Vaskod3678892020-05-21 10:06:58 +02007586 /* '[' */
Michal Vasko004d3152020-06-11 19:59:22 +02007587 if (lyxp_check_token(NULL, exp, e_idx, LYXP_TOKEN_BRACK1)) {
7588 return LY_EINVAL;
7589 }
7590 ++e_idx;
Michal Vaskod3678892020-05-21 10:06:58 +02007591
Michal Vasko3354d272021-04-06 09:40:06 +02007592 if (lyxp_check_token(NULL, exp, e_idx, LYXP_TOKEN_DOT)) {
7593 /* definitely not the value */
7594 return LY_EINVAL;
7595 }
7596
Michal Vaskod3678892020-05-21 10:06:58 +02007597 /* ']' */
Michal Vasko004d3152020-06-11 19:59:22 +02007598 while (lyxp_check_token(NULL, exp, e_idx, LYXP_TOKEN_BRACK2)) {
7599 ++e_idx;
7600 }
7601 ++e_idx;
Michal Vaskod3678892020-05-21 10:06:58 +02007602 }
7603
Michal Vasko004d3152020-06-11 19:59:22 +02007604 /* get the joined length of all the keys predicates/of the single leaf-list predicate */
7605 pred_len = (exp->tok_pos[e_idx - 1] + exp->tok_len[e_idx - 1]) - exp->tok_pos[*tok_idx];
7606
7607 /* turn logging off */
7608 prev_lo = ly_log_options(0);
7609
7610 /* parse the predicate(s) */
Michal Vasko5d24f6c2020-10-13 13:49:06 +02007611 LY_CHECK_GOTO(ret = ly_path_parse_predicate(ctx_node->module->ctx, ctx_node, exp->expr + exp->tok_pos[*tok_idx],
7612 pred_len, LY_PATH_PREFIX_OPTIONAL, LY_PATH_PRED_SIMPLE, &exp2), cleanup);
Michal Vasko004d3152020-06-11 19:59:22 +02007613
7614 /* compile */
Michal Vasko5d24f6c2020-10-13 13:49:06 +02007615 ret = ly_path_compile_predicate(ctx_node->module->ctx, cur_node, cur_mod, ctx_node, exp2, &pred_idx, format,
7616 prefix_data, predicates, pred_type);
Michal Vasko004d3152020-06-11 19:59:22 +02007617 LY_CHECK_GOTO(ret, cleanup);
7618
7619 /* success, the predicate must include all the needed information for hash-based search */
7620 *tok_idx = e_idx;
7621
7622cleanup:
7623 ly_log_options(prev_lo);
Michal Vasko5d24f6c2020-10-13 13:49:06 +02007624 lyxp_expr_free(ctx_node->module->ctx, exp2);
Michal Vasko004d3152020-06-11 19:59:22 +02007625 return ret;
Michal Vaskod3678892020-05-21 10:06:58 +02007626}
7627
7628/**
Michal Vasko5d24f6c2020-10-13 13:49:06 +02007629 * @brief Search for/check the next schema node that could be the only matching schema node meaning the
7630 * data node(s) could be found using a single hash-based search.
7631 *
Michal Vaskoc71c37b2021-01-11 13:40:02 +01007632 * @param[in] ctx libyang context.
Michal Vasko5d24f6c2020-10-13 13:49:06 +02007633 * @param[in] node Next context node to check.
7634 * @param[in] name Expected node name.
7635 * @param[in] name_len Length of @p name.
Michal Vaskoc71c37b2021-01-11 13:40:02 +01007636 * @param[in] moveto_mod Expected node module, can be NULL for JSON format with no prefix.
Michal Vasko5d24f6c2020-10-13 13:49:06 +02007637 * @param[in] root_type XPath root type.
Michal Vasko5d24f6c2020-10-13 13:49:06 +02007638 * @param[in] format Prefix format.
7639 * @param[in,out] found Previously found node, is updated.
7640 * @return LY_SUCCESS on success,
7641 * @return LY_ENOT if the whole check failed and hashes cannot be used.
7642 */
7643static LY_ERR
Michal Vaskoc71c37b2021-01-11 13:40:02 +01007644eval_name_test_with_predicate_get_scnode(const struct ly_ctx *ctx, const struct lyd_node *node, const char *name,
Radek Krejci8df109d2021-04-23 12:19:08 +02007645 uint16_t name_len, const struct lys_module *moveto_mod, enum lyxp_node_type root_type, LY_VALUE_FORMAT format,
Michal Vasko5d24f6c2020-10-13 13:49:06 +02007646 const struct lysc_node **found)
7647{
7648 const struct lysc_node *scnode;
7649 const struct lys_module *mod;
7650 uint32_t idx = 0;
7651
Radek Krejci8df109d2021-04-23 12:19:08 +02007652 assert((format == LY_VALUE_JSON) || moveto_mod);
Michal Vaskoc71c37b2021-01-11 13:40:02 +01007653
Michal Vasko5d24f6c2020-10-13 13:49:06 +02007654continue_search:
Michal Vasko7d1d0912020-10-16 08:38:30 +02007655 scnode = NULL;
Michal Vasko5d24f6c2020-10-13 13:49:06 +02007656 if (!node) {
Radek Krejci8df109d2021-04-23 12:19:08 +02007657 if ((format == LY_VALUE_JSON) && !moveto_mod) {
Michal Vasko5d24f6c2020-10-13 13:49:06 +02007658 /* search all modules for a single match */
Michal Vaskoc71c37b2021-01-11 13:40:02 +01007659 while ((mod = ly_ctx_get_module_iter(ctx, &idx))) {
Michal Vasko35a3b1d2021-07-14 09:34:16 +02007660 if (!mod->implemented) {
7661 continue;
7662 }
7663
Michal Vasko5d24f6c2020-10-13 13:49:06 +02007664 scnode = lys_find_child(NULL, mod, name, name_len, 0, 0);
7665 if (scnode) {
7666 /* we have found a match */
7667 break;
7668 }
7669 }
7670
Michal Vasko7d1d0912020-10-16 08:38:30 +02007671 if (!scnode) {
7672 /* all modules searched */
7673 idx = 0;
7674 }
Michal Vasko5d24f6c2020-10-13 13:49:06 +02007675 } else {
7676 /* search in top-level */
7677 scnode = lys_find_child(NULL, moveto_mod, name, name_len, 0, 0);
7678 }
7679 } else if (!*found || (lysc_data_parent(*found) != node->schema)) {
Radek Krejci8df109d2021-04-23 12:19:08 +02007680 if ((format == LY_VALUE_JSON) && !moveto_mod) {
Michal Vasko5d24f6c2020-10-13 13:49:06 +02007681 /* we must adjust the module to inherit the one from the context node */
7682 moveto_mod = node->schema->module;
7683 }
7684
7685 /* search in children, do not repeat the same search */
7686 scnode = lys_find_child(node->schema, moveto_mod, name, name_len, 0, 0);
Michal Vasko7d1d0912020-10-16 08:38:30 +02007687 } /* else skip redundant search */
Michal Vasko5d24f6c2020-10-13 13:49:06 +02007688
7689 /* additional context check */
7690 if (scnode && (root_type == LYXP_NODE_ROOT_CONFIG) && (scnode->flags & LYS_CONFIG_R)) {
7691 scnode = NULL;
7692 }
7693
7694 if (scnode) {
7695 if (*found) {
7696 /* we found a schema node with the same name but at different level, give up, too complicated
7697 * (more hash-based searches would be required, not supported) */
7698 return LY_ENOT;
7699 } else {
7700 /* remember the found schema node and continue to make sure it can be used */
7701 *found = scnode;
7702 }
7703 }
7704
7705 if (idx) {
7706 /* continue searching all the following models */
7707 goto continue_search;
7708 }
7709
7710 return LY_SUCCESS;
7711}
7712
7713/**
Michal Vasko4ad69e72021-10-26 16:25:55 +02007714 * @brief Generate message when no matching schema nodes were found for a path segment.
7715 *
7716 * @param[in] set XPath set.
7717 * @param[in] scparent Previous schema parent in the context, if only one.
7718 * @param[in] ncname XPath NCName being evaluated.
7719 * @param[in] ncname_len Length of @p ncname.
7720 * @param[in] expr Whole XPath expression.
7721 * @param[in] options XPath options.
7722 */
7723static void
7724eval_name_test_scnode_no_match_msg(struct lyxp_set *set, const struct lyxp_set_scnode *scparent, const char *ncname,
7725 uint16_t ncname_len, const char *expr, uint32_t options)
7726{
7727 const char *format;
7728 char *path = NULL, *ppath = NULL;
7729
7730 path = lysc_path(set->cur_scnode, LYSC_PATH_LOG, NULL, 0);
7731 if (scparent) {
7732 /* generate path for the parent */
7733 if (scparent->type == LYXP_NODE_ELEM) {
7734 ppath = lysc_path(scparent->scnode, LYSC_PATH_LOG, NULL, 0);
7735 } else if (scparent->type == LYXP_NODE_ROOT) {
7736 ppath = strdup("<root>");
7737 } else if (scparent->type == LYXP_NODE_ROOT_CONFIG) {
7738 ppath = strdup("<config-root>");
7739 }
7740 }
7741 if (ppath) {
7742 format = "Schema node \"%.*s\" for parent \"%s\" not found; in expr \"%.*s\" with context node \"%s\".";
7743 if (options & LYXP_SCNODE_ERROR) {
7744 LOGERR(set->ctx, LY_EVALID, format, ncname_len, ncname, ppath, (ncname - expr) + ncname_len, expr, path);
7745 } else {
7746 LOGWRN(set->ctx, format, ncname_len, ncname, ppath, (ncname - expr) + ncname_len, expr, path);
7747 }
7748 } else {
7749 format = "Schema node \"%.*s\" not found; in expr \"%.*s\" with context node \"%s\".";
7750 if (options & LYXP_SCNODE_ERROR) {
7751 LOGERR(set->ctx, LY_EVALID, format, ncname_len, ncname, (ncname - expr) + ncname_len, expr, path);
7752 } else {
7753 LOGWRN(set->ctx, format, ncname_len, ncname, (ncname - expr) + ncname_len, expr, path);
7754 }
7755 }
7756 free(path);
7757 free(ppath);
7758}
7759
7760/**
Michal Vaskod3678892020-05-21 10:06:58 +02007761 * @brief Evaluate NameTest and any following Predicates. Logs directly on error.
7762 *
7763 * [5] Step ::= '@'? NodeTest Predicate* | '.' | '..'
7764 * [6] NodeTest ::= NameTest | NodeType '(' ')'
7765 * [7] NameTest ::= '*' | NCName ':' '*' | QName
7766 *
7767 * @param[in] exp Parsed XPath expression.
Michal Vasko004d3152020-06-11 19:59:22 +02007768 * @param[in] tok_idx Position in the expression @p exp.
Michal Vasko49fec8e2022-05-24 10:28:33 +02007769 * @param[in] axis What axis to search on.
Michal Vaskod3678892020-05-21 10:06:58 +02007770 * @param[in] all_desc Whether to search all the descendants or children only.
aPiecek8b0cc152021-05-31 16:40:31 +02007771 * @param[in,out] set Context and result set.
Michal Vaskod3678892020-05-21 10:06:58 +02007772 * @param[in] options XPath options.
Michal Vaskoa036a6b2021-10-12 16:05:23 +02007773 * @return LY_ERR (LY_EINCOMPLETE on unresolved when, LY_ENOT for not found schema node)
Michal Vaskod3678892020-05-21 10:06:58 +02007774 */
7775static LY_ERR
Michal Vasko49fec8e2022-05-24 10:28:33 +02007776eval_name_test_with_predicate(const struct lyxp_expr *exp, uint16_t *tok_idx, enum lyxp_axis axis, ly_bool all_desc,
Michal Vasko5d24f6c2020-10-13 13:49:06 +02007777 struct lyxp_set *set, uint32_t options)
Michal Vaskod3678892020-05-21 10:06:58 +02007778{
Michal Vaskoa036a6b2021-10-12 16:05:23 +02007779 LY_ERR rc = LY_SUCCESS, r;
Michal Vasko004d3152020-06-11 19:59:22 +02007780 const char *ncname, *ncname_dict = NULL;
7781 uint16_t ncname_len;
Michal Vasko5d24f6c2020-10-13 13:49:06 +02007782 const struct lys_module *moveto_mod = NULL;
Michal Vaskoc71c37b2021-01-11 13:40:02 +01007783 const struct lysc_node *scnode = NULL;
Michal Vasko004d3152020-06-11 19:59:22 +02007784 struct ly_path_predicate *predicates = NULL;
7785 enum ly_path_pred_type pred_type = 0;
Michal Vaskoa036a6b2021-10-12 16:05:23 +02007786 int scnode_skip_pred = 0;
Michal Vaskod3678892020-05-21 10:06:58 +02007787
aPiecek8b0cc152021-05-31 16:40:31 +02007788 LOGDBG(LY_LDGXPATH, "%-27s %s %s[%u]", __func__, (options & LYXP_SKIP_EXPR ? "skipped" : "parsed"),
Michal Vasko49fec8e2022-05-24 10:28:33 +02007789 lyxp_token2str(exp->tokens[*tok_idx]), exp->tok_pos[*tok_idx]);
Michal Vasko004d3152020-06-11 19:59:22 +02007790 ++(*tok_idx);
Michal Vaskod3678892020-05-21 10:06:58 +02007791
aPiecek8b0cc152021-05-31 16:40:31 +02007792 if (options & LYXP_SKIP_EXPR) {
Michal Vaskod3678892020-05-21 10:06:58 +02007793 goto moveto;
7794 }
7795
Michal Vasko004d3152020-06-11 19:59:22 +02007796 ncname = &exp->expr[exp->tok_pos[*tok_idx - 1]];
7797 ncname_len = exp->tok_len[*tok_idx - 1];
Michal Vaskod3678892020-05-21 10:06:58 +02007798
Michal Vaskoc71c37b2021-01-11 13:40:02 +01007799 if ((ncname[0] == '*') && (ncname_len == 1)) {
7800 /* all nodes will match */
7801 goto moveto;
Michal Vasko5d24f6c2020-10-13 13:49:06 +02007802 }
7803
Michal Vaskoc71c37b2021-01-11 13:40:02 +01007804 /* parse (and skip) module name */
7805 rc = moveto_resolve_model(&ncname, &ncname_len, set, NULL, &moveto_mod);
Michal Vaskod3678892020-05-21 10:06:58 +02007806 LY_CHECK_GOTO(rc, cleanup);
7807
Michal Vasko49fec8e2022-05-24 10:28:33 +02007808 if ((ncname[0] == '*') && (ncname_len == 1)) {
7809 /* all nodes from the module will match */
7810 goto moveto;
7811 }
7812
7813 if (((set->format == LY_VALUE_JSON) || moveto_mod) && (axis == LYXP_AXIS_CHILD) && !all_desc &&
7814 (set->type == LYXP_SET_NODE_SET)) {
Michal Vaskod3678892020-05-21 10:06:58 +02007815 /* find the matching schema node in some parent in the context */
Radek Krejci1deb5be2020-08-26 16:43:36 +02007816 for (uint32_t i = 0; i < set->used; ++i) {
Michal Vaskoc71c37b2021-01-11 13:40:02 +01007817 if (eval_name_test_with_predicate_get_scnode(set->ctx, set->val.nodes[i].node, ncname, ncname_len,
7818 moveto_mod, set->root_type, set->format, &scnode)) {
Michal Vasko5d24f6c2020-10-13 13:49:06 +02007819 /* check failed */
7820 scnode = NULL;
7821 break;
Michal Vaskod3678892020-05-21 10:06:58 +02007822 }
7823 }
7824
Michal Vasko004d3152020-06-11 19:59:22 +02007825 if (scnode && (scnode->nodetype & (LYS_LIST | LYS_LEAFLIST))) {
7826 /* try to create the predicates */
Michal Vasko5d24f6c2020-10-13 13:49:06 +02007827 if (eval_name_test_try_compile_predicates(exp, tok_idx, scnode, set->cur_mod, set->cur_node ?
7828 set->cur_node->schema : NULL, set->format, set->prefix_data, &predicates, &pred_type)) {
Michal Vasko004d3152020-06-11 19:59:22 +02007829 /* hashes cannot be used */
Michal Vaskod3678892020-05-21 10:06:58 +02007830 scnode = NULL;
7831 }
7832 }
7833 }
7834
Michal Vaskoc71c37b2021-01-11 13:40:02 +01007835 if (!scnode) {
7836 /* we are not able to match based on a schema node and not all the modules match ("*"),
Michal Vasko004d3152020-06-11 19:59:22 +02007837 * use dictionary for efficient comparison */
Radek Krejci011e4aa2020-09-04 15:22:31 +02007838 LY_CHECK_GOTO(rc = lydict_insert(set->ctx, ncname, ncname_len, &ncname_dict), cleanup);
Michal Vaskod3678892020-05-21 10:06:58 +02007839 }
7840
7841moveto:
7842 /* move to the attribute(s), data node(s), or schema node(s) */
Michal Vasko49fec8e2022-05-24 10:28:33 +02007843 if (axis == LYXP_AXIS_ATTRIBUTE) {
aPiecek8b0cc152021-05-31 16:40:31 +02007844 if (!(options & LYXP_SKIP_EXPR) && (options & LYXP_SCNODE_ALL)) {
Michal Vasko1a09b212021-05-06 13:00:10 +02007845 set_scnode_clear_ctx(set, LYXP_SET_SCNODE_ATOM_NODE);
Michal Vaskod3678892020-05-21 10:06:58 +02007846 } else {
7847 if (all_desc) {
Michal Vaskocdad7122020-11-09 21:04:44 +01007848 rc = moveto_attr_alldesc(set, moveto_mod, ncname_dict, options);
Michal Vaskod3678892020-05-21 10:06:58 +02007849 } else {
aPiecek8b0cc152021-05-31 16:40:31 +02007850 rc = moveto_attr(set, moveto_mod, ncname_dict, options);
Michal Vaskod3678892020-05-21 10:06:58 +02007851 }
7852 LY_CHECK_GOTO(rc, cleanup);
7853 }
7854 } else {
aPiecek8b0cc152021-05-31 16:40:31 +02007855 if (!(options & LYXP_SKIP_EXPR) && (options & LYXP_SCNODE_ALL)) {
Radek Krejci1deb5be2020-08-26 16:43:36 +02007856 int64_t i;
Michal Vaskoa036a6b2021-10-12 16:05:23 +02007857 const struct lyxp_set_scnode *scparent = NULL;
Michal Vaskoa036a6b2021-10-12 16:05:23 +02007858
7859 /* remember parent if there is only one, to print in the warning */
7860 for (i = 0; i < set->used; ++i) {
7861 if (set->val.scnodes[i].in_ctx == LYXP_SET_SCNODE_ATOM_CTX) {
7862 if (!scparent) {
7863 /* remember the context node */
7864 scparent = &set->val.scnodes[i];
7865 } else {
7866 /* several context nodes, no reasonable error possible */
7867 scparent = NULL;
7868 break;
7869 }
7870 }
7871 }
Radek Krejci1deb5be2020-08-26 16:43:36 +02007872
Michal Vasko49fec8e2022-05-24 10:28:33 +02007873 if (all_desc && (axis == LYXP_AXIS_CHILD)) {
7874 /* efficient evaluation that does not add all the descendants into the set */
7875 rc = moveto_scnode_alldesc_child(set, moveto_mod, ncname_dict, options);
Michal Vaskod3678892020-05-21 10:06:58 +02007876 } else {
Michal Vasko49fec8e2022-05-24 10:28:33 +02007877 if (all_desc) {
7878 /* "//" == "/descendant-or-self::node()/" */
7879 rc = xpath_pi_node(set, LYXP_AXIS_DESCENDANT_OR_SELF, options);
7880 LY_CHECK_GOTO(rc, cleanup);
7881 }
7882 rc = moveto_scnode(set, moveto_mod, ncname_dict, axis, options);
Michal Vaskod3678892020-05-21 10:06:58 +02007883 }
7884 LY_CHECK_GOTO(rc, cleanup);
7885
7886 for (i = set->used - 1; i > -1; --i) {
Michal Vasko1a09b212021-05-06 13:00:10 +02007887 if (set->val.scnodes[i].in_ctx > LYXP_SET_SCNODE_ATOM_NODE) {
Michal Vaskod3678892020-05-21 10:06:58 +02007888 break;
7889 }
7890 }
7891 if (i == -1) {
Michal Vasko4ad69e72021-10-26 16:25:55 +02007892 /* generate message */
7893 eval_name_test_scnode_no_match_msg(set, scparent, ncname, ncname_len, exp->expr, options);
7894
7895 if (options & LYXP_SCNODE_ERROR) {
7896 /* error */
7897 rc = LY_EVALID;
7898 goto cleanup;
Michal Vaskoa036a6b2021-10-12 16:05:23 +02007899 }
Michal Vaskoa036a6b2021-10-12 16:05:23 +02007900
7901 /* skip the predicates and the rest of this path to not generate invalid warnings */
7902 rc = LY_ENOT;
7903 scnode_skip_pred = 1;
Michal Vaskod3678892020-05-21 10:06:58 +02007904 }
7905 } else {
Michal Vasko49fec8e2022-05-24 10:28:33 +02007906 if (all_desc && (axis == LYXP_AXIS_CHILD)) {
7907 /* efficient evaluation */
7908 rc = moveto_node_alldesc_child(set, moveto_mod, ncname_dict, options);
7909 } else if (scnode && (axis == LYXP_AXIS_CHILD)) {
7910 /* we can find the child nodes using hashes */
7911 rc = moveto_node_hash_child(set, scnode, predicates, options);
Michal Vaskod3678892020-05-21 10:06:58 +02007912 } else {
Michal Vasko49fec8e2022-05-24 10:28:33 +02007913 if (all_desc) {
7914 /* "//" == "/descendant-or-self::node()/" */
7915 rc = xpath_pi_node(set, LYXP_AXIS_DESCENDANT_OR_SELF, options);
7916 LY_CHECK_GOTO(rc, cleanup);
Michal Vaskod3678892020-05-21 10:06:58 +02007917 }
Michal Vasko49fec8e2022-05-24 10:28:33 +02007918 rc = moveto_node(set, moveto_mod, ncname_dict, axis, options);
Michal Vaskod3678892020-05-21 10:06:58 +02007919 }
7920 LY_CHECK_GOTO(rc, cleanup);
7921 }
7922 }
7923
Michal Vaskoa036a6b2021-10-12 16:05:23 +02007924 if (scnode_skip_pred) {
7925 /* skip predicates */
7926 options |= LYXP_SKIP_EXPR;
7927 }
7928
Michal Vaskod3678892020-05-21 10:06:58 +02007929 /* Predicate* */
Michal Vasko004d3152020-06-11 19:59:22 +02007930 while (!lyxp_check_token(NULL, exp, *tok_idx, LYXP_TOKEN_BRACK1)) {
Michal Vasko49fec8e2022-05-24 10:28:33 +02007931 r = eval_predicate(exp, tok_idx, set, options, axis);
Michal Vaskoa036a6b2021-10-12 16:05:23 +02007932 LY_CHECK_ERR_GOTO(r, rc = r, cleanup);
Michal Vaskod3678892020-05-21 10:06:58 +02007933 }
7934
7935cleanup:
Michal Vaskoa036a6b2021-10-12 16:05:23 +02007936 if (scnode_skip_pred) {
7937 /* restore options */
7938 options &= ~LYXP_SKIP_EXPR;
7939 }
aPiecek8b0cc152021-05-31 16:40:31 +02007940 if (!(options & LYXP_SKIP_EXPR)) {
Michal Vasko004d3152020-06-11 19:59:22 +02007941 lydict_remove(set->ctx, ncname_dict);
Michal Vaskof7e16e22020-10-21 09:24:39 +02007942 ly_path_predicates_free(set->ctx, pred_type, predicates);
Michal Vaskodb51a8d2020-05-27 15:22:29 +02007943 }
Michal Vaskod3678892020-05-21 10:06:58 +02007944 return rc;
7945}
7946
7947/**
7948 * @brief Evaluate NodeType and any following Predicates. Logs directly on error.
7949 *
7950 * [5] Step ::= '@'? NodeTest Predicate* | '.' | '..'
7951 * [6] NodeTest ::= NameTest | NodeType '(' ')'
7952 * [8] NodeType ::= 'text' | 'node'
7953 *
7954 * @param[in] exp Parsed XPath expression.
Michal Vasko004d3152020-06-11 19:59:22 +02007955 * @param[in] tok_idx Position in the expression @p exp.
Michal Vasko49fec8e2022-05-24 10:28:33 +02007956 * @param[in] axis Axis to search on.
7957 * @param[in] all_desc Whether to search all the descendants or axis only.
aPiecek8b0cc152021-05-31 16:40:31 +02007958 * @param[in,out] set Context and result set.
Michal Vaskod3678892020-05-21 10:06:58 +02007959 * @param[in] options XPath options.
7960 * @return LY_ERR (LY_EINCOMPLETE on unresolved when)
7961 */
7962static LY_ERR
Michal Vasko49fec8e2022-05-24 10:28:33 +02007963eval_node_type_with_predicate(const struct lyxp_expr *exp, uint16_t *tok_idx, enum lyxp_axis axis, ly_bool all_desc,
Radek Krejci1deb5be2020-08-26 16:43:36 +02007964 struct lyxp_set *set, uint32_t options)
Michal Vaskod3678892020-05-21 10:06:58 +02007965{
7966 LY_ERR rc;
7967
Michal Vaskod3678892020-05-21 10:06:58 +02007968 (void)all_desc;
7969
aPiecek8b0cc152021-05-31 16:40:31 +02007970 if (!(options & LYXP_SKIP_EXPR)) {
Michal Vasko004d3152020-06-11 19:59:22 +02007971 assert(exp->tok_len[*tok_idx] == 4);
Michal Vasko49fec8e2022-05-24 10:28:33 +02007972 if (!strncmp(&exp->expr[exp->tok_pos[*tok_idx]], "node", 4)) {
7973 rc = xpath_pi_node(set, axis, options);
Michal Vaskod3678892020-05-21 10:06:58 +02007974 } else {
Michal Vasko49fec8e2022-05-24 10:28:33 +02007975 assert(!strncmp(&exp->expr[exp->tok_pos[*tok_idx]], "text", 4));
7976 rc = xpath_pi_text(set, axis, options);
Michal Vaskod3678892020-05-21 10:06:58 +02007977 }
Michal Vasko49fec8e2022-05-24 10:28:33 +02007978 LY_CHECK_RET(rc);
Michal Vaskod3678892020-05-21 10:06:58 +02007979 }
aPiecek8b0cc152021-05-31 16:40:31 +02007980 LOGDBG(LY_LDGXPATH, "%-27s %s %s[%u]", __func__, (options & LYXP_SKIP_EXPR ? "skipped" : "parsed"),
Michal Vasko49fec8e2022-05-24 10:28:33 +02007981 lyxp_token2str(exp->tokens[*tok_idx]), exp->tok_pos[*tok_idx]);
Michal Vasko004d3152020-06-11 19:59:22 +02007982 ++(*tok_idx);
Michal Vaskod3678892020-05-21 10:06:58 +02007983
7984 /* '(' */
Michal Vasko004d3152020-06-11 19:59:22 +02007985 assert(exp->tokens[*tok_idx] == LYXP_TOKEN_PAR1);
aPiecek8b0cc152021-05-31 16:40:31 +02007986 LOGDBG(LY_LDGXPATH, "%-27s %s %s[%u]", __func__, (options & LYXP_SKIP_EXPR ? "skipped" : "parsed"),
Michal Vasko49fec8e2022-05-24 10:28:33 +02007987 lyxp_token2str(exp->tokens[*tok_idx]), exp->tok_pos[*tok_idx]);
Michal Vasko004d3152020-06-11 19:59:22 +02007988 ++(*tok_idx);
Michal Vaskod3678892020-05-21 10:06:58 +02007989
7990 /* ')' */
Michal Vasko004d3152020-06-11 19:59:22 +02007991 assert(exp->tokens[*tok_idx] == LYXP_TOKEN_PAR2);
aPiecek8b0cc152021-05-31 16:40:31 +02007992 LOGDBG(LY_LDGXPATH, "%-27s %s %s[%u]", __func__, (options & LYXP_SKIP_EXPR ? "skipped" : "parsed"),
Michal Vasko49fec8e2022-05-24 10:28:33 +02007993 lyxp_token2str(exp->tokens[*tok_idx]), exp->tok_pos[*tok_idx]);
Michal Vasko004d3152020-06-11 19:59:22 +02007994 ++(*tok_idx);
Michal Vaskod3678892020-05-21 10:06:58 +02007995
7996 /* Predicate* */
Michal Vasko004d3152020-06-11 19:59:22 +02007997 while (!lyxp_check_token(NULL, exp, *tok_idx, LYXP_TOKEN_BRACK1)) {
Michal Vasko49fec8e2022-05-24 10:28:33 +02007998 rc = eval_predicate(exp, tok_idx, set, options, axis);
Michal Vaskod3678892020-05-21 10:06:58 +02007999 LY_CHECK_RET(rc);
8000 }
8001
8002 return LY_SUCCESS;
8003}
8004
8005/**
Michal Vasko03ff5a72019-09-11 13:49:33 +02008006 * @brief Evaluate RelativeLocationPath. Logs directly on error.
8007 *
8008 * [4] RelativeLocationPath ::= Step | RelativeLocationPath '/' Step | RelativeLocationPath '//' Step
8009 * [5] Step ::= '@'? NodeTest Predicate* | '.' | '..'
Michal Vaskod3678892020-05-21 10:06:58 +02008010 * [6] NodeTest ::= NameTest | NodeType '(' ')'
Michal Vasko03ff5a72019-09-11 13:49:33 +02008011 *
8012 * @param[in] exp Parsed XPath expression.
Michal Vasko004d3152020-06-11 19:59:22 +02008013 * @param[in] tok_idx Position in the expression @p exp.
Michal Vasko03ff5a72019-09-11 13:49:33 +02008014 * @param[in] all_desc Whether to search all the descendants or children only.
aPiecek8b0cc152021-05-31 16:40:31 +02008015 * @param[in,out] set Context and result set.
Michal Vasko03ff5a72019-09-11 13:49:33 +02008016 * @param[in] options XPath options.
8017 * @return LY_ERR (YL_EINCOMPLETE on unresolved when)
8018 */
8019static LY_ERR
Michal Vasko40308e72020-10-20 16:38:40 +02008020eval_relative_location_path(const struct lyxp_expr *exp, uint16_t *tok_idx, ly_bool all_desc, struct lyxp_set *set,
8021 uint32_t options)
Michal Vasko03ff5a72019-09-11 13:49:33 +02008022{
Michal Vaskoa036a6b2021-10-12 16:05:23 +02008023 LY_ERR rc = LY_SUCCESS;
Michal Vasko49fec8e2022-05-24 10:28:33 +02008024 enum lyxp_axis axis;
Michal Vaskoa036a6b2021-10-12 16:05:23 +02008025 int scnode_skip_path = 0;
Michal Vasko03ff5a72019-09-11 13:49:33 +02008026
8027 goto step;
8028 do {
8029 /* evaluate '/' or '//' */
Michal Vasko004d3152020-06-11 19:59:22 +02008030 if (exp->tok_len[*tok_idx] == 1) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02008031 all_desc = 0;
8032 } else {
Michal Vasko004d3152020-06-11 19:59:22 +02008033 assert(exp->tok_len[*tok_idx] == 2);
Michal Vasko03ff5a72019-09-11 13:49:33 +02008034 all_desc = 1;
8035 }
aPiecek8b0cc152021-05-31 16:40:31 +02008036 LOGDBG(LY_LDGXPATH, "%-27s %s %s[%u]", __func__, (options & LYXP_SKIP_EXPR ? "skipped" : "parsed"),
Michal Vasko49fec8e2022-05-24 10:28:33 +02008037 lyxp_token2str(exp->tokens[*tok_idx]), exp->tok_pos[*tok_idx]);
Michal Vasko004d3152020-06-11 19:59:22 +02008038 ++(*tok_idx);
Michal Vasko03ff5a72019-09-11 13:49:33 +02008039
8040step:
Michal Vasko49fec8e2022-05-24 10:28:33 +02008041 /* AxisSpecifier */
8042 if (exp->tokens[*tok_idx] == LYXP_TOKEN_AXISNAME) {
8043 axis = str2axis(exp->expr + exp->tok_pos[*tok_idx], exp->tok_len[*tok_idx]);
Michal Vaskod3678892020-05-21 10:06:58 +02008044
aPiecek8b0cc152021-05-31 16:40:31 +02008045 LOGDBG(LY_LDGXPATH, "%-27s %s %s[%u]", __func__, (options & LYXP_SKIP_EXPR ? "skipped" : "parsed"),
Michal Vasko49fec8e2022-05-24 10:28:33 +02008046 lyxp_token2str(exp->tokens[*tok_idx]), exp->tok_pos[*tok_idx]);
8047 ++(*tok_idx);
8048
8049 assert(exp->tokens[*tok_idx] == LYXP_TOKEN_DCOLON);
8050 LOGDBG(LY_LDGXPATH, "%-27s %s %s[%u]", __func__, (options & LYXP_SKIP_EXPR ? "skipped" : "parsed"),
8051 lyxp_token2str(exp->tokens[*tok_idx]), exp->tok_pos[*tok_idx]);
8052 ++(*tok_idx);
8053 } else if (exp->tokens[*tok_idx] == LYXP_TOKEN_AT) {
8054 axis = LYXP_AXIS_ATTRIBUTE;
8055
8056 LOGDBG(LY_LDGXPATH, "%-27s %s %s[%u]", __func__, (options & LYXP_SKIP_EXPR ? "skipped" : "parsed"),
8057 lyxp_token2str(exp->tokens[*tok_idx]), exp->tok_pos[*tok_idx]);
Michal Vasko004d3152020-06-11 19:59:22 +02008058 ++(*tok_idx);
Michal Vaskod3678892020-05-21 10:06:58 +02008059 } else {
Michal Vasko49fec8e2022-05-24 10:28:33 +02008060 /* default */
8061 axis = LYXP_AXIS_CHILD;
Michal Vaskod3678892020-05-21 10:06:58 +02008062 }
8063
Michal Vasko49fec8e2022-05-24 10:28:33 +02008064 /* NodeTest Predicate* */
Michal Vasko004d3152020-06-11 19:59:22 +02008065 switch (exp->tokens[*tok_idx]) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02008066 case LYXP_TOKEN_DOT:
8067 /* evaluate '.' */
Michal Vasko49fec8e2022-05-24 10:28:33 +02008068 if (!(options & LYXP_SKIP_EXPR)) {
8069 if (((options & LYXP_SCNODE_ALL) && (set->type != LYXP_SET_SCNODE_SET)) ||
8070 (!(options & LYXP_SCNODE_ALL) && (set->type != LYXP_SET_NODE_SET))) {
8071 LOGVAL(set->ctx, LY_VCODE_XP_INOP_1, "path operator", print_set_type(set));
8072 rc = LY_EVALID;
8073 goto cleanup;
8074 }
8075
8076 if (all_desc) {
8077 rc = xpath_pi_node(set, LYXP_AXIS_DESCENDANT_OR_SELF, options);
8078 LY_CHECK_GOTO(rc, cleanup);
8079 }
8080 rc = xpath_pi_node(set, LYXP_AXIS_SELF, options);
8081 LY_CHECK_GOTO(rc, cleanup);
Michal Vasko03ff5a72019-09-11 13:49:33 +02008082 }
Michal Vasko03ff5a72019-09-11 13:49:33 +02008083 LOGDBG(LY_LDGXPATH, "%-27s %s %s[%u]", __func__, (set ? "parsed" : "skipped"),
Michal Vasko49fec8e2022-05-24 10:28:33 +02008084 lyxp_token2str(exp->tokens[*tok_idx]), exp->tok_pos[*tok_idx]);
Michal Vasko004d3152020-06-11 19:59:22 +02008085 ++(*tok_idx);
Michal Vasko03ff5a72019-09-11 13:49:33 +02008086 break;
8087
8088 case LYXP_TOKEN_DDOT:
8089 /* evaluate '..' */
Michal Vasko49fec8e2022-05-24 10:28:33 +02008090 if (!(options & LYXP_SKIP_EXPR)) {
8091 if (((options & LYXP_SCNODE_ALL) && (set->type != LYXP_SET_SCNODE_SET)) ||
8092 (!(options & LYXP_SCNODE_ALL) && (set->type != LYXP_SET_NODE_SET))) {
8093 LOGVAL(set->ctx, LY_VCODE_XP_INOP_1, "path operator", print_set_type(set));
8094 rc = LY_EVALID;
8095 goto cleanup;
8096 }
8097
8098 if (all_desc) {
8099 rc = xpath_pi_node(set, LYXP_AXIS_DESCENDANT_OR_SELF, options);
8100 LY_CHECK_GOTO(rc, cleanup);
8101 }
8102 rc = xpath_pi_node(set, LYXP_AXIS_PARENT, options);
8103 LY_CHECK_GOTO(rc, cleanup);
Michal Vasko03ff5a72019-09-11 13:49:33 +02008104 }
aPiecek8b0cc152021-05-31 16:40:31 +02008105 LOGDBG(LY_LDGXPATH, "%-27s %s %s[%u]", __func__, (options & LYXP_SKIP_EXPR ? "skipped" : "parsed"),
Michal Vasko49fec8e2022-05-24 10:28:33 +02008106 lyxp_token2str(exp->tokens[*tok_idx]), exp->tok_pos[*tok_idx]);
Michal Vasko004d3152020-06-11 19:59:22 +02008107 ++(*tok_idx);
Michal Vasko03ff5a72019-09-11 13:49:33 +02008108 break;
8109
Michal Vasko03ff5a72019-09-11 13:49:33 +02008110 case LYXP_TOKEN_NAMETEST:
Michal Vaskod3678892020-05-21 10:06:58 +02008111 /* evaluate NameTest Predicate* */
Michal Vasko49fec8e2022-05-24 10:28:33 +02008112 rc = eval_name_test_with_predicate(exp, tok_idx, axis, all_desc, set, options);
Michal Vaskoa036a6b2021-10-12 16:05:23 +02008113 if (rc == LY_ENOT) {
8114 assert(options & LYXP_SCNODE_ALL);
8115 /* skip the rest of this path */
8116 rc = LY_SUCCESS;
8117 scnode_skip_path = 1;
8118 options |= LYXP_SKIP_EXPR;
8119 }
8120 LY_CHECK_GOTO(rc, cleanup);
Michal Vaskod3678892020-05-21 10:06:58 +02008121 break;
Michal Vasko03ff5a72019-09-11 13:49:33 +02008122
Michal Vaskod3678892020-05-21 10:06:58 +02008123 case LYXP_TOKEN_NODETYPE:
8124 /* evaluate NodeType Predicate* */
Michal Vasko49fec8e2022-05-24 10:28:33 +02008125 rc = eval_node_type_with_predicate(exp, tok_idx, axis, all_desc, set, options);
Michal Vaskoa036a6b2021-10-12 16:05:23 +02008126 LY_CHECK_GOTO(rc, cleanup);
Michal Vasko03ff5a72019-09-11 13:49:33 +02008127 break;
8128
8129 default:
Michal Vaskoa036a6b2021-10-12 16:05:23 +02008130 LOGINT(set->ctx);
8131 rc = LY_EINT;
8132 goto cleanup;
Michal Vasko03ff5a72019-09-11 13:49:33 +02008133 }
Michal Vasko004d3152020-06-11 19:59:22 +02008134 } while (!exp_check_token2(NULL, exp, *tok_idx, LYXP_TOKEN_OPER_PATH, LYXP_TOKEN_OPER_RPATH));
Michal Vasko03ff5a72019-09-11 13:49:33 +02008135
Michal Vaskoa036a6b2021-10-12 16:05:23 +02008136cleanup:
8137 if (scnode_skip_path) {
8138 options &= ~LYXP_SKIP_EXPR;
8139 }
8140 return rc;
Michal Vasko03ff5a72019-09-11 13:49:33 +02008141}
8142
8143/**
8144 * @brief Evaluate AbsoluteLocationPath. Logs directly on error.
8145 *
8146 * [3] AbsoluteLocationPath ::= '/' RelativeLocationPath? | '//' RelativeLocationPath
8147 *
8148 * @param[in] exp Parsed XPath expression.
Michal Vasko004d3152020-06-11 19:59:22 +02008149 * @param[in] tok_idx Position in the expression @p exp.
aPiecek8b0cc152021-05-31 16:40:31 +02008150 * @param[in,out] set Context and result set.
Michal Vasko03ff5a72019-09-11 13:49:33 +02008151 * @param[in] options XPath options.
8152 * @return LY_ERR (LY_EINCOMPLETE on unresolved when)
8153 */
8154static LY_ERR
Michal Vasko40308e72020-10-20 16:38:40 +02008155eval_absolute_location_path(const struct lyxp_expr *exp, uint16_t *tok_idx, struct lyxp_set *set, uint32_t options)
Michal Vasko03ff5a72019-09-11 13:49:33 +02008156{
Radek Krejci857189e2020-09-01 13:26:36 +02008157 ly_bool all_desc;
Michal Vasko03ff5a72019-09-11 13:49:33 +02008158
aPiecek8b0cc152021-05-31 16:40:31 +02008159 if (!(options & LYXP_SKIP_EXPR)) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02008160 /* no matter what tokens follow, we need to be at the root */
Michal Vaskob0099a92020-08-31 14:55:23 +02008161 LY_CHECK_RET(moveto_root(set, options));
Michal Vasko03ff5a72019-09-11 13:49:33 +02008162 }
8163
8164 /* '/' RelativeLocationPath? */
Michal Vasko004d3152020-06-11 19:59:22 +02008165 if (exp->tok_len[*tok_idx] == 1) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02008166 /* evaluate '/' - deferred */
8167 all_desc = 0;
aPiecek8b0cc152021-05-31 16:40:31 +02008168 LOGDBG(LY_LDGXPATH, "%-27s %s %s[%u]", __func__, (options & LYXP_SKIP_EXPR ? "skipped" : "parsed"),
Michal Vasko49fec8e2022-05-24 10:28:33 +02008169 lyxp_token2str(exp->tokens[*tok_idx]), exp->tok_pos[*tok_idx]);
Michal Vasko004d3152020-06-11 19:59:22 +02008170 ++(*tok_idx);
Michal Vasko03ff5a72019-09-11 13:49:33 +02008171
Michal Vasko004d3152020-06-11 19:59:22 +02008172 if (lyxp_check_token(NULL, exp, *tok_idx, LYXP_TOKEN_NONE)) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02008173 return LY_SUCCESS;
8174 }
Michal Vasko004d3152020-06-11 19:59:22 +02008175 switch (exp->tokens[*tok_idx]) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02008176 case LYXP_TOKEN_DOT:
8177 case LYXP_TOKEN_DDOT:
Michal Vasko49fec8e2022-05-24 10:28:33 +02008178 case LYXP_TOKEN_AXISNAME:
Michal Vasko03ff5a72019-09-11 13:49:33 +02008179 case LYXP_TOKEN_AT:
8180 case LYXP_TOKEN_NAMETEST:
8181 case LYXP_TOKEN_NODETYPE:
Michal Vaskob0099a92020-08-31 14:55:23 +02008182 LY_CHECK_RET(eval_relative_location_path(exp, tok_idx, all_desc, set, options));
Michal Vasko03ff5a72019-09-11 13:49:33 +02008183 break;
8184 default:
8185 break;
8186 }
8187
Michal Vasko03ff5a72019-09-11 13:49:33 +02008188 } else {
Radek Krejcif6a11002020-08-21 13:29:07 +02008189 /* '//' RelativeLocationPath */
Michal Vasko03ff5a72019-09-11 13:49:33 +02008190 /* evaluate '//' - deferred so as not to waste memory by remembering all the nodes */
8191 all_desc = 1;
aPiecek8b0cc152021-05-31 16:40:31 +02008192 LOGDBG(LY_LDGXPATH, "%-27s %s %s[%u]", __func__, (options & LYXP_SKIP_EXPR ? "skipped" : "parsed"),
Michal Vasko49fec8e2022-05-24 10:28:33 +02008193 lyxp_token2str(exp->tokens[*tok_idx]), exp->tok_pos[*tok_idx]);
Michal Vasko004d3152020-06-11 19:59:22 +02008194 ++(*tok_idx);
Michal Vasko03ff5a72019-09-11 13:49:33 +02008195
Michal Vaskob0099a92020-08-31 14:55:23 +02008196 LY_CHECK_RET(eval_relative_location_path(exp, tok_idx, all_desc, set, options));
Michal Vasko03ff5a72019-09-11 13:49:33 +02008197 }
8198
8199 return LY_SUCCESS;
8200}
8201
8202/**
8203 * @brief Evaluate FunctionCall. Logs directly on error.
8204 *
Michal Vaskod3678892020-05-21 10:06:58 +02008205 * [11] FunctionCall ::= FunctionName '(' ( Expr ( ',' Expr )* )? ')'
Michal Vasko03ff5a72019-09-11 13:49:33 +02008206 *
8207 * @param[in] exp Parsed XPath expression.
Michal Vasko004d3152020-06-11 19:59:22 +02008208 * @param[in] tok_idx Position in the expression @p exp.
aPiecek8b0cc152021-05-31 16:40:31 +02008209 * @param[in,out] set Context and result set.
Michal Vasko03ff5a72019-09-11 13:49:33 +02008210 * @param[in] options XPath options.
8211 * @return LY_ERR (LY_EINCOMPLETE on unresolved when)
8212 */
8213static LY_ERR
Michal Vasko40308e72020-10-20 16:38:40 +02008214eval_function_call(const struct lyxp_expr *exp, uint16_t *tok_idx, struct lyxp_set *set, uint32_t options)
Michal Vasko03ff5a72019-09-11 13:49:33 +02008215{
8216 LY_ERR rc;
Michal Vasko69730152020-10-09 16:30:07 +02008217
Radek Krejci1deb5be2020-08-26 16:43:36 +02008218 LY_ERR (*xpath_func)(struct lyxp_set **, uint16_t, struct lyxp_set *, uint32_t) = NULL;
Michal Vasko0cbf54f2019-12-16 10:01:06 +01008219 uint16_t arg_count = 0, i;
Michal Vasko03ff5a72019-09-11 13:49:33 +02008220 struct lyxp_set **args = NULL, **args_aux;
8221
aPiecek8b0cc152021-05-31 16:40:31 +02008222 if (!(options & LYXP_SKIP_EXPR)) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02008223 /* FunctionName */
Michal Vasko004d3152020-06-11 19:59:22 +02008224 switch (exp->tok_len[*tok_idx]) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02008225 case 3:
Michal Vasko004d3152020-06-11 19:59:22 +02008226 if (!strncmp(&exp->expr[exp->tok_pos[*tok_idx]], "not", 3)) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02008227 xpath_func = &xpath_not;
Michal Vasko004d3152020-06-11 19:59:22 +02008228 } else if (!strncmp(&exp->expr[exp->tok_pos[*tok_idx]], "sum", 3)) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02008229 xpath_func = &xpath_sum;
8230 }
8231 break;
8232 case 4:
Michal Vasko004d3152020-06-11 19:59:22 +02008233 if (!strncmp(&exp->expr[exp->tok_pos[*tok_idx]], "lang", 4)) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02008234 xpath_func = &xpath_lang;
Michal Vasko004d3152020-06-11 19:59:22 +02008235 } else if (!strncmp(&exp->expr[exp->tok_pos[*tok_idx]], "last", 4)) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02008236 xpath_func = &xpath_last;
Michal Vasko004d3152020-06-11 19:59:22 +02008237 } else if (!strncmp(&exp->expr[exp->tok_pos[*tok_idx]], "name", 4)) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02008238 xpath_func = &xpath_name;
Michal Vasko004d3152020-06-11 19:59:22 +02008239 } else if (!strncmp(&exp->expr[exp->tok_pos[*tok_idx]], "true", 4)) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02008240 xpath_func = &xpath_true;
8241 }
8242 break;
8243 case 5:
Michal Vasko004d3152020-06-11 19:59:22 +02008244 if (!strncmp(&exp->expr[exp->tok_pos[*tok_idx]], "count", 5)) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02008245 xpath_func = &xpath_count;
Michal Vasko004d3152020-06-11 19:59:22 +02008246 } else if (!strncmp(&exp->expr[exp->tok_pos[*tok_idx]], "false", 5)) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02008247 xpath_func = &xpath_false;
Michal Vasko004d3152020-06-11 19:59:22 +02008248 } else if (!strncmp(&exp->expr[exp->tok_pos[*tok_idx]], "floor", 5)) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02008249 xpath_func = &xpath_floor;
Michal Vasko004d3152020-06-11 19:59:22 +02008250 } else if (!strncmp(&exp->expr[exp->tok_pos[*tok_idx]], "round", 5)) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02008251 xpath_func = &xpath_round;
Michal Vasko004d3152020-06-11 19:59:22 +02008252 } else if (!strncmp(&exp->expr[exp->tok_pos[*tok_idx]], "deref", 5)) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02008253 xpath_func = &xpath_deref;
8254 }
8255 break;
8256 case 6:
Michal Vasko004d3152020-06-11 19:59:22 +02008257 if (!strncmp(&exp->expr[exp->tok_pos[*tok_idx]], "concat", 6)) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02008258 xpath_func = &xpath_concat;
Michal Vasko004d3152020-06-11 19:59:22 +02008259 } else if (!strncmp(&exp->expr[exp->tok_pos[*tok_idx]], "number", 6)) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02008260 xpath_func = &xpath_number;
Michal Vasko004d3152020-06-11 19:59:22 +02008261 } else if (!strncmp(&exp->expr[exp->tok_pos[*tok_idx]], "string", 6)) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02008262 xpath_func = &xpath_string;
8263 }
8264 break;
8265 case 7:
Michal Vasko004d3152020-06-11 19:59:22 +02008266 if (!strncmp(&exp->expr[exp->tok_pos[*tok_idx]], "boolean", 7)) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02008267 xpath_func = &xpath_boolean;
Michal Vasko004d3152020-06-11 19:59:22 +02008268 } else if (!strncmp(&exp->expr[exp->tok_pos[*tok_idx]], "ceiling", 7)) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02008269 xpath_func = &xpath_ceiling;
Michal Vasko004d3152020-06-11 19:59:22 +02008270 } else if (!strncmp(&exp->expr[exp->tok_pos[*tok_idx]], "current", 7)) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02008271 xpath_func = &xpath_current;
8272 }
8273 break;
8274 case 8:
Michal Vasko004d3152020-06-11 19:59:22 +02008275 if (!strncmp(&exp->expr[exp->tok_pos[*tok_idx]], "contains", 8)) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02008276 xpath_func = &xpath_contains;
Michal Vasko004d3152020-06-11 19:59:22 +02008277 } else if (!strncmp(&exp->expr[exp->tok_pos[*tok_idx]], "position", 8)) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02008278 xpath_func = &xpath_position;
Michal Vasko004d3152020-06-11 19:59:22 +02008279 } else if (!strncmp(&exp->expr[exp->tok_pos[*tok_idx]], "re-match", 8)) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02008280 xpath_func = &xpath_re_match;
8281 }
8282 break;
8283 case 9:
Michal Vasko004d3152020-06-11 19:59:22 +02008284 if (!strncmp(&exp->expr[exp->tok_pos[*tok_idx]], "substring", 9)) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02008285 xpath_func = &xpath_substring;
Michal Vasko004d3152020-06-11 19:59:22 +02008286 } else if (!strncmp(&exp->expr[exp->tok_pos[*tok_idx]], "translate", 9)) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02008287 xpath_func = &xpath_translate;
8288 }
8289 break;
8290 case 10:
Michal Vasko004d3152020-06-11 19:59:22 +02008291 if (!strncmp(&exp->expr[exp->tok_pos[*tok_idx]], "local-name", 10)) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02008292 xpath_func = &xpath_local_name;
Michal Vasko004d3152020-06-11 19:59:22 +02008293 } else if (!strncmp(&exp->expr[exp->tok_pos[*tok_idx]], "enum-value", 10)) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02008294 xpath_func = &xpath_enum_value;
Michal Vasko004d3152020-06-11 19:59:22 +02008295 } else if (!strncmp(&exp->expr[exp->tok_pos[*tok_idx]], "bit-is-set", 10)) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02008296 xpath_func = &xpath_bit_is_set;
8297 }
8298 break;
8299 case 11:
Michal Vasko004d3152020-06-11 19:59:22 +02008300 if (!strncmp(&exp->expr[exp->tok_pos[*tok_idx]], "starts-with", 11)) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02008301 xpath_func = &xpath_starts_with;
8302 }
8303 break;
8304 case 12:
Michal Vasko004d3152020-06-11 19:59:22 +02008305 if (!strncmp(&exp->expr[exp->tok_pos[*tok_idx]], "derived-from", 12)) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02008306 xpath_func = &xpath_derived_from;
8307 }
8308 break;
8309 case 13:
Michal Vasko004d3152020-06-11 19:59:22 +02008310 if (!strncmp(&exp->expr[exp->tok_pos[*tok_idx]], "namespace-uri", 13)) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02008311 xpath_func = &xpath_namespace_uri;
Michal Vasko004d3152020-06-11 19:59:22 +02008312 } else if (!strncmp(&exp->expr[exp->tok_pos[*tok_idx]], "string-length", 13)) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02008313 xpath_func = &xpath_string_length;
8314 }
8315 break;
8316 case 15:
Michal Vasko004d3152020-06-11 19:59:22 +02008317 if (!strncmp(&exp->expr[exp->tok_pos[*tok_idx]], "normalize-space", 15)) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02008318 xpath_func = &xpath_normalize_space;
Michal Vasko004d3152020-06-11 19:59:22 +02008319 } else if (!strncmp(&exp->expr[exp->tok_pos[*tok_idx]], "substring-after", 15)) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02008320 xpath_func = &xpath_substring_after;
8321 }
8322 break;
8323 case 16:
Michal Vasko004d3152020-06-11 19:59:22 +02008324 if (!strncmp(&exp->expr[exp->tok_pos[*tok_idx]], "substring-before", 16)) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02008325 xpath_func = &xpath_substring_before;
8326 }
8327 break;
8328 case 20:
Michal Vasko004d3152020-06-11 19:59:22 +02008329 if (!strncmp(&exp->expr[exp->tok_pos[*tok_idx]], "derived-from-or-self", 20)) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02008330 xpath_func = &xpath_derived_from_or_self;
8331 }
8332 break;
8333 }
8334
8335 if (!xpath_func) {
Radek Krejci2efc45b2020-12-22 16:25:44 +01008336 LOGVAL(set->ctx, LY_VCODE_XP_INFUNC, exp->tok_len[*tok_idx], &exp->expr[exp->tok_pos[*tok_idx]]);
Michal Vasko03ff5a72019-09-11 13:49:33 +02008337 return LY_EVALID;
8338 }
8339 }
8340
aPiecek8b0cc152021-05-31 16:40:31 +02008341 LOGDBG(LY_LDGXPATH, "%-27s %s %s[%u]", __func__, (options & LYXP_SKIP_EXPR ? "skipped" : "parsed"),
Michal Vasko49fec8e2022-05-24 10:28:33 +02008342 lyxp_token2str(exp->tokens[*tok_idx]), exp->tok_pos[*tok_idx]);
Michal Vasko004d3152020-06-11 19:59:22 +02008343 ++(*tok_idx);
Michal Vasko03ff5a72019-09-11 13:49:33 +02008344
8345 /* '(' */
Michal Vasko004d3152020-06-11 19:59:22 +02008346 assert(exp->tokens[*tok_idx] == LYXP_TOKEN_PAR1);
aPiecek8b0cc152021-05-31 16:40:31 +02008347 LOGDBG(LY_LDGXPATH, "%-27s %s %s[%u]", __func__, (options & LYXP_SKIP_EXPR ? "skipped" : "parsed"),
Michal Vasko49fec8e2022-05-24 10:28:33 +02008348 lyxp_token2str(exp->tokens[*tok_idx]), exp->tok_pos[*tok_idx]);
Michal Vasko004d3152020-06-11 19:59:22 +02008349 ++(*tok_idx);
Michal Vasko03ff5a72019-09-11 13:49:33 +02008350
8351 /* ( Expr ( ',' Expr )* )? */
Michal Vasko004d3152020-06-11 19:59:22 +02008352 if (exp->tokens[*tok_idx] != LYXP_TOKEN_PAR2) {
aPiecek8b0cc152021-05-31 16:40:31 +02008353 if (!(options & LYXP_SKIP_EXPR)) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02008354 args = malloc(sizeof *args);
8355 LY_CHECK_ERR_GOTO(!args, LOGMEM(set->ctx); rc = LY_EMEM, cleanup);
8356 arg_count = 1;
8357 args[0] = set_copy(set);
8358 if (!args[0]) {
8359 rc = LY_EMEM;
8360 goto cleanup;
8361 }
8362
Michal Vasko004d3152020-06-11 19:59:22 +02008363 rc = eval_expr_select(exp, tok_idx, 0, args[0], options);
Michal Vasko03ff5a72019-09-11 13:49:33 +02008364 LY_CHECK_GOTO(rc, cleanup);
8365 } else {
aPiecek8b0cc152021-05-31 16:40:31 +02008366 rc = eval_expr_select(exp, tok_idx, 0, set, options | LYXP_SKIP_EXPR);
Michal Vasko03ff5a72019-09-11 13:49:33 +02008367 LY_CHECK_GOTO(rc, cleanup);
8368 }
8369 }
Michal Vasko004d3152020-06-11 19:59:22 +02008370 while (!lyxp_check_token(NULL, exp, *tok_idx, LYXP_TOKEN_COMMA)) {
aPiecek8b0cc152021-05-31 16:40:31 +02008371 LOGDBG(LY_LDGXPATH, "%-27s %s %s[%u]", __func__, (options & LYXP_SKIP_EXPR ? "skipped" : "parsed"),
Michal Vasko49fec8e2022-05-24 10:28:33 +02008372 lyxp_token2str(exp->tokens[*tok_idx]), exp->tok_pos[*tok_idx]);
Michal Vasko004d3152020-06-11 19:59:22 +02008373 ++(*tok_idx);
Michal Vasko03ff5a72019-09-11 13:49:33 +02008374
aPiecek8b0cc152021-05-31 16:40:31 +02008375 if (!(options & LYXP_SKIP_EXPR)) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02008376 ++arg_count;
8377 args_aux = realloc(args, arg_count * sizeof *args);
8378 LY_CHECK_ERR_GOTO(!args_aux, arg_count--; LOGMEM(set->ctx); rc = LY_EMEM, cleanup);
8379 args = args_aux;
8380 args[arg_count - 1] = set_copy(set);
8381 if (!args[arg_count - 1]) {
8382 rc = LY_EMEM;
8383 goto cleanup;
8384 }
8385
Michal Vasko004d3152020-06-11 19:59:22 +02008386 rc = eval_expr_select(exp, tok_idx, 0, args[arg_count - 1], options);
Michal Vasko03ff5a72019-09-11 13:49:33 +02008387 LY_CHECK_GOTO(rc, cleanup);
8388 } else {
aPiecek8b0cc152021-05-31 16:40:31 +02008389 rc = eval_expr_select(exp, tok_idx, 0, set, options | LYXP_SKIP_EXPR);
Michal Vasko03ff5a72019-09-11 13:49:33 +02008390 LY_CHECK_GOTO(rc, cleanup);
8391 }
8392 }
8393
8394 /* ')' */
Michal Vasko004d3152020-06-11 19:59:22 +02008395 assert(exp->tokens[*tok_idx] == LYXP_TOKEN_PAR2);
aPiecek8b0cc152021-05-31 16:40:31 +02008396 LOGDBG(LY_LDGXPATH, "%-27s %s %s[%u]", __func__, (options & LYXP_SKIP_EXPR ? "skipped" : "parsed"),
Michal Vasko49fec8e2022-05-24 10:28:33 +02008397 lyxp_token2str(exp->tokens[*tok_idx]), exp->tok_pos[*tok_idx]);
Michal Vasko004d3152020-06-11 19:59:22 +02008398 ++(*tok_idx);
Michal Vasko03ff5a72019-09-11 13:49:33 +02008399
aPiecek8b0cc152021-05-31 16:40:31 +02008400 if (!(options & LYXP_SKIP_EXPR)) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02008401 /* evaluate function */
8402 rc = xpath_func(args, arg_count, set, options);
8403
8404 if (options & LYXP_SCNODE_ALL) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02008405 /* merge all nodes from arg evaluations */
8406 for (i = 0; i < arg_count; ++i) {
Michal Vasko1a09b212021-05-06 13:00:10 +02008407 set_scnode_clear_ctx(args[i], LYXP_SET_SCNODE_ATOM_NODE);
Michal Vaskoecd62de2019-11-13 12:35:11 +01008408 lyxp_set_scnode_merge(set, args[i]);
Michal Vasko03ff5a72019-09-11 13:49:33 +02008409 }
8410 }
8411 } else {
8412 rc = LY_SUCCESS;
8413 }
8414
8415cleanup:
8416 for (i = 0; i < arg_count; ++i) {
8417 lyxp_set_free(args[i]);
8418 }
8419 free(args);
Michal Vasko03ff5a72019-09-11 13:49:33 +02008420 return rc;
8421}
8422
8423/**
8424 * @brief Evaluate Number. Logs directly on error.
8425 *
8426 * @param[in] ctx Context for errors.
8427 * @param[in] exp Parsed XPath expression.
Michal Vasko004d3152020-06-11 19:59:22 +02008428 * @param[in] tok_idx Position in the expression @p exp.
Michal Vasko03ff5a72019-09-11 13:49:33 +02008429 * @param[in,out] set Context and result set. On NULL the rule is only parsed.
8430 * @return LY_ERR
8431 */
8432static LY_ERR
Michal Vasko40308e72020-10-20 16:38:40 +02008433eval_number(struct ly_ctx *ctx, const struct lyxp_expr *exp, uint16_t *tok_idx, struct lyxp_set *set)
Michal Vasko03ff5a72019-09-11 13:49:33 +02008434{
8435 long double num;
8436 char *endptr;
8437
8438 if (set) {
8439 errno = 0;
Michal Vasko004d3152020-06-11 19:59:22 +02008440 num = strtold(&exp->expr[exp->tok_pos[*tok_idx]], &endptr);
Michal Vasko03ff5a72019-09-11 13:49:33 +02008441 if (errno) {
Radek Krejci2efc45b2020-12-22 16:25:44 +01008442 LOGVAL(ctx, LY_VCODE_XP_INTOK, "Unknown", &exp->expr[exp->tok_pos[*tok_idx]]);
8443 LOGVAL(ctx, LYVE_XPATH, "Failed to convert \"%.*s\" into a long double (%s).",
Michal Vasko69730152020-10-09 16:30:07 +02008444 exp->tok_len[*tok_idx], &exp->expr[exp->tok_pos[*tok_idx]], strerror(errno));
Michal Vasko03ff5a72019-09-11 13:49:33 +02008445 return LY_EVALID;
Michal Vasko004d3152020-06-11 19:59:22 +02008446 } else if (endptr - &exp->expr[exp->tok_pos[*tok_idx]] != exp->tok_len[*tok_idx]) {
Radek Krejci2efc45b2020-12-22 16:25:44 +01008447 LOGVAL(ctx, LY_VCODE_XP_INTOK, "Unknown", &exp->expr[exp->tok_pos[*tok_idx]]);
8448 LOGVAL(ctx, LYVE_XPATH, "Failed to convert \"%.*s\" into a long double.",
Michal Vasko69730152020-10-09 16:30:07 +02008449 exp->tok_len[*tok_idx], &exp->expr[exp->tok_pos[*tok_idx]]);
Michal Vasko03ff5a72019-09-11 13:49:33 +02008450 return LY_EVALID;
8451 }
8452
8453 set_fill_number(set, num);
8454 }
8455
8456 LOGDBG(LY_LDGXPATH, "%-27s %s %s[%u]", __func__, (set ? "parsed" : "skipped"),
Michal Vasko49fec8e2022-05-24 10:28:33 +02008457 lyxp_token2str(exp->tokens[*tok_idx]), exp->tok_pos[*tok_idx]);
Michal Vasko004d3152020-06-11 19:59:22 +02008458 ++(*tok_idx);
Michal Vasko03ff5a72019-09-11 13:49:33 +02008459 return LY_SUCCESS;
8460}
8461
aPiecekdf23eee2021-10-07 12:21:50 +02008462LY_ERR
8463lyxp_vars_find(struct lyxp_var *vars, const char *name, size_t name_len, struct lyxp_var **var)
8464{
8465 LY_ERR ret = LY_ENOTFOUND;
8466 LY_ARRAY_COUNT_TYPE u;
8467
8468 assert(vars && name);
8469
8470 name_len = name_len ? name_len : strlen(name);
8471
8472 LY_ARRAY_FOR(vars, u) {
8473 if (!strncmp(vars[u].name, name, name_len)) {
8474 ret = LY_SUCCESS;
8475 break;
8476 }
8477 }
8478
8479 if (var && !ret) {
8480 *var = &vars[u];
8481 }
8482
8483 return ret;
8484}
8485
Michal Vasko03ff5a72019-09-11 13:49:33 +02008486/**
aPiecekfba75362021-10-07 12:39:48 +02008487 * @brief Evaluate VariableReference.
8488 *
8489 * @param[in] exp Parsed XPath expression.
8490 * @param[in] tok_idx Position in the expression @p exp.
8491 * @param[in] vars [Sized array](@ref sizedarrays) of XPath variables.
8492 * @param[in,out] set Context and result set.
8493 * @param[in] options XPath options.
8494 * @return LY_ERR value.
8495 */
8496static LY_ERR
8497eval_variable_reference(const struct lyxp_expr *exp, uint16_t *tok_idx, struct lyxp_set *set, uint32_t options)
8498{
8499 LY_ERR ret;
8500 const char *name;
8501 struct lyxp_var *var;
8502 const struct lyxp_var *vars;
8503 struct lyxp_expr *tokens = NULL;
8504 uint16_t token_index;
8505
8506 vars = set->vars;
8507
Michal Vasko49fec8e2022-05-24 10:28:33 +02008508 /* find out the name and value of the variable */
aPiecekfba75362021-10-07 12:39:48 +02008509 name = &exp->expr[exp->tok_pos[*tok_idx]];
8510 ret = lyxp_vars_find((struct lyxp_var *)vars, name, exp->tok_len[*tok_idx], &var);
8511 LY_CHECK_ERR_RET(ret, LOGERR(set->ctx, ret,
8512 "XPath variable \"%s\" not defined.", name), ret);
8513
Michal Vasko49fec8e2022-05-24 10:28:33 +02008514 /* parse value */
aPiecekfba75362021-10-07 12:39:48 +02008515 ret = lyxp_expr_parse(set->ctx, var->value, 0, 1, &tokens);
8516 LY_CHECK_GOTO(ret, cleanup);
8517
Michal Vasko49fec8e2022-05-24 10:28:33 +02008518 /* evaluate value */
aPiecekfba75362021-10-07 12:39:48 +02008519 token_index = 0;
8520 ret = eval_expr_select(tokens, &token_index, 0, set, options);
8521 LY_CHECK_GOTO(ret, cleanup);
8522
8523cleanup:
8524 lyxp_expr_free(set->ctx, tokens);
8525
8526 return ret;
8527}
8528
8529/**
Michal Vasko03ff5a72019-09-11 13:49:33 +02008530 * @brief Evaluate PathExpr. Logs directly on error.
8531 *
Michal Vaskod3678892020-05-21 10:06:58 +02008532 * [12] PathExpr ::= LocationPath | PrimaryExpr Predicate*
Michal Vasko03ff5a72019-09-11 13:49:33 +02008533 * | PrimaryExpr Predicate* '/' RelativeLocationPath
8534 * | PrimaryExpr Predicate* '//' RelativeLocationPath
8535 * [2] LocationPath ::= RelativeLocationPath | AbsoluteLocationPath
aPiecekfba75362021-10-07 12:39:48 +02008536 * [10] PrimaryExpr ::= VariableReference | '(' Expr ')' | Literal | Number | FunctionCall
Michal Vasko03ff5a72019-09-11 13:49:33 +02008537 *
8538 * @param[in] exp Parsed XPath expression.
Michal Vasko004d3152020-06-11 19:59:22 +02008539 * @param[in] tok_idx Position in the expression @p exp.
aPiecek8b0cc152021-05-31 16:40:31 +02008540 * @param[in,out] set Context and result set.
Michal Vasko03ff5a72019-09-11 13:49:33 +02008541 * @param[in] options XPath options.
8542 * @return LY_ERR (LY_EINCOMPLETE on unresolved when)
8543 */
8544static LY_ERR
Michal Vasko40308e72020-10-20 16:38:40 +02008545eval_path_expr(const struct lyxp_expr *exp, uint16_t *tok_idx, struct lyxp_set *set, uint32_t options)
Michal Vasko03ff5a72019-09-11 13:49:33 +02008546{
Michal Vasko49fec8e2022-05-24 10:28:33 +02008547 ly_bool all_desc;
Michal Vasko03ff5a72019-09-11 13:49:33 +02008548 LY_ERR rc;
8549
Michal Vasko004d3152020-06-11 19:59:22 +02008550 switch (exp->tokens[*tok_idx]) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02008551 case LYXP_TOKEN_PAR1:
8552 /* '(' Expr ')' */
8553
8554 /* '(' */
aPiecek8b0cc152021-05-31 16:40:31 +02008555 LOGDBG(LY_LDGXPATH, "%-27s %s %s[%u]", __func__, (options & LYXP_SKIP_EXPR ? "skipped" : "parsed"),
Michal Vasko49fec8e2022-05-24 10:28:33 +02008556 lyxp_token2str(exp->tokens[*tok_idx]), exp->tok_pos[*tok_idx]);
Michal Vasko004d3152020-06-11 19:59:22 +02008557 ++(*tok_idx);
Michal Vasko03ff5a72019-09-11 13:49:33 +02008558
8559 /* Expr */
Michal Vasko004d3152020-06-11 19:59:22 +02008560 rc = eval_expr_select(exp, tok_idx, 0, set, options);
Michal Vasko03ff5a72019-09-11 13:49:33 +02008561 LY_CHECK_RET(rc);
8562
8563 /* ')' */
Michal Vasko004d3152020-06-11 19:59:22 +02008564 assert(exp->tokens[*tok_idx] == LYXP_TOKEN_PAR2);
aPiecek8b0cc152021-05-31 16:40:31 +02008565 LOGDBG(LY_LDGXPATH, "%-27s %s %s[%u]", __func__, (options & LYXP_SKIP_EXPR ? "skipped" : "parsed"),
Michal Vasko49fec8e2022-05-24 10:28:33 +02008566 lyxp_token2str(exp->tokens[*tok_idx]), exp->tok_pos[*tok_idx]);
Michal Vasko004d3152020-06-11 19:59:22 +02008567 ++(*tok_idx);
Michal Vasko03ff5a72019-09-11 13:49:33 +02008568
Michal Vasko03ff5a72019-09-11 13:49:33 +02008569 goto predicate;
8570
8571 case LYXP_TOKEN_DOT:
8572 case LYXP_TOKEN_DDOT:
Michal Vasko49fec8e2022-05-24 10:28:33 +02008573 case LYXP_TOKEN_AXISNAME:
Michal Vasko03ff5a72019-09-11 13:49:33 +02008574 case LYXP_TOKEN_AT:
8575 case LYXP_TOKEN_NAMETEST:
8576 case LYXP_TOKEN_NODETYPE:
8577 /* RelativeLocationPath */
Michal Vasko004d3152020-06-11 19:59:22 +02008578 rc = eval_relative_location_path(exp, tok_idx, 0, set, options);
Michal Vasko03ff5a72019-09-11 13:49:33 +02008579 LY_CHECK_RET(rc);
8580 break;
8581
aPiecekfba75362021-10-07 12:39:48 +02008582 case LYXP_TOKEN_VARREF:
8583 /* VariableReference */
8584 rc = eval_variable_reference(exp, tok_idx, set, options);
8585 LY_CHECK_RET(rc);
8586 ++(*tok_idx);
8587
aPiecekfba75362021-10-07 12:39:48 +02008588 goto predicate;
8589
Michal Vasko03ff5a72019-09-11 13:49:33 +02008590 case LYXP_TOKEN_FUNCNAME:
8591 /* FunctionCall */
aPiecek8b0cc152021-05-31 16:40:31 +02008592 rc = eval_function_call(exp, tok_idx, set, options);
Michal Vasko03ff5a72019-09-11 13:49:33 +02008593 LY_CHECK_RET(rc);
8594
Michal Vasko03ff5a72019-09-11 13:49:33 +02008595 goto predicate;
8596
Michal Vasko3e48bf32020-06-01 08:39:07 +02008597 case LYXP_TOKEN_OPER_PATH:
8598 case LYXP_TOKEN_OPER_RPATH:
Michal Vasko03ff5a72019-09-11 13:49:33 +02008599 /* AbsoluteLocationPath */
Michal Vasko004d3152020-06-11 19:59:22 +02008600 rc = eval_absolute_location_path(exp, tok_idx, set, options);
Michal Vasko03ff5a72019-09-11 13:49:33 +02008601 LY_CHECK_RET(rc);
8602 break;
8603
8604 case LYXP_TOKEN_LITERAL:
8605 /* Literal */
aPiecek8b0cc152021-05-31 16:40:31 +02008606 if ((options & LYXP_SKIP_EXPR) || (options & LYXP_SCNODE_ALL)) {
8607 if (!(options & LYXP_SKIP_EXPR)) {
Michal Vasko1a09b212021-05-06 13:00:10 +02008608 set_scnode_clear_ctx(set, LYXP_SET_SCNODE_ATOM_VAL);
Michal Vasko03ff5a72019-09-11 13:49:33 +02008609 }
Michal Vasko004d3152020-06-11 19:59:22 +02008610 eval_literal(exp, tok_idx, NULL);
Michal Vasko03ff5a72019-09-11 13:49:33 +02008611 } else {
Michal Vasko004d3152020-06-11 19:59:22 +02008612 eval_literal(exp, tok_idx, set);
Michal Vasko03ff5a72019-09-11 13:49:33 +02008613 }
8614
Michal Vasko03ff5a72019-09-11 13:49:33 +02008615 goto predicate;
8616
8617 case LYXP_TOKEN_NUMBER:
8618 /* Number */
aPiecek8b0cc152021-05-31 16:40:31 +02008619 if ((options & LYXP_SKIP_EXPR) || (options & LYXP_SCNODE_ALL)) {
8620 if (!(options & LYXP_SKIP_EXPR)) {
Michal Vasko1a09b212021-05-06 13:00:10 +02008621 set_scnode_clear_ctx(set, LYXP_SET_SCNODE_ATOM_VAL);
Michal Vasko03ff5a72019-09-11 13:49:33 +02008622 }
Michal Vasko004d3152020-06-11 19:59:22 +02008623 rc = eval_number(NULL, exp, tok_idx, NULL);
Michal Vasko03ff5a72019-09-11 13:49:33 +02008624 } else {
Michal Vasko004d3152020-06-11 19:59:22 +02008625 rc = eval_number(set->ctx, exp, tok_idx, set);
Michal Vasko03ff5a72019-09-11 13:49:33 +02008626 }
8627 LY_CHECK_RET(rc);
8628
Michal Vasko03ff5a72019-09-11 13:49:33 +02008629 goto predicate;
8630
8631 default:
Michal Vasko49fec8e2022-05-24 10:28:33 +02008632 LOGVAL(set->ctx, LY_VCODE_XP_INTOK, lyxp_token2str(exp->tokens[*tok_idx]), &exp->expr[exp->tok_pos[*tok_idx]]);
Michal Vasko03ff5a72019-09-11 13:49:33 +02008633 return LY_EVALID;
8634 }
8635
8636 return LY_SUCCESS;
8637
8638predicate:
8639 /* Predicate* */
Michal Vasko004d3152020-06-11 19:59:22 +02008640 while (!lyxp_check_token(NULL, exp, *tok_idx, LYXP_TOKEN_BRACK1)) {
Michal Vasko49fec8e2022-05-24 10:28:33 +02008641 rc = eval_predicate(exp, tok_idx, set, options, LYXP_AXIS_CHILD);
Michal Vasko03ff5a72019-09-11 13:49:33 +02008642 LY_CHECK_RET(rc);
8643 }
8644
8645 /* ('/' or '//') RelativeLocationPath */
Michal Vasko004d3152020-06-11 19:59:22 +02008646 if (!exp_check_token2(NULL, exp, *tok_idx, LYXP_TOKEN_OPER_PATH, LYXP_TOKEN_OPER_RPATH)) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02008647
8648 /* evaluate '/' or '//' */
Michal Vasko004d3152020-06-11 19:59:22 +02008649 if (exp->tokens[*tok_idx] == LYXP_TOKEN_OPER_PATH) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02008650 all_desc = 0;
8651 } else {
Michal Vasko03ff5a72019-09-11 13:49:33 +02008652 all_desc = 1;
8653 }
8654
aPiecek8b0cc152021-05-31 16:40:31 +02008655 LOGDBG(LY_LDGXPATH, "%-27s %s %s[%u]", __func__, (options & LYXP_SKIP_EXPR ? "skipped" : "parsed"),
Michal Vasko49fec8e2022-05-24 10:28:33 +02008656 lyxp_token2str(exp->tokens[*tok_idx]), exp->tok_pos[*tok_idx]);
Michal Vasko004d3152020-06-11 19:59:22 +02008657 ++(*tok_idx);
Michal Vasko03ff5a72019-09-11 13:49:33 +02008658
Michal Vasko004d3152020-06-11 19:59:22 +02008659 rc = eval_relative_location_path(exp, tok_idx, all_desc, set, options);
Michal Vasko03ff5a72019-09-11 13:49:33 +02008660 LY_CHECK_RET(rc);
8661 }
8662
8663 return LY_SUCCESS;
8664}
8665
8666/**
8667 * @brief Evaluate UnionExpr. Logs directly on error.
8668 *
Michal Vaskod3678892020-05-21 10:06:58 +02008669 * [20] UnionExpr ::= PathExpr | UnionExpr '|' PathExpr
Michal Vasko03ff5a72019-09-11 13:49:33 +02008670 *
8671 * @param[in] exp Parsed XPath expression.
Michal Vasko004d3152020-06-11 19:59:22 +02008672 * @param[in] tok_idx Position in the expression @p exp.
Michal Vasko03ff5a72019-09-11 13:49:33 +02008673 * @param[in] repeat How many times this expression is repeated.
aPiecek8b0cc152021-05-31 16:40:31 +02008674 * @param[in,out] set Context and result set.
Michal Vasko03ff5a72019-09-11 13:49:33 +02008675 * @param[in] options XPath options.
8676 * @return LY_ERR (LY_EINCOMPLETE on unresolved when)
8677 */
8678static LY_ERR
Michal Vasko40308e72020-10-20 16:38:40 +02008679eval_union_expr(const struct lyxp_expr *exp, uint16_t *tok_idx, uint16_t repeat, struct lyxp_set *set, uint32_t options)
Michal Vasko03ff5a72019-09-11 13:49:33 +02008680{
8681 LY_ERR rc = LY_SUCCESS;
8682 struct lyxp_set orig_set, set2;
8683 uint16_t i;
8684
8685 assert(repeat);
8686
8687 set_init(&orig_set, set);
8688 set_init(&set2, set);
8689
8690 set_fill_set(&orig_set, set);
8691
Michal Vasko004d3152020-06-11 19:59:22 +02008692 rc = eval_expr_select(exp, tok_idx, LYXP_EXPR_UNION, set, options);
Michal Vasko03ff5a72019-09-11 13:49:33 +02008693 LY_CHECK_GOTO(rc, cleanup);
8694
8695 /* ('|' PathExpr)* */
8696 for (i = 0; i < repeat; ++i) {
Michal Vasko004d3152020-06-11 19:59:22 +02008697 assert(exp->tokens[*tok_idx] == LYXP_TOKEN_OPER_UNI);
aPiecek8b0cc152021-05-31 16:40:31 +02008698 LOGDBG(LY_LDGXPATH, "%-27s %s %s[%u]", __func__, (options & LYXP_SKIP_EXPR ? "skipped" : "parsed"),
Michal Vasko49fec8e2022-05-24 10:28:33 +02008699 lyxp_token2str(exp->tokens[*tok_idx]), exp->tok_pos[*tok_idx]);
Michal Vasko004d3152020-06-11 19:59:22 +02008700 ++(*tok_idx);
Michal Vasko03ff5a72019-09-11 13:49:33 +02008701
aPiecek8b0cc152021-05-31 16:40:31 +02008702 if (options & LYXP_SKIP_EXPR) {
8703 rc = eval_expr_select(exp, tok_idx, LYXP_EXPR_UNION, set, options);
Michal Vasko03ff5a72019-09-11 13:49:33 +02008704 LY_CHECK_GOTO(rc, cleanup);
8705 continue;
8706 }
8707
8708 set_fill_set(&set2, &orig_set);
Michal Vasko004d3152020-06-11 19:59:22 +02008709 rc = eval_expr_select(exp, tok_idx, LYXP_EXPR_UNION, &set2, options);
Michal Vasko03ff5a72019-09-11 13:49:33 +02008710 LY_CHECK_GOTO(rc, cleanup);
8711
8712 /* eval */
8713 if (options & LYXP_SCNODE_ALL) {
Michal Vaskoecd62de2019-11-13 12:35:11 +01008714 lyxp_set_scnode_merge(set, &set2);
Michal Vasko03ff5a72019-09-11 13:49:33 +02008715 } else {
Michal Vasko5e0e6eb2019-11-06 15:47:50 +01008716 rc = moveto_union(set, &set2);
Michal Vasko03ff5a72019-09-11 13:49:33 +02008717 LY_CHECK_GOTO(rc, cleanup);
8718 }
8719 }
8720
8721cleanup:
Michal Vaskod3678892020-05-21 10:06:58 +02008722 lyxp_set_free_content(&orig_set);
8723 lyxp_set_free_content(&set2);
Michal Vasko03ff5a72019-09-11 13:49:33 +02008724 return rc;
8725}
8726
8727/**
8728 * @brief Evaluate UnaryExpr. Logs directly on error.
8729 *
Michal Vaskod3678892020-05-21 10:06:58 +02008730 * [19] UnaryExpr ::= UnionExpr | '-' UnaryExpr
Michal Vasko03ff5a72019-09-11 13:49:33 +02008731 *
8732 * @param[in] exp Parsed XPath expression.
Michal Vasko004d3152020-06-11 19:59:22 +02008733 * @param[in] tok_idx Position in the expression @p exp.
Michal Vasko03ff5a72019-09-11 13:49:33 +02008734 * @param[in] repeat How many times this expression is repeated.
aPiecek8b0cc152021-05-31 16:40:31 +02008735 * @param[in,out] set Context and result set.
Michal Vasko03ff5a72019-09-11 13:49:33 +02008736 * @param[in] options XPath options.
8737 * @return LY_ERR (LY_EINCOMPLETE on unresolved when)
8738 */
8739static LY_ERR
Michal Vasko40308e72020-10-20 16:38:40 +02008740eval_unary_expr(const struct lyxp_expr *exp, uint16_t *tok_idx, uint16_t repeat, struct lyxp_set *set, uint32_t options)
Michal Vasko03ff5a72019-09-11 13:49:33 +02008741{
8742 LY_ERR rc;
8743 uint16_t this_op, i;
8744
8745 assert(repeat);
8746
8747 /* ('-')+ */
Michal Vasko004d3152020-06-11 19:59:22 +02008748 this_op = *tok_idx;
Michal Vasko03ff5a72019-09-11 13:49:33 +02008749 for (i = 0; i < repeat; ++i) {
Michal Vasko004d3152020-06-11 19:59:22 +02008750 assert(!lyxp_check_token(NULL, exp, *tok_idx, LYXP_TOKEN_OPER_MATH) && (exp->expr[exp->tok_pos[*tok_idx]] == '-'));
Michal Vasko03ff5a72019-09-11 13:49:33 +02008751
aPiecek8b0cc152021-05-31 16:40:31 +02008752 LOGDBG(LY_LDGXPATH, "%-27s %s %s[%u]", __func__, (options & LYXP_SKIP_EXPR ? "skipped" : "parsed"),
Michal Vasko49fec8e2022-05-24 10:28:33 +02008753 lyxp_token2str(exp->tokens[*tok_idx]), exp->tok_pos[*tok_idx]);
Michal Vasko004d3152020-06-11 19:59:22 +02008754 ++(*tok_idx);
Michal Vasko03ff5a72019-09-11 13:49:33 +02008755 }
8756
Michal Vasko004d3152020-06-11 19:59:22 +02008757 rc = eval_expr_select(exp, tok_idx, LYXP_EXPR_UNARY, set, options);
Michal Vasko03ff5a72019-09-11 13:49:33 +02008758 LY_CHECK_RET(rc);
8759
aPiecek8b0cc152021-05-31 16:40:31 +02008760 if (!(options & LYXP_SKIP_EXPR) && (repeat % 2)) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02008761 if (options & LYXP_SCNODE_ALL) {
8762 warn_operands(set->ctx, set, NULL, 1, exp->expr, exp->tok_pos[this_op]);
8763 } else {
Michal Vasko5e0e6eb2019-11-06 15:47:50 +01008764 rc = moveto_op_math(set, NULL, &exp->expr[exp->tok_pos[this_op]]);
Michal Vasko03ff5a72019-09-11 13:49:33 +02008765 LY_CHECK_RET(rc);
8766 }
8767 }
8768
8769 return LY_SUCCESS;
8770}
8771
8772/**
8773 * @brief Evaluate MultiplicativeExpr. Logs directly on error.
8774 *
Michal Vaskod3678892020-05-21 10:06:58 +02008775 * [18] MultiplicativeExpr ::= UnaryExpr
Michal Vasko03ff5a72019-09-11 13:49:33 +02008776 * | MultiplicativeExpr '*' UnaryExpr
8777 * | MultiplicativeExpr 'div' UnaryExpr
8778 * | MultiplicativeExpr 'mod' UnaryExpr
8779 *
8780 * @param[in] exp Parsed XPath expression.
Michal Vasko004d3152020-06-11 19:59:22 +02008781 * @param[in] tok_idx Position in the expression @p exp.
Michal Vasko03ff5a72019-09-11 13:49:33 +02008782 * @param[in] repeat How many times this expression is repeated.
aPiecek8b0cc152021-05-31 16:40:31 +02008783 * @param[in,out] set Context and result set.
Michal Vasko03ff5a72019-09-11 13:49:33 +02008784 * @param[in] options XPath options.
8785 * @return LY_ERR (LY_EINCOMPLETE on unresolved when)
8786 */
8787static LY_ERR
Michal Vasko40308e72020-10-20 16:38:40 +02008788eval_multiplicative_expr(const struct lyxp_expr *exp, uint16_t *tok_idx, uint16_t repeat, struct lyxp_set *set,
8789 uint32_t options)
Michal Vasko03ff5a72019-09-11 13:49:33 +02008790{
8791 LY_ERR rc;
8792 uint16_t this_op;
8793 struct lyxp_set orig_set, set2;
8794 uint16_t i;
8795
8796 assert(repeat);
8797
8798 set_init(&orig_set, set);
8799 set_init(&set2, set);
8800
8801 set_fill_set(&orig_set, set);
8802
Michal Vasko004d3152020-06-11 19:59:22 +02008803 rc = eval_expr_select(exp, tok_idx, LYXP_EXPR_MULTIPLICATIVE, set, options);
Michal Vasko03ff5a72019-09-11 13:49:33 +02008804 LY_CHECK_GOTO(rc, cleanup);
8805
8806 /* ('*' / 'div' / 'mod' UnaryExpr)* */
8807 for (i = 0; i < repeat; ++i) {
Michal Vasko004d3152020-06-11 19:59:22 +02008808 this_op = *tok_idx;
Michal Vasko03ff5a72019-09-11 13:49:33 +02008809
Michal Vasko004d3152020-06-11 19:59:22 +02008810 assert(exp->tokens[*tok_idx] == LYXP_TOKEN_OPER_MATH);
aPiecek8b0cc152021-05-31 16:40:31 +02008811 LOGDBG(LY_LDGXPATH, "%-27s %s %s[%u]", __func__, (options & LYXP_SKIP_EXPR ? "skipped" : "parsed"),
Michal Vasko49fec8e2022-05-24 10:28:33 +02008812 lyxp_token2str(exp->tokens[*tok_idx]), exp->tok_pos[*tok_idx]);
Michal Vasko004d3152020-06-11 19:59:22 +02008813 ++(*tok_idx);
Michal Vasko03ff5a72019-09-11 13:49:33 +02008814
aPiecek8b0cc152021-05-31 16:40:31 +02008815 if (options & LYXP_SKIP_EXPR) {
8816 rc = eval_expr_select(exp, tok_idx, LYXP_EXPR_MULTIPLICATIVE, set, options);
Michal Vasko03ff5a72019-09-11 13:49:33 +02008817 LY_CHECK_GOTO(rc, cleanup);
8818 continue;
8819 }
8820
8821 set_fill_set(&set2, &orig_set);
Michal Vasko004d3152020-06-11 19:59:22 +02008822 rc = eval_expr_select(exp, tok_idx, LYXP_EXPR_MULTIPLICATIVE, &set2, options);
Michal Vasko03ff5a72019-09-11 13:49:33 +02008823 LY_CHECK_GOTO(rc, cleanup);
8824
8825 /* eval */
8826 if (options & LYXP_SCNODE_ALL) {
8827 warn_operands(set->ctx, set, &set2, 1, exp->expr, exp->tok_pos[this_op - 1]);
Michal Vaskoecd62de2019-11-13 12:35:11 +01008828 lyxp_set_scnode_merge(set, &set2);
Michal Vasko1a09b212021-05-06 13:00:10 +02008829 set_scnode_clear_ctx(set, LYXP_SET_SCNODE_ATOM_VAL);
Michal Vasko03ff5a72019-09-11 13:49:33 +02008830 } else {
Michal Vasko5e0e6eb2019-11-06 15:47:50 +01008831 rc = moveto_op_math(set, &set2, &exp->expr[exp->tok_pos[this_op]]);
Michal Vasko03ff5a72019-09-11 13:49:33 +02008832 LY_CHECK_GOTO(rc, cleanup);
8833 }
8834 }
8835
8836cleanup:
Michal Vaskod3678892020-05-21 10:06:58 +02008837 lyxp_set_free_content(&orig_set);
8838 lyxp_set_free_content(&set2);
Michal Vasko03ff5a72019-09-11 13:49:33 +02008839 return rc;
8840}
8841
8842/**
8843 * @brief Evaluate AdditiveExpr. Logs directly on error.
8844 *
Michal Vaskod3678892020-05-21 10:06:58 +02008845 * [17] AdditiveExpr ::= MultiplicativeExpr
Michal Vasko03ff5a72019-09-11 13:49:33 +02008846 * | AdditiveExpr '+' MultiplicativeExpr
8847 * | AdditiveExpr '-' MultiplicativeExpr
8848 *
8849 * @param[in] exp Parsed XPath expression.
Michal Vasko004d3152020-06-11 19:59:22 +02008850 * @param[in] tok_idx Position in the expression @p exp.
Michal Vasko03ff5a72019-09-11 13:49:33 +02008851 * @param[in] repeat How many times this expression is repeated.
aPiecek8b0cc152021-05-31 16:40:31 +02008852 * @param[in,out] set Context and result set.
Michal Vasko03ff5a72019-09-11 13:49:33 +02008853 * @param[in] options XPath options.
8854 * @return LY_ERR (LY_EINCOMPLETE on unresolved when)
8855 */
8856static LY_ERR
Michal Vasko40308e72020-10-20 16:38:40 +02008857eval_additive_expr(const struct lyxp_expr *exp, uint16_t *tok_idx, uint16_t repeat, struct lyxp_set *set, uint32_t options)
Michal Vasko03ff5a72019-09-11 13:49:33 +02008858{
8859 LY_ERR rc;
8860 uint16_t this_op;
8861 struct lyxp_set orig_set, set2;
8862 uint16_t i;
8863
8864 assert(repeat);
8865
8866 set_init(&orig_set, set);
8867 set_init(&set2, set);
8868
8869 set_fill_set(&orig_set, set);
8870
Michal Vasko004d3152020-06-11 19:59:22 +02008871 rc = eval_expr_select(exp, tok_idx, LYXP_EXPR_ADDITIVE, set, options);
Michal Vasko03ff5a72019-09-11 13:49:33 +02008872 LY_CHECK_GOTO(rc, cleanup);
8873
8874 /* ('+' / '-' MultiplicativeExpr)* */
8875 for (i = 0; i < repeat; ++i) {
Michal Vasko004d3152020-06-11 19:59:22 +02008876 this_op = *tok_idx;
Michal Vasko03ff5a72019-09-11 13:49:33 +02008877
Michal Vasko004d3152020-06-11 19:59:22 +02008878 assert(exp->tokens[*tok_idx] == LYXP_TOKEN_OPER_MATH);
Michal Vasko03ff5a72019-09-11 13:49:33 +02008879 LOGDBG(LY_LDGXPATH, "%-27s %s %s[%u]", __func__, (set ? "parsed" : "skipped"),
Michal Vasko49fec8e2022-05-24 10:28:33 +02008880 lyxp_token2str(exp->tokens[*tok_idx]), exp->tok_pos[*tok_idx]);
Michal Vasko004d3152020-06-11 19:59:22 +02008881 ++(*tok_idx);
Michal Vasko03ff5a72019-09-11 13:49:33 +02008882
aPiecek8b0cc152021-05-31 16:40:31 +02008883 if (options & LYXP_SKIP_EXPR) {
8884 rc = eval_expr_select(exp, tok_idx, LYXP_EXPR_ADDITIVE, set, options);
Michal Vasko03ff5a72019-09-11 13:49:33 +02008885 LY_CHECK_GOTO(rc, cleanup);
8886 continue;
8887 }
8888
8889 set_fill_set(&set2, &orig_set);
Michal Vasko004d3152020-06-11 19:59:22 +02008890 rc = eval_expr_select(exp, tok_idx, LYXP_EXPR_ADDITIVE, &set2, options);
Michal Vasko03ff5a72019-09-11 13:49:33 +02008891 LY_CHECK_GOTO(rc, cleanup);
8892
8893 /* eval */
8894 if (options & LYXP_SCNODE_ALL) {
8895 warn_operands(set->ctx, set, &set2, 1, exp->expr, exp->tok_pos[this_op - 1]);
Michal Vaskoecd62de2019-11-13 12:35:11 +01008896 lyxp_set_scnode_merge(set, &set2);
Michal Vasko1a09b212021-05-06 13:00:10 +02008897 set_scnode_clear_ctx(set, LYXP_SET_SCNODE_ATOM_VAL);
Michal Vasko03ff5a72019-09-11 13:49:33 +02008898 } else {
Michal Vasko5e0e6eb2019-11-06 15:47:50 +01008899 rc = moveto_op_math(set, &set2, &exp->expr[exp->tok_pos[this_op]]);
Michal Vasko03ff5a72019-09-11 13:49:33 +02008900 LY_CHECK_GOTO(rc, cleanup);
8901 }
8902 }
8903
8904cleanup:
Michal Vaskod3678892020-05-21 10:06:58 +02008905 lyxp_set_free_content(&orig_set);
8906 lyxp_set_free_content(&set2);
Michal Vasko03ff5a72019-09-11 13:49:33 +02008907 return rc;
8908}
8909
8910/**
8911 * @brief Evaluate RelationalExpr. Logs directly on error.
8912 *
Michal Vaskod3678892020-05-21 10:06:58 +02008913 * [16] RelationalExpr ::= AdditiveExpr
Michal Vasko03ff5a72019-09-11 13:49:33 +02008914 * | RelationalExpr '<' AdditiveExpr
8915 * | RelationalExpr '>' AdditiveExpr
8916 * | RelationalExpr '<=' AdditiveExpr
8917 * | RelationalExpr '>=' AdditiveExpr
8918 *
8919 * @param[in] exp Parsed XPath expression.
Michal Vasko004d3152020-06-11 19:59:22 +02008920 * @param[in] tok_idx Position in the expression @p exp.
Michal Vasko03ff5a72019-09-11 13:49:33 +02008921 * @param[in] repeat How many times this expression is repeated.
aPiecek8b0cc152021-05-31 16:40:31 +02008922 * @param[in,out] set Context and result set.
Michal Vasko03ff5a72019-09-11 13:49:33 +02008923 * @param[in] options XPath options.
8924 * @return LY_ERR (LY_EINCOMPLETE on unresolved when)
8925 */
8926static LY_ERR
Michal Vasko40308e72020-10-20 16:38:40 +02008927eval_relational_expr(const struct lyxp_expr *exp, uint16_t *tok_idx, uint16_t repeat, struct lyxp_set *set, uint32_t options)
Michal Vasko03ff5a72019-09-11 13:49:33 +02008928{
8929 LY_ERR rc;
8930 uint16_t this_op;
8931 struct lyxp_set orig_set, set2;
8932 uint16_t i;
8933
8934 assert(repeat);
8935
8936 set_init(&orig_set, set);
8937 set_init(&set2, set);
8938
8939 set_fill_set(&orig_set, set);
8940
Michal Vasko004d3152020-06-11 19:59:22 +02008941 rc = eval_expr_select(exp, tok_idx, LYXP_EXPR_RELATIONAL, set, options);
Michal Vasko03ff5a72019-09-11 13:49:33 +02008942 LY_CHECK_GOTO(rc, cleanup);
8943
8944 /* ('<' / '>' / '<=' / '>=' AdditiveExpr)* */
8945 for (i = 0; i < repeat; ++i) {
Michal Vasko004d3152020-06-11 19:59:22 +02008946 this_op = *tok_idx;
Michal Vasko03ff5a72019-09-11 13:49:33 +02008947
Michal Vasko004d3152020-06-11 19:59:22 +02008948 assert(exp->tokens[*tok_idx] == LYXP_TOKEN_OPER_COMP);
aPiecek8b0cc152021-05-31 16:40:31 +02008949 LOGDBG(LY_LDGXPATH, "%-27s %s %s[%u]", __func__, (options & LYXP_SKIP_EXPR ? "skipped" : "parsed"),
Michal Vasko49fec8e2022-05-24 10:28:33 +02008950 lyxp_token2str(exp->tokens[*tok_idx]), exp->tok_pos[*tok_idx]);
Michal Vasko004d3152020-06-11 19:59:22 +02008951 ++(*tok_idx);
Michal Vasko03ff5a72019-09-11 13:49:33 +02008952
aPiecek8b0cc152021-05-31 16:40:31 +02008953 if (options & LYXP_SKIP_EXPR) {
8954 rc = eval_expr_select(exp, tok_idx, LYXP_EXPR_RELATIONAL, set, options);
Michal Vasko03ff5a72019-09-11 13:49:33 +02008955 LY_CHECK_GOTO(rc, cleanup);
8956 continue;
8957 }
8958
8959 set_fill_set(&set2, &orig_set);
Michal Vasko004d3152020-06-11 19:59:22 +02008960 rc = eval_expr_select(exp, tok_idx, LYXP_EXPR_RELATIONAL, &set2, options);
Michal Vasko03ff5a72019-09-11 13:49:33 +02008961 LY_CHECK_GOTO(rc, cleanup);
8962
8963 /* eval */
8964 if (options & LYXP_SCNODE_ALL) {
8965 warn_operands(set->ctx, set, &set2, 1, exp->expr, exp->tok_pos[this_op - 1]);
Michal Vaskoecd62de2019-11-13 12:35:11 +01008966 lyxp_set_scnode_merge(set, &set2);
Michal Vasko1a09b212021-05-06 13:00:10 +02008967 set_scnode_clear_ctx(set, LYXP_SET_SCNODE_ATOM_VAL);
Michal Vasko03ff5a72019-09-11 13:49:33 +02008968 } else {
8969 rc = moveto_op_comp(set, &set2, &exp->expr[exp->tok_pos[this_op]], options);
8970 LY_CHECK_GOTO(rc, cleanup);
8971 }
8972 }
8973
8974cleanup:
Michal Vaskod3678892020-05-21 10:06:58 +02008975 lyxp_set_free_content(&orig_set);
8976 lyxp_set_free_content(&set2);
Michal Vasko03ff5a72019-09-11 13:49:33 +02008977 return rc;
8978}
8979
8980/**
8981 * @brief Evaluate EqualityExpr. Logs directly on error.
8982 *
Michal Vaskod3678892020-05-21 10:06:58 +02008983 * [15] EqualityExpr ::= RelationalExpr | EqualityExpr '=' RelationalExpr
Michal Vasko03ff5a72019-09-11 13:49:33 +02008984 * | EqualityExpr '!=' RelationalExpr
8985 *
8986 * @param[in] exp Parsed XPath expression.
Michal Vasko004d3152020-06-11 19:59:22 +02008987 * @param[in] tok_idx Position in the expression @p exp.
Michal Vasko03ff5a72019-09-11 13:49:33 +02008988 * @param[in] repeat How many times this expression is repeated.
aPiecek8b0cc152021-05-31 16:40:31 +02008989 * @param[in,out] set Context and result set.
Michal Vasko03ff5a72019-09-11 13:49:33 +02008990 * @param[in] options XPath options.
8991 * @return LY_ERR (LY_EINCOMPLETE on unresolved when)
8992 */
8993static LY_ERR
Michal Vasko40308e72020-10-20 16:38:40 +02008994eval_equality_expr(const struct lyxp_expr *exp, uint16_t *tok_idx, uint16_t repeat, struct lyxp_set *set, uint32_t options)
Michal Vasko03ff5a72019-09-11 13:49:33 +02008995{
8996 LY_ERR rc;
8997 uint16_t this_op;
8998 struct lyxp_set orig_set, set2;
8999 uint16_t i;
9000
9001 assert(repeat);
9002
9003 set_init(&orig_set, set);
9004 set_init(&set2, set);
9005
9006 set_fill_set(&orig_set, set);
9007
Michal Vasko004d3152020-06-11 19:59:22 +02009008 rc = eval_expr_select(exp, tok_idx, LYXP_EXPR_EQUALITY, set, options);
Michal Vasko03ff5a72019-09-11 13:49:33 +02009009 LY_CHECK_GOTO(rc, cleanup);
9010
9011 /* ('=' / '!=' RelationalExpr)* */
9012 for (i = 0; i < repeat; ++i) {
Michal Vasko004d3152020-06-11 19:59:22 +02009013 this_op = *tok_idx;
Michal Vasko03ff5a72019-09-11 13:49:33 +02009014
Michal Vasko004d3152020-06-11 19:59:22 +02009015 assert((exp->tokens[*tok_idx] == LYXP_TOKEN_OPER_EQUAL) || (exp->tokens[*tok_idx] == LYXP_TOKEN_OPER_NEQUAL));
aPiecek8b0cc152021-05-31 16:40:31 +02009016 LOGDBG(LY_LDGXPATH, "%-27s %s %s[%u]", __func__, (options & LYXP_SKIP_EXPR ? "skipped" : "parsed"),
Michal Vasko49fec8e2022-05-24 10:28:33 +02009017 lyxp_token2str(exp->tokens[*tok_idx]), exp->tok_pos[*tok_idx]);
Michal Vasko004d3152020-06-11 19:59:22 +02009018 ++(*tok_idx);
Michal Vasko03ff5a72019-09-11 13:49:33 +02009019
aPiecek8b0cc152021-05-31 16:40:31 +02009020 if (options & LYXP_SKIP_EXPR) {
9021 rc = eval_expr_select(exp, tok_idx, LYXP_EXPR_EQUALITY, set, options);
Michal Vasko03ff5a72019-09-11 13:49:33 +02009022 LY_CHECK_GOTO(rc, cleanup);
9023 continue;
9024 }
9025
9026 set_fill_set(&set2, &orig_set);
Michal Vasko004d3152020-06-11 19:59:22 +02009027 rc = eval_expr_select(exp, tok_idx, LYXP_EXPR_EQUALITY, &set2, options);
Michal Vasko03ff5a72019-09-11 13:49:33 +02009028 LY_CHECK_GOTO(rc, cleanup);
9029
9030 /* eval */
9031 if (options & LYXP_SCNODE_ALL) {
9032 warn_operands(set->ctx, set, &set2, 0, exp->expr, exp->tok_pos[this_op - 1]);
Michal Vasko004d3152020-06-11 19:59:22 +02009033 warn_equality_value(exp, set, *tok_idx - 1, this_op - 1, *tok_idx - 1);
9034 warn_equality_value(exp, &set2, this_op - 1, this_op - 1, *tok_idx - 1);
Michal Vaskoecd62de2019-11-13 12:35:11 +01009035 lyxp_set_scnode_merge(set, &set2);
Michal Vasko1a09b212021-05-06 13:00:10 +02009036 set_scnode_clear_ctx(set, LYXP_SET_SCNODE_ATOM_VAL);
Michal Vasko03ff5a72019-09-11 13:49:33 +02009037 } else {
Michal Vasko03ff5a72019-09-11 13:49:33 +02009038 rc = moveto_op_comp(set, &set2, &exp->expr[exp->tok_pos[this_op]], options);
9039 LY_CHECK_GOTO(rc, cleanup);
9040 }
9041 }
9042
9043cleanup:
Michal Vaskod3678892020-05-21 10:06:58 +02009044 lyxp_set_free_content(&orig_set);
9045 lyxp_set_free_content(&set2);
Michal Vasko03ff5a72019-09-11 13:49:33 +02009046 return rc;
9047}
9048
9049/**
9050 * @brief Evaluate AndExpr. Logs directly on error.
9051 *
Michal Vaskod3678892020-05-21 10:06:58 +02009052 * [14] AndExpr ::= EqualityExpr | AndExpr 'and' EqualityExpr
Michal Vasko03ff5a72019-09-11 13:49:33 +02009053 *
9054 * @param[in] exp Parsed XPath expression.
Michal Vasko004d3152020-06-11 19:59:22 +02009055 * @param[in] tok_idx Position in the expression @p exp.
Michal Vasko03ff5a72019-09-11 13:49:33 +02009056 * @param[in] repeat How many times this expression is repeated.
aPiecek8b0cc152021-05-31 16:40:31 +02009057 * @param[in,out] set Context and result set.
Michal Vasko03ff5a72019-09-11 13:49:33 +02009058 * @param[in] options XPath options.
9059 * @return LY_ERR (LY_EINCOMPLETE on unresolved when)
9060 */
9061static LY_ERR
Michal Vasko40308e72020-10-20 16:38:40 +02009062eval_and_expr(const struct lyxp_expr *exp, uint16_t *tok_idx, uint16_t repeat, struct lyxp_set *set, uint32_t options)
Michal Vasko03ff5a72019-09-11 13:49:33 +02009063{
9064 LY_ERR rc;
9065 struct lyxp_set orig_set, set2;
9066 uint16_t i;
9067
9068 assert(repeat);
9069
9070 set_init(&orig_set, set);
9071 set_init(&set2, set);
9072
9073 set_fill_set(&orig_set, set);
9074
Michal Vasko004d3152020-06-11 19:59:22 +02009075 rc = eval_expr_select(exp, tok_idx, LYXP_EXPR_AND, set, options);
Michal Vasko03ff5a72019-09-11 13:49:33 +02009076 LY_CHECK_GOTO(rc, cleanup);
9077
9078 /* cast to boolean, we know that will be the final result */
aPiecek8b0cc152021-05-31 16:40:31 +02009079 if (!(options & LYXP_SKIP_EXPR) && (options & LYXP_SCNODE_ALL)) {
Michal Vasko1a09b212021-05-06 13:00:10 +02009080 set_scnode_clear_ctx(set, LYXP_SET_SCNODE_ATOM_NODE);
Michal Vasko03ff5a72019-09-11 13:49:33 +02009081 } else {
Michal Vasko5e0e6eb2019-11-06 15:47:50 +01009082 lyxp_set_cast(set, LYXP_SET_BOOLEAN);
Michal Vasko03ff5a72019-09-11 13:49:33 +02009083 }
9084
9085 /* ('and' EqualityExpr)* */
9086 for (i = 0; i < repeat; ++i) {
Michal Vasko004d3152020-06-11 19:59:22 +02009087 assert(exp->tokens[*tok_idx] == LYXP_TOKEN_OPER_LOG);
aPiecek8b0cc152021-05-31 16:40:31 +02009088 LOGDBG(LY_LDGXPATH, "%-27s %s %s[%u]", __func__, ((options & LYXP_SKIP_EXPR) || !set->val.bln ? "skipped" : "parsed"),
Michal Vasko49fec8e2022-05-24 10:28:33 +02009089 lyxp_token2str(exp->tokens[*tok_idx]), exp->tok_pos[*tok_idx]);
Michal Vasko004d3152020-06-11 19:59:22 +02009090 ++(*tok_idx);
Michal Vasko03ff5a72019-09-11 13:49:33 +02009091
9092 /* lazy evaluation */
aPiecek8b0cc152021-05-31 16:40:31 +02009093 if ((options & LYXP_SKIP_EXPR) || ((set->type == LYXP_SET_BOOLEAN) && !set->val.bln)) {
9094 rc = eval_expr_select(exp, tok_idx, LYXP_EXPR_AND, set, options | LYXP_SKIP_EXPR);
Michal Vasko03ff5a72019-09-11 13:49:33 +02009095 LY_CHECK_GOTO(rc, cleanup);
9096 continue;
9097 }
9098
9099 set_fill_set(&set2, &orig_set);
Michal Vasko004d3152020-06-11 19:59:22 +02009100 rc = eval_expr_select(exp, tok_idx, LYXP_EXPR_AND, &set2, options);
Michal Vasko03ff5a72019-09-11 13:49:33 +02009101 LY_CHECK_GOTO(rc, cleanup);
9102
9103 /* eval - just get boolean value actually */
9104 if (set->type == LYXP_SET_SCNODE_SET) {
Michal Vasko1a09b212021-05-06 13:00:10 +02009105 set_scnode_clear_ctx(&set2, LYXP_SET_SCNODE_ATOM_NODE);
Michal Vaskoecd62de2019-11-13 12:35:11 +01009106 lyxp_set_scnode_merge(set, &set2);
Michal Vasko03ff5a72019-09-11 13:49:33 +02009107 } else {
Michal Vasko5e0e6eb2019-11-06 15:47:50 +01009108 lyxp_set_cast(&set2, LYXP_SET_BOOLEAN);
Michal Vasko03ff5a72019-09-11 13:49:33 +02009109 set_fill_set(set, &set2);
9110 }
9111 }
9112
9113cleanup:
Michal Vaskod3678892020-05-21 10:06:58 +02009114 lyxp_set_free_content(&orig_set);
9115 lyxp_set_free_content(&set2);
Michal Vasko03ff5a72019-09-11 13:49:33 +02009116 return rc;
9117}
9118
9119/**
9120 * @brief Evaluate OrExpr. Logs directly on error.
9121 *
Michal Vaskod3678892020-05-21 10:06:58 +02009122 * [13] OrExpr ::= AndExpr | OrExpr 'or' AndExpr
Michal Vasko03ff5a72019-09-11 13:49:33 +02009123 *
9124 * @param[in] exp Parsed XPath expression.
Michal Vasko004d3152020-06-11 19:59:22 +02009125 * @param[in] tok_idx Position in the expression @p exp.
Michal Vasko03ff5a72019-09-11 13:49:33 +02009126 * @param[in] repeat How many times this expression is repeated.
aPiecek8b0cc152021-05-31 16:40:31 +02009127 * @param[in,out] set Context and result set.
Michal Vasko03ff5a72019-09-11 13:49:33 +02009128 * @param[in] options XPath options.
9129 * @return LY_ERR (LY_EINCOMPLETE on unresolved when)
9130 */
9131static LY_ERR
Michal Vasko40308e72020-10-20 16:38:40 +02009132eval_or_expr(const struct lyxp_expr *exp, uint16_t *tok_idx, uint16_t repeat, struct lyxp_set *set, uint32_t options)
Michal Vasko03ff5a72019-09-11 13:49:33 +02009133{
9134 LY_ERR rc;
9135 struct lyxp_set orig_set, set2;
9136 uint16_t i;
9137
9138 assert(repeat);
9139
9140 set_init(&orig_set, set);
9141 set_init(&set2, set);
9142
9143 set_fill_set(&orig_set, set);
9144
Michal Vasko004d3152020-06-11 19:59:22 +02009145 rc = eval_expr_select(exp, tok_idx, LYXP_EXPR_OR, set, options);
Michal Vasko03ff5a72019-09-11 13:49:33 +02009146 LY_CHECK_GOTO(rc, cleanup);
9147
9148 /* cast to boolean, we know that will be the final result */
aPiecek8b0cc152021-05-31 16:40:31 +02009149 if (!(options & LYXP_SKIP_EXPR) && (options & LYXP_SCNODE_ALL)) {
Michal Vasko1a09b212021-05-06 13:00:10 +02009150 set_scnode_clear_ctx(set, LYXP_SET_SCNODE_ATOM_NODE);
Michal Vasko03ff5a72019-09-11 13:49:33 +02009151 } else {
Michal Vasko5e0e6eb2019-11-06 15:47:50 +01009152 lyxp_set_cast(set, LYXP_SET_BOOLEAN);
Michal Vasko03ff5a72019-09-11 13:49:33 +02009153 }
9154
9155 /* ('or' AndExpr)* */
9156 for (i = 0; i < repeat; ++i) {
Michal Vasko004d3152020-06-11 19:59:22 +02009157 assert(exp->tokens[*tok_idx] == LYXP_TOKEN_OPER_LOG);
aPiecek8b0cc152021-05-31 16:40:31 +02009158 LOGDBG(LY_LDGXPATH, "%-27s %s %s[%u]", __func__, ((options & LYXP_SKIP_EXPR) || set->val.bln ? "skipped" : "parsed"),
Michal Vasko49fec8e2022-05-24 10:28:33 +02009159 lyxp_token2str(exp->tokens[*tok_idx]), exp->tok_pos[*tok_idx]);
Michal Vasko004d3152020-06-11 19:59:22 +02009160 ++(*tok_idx);
Michal Vasko03ff5a72019-09-11 13:49:33 +02009161
9162 /* lazy evaluation */
aPiecek8b0cc152021-05-31 16:40:31 +02009163 if ((options & LYXP_SKIP_EXPR) || ((set->type == LYXP_SET_BOOLEAN) && set->val.bln)) {
9164 rc = eval_expr_select(exp, tok_idx, LYXP_EXPR_OR, set, options | LYXP_SKIP_EXPR);
Michal Vasko03ff5a72019-09-11 13:49:33 +02009165 LY_CHECK_GOTO(rc, cleanup);
9166 continue;
9167 }
9168
9169 set_fill_set(&set2, &orig_set);
9170 /* expr_type cound have been LYXP_EXPR_NONE in all these later calls (except for the first one),
9171 * but it does not matter */
Michal Vasko004d3152020-06-11 19:59:22 +02009172 rc = eval_expr_select(exp, tok_idx, LYXP_EXPR_OR, &set2, options);
Michal Vasko03ff5a72019-09-11 13:49:33 +02009173 LY_CHECK_GOTO(rc, cleanup);
9174
9175 /* eval - just get boolean value actually */
9176 if (set->type == LYXP_SET_SCNODE_SET) {
Michal Vasko1a09b212021-05-06 13:00:10 +02009177 set_scnode_clear_ctx(&set2, LYXP_SET_SCNODE_ATOM_NODE);
Michal Vaskoecd62de2019-11-13 12:35:11 +01009178 lyxp_set_scnode_merge(set, &set2);
Michal Vasko03ff5a72019-09-11 13:49:33 +02009179 } else {
Michal Vasko5e0e6eb2019-11-06 15:47:50 +01009180 lyxp_set_cast(&set2, LYXP_SET_BOOLEAN);
Michal Vasko03ff5a72019-09-11 13:49:33 +02009181 set_fill_set(set, &set2);
9182 }
9183 }
9184
9185cleanup:
Michal Vaskod3678892020-05-21 10:06:58 +02009186 lyxp_set_free_content(&orig_set);
9187 lyxp_set_free_content(&set2);
Michal Vasko03ff5a72019-09-11 13:49:33 +02009188 return rc;
9189}
9190
9191/**
Michal Vasko004d3152020-06-11 19:59:22 +02009192 * @brief Decide what expression is at the pointer @p tok_idx and evaluate it accordingly.
Michal Vasko03ff5a72019-09-11 13:49:33 +02009193 *
9194 * @param[in] exp Parsed XPath expression.
Michal Vasko004d3152020-06-11 19:59:22 +02009195 * @param[in] tok_idx Position in the expression @p exp.
Michal Vasko03ff5a72019-09-11 13:49:33 +02009196 * @param[in] etype Expression type to evaluate.
aPiecek8b0cc152021-05-31 16:40:31 +02009197 * @param[in,out] set Context and result set.
Michal Vasko03ff5a72019-09-11 13:49:33 +02009198 * @param[in] options XPath options.
9199 * @return LY_ERR (LY_EINCOMPLETE on unresolved when)
9200 */
9201static LY_ERR
Michal Vasko40308e72020-10-20 16:38:40 +02009202eval_expr_select(const struct lyxp_expr *exp, uint16_t *tok_idx, enum lyxp_expr_type etype, struct lyxp_set *set,
9203 uint32_t options)
Michal Vasko03ff5a72019-09-11 13:49:33 +02009204{
9205 uint16_t i, count;
9206 enum lyxp_expr_type next_etype;
9207 LY_ERR rc;
9208
9209 /* process operator repeats */
Michal Vasko004d3152020-06-11 19:59:22 +02009210 if (!exp->repeat[*tok_idx]) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02009211 next_etype = LYXP_EXPR_NONE;
9212 } else {
9213 /* find etype repeat */
Radek Krejci1e008d22020-08-17 11:37:37 +02009214 for (i = 0; exp->repeat[*tok_idx][i] > etype; ++i) {}
Michal Vasko03ff5a72019-09-11 13:49:33 +02009215
9216 /* select one-priority lower because etype expression called us */
9217 if (i) {
Michal Vasko004d3152020-06-11 19:59:22 +02009218 next_etype = exp->repeat[*tok_idx][i - 1];
Michal Vasko03ff5a72019-09-11 13:49:33 +02009219 /* count repeats for that expression */
Radek Krejci1e008d22020-08-17 11:37:37 +02009220 for (count = 0; i && exp->repeat[*tok_idx][i - 1] == next_etype; ++count, --i) {}
Michal Vasko03ff5a72019-09-11 13:49:33 +02009221 } else {
9222 next_etype = LYXP_EXPR_NONE;
9223 }
9224 }
9225
9226 /* decide what expression are we parsing based on the repeat */
9227 switch (next_etype) {
9228 case LYXP_EXPR_OR:
Michal Vasko004d3152020-06-11 19:59:22 +02009229 rc = eval_or_expr(exp, tok_idx, count, set, options);
Michal Vasko03ff5a72019-09-11 13:49:33 +02009230 break;
9231 case LYXP_EXPR_AND:
Michal Vasko004d3152020-06-11 19:59:22 +02009232 rc = eval_and_expr(exp, tok_idx, count, set, options);
Michal Vasko03ff5a72019-09-11 13:49:33 +02009233 break;
9234 case LYXP_EXPR_EQUALITY:
Michal Vasko004d3152020-06-11 19:59:22 +02009235 rc = eval_equality_expr(exp, tok_idx, count, set, options);
Michal Vasko03ff5a72019-09-11 13:49:33 +02009236 break;
9237 case LYXP_EXPR_RELATIONAL:
Michal Vasko004d3152020-06-11 19:59:22 +02009238 rc = eval_relational_expr(exp, tok_idx, count, set, options);
Michal Vasko03ff5a72019-09-11 13:49:33 +02009239 break;
9240 case LYXP_EXPR_ADDITIVE:
Michal Vasko004d3152020-06-11 19:59:22 +02009241 rc = eval_additive_expr(exp, tok_idx, count, set, options);
Michal Vasko03ff5a72019-09-11 13:49:33 +02009242 break;
9243 case LYXP_EXPR_MULTIPLICATIVE:
Michal Vasko004d3152020-06-11 19:59:22 +02009244 rc = eval_multiplicative_expr(exp, tok_idx, count, set, options);
Michal Vasko03ff5a72019-09-11 13:49:33 +02009245 break;
9246 case LYXP_EXPR_UNARY:
Michal Vasko004d3152020-06-11 19:59:22 +02009247 rc = eval_unary_expr(exp, tok_idx, count, set, options);
Michal Vasko03ff5a72019-09-11 13:49:33 +02009248 break;
9249 case LYXP_EXPR_UNION:
Michal Vasko004d3152020-06-11 19:59:22 +02009250 rc = eval_union_expr(exp, tok_idx, count, set, options);
Michal Vasko03ff5a72019-09-11 13:49:33 +02009251 break;
9252 case LYXP_EXPR_NONE:
Michal Vasko004d3152020-06-11 19:59:22 +02009253 rc = eval_path_expr(exp, tok_idx, set, options);
Michal Vasko03ff5a72019-09-11 13:49:33 +02009254 break;
9255 default:
9256 LOGINT_RET(set->ctx);
9257 }
9258
9259 return rc;
9260}
9261
Michal Vasko5e0e6eb2019-11-06 15:47:50 +01009262/**
9263 * @brief Get root type.
9264 *
9265 * @param[in] ctx_node Context node.
9266 * @param[in] ctx_scnode Schema context node.
9267 * @param[in] options XPath options.
9268 * @return Root type.
9269 */
9270static enum lyxp_node_type
Radek Krejci1deb5be2020-08-26 16:43:36 +02009271lyxp_get_root_type(const struct lyd_node *ctx_node, const struct lysc_node *ctx_scnode, uint32_t options)
Michal Vasko5e0e6eb2019-11-06 15:47:50 +01009272{
Michal Vasko6b26e742020-07-17 15:02:10 +02009273 const struct lysc_node *op;
9274
Michal Vaskoa27245c2022-05-02 09:01:35 +02009275 /* explicit */
9276 if (options & LYXP_ACCESS_TREE_ALL) {
9277 return LYXP_NODE_ROOT;
9278 } else if (options & LYXP_ACCESS_TREE_CONFIG) {
9279 return LYXP_NODE_ROOT_CONFIG;
9280 }
9281
Michal Vasko5e0e6eb2019-11-06 15:47:50 +01009282 if (options & LYXP_SCNODE_ALL) {
Michal Vasko5d24f6c2020-10-13 13:49:06 +02009283 /* schema */
Radek Krejci1e008d22020-08-17 11:37:37 +02009284 for (op = ctx_scnode; op && !(op->nodetype & (LYS_RPC | LYS_ACTION | LYS_NOTIF)); op = op->parent) {}
Michal Vasko6b26e742020-07-17 15:02:10 +02009285
9286 if (op || (options & LYXP_SCNODE)) {
Michal Vasko5e0e6eb2019-11-06 15:47:50 +01009287 /* general root that can access everything */
9288 return LYXP_NODE_ROOT;
9289 } else if (!ctx_scnode || (ctx_scnode->flags & LYS_CONFIG_W)) {
9290 /* root context node can access only config data (because we said so, it is unspecified) */
9291 return LYXP_NODE_ROOT_CONFIG;
Michal Vasko5e0e6eb2019-11-06 15:47:50 +01009292 }
Michal Vasko6b26e742020-07-17 15:02:10 +02009293 return LYXP_NODE_ROOT;
Michal Vasko5e0e6eb2019-11-06 15:47:50 +01009294 }
9295
Michal Vasko5d24f6c2020-10-13 13:49:06 +02009296 /* data */
Michal Vasko6b26e742020-07-17 15:02:10 +02009297 op = ctx_node ? ctx_node->schema : NULL;
Michal Vaskod989ba02020-08-24 10:59:24 +02009298 for ( ; op && !(op->nodetype & (LYS_RPC | LYS_ACTION | LYS_NOTIF)); op = op->parent) {}
Michal Vasko6b26e742020-07-17 15:02:10 +02009299
Michal Vasko5d24f6c2020-10-13 13:49:06 +02009300 if (op || !(options & LYXP_SCHEMA)) {
9301 /* general root that can access everything */
9302 return LYXP_NODE_ROOT;
Christian Hoppsb6ecaea2021-02-06 09:45:38 -05009303 } else if (!ctx_node || !ctx_node->schema || (ctx_node->schema->flags & LYS_CONFIG_W)) {
Michal Vasko5e0e6eb2019-11-06 15:47:50 +01009304 /* root context node can access only config data (because we said so, it is unspecified) */
9305 return LYXP_NODE_ROOT_CONFIG;
9306 }
Michal Vasko5e0e6eb2019-11-06 15:47:50 +01009307 return LYXP_NODE_ROOT;
9308}
9309
Michal Vasko03ff5a72019-09-11 13:49:33 +02009310LY_ERR
Michal Vasko400e9672021-01-11 13:39:17 +01009311lyxp_eval(const struct ly_ctx *ctx, const struct lyxp_expr *exp, const struct lys_module *cur_mod,
Radek Krejci8df109d2021-04-23 12:19:08 +02009312 LY_VALUE_FORMAT format, void *prefix_data, const struct lyd_node *ctx_node, const struct lyd_node *tree,
aPiecekfba75362021-10-07 12:39:48 +02009313 const struct lyxp_var *vars, struct lyxp_set *set, uint32_t options)
Michal Vasko03ff5a72019-09-11 13:49:33 +02009314{
Michal Vasko004d3152020-06-11 19:59:22 +02009315 uint16_t tok_idx = 0;
Michal Vaskoddd76592022-01-17 13:34:48 +01009316 const struct lysc_node *snode;
Michal Vasko03ff5a72019-09-11 13:49:33 +02009317 LY_ERR rc;
9318
Michal Vasko400e9672021-01-11 13:39:17 +01009319 LY_CHECK_ARG_RET(ctx, ctx, exp, set, LY_EINVAL);
Radek Krejci8df109d2021-04-23 12:19:08 +02009320 if (!cur_mod && ((format == LY_VALUE_SCHEMA) || (format == LY_VALUE_SCHEMA_RESOLVED))) {
Michal Vaskoddd76592022-01-17 13:34:48 +01009321 LOGERR(ctx, LY_EINVAL, "Current module must be set if schema format is used.");
Michal Vasko5d24f6c2020-10-13 13:49:06 +02009322 return LY_EINVAL;
Michal Vasko004d3152020-06-11 19:59:22 +02009323 }
Michal Vasko03ff5a72019-09-11 13:49:33 +02009324
Michal Vaskod3bb12f2020-12-04 14:33:09 +01009325 if (tree) {
9326 /* adjust the pointer to be the first top-level sibling */
9327 while (tree->parent) {
9328 tree = lyd_parent(tree);
9329 }
9330 tree = lyd_first_sibling(tree);
Michal Vaskoddd76592022-01-17 13:34:48 +01009331
9332 for (snode = tree->schema->parent; snode && (snode->nodetype & (LYS_CASE | LYS_CHOICE)); snode = snode->parent) {}
9333 if (snode) {
9334 /* unable to evaluate absolute paths */
9335 LOGERR(ctx, LY_EINVAL, "Data node \"%s\" has no parent but is not instance of a top-level schema node.",
9336 LYD_NAME(tree));
9337 return LY_EINVAL;
9338 }
Michal Vaskod3bb12f2020-12-04 14:33:09 +01009339 }
9340
Michal Vasko03ff5a72019-09-11 13:49:33 +02009341 /* prepare set for evaluation */
Michal Vasko03ff5a72019-09-11 13:49:33 +02009342 memset(set, 0, sizeof *set);
Michal Vaskod3678892020-05-21 10:06:58 +02009343 set->type = LYXP_SET_NODE_SET;
Michal Vasko5d24f6c2020-10-13 13:49:06 +02009344 set->root_type = lyxp_get_root_type(ctx_node, NULL, options);
9345 set_insert_node(set, (struct lyd_node *)ctx_node, 0, ctx_node ? LYXP_NODE_ELEM : set->root_type, 0);
9346
Michal Vasko400e9672021-01-11 13:39:17 +01009347 set->ctx = (struct ly_ctx *)ctx;
Michal Vasko5d24f6c2020-10-13 13:49:06 +02009348 set->cur_node = ctx_node;
9349 for (set->context_op = ctx_node ? ctx_node->schema : NULL;
Radek Krejci0f969882020-08-21 16:56:47 +02009350 set->context_op && !(set->context_op->nodetype & (LYS_RPC | LYS_ACTION | LYS_NOTIF));
9351 set->context_op = set->context_op->parent) {}
Michal Vaskof03ed032020-03-04 13:31:44 +01009352 set->tree = tree;
Michal Vasko5d24f6c2020-10-13 13:49:06 +02009353 set->cur_mod = cur_mod;
Michal Vasko03ff5a72019-09-11 13:49:33 +02009354 set->format = format;
Michal Vasko5d24f6c2020-10-13 13:49:06 +02009355 set->prefix_data = prefix_data;
aPiecekfba75362021-10-07 12:39:48 +02009356 set->vars = vars;
Michal Vasko03ff5a72019-09-11 13:49:33 +02009357
Radek Krejciddace2c2021-01-08 11:30:56 +01009358 LOG_LOCSET(NULL, set->cur_node, NULL, NULL);
Radek Krejci2efc45b2020-12-22 16:25:44 +01009359
Michal Vasko03ff5a72019-09-11 13:49:33 +02009360 /* evaluate */
Michal Vasko004d3152020-06-11 19:59:22 +02009361 rc = eval_expr_select(exp, &tok_idx, 0, set, options);
Michal Vasko03ff5a72019-09-11 13:49:33 +02009362 if (rc != LY_SUCCESS) {
Michal Vaskod3678892020-05-21 10:06:58 +02009363 lyxp_set_free_content(set);
Michal Vasko03ff5a72019-09-11 13:49:33 +02009364 }
9365
Michal Vasko4a7d4d62021-12-13 17:05:06 +01009366 if (set->cur_node) {
9367 LOG_LOCBACK(0, 1, 0, 0);
9368 }
Michal Vasko03ff5a72019-09-11 13:49:33 +02009369 return rc;
9370}
9371
9372#if 0
9373
9374/* full xml printing of set elements, not used currently */
9375
9376void
9377lyxp_set_print_xml(FILE *f, struct lyxp_set *set)
9378{
9379 uint32_t i;
9380 char *str_num;
9381 struct lyout out;
9382
9383 memset(&out, 0, sizeof out);
9384
9385 out.type = LYOUT_STREAM;
9386 out.method.f = f;
9387
9388 switch (set->type) {
9389 case LYXP_SET_EMPTY:
Michal Vasko5233e962020-08-14 14:26:20 +02009390 ly_print_(&out, "Empty XPath set\n\n");
Michal Vasko03ff5a72019-09-11 13:49:33 +02009391 break;
9392 case LYXP_SET_BOOLEAN:
Michal Vasko5233e962020-08-14 14:26:20 +02009393 ly_print_(&out, "Boolean XPath set:\n");
9394 ly_print_(&out, "%s\n\n", set->value.bool ? "true" : "false");
Michal Vasko03ff5a72019-09-11 13:49:33 +02009395 break;
9396 case LYXP_SET_STRING:
Michal Vasko5233e962020-08-14 14:26:20 +02009397 ly_print_(&out, "String XPath set:\n");
9398 ly_print_(&out, "\"%s\"\n\n", set->value.str);
Michal Vasko03ff5a72019-09-11 13:49:33 +02009399 break;
9400 case LYXP_SET_NUMBER:
Michal Vasko5233e962020-08-14 14:26:20 +02009401 ly_print_(&out, "Number XPath set:\n");
Michal Vasko03ff5a72019-09-11 13:49:33 +02009402
9403 if (isnan(set->value.num)) {
9404 str_num = strdup("NaN");
9405 } else if ((set->value.num == 0) || (set->value.num == -0.0f)) {
9406 str_num = strdup("0");
9407 } else if (isinf(set->value.num) && !signbit(set->value.num)) {
9408 str_num = strdup("Infinity");
9409 } else if (isinf(set->value.num) && signbit(set->value.num)) {
9410 str_num = strdup("-Infinity");
9411 } else if ((long long)set->value.num == set->value.num) {
9412 if (asprintf(&str_num, "%lld", (long long)set->value.num) == -1) {
9413 str_num = NULL;
9414 }
9415 } else {
9416 if (asprintf(&str_num, "%03.1Lf", set->value.num) == -1) {
9417 str_num = NULL;
9418 }
9419 }
9420 if (!str_num) {
9421 LOGMEM;
9422 return;
9423 }
Michal Vasko5233e962020-08-14 14:26:20 +02009424 ly_print_(&out, "%s\n\n", str_num);
Michal Vasko03ff5a72019-09-11 13:49:33 +02009425 free(str_num);
9426 break;
9427 case LYXP_SET_NODE_SET:
Michal Vasko5233e962020-08-14 14:26:20 +02009428 ly_print_(&out, "Node XPath set:\n");
Michal Vasko03ff5a72019-09-11 13:49:33 +02009429
9430 for (i = 0; i < set->used; ++i) {
Michal Vasko5233e962020-08-14 14:26:20 +02009431 ly_print_(&out, "%d. ", i + 1);
Michal Vasko03ff5a72019-09-11 13:49:33 +02009432 switch (set->node_type[i]) {
9433 case LYXP_NODE_ROOT_ALL:
Michal Vasko5233e962020-08-14 14:26:20 +02009434 ly_print_(&out, "ROOT all\n\n");
Michal Vasko03ff5a72019-09-11 13:49:33 +02009435 break;
9436 case LYXP_NODE_ROOT_CONFIG:
Michal Vasko5233e962020-08-14 14:26:20 +02009437 ly_print_(&out, "ROOT config\n\n");
Michal Vasko03ff5a72019-09-11 13:49:33 +02009438 break;
9439 case LYXP_NODE_ROOT_STATE:
Michal Vasko5233e962020-08-14 14:26:20 +02009440 ly_print_(&out, "ROOT state\n\n");
Michal Vasko03ff5a72019-09-11 13:49:33 +02009441 break;
9442 case LYXP_NODE_ROOT_NOTIF:
Michal Vasko5233e962020-08-14 14:26:20 +02009443 ly_print_(&out, "ROOT notification \"%s\"\n\n", set->value.nodes[i]->schema->name);
Michal Vasko03ff5a72019-09-11 13:49:33 +02009444 break;
9445 case LYXP_NODE_ROOT_RPC:
Michal Vasko5233e962020-08-14 14:26:20 +02009446 ly_print_(&out, "ROOT rpc \"%s\"\n\n", set->value.nodes[i]->schema->name);
Michal Vasko03ff5a72019-09-11 13:49:33 +02009447 break;
9448 case LYXP_NODE_ROOT_OUTPUT:
Michal Vasko5233e962020-08-14 14:26:20 +02009449 ly_print_(&out, "ROOT output \"%s\"\n\n", set->value.nodes[i]->schema->name);
Michal Vasko03ff5a72019-09-11 13:49:33 +02009450 break;
9451 case LYXP_NODE_ELEM:
Michal Vasko5233e962020-08-14 14:26:20 +02009452 ly_print_(&out, "ELEM \"%s\"\n", set->value.nodes[i]->schema->name);
Michal Vasko03ff5a72019-09-11 13:49:33 +02009453 xml_print_node(&out, 1, set->value.nodes[i], 1, LYP_FORMAT);
Michal Vasko5233e962020-08-14 14:26:20 +02009454 ly_print_(&out, "\n");
Michal Vasko03ff5a72019-09-11 13:49:33 +02009455 break;
9456 case LYXP_NODE_TEXT:
Michal Vasko5233e962020-08-14 14:26:20 +02009457 ly_print_(&out, "TEXT \"%s\"\n\n", ((struct lyd_node_leaf_list *)set->value.nodes[i])->value_str);
Michal Vasko03ff5a72019-09-11 13:49:33 +02009458 break;
9459 case LYXP_NODE_ATTR:
Michal Vasko5233e962020-08-14 14:26:20 +02009460 ly_print_(&out, "ATTR \"%s\" = \"%s\"\n\n", set->value.attrs[i]->name, set->value.attrs[i]->value);
Michal Vasko03ff5a72019-09-11 13:49:33 +02009461 break;
9462 }
9463 }
9464 break;
9465 }
9466}
9467
9468#endif
9469
9470LY_ERR
Michal Vasko5e0e6eb2019-11-06 15:47:50 +01009471lyxp_set_cast(struct lyxp_set *set, enum lyxp_set_type target)
Michal Vasko03ff5a72019-09-11 13:49:33 +02009472{
9473 long double num;
9474 char *str;
9475 LY_ERR rc;
9476
9477 if (!set || (set->type == target)) {
9478 return LY_SUCCESS;
9479 }
9480
9481 /* it's not possible to convert anything into a node set */
Michal Vaskod3678892020-05-21 10:06:58 +02009482 assert(target != LYXP_SET_NODE_SET);
Michal Vasko03ff5a72019-09-11 13:49:33 +02009483
9484 if (set->type == LYXP_SET_SCNODE_SET) {
Michal Vaskod3678892020-05-21 10:06:58 +02009485 lyxp_set_free_content(set);
Michal Vasko03ff5a72019-09-11 13:49:33 +02009486 return LY_EINVAL;
9487 }
9488
9489 /* to STRING */
Michal Vaskod3678892020-05-21 10:06:58 +02009490 if ((target == LYXP_SET_STRING) || ((target == LYXP_SET_NUMBER) && (set->type == LYXP_SET_NODE_SET))) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02009491 switch (set->type) {
9492 case LYXP_SET_NUMBER:
9493 if (isnan(set->val.num)) {
9494 set->val.str = strdup("NaN");
9495 LY_CHECK_ERR_RET(!set->val.str, LOGMEM(set->ctx), -1);
9496 } else if ((set->val.num == 0) || (set->val.num == -0.0f)) {
9497 set->val.str = strdup("0");
9498 LY_CHECK_ERR_RET(!set->val.str, LOGMEM(set->ctx), -1);
9499 } else if (isinf(set->val.num) && !signbit(set->val.num)) {
9500 set->val.str = strdup("Infinity");
9501 LY_CHECK_ERR_RET(!set->val.str, LOGMEM(set->ctx), -1);
9502 } else if (isinf(set->val.num) && signbit(set->val.num)) {
9503 set->val.str = strdup("-Infinity");
9504 LY_CHECK_ERR_RET(!set->val.str, LOGMEM(set->ctx), -1);
9505 } else if ((long long)set->val.num == set->val.num) {
9506 if (asprintf(&str, "%lld", (long long)set->val.num) == -1) {
9507 LOGMEM_RET(set->ctx);
9508 }
9509 set->val.str = str;
9510 } else {
9511 if (asprintf(&str, "%03.1Lf", set->val.num) == -1) {
9512 LOGMEM_RET(set->ctx);
9513 }
9514 set->val.str = str;
9515 }
9516 break;
9517 case LYXP_SET_BOOLEAN:
Michal Vasko004d3152020-06-11 19:59:22 +02009518 if (set->val.bln) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02009519 set->val.str = strdup("true");
9520 } else {
9521 set->val.str = strdup("false");
9522 }
9523 LY_CHECK_ERR_RET(!set->val.str, LOGMEM(set->ctx), LY_EMEM);
9524 break;
9525 case LYXP_SET_NODE_SET:
Michal Vasko03ff5a72019-09-11 13:49:33 +02009526 /* we need the set sorted, it affects the result */
Michal Vasko5e0e6eb2019-11-06 15:47:50 +01009527 assert(!set_sort(set));
Michal Vasko03ff5a72019-09-11 13:49:33 +02009528
Michal Vasko5e0e6eb2019-11-06 15:47:50 +01009529 rc = cast_node_set_to_string(set, &str);
Michal Vasko03ff5a72019-09-11 13:49:33 +02009530 LY_CHECK_RET(rc);
Michal Vaskod3678892020-05-21 10:06:58 +02009531 lyxp_set_free_content(set);
Michal Vasko03ff5a72019-09-11 13:49:33 +02009532 set->val.str = str;
9533 break;
Michal Vasko03ff5a72019-09-11 13:49:33 +02009534 default:
9535 LOGINT_RET(set->ctx);
9536 }
9537 set->type = LYXP_SET_STRING;
9538 }
9539
9540 /* to NUMBER */
9541 if (target == LYXP_SET_NUMBER) {
9542 switch (set->type) {
9543 case LYXP_SET_STRING:
9544 num = cast_string_to_number(set->val.str);
Michal Vaskod3678892020-05-21 10:06:58 +02009545 lyxp_set_free_content(set);
Michal Vasko03ff5a72019-09-11 13:49:33 +02009546 set->val.num = num;
9547 break;
9548 case LYXP_SET_BOOLEAN:
Michal Vasko004d3152020-06-11 19:59:22 +02009549 if (set->val.bln) {
Michal Vasko03ff5a72019-09-11 13:49:33 +02009550 set->val.num = 1;
9551 } else {
9552 set->val.num = 0;
9553 }
9554 break;
9555 default:
9556 LOGINT_RET(set->ctx);
9557 }
9558 set->type = LYXP_SET_NUMBER;
9559 }
9560
9561 /* to BOOLEAN */
9562 if (target == LYXP_SET_BOOLEAN) {
9563 switch (set->type) {
9564 case LYXP_SET_NUMBER:
9565 if ((set->val.num == 0) || (set->val.num == -0.0f) || isnan(set->val.num)) {
Michal Vasko004d3152020-06-11 19:59:22 +02009566 set->val.bln = 0;
Michal Vasko03ff5a72019-09-11 13:49:33 +02009567 } else {
Michal Vasko004d3152020-06-11 19:59:22 +02009568 set->val.bln = 1;
Michal Vasko03ff5a72019-09-11 13:49:33 +02009569 }
9570 break;
9571 case LYXP_SET_STRING:
9572 if (set->val.str[0]) {
Michal Vaskod3678892020-05-21 10:06:58 +02009573 lyxp_set_free_content(set);
Michal Vasko004d3152020-06-11 19:59:22 +02009574 set->val.bln = 1;
Michal Vasko03ff5a72019-09-11 13:49:33 +02009575 } else {
Michal Vaskod3678892020-05-21 10:06:58 +02009576 lyxp_set_free_content(set);
Michal Vasko004d3152020-06-11 19:59:22 +02009577 set->val.bln = 0;
Michal Vasko03ff5a72019-09-11 13:49:33 +02009578 }
9579 break;
9580 case LYXP_SET_NODE_SET:
Michal Vaskod3678892020-05-21 10:06:58 +02009581 if (set->used) {
9582 lyxp_set_free_content(set);
Michal Vasko004d3152020-06-11 19:59:22 +02009583 set->val.bln = 1;
Michal Vaskod3678892020-05-21 10:06:58 +02009584 } else {
9585 lyxp_set_free_content(set);
Michal Vasko004d3152020-06-11 19:59:22 +02009586 set->val.bln = 0;
Michal Vaskod3678892020-05-21 10:06:58 +02009587 }
Michal Vasko03ff5a72019-09-11 13:49:33 +02009588 break;
9589 default:
9590 LOGINT_RET(set->ctx);
9591 }
9592 set->type = LYXP_SET_BOOLEAN;
9593 }
9594
Michal Vasko03ff5a72019-09-11 13:49:33 +02009595 return LY_SUCCESS;
9596}
9597
9598LY_ERR
Michal Vasko400e9672021-01-11 13:39:17 +01009599lyxp_atomize(const struct ly_ctx *ctx, const struct lyxp_expr *exp, const struct lys_module *cur_mod,
Radek Krejci8df109d2021-04-23 12:19:08 +02009600 LY_VALUE_FORMAT format, void *prefix_data, const struct lysc_node *ctx_scnode, struct lyxp_set *set,
Michal Vasko400e9672021-01-11 13:39:17 +01009601 uint32_t options)
Michal Vasko03ff5a72019-09-11 13:49:33 +02009602{
Radek Krejci2efc45b2020-12-22 16:25:44 +01009603 LY_ERR ret;
Michal Vasko004d3152020-06-11 19:59:22 +02009604 uint16_t tok_idx = 0;
Michal Vasko03ff5a72019-09-11 13:49:33 +02009605
Michal Vasko400e9672021-01-11 13:39:17 +01009606 LY_CHECK_ARG_RET(ctx, ctx, exp, set, LY_EINVAL);
Radek Krejci8df109d2021-04-23 12:19:08 +02009607 if (!cur_mod && ((format == LY_VALUE_SCHEMA) || (format == LY_VALUE_SCHEMA_RESOLVED))) {
Michal Vasko5d24f6c2020-10-13 13:49:06 +02009608 LOGARG(NULL, "Current module must be set if schema format is used.");
9609 return LY_EINVAL;
Michal Vasko004d3152020-06-11 19:59:22 +02009610 }
Michal Vasko03ff5a72019-09-11 13:49:33 +02009611
9612 /* prepare set for evaluation */
Michal Vasko03ff5a72019-09-11 13:49:33 +02009613 memset(set, 0, sizeof *set);
9614 set->type = LYXP_SET_SCNODE_SET;
Michal Vasko5d24f6c2020-10-13 13:49:06 +02009615 set->root_type = lyxp_get_root_type(NULL, ctx_scnode, options);
9616 LY_CHECK_RET(lyxp_set_scnode_insert_node(set, ctx_scnode, ctx_scnode ? LYXP_NODE_ELEM : set->root_type, NULL));
Radek Krejcif13b87b2020-12-01 22:02:17 +01009617 set->val.scnodes[0].in_ctx = LYXP_SET_SCNODE_START;
Michal Vasko5d24f6c2020-10-13 13:49:06 +02009618
Michal Vasko400e9672021-01-11 13:39:17 +01009619 set->ctx = (struct ly_ctx *)ctx;
Michal Vasko5d24f6c2020-10-13 13:49:06 +02009620 set->cur_scnode = ctx_scnode;
Michal Vasko6b26e742020-07-17 15:02:10 +02009621 for (set->context_op = ctx_scnode;
Radek Krejci0f969882020-08-21 16:56:47 +02009622 set->context_op && !(set->context_op->nodetype & (LYS_RPC | LYS_ACTION | LYS_NOTIF));
9623 set->context_op = set->context_op->parent) {}
Michal Vasko5d24f6c2020-10-13 13:49:06 +02009624 set->cur_mod = cur_mod;
Michal Vasko03ff5a72019-09-11 13:49:33 +02009625 set->format = format;
Michal Vasko5d24f6c2020-10-13 13:49:06 +02009626 set->prefix_data = prefix_data;
Michal Vasko03ff5a72019-09-11 13:49:33 +02009627
Radek Krejciddace2c2021-01-08 11:30:56 +01009628 LOG_LOCSET(set->cur_scnode, NULL, NULL, NULL);
Radek Krejci2efc45b2020-12-22 16:25:44 +01009629
Michal Vasko03ff5a72019-09-11 13:49:33 +02009630 /* evaluate */
Radek Krejci2efc45b2020-12-22 16:25:44 +01009631 ret = eval_expr_select(exp, &tok_idx, 0, set, options);
9632
Radek Krejciddace2c2021-01-08 11:30:56 +01009633 LOG_LOCBACK(1, 0, 0, 0);
Radek Krejci2efc45b2020-12-22 16:25:44 +01009634 return ret;
Michal Vasko03ff5a72019-09-11 13:49:33 +02009635}
Michal Vaskod43d71a2020-08-07 14:54:58 +02009636
Jan Kundrátc53a7ec2021-12-09 16:01:19 +01009637LIBYANG_API_DEF const char *
Michal Vaskod43d71a2020-08-07 14:54:58 +02009638lyxp_get_expr(const struct lyxp_expr *path)
9639{
9640 if (!path) {
9641 return NULL;
9642 }
9643
9644 return path->expr;
9645}