blob: fc4bf1c03cf4a4781dc804d97340f174b138dfa5 [file] [log] [blame]
Radek Krejcie7b95092019-05-15 11:03:07 +02001/**
2 * @file tree_data_helpers.c
3 * @author Radek Krejci <rkrejci@cesnet.cz>
4 * @brief Parsing and validation helper functions for data trees
5 *
6 * Copyright (c) 2015 - 2018 CESNET, z.s.p.o.
7 *
8 * This source code is licensed under BSD 3-Clause License (the "License").
9 * You may not use this file except in compliance with the License.
10 * You may obtain a copy of the License at
11 *
12 * https://opensource.org/licenses/BSD-3-Clause
13 */
14#include "common.h"
15
16#include <assert.h>
17#include <stdlib.h>
18
19#include "log.h"
20#include "dict.h"
21#include "plugins_types.h"
22#include "tree_data.h"
23#include "tree_schema.h"
24
25struct lyd_node **
26lyd_node_children_p(struct lyd_node *node)
27{
28 assert(node);
29 switch (node->schema->nodetype) {
30 case LYS_CONTAINER:
31 case LYS_LIST:
32 return &((struct lyd_node_inner*)node)->child;
33 default:
34 return NULL;
35 }
36}
37
38API const struct lyd_node *
39lyd_node_children(const struct lyd_node *node)
40{
41 struct lyd_node **children;
42
43 if (!node) {
44 return NULL;
45 }
46
47 children = lyd_node_children_p((struct lyd_node*)node);
48 if (children) {
49 return *children;
50 } else {
51 return NULL;
52 }
53}
54
55LY_ERR
56lyd_parse_check_options(struct ly_ctx *ctx, int options, const char *func)
57{
58 int x = options & LYD_OPT_TYPEMASK;
59
60 /* LYD_OPT_WHENAUTODEL can be used only with LYD_OPT_DATA or LYD_OPT_CONFIG */
61 if (options & LYD_OPT_WHENAUTODEL) {
62 if ((x == LYD_OPT_EDIT) || (x == LYD_OPT_NOTIF_FILTER)) {
63 LOGERR(ctx, LY_EINVAL, "%s: Invalid options 0x%x (LYD_OPT_DATA_WHENAUTODEL can be used only with LYD_OPT_DATA or LYD_OPT_CONFIG)",
64 func, options);
65 return LY_EINVAL;
66 }
67 }
68
69 if (options & (LYD_OPT_DATA_ADD_YANGLIB | LYD_OPT_DATA_NO_YANGLIB)) {
70 if (x != LYD_OPT_DATA) {
71 LOGERR(ctx, LY_EINVAL, "%s: Invalid options 0x%x (LYD_OPT_DATA_*_YANGLIB can be used only with LYD_OPT_DATA)",
72 func, options);
73 return LY_EINVAL;
74 }
75 }
76
77 /* "is power of 2" algorithm, with 0 exception */
78 if (x && !(x && !(x & (x - 1)))) {
79 LOGERR(ctx, LY_EINVAL, "%s: Invalid options 0x%x (multiple data type flags set).", func, options);
80 return LY_EINVAL;
81 }
82
83 return LY_SUCCESS;
84}
85
86LY_ERR
87lyd_value_validate(struct lyd_node_term *node, const char *value, size_t value_len, int options)
88{
89 LY_ERR ret = LY_SUCCESS;
90 struct ly_err_item *err = NULL;
91 struct ly_ctx *ctx;
92 struct lysc_type *type;
93
94 assert(node);
95
96 ctx = node->schema->module->ctx;
97 type = ((struct lysc_node_leaf*)node->schema)->type;
98 if (type->plugin->validate) {
99 ret = type->plugin->validate(ctx, type, value, value_len, options, &node->value.canonized, &err);
100 if (ret) {
101 ly_err_print(err);
102 LOGVAL(ctx, LY_VLOG_STR, err->path, err->vecode, err->msg);
103 ly_err_free(err);
104 }
105 } else if (options & LY_TYPE_VALIDATE_CANONIZE) {
106 node->value.canonized = lydict_insert(ctx, value, value_len);
107 }
108
109 return ret;
110}
111