| /** |
| * @file test_parser_yin.c |
| * @author David Sedlák <xsedla1d@stud.fit.vutbr.cz> |
| * @brief unit tests for functions from parser_yin.c |
| * |
| * Copyright (c) 2015 - 2019 CESNET, z.s.p.o. |
| * |
| * This source code is licensed under BSD 3-Clause License (the "License"). |
| * You may not use this file except in compliance with the License. |
| * You may obtain a copy of the License at |
| * |
| * https://opensource.org/licenses/BSD-3-Clause |
| */ |
| |
| #include <stdarg.h> |
| #include <stddef.h> |
| #include <setjmp.h> |
| #include <cmocka.h> |
| |
| #include <stdio.h> |
| #include <string.h> |
| #include <stdbool.h> |
| |
| #include "common.h" |
| #include "in.h" |
| #include "parser_internal.h" |
| #include "tree_schema.h" |
| #include "tree_schema_internal.h" |
| #include "xml.h" |
| #include "xpath.h" |
| |
| /* copied from parser_yin.c */ |
| enum yin_argument { |
| YIN_ARG_UNKNOWN = 0, /**< parsed argument can not be matched with any supported yin argument keyword */ |
| YIN_ARG_NAME, /**< argument name */ |
| YIN_ARG_TARGET_NODE, /**< argument target-node */ |
| YIN_ARG_MODULE, /**< argument module */ |
| YIN_ARG_VALUE, /**< argument value */ |
| YIN_ARG_TEXT, /**< argument text */ |
| YIN_ARG_CONDITION, /**< argument condition */ |
| YIN_ARG_URI, /**< argument uri */ |
| YIN_ARG_DATE, /**< argument data */ |
| YIN_ARG_TAG, /**< argument tag */ |
| YIN_ARG_NONE /**< empty (special value) */ |
| }; |
| |
| struct yin_subelement { |
| enum ly_stmt type; /**< type of keyword */ |
| void *dest; /**< meta infromation passed to responsible function (mostly information about where parsed subelement should be stored) */ |
| uint16_t flags; /**< describes constraints of subelement can be set to YIN_SUBELEM_MANDATORY, YIN_SUBELEM_UNIQUE, YIN_SUBELEM_FIRST, YIN_SUBELEM_VER2, and YIN_SUBELEM_DEFAULT_TEXT */ |
| }; |
| |
| struct import_meta { |
| const char *prefix; /**< module prefix. */ |
| struct lysp_import **imports; /**< imports to add to. */ |
| }; |
| |
| struct yin_argument_meta { |
| uint16_t *flags; /**< Argument flags */ |
| const char **argument; /**< Argument value */ |
| }; |
| |
| struct tree_node_meta { |
| struct lysp_node *parent; /**< parent node */ |
| struct lysp_node **nodes; /**< linked list of siblings */ |
| }; |
| |
| struct include_meta { |
| const char *name; /**< Module/submodule name. */ |
| struct lysp_include **includes; /**< [Sized array](@ref sizedarrays) of parsed includes to add to. */ |
| }; |
| |
| struct inout_meta { |
| struct lysp_node *parent; /**< Parent node. */ |
| struct lysp_action_inout *inout_p; /**< inout_p Input/output pointer to write to. */ |
| }; |
| |
| struct minmax_dev_meta { |
| uint32_t *lim; /**< min/max value to write to. */ |
| uint16_t *flags; /**< min/max flags to write to. */ |
| struct lysp_ext_instance **exts; /**< extension instances to add to. */ |
| }; |
| |
| #define YIN_SUBELEM_MANDATORY 0x01 |
| #define YIN_SUBELEM_UNIQUE 0x02 |
| #define YIN_SUBELEM_FIRST 0x04 |
| #define YIN_SUBELEM_VER2 0x08 |
| |
| #define YIN_SUBELEM_PARSED 0x80 |
| |
| /* prototypes of static functions */ |
| enum yin_argument yin_match_argument_name(const char *name, size_t len); |
| LY_ERR yin_parse_content(struct lys_yin_parser_ctx *ctx, struct yin_subelement *subelem_info, size_t subelem_info_size, |
| enum ly_stmt current_element, const char **text_content, struct lysp_ext_instance **exts); |
| LY_ERR yin_validate_value(struct lys_yin_parser_ctx *ctx, enum yang_arg val_type); |
| enum ly_stmt yin_match_keyword(struct lys_yin_parser_ctx *ctx, const char *name, size_t name_len, |
| const char *prefix, size_t prefix_len, enum ly_stmt parrent); |
| LY_ERR yin_parse_extension_instance(struct lys_yin_parser_ctx *ctx, LYEXT_SUBSTMT subelem, LY_ARRAY_COUNT_TYPE subelem_index, |
| struct lysp_ext_instance **exts); |
| LY_ERR yin_parse_element_generic(struct lys_yin_parser_ctx *ctx, enum ly_stmt parent, struct lysp_stmt **element); |
| LY_ERR yin_parse_mod(struct lys_yin_parser_ctx *ctx, struct lysp_module *mod); |
| LY_ERR yin_parse_submod(struct lys_yin_parser_ctx *ctx, struct lysp_submodule *submod); |
| |
| void lysp_ext_instance_free(struct ly_ctx *ctx, struct lysp_ext_instance *ext); |
| void lysp_ext_free(struct ly_ctx *ctx, struct lysp_ext *ext); |
| void lysp_when_free(struct ly_ctx *ctx, struct lysp_when *when); |
| void lysp_type_free(struct ly_ctx *ctx, struct lysp_type *type); |
| void lysp_node_free(struct ly_ctx *ctx, struct lysp_node *node); |
| void lysp_tpdf_free(struct ly_ctx *ctx, struct lysp_tpdf *tpdf); |
| void lysp_refine_free(struct ly_ctx *ctx, struct lysp_refine *ref); |
| void lysp_revision_free(struct ly_ctx *ctx, struct lysp_revision *rev); |
| void lysp_include_free(struct ly_ctx *ctx, struct lysp_include *include); |
| void lysp_feature_free(struct ly_ctx *ctx, struct lysp_feature *feat); |
| void lysp_ident_free(struct ly_ctx *ctx, struct lysp_ident *ident); |
| void lysp_notif_free(struct ly_ctx *ctx, struct lysp_notif *notif); |
| void lysp_grp_free(struct ly_ctx *ctx, struct lysp_grp *grp); |
| void lysp_action_inout_free(struct ly_ctx *ctx, struct lysp_action_inout *inout); |
| void lysp_action_free(struct ly_ctx *ctx, struct lysp_action *action); |
| void lysp_augment_free(struct ly_ctx *ctx, struct lysp_augment *augment); |
| void lysp_deviate_free(struct ly_ctx *ctx, struct lysp_deviate *d); |
| void lysp_deviation_free(struct ly_ctx *ctx, struct lysp_deviation *dev); |
| void lysp_import_free(struct ly_ctx *ctx, struct lysp_import *import); |
| |
| /* wrapping element used for mocking has nothing to do with real module structure */ |
| #define ELEMENT_WRAPPER_START "<status xmlns=\"urn:ietf:params:xml:ns:yang:yin:1\">" |
| #define ELEMENT_WRAPPER_END "</status>" |
| |
| struct test_parser_yin_state { |
| struct ly_ctx *ctx; |
| struct lys_yin_parser_ctx *yin_ctx; |
| struct ly_in *in; |
| bool finished_correctly; |
| }; |
| |
| #define BUFSIZE 1024 |
| char logbuf[BUFSIZE] = {0}; |
| int store = -1; /* negative for infinite logging, positive for limited logging */ |
| |
| /* set to 0 to printing error messages to stderr instead of checking them in code */ |
| #define ENABLE_LOGGER_CHECKING 1 |
| |
| #if ENABLE_LOGGER_CHECKING |
| static void |
| logger(LY_LOG_LEVEL level, const char *msg, const char *path) |
| { |
| (void) level; /* unused */ |
| if (store) { |
| if (path && path[0]) { |
| snprintf(logbuf, BUFSIZE - 1, "%s %s", msg, path); |
| } else { |
| strncpy(logbuf, msg, BUFSIZE - 1); |
| } |
| if (store > 0) { |
| --store; |
| } |
| } |
| } |
| #endif |
| |
| #if ENABLE_LOGGER_CHECKING |
| # define logbuf_assert(str) assert_string_equal(logbuf, str) |
| #else |
| # define logbuf_assert(str) |
| #endif |
| |
| #define TEST_DUP_GENERIC(PREFIX, MEMBER, VALUE1, VALUE2, FUNC, RESULT, LINE, CLEANUP) \ |
| str = PREFIX MEMBER" "VALUE1";"MEMBER" "VALUE2";} ..."; \ |
| assert_int_equal(LY_EVALID, FUNC(&ctx, &str, RESULT)); \ |
| logbuf_assert("Duplicate keyword \""MEMBER"\". Line number "LINE"."); \ |
| CLEANUP |
| |
| int |
| setup_ly_ctx(void **state) |
| { |
| struct test_parser_yin_state *st = NULL; |
| |
| /* allocate state variable */ |
| (*state) = st = calloc(1, sizeof(*st)); |
| if (!st) { |
| fprintf(stderr, "Memmory allocation failed"); |
| return EXIT_FAILURE; |
| } |
| |
| /* create new libyang context */ |
| ly_ctx_new(NULL, 0, &st->ctx); |
| |
| return EXIT_SUCCESS; |
| } |
| |
| int |
| destroy_ly_ctx(void **state) |
| { |
| struct test_parser_yin_state *st = *state; |
| ly_ctx_destroy(st->ctx, NULL); |
| free(st); |
| |
| return EXIT_SUCCESS; |
| } |
| |
| static int |
| setup_f(void **state) |
| { |
| struct test_parser_yin_state *st = *state; |
| |
| #if ENABLE_LOGGER_CHECKING |
| /* setup logger */ |
| ly_set_log_clb(logger, 1); |
| #endif |
| |
| /* allocate parser context */ |
| st->yin_ctx = calloc(1, sizeof(*st->yin_ctx)); |
| st->yin_ctx->format = LYS_IN_YIN; |
| |
| /* allocate new parsed module */ |
| st->yin_ctx->parsed_mod = calloc(1, sizeof *st->yin_ctx->parsed_mod); |
| |
| /* allocate new module */ |
| st->yin_ctx->parsed_mod->mod = calloc(1, sizeof *st->yin_ctx->parsed_mod->mod); |
| st->yin_ctx->parsed_mod->mod->ctx = st->ctx; |
| st->yin_ctx->parsed_mod->mod->parsed = st->yin_ctx->parsed_mod; |
| |
| st->in = NULL; |
| |
| return EXIT_SUCCESS; |
| } |
| |
| static int |
| teardown_f(void **state) |
| { |
| struct test_parser_yin_state *st = *(struct test_parser_yin_state **)state; |
| |
| #if ENABLE_LOGGER_CHECKING |
| /* teardown logger */ |
| if (!st->finished_correctly && logbuf[0] != '\0') { |
| fprintf(stderr, "%s\n", logbuf); |
| } |
| #endif |
| |
| lyxml_ctx_free(st->yin_ctx->xmlctx); |
| lys_module_free(st->yin_ctx->parsed_mod->mod, NULL); |
| free(st->yin_ctx); |
| ly_in_free(st->in, 0); |
| |
| return EXIT_SUCCESS; |
| } |
| |
| static struct test_parser_yin_state* |
| reset_state(void **state) |
| { |
| ((struct test_parser_yin_state *)*state)->finished_correctly = true; |
| logbuf[0] = '\0'; |
| teardown_f(state); |
| setup_f(state); |
| |
| return *state; |
| } |
| |
| void |
| logbuf_clean(void) |
| { |
| logbuf[0] = '\0'; |
| } |
| |
| static int |
| setup_element_test(void **state) |
| { |
| struct test_parser_yin_state *st; |
| |
| setup_f(state); |
| st = *state; |
| |
| lydict_insert(st->ctx, "module-name", 0, &st->yin_ctx->parsed_mod->mod->name); |
| |
| return EXIT_SUCCESS; |
| } |
| |
| static void |
| test_yin_match_keyword(void **state) |
| { |
| struct test_parser_yin_state *st = *state; |
| |
| const char *prefix; |
| size_t prefix_len; |
| /* create mock yin namespace in xml context */ |
| const char *data = "<module xmlns=\"urn:ietf:params:xml:ns:yang:yin:1\" />"; |
| ly_in_new_memory(data, &st->in); |
| lyxml_ctx_new(st->ctx, st->in, &st->yin_ctx->xmlctx); |
| prefix = st->yin_ctx->xmlctx->prefix; |
| prefix_len = st->yin_ctx->xmlctx->prefix_len; |
| |
| assert_int_equal(yin_match_keyword(st->yin_ctx, "anydatax", strlen("anydatax"), prefix, prefix_len, LY_STMT_NONE), LY_STMT_NONE); |
| assert_int_equal(yin_match_keyword(st->yin_ctx, "asdasd", strlen("asdasd"), prefix, prefix_len, LY_STMT_NONE), LY_STMT_NONE); |
| assert_int_equal(yin_match_keyword(st->yin_ctx, "", 0, prefix, prefix_len, LY_STMT_NONE), LY_STMT_NONE); |
| assert_int_equal(yin_match_keyword(st->yin_ctx, "anydata", strlen("anydata"), prefix, prefix_len, LY_STMT_NONE), LY_STMT_ANYDATA); |
| assert_int_equal(yin_match_keyword(st->yin_ctx, "anyxml", strlen("anyxml"), prefix, prefix_len, LY_STMT_NONE), LY_STMT_ANYXML); |
| assert_int_equal(yin_match_keyword(st->yin_ctx, "argument", strlen("argument"), prefix, prefix_len, LY_STMT_NONE), LY_STMT_ARGUMENT); |
| assert_int_equal(yin_match_keyword(st->yin_ctx, "augment", strlen("augment"), prefix, prefix_len, LY_STMT_NONE), LY_STMT_AUGMENT); |
| assert_int_equal(yin_match_keyword(st->yin_ctx, "base", strlen("base"), prefix, prefix_len, LY_STMT_NONE), LY_STMT_BASE); |
| assert_int_equal(yin_match_keyword(st->yin_ctx, "belongs-to", strlen("belongs-to"), prefix, prefix_len, LY_STMT_NONE), LY_STMT_BELONGS_TO); |
| assert_int_equal(yin_match_keyword(st->yin_ctx, "bit", strlen("bit"), prefix, prefix_len, LY_STMT_NONE), LY_STMT_BIT); |
| assert_int_equal(yin_match_keyword(st->yin_ctx, "case", strlen("case"), prefix, prefix_len, LY_STMT_NONE), LY_STMT_CASE); |
| assert_int_equal(yin_match_keyword(st->yin_ctx, "choice", strlen("choice"), prefix, prefix_len, LY_STMT_NONE), LY_STMT_CHOICE); |
| assert_int_equal(yin_match_keyword(st->yin_ctx, "config", strlen("config"), prefix, prefix_len, LY_STMT_NONE), LY_STMT_CONFIG); |
| assert_int_equal(yin_match_keyword(st->yin_ctx, "contact", strlen("contact"), prefix, prefix_len, LY_STMT_NONE), LY_STMT_CONTACT); |
| assert_int_equal(yin_match_keyword(st->yin_ctx, "container", strlen("container"), prefix, prefix_len, LY_STMT_NONE), LY_STMT_CONTAINER); |
| assert_int_equal(yin_match_keyword(st->yin_ctx, "default", strlen("default"), prefix, prefix_len, LY_STMT_NONE), LY_STMT_DEFAULT); |
| assert_int_equal(yin_match_keyword(st->yin_ctx, "description", strlen("description"), prefix, prefix_len, LY_STMT_NONE), LY_STMT_DESCRIPTION); |
| assert_int_equal(yin_match_keyword(st->yin_ctx, "deviate", strlen("deviate"), prefix, prefix_len, LY_STMT_NONE), LY_STMT_DEVIATE); |
| assert_int_equal(yin_match_keyword(st->yin_ctx, "deviation", strlen("deviation"), prefix, prefix_len, LY_STMT_NONE), LY_STMT_DEVIATION); |
| assert_int_equal(yin_match_keyword(st->yin_ctx, "enum", strlen("enum"), prefix, prefix_len, LY_STMT_NONE), LY_STMT_ENUM); |
| assert_int_equal(yin_match_keyword(st->yin_ctx, "error-app-tag", strlen("error-app-tag"), prefix, prefix_len, LY_STMT_NONE), LY_STMT_ERROR_APP_TAG); |
| assert_int_equal(yin_match_keyword(st->yin_ctx, "error-message", strlen("error-message"), prefix, prefix_len, LY_STMT_NONE), LY_STMT_ERROR_MESSAGE); |
| assert_int_equal(yin_match_keyword(st->yin_ctx, "extension", strlen("extension"), prefix, prefix_len, LY_STMT_NONE), LY_STMT_EXTENSION); |
| assert_int_equal(yin_match_keyword(st->yin_ctx, "feature", strlen("feature"), prefix, prefix_len, LY_STMT_NONE), LY_STMT_FEATURE); |
| assert_int_equal(yin_match_keyword(st->yin_ctx, "fraction-digits", strlen("fraction-digits"), prefix, prefix_len, LY_STMT_NONE), LY_STMT_FRACTION_DIGITS); |
| assert_int_equal(yin_match_keyword(st->yin_ctx, "grouping", strlen("grouping"), prefix, prefix_len, LY_STMT_NONE), LY_STMT_GROUPING); |
| assert_int_equal(yin_match_keyword(st->yin_ctx, "identity", strlen("identity"), prefix, prefix_len, LY_STMT_NONE), LY_STMT_IDENTITY); |
| assert_int_equal(yin_match_keyword(st->yin_ctx, "if-feature", strlen("if-feature"), prefix, prefix_len, LY_STMT_NONE), LY_STMT_IF_FEATURE); |
| assert_int_equal(yin_match_keyword(st->yin_ctx, "import", strlen("import"), prefix, prefix_len, LY_STMT_NONE), LY_STMT_IMPORT); |
| assert_int_equal(yin_match_keyword(st->yin_ctx, "include", strlen("include"), prefix, prefix_len, LY_STMT_NONE), LY_STMT_INCLUDE); |
| assert_int_equal(yin_match_keyword(st->yin_ctx, "input", strlen("input"), prefix, prefix_len, LY_STMT_NONE), LY_STMT_INPUT); |
| assert_int_equal(yin_match_keyword(st->yin_ctx, "key", strlen("key"), prefix, prefix_len, LY_STMT_NONE), LY_STMT_KEY); |
| assert_int_equal(yin_match_keyword(st->yin_ctx, "leaf", strlen("leaf"), prefix, prefix_len, LY_STMT_NONE), LY_STMT_LEAF); |
| assert_int_equal(yin_match_keyword(st->yin_ctx, "leaf-list", strlen("leaf-list"), prefix, prefix_len, LY_STMT_NONE), LY_STMT_LEAF_LIST); |
| assert_int_equal(yin_match_keyword(st->yin_ctx, "length", strlen("length"), prefix, prefix_len, LY_STMT_NONE), LY_STMT_LENGTH); |
| assert_int_equal(yin_match_keyword(st->yin_ctx, "list", strlen("list"), prefix, prefix_len, LY_STMT_NONE), LY_STMT_LIST); |
| assert_int_equal(yin_match_keyword(st->yin_ctx, "mandatory", strlen("mandatory"), prefix, prefix_len, LY_STMT_NONE), LY_STMT_MANDATORY); |
| assert_int_equal(yin_match_keyword(st->yin_ctx, "max-elements", strlen("max-elements"), prefix, prefix_len, LY_STMT_NONE), LY_STMT_MAX_ELEMENTS); |
| assert_int_equal(yin_match_keyword(st->yin_ctx, "min-elements", strlen("min-elements"), prefix, prefix_len, LY_STMT_NONE), LY_STMT_MIN_ELEMENTS); |
| assert_int_equal(yin_match_keyword(st->yin_ctx, "modifier", strlen("modifier"), prefix, prefix_len, LY_STMT_NONE), LY_STMT_MODIFIER); |
| assert_int_equal(yin_match_keyword(st->yin_ctx, "module", strlen("module"), prefix, prefix_len, LY_STMT_NONE), LY_STMT_MODULE); |
| assert_int_equal(yin_match_keyword(st->yin_ctx, "must", strlen("must"), prefix, prefix_len, LY_STMT_NONE), LY_STMT_MUST); |
| assert_int_equal(yin_match_keyword(st->yin_ctx, "namespace", strlen("namespace"), prefix, prefix_len, LY_STMT_NONE), LY_STMT_NAMESPACE); |
| assert_int_equal(yin_match_keyword(st->yin_ctx, "notification", strlen("notification"), prefix, prefix_len, LY_STMT_NONE), LY_STMT_NOTIFICATION); |
| assert_int_equal(yin_match_keyword(st->yin_ctx, "ordered-by", strlen("ordered-by"), prefix, prefix_len, LY_STMT_NONE), LY_STMT_ORDERED_BY); |
| assert_int_equal(yin_match_keyword(st->yin_ctx, "organization", strlen("organization"), prefix, prefix_len, LY_STMT_NONE), LY_STMT_ORGANIZATION); |
| assert_int_equal(yin_match_keyword(st->yin_ctx, "output", strlen("output"), prefix, prefix_len, LY_STMT_NONE), LY_STMT_OUTPUT); |
| assert_int_equal(yin_match_keyword(st->yin_ctx, "path", strlen("path"), prefix, prefix_len, LY_STMT_NONE), LY_STMT_PATH); |
| assert_int_equal(yin_match_keyword(st->yin_ctx, "pattern", strlen("pattern"), prefix, prefix_len, LY_STMT_NONE), LY_STMT_PATTERN); |
| assert_int_equal(yin_match_keyword(st->yin_ctx, "position", strlen("position"), prefix, prefix_len, LY_STMT_NONE), LY_STMT_POSITION); |
| assert_int_equal(yin_match_keyword(st->yin_ctx, "prefix", strlen("prefix"), prefix, prefix_len, LY_STMT_NONE), LY_STMT_PREFIX); |
| assert_int_equal(yin_match_keyword(st->yin_ctx, "presence", strlen("presence"), prefix, prefix_len, LY_STMT_NONE), LY_STMT_PRESENCE); |
| assert_int_equal(yin_match_keyword(st->yin_ctx, "range", strlen("range"), prefix, prefix_len, LY_STMT_NONE), LY_STMT_RANGE); |
| assert_int_equal(yin_match_keyword(st->yin_ctx, "reference", strlen("reference"), prefix, prefix_len, LY_STMT_NONE), LY_STMT_REFERENCE); |
| assert_int_equal(yin_match_keyword(st->yin_ctx, "refine", strlen("refine"), prefix, prefix_len, LY_STMT_NONE), LY_STMT_REFINE); |
| assert_int_equal(yin_match_keyword(st->yin_ctx, "require-instance", strlen("require-instance"), prefix, prefix_len, LY_STMT_NONE), LY_STMT_REQUIRE_INSTANCE); |
| assert_int_equal(yin_match_keyword(st->yin_ctx, "revision", strlen("revision"), prefix, prefix_len, LY_STMT_NONE), LY_STMT_REVISION); |
| assert_int_equal(yin_match_keyword(st->yin_ctx, "revision-date", strlen("revision-date"), prefix, prefix_len, LY_STMT_NONE), LY_STMT_REVISION_DATE); |
| assert_int_equal(yin_match_keyword(st->yin_ctx, "rpc", strlen("rpc"), prefix, prefix_len, LY_STMT_NONE), LY_STMT_RPC); |
| assert_int_equal(yin_match_keyword(st->yin_ctx, "status", strlen("status"), prefix, prefix_len, LY_STMT_NONE), LY_STMT_STATUS); |
| assert_int_equal(yin_match_keyword(st->yin_ctx, "submodule", strlen("submodule"), prefix, prefix_len, LY_STMT_NONE), LY_STMT_SUBMODULE); |
| assert_int_equal(yin_match_keyword(st->yin_ctx, "type", strlen("type"), prefix, prefix_len, LY_STMT_NONE), LY_STMT_TYPE); |
| assert_int_equal(yin_match_keyword(st->yin_ctx, "typedef", strlen("typedef"), prefix, prefix_len, LY_STMT_NONE), LY_STMT_TYPEDEF); |
| assert_int_equal(yin_match_keyword(st->yin_ctx, "unique", strlen("unique"), prefix, prefix_len, LY_STMT_NONE), LY_STMT_UNIQUE); |
| assert_int_equal(yin_match_keyword(st->yin_ctx, "units", strlen("units"), prefix, prefix_len, LY_STMT_NONE), LY_STMT_UNITS); |
| assert_int_equal(yin_match_keyword(st->yin_ctx, "uses", strlen("uses"), prefix, prefix_len, LY_STMT_NONE), LY_STMT_USES); |
| assert_int_equal(yin_match_keyword(st->yin_ctx, "value", strlen("value"), prefix, prefix_len, LY_STMT_NONE), LY_STMT_VALUE); |
| assert_int_equal(yin_match_keyword(st->yin_ctx, "when", strlen("when"), prefix, prefix_len, LY_STMT_NONE), LY_STMT_WHEN); |
| assert_int_equal(yin_match_keyword(st->yin_ctx, "yang-version", strlen("yang-version"), prefix, prefix_len, LY_STMT_NONE), LY_STMT_YANG_VERSION); |
| assert_int_equal(yin_match_keyword(st->yin_ctx, "yin-element", strlen("yin-element"), prefix, prefix_len, LY_STMT_NONE), LY_STMT_YIN_ELEMENT); |
| |
| st->finished_correctly = true; |
| } |
| |
| static void |
| test_yin_match_argument_name(void **state) |
| { |
| (void)state; /* unused */ |
| |
| assert_int_equal(yin_match_argument_name("", 5), YIN_ARG_UNKNOWN); |
| assert_int_equal(yin_match_argument_name("qwertyasd", 5), YIN_ARG_UNKNOWN); |
| assert_int_equal(yin_match_argument_name("conditionasd", 8), YIN_ARG_UNKNOWN); |
| assert_int_equal(yin_match_argument_name("condition", 9), YIN_ARG_CONDITION); |
| assert_int_equal(yin_match_argument_name("date", 4), YIN_ARG_DATE); |
| assert_int_equal(yin_match_argument_name("module", 6), YIN_ARG_MODULE); |
| assert_int_equal(yin_match_argument_name("name", 4), YIN_ARG_NAME); |
| assert_int_equal(yin_match_argument_name("tag", 3), YIN_ARG_TAG); |
| assert_int_equal(yin_match_argument_name("target-node", 11), YIN_ARG_TARGET_NODE); |
| assert_int_equal(yin_match_argument_name("text", 4), YIN_ARG_TEXT); |
| assert_int_equal(yin_match_argument_name("uri", 3), YIN_ARG_URI); |
| assert_int_equal(yin_match_argument_name("value", 5), YIN_ARG_VALUE); |
| } |
| |
| static void |
| test_yin_parse_element_generic(void **state) |
| { |
| struct test_parser_yin_state *st = *state; |
| struct lysp_ext_instance exts; |
| LY_ERR ret; |
| |
| memset(&exts, 0, sizeof(exts)); |
| |
| const char *data = "<myext:elem attr=\"value\" xmlns:myext=\"urn:example:extensions\">text_value</myext:elem>"; |
| ly_in_new_memory(data, &st->in); |
| lyxml_ctx_new(st->ctx, st->in, &st->yin_ctx->xmlctx); |
| |
| ret = yin_parse_element_generic(st->yin_ctx, LY_STMT_EXTENSION_INSTANCE, &exts.child); |
| assert_int_equal(ret, LY_SUCCESS); |
| assert_int_equal(st->yin_ctx->xmlctx->status, LYXML_ELEM_CLOSE); |
| assert_string_equal(exts.child->stmt, "urn:example:extensions:elem"); |
| assert_string_equal(exts.child->arg, "text_value"); |
| assert_string_equal(exts.child->child->stmt, "attr"); |
| assert_string_equal(exts.child->child->arg, "value"); |
| assert_true(exts.child->child->flags & LYS_YIN_ATTR); |
| lysp_ext_instance_free(st->ctx, &exts); |
| st = reset_state(state); |
| |
| data = "<myext:elem xmlns:myext=\"urn:example:extensions\"></myext:elem>"; |
| ly_in_new_memory(data, &st->in); |
| lyxml_ctx_new(st->ctx, st->in, &st->yin_ctx->xmlctx); |
| |
| ret = yin_parse_element_generic(st->yin_ctx, LY_STMT_EXTENSION_INSTANCE, &exts.child); |
| assert_int_equal(ret, LY_SUCCESS); |
| assert_int_equal(st->yin_ctx->xmlctx->status, LYXML_ELEM_CLOSE); |
| assert_string_equal(exts.child->stmt, "urn:example:extensions:elem"); |
| assert_null(exts.child->child); |
| assert_null(exts.child->arg); |
| lysp_ext_instance_free(st->ctx, &exts); |
| |
| st->finished_correctly = true; |
| } |
| |
| static void |
| test_yin_parse_extension_instance(void **state) |
| { |
| LY_ERR ret; |
| struct test_parser_yin_state *st = *state; |
| struct lysp_ext_instance *exts = NULL; |
| const char *data = "<myext:ext value1=\"test\" value=\"test2\" xmlns:myext=\"urn:example:extensions\"><myext:subelem>text</myext:subelem></myext:ext>"; |
| ly_in_new_memory(data, &st->in); |
| lyxml_ctx_new(st->ctx, st->in, &st->yin_ctx->xmlctx); |
| |
| ret = yin_parse_extension_instance(st->yin_ctx, LYEXT_SUBSTMT_CONTACT, 0, &exts); |
| assert_int_equal(ret, LY_SUCCESS); |
| assert_string_equal(exts->name, "urn:example:extensions:ext"); |
| assert_int_equal(exts->insubstmt_index, 0); |
| assert_true(exts->insubstmt == LYEXT_SUBSTMT_CONTACT); |
| assert_true(exts->yin & LYS_YIN); |
| assert_string_equal(exts->child->stmt, "value1"); |
| assert_string_equal(exts->child->arg, "test"); |
| assert_null(exts->child->child); |
| assert_true(exts->child->flags & LYS_YIN_ATTR); |
| assert_string_equal(exts->child->next->stmt, "value"); |
| assert_string_equal(exts->child->next->arg, "test2"); |
| assert_null(exts->child->next->child); |
| assert_true(exts->child->next->flags & LYS_YIN_ATTR); |
| |
| assert_string_equal(exts->child->next->next->stmt, "urn:example:extensions:subelem"); |
| assert_string_equal(exts->child->next->next->arg, "text"); |
| assert_null(exts->child->next->next->child); |
| assert_null(exts->child->next->next->next); |
| assert_false(exts->child->next->next->flags & LYS_YIN_ATTR); |
| lysp_ext_instance_free(st->ctx, exts); |
| LY_ARRAY_FREE(exts); |
| exts = NULL; |
| st = reset_state(state); |
| |
| data = "<myext:extension-elem xmlns:myext=\"urn:example:extensions\" />"; |
| ly_in_new_memory(data, &st->in); |
| lyxml_ctx_new(st->ctx, st->in, &st->yin_ctx->xmlctx); |
| |
| ret = yin_parse_extension_instance(st->yin_ctx, LYEXT_SUBSTMT_CONTACT, 0, &exts); |
| assert_int_equal(ret, LY_SUCCESS); |
| assert_string_equal(exts->name, "urn:example:extensions:extension-elem"); |
| assert_null(exts->argument); |
| assert_null(exts->child); |
| assert_int_equal(exts->insubstmt, LYEXT_SUBSTMT_CONTACT); |
| assert_int_equal(exts->insubstmt_index, 0); |
| assert_true(exts->yin & LYS_YIN); |
| lysp_ext_instance_free(st->ctx, exts); |
| LY_ARRAY_FREE(exts); |
| exts = NULL; |
| st = reset_state(state); |
| |
| data = "<myext:ext attr1=\"text1\" attr2=\"text2\" xmlns:myext=\"urn:example:extensions\">" |
| "<myext:ext-sub1/>" |
| "<myext:ext-sub2 sattr1=\"stext2\">" |
| "<myext:ext-sub21>" |
| "<myext:ext-sub211 sattr21=\"text21\"/>" |
| "</myext:ext-sub21>" |
| "</myext:ext-sub2>" |
| "<myext:ext-sub3 attr3=\"text3\"></myext:ext-sub3>" |
| "</myext:ext>"; |
| ly_in_new_memory(data, &st->in); |
| lyxml_ctx_new(st->ctx, st->in, &st->yin_ctx->xmlctx); |
| |
| ret = yin_parse_extension_instance(st->yin_ctx, LYEXT_SUBSTMT_CONTACT, 0, &exts); |
| assert_int_equal(ret, LY_SUCCESS); |
| |
| assert_string_equal(exts->name, "urn:example:extensions:ext"); |
| assert_null(exts->argument); |
| assert_int_equal(exts->insubstmt, LYEXT_SUBSTMT_CONTACT); |
| assert_int_equal(exts->insubstmt_index, 0); |
| assert_true(exts->yin & LYS_YIN); |
| assert_string_equal(exts->child->stmt, "attr1"); |
| assert_string_equal(exts->child->arg, "text1"); |
| assert_null(exts->child->child); |
| assert_true(exts->child->flags & LYS_YIN_ATTR); |
| |
| assert_string_equal(exts->child->next->stmt, "attr2"); |
| assert_string_equal(exts->child->next->arg, "text2"); |
| assert_null(exts->child->next->child); |
| assert_true(exts->child->next->flags & LYS_YIN_ATTR); |
| |
| assert_string_equal(exts->child->next->next->stmt, "urn:example:extensions:ext-sub1"); |
| assert_null(exts->child->next->next->arg); |
| assert_null(exts->child->next->next->child); |
| assert_int_equal(exts->child->next->next->flags, 0); |
| |
| assert_string_equal(exts->child->next->next->next->stmt, "urn:example:extensions:ext-sub2"); |
| assert_null(exts->child->next->next->next->arg); |
| assert_int_equal(exts->child->next->next->next->flags, 0); |
| assert_string_equal(exts->child->next->next->next->child->stmt, "sattr1"); |
| assert_string_equal(exts->child->next->next->next->child->arg, "stext2"); |
| assert_null(exts->child->next->next->next->child->child); |
| assert_true(exts->child->next->next->next->child->flags & LYS_YIN_ATTR); |
| |
| assert_string_equal(exts->child->next->next->next->child->next->stmt, "urn:example:extensions:ext-sub21"); |
| assert_null(exts->child->next->next->next->child->next->arg); |
| assert_null(exts->child->next->next->next->child->next->next); |
| assert_int_equal(exts->child->next->next->next->child->next->flags, 0); |
| |
| assert_string_equal(exts->child->next->next->next->child->next->child->stmt, "urn:example:extensions:ext-sub211"); |
| assert_null(exts->child->next->next->next->child->next->child->arg); |
| assert_int_equal(exts->child->next->next->next->child->next->child->flags, 0); |
| assert_null(exts->child->next->next->next->child->next->child->next); |
| |
| assert_string_equal(exts->child->next->next->next->child->next->child->child->stmt, "sattr21"); |
| assert_string_equal(exts->child->next->next->next->child->next->child->child->arg, "text21"); |
| assert_null(exts->child->next->next->next->child->next->child->child->next); |
| assert_null(exts->child->next->next->next->child->next->child->child->child); |
| assert_true(exts->child->next->next->next->child->next->child->child->flags & LYS_YIN_ATTR); |
| |
| assert_string_equal(exts->child->next->next->next->next->stmt, "urn:example:extensions:ext-sub3"); |
| assert_null(exts->child->next->next->next->next->arg); |
| assert_null(exts->child->next->next->next->next->next); |
| assert_int_equal(exts->child->next->next->next->next->flags, 0); |
| |
| assert_string_equal(exts->child->next->next->next->next->child->stmt, "attr3"); |
| assert_string_equal(exts->child->next->next->next->next->child->arg, "text3"); |
| assert_null(exts->child->next->next->next->next->child->next); |
| assert_null(exts->child->next->next->next->next->child->child); |
| assert_true(exts->child->next->next->next->next->child->flags & LYS_YIN_ATTR); |
| |
| lysp_ext_instance_free(st->ctx, exts); |
| LY_ARRAY_FREE(exts); |
| exts = NULL; |
| st = reset_state(state); |
| |
| data = "<myext:extension-elem xmlns:myext=\"urn:example:extensions\" xmlns:yin=\"urn:ietf:params:xml:ns:yang:yin:1\">" |
| "<yin:action name=\"act-name\" pre:prefixed=\"ignored\"/>" |
| "<yin:augment target-node=\"target\"/>" |
| "<yin:status value=\"value\"/>" |
| "<yin:include module=\"mod\"/>" |
| "<yin:input />" |
| "<yin:must condition=\"cond\"/>" |
| "<yin:namespace uri=\"uri\"/>" |
| "<yin:revision date=\"data\"/>" |
| "<yin:unique tag=\"tag\"/>" |
| "<yin:description><yin:text>contact-val</yin:text></yin:description>" |
| "<yin:error-message><yin:value>err-msg</yin:value></yin:error-message>" |
| "</myext:extension-elem>"; |
| ly_in_new_memory(data, &st->in); |
| lyxml_ctx_new(st->ctx, st->in, &st->yin_ctx->xmlctx); |
| |
| ret = yin_parse_extension_instance(st->yin_ctx, LYEXT_SUBSTMT_CONTACT, 0, &exts); |
| assert_int_equal(ret, LY_SUCCESS); |
| assert_string_equal(exts->child->arg, "act-name"); |
| assert_string_equal(exts->child->next->arg, "target"); |
| assert_string_equal(exts->child->next->next->arg, "value"); |
| assert_string_equal(exts->child->next->next->next->arg, "mod"); |
| assert_null(exts->child->next->next->next->next->arg); |
| assert_string_equal(exts->child->next->next->next->next->next->arg, "cond"); |
| assert_string_equal(exts->child->next->next->next->next->next->next->arg, "uri"); |
| assert_string_equal(exts->child->next->next->next->next->next->next->next->arg, "data"); |
| assert_string_equal(exts->child->next->next->next->next->next->next->next->next->arg, "tag"); |
| assert_string_equal(exts->child->next->next->next->next->next->next->next->next->next->arg, "contact-val"); |
| lysp_ext_instance_free(st->ctx, exts); |
| LY_ARRAY_FREE(exts); |
| exts = NULL; |
| st = reset_state(state); |
| |
| st->finished_correctly = true; |
| } |
| |
| static void |
| test_yin_parse_content(void **state) |
| { |
| struct test_parser_yin_state *st = *state; |
| LY_ERR ret = LY_SUCCESS; |
| const char *data = "<prefix value=\"a_mod\" xmlns=\"urn:ietf:params:xml:ns:yang:yin:1\">" |
| "<myext:custom xmlns:myext=\"urn:example:extensions\">" |
| "totally amazing extension" |
| "</myext:custom>" |
| "<extension name=\"ext\">" |
| "<argument name=\"argname\"></argument>" |
| "<description><text>desc</text></description>" |
| "<reference><text>ref</text></reference>" |
| "<status value=\"deprecated\"></status>" |
| "</extension>" |
| "<text xmlns=\"urn:ietf:params:xml:ns:yang:yin:1\">wsefsdf</text>" |
| "<if-feature name=\"foo\"></if-feature>" |
| "<when condition=\"condition...\">" |
| "<reference><text>when_ref</text></reference>" |
| "<description><text>when_desc</text></description>" |
| "</when>" |
| "<config value=\"true\"/>" |
| "<error-message>" |
| "<value>error-msg</value>" |
| "</error-message>" |
| "<error-app-tag value=\"err-app-tag\"/>" |
| "<units name=\"radians\"></units>" |
| "<default value=\"default-value\"/>" |
| "<position value=\"25\"></position>" |
| "<value value=\"-5\"/>" |
| "<require-instance value=\"true\"></require-instance>" |
| "<range value=\"5..10\" />" |
| "<length value=\"baf\"/>" |
| "<pattern value='pattern'>" |
| "<modifier value='invert-match'/>" |
| "</pattern>" |
| "<enum name=\"yay\">" |
| "</enum>" |
| "</prefix>"; |
| struct lysp_ext_instance *exts = NULL; |
| const char **if_features = NULL; |
| const char *value, *err_msg, *app_tag, *units; |
| struct lysp_qname def = {0}; |
| struct lysp_ext *ext_def = NULL; |
| struct lysp_when *when_p = NULL; |
| struct lysp_type_enum pos_enum = {}, val_enum = {}; |
| struct lysp_type req_type = {}, range_type = {}, len_type = {}, patter_type = {}, enum_type = {}; |
| uint8_t config = 0; |
| |
| ly_in_new_memory(data, &st->in); |
| lyxml_ctx_new(st->ctx, st->in, &st->yin_ctx->xmlctx); |
| lyxml_ctx_next(st->yin_ctx->xmlctx); |
| lyxml_ctx_next(st->yin_ctx->xmlctx); |
| lyxml_ctx_next(st->yin_ctx->xmlctx); |
| |
| struct yin_subelement subelems[17] = { |
| {LY_STMT_CONFIG, &config, 0}, |
| {LY_STMT_DEFAULT, &def, YIN_SUBELEM_UNIQUE}, |
| {LY_STMT_ENUM, &enum_type, 0}, |
| {LY_STMT_ERROR_APP_TAG, &app_tag, YIN_SUBELEM_UNIQUE}, |
| {LY_STMT_ERROR_MESSAGE, &err_msg, 0}, |
| {LY_STMT_EXTENSION, &ext_def, 0}, |
| {LY_STMT_IF_FEATURE, &if_features, 0}, |
| {LY_STMT_LENGTH, &len_type, 0}, |
| {LY_STMT_PATTERN, &patter_type, 0}, |
| {LY_STMT_POSITION, &pos_enum, 0}, |
| {LY_STMT_RANGE, &range_type, 0}, |
| {LY_STMT_REQUIRE_INSTANCE, &req_type, 0}, |
| {LY_STMT_UNITS, &units, YIN_SUBELEM_UNIQUE}, |
| {LY_STMT_VALUE, &val_enum, 0}, |
| {LY_STMT_WHEN, &when_p, 0}, |
| {LY_STMT_EXTENSION_INSTANCE, NULL, 0}, |
| {LY_STMT_ARG_TEXT, &value, 0} |
| }; |
| ret = yin_parse_content(st->yin_ctx, subelems, 17, LY_STMT_PREFIX, NULL, &exts); |
| assert_int_equal(ret, LY_SUCCESS); |
| /* check parsed values */ |
| assert_string_equal(def.str, "default-value"); |
| assert_string_equal(exts->name, "urn:example:extensions:custom"); |
| assert_string_equal(exts->argument, "totally amazing extension"); |
| assert_string_equal(value, "wsefsdf"); |
| assert_string_equal(units, "radians"); |
| assert_string_equal(when_p->cond, "condition..."); |
| assert_string_equal(when_p->dsc, "when_desc"); |
| assert_string_equal(when_p->ref, "when_ref"); |
| assert_int_equal(config, LYS_CONFIG_W); |
| assert_int_equal(pos_enum.value, 25); |
| assert_true(pos_enum.flags & LYS_SET_VALUE); |
| assert_int_equal(val_enum.value, -5); |
| assert_true(val_enum.flags & LYS_SET_VALUE); |
| assert_int_equal(req_type.require_instance, 1); |
| assert_true(req_type.flags &= LYS_SET_REQINST); |
| assert_string_equal(range_type.range->arg.str, "5..10"); |
| assert_true(range_type.flags & LYS_SET_RANGE); |
| assert_string_equal(err_msg, "error-msg"); |
| assert_string_equal(app_tag, "err-app-tag"); |
| assert_string_equal(enum_type.enums->name, "yay"); |
| assert_string_equal(len_type.length->arg.str, "baf"); |
| assert_true(len_type.flags & LYS_SET_LENGTH); |
| assert_string_equal(patter_type.patterns->arg.str, "\x015pattern"); |
| assert_true(patter_type.flags & LYS_SET_PATTERN); |
| /* cleanup */ |
| lysp_ext_instance_free(st->ctx, exts); |
| lysp_when_free(st->ctx, when_p); |
| lysp_ext_free(st->ctx, ext_def); |
| FREE_STRING(st->ctx, *if_features); |
| FREE_STRING(st->ctx, err_msg); |
| FREE_STRING(st->ctx, app_tag); |
| FREE_STRING(st->ctx, units); |
| FREE_STRING(st->ctx, patter_type.patterns->arg.str); |
| FREE_STRING(st->ctx, def.str); |
| FREE_STRING(st->ctx, range_type.range->arg.str); |
| FREE_STRING(st->ctx, len_type.length->arg.str); |
| FREE_STRING(st->ctx, enum_type.enums->name); |
| FREE_STRING(st->ctx, value); |
| LY_ARRAY_FREE(if_features); |
| LY_ARRAY_FREE(exts); |
| LY_ARRAY_FREE(ext_def); |
| LY_ARRAY_FREE(patter_type.patterns); |
| LY_ARRAY_FREE(enum_type.enums); |
| free(when_p); |
| free(range_type.range); |
| free(len_type.length); |
| st = reset_state(state); |
| |
| /* test unique subelem */ |
| const char *prefix_value; |
| struct yin_subelement subelems2[2] = {{LY_STMT_PREFIX, &prefix_value, YIN_SUBELEM_UNIQUE}, |
| {LY_STMT_ARG_TEXT, &value, YIN_SUBELEM_UNIQUE}}; |
| data = ELEMENT_WRAPPER_START |
| "<prefix value=\"inv_mod\" />" |
| "<text xmlns=\"urn:ietf:params:xml:ns:yang:yin:1\">wsefsdf</text>" |
| "<text xmlns=\"urn:ietf:params:xml:ns:yang:yin:1\">wsefsdf</text>" |
| ELEMENT_WRAPPER_END; |
| ly_in_new_memory(data, &st->in); |
| lyxml_ctx_new(st->ctx, st->in, &st->yin_ctx->xmlctx); |
| lyxml_ctx_next(st->yin_ctx->xmlctx); |
| |
| ret = yin_parse_content(st->yin_ctx, subelems2, 2, LY_STMT_STATUS, NULL, &exts); |
| assert_int_equal(ret, LY_EVALID); |
| logbuf_assert("Redefinition of \"text\" sub-element in \"status\" element. Line number 1."); |
| lydict_remove(st->ctx, prefix_value); |
| lydict_remove(st->ctx, value); |
| st = reset_state(state); |
| |
| /* test first subelem */ |
| data = ELEMENT_WRAPPER_START |
| "<prefix value=\"inv_mod\" />" |
| "<text xmlns=\"urn:ietf:params:xml:ns:yang:yin:1\">wsefsdf</text>" |
| "<text xmlns=\"urn:ietf:params:xml:ns:yang:yin:1\">wsefsdf</text>" |
| ELEMENT_WRAPPER_END; |
| struct yin_subelement subelems3[2] = {{LY_STMT_PREFIX, &prefix_value, YIN_SUBELEM_UNIQUE}, |
| {LY_STMT_ARG_TEXT, &value, YIN_SUBELEM_FIRST}}; |
| ly_in_new_memory(data, &st->in); |
| lyxml_ctx_new(st->ctx, st->in, &st->yin_ctx->xmlctx); |
| lyxml_ctx_next(st->yin_ctx->xmlctx); |
| |
| ret = yin_parse_content(st->yin_ctx, subelems3, 2, LY_STMT_STATUS, NULL, &exts); |
| assert_int_equal(ret, LY_EVALID); |
| logbuf_assert("Sub-element \"text\" of \"status\" element must be defined as it's first sub-element. Line number 1."); |
| lydict_remove(st->ctx, prefix_value); |
| st = reset_state(state); |
| |
| /* test mandatory subelem */ |
| data = ELEMENT_WRAPPER_START ELEMENT_WRAPPER_END; |
| struct yin_subelement subelems4[1] = {{LY_STMT_PREFIX, &prefix_value, YIN_SUBELEM_MANDATORY | YIN_SUBELEM_UNIQUE}}; |
| ly_in_new_memory(data, &st->in); |
| lyxml_ctx_new(st->ctx, st->in, &st->yin_ctx->xmlctx); |
| lyxml_ctx_next(st->yin_ctx->xmlctx); |
| |
| ret = yin_parse_content(st->yin_ctx, subelems4, 1, LY_STMT_STATUS, NULL, &exts); |
| assert_int_equal(ret, LY_EVALID); |
| logbuf_assert("Missing mandatory sub-element \"prefix\" of \"status\" element. Line number 1."); |
| |
| st->finished_correctly = true; |
| } |
| |
| static void |
| test_validate_value(void **state) |
| { |
| struct test_parser_yin_state *st = *state; |
| const char *data = ELEMENT_WRAPPER_START ELEMENT_WRAPPER_END; |
| |
| /* create some XML context */ |
| ly_in_new_memory(data, &st->in); |
| lyxml_ctx_new(st->ctx, st->in, &st->yin_ctx->xmlctx); |
| st->yin_ctx->xmlctx->status = LYXML_ELEM_CONTENT; |
| st->yin_ctx->xmlctx->dynamic = 0; |
| |
| st->yin_ctx->xmlctx->value = "#invalid"; |
| st->yin_ctx->xmlctx->value_len = 8; |
| assert_int_equal(yin_validate_value(st->yin_ctx, Y_IDENTIF_ARG), LY_EVALID); |
| logbuf_assert("Invalid identifier character '#' (0x0023). Line number 1."); |
| |
| st->yin_ctx->xmlctx->value = ""; |
| st->yin_ctx->xmlctx->value_len = 0; |
| assert_int_equal(yin_validate_value(st->yin_ctx, Y_STR_ARG), LY_SUCCESS); |
| |
| st->yin_ctx->xmlctx->value = "pre:b"; |
| st->yin_ctx->xmlctx->value_len = 5; |
| assert_int_equal(yin_validate_value(st->yin_ctx, Y_IDENTIF_ARG), LY_EVALID); |
| assert_int_equal(yin_validate_value(st->yin_ctx, Y_PREF_IDENTIF_ARG), LY_SUCCESS); |
| |
| st->yin_ctx->xmlctx->value = "pre:pre:b"; |
| st->yin_ctx->xmlctx->value_len = 9; |
| assert_int_equal(yin_validate_value(st->yin_ctx, Y_PREF_IDENTIF_ARG), LY_EVALID); |
| |
| st->finished_correctly = true; |
| } |
| |
| /* helper function to simplify unit test of each element using parse_content function */ |
| LY_ERR |
| test_element_helper(struct test_parser_yin_state *st, const char *data, void *dest, const char **text, struct lysp_ext_instance **exts) |
| { |
| const char *name, *prefix; |
| size_t name_len, prefix_len; |
| LY_ERR ret = LY_SUCCESS; |
| struct yin_subelement subelems[71] = { |
| {LY_STMT_ACTION, dest, 0}, |
| {LY_STMT_ANYDATA, dest, 0}, |
| {LY_STMT_ANYXML, dest, 0}, |
| {LY_STMT_ARGUMENT,dest, 0}, |
| {LY_STMT_AUGMENT, dest, 0}, |
| {LY_STMT_BASE, dest, 0}, |
| {LY_STMT_BELONGS_TO, dest, 0}, |
| {LY_STMT_BIT, dest, 0}, |
| {LY_STMT_CASE, dest, 0}, |
| {LY_STMT_CHOICE, dest, 0}, |
| {LY_STMT_CONFIG, dest, 0}, |
| {LY_STMT_CONTACT, dest, 0}, |
| {LY_STMT_CONTAINER, dest, 0}, |
| {LY_STMT_DEFAULT, dest, YIN_SUBELEM_UNIQUE}, |
| {LY_STMT_DESCRIPTION, dest, 0}, |
| {LY_STMT_DEVIATE, dest, 0}, |
| {LY_STMT_DEVIATION, dest, 0}, |
| {LY_STMT_ENUM, dest, 0}, |
| {LY_STMT_ERROR_APP_TAG, dest, YIN_SUBELEM_UNIQUE}, |
| {LY_STMT_ERROR_MESSAGE, dest, 0}, |
| {LY_STMT_EXTENSION, dest, 0}, |
| {LY_STMT_FEATURE, dest, 0}, |
| {LY_STMT_FRACTION_DIGITS, dest, 0}, |
| {LY_STMT_GROUPING, dest, 0}, |
| {LY_STMT_IDENTITY, dest, 0}, |
| {LY_STMT_IF_FEATURE, dest, 0}, |
| {LY_STMT_IMPORT, dest, 0}, |
| {LY_STMT_INCLUDE, dest, 0}, |
| {LY_STMT_INPUT, dest, 0}, |
| {LY_STMT_KEY, dest, YIN_SUBELEM_UNIQUE}, |
| {LY_STMT_LEAF, dest, 0}, |
| {LY_STMT_LEAF_LIST, dest, 0}, |
| {LY_STMT_LENGTH, dest, 0}, |
| {LY_STMT_LIST, dest, 0}, |
| {LY_STMT_MANDATORY, dest, 0}, |
| {LY_STMT_MAX_ELEMENTS, dest, 0}, |
| {LY_STMT_MIN_ELEMENTS, dest, 0}, |
| {LY_STMT_MODIFIER, dest, 0}, |
| {LY_STMT_MODULE, dest, 0}, |
| {LY_STMT_MUST, dest, 0}, |
| {LY_STMT_NAMESPACE, dest, YIN_SUBELEM_UNIQUE}, |
| {LY_STMT_NOTIFICATION, dest, 0}, |
| {LY_STMT_ORDERED_BY, dest, 0}, |
| {LY_STMT_ORGANIZATION, dest, 0}, |
| {LY_STMT_OUTPUT, dest, 0}, |
| {LY_STMT_PATH, dest, 0}, |
| {LY_STMT_PATTERN, dest, 0}, |
| {LY_STMT_POSITION, dest, 0}, |
| {LY_STMT_PREFIX, dest, YIN_SUBELEM_UNIQUE}, |
| {LY_STMT_PRESENCE, dest, YIN_SUBELEM_UNIQUE}, |
| {LY_STMT_RANGE, dest, 0}, |
| {LY_STMT_REFERENCE, dest, 0}, |
| {LY_STMT_REFINE, dest, 0}, |
| {LY_STMT_REQUIRE_INSTANCE, dest, 0}, |
| {LY_STMT_REVISION, dest, 0}, |
| {LY_STMT_REVISION_DATE, dest, 0}, |
| {LY_STMT_RPC, dest, 0}, |
| {LY_STMT_STATUS, dest, 0}, |
| {LY_STMT_SUBMODULE, dest, 0}, |
| {LY_STMT_TYPE, dest, 0}, |
| {LY_STMT_TYPEDEF, dest, 0}, |
| {LY_STMT_UNIQUE, dest, 0}, |
| {LY_STMT_UNITS, dest, YIN_SUBELEM_UNIQUE}, |
| {LY_STMT_USES, dest, 0}, |
| {LY_STMT_VALUE, dest, 0}, |
| {LY_STMT_WHEN, dest, 0}, |
| {LY_STMT_YANG_VERSION, dest, 0}, |
| {LY_STMT_YIN_ELEMENT, dest, 0}, |
| {LY_STMT_EXTENSION_INSTANCE, dest, 0}, |
| {LY_STMT_ARG_TEXT, dest, 0}, |
| {LY_STMT_ARG_VALUE, dest, 0} |
| }; |
| ly_in_new_memory(data, &st->in); |
| lyxml_ctx_new(st->ctx, st->in, &st->yin_ctx->xmlctx); |
| prefix = st->yin_ctx->xmlctx->prefix; |
| prefix_len = st->yin_ctx->xmlctx->prefix_len; |
| name = st->yin_ctx->xmlctx->name; |
| name_len = st->yin_ctx->xmlctx->name_len; |
| lyxml_ctx_next(st->yin_ctx->xmlctx); |
| |
| ret = yin_parse_content(st->yin_ctx, subelems, 71, yin_match_keyword(st->yin_ctx, name, name_len, prefix, prefix_len, LY_STMT_NONE), text, exts); |
| |
| /* free parser and input */ |
| lyxml_ctx_free(st->yin_ctx->xmlctx); |
| st->yin_ctx->xmlctx = NULL; |
| ly_in_free(st->in, 0); |
| st->in = NULL; |
| return ret; |
| } |
| |
| #define EXT_SUBELEM "<myext:c-define name=\"MY_MTU\" xmlns:myext=\"urn:example:extensions\"/>" |
| |
| static void |
| test_enum_elem(void **state) |
| { |
| struct test_parser_yin_state *st = *state; |
| struct lysp_type type = {}; |
| const char *data; |
| data = ELEMENT_WRAPPER_START |
| "<enum name=\"enum-name\">" |
| "<if-feature name=\"feature\" />" |
| "<value value=\"55\" />" |
| "<status value=\"deprecated\" />" |
| "<description><text>desc...</text></description>" |
| "<reference><text>ref...</text></reference>" |
| EXT_SUBELEM |
| "</enum>" |
| ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &type, NULL, NULL), LY_SUCCESS); |
| assert_string_equal(type.enums->name, "enum-name"); |
| assert_string_equal(type.enums->iffeatures[0].str, "feature"); |
| assert_int_equal(type.enums->value, 55); |
| assert_true((type.enums->flags & LYS_STATUS_DEPRC) && (type.enums->flags & LYS_SET_VALUE)); |
| assert_string_equal(type.enums->dsc, "desc..."); |
| assert_string_equal(type.enums->ref, "ref..."); |
| assert_string_equal(type.enums->exts->name, "urn:example:extensions:c-define"); |
| assert_int_equal(type.enums->exts->insubstmt_index, 0); |
| assert_int_equal(type.enums->exts->insubstmt, LYEXT_SUBSTMT_SELF); |
| lysp_type_free(st->ctx, &type); |
| memset(&type, 0, sizeof type); |
| |
| data = ELEMENT_WRAPPER_START |
| "<enum name=\"enum-name\"></enum>" |
| ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &type, NULL, NULL), LY_SUCCESS); |
| assert_string_equal(type.enums->name, "enum-name"); |
| lysp_type_free(st->ctx, &type); |
| memset(&type, 0, sizeof type); |
| |
| st->finished_correctly = true; |
| } |
| |
| static void |
| test_bit_elem(void **state) |
| { |
| struct test_parser_yin_state *st = *state; |
| struct lysp_type type = {}; |
| const char *data; |
| data = ELEMENT_WRAPPER_START |
| "<bit name=\"bit-name\">" |
| "<if-feature name=\"feature\" />" |
| "<position value=\"55\" />" |
| "<status value=\"deprecated\" />" |
| "<description><text>desc...</text></description>" |
| "<reference><text>ref...</text></reference>" |
| EXT_SUBELEM |
| "</bit>" |
| ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &type, NULL, NULL), LY_SUCCESS); |
| assert_string_equal(type.bits->name, "bit-name"); |
| assert_string_equal(type.bits->iffeatures[0].str, "feature"); |
| assert_int_equal(type.bits->value, 55); |
| assert_true((type.bits->flags & LYS_STATUS_DEPRC) && (type.bits->flags & LYS_SET_VALUE)); |
| assert_string_equal(type.bits->dsc, "desc..."); |
| assert_string_equal(type.bits->ref, "ref..."); |
| assert_string_equal(type.bits->exts->name, "urn:example:extensions:c-define"); |
| assert_int_equal(type.bits->exts->insubstmt_index, 0); |
| assert_int_equal(type.bits->exts->insubstmt, LYEXT_SUBSTMT_SELF); |
| lysp_type_free(st->ctx, &type); |
| memset(&type, 0, sizeof type); |
| |
| data = ELEMENT_WRAPPER_START |
| "<bit name=\"bit-name\"> </bit>" |
| ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &type, NULL, NULL), LY_SUCCESS); |
| assert_string_equal(type.bits->name, "bit-name"); |
| lysp_type_free(st->ctx, &type); |
| memset(&type, 0, sizeof type); |
| |
| st->finished_correctly = true; |
| } |
| |
| static void |
| test_meta_elem(void **state) |
| { |
| struct test_parser_yin_state *st = *state; |
| char *value = NULL; |
| const char *data; |
| struct lysp_ext_instance *exts = NULL; |
| |
| /* organization element */ |
| data = ELEMENT_WRAPPER_START |
| "<organization><text>organization...</text>" EXT_SUBELEM EXT_SUBELEM "</organization>" |
| ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &value, NULL, &exts), LY_SUCCESS); |
| assert_string_equal(exts[0].name, "urn:example:extensions:c-define"); |
| assert_int_equal(exts[0].insubstmt_index, 0); |
| assert_int_equal(exts[0].insubstmt, LYEXT_SUBSTMT_ORGANIZATION); |
| assert_string_equal(exts[1].name, "urn:example:extensions:c-define"); |
| assert_int_equal(exts[1].insubstmt_index, 0); |
| assert_int_equal(exts[1].insubstmt, LYEXT_SUBSTMT_ORGANIZATION); |
| assert_string_equal(value, "organization..."); |
| FREE_STRING(st->ctx, value); |
| FREE_ARRAY(st->ctx, exts, lysp_ext_instance_free); |
| value = NULL; |
| exts = NULL; |
| |
| /* contact element */ |
| data = ELEMENT_WRAPPER_START |
| "<contact><text>contact...</text>" EXT_SUBELEM "</contact>" |
| ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &value, NULL, &exts), LY_SUCCESS); |
| assert_string_equal(exts[0].name, "urn:example:extensions:c-define"); |
| assert_int_equal(exts[0].insubstmt_index, 0); |
| assert_int_equal(exts[0].insubstmt, LYEXT_SUBSTMT_CONTACT); |
| assert_string_equal(value, "contact..."); |
| FREE_STRING(st->ctx, value); |
| FREE_ARRAY(st->ctx, exts, lysp_ext_instance_free); |
| exts = NULL; |
| value = NULL; |
| |
| /* description element */ |
| data = ELEMENT_WRAPPER_START |
| "<description><text>description...</text>" EXT_SUBELEM "</description>" |
| ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &value, NULL, &exts), LY_SUCCESS); |
| assert_string_equal(value, "description..."); |
| assert_string_equal(exts[0].name, "urn:example:extensions:c-define"); |
| assert_int_equal(exts[0].insubstmt_index, 0); |
| assert_int_equal(exts[0].insubstmt, LYEXT_SUBSTMT_DESCRIPTION); |
| FREE_STRING(st->ctx, value); |
| value = NULL; |
| FREE_ARRAY(st->ctx, exts, lysp_ext_instance_free); |
| exts = NULL; |
| |
| /* reference element */ |
| data = ELEMENT_WRAPPER_START |
| "<reference><text>reference...</text>" EXT_SUBELEM "</reference>" |
| ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &value, NULL, &exts), LY_SUCCESS); |
| assert_string_equal(value, "reference..."); |
| assert_string_equal(exts[0].name, "urn:example:extensions:c-define"); |
| assert_int_equal(exts[0].insubstmt_index, 0); |
| assert_int_equal(exts[0].insubstmt, LYEXT_SUBSTMT_REFERENCE); |
| FREE_STRING(st->ctx, value); |
| value = NULL; |
| FREE_ARRAY(st->ctx, exts, lysp_ext_instance_free); |
| exts = NULL; |
| |
| /* reference element */ |
| data = ELEMENT_WRAPPER_START |
| "<reference invalid=\"text\"><text>reference...</text>""</reference>" |
| ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &value, NULL, &exts), LY_EVALID); |
| logbuf_assert("Unexpected attribute \"invalid\" of \"reference\" element. Line number 1."); |
| FREE_STRING(st->ctx, value); |
| value = NULL; |
| FREE_ARRAY(st->ctx, exts, lysp_ext_instance_free); |
| exts = NULL; |
| |
| /* missing text subelement */ |
| data = ELEMENT_WRAPPER_START |
| "<reference>reference...</reference>" |
| ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &value, NULL, &exts), LY_EVALID); |
| logbuf_assert("Missing mandatory sub-element \"text\" of \"reference\" element. Line number 1."); |
| |
| /* reference element */ |
| data = ELEMENT_WRAPPER_START |
| "<reference>" EXT_SUBELEM "<text>reference...</text></reference>" |
| ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &value, NULL, &exts), LY_EVALID); |
| logbuf_assert("Sub-element \"text\" of \"reference\" element must be defined as it's first sub-element. Line number 1."); |
| FREE_STRING(st->ctx, value); |
| value = NULL; |
| FREE_ARRAY(st->ctx, exts, lysp_ext_instance_free); |
| exts = NULL; |
| |
| st->finished_correctly = true; |
| } |
| |
| static void |
| test_import_elem(void **state) |
| { |
| struct test_parser_yin_state *st = *state; |
| const char *data; |
| struct lysp_import *imports = NULL; |
| struct import_meta imp_meta = {"prefix", &imports}; |
| |
| /* max subelems */ |
| data = ELEMENT_WRAPPER_START |
| "<import module=\"a\">" |
| EXT_SUBELEM |
| "<prefix value=\"a_mod\"/>" |
| "<revision-date date=\"2015-01-01\"></revision-date>" |
| "<description><text>import description</text></description>" |
| "<reference><text>import reference</text></reference>" |
| "</import>" |
| ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &imp_meta, NULL, NULL), LY_SUCCESS); |
| assert_string_equal(imports->name, "a"); |
| assert_string_equal(imports->prefix, "a_mod"); |
| assert_string_equal(imports->rev, "2015-01-01"); |
| assert_string_equal(imports->dsc, "import description"); |
| assert_string_equal(imports->ref, "import reference"); |
| assert_string_equal(imports->exts->name, "urn:example:extensions:c-define"); |
| assert_int_equal(imports->exts->insubstmt, LYEXT_SUBSTMT_SELF); |
| assert_int_equal(imports->exts->insubstmt_index, 0); |
| FREE_ARRAY(st->ctx, imports, lysp_import_free); |
| imports = NULL; |
| |
| /* min subelems */ |
| data = ELEMENT_WRAPPER_START |
| "<import module=\"a\">" |
| "<prefix value=\"a_mod\"/>" |
| "</import>" |
| ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &imp_meta, NULL, NULL), LY_SUCCESS); |
| assert_string_equal(imports->prefix, "a_mod"); |
| FREE_ARRAY(st->ctx, imports, lysp_import_free); |
| imports = NULL; |
| |
| /* invalid (missing prefix) */ |
| data = ELEMENT_WRAPPER_START "<import module=\"a\"></import>" ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &imp_meta, NULL, NULL), LY_EVALID); |
| logbuf_assert("Missing mandatory sub-element \"prefix\" of \"import\" element. Line number 1."); |
| FREE_ARRAY(st->ctx, imports, lysp_import_free); |
| imports = NULL; |
| |
| /* invalid reused prefix */ |
| data = ELEMENT_WRAPPER_START |
| "<import module=\"a\">" |
| "<prefix value=\"prefix\"/>" |
| "</import>" |
| "<import module=\"a\">" |
| "<prefix value=\"prefix\"/>" |
| "</import>" |
| ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &imp_meta, NULL, NULL), LY_EVALID); |
| logbuf_assert("Prefix \"prefix\" already used as module prefix. Line number 1."); |
| FREE_ARRAY(st->ctx, imports, lysp_import_free); |
| imports = NULL; |
| |
| st->finished_correctly = true; |
| } |
| |
| static void |
| test_status_elem(void **state) |
| { |
| struct test_parser_yin_state *st = *state; |
| const char *data; |
| uint16_t flags = 0; |
| struct lysp_ext_instance *exts = NULL; |
| |
| /* test valid values */ |
| data = ELEMENT_WRAPPER_START "<status value=\"current\" />" ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &flags, NULL, NULL), LY_SUCCESS); |
| assert_true(flags & LYS_STATUS_CURR); |
| |
| data = ELEMENT_WRAPPER_START "<status value=\"deprecated\" />" ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &flags, NULL, NULL), LY_SUCCESS); |
| assert_true(flags & LYS_STATUS_DEPRC); |
| |
| data = ELEMENT_WRAPPER_START "<status value=\"obsolete\">"EXT_SUBELEM"</status>" ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &flags, NULL, &exts), LY_SUCCESS); |
| assert_true(flags & LYS_STATUS_OBSLT); |
| assert_string_equal(exts[0].name, "urn:example:extensions:c-define"); |
| assert_int_equal(exts[0].insubstmt_index, 0); |
| assert_int_equal(exts[0].insubstmt, LYEXT_SUBSTMT_STATUS); |
| FREE_ARRAY(st->ctx, exts, lysp_ext_instance_free); |
| exts = NULL; |
| |
| /* test invalid value */ |
| data = ELEMENT_WRAPPER_START "<status value=\"invalid\"></status>" ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &flags, NULL, NULL), LY_EVALID); |
| logbuf_assert("Invalid value \"invalid\" of \"value\" attribute in \"status\" element. Valid values are \"current\", \"deprecated\" and \"obsolete\". Line number 1."); |
| st->finished_correctly = true; |
| } |
| |
| static void |
| test_ext_elem(void **state) |
| { |
| struct test_parser_yin_state *st = *state; |
| const char *data; |
| struct lysp_ext *ext = NULL; |
| |
| /* max subelems */ |
| data = ELEMENT_WRAPPER_START |
| "<extension name=\"ext_name\">" |
| "<argument name=\"arg\"></argument>" |
| "<status value=\"current\"/>" |
| "<description><text>ext_desc</text></description>" |
| "<reference><text>ext_ref</text></reference>" |
| EXT_SUBELEM |
| "</extension>" |
| ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &ext, NULL, NULL), LY_SUCCESS); |
| assert_string_equal(ext->name, "ext_name"); |
| assert_string_equal(ext->argument, "arg"); |
| assert_true(ext->flags & LYS_STATUS_CURR); |
| assert_string_equal(ext->dsc, "ext_desc"); |
| assert_string_equal(ext->ref, "ext_ref"); |
| assert_string_equal(ext->exts[0].name, "urn:example:extensions:c-define"); |
| assert_int_equal(ext->exts[0].insubstmt_index, 0); |
| assert_int_equal(ext->exts[0].insubstmt, LYEXT_SUBSTMT_SELF); |
| lysp_ext_free(st->ctx, ext); |
| LY_ARRAY_FREE(ext); |
| ext = NULL; |
| |
| /* min subelems */ |
| data = ELEMENT_WRAPPER_START "<extension name=\"ext_name\"></extension>" ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &ext, NULL, NULL), LY_SUCCESS); |
| assert_string_equal(ext->name, "ext_name"); |
| lysp_ext_free(st->ctx, ext); |
| LY_ARRAY_FREE(ext); |
| ext = NULL; |
| |
| st->finished_correctly = true; |
| } |
| |
| static void |
| test_yin_element_elem(void **state) |
| { |
| struct test_parser_yin_state *st = *state; |
| const char *data; |
| uint16_t flags = 0; |
| struct lysp_ext_instance *exts = NULL; |
| |
| data = ELEMENT_WRAPPER_START "<yin-element value=\"true\" />" ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &flags, NULL, NULL), LY_SUCCESS); |
| assert_true(flags & LYS_YINELEM_TRUE); |
| |
| data = ELEMENT_WRAPPER_START "<yin-element value=\"false\">" EXT_SUBELEM "</yin-element>" ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &flags, NULL, &exts), LY_SUCCESS); |
| assert_true(flags & LYS_YINELEM_TRUE); |
| assert_string_equal(exts[0].name, "urn:example:extensions:c-define"); |
| assert_int_equal(exts[0].insubstmt_index, 0); |
| assert_int_equal(exts[0].insubstmt, LYEXT_SUBSTMT_YINELEM); |
| FREE_ARRAY(st->ctx, exts, lysp_ext_instance_free); |
| |
| data = ELEMENT_WRAPPER_START "<yin-element value=\"invalid\" />" ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &flags, NULL, NULL), LY_EVALID); |
| assert_true(flags & LYS_YINELEM_TRUE); |
| logbuf_assert("Invalid value \"invalid\" of \"value\" attribute in \"yin-element\" element. Valid values are \"true\" and \"false\". Line number 1."); |
| st->finished_correctly = true; |
| } |
| |
| static void |
| test_yangversion_elem(void **state) |
| { |
| struct test_parser_yin_state *st = *state; |
| const char *data; |
| uint8_t version = 0; |
| struct lysp_ext_instance *exts = NULL; |
| |
| /* valid values */ |
| data = ELEMENT_WRAPPER_START "<yang-version value=\"1\" />" ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &version, NULL, NULL), LY_SUCCESS); |
| assert_true(version & LYS_VERSION_1_0); |
| |
| data = ELEMENT_WRAPPER_START "<yang-version value=\"1.1\">" EXT_SUBELEM "</yang-version>" ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &version, NULL, &exts), LY_SUCCESS); |
| assert_true(version & LYS_VERSION_1_1); |
| assert_string_equal(exts[0].name, "urn:example:extensions:c-define"); |
| assert_int_equal(exts[0].insubstmt_index, 0); |
| assert_int_equal(exts[0].insubstmt, LYEXT_SUBSTMT_VERSION); |
| FREE_ARRAY(st->ctx, exts, lysp_ext_instance_free); |
| |
| /* invalid value */ |
| data = ELEMENT_WRAPPER_START "<yang-version value=\"version\" />" ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &version, NULL, NULL), LY_EVALID); |
| logbuf_assert("Invalid value \"version\" of \"value\" attribute in \"yang-version\" element. Valid values are \"1\" and \"1.1\". Line number 1."); |
| |
| st->finished_correctly = true; |
| } |
| |
| static void |
| test_mandatory_elem(void **state) |
| { |
| struct test_parser_yin_state *st = *state; |
| const char *data; |
| uint16_t man = 0; |
| struct lysp_ext_instance *exts = NULL; |
| |
| /* valid values */ |
| data = ELEMENT_WRAPPER_START "<mandatory value=\"true\" />" ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &man, NULL, NULL), LY_SUCCESS); |
| assert_int_equal(man, LYS_MAND_TRUE); |
| man = 0; |
| |
| data = ELEMENT_WRAPPER_START "<mandatory value=\"false\">" EXT_SUBELEM "</mandatory>" ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &man, NULL, &exts), LY_SUCCESS); |
| assert_int_equal(man, LYS_MAND_FALSE); |
| assert_string_equal(exts[0].name, "urn:example:extensions:c-define"); |
| assert_int_equal(exts[0].insubstmt_index, 0); |
| assert_int_equal(exts[0].insubstmt, LYEXT_SUBSTMT_MANDATORY); |
| FREE_ARRAY(st->ctx, exts, lysp_ext_instance_free); |
| |
| data = ELEMENT_WRAPPER_START "<mandatory value=\"invalid\" />" ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &man, NULL, NULL), LY_EVALID); |
| logbuf_assert("Invalid value \"invalid\" of \"value\" attribute in \"mandatory\" element. Valid values are \"true\" and \"false\". Line number 1."); |
| |
| st->finished_correctly = true; |
| } |
| |
| static void |
| test_argument_elem(void **state) |
| { |
| struct test_parser_yin_state *st = *state; |
| const char *data; |
| uint16_t flags = 0; |
| const char *arg; |
| struct yin_argument_meta arg_meta = {&flags, &arg}; |
| struct lysp_ext_instance *exts = NULL; |
| |
| /* max subelems */ |
| data = ELEMENT_WRAPPER_START |
| "<argument name=\"arg-name\">" |
| "<yin-element value=\"true\" />" |
| EXT_SUBELEM |
| "</argument>" |
| ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &arg_meta, NULL, &exts), LY_SUCCESS); |
| assert_string_equal(arg, "arg-name"); |
| assert_true(flags & LYS_YINELEM_TRUE); |
| assert_string_equal(exts[0].name, "urn:example:extensions:c-define"); |
| assert_int_equal(exts[0].insubstmt_index, 0); |
| assert_int_equal(exts[0].insubstmt, LYEXT_SUBSTMT_ARGUMENT); |
| FREE_ARRAY(st->ctx, exts, lysp_ext_instance_free); |
| exts = NULL; |
| flags = 0; |
| FREE_STRING(st->ctx, arg); |
| arg = NULL; |
| |
| /* min subelems */ |
| data = ELEMENT_WRAPPER_START |
| "<argument name=\"arg\">" |
| "</argument>" |
| ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &arg_meta, NULL, NULL), LY_SUCCESS); |
| assert_string_equal(arg, "arg"); |
| assert_true(flags == 0); |
| FREE_STRING(st->ctx, arg); |
| |
| st->finished_correctly = true; |
| } |
| |
| static void |
| test_base_elem(void **state) |
| { |
| struct test_parser_yin_state *st = *state; |
| const char *data; |
| const char **bases = NULL; |
| struct lysp_ext_instance *exts = NULL; |
| struct lysp_type type = {}; |
| |
| /* as identity subelement */ |
| data = "<identity xmlns=\"urn:ietf:params:xml:ns:yang:yin:1\">" |
| "<base name=\"base-name\">" |
| EXT_SUBELEM |
| "</base>" |
| "</identity>"; |
| assert_int_equal(test_element_helper(st, data, &bases, NULL, &exts), LY_SUCCESS); |
| assert_string_equal(*bases, "base-name"); |
| assert_string_equal(exts[0].name, "urn:example:extensions:c-define"); |
| assert_int_equal(exts[0].insubstmt_index, 0); |
| assert_int_equal(exts[0].insubstmt, LYEXT_SUBSTMT_BASE); |
| FREE_ARRAY(st->ctx, exts, lysp_ext_instance_free); |
| exts = NULL; |
| FREE_STRING(st->ctx, *bases); |
| LY_ARRAY_FREE(bases); |
| |
| /* as type subelement */ |
| data = "<type xmlns=\"urn:ietf:params:xml:ns:yang:yin:1\">" |
| "<base name=\"base-name\">" |
| EXT_SUBELEM |
| "</base>" |
| "</type>"; |
| assert_int_equal(test_element_helper(st, data, &type, NULL, &exts), LY_SUCCESS); |
| assert_string_equal(*type.bases, "base-name"); |
| assert_true(type.flags & LYS_SET_BASE); |
| assert_string_equal(exts[0].name, "urn:example:extensions:c-define"); |
| assert_int_equal(exts[0].insubstmt_index, 0); |
| assert_int_equal(exts[0].insubstmt, LYEXT_SUBSTMT_BASE); |
| FREE_ARRAY(st->ctx, exts, lysp_ext_instance_free); |
| exts = NULL; |
| FREE_STRING(st->ctx, *type.bases); |
| LY_ARRAY_FREE(type.bases); |
| |
| st->finished_correctly = true; |
| } |
| |
| static void |
| test_belongsto_elem(void **state) |
| { |
| struct test_parser_yin_state *st = *state; |
| const char *data; |
| struct lysp_submodule submod; |
| struct lysp_ext_instance *exts = NULL; |
| |
| data = ELEMENT_WRAPPER_START |
| "<belongs-to module=\"module-name\"><prefix value=\"pref\"/>"EXT_SUBELEM"</belongs-to>" |
| ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &submod, NULL, &exts), LY_SUCCESS); |
| assert_string_equal(submod.prefix, "pref"); |
| assert_string_equal(exts[0].name, "urn:example:extensions:c-define"); |
| assert_int_equal(exts[0].insubstmt_index, 0); |
| assert_int_equal(exts[0].insubstmt, LYEXT_SUBSTMT_BELONGSTO); |
| FREE_ARRAY(st->ctx, exts, lysp_ext_instance_free); |
| exts = NULL; |
| FREE_STRING(st->ctx, submod.prefix); |
| |
| data = ELEMENT_WRAPPER_START "<belongs-to module=\"module-name\"></belongs-to>" ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &submod, NULL, NULL), LY_EVALID); |
| logbuf_assert("Missing mandatory sub-element \"prefix\" of \"belongs-to\" element. Line number 1."); |
| |
| st->finished_correctly = true; |
| } |
| |
| static void |
| test_config_elem(void **state) |
| { |
| struct test_parser_yin_state *st = *state; |
| const char *data; |
| uint16_t flags = 0; |
| struct lysp_ext_instance *exts = NULL; |
| |
| data = ELEMENT_WRAPPER_START "<config value=\"true\">" EXT_SUBELEM "</config>" ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &flags, NULL, &exts), LY_SUCCESS); |
| assert_true(flags & LYS_CONFIG_W); |
| assert_string_equal(exts[0].name, "urn:example:extensions:c-define"); |
| assert_int_equal(exts[0].insubstmt_index, 0); |
| assert_int_equal(exts[0].insubstmt, LYEXT_SUBSTMT_CONFIG); |
| FREE_ARRAY(st->ctx, exts, lysp_ext_instance_free); |
| exts = NULL; |
| flags = 0; |
| |
| data = ELEMENT_WRAPPER_START "<config value=\"false\"/>" ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &flags, NULL, NULL), LY_SUCCESS); |
| assert_true(flags & LYS_CONFIG_R); |
| flags = 0; |
| |
| data = ELEMENT_WRAPPER_START "<config value=\"invalid\"/>" ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &flags, NULL, NULL), LY_EVALID); |
| logbuf_assert("Invalid value \"invalid\" of \"value\" attribute in \"config\" element. Valid values are \"true\" and \"false\". Line number 1."); |
| |
| st->finished_correctly = true; |
| } |
| |
| static void |
| test_default_elem(void **state) |
| { |
| struct test_parser_yin_state *st = *state; |
| const char *data; |
| struct lysp_qname val = {0}; |
| struct lysp_ext_instance *exts = NULL; |
| |
| data = ELEMENT_WRAPPER_START "<default value=\"defaul-value\">"EXT_SUBELEM"</default>" ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &val, NULL, &exts), LY_SUCCESS); |
| assert_string_equal(val.str, "defaul-value"); |
| assert_string_equal(exts[0].name, "urn:example:extensions:c-define"); |
| assert_int_equal(exts[0].insubstmt_index, 0); |
| assert_int_equal(exts[0].insubstmt, LYEXT_SUBSTMT_DEFAULT); |
| FREE_ARRAY(st->ctx, exts, lysp_ext_instance_free); |
| exts = NULL; |
| FREE_STRING(st->ctx, val.str); |
| val.str = NULL; |
| |
| data = ELEMENT_WRAPPER_START "<default/>" ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &val, NULL, NULL), LY_EVALID); |
| logbuf_assert("Missing mandatory attribute value of default element. Line number 1."); |
| |
| st->finished_correctly = true; |
| } |
| |
| static void |
| test_err_app_tag_elem(void **state) |
| { |
| struct test_parser_yin_state *st = *state; |
| const char *data; |
| const char *val = NULL; |
| struct lysp_ext_instance *exts = NULL; |
| |
| data = ELEMENT_WRAPPER_START "<error-app-tag value=\"val\">"EXT_SUBELEM"</error-app-tag>" ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &val, NULL, &exts), LY_SUCCESS); |
| assert_string_equal(val, "val"); |
| assert_string_equal(exts[0].name, "urn:example:extensions:c-define"); |
| assert_int_equal(exts[0].insubstmt_index, 0); |
| assert_int_equal(exts[0].insubstmt, LYEXT_SUBSTMT_ERRTAG); |
| FREE_ARRAY(st->ctx, exts, lysp_ext_instance_free); |
| exts = NULL; |
| FREE_STRING(st->ctx, val); |
| val = NULL; |
| |
| data = ELEMENT_WRAPPER_START "<error-app-tag/>" ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &val, NULL, NULL), LY_EVALID); |
| logbuf_assert("Missing mandatory attribute value of error-app-tag element. Line number 1."); |
| |
| st->finished_correctly = true; |
| } |
| |
| static void |
| test_err_msg_elem(void **state) |
| { |
| struct test_parser_yin_state *st = *state; |
| const char *data; |
| const char *val = NULL; |
| struct lysp_ext_instance *exts = NULL; |
| |
| data = ELEMENT_WRAPPER_START "<error-message><value>val</value>"EXT_SUBELEM"</error-message>" ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &val, NULL, &exts), LY_SUCCESS); |
| assert_string_equal(val, "val"); |
| assert_string_equal(exts[0].name, "urn:example:extensions:c-define"); |
| assert_int_equal(exts[0].insubstmt_index, 0); |
| assert_int_equal(exts[0].insubstmt, LYEXT_SUBSTMT_ERRMSG); |
| FREE_ARRAY(st->ctx, exts, lysp_ext_instance_free); |
| exts = NULL; |
| FREE_STRING(st->ctx, val); |
| |
| data = ELEMENT_WRAPPER_START "<error-message></error-message>" ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &val, NULL, NULL), LY_EVALID); |
| logbuf_assert("Missing mandatory sub-element \"value\" of \"error-message\" element. Line number 1."); |
| |
| data = ELEMENT_WRAPPER_START "<error-message invalid=\"text\"/>" ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &val, NULL, NULL), LY_EVALID); |
| logbuf_assert("Unexpected attribute \"invalid\" of \"error-message\" element. Line number 1."); |
| |
| st->finished_correctly = true; |
| } |
| |
| static void |
| test_fracdigits_elem(void **state) |
| { |
| struct test_parser_yin_state *st = *state; |
| const char *data; |
| struct lysp_type type = {}; |
| |
| /* valid value */ |
| data = ELEMENT_WRAPPER_START "<fraction-digits value=\"10\">"EXT_SUBELEM"</fraction-digits>" ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &type, NULL, NULL), LY_SUCCESS); |
| assert_string_equal(type.exts[0].name, "urn:example:extensions:c-define"); |
| assert_int_equal(type.exts[0].insubstmt_index, 0); |
| assert_int_equal(type.exts[0].insubstmt, LYEXT_SUBSTMT_FRACDIGITS); |
| assert_int_equal(type.fraction_digits, 10); |
| assert_true(type.flags & LYS_SET_FRDIGITS); |
| FREE_ARRAY(st->ctx, type.exts, lysp_ext_instance_free); |
| |
| /* invalid values */ |
| data = ELEMENT_WRAPPER_START "<fraction-digits value=\"-1\"></fraction-digits>" ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &type, NULL, NULL), LY_EVALID); |
| logbuf_assert("Invalid value \"-1\" of \"value\" attribute in \"fraction-digits\" element. Line number 1."); |
| |
| data = ELEMENT_WRAPPER_START "<fraction-digits value=\"02\"></fraction-digits>" ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &type, NULL, NULL), LY_EVALID); |
| logbuf_assert("Invalid value \"02\" of \"value\" attribute in \"fraction-digits\" element. Line number 1."); |
| |
| data = ELEMENT_WRAPPER_START "<fraction-digits value=\"1p\"></fraction-digits>" ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &type, NULL, NULL), LY_EVALID); |
| logbuf_assert("Invalid value \"1p\" of \"value\" attribute in \"fraction-digits\" element. Line number 1."); |
| |
| data = ELEMENT_WRAPPER_START "<fraction-digits value=\"19\"></fraction-digits>" ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &type, NULL, NULL), LY_EVALID); |
| logbuf_assert("Invalid value \"19\" of \"value\" attribute in \"fraction-digits\" element. Line number 1."); |
| |
| data = ELEMENT_WRAPPER_START "<fraction-digits value=\"999999999999999999\"></fraction-digits>" ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &type, NULL, NULL), LY_EVALID); |
| logbuf_assert("Invalid value \"999999999999999999\" of \"value\" attribute in \"fraction-digits\" element. Line number 1."); |
| |
| st->finished_correctly = true; |
| } |
| |
| static void |
| test_iffeature_elem(void **state) |
| { |
| struct test_parser_yin_state *st = *state; |
| const char *data; |
| const char **iffeatures = NULL; |
| struct lysp_ext_instance *exts = NULL; |
| |
| data = ELEMENT_WRAPPER_START "<if-feature name=\"local-storage\">"EXT_SUBELEM"</if-feature>" ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &iffeatures, NULL, &exts), LY_SUCCESS); |
| assert_string_equal(*iffeatures, "local-storage"); |
| assert_string_equal(exts[0].name, "urn:example:extensions:c-define"); |
| assert_int_equal(exts[0].insubstmt_index, 0); |
| assert_int_equal(exts[0].insubstmt, LYEXT_SUBSTMT_IFFEATURE); |
| FREE_ARRAY(st->ctx, exts, lysp_ext_instance_free); |
| exts = NULL; |
| FREE_STRING(st->ctx, *iffeatures); |
| LY_ARRAY_FREE(iffeatures); |
| iffeatures = NULL; |
| |
| data = ELEMENT_WRAPPER_START "<if-feature/>" ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &iffeatures, NULL, NULL), LY_EVALID); |
| logbuf_assert("Missing mandatory attribute name of if-feature element. Line number 1."); |
| LY_ARRAY_FREE(iffeatures); |
| iffeatures = NULL; |
| |
| st->finished_correctly = true; |
| } |
| |
| static void |
| test_length_elem(void **state) |
| { |
| struct test_parser_yin_state *st = *state; |
| const char *data; |
| struct lysp_type type = {}; |
| |
| /* max subelems */ |
| data = ELEMENT_WRAPPER_START |
| "<length value=\"length-str\">" |
| "<error-message><value>err-msg</value></error-message>" |
| "<error-app-tag value=\"err-app-tag\"/>" |
| "<description><text>desc</text></description>" |
| "<reference><text>ref</text></reference>" |
| EXT_SUBELEM |
| "</length>" |
| ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &type, NULL, NULL), LY_SUCCESS); |
| assert_string_equal(type.length->arg.str, "length-str"); |
| assert_string_equal(type.length->emsg, "err-msg"); |
| assert_string_equal(type.length->eapptag, "err-app-tag"); |
| assert_string_equal(type.length->dsc, "desc"); |
| assert_string_equal(type.length->ref, "ref"); |
| assert_true(type.flags & LYS_SET_LENGTH); |
| assert_string_equal(type.length->exts[0].name, "urn:example:extensions:c-define"); |
| assert_int_equal(type.length->exts[0].insubstmt_index, 0); |
| assert_int_equal(type.length->exts[0].insubstmt, LYEXT_SUBSTMT_SELF); |
| lysp_type_free(st->ctx, &type); |
| memset(&type, 0, sizeof(type)); |
| |
| /* min subelems */ |
| data = ELEMENT_WRAPPER_START |
| "<length value=\"length-str\">" |
| "</length>" |
| ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &type, NULL, NULL), LY_SUCCESS); |
| assert_string_equal(type.length->arg.str, "length-str"); |
| lysp_type_free(st->ctx, &type); |
| assert_true(type.flags & LYS_SET_LENGTH); |
| memset(&type, 0, sizeof(type)); |
| |
| data = ELEMENT_WRAPPER_START "<length></length>" ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &type, NULL, NULL), LY_EVALID); |
| logbuf_assert("Missing mandatory attribute value of length element. Line number 1."); |
| lysp_type_free(st->ctx, &type); |
| memset(&type, 0, sizeof(type)); |
| |
| st->finished_correctly = true; |
| } |
| |
| static void |
| test_modifier_elem(void **state) |
| { |
| struct test_parser_yin_state *st = *state; |
| const char *data; |
| const char *pat; |
| struct lysp_ext_instance *exts = NULL; |
| |
| assert_int_equal(LY_SUCCESS, lydict_insert(st->ctx, "\006pattern", 8, &pat)); |
| data = ELEMENT_WRAPPER_START "<modifier value=\"invert-match\">" EXT_SUBELEM "</modifier>" ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &pat, NULL, &exts), LY_SUCCESS); |
| assert_string_equal(pat, "\x015pattern"); |
| assert_string_equal(exts[0].name, "urn:example:extensions:c-define"); |
| assert_int_equal(exts[0].insubstmt_index, 0); |
| assert_int_equal(exts[0].insubstmt, LYEXT_SUBSTMT_MODIFIER); |
| FREE_ARRAY(st->ctx, exts, lysp_ext_instance_free); |
| exts = NULL; |
| FREE_STRING(st->ctx, pat); |
| |
| assert_int_equal(LY_SUCCESS, lydict_insert(st->ctx, "\006pattern", 8, &pat)); |
| data = ELEMENT_WRAPPER_START "<modifier value=\"invert\" />" ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &pat, NULL, NULL), LY_EVALID); |
| logbuf_assert("Invalid value \"invert\" of \"value\" attribute in \"modifier\" element. Only valid value is \"invert-match\". Line number 1."); |
| FREE_STRING(st->ctx, pat); |
| |
| st->finished_correctly = true; |
| } |
| |
| static void |
| test_namespace_elem(void **state) |
| { |
| struct test_parser_yin_state *st = *state; |
| const char *data; |
| const char *ns; |
| struct lysp_ext_instance *exts = NULL; |
| |
| data = ELEMENT_WRAPPER_START "<namespace uri=\"ns\">" EXT_SUBELEM "</namespace>" ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &ns, NULL, &exts), LY_SUCCESS); |
| assert_string_equal(ns, "ns"); |
| assert_string_equal(exts[0].name, "urn:example:extensions:c-define"); |
| assert_int_equal(exts[0].insubstmt_index, 0); |
| assert_int_equal(exts[0].insubstmt, LYEXT_SUBSTMT_NAMESPACE); |
| FREE_ARRAY(st->ctx, exts, lysp_ext_instance_free); |
| exts = NULL; |
| FREE_STRING(st->ctx, ns); |
| |
| data = ELEMENT_WRAPPER_START "<namespace/>" ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &ns, NULL, NULL), LY_EVALID); |
| logbuf_assert("Missing mandatory attribute uri of namespace element. Line number 1."); |
| |
| st->finished_correctly = true; |
| } |
| |
| static void |
| test_pattern_elem(void **state) |
| { |
| struct test_parser_yin_state *st = *state; |
| const char *data; |
| struct lysp_type type = {}; |
| |
| /* max subelems */ |
| data = ELEMENT_WRAPPER_START |
| "<pattern value=\"super_pattern\">" |
| "<modifier value=\"invert-match\"/>" |
| "<error-message><value>err-msg-value</value></error-message>" |
| "<error-app-tag value=\"err-app-tag-value\"/>" |
| "<description><text>"pattern-desc"</text></description>" |
| "<reference><text>pattern-ref</text></reference>" |
| EXT_SUBELEM |
| "</pattern>" |
| ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &type, NULL, NULL), LY_SUCCESS); |
| assert_true(type.flags & LYS_SET_PATTERN); |
| assert_string_equal(type.patterns->arg.str, "\x015super_pattern"); |
| assert_string_equal(type.patterns->dsc, "\"pattern-desc\""); |
| assert_string_equal(type.patterns->eapptag, "err-app-tag-value"); |
| assert_string_equal(type.patterns->emsg, "err-msg-value"); |
| assert_string_equal(type.patterns->ref, "pattern-ref"); |
| assert_string_equal(type.patterns->exts[0].name, "urn:example:extensions:c-define"); |
| assert_int_equal(type.patterns->exts[0].insubstmt_index, 0); |
| assert_int_equal(type.patterns->exts[0].insubstmt, LYEXT_SUBSTMT_SELF); |
| lysp_type_free(st->ctx, &type); |
| memset(&type, 0, sizeof(type)); |
| |
| /* min subelems */ |
| data = ELEMENT_WRAPPER_START "<pattern value=\"pattern\"> </pattern>" ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &type, NULL, NULL), LY_SUCCESS); |
| assert_string_equal(type.patterns->arg.str, "\x006pattern"); |
| lysp_type_free(st->ctx, &type); |
| memset(&type, 0, sizeof(type)); |
| |
| st->finished_correctly = true; |
| } |
| |
| static void |
| test_value_position_elem(void **state) |
| { |
| struct test_parser_yin_state *st = *state; |
| const char *data; |
| struct lysp_type_enum en = {}; |
| |
| /* valid values */ |
| data = ELEMENT_WRAPPER_START "<value value=\"55\">" EXT_SUBELEM "</value>" ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &en, NULL, NULL), LY_SUCCESS); |
| assert_int_equal(en.value, 55); |
| assert_true(en.flags & LYS_SET_VALUE); |
| assert_string_equal(en.exts[0].name, "urn:example:extensions:c-define"); |
| assert_int_equal(en.exts[0].insubstmt_index, 0); |
| assert_int_equal(en.exts[0].insubstmt, LYEXT_SUBSTMT_VALUE); |
| FREE_ARRAY(st->ctx, en.exts, lysp_ext_instance_free); |
| memset(&en, 0, sizeof(en)); |
| |
| data = ELEMENT_WRAPPER_START "<value value=\"-55\"/>" ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &en, NULL, NULL), LY_SUCCESS); |
| assert_int_equal(en.value, -55); |
| assert_true(en.flags & LYS_SET_VALUE); |
| memset(&en, 0, sizeof(en)); |
| |
| data = ELEMENT_WRAPPER_START "<value value=\"0\"/>" ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &en, NULL, NULL), LY_SUCCESS); |
| assert_int_equal(en.value, 0); |
| assert_true(en.flags & LYS_SET_VALUE); |
| memset(&en, 0, sizeof(en)); |
| |
| data = ELEMENT_WRAPPER_START "<value value=\"-0\"/>" ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &en, NULL, NULL), LY_SUCCESS); |
| assert_int_equal(en.value, 0); |
| assert_true(en.flags & LYS_SET_VALUE); |
| memset(&en, 0, sizeof(en)); |
| |
| /* valid positions */ |
| data = ELEMENT_WRAPPER_START "<position value=\"55\">" EXT_SUBELEM "</position>" ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &en, NULL, NULL), LY_SUCCESS); |
| assert_int_equal(en.value, 55); |
| assert_true(en.flags & LYS_SET_VALUE); |
| assert_string_equal(en.exts[0].name, "urn:example:extensions:c-define"); |
| assert_int_equal(en.exts[0].insubstmt_index, 0); |
| assert_int_equal(en.exts[0].insubstmt, LYEXT_SUBSTMT_POSITION); |
| FREE_ARRAY(st->ctx, en.exts, lysp_ext_instance_free); |
| memset(&en, 0, sizeof(en)); |
| |
| data = ELEMENT_WRAPPER_START "<position value=\"0\" />" ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &en, NULL, NULL), LY_SUCCESS); |
| assert_int_equal(en.value, 0); |
| assert_true(en.flags & LYS_SET_VALUE); |
| memset(&en, 0, sizeof(en)); |
| |
| /* invalid values */ |
| data = ELEMENT_WRAPPER_START "<value value=\"99999999999999999999999\"/>" ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &en, NULL, NULL), LY_EVALID); |
| logbuf_assert("Invalid value \"99999999999999999999999\" of \"value\" attribute in \"value\" element. Line number 1."); |
| |
| data = ELEMENT_WRAPPER_START "<value value=\"1k\"/>" ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &en, NULL, NULL), LY_EVALID); |
| logbuf_assert("Invalid value \"1k\" of \"value\" attribute in \"value\" element. Line number 1."); |
| |
| data = ELEMENT_WRAPPER_START "<value value=\"\"/>" ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &en, NULL, NULL), LY_EVALID); |
| logbuf_assert("Invalid value \"\" of \"value\" attribute in \"value\" element. Line number 1."); |
| |
| /*invalid positions */ |
| data = ELEMENT_WRAPPER_START "<position value=\"-5\"/>" ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &en, NULL, NULL), LY_EVALID); |
| logbuf_assert("Invalid value \"-5\" of \"value\" attribute in \"position\" element. Line number 1."); |
| |
| data = ELEMENT_WRAPPER_START "<position value=\"-0\"/>" ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &en, NULL, NULL), LY_EVALID); |
| logbuf_assert("Invalid value \"-0\" of \"value\" attribute in \"position\" element. Line number 1."); |
| |
| data = ELEMENT_WRAPPER_START "<position value=\"99999999999999999999\"/>" ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &en, NULL, NULL), LY_EVALID); |
| logbuf_assert("Invalid value \"99999999999999999999\" of \"value\" attribute in \"position\" element. Line number 1."); |
| |
| data = ELEMENT_WRAPPER_START "<position value=\"\"/>" ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &en, NULL, NULL), LY_EVALID); |
| logbuf_assert("Invalid value \"\" of \"value\" attribute in \"position\" element. Line number 1."); |
| |
| st->finished_correctly = true; |
| } |
| |
| static void |
| test_prefix_elem(void **state) |
| { |
| struct test_parser_yin_state *st = *state; |
| const char *data; |
| const char *value = NULL; |
| struct lysp_ext_instance *exts = NULL; |
| |
| data = ELEMENT_WRAPPER_START "<prefix value=\"pref\">" EXT_SUBELEM "</prefix>" ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &value, NULL, &exts), LY_SUCCESS); |
| assert_string_equal(value, "pref"); |
| assert_string_equal(exts[0].name, "urn:example:extensions:c-define"); |
| assert_int_equal(exts[0].insubstmt_index, 0); |
| assert_int_equal(exts[0].insubstmt, LYEXT_SUBSTMT_PREFIX); |
| FREE_ARRAY(st->ctx, exts, lysp_ext_instance_free); |
| exts = NULL; |
| FREE_STRING(st->ctx, value); |
| |
| data = ELEMENT_WRAPPER_START "<prefix value=\"pref\"/>" ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &value, NULL, NULL), LY_SUCCESS); |
| assert_string_equal(value, "pref"); |
| FREE_STRING(st->ctx, value); |
| |
| st->finished_correctly = true; |
| } |
| |
| static void |
| test_range_elem(void **state) |
| { |
| struct test_parser_yin_state *st = *state; |
| const char *data; |
| struct lysp_type type = {}; |
| |
| /* max subelems */ |
| data = ELEMENT_WRAPPER_START |
| "<range value=\"range-str\">" |
| "<error-message><value>err-msg</value></error-message>" |
| "<error-app-tag value=\"err-app-tag\" />" |
| "<description><text>desc</text></description>" |
| "<reference><text>ref</text></reference>" |
| EXT_SUBELEM |
| "</range>" |
| ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &type, NULL, NULL), LY_SUCCESS); |
| assert_string_equal(type.range->arg.str, "range-str"); |
| assert_string_equal(type.range->dsc, "desc"); |
| assert_string_equal(type.range->eapptag, "err-app-tag"); |
| assert_string_equal(type.range->emsg, "err-msg"); |
| assert_string_equal(type.range->ref, "ref"); |
| assert_true(type.flags & LYS_SET_RANGE); |
| assert_string_equal(type.range->exts[0].name, "urn:example:extensions:c-define"); |
| assert_int_equal(type.range->exts[0].insubstmt_index, 0); |
| assert_int_equal(type.range->exts[0].insubstmt, LYEXT_SUBSTMT_SELF); |
| lysp_type_free(st->ctx, &type); |
| memset(&type, 0, sizeof(type)); |
| |
| /* min subelems */ |
| data = ELEMENT_WRAPPER_START "<range value=\"range-str\"/>" ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &type, NULL, NULL), LY_SUCCESS); |
| assert_string_equal(type.range->arg.str, "range-str"); |
| lysp_type_free(st->ctx, &type); |
| memset(&type, 0, sizeof(type)); |
| |
| st->finished_correctly = true; |
| } |
| |
| static void |
| test_reqinstance_elem(void **state) |
| { |
| struct test_parser_yin_state *st = *state; |
| const char *data; |
| struct lysp_type type = {}; |
| |
| data = ELEMENT_WRAPPER_START "<require-instance value=\"true\">" EXT_SUBELEM "</require-instance>" ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &type, NULL, NULL), LY_SUCCESS); |
| assert_int_equal(type.require_instance, 1); |
| assert_true(type.flags & LYS_SET_REQINST); |
| assert_string_equal(type.exts[0].name, "urn:example:extensions:c-define"); |
| assert_int_equal(type.exts[0].insubstmt_index, 0); |
| assert_int_equal(type.exts[0].insubstmt, LYEXT_SUBSTMT_REQINSTANCE); |
| lysp_type_free(st->ctx, &type); |
| memset(&type, 0, sizeof(type)); |
| |
| data = ELEMENT_WRAPPER_START "<require-instance value=\"false\"/>" ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &type, NULL, NULL), LY_SUCCESS); |
| assert_int_equal(type.require_instance, 0); |
| assert_true(type.flags & LYS_SET_REQINST); |
| memset(&type, 0, sizeof(type)); |
| |
| data = ELEMENT_WRAPPER_START "<require-instance value=\"invalid\"/>" ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &type, NULL, NULL), LY_EVALID); |
| memset(&type, 0, sizeof(type)); |
| logbuf_assert("Invalid value \"invalid\" of \"value\" attribute in \"require-instance\" element. Valid values are \"true\" and \"false\". Line number 1."); |
| |
| st->finished_correctly = true; |
| } |
| |
| static void |
| test_revision_date_elem(void **state) |
| { |
| struct test_parser_yin_state *st = *state; |
| const char *data; |
| char rev[LY_REV_SIZE]; |
| struct lysp_ext_instance *exts = NULL; |
| |
| data = ELEMENT_WRAPPER_START "<revision-date date=\"2000-01-01\">"EXT_SUBELEM"</revision-date>" ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, rev, NULL, &exts), LY_SUCCESS); |
| assert_string_equal(rev, "2000-01-01"); |
| assert_string_equal(exts[0].name, "urn:example:extensions:c-define"); |
| assert_int_equal(exts[0].insubstmt_index, 0); |
| assert_int_equal(exts[0].insubstmt, LYEXT_SUBSTMT_REVISIONDATE); |
| FREE_ARRAY(st->ctx, exts, lysp_ext_instance_free); |
| |
| data = ELEMENT_WRAPPER_START "<revision-date date=\"2000-01-01\"/>" ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, rev, NULL, NULL), LY_SUCCESS); |
| assert_string_equal(rev, "2000-01-01"); |
| |
| data = ELEMENT_WRAPPER_START "<revision-date date=\"2000-50-05\"/>" ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, rev, NULL, NULL), LY_EVALID); |
| logbuf_assert("Invalid value \"2000-50-05\" of \"revision-date\". Line number 1."); |
| |
| st->finished_correctly = true; |
| } |
| |
| static void |
| test_unique_elem(void **state) |
| { |
| struct test_parser_yin_state *st = *state; |
| const char *data; |
| const char **values = NULL; |
| struct lysp_ext_instance *exts = NULL; |
| |
| data = ELEMENT_WRAPPER_START "<unique tag=\"tag\">"EXT_SUBELEM"</unique>" ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &values, NULL, &exts), LY_SUCCESS); |
| assert_string_equal(*values, "tag"); |
| assert_string_equal(exts[0].name, "urn:example:extensions:c-define"); |
| assert_int_equal(exts[0].insubstmt_index, 0); |
| assert_int_equal(exts[0].insubstmt, LYEXT_SUBSTMT_UNIQUE); |
| FREE_ARRAY(st->ctx, exts, lysp_ext_instance_free); |
| FREE_STRING(st->ctx, *values); |
| LY_ARRAY_FREE(values); |
| values = NULL; |
| |
| data = ELEMENT_WRAPPER_START "<unique tag=\"tag\"/>" ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &values, NULL, NULL), LY_SUCCESS); |
| assert_string_equal(*values, "tag"); |
| FREE_STRING(st->ctx, *values); |
| LY_ARRAY_FREE(values); |
| values = NULL; |
| |
| st->finished_correctly = true; |
| } |
| |
| static void |
| test_units_elem(void **state) |
| { |
| struct test_parser_yin_state *st = *state; |
| const char *data; |
| const char *values = NULL; |
| struct lysp_ext_instance *exts = NULL; |
| |
| data = ELEMENT_WRAPPER_START "<units name=\"name\">"EXT_SUBELEM"</units>" ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &values, NULL, &exts), LY_SUCCESS); |
| assert_string_equal(values, "name"); |
| assert_string_equal(exts[0].name, "urn:example:extensions:c-define"); |
| assert_int_equal(exts[0].insubstmt_index, 0); |
| assert_int_equal(exts[0].insubstmt, LYEXT_SUBSTMT_UNITS); |
| FREE_ARRAY(st->ctx, exts, lysp_ext_instance_free); |
| FREE_STRING(st->ctx, values); |
| values = NULL; |
| |
| data = ELEMENT_WRAPPER_START "<units name=\"name\"/>" ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &values, NULL, NULL), LY_SUCCESS); |
| assert_string_equal(values, "name"); |
| FREE_STRING(st->ctx, values); |
| values = NULL; |
| |
| st->finished_correctly = true; |
| } |
| |
| static void |
| test_when_elem(void **state) |
| { |
| struct test_parser_yin_state *st = *state; |
| const char *data; |
| struct lysp_when *when = NULL; |
| |
| data = ELEMENT_WRAPPER_START |
| "<when condition=\"cond\">" |
| "<description><text>desc</text></description>" |
| "<reference><text>ref</text></reference>" |
| EXT_SUBELEM |
| "</when>" |
| ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &when, NULL, NULL), LY_SUCCESS); |
| assert_string_equal(when->cond, "cond"); |
| assert_string_equal(when->dsc, "desc"); |
| assert_string_equal(when->ref, "ref"); |
| assert_string_equal(when->exts[0].name, "urn:example:extensions:c-define"); |
| assert_int_equal(when->exts[0].insubstmt_index, 0); |
| assert_int_equal(when->exts[0].insubstmt, LYEXT_SUBSTMT_SELF); |
| lysp_when_free(st->ctx, when); |
| free(when); |
| when = NULL; |
| |
| data = ELEMENT_WRAPPER_START "<when condition=\"cond\" />" ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &when, NULL, NULL), LY_SUCCESS); |
| assert_string_equal(when->cond, "cond"); |
| lysp_when_free(st->ctx, when); |
| free(when); |
| when = NULL; |
| |
| st->finished_correctly = true; |
| } |
| |
| static void |
| test_yin_text_value_elem(void **state) |
| { |
| struct test_parser_yin_state *st = *state; |
| const char *data; |
| const char *val; |
| |
| data = ELEMENT_WRAPPER_START "<text>text</text>" ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &val, NULL, NULL), LY_SUCCESS); |
| assert_string_equal(val, "text"); |
| FREE_STRING(st->ctx, val); |
| |
| data = "<error-message xmlns=\"urn:ietf:params:xml:ns:yang:yin:1\"> <value>text</value> </error-message>"; |
| assert_int_equal(test_element_helper(st, data, &val, NULL, NULL), LY_SUCCESS); |
| assert_string_equal(val, "text"); |
| FREE_STRING(st->ctx, val); |
| |
| data = ELEMENT_WRAPPER_START "<text></text>" ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &val, NULL, NULL), LY_SUCCESS); |
| assert_string_equal("", val); |
| FREE_STRING(st->ctx, val); |
| |
| st->finished_correctly = true; |
| } |
| |
| static void |
| test_type_elem(void **state) |
| { |
| struct test_parser_yin_state *st = *state; |
| const char *data; |
| struct lysp_type type = {}; |
| |
| /* max subelems */ |
| data = ELEMENT_WRAPPER_START |
| "<type name=\"type-name\">" |
| "<base name=\"base-name\"/>" |
| "<bit name=\"bit\"/>" |
| "<enum name=\"enum\"/>" |
| "<fraction-digits value=\"2\"/>" |
| "<length value=\"length\"/>" |
| "<path value=\"/path\"/>" |
| "<pattern value=\"pattern\"/>" |
| "<range value=\"range\" />" |
| "<require-instance value=\"true\"/>" |
| "<type name=\"sub-type-name\"/>" |
| EXT_SUBELEM |
| "</type>" |
| ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &type, NULL, NULL), LY_SUCCESS); |
| assert_string_equal(type.name, "type-name"); |
| assert_string_equal(*type.bases, "base-name"); |
| assert_string_equal(type.bits->name, "bit"); |
| assert_string_equal(type.enums->name, "enum"); |
| assert_int_equal(type.fraction_digits, 2); |
| assert_string_equal(type.length->arg.str, "length"); |
| assert_string_equal(type.path->expr, "/path"); |
| assert_string_equal(type.patterns->arg.str, "\006pattern"); |
| assert_string_equal(type.range->arg.str, "range"); |
| assert_int_equal(type.require_instance, 1); |
| assert_string_equal(type.types->name, "sub-type-name"); |
| assert_string_equal(type.exts[0].name, "urn:example:extensions:c-define"); |
| assert_int_equal(type.exts[0].insubstmt_index, 0); |
| assert_int_equal(type.exts[0].insubstmt, LYEXT_SUBSTMT_SELF); |
| assert_true(type.flags & LYS_SET_BASE); |
| assert_true(type.flags & LYS_SET_BIT); |
| assert_true(type.flags & LYS_SET_ENUM); |
| assert_true(type.flags & LYS_SET_FRDIGITS); |
| assert_true(type.flags & LYS_SET_LENGTH); |
| assert_true(type.flags & LYS_SET_PATH); |
| assert_true(type.flags & LYS_SET_PATTERN); |
| assert_true(type.flags & LYS_SET_RANGE); |
| assert_true(type.flags & LYS_SET_REQINST); |
| assert_true(type.flags & LYS_SET_TYPE); |
| lysp_type_free(st->ctx, &type); |
| memset(&type, 0, sizeof(type)); |
| |
| /* min subelems */ |
| data = ELEMENT_WRAPPER_START "<type name=\"type-name\"/>" ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &type, NULL, NULL), LY_SUCCESS); |
| lysp_type_free(st->ctx, &type); |
| memset(&type, 0, sizeof(type)); |
| |
| st->finished_correctly = true; |
| } |
| |
| static void |
| test_max_elems_elem(void **state) |
| { |
| struct test_parser_yin_state *st = *state; |
| const char *data; |
| struct lysp_node_list list = {}; |
| struct lysp_node_leaflist llist = {}; |
| struct lysp_refine refine = {}; |
| |
| data = "<refine xmlns=\"urn:ietf:params:xml:ns:yang:yin:1\"> <max-elements value=\"unbounded\">"EXT_SUBELEM"</max-elements> </refine>"; |
| assert_int_equal(test_element_helper(st, data, &refine, NULL, NULL), LY_SUCCESS); |
| assert_int_equal(refine.max, 0); |
| assert_true(refine.flags & LYS_SET_MAX); |
| assert_string_equal(refine.exts[0].name, "urn:example:extensions:c-define"); |
| assert_int_equal(refine.exts[0].insubstmt_index, 0); |
| assert_int_equal(refine.exts[0].insubstmt, LYEXT_SUBSTMT_MAX); |
| FREE_ARRAY(st->ctx, refine.exts, lysp_ext_instance_free); |
| |
| data = "<list xmlns=\"urn:ietf:params:xml:ns:yang:yin:1\"> <max-elements value=\"5\">"EXT_SUBELEM"</max-elements> </list>"; |
| assert_int_equal(test_element_helper(st, data, &list, NULL, NULL), LY_SUCCESS); |
| assert_int_equal(list.max, 5); |
| assert_true(list.flags & LYS_SET_MAX); |
| assert_string_equal(list.exts[0].name, "urn:example:extensions:c-define"); |
| assert_int_equal(list.exts[0].insubstmt_index, 0); |
| assert_int_equal(list.exts[0].insubstmt, LYEXT_SUBSTMT_MAX); |
| FREE_ARRAY(st->ctx, list.exts, lysp_ext_instance_free); |
| |
| data = "<leaf-list xmlns=\"urn:ietf:params:xml:ns:yang:yin:1\"> <max-elements value=\"85\">"EXT_SUBELEM"</max-elements> </leaf-list>"; |
| assert_int_equal(test_element_helper(st, data, &llist, NULL, NULL), LY_SUCCESS); |
| assert_int_equal(llist.max, 85); |
| assert_true(llist.flags & LYS_SET_MAX); |
| assert_string_equal(llist.exts[0].name, "urn:example:extensions:c-define"); |
| assert_int_equal(llist.exts[0].insubstmt_index, 0); |
| assert_int_equal(llist.exts[0].insubstmt, LYEXT_SUBSTMT_MAX); |
| FREE_ARRAY(st->ctx, llist.exts, lysp_ext_instance_free); |
| |
| data = "<refine xmlns=\"urn:ietf:params:xml:ns:yang:yin:1\"> <max-elements value=\"10\"/> </refine>"; |
| assert_int_equal(test_element_helper(st, data, &refine, NULL, NULL), LY_SUCCESS); |
| assert_int_equal(refine.max, 10); |
| assert_true(refine.flags & LYS_SET_MAX); |
| |
| data = "<list xmlns=\"urn:ietf:params:xml:ns:yang:yin:1\"> <max-elements value=\"0\"/> </list>"; |
| assert_int_equal(test_element_helper(st, data, &list, NULL, NULL), LY_EVALID); |
| logbuf_assert("Invalid value \"0\" of \"value\" attribute in \"max-elements\" element. Line number 1."); |
| |
| data = "<list xmlns=\"urn:ietf:params:xml:ns:yang:yin:1\"> <max-elements value=\"-10\"/> </list>"; |
| assert_int_equal(test_element_helper(st, data, &list, NULL, NULL), LY_EVALID); |
| logbuf_assert("Invalid value \"-10\" of \"value\" attribute in \"max-elements\" element. Line number 1."); |
| |
| data = "<list xmlns=\"urn:ietf:params:xml:ns:yang:yin:1\"> <max-elements value=\"k\"/> </list>"; |
| assert_int_equal(test_element_helper(st, data, &list, NULL, NULL), LY_EVALID); |
| logbuf_assert("Invalid value \"k\" of \"value\" attribute in \"max-elements\" element. Line number 1."); |
| |
| data = "<list xmlns=\"urn:ietf:params:xml:ns:yang:yin:1\"> <max-elements value=\"u12\"/> </list>"; |
| assert_int_equal(test_element_helper(st, data, &list, NULL, NULL), LY_EVALID); |
| logbuf_assert("Invalid value \"u12\" of \"value\" attribute in \"max-elements\" element. Line number 1."); |
| |
| st->finished_correctly = true; |
| } |
| |
| static void |
| test_min_elems_elem(void **state) |
| { |
| struct test_parser_yin_state *st = *state; |
| const char *data; |
| struct lysp_node_list list = {}; |
| struct lysp_node_leaflist llist = {}; |
| struct lysp_refine refine = {}; |
| |
| data = "<refine xmlns=\"urn:ietf:params:xml:ns:yang:yin:1\"> <min-elements value=\"0\">"EXT_SUBELEM"</min-elements> </refine>"; |
| assert_int_equal(test_element_helper(st, data, &refine, NULL, NULL), LY_SUCCESS); |
| assert_int_equal(refine.min, 0); |
| assert_true(refine.flags & LYS_SET_MIN); |
| assert_string_equal(refine.exts[0].name, "urn:example:extensions:c-define"); |
| assert_int_equal(refine.exts[0].insubstmt_index, 0); |
| assert_int_equal(refine.exts[0].insubstmt, LYEXT_SUBSTMT_MIN); |
| FREE_ARRAY(st->ctx, refine.exts, lysp_ext_instance_free); |
| |
| data = "<list xmlns=\"urn:ietf:params:xml:ns:yang:yin:1\"> <min-elements value=\"41\">"EXT_SUBELEM"</min-elements> </list>"; |
| assert_int_equal(test_element_helper(st, data, &list, NULL, NULL), LY_SUCCESS); |
| assert_int_equal(list.min, 41); |
| assert_true(list.flags & LYS_SET_MIN); |
| assert_string_equal(list.exts[0].name, "urn:example:extensions:c-define"); |
| assert_int_equal(list.exts[0].insubstmt_index, 0); |
| assert_int_equal(list.exts[0].insubstmt, LYEXT_SUBSTMT_MIN); |
| FREE_ARRAY(st->ctx, list.exts, lysp_ext_instance_free); |
| |
| data = "<leaf-list xmlns=\"urn:ietf:params:xml:ns:yang:yin:1\"> <min-elements value=\"50\">"EXT_SUBELEM"</min-elements> </leaf-list>"; |
| assert_int_equal(test_element_helper(st, data, &llist, NULL, NULL), LY_SUCCESS); |
| assert_int_equal(llist.min, 50); |
| assert_true(llist.flags & LYS_SET_MIN); |
| assert_string_equal(llist.exts[0].name, "urn:example:extensions:c-define"); |
| assert_int_equal(llist.exts[0].insubstmt_index, 0); |
| assert_int_equal(llist.exts[0].insubstmt, LYEXT_SUBSTMT_MIN); |
| FREE_ARRAY(st->ctx, llist.exts, lysp_ext_instance_free); |
| |
| data = "<leaf-list xmlns=\"urn:ietf:params:xml:ns:yang:yin:1\"> <min-elements value=\"-5\"/> </leaf-list>"; |
| assert_int_equal(test_element_helper(st, data, &llist, NULL, NULL), LY_EVALID); |
| logbuf_assert("Value \"-5\" of \"value\" attribute in \"min-elements\" element is out of bounds. Line number 1."); |
| |
| data = "<leaf-list xmlns=\"urn:ietf:params:xml:ns:yang:yin:1\"> <min-elements value=\"99999999999999999\"/> </leaf-list>"; |
| assert_int_equal(test_element_helper(st, data, &llist, NULL, NULL), LY_EVALID); |
| logbuf_assert("Value \"99999999999999999\" of \"value\" attribute in \"min-elements\" element is out of bounds. Line number 1."); |
| |
| data = "<leaf-list xmlns=\"urn:ietf:params:xml:ns:yang:yin:1\"> <min-elements value=\"5k\"/> </leaf-list>"; |
| assert_int_equal(test_element_helper(st, data, &llist, NULL, NULL), LY_EVALID); |
| logbuf_assert("Invalid value \"5k\" of \"value\" attribute in \"min-elements\" element. Line number 1."); |
| |
| data = "<leaf-list xmlns=\"urn:ietf:params:xml:ns:yang:yin:1\"> <min-elements value=\"05\"/> </leaf-list>"; |
| assert_int_equal(test_element_helper(st, data, &llist, NULL, NULL), LY_EVALID); |
| logbuf_assert("Invalid value \"05\" of \"value\" attribute in \"min-elements\" element. Line number 1."); |
| |
| st->finished_correctly = true; |
| } |
| |
| static void |
| test_ordby_elem(void **state) |
| { |
| struct test_parser_yin_state *st = *state; |
| const char *data; |
| uint16_t flags = 0; |
| struct lysp_ext_instance *exts = NULL; |
| |
| data = ELEMENT_WRAPPER_START "<ordered-by value=\"system\">"EXT_SUBELEM"</ordered-by>" ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &flags, NULL, &exts), LY_SUCCESS); |
| assert_true(flags & LYS_ORDBY_SYSTEM); |
| assert_string_equal(exts[0].name, "urn:example:extensions:c-define"); |
| assert_int_equal(exts[0].insubstmt_index, 0); |
| assert_int_equal(exts[0].insubstmt, LYEXT_SUBSTMT_ORDEREDBY); |
| FREE_ARRAY(st->ctx, exts, lysp_ext_instance_free); |
| |
| data = ELEMENT_WRAPPER_START "<ordered-by value=\"user\"/>" ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &flags, NULL, NULL), LY_SUCCESS); |
| assert_true(flags & LYS_ORDBY_USER); |
| |
| data = ELEMENT_WRAPPER_START "<ordered-by value=\"inv\"/>" ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &flags, NULL, NULL), LY_EVALID); |
| logbuf_assert("Invalid value \"inv\" of \"value\" attribute in \"ordered-by\" element. Valid values are \"system\" and \"user\". Line number 1."); |
| |
| st->finished_correctly = true; |
| } |
| |
| static void |
| test_any_elem(void **state) |
| { |
| struct test_parser_yin_state *st = *state; |
| const char *data; |
| struct lysp_node *siblings = NULL; |
| struct tree_node_meta node_meta = {.parent = NULL, .nodes = &siblings}; |
| struct lysp_node_anydata *parsed = NULL; |
| |
| /* anyxml max subelems */ |
| data = ELEMENT_WRAPPER_START |
| "<anyxml name=\"any-name\">" |
| "<config value=\"true\" />" |
| "<description><text>desc</text></description>" |
| "<if-feature name=\"feature\" />" |
| "<mandatory value=\"true\" />" |
| "<must condition=\"must-cond\" />" |
| "<reference><text>ref</text></reference>" |
| "<status value=\"deprecated\"/>" |
| "<when condition=\"when-cond\"/>" |
| EXT_SUBELEM |
| "</anyxml>" |
| ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &node_meta, NULL, NULL), LY_SUCCESS); |
| parsed = (struct lysp_node_anydata *)siblings; |
| assert_null(parsed->parent); |
| assert_int_equal(parsed->nodetype, LYS_ANYXML); |
| assert_true(parsed->flags & LYS_CONFIG_W); |
| assert_true(parsed->flags & LYS_MAND_TRUE); |
| assert_true(parsed->flags & LYS_STATUS_DEPRC); |
| assert_null(parsed->next); |
| assert_string_equal(parsed->name, "any-name"); |
| assert_string_equal(parsed->dsc, "desc"); |
| assert_string_equal(parsed->ref, "ref"); |
| assert_string_equal(parsed->when->cond, "when-cond"); |
| assert_string_equal(parsed->iffeatures[0].str, "feature"); |
| assert_string_equal(parsed->exts[0].name, "urn:example:extensions:c-define"); |
| assert_int_equal(parsed->exts[0].insubstmt_index, 0); |
| assert_int_equal(parsed->exts[0].insubstmt, LYEXT_SUBSTMT_SELF); |
| lysp_node_free(st->ctx, siblings); |
| siblings = NULL; |
| |
| /* anydata max subelems */ |
| data = ELEMENT_WRAPPER_START |
| "<anydata name=\"any-name\">" |
| "<config value=\"true\" />" |
| "<description><text>desc</text></description>" |
| "<if-feature name=\"feature\" />" |
| "<mandatory value=\"true\" />" |
| "<must condition=\"must-cond\" />" |
| "<reference><text>ref</text></reference>" |
| "<status value=\"deprecated\"/>" |
| "<when condition=\"when-cond\"/>" |
| EXT_SUBELEM |
| "</anydata>" |
| ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &node_meta, NULL, NULL), LY_SUCCESS); |
| parsed = (struct lysp_node_anydata *)siblings; |
| assert_null(parsed->parent); |
| assert_int_equal(parsed->nodetype, LYS_ANYDATA); |
| assert_true(parsed->flags & LYS_CONFIG_W); |
| assert_true(parsed->flags & LYS_MAND_TRUE); |
| assert_true(parsed->flags & LYS_STATUS_DEPRC); |
| assert_null(parsed->next); |
| assert_string_equal(parsed->name, "any-name"); |
| assert_string_equal(parsed->dsc, "desc"); |
| assert_string_equal(parsed->ref, "ref"); |
| assert_string_equal(parsed->when->cond, "when-cond"); |
| assert_string_equal(parsed->iffeatures[0].str, "feature"); |
| assert_string_equal(parsed->exts[0].name, "urn:example:extensions:c-define"); |
| assert_int_equal(parsed->exts[0].insubstmt_index, 0); |
| assert_int_equal(parsed->exts[0].insubstmt, LYEXT_SUBSTMT_SELF); |
| lysp_node_free(st->ctx, siblings); |
| siblings = NULL; |
| |
| /* min subelems */ |
| node_meta.parent = (void *)0x10; |
| data = ELEMENT_WRAPPER_START "<anydata name=\"any-name\"> </anydata>" ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &node_meta, NULL, NULL), LY_SUCCESS); |
| parsed = (struct lysp_node_anydata *)siblings; |
| assert_ptr_equal(parsed->parent, node_meta.parent); |
| assert_int_equal(parsed->nodetype, LYS_ANYDATA); |
| assert_null(parsed->next); |
| assert_null(parsed->exts); |
| lysp_node_free(st->ctx, siblings); |
| |
| st->finished_correctly = true; |
| } |
| |
| static void |
| test_leaf_elem(void **state) |
| { |
| struct test_parser_yin_state *st = *state; |
| const char *data; |
| struct lysp_node *siblings = NULL; |
| struct tree_node_meta node_meta = {.parent = NULL, .nodes = &siblings}; |
| struct lysp_node_leaf *parsed = NULL; |
| |
| /* max elements */ |
| data = ELEMENT_WRAPPER_START |
| "<leaf name=\"leaf\">" |
| "<config value=\"true\" />" |
| "<default value=\"def-val\"/>" |
| "<description><text>desc</text></description>" |
| "<if-feature name=\"feature\" />" |
| "<mandatory value=\"true\" />" |
| "<must condition=\"must-cond\" />" |
| "<reference><text>ref</text></reference>" |
| "<status value=\"deprecated\"/>" |
| "<type name=\"type\"/>" |
| "<units name=\"uni\"/>" |
| "<when condition=\"when-cond\"/>" |
| EXT_SUBELEM |
| "</leaf>" |
| ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &node_meta, NULL, NULL), LY_SUCCESS); |
| parsed = (struct lysp_node_leaf *)siblings; |
| assert_null(parsed->parent); |
| assert_int_equal(parsed->nodetype, LYS_LEAF); |
| assert_true(parsed->flags & LYS_CONFIG_W); |
| assert_true(parsed->flags & LYS_MAND_TRUE); |
| assert_true(parsed->flags & LYS_STATUS_DEPRC); |
| assert_null(parsed->next); |
| assert_string_equal(parsed->name, "leaf"); |
| assert_string_equal(parsed->dsc, "desc"); |
| assert_string_equal(parsed->ref, "ref"); |
| assert_string_equal(parsed->when->cond, "when-cond"); |
| assert_string_equal(parsed->iffeatures[0].str, "feature"); |
| assert_string_equal(parsed->exts[0].name, "urn:example:extensions:c-define"); |
| assert_int_equal(parsed->exts[0].insubstmt_index, 0); |
| assert_int_equal(parsed->exts[0].insubstmt, LYEXT_SUBSTMT_SELF); |
| assert_string_equal(parsed->musts->arg.str, "must-cond"); |
| assert_string_equal(parsed->type.name, "type"); |
| assert_string_equal(parsed->units, "uni"); |
| assert_string_equal(parsed->dflt.str, "def-val"); |
| lysp_node_free(st->ctx, siblings); |
| siblings = NULL; |
| |
| /* min elements */ |
| data = ELEMENT_WRAPPER_START "<leaf name=\"leaf\"> <type name=\"type\"/> </leaf>" ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &node_meta, NULL, NULL), LY_SUCCESS); |
| parsed = (struct lysp_node_leaf *)siblings; |
| assert_string_equal(parsed->name, "leaf"); |
| assert_string_equal(parsed->type.name, "type"); |
| lysp_node_free(st->ctx, siblings); |
| siblings = NULL; |
| |
| st->finished_correctly = true; |
| } |
| |
| static void |
| test_leaf_list_elem(void **state) |
| { |
| struct test_parser_yin_state *st = *state; |
| const char *data; |
| struct lysp_node *siblings = NULL; |
| struct tree_node_meta node_meta = {.parent = NULL, .nodes = &siblings}; |
| struct lysp_node_leaflist *parsed = NULL; |
| |
| data = ELEMENT_WRAPPER_START |
| "<leaf-list name=\"llist\">" |
| "<config value=\"true\" />" |
| "<default value=\"def-val0\"/>" |
| "<default value=\"def-val1\"/>" |
| "<description><text>desc</text></description>" |
| "<if-feature name=\"feature\"/>" |
| "<max-elements value=\"5\"/>" |
| "<must condition=\"must-cond\"/>" |
| "<ordered-by value=\"user\" />" |
| "<reference><text>ref</text></reference>" |
| "<status value=\"current\"/>" |
| "<type name=\"type\"/>" |
| "<units name=\"uni\"/>" |
| "<when condition=\"when-cond\"/>" |
| EXT_SUBELEM |
| "</leaf-list>" |
| ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &node_meta, NULL, NULL), LY_SUCCESS); |
| parsed = (struct lysp_node_leaflist *)siblings; |
| assert_string_equal(parsed->dflts[0].str, "def-val0"); |
| assert_string_equal(parsed->dflts[1].str, "def-val1"); |
| assert_string_equal(parsed->dsc, "desc"); |
| assert_string_equal(parsed->iffeatures[0].str, "feature"); |
| assert_int_equal(parsed->max, 5); |
| assert_string_equal(parsed->musts->arg.str, "must-cond"); |
| assert_string_equal(parsed->name, "llist"); |
| assert_null(parsed->next); |
| assert_int_equal(parsed->nodetype, LYS_LEAFLIST); |
| assert_null(parsed->parent); |
| assert_string_equal(parsed->ref, "ref"); |
| assert_string_equal(parsed->type.name, "type"); |
| assert_string_equal(parsed->units, "uni"); |
| assert_string_equal(parsed->when->cond, "when-cond"); |
| assert_true(parsed->flags & LYS_CONFIG_W); |
| assert_true(parsed->flags & LYS_ORDBY_USER); |
| assert_true(parsed->flags & LYS_STATUS_CURR); |
| assert_string_equal(parsed->exts[0].name, "urn:example:extensions:c-define"); |
| assert_int_equal(parsed->exts[0].insubstmt_index, 0); |
| assert_int_equal(parsed->exts[0].insubstmt, LYEXT_SUBSTMT_SELF); |
| lysp_node_free(st->ctx, siblings); |
| siblings = NULL; |
| |
| data = ELEMENT_WRAPPER_START |
| "<leaf-list name=\"llist\">" |
| "<config value=\"true\" />" |
| "<description><text>desc</text></description>" |
| "<if-feature name=\"feature\"/>" |
| "<min-elements value=\"5\"/>" |
| "<must condition=\"must-cond\"/>" |
| "<ordered-by value=\"user\" />" |
| "<reference><text>ref</text></reference>" |
| "<status value=\"current\"/>" |
| "<type name=\"type\"/>" |
| "<units name=\"uni\"/>" |
| "<when condition=\"when-cond\"/>" |
| EXT_SUBELEM |
| "</leaf-list>" |
| ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &node_meta, NULL, NULL), LY_SUCCESS); |
| parsed = (struct lysp_node_leaflist *)siblings; |
| assert_string_equal(parsed->dsc, "desc"); |
| assert_string_equal(parsed->iffeatures[0].str, "feature"); |
| assert_int_equal(parsed->min, 5); |
| assert_string_equal(parsed->musts->arg.str, "must-cond"); |
| assert_string_equal(parsed->name, "llist"); |
| assert_null(parsed->next); |
| assert_int_equal(parsed->nodetype, LYS_LEAFLIST); |
| assert_null(parsed->parent); |
| assert_string_equal(parsed->ref, "ref"); |
| assert_string_equal(parsed->type.name, "type"); |
| assert_string_equal(parsed->units, "uni"); |
| assert_string_equal(parsed->when->cond, "when-cond"); |
| assert_true(parsed->flags & LYS_CONFIG_W); |
| assert_true(parsed->flags & LYS_ORDBY_USER); |
| assert_true(parsed->flags & LYS_STATUS_CURR); |
| assert_string_equal(parsed->exts[0].name, "urn:example:extensions:c-define"); |
| assert_int_equal(parsed->exts[0].insubstmt_index, 0); |
| assert_int_equal(parsed->exts[0].insubstmt, LYEXT_SUBSTMT_SELF); |
| lysp_node_free(st->ctx, siblings); |
| siblings = NULL; |
| |
| data = ELEMENT_WRAPPER_START |
| "<leaf-list name=\"llist\">" |
| "<config value=\"true\" />" |
| "<description><text>desc</text></description>" |
| "<if-feature name=\"feature\"/>" |
| "<max-elements value=\"15\"/>" |
| "<min-elements value=\"5\"/>" |
| "<must condition=\"must-cond\"/>" |
| "<ordered-by value=\"user\" />" |
| "<reference><text>ref</text></reference>" |
| "<status value=\"current\"/>" |
| "<type name=\"type\"/>" |
| "<units name=\"uni\"/>" |
| "<when condition=\"when-cond\"/>" |
| "</leaf-list>" |
| ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &node_meta, NULL, NULL), LY_SUCCESS); |
| parsed = (struct lysp_node_leaflist *)siblings; |
| assert_string_equal(parsed->dsc, "desc"); |
| assert_string_equal(parsed->iffeatures[0].str, "feature"); |
| assert_int_equal(parsed->min, 5); |
| assert_int_equal(parsed->max, 15); |
| assert_string_equal(parsed->musts->arg.str, "must-cond"); |
| assert_string_equal(parsed->name, "llist"); |
| assert_null(parsed->next); |
| assert_int_equal(parsed->nodetype, LYS_LEAFLIST); |
| assert_null(parsed->parent); |
| assert_string_equal(parsed->ref, "ref"); |
| assert_string_equal(parsed->type.name, "type"); |
| assert_string_equal(parsed->units, "uni"); |
| assert_string_equal(parsed->when->cond, "when-cond"); |
| assert_true(parsed->flags & LYS_CONFIG_W); |
| assert_true(parsed->flags & LYS_ORDBY_USER); |
| assert_true(parsed->flags & LYS_STATUS_CURR); |
| lysp_node_free(st->ctx, siblings); |
| siblings = NULL; |
| |
| data = ELEMENT_WRAPPER_START |
| "<leaf-list name=\"llist\">" |
| "<type name=\"type\"/>" |
| "</leaf-list>" |
| ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &node_meta, NULL, NULL), LY_SUCCESS); |
| parsed = (struct lysp_node_leaflist *)siblings; |
| assert_string_equal(parsed->name, "llist"); |
| assert_string_equal(parsed->type.name, "type"); |
| lysp_node_free(st->ctx, siblings); |
| siblings = NULL; |
| |
| /* invalid combinations */ |
| data = ELEMENT_WRAPPER_START |
| "<leaf-list name=\"llist\">" |
| "<max-elements value=\"5\"/>" |
| "<min-elements value=\"15\"/>" |
| "<type name=\"type\"/>" |
| "</leaf-list>" |
| ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &node_meta, NULL, NULL), LY_EVALID); |
| logbuf_assert("Invalid combination of min-elements and max-elements: min value 15 is bigger than the max value 5. Line number 1."); |
| lysp_node_free(st->ctx, siblings); |
| siblings = NULL; |
| |
| data = ELEMENT_WRAPPER_START |
| "<leaf-list name=\"llist\">" |
| "<default value=\"def-val1\"/>" |
| "<min-elements value=\"15\"/>" |
| "<type name=\"type\"/>" |
| "</leaf-list>" |
| ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &node_meta, NULL, NULL), LY_EVALID); |
| logbuf_assert("Invalid combination of sub-elemnts \"min-elements\" and \"default\" in \"leaf-list\" element. Line number 1."); |
| lysp_node_free(st->ctx, siblings); |
| siblings = NULL; |
| |
| data = ELEMENT_WRAPPER_START |
| "<leaf-list name=\"llist\">" |
| "</leaf-list>" |
| ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &node_meta, NULL, NULL), LY_EVALID); |
| logbuf_assert("Missing mandatory sub-element \"type\" of \"leaf-list\" element. Line number 1."); |
| lysp_node_free(st->ctx, siblings); |
| siblings = NULL; |
| |
| st->finished_correctly = true; |
| } |
| |
| static void |
| test_presence_elem(void **state) |
| { |
| struct test_parser_yin_state *st = *state; |
| const char *data; |
| const char *val; |
| struct lysp_ext_instance *exts = NULL; |
| |
| data = ELEMENT_WRAPPER_START "<presence value=\"presence-val\">"EXT_SUBELEM"</presence>" ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &val, NULL, &exts), LY_SUCCESS); |
| assert_string_equal(val, "presence-val"); |
| assert_string_equal(exts[0].name, "urn:example:extensions:c-define"); |
| assert_int_equal(exts[0].insubstmt_index, 0); |
| assert_int_equal(exts[0].insubstmt, LYEXT_SUBSTMT_PRESENCE); |
| FREE_ARRAY(st->ctx, exts, lysp_ext_instance_free); |
| FREE_STRING(st->ctx, val); |
| |
| data = ELEMENT_WRAPPER_START "<presence value=\"presence-val\"/>" ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &val, NULL, NULL), LY_SUCCESS); |
| assert_string_equal(val, "presence-val"); |
| FREE_STRING(st->ctx, val); |
| |
| data = ELEMENT_WRAPPER_START "<presence/>" ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &val, NULL, NULL), LY_EVALID); |
| logbuf_assert("Missing mandatory attribute value of presence element. Line number 1."); |
| |
| st->finished_correctly = true; |
| } |
| |
| static void |
| test_key_elem(void **state) |
| { |
| struct test_parser_yin_state *st = *state; |
| const char *data; |
| const char *val; |
| struct lysp_ext_instance *exts = NULL; |
| |
| data = ELEMENT_WRAPPER_START "<key value=\"key-value\">"EXT_SUBELEM"</key>" ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &val, NULL, &exts), LY_SUCCESS); |
| assert_string_equal(val, "key-value"); |
| assert_string_equal(exts[0].name, "urn:example:extensions:c-define"); |
| assert_int_equal(exts[0].insubstmt_index, 0); |
| assert_int_equal(exts[0].insubstmt, LYEXT_SUBSTMT_KEY); |
| FREE_ARRAY(st->ctx, exts, lysp_ext_instance_free); |
| FREE_STRING(st->ctx, val); |
| |
| data = ELEMENT_WRAPPER_START "<key value=\"key-value\"/>" ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &val, NULL, NULL), LY_SUCCESS); |
| assert_string_equal(val, "key-value"); |
| FREE_STRING(st->ctx, val); |
| |
| data = ELEMENT_WRAPPER_START "<key/>" ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &val, NULL, NULL), LY_EVALID); |
| logbuf_assert("Missing mandatory attribute value of key element. Line number 1."); |
| |
| st->finished_correctly = true; |
| } |
| |
| static void |
| test_typedef_elem(void **state) |
| { |
| struct test_parser_yin_state *st = *state; |
| const char *data; |
| struct lysp_tpdf *tpdfs = NULL; |
| struct tree_node_meta typdef_meta = {NULL, (struct lysp_node **)&tpdfs}; |
| |
| data = ELEMENT_WRAPPER_START |
| "<typedef name=\"tpdf-name\">" |
| "<default value=\"def-val\"/>" |
| "<description><text>desc-text</text></description>" |
| "<reference><text>ref-text</text></reference>" |
| "<status value=\"current\"/>" |
| "<type name=\"type\"/>" |
| "<units name=\"uni\"/>" |
| EXT_SUBELEM |
| "</typedef>" |
| ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &typdef_meta, NULL, NULL), LY_SUCCESS); |
| assert_string_equal(tpdfs[0].dflt.str, "def-val"); |
| assert_string_equal(tpdfs[0].dsc, "desc-text"); |
| assert_string_equal(tpdfs[0].name, "tpdf-name"); |
| assert_string_equal(tpdfs[0].ref, "ref-text"); |
| assert_string_equal(tpdfs[0].type.name, "type"); |
| assert_string_equal(tpdfs[0].units, "uni"); |
| assert_true(tpdfs[0].flags & LYS_STATUS_CURR); |
| assert_string_equal(tpdfs[0].exts[0].name, "urn:example:extensions:c-define"); |
| assert_int_equal(tpdfs[0].exts[0].insubstmt_index, 0); |
| assert_int_equal(tpdfs[0].exts[0].insubstmt, LYEXT_SUBSTMT_SELF); |
| FREE_ARRAY(st->ctx, tpdfs, lysp_tpdf_free); |
| tpdfs = NULL; |
| |
| data = ELEMENT_WRAPPER_START |
| "<typedef name=\"tpdf-name\">" |
| "<type name=\"type\"/>" |
| "</typedef>" |
| ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &typdef_meta, NULL, NULL), LY_SUCCESS); |
| assert_string_equal(tpdfs[0].name, "tpdf-name"); |
| assert_string_equal(tpdfs[0].type.name, "type"); |
| FREE_ARRAY(st->ctx, tpdfs, lysp_tpdf_free); |
| tpdfs = NULL; |
| |
| st->finished_correctly = true; |
| } |
| |
| static void |
| test_refine_elem(void **state) |
| { |
| struct test_parser_yin_state *st = *state; |
| const char *data; |
| struct lysp_refine *refines = NULL; |
| |
| /* max subelems */ |
| data = ELEMENT_WRAPPER_START |
| "<refine target-node=\"target\">" |
| "<if-feature name=\"feature\" />" |
| "<must condition=\"cond\" />" |
| "<presence value=\"presence\" />" |
| "<default value=\"def\" />" |
| "<config value=\"true\" />" |
| "<mandatory value=\"true\" />" |
| "<min-elements value=\"10\" />" |
| "<max-elements value=\"20\" />" |
| "<description><text>desc</text></description>" |
| "<reference><text>ref</text></reference>" |
| EXT_SUBELEM |
| "</refine>" |
| ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &refines, NULL, NULL), LY_SUCCESS); |
| assert_string_equal(refines->nodeid, "target"); |
| assert_string_equal(refines->dflts[0].str, "def"); |
| assert_string_equal(refines->dsc, "desc"); |
| assert_true(refines->flags & LYS_CONFIG_W); |
| assert_true(refines->flags & LYS_MAND_TRUE); |
| assert_string_equal(refines->iffeatures[0].str, "feature"); |
| assert_int_equal(refines->max, 20); |
| assert_int_equal(refines->min, 10); |
| assert_string_equal(refines->musts->arg.str, "cond"); |
| assert_string_equal(refines->presence, "presence"); |
| assert_string_equal(refines->ref, "ref"); |
| assert_string_equal(refines->exts[0].name, "urn:example:extensions:c-define"); |
| assert_int_equal(refines->exts[0].insubstmt_index, 0); |
| assert_int_equal(refines->exts[0].insubstmt, LYEXT_SUBSTMT_SELF); |
| FREE_ARRAY(st->ctx, refines, lysp_refine_free); |
| refines = NULL; |
| |
| /* min subelems */ |
| data = ELEMENT_WRAPPER_START "<refine target-node=\"target\" />" ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &refines, NULL, NULL), LY_SUCCESS); |
| assert_string_equal(refines->nodeid, "target"); |
| FREE_ARRAY(st->ctx, refines, lysp_refine_free); |
| refines = NULL; |
| |
| st->finished_correctly = true; |
| } |
| |
| static void |
| test_uses_elem(void **state) |
| { |
| struct test_parser_yin_state *st = *state; |
| const char *data; |
| struct lysp_node *siblings = NULL; |
| struct tree_node_meta node_meta = {NULL, &siblings}; |
| struct lysp_node_uses *parsed = NULL; |
| |
| /* max subelems */ |
| data = ELEMENT_WRAPPER_START |
| "<uses name=\"uses-name\">" |
| "<when condition=\"cond\" />" |
| "<if-feature name=\"feature\" />" |
| "<status value=\"obsolete\" />" |
| "<description><text>desc</text></description>" |
| "<reference><text>ref</text></reference>" |
| "<refine target-node=\"target\"/>" |
| "<augment target-node=\"target\" />" |
| EXT_SUBELEM |
| "</uses>" |
| ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &node_meta, NULL, NULL), LY_SUCCESS); |
| parsed = (struct lysp_node_uses *)&siblings[0]; |
| assert_string_equal(parsed->name, "uses-name"); |
| assert_string_equal(parsed->dsc, "desc"); |
| assert_true(parsed->flags & LYS_STATUS_OBSLT); |
| assert_string_equal(parsed->iffeatures[0].str, "feature"); |
| assert_null(parsed->next); |
| assert_int_equal(parsed->nodetype, LYS_USES); |
| assert_null(parsed->parent); |
| assert_string_equal(parsed->ref, "ref"); |
| assert_string_equal(parsed->refines->nodeid, "target"); |
| assert_string_equal(parsed->when->cond, "cond"); |
| assert_string_equal(parsed->augments->nodeid, "target"); |
| assert_string_equal(parsed->exts[0].name, "urn:example:extensions:c-define"); |
| assert_int_equal(parsed->exts[0].insubstmt_index, 0); |
| assert_int_equal(parsed->exts[0].insubstmt, LYEXT_SUBSTMT_SELF); |
| lysp_node_free(st->ctx, siblings); |
| siblings = NULL; |
| |
| /* min subelems */ |
| data = ELEMENT_WRAPPER_START "<uses name=\"uses-name\"/>" ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &node_meta, NULL, NULL), LY_SUCCESS); |
| assert_string_equal(siblings[0].name, "uses-name"); |
| lysp_node_free(st->ctx, siblings); |
| siblings = NULL; |
| |
| st->finished_correctly = true; |
| } |
| |
| static void |
| test_revision_elem(void **state) |
| { |
| struct test_parser_yin_state *st = *state; |
| const char *data; |
| struct lysp_revision *revs = NULL; |
| |
| /* max subelems */ |
| data = ELEMENT_WRAPPER_START |
| "<revision date=\"2018-12-25\">" |
| "<description><text>desc</text></description>" |
| "<reference><text>ref</text></reference>" |
| EXT_SUBELEM |
| "</revision>" |
| ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &revs, NULL, NULL), LY_SUCCESS); |
| assert_string_equal(revs->date, "2018-12-25"); |
| assert_string_equal(revs->dsc, "desc"); |
| assert_string_equal(revs->ref, "ref"); |
| assert_string_equal(revs->exts[0].name, "urn:example:extensions:c-define"); |
| assert_int_equal(revs->exts[0].insubstmt_index, 0); |
| assert_int_equal(revs->exts[0].insubstmt, LYEXT_SUBSTMT_SELF); |
| FREE_ARRAY(st->ctx, revs, lysp_revision_free); |
| revs = NULL; |
| |
| /* min subelems */ |
| data = ELEMENT_WRAPPER_START "<revision date=\"2005-05-05\" />" ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &revs, NULL, NULL), LY_SUCCESS); |
| assert_string_equal(revs->date, "2005-05-05"); |
| FREE_ARRAY(st->ctx, revs, lysp_revision_free); |
| revs = NULL; |
| |
| /* invalid value */ |
| data = ELEMENT_WRAPPER_START "<revision date=\"05-05-2005\" />" ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &revs, NULL, NULL), LY_EVALID); |
| logbuf_assert("Invalid value \"05-05-2005\" of \"revision\". Line number 1."); |
| FREE_ARRAY(st->ctx, revs, lysp_revision_free); |
| revs = NULL; |
| |
| st->finished_correctly = true; |
| } |
| |
| static void |
| test_include_elem(void **state) |
| { |
| struct test_parser_yin_state *st = *state; |
| const char *data; |
| struct lysp_include *includes = NULL; |
| struct include_meta inc_meta = {"module-name", &includes}; |
| |
| /* max subelems */ |
| st->yin_ctx->parsed_mod->version = LYS_VERSION_1_1; |
| data = ELEMENT_WRAPPER_START |
| "<include module=\"mod\">" |
| "<description><text>desc</text></description>" |
| "<reference><text>ref</text></reference>" |
| "<revision-date date=\"1999-09-09\"/>" |
| EXT_SUBELEM |
| "</include>" |
| ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &inc_meta, NULL, NULL), LY_SUCCESS); |
| assert_string_equal(includes->name, "mod"); |
| assert_string_equal(includes->dsc, "desc"); |
| assert_string_equal(includes->ref, "ref"); |
| assert_string_equal(includes->rev, "1999-09-09"); |
| assert_string_equal(includes->exts[0].name, "urn:example:extensions:c-define"); |
| assert_int_equal(includes->exts[0].insubstmt_index, 0); |
| assert_int_equal(includes->exts[0].insubstmt, LYEXT_SUBSTMT_SELF); |
| FREE_ARRAY(st->ctx, includes, lysp_include_free); |
| includes = NULL; |
| |
| /* min subelems */ |
| data = ELEMENT_WRAPPER_START "<include module=\"mod\"/>" ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &inc_meta, NULL, NULL), LY_SUCCESS); |
| assert_string_equal(includes->name, "mod"); |
| FREE_ARRAY(st->ctx, includes, lysp_include_free); |
| includes = NULL; |
| |
| /* invalid combinations */ |
| st->yin_ctx->parsed_mod->version = LYS_VERSION_1_0; |
| data = ELEMENT_WRAPPER_START |
| "<include module=\"mod\">" |
| "<description><text>desc</text></description>" |
| "<revision-date date=\"1999-09-09\"/>" |
| "</include>" |
| ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &inc_meta, NULL, NULL), LY_EVALID); |
| logbuf_assert("Invalid sub-elemnt \"description\" of \"include\" element - this sub-element is allowed only in modules with version 1.1 or newer. Line number 1."); |
| FREE_ARRAY(st->ctx, includes, lysp_include_free); |
| includes = NULL; |
| |
| st->yin_ctx->parsed_mod->version = LYS_VERSION_1_0; |
| data = ELEMENT_WRAPPER_START |
| "<include module=\"mod\">" |
| "<reference><text>ref</text></reference>" |
| "<revision-date date=\"1999-09-09\"/>" |
| "</include>" |
| ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &inc_meta, NULL, NULL), LY_EVALID); |
| logbuf_assert("Invalid sub-elemnt \"reference\" of \"include\" element - this sub-element is allowed only in modules with version 1.1 or newer. Line number 1."); |
| FREE_ARRAY(st->ctx, includes, lysp_include_free); |
| includes = NULL; |
| |
| st->finished_correctly = true; |
| } |
| |
| static void |
| test_list_elem(void **state) |
| { |
| struct test_parser_yin_state *st = *state; |
| const char *data; |
| struct lysp_node *siblings = NULL; |
| struct tree_node_meta node_meta = {NULL, &siblings}; |
| struct lysp_node_list *parsed = NULL; |
| |
| /* max subelems */ |
| data = ELEMENT_WRAPPER_START |
| "<list name=\"list-name\">" |
| "<when condition=\"when\"/>" |
| "<if-feature name=\"iff\"/>" |
| "<must condition=\"must-cond\"/>" |
| "<key value=\"key\"/>" |
| "<unique tag=\"utag\"/>" |
| "<config value=\"true\"/>" |
| "<min-elements value=\"10\"/>" |
| "<ordered-by value=\"user\"/>" |
| "<status value=\"deprecated\"/>" |
| "<description><text>desc</text></description>" |
| "<reference><text>ref</text></reference>" |
| "<anydata name=\"anyd\"/>" |
| "<anyxml name=\"anyx\"/>" |
| "<container name=\"cont\"/>" |
| "<choice name=\"choice\"/>" |
| "<action name=\"action\"/>" |
| "<grouping name=\"grp\"/>" |
| "<notification name=\"notf\"/>" |
| "<leaf name=\"leaf\"> <type name=\"type\"/> </leaf>" |
| "<leaf-list name=\"llist\"> <type name=\"type\"/> </leaf-list>" |
| "<list name=\"sub-list\"/>" |
| "<typedef name=\"tpdf\"> <type name=\"type\"/> </typedef>" |
| "<uses name=\"uses-name\"/>" |
| EXT_SUBELEM |
| "</list>" |
| ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &node_meta, NULL, NULL), LY_SUCCESS); |
| parsed = (struct lysp_node_list *)&siblings[0]; |
| assert_string_equal(parsed->dsc, "desc"); |
| assert_string_equal(parsed->child->name, "anyd"); |
| assert_int_equal(parsed->child->nodetype, LYS_ANYDATA); |
| assert_string_equal(parsed->child->next->name, "anyx"); |
| assert_int_equal(parsed->child->next->nodetype, LYS_ANYXML); |
| assert_string_equal(parsed->child->next->next->name, "cont"); |
| assert_int_equal(parsed->child->next->next->nodetype, LYS_CONTAINER); |
| assert_string_equal(parsed->child->next->next->next->name, "choice"); |
| assert_int_equal(parsed->child->next->next->next->nodetype, LYS_CHOICE); |
| assert_string_equal(parsed->child->next->next->next->next->name, "leaf"); |
| assert_int_equal(parsed->child->next->next->next->next->nodetype, LYS_LEAF); |
| assert_string_equal(parsed->child->next->next->next->next->next->name, "llist"); |
| assert_int_equal(parsed->child->next->next->next->next->next->nodetype, LYS_LEAFLIST); |
| assert_string_equal(parsed->child->next->next->next->next->next->next->name, "sub-list"); |
| assert_int_equal(parsed->child->next->next->next->next->next->next->nodetype, LYS_LIST); |
| assert_string_equal(parsed->child->next->next->next->next->next->next->next->name, "uses-name"); |
| assert_int_equal(parsed->child->next->next->next->next->next->next->next->nodetype, LYS_USES); |
| assert_null(parsed->child->next->next->next->next->next->next->next->next); |
| assert_string_equal(parsed->groupings->name, "grp"); |
| assert_string_equal(parsed->actions->name, "action"); |
| assert_int_equal(parsed->groupings->nodetype, LYS_GROUPING); |
| assert_string_equal(parsed->notifs->name, "notf"); |
| assert_true(parsed->flags & LYS_ORDBY_USER); |
| assert_true(parsed->flags & LYS_STATUS_DEPRC); |
| assert_true(parsed->flags & LYS_CONFIG_W); |
| assert_string_equal(parsed->iffeatures[0].str, "iff"); |
| assert_string_equal(parsed->key, "key"); |
| assert_int_equal(parsed->min, 10); |
| assert_string_equal(parsed->musts->arg.str, "must-cond"); |
| assert_string_equal(parsed->name, "list-name"); |
| assert_null(parsed->next); |
| assert_int_equal(parsed->nodetype, LYS_LIST); |
| assert_null(parsed->parent); |
| assert_string_equal(parsed->ref, "ref"); |
| assert_string_equal(parsed->typedefs->name, "tpdf"); |
| assert_string_equal(parsed->uniques->str, "utag"); |
| assert_string_equal(parsed->when->cond, "when"); |
| assert_string_equal(parsed->exts[0].name, "urn:example:extensions:c-define"); |
| assert_int_equal(parsed->exts[0].insubstmt_index, 0); |
| assert_int_equal(parsed->exts[0].insubstmt, LYEXT_SUBSTMT_SELF); |
| lysp_node_free(st->ctx, siblings); |
| ly_set_erase(&st->yin_ctx->tpdfs_nodes, NULL); |
| siblings = NULL; |
| |
| /* min subelems */ |
| data = ELEMENT_WRAPPER_START "<list name=\"list-name\" />" ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &node_meta, NULL, NULL), LY_SUCCESS); |
| parsed = (struct lysp_node_list *)&siblings[0]; |
| assert_string_equal(parsed->name, "list-name"); |
| lysp_node_free(st->ctx, siblings); |
| siblings = NULL; |
| |
| st->finished_correctly = true; |
| } |
| |
| static void |
| test_notification_elem(void **state) |
| { |
| struct test_parser_yin_state *st = *state; |
| const char *data; |
| struct lysp_notif *notifs = NULL; |
| struct tree_node_meta notif_meta = {NULL, (struct lysp_node **)¬ifs}; |
| |
| /* max subelems */ |
| st->yin_ctx->parsed_mod->version = LYS_VERSION_1_1; |
| data = ELEMENT_WRAPPER_START |
| "<notification name=\"notif-name\">" |
| "<anydata name=\"anyd\"/>" |
| "<anyxml name=\"anyx\"/>" |
| "<description><text>desc</text></description>" |
| "<if-feature name=\"iff\"/>" |
| "<leaf name=\"leaf\"> <type name=\"type\"/> </leaf>" |
| "<leaf-list name=\"llist\"> <type name=\"type\"/> </leaf-list>" |
| "<list name=\"sub-list\"/>" |
| "<must condition=\"cond\"/>" |
| "<reference><text>ref</text></reference>" |
| "<status value=\"deprecated\"/>" |
| "<typedef name=\"tpdf\"> <type name=\"type\"/> </typedef>" |
| "<uses name=\"uses-name\"/>" |
| "<container name=\"cont\"/>" |
| "<choice name=\"choice\"/>" |
| "<grouping name=\"grp\"/>" |
| EXT_SUBELEM |
| "</notification>" |
| ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, ¬if_meta, NULL, NULL), LY_SUCCESS); |
| assert_string_equal(notifs->name, "notif-name"); |
| assert_string_equal(notifs->data->name, "anyd"); |
| assert_int_equal(notifs->data->nodetype, LYS_ANYDATA); |
| assert_string_equal(notifs->data->next->name, "anyx"); |
| assert_int_equal(notifs->data->next->nodetype, LYS_ANYXML); |
| assert_string_equal(notifs->data->next->next->name, "leaf"); |
| assert_int_equal(notifs->data->next->next->nodetype, LYS_LEAF); |
| assert_string_equal(notifs->data->next->next->next->name, "llist"); |
| assert_int_equal(notifs->data->next->next->next->nodetype, LYS_LEAFLIST); |
| assert_string_equal(notifs->data->next->next->next->next->name, "sub-list"); |
| assert_int_equal(notifs->data->next->next->next->next->nodetype, LYS_LIST); |
| assert_true(notifs->flags & LYS_STATUS_DEPRC); |
| assert_string_equal(notifs->groupings->name, "grp"); |
| assert_int_equal(notifs->groupings->nodetype, LYS_GROUPING); |
| assert_string_equal(notifs->data->next->next->next->next->next->name, "uses-name"); |
| assert_int_equal(notifs->data->next->next->next->next->next->nodetype, LYS_USES); |
| assert_string_equal(notifs->data->next->next->next->next->next->next->name, "cont"); |
| assert_int_equal(notifs->data->next->next->next->next->next->next->nodetype, LYS_CONTAINER); |
| assert_int_equal(notifs->data->next->next->next->next->next->next->next->nodetype, LYS_CHOICE); |
| assert_string_equal(notifs->data->next->next->next->next->next->next->next->name, "choice"); |
| assert_null(notifs->data->next->next->next->next->next->next->next->next); |
| assert_string_equal(notifs->iffeatures[0].str, "iff"); |
| assert_string_equal(notifs->musts->arg.str, "cond"); |
| assert_int_equal(notifs->nodetype, LYS_NOTIF); |
| assert_null(notifs->parent); |
| assert_string_equal(notifs->ref, "ref"); |
| assert_string_equal(notifs->typedefs->name, "tpdf"); |
| assert_string_equal(notifs->exts[0].name, "urn:example:extensions:c-define"); |
| assert_int_equal(notifs->exts[0].insubstmt_index, 0); |
| assert_int_equal(notifs->exts[0].insubstmt, LYEXT_SUBSTMT_SELF); |
| FREE_ARRAY(st->ctx, notifs, lysp_notif_free); |
| notifs = NULL; |
| |
| /* min subelems */ |
| data = ELEMENT_WRAPPER_START "<notification name=\"notif-name\" />" ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, ¬if_meta, NULL, NULL), LY_SUCCESS); |
| assert_string_equal(notifs->name, "notif-name"); |
| FREE_ARRAY(st->ctx, notifs, lysp_notif_free); |
| notifs = NULL; |
| |
| st->finished_correctly = true; |
| } |
| |
| static void |
| test_grouping_elem(void **state) |
| { |
| struct test_parser_yin_state *st = *state; |
| const char *data; |
| struct lysp_grp *grps = NULL; |
| struct tree_node_meta grp_meta = {NULL, (struct lysp_node **)&grps}; |
| |
| /* max subelems */ |
| data = ELEMENT_WRAPPER_START |
| "<grouping name=\"grp-name\">" |
| "<anydata name=\"anyd\"/>" |
| "<anyxml name=\"anyx\"/>" |
| "<description><text>desc</text></description>" |
| "<grouping name=\"sub-grp\"/>" |
| "<leaf name=\"leaf\"> <type name=\"type\"/> </leaf>" |
| "<leaf-list name=\"llist\"> <type name=\"type\"/> </leaf-list>" |
| "<list name=\"list\"/>" |
| "<notification name=\"notf\"/>" |
| "<reference><text>ref</text></reference>" |
| "<status value=\"current\"/>" |
| "<typedef name=\"tpdf\"> <type name=\"type\"/> </typedef>" |
| "<uses name=\"uses-name\"/>" |
| "<action name=\"act\"/>" |
| "<container name=\"cont\"/>" |
| "<choice name=\"choice\"/>" |
| EXT_SUBELEM |
| "</grouping>" |
| ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &grp_meta, NULL, NULL), LY_SUCCESS); |
| assert_string_equal(grps->name, "grp-name"); |
| assert_string_equal(grps->data->name, "anyd"); |
| assert_string_equal(grps->data->next->name, "anyx"); |
| assert_string_equal(grps->data->next->next->name, "leaf"); |
| assert_string_equal(grps->data->next->next->next->name, "llist"); |
| assert_string_equal(grps->data->next->next->next->next->name, "list"); |
| assert_string_equal(grps->dsc, "desc"); |
| assert_true(grps->flags & LYS_STATUS_CURR); |
| assert_string_equal(grps->groupings->name, "sub-grp"); |
| assert_int_equal(grps->nodetype, LYS_GROUPING); |
| assert_string_equal(grps->notifs->name, "notf"); |
| assert_null(grps->parent); |
| assert_string_equal(grps->ref, "ref"); |
| assert_string_equal(grps->typedefs->name, "tpdf"); |
| assert_string_equal(grps->actions->name, "act"); |
| assert_string_equal(grps->data->next->next->next->next->next->name, "uses-name"); |
| assert_int_equal(grps->data->next->next->next->next->next->nodetype, LYS_USES); |
| assert_string_equal(grps->data->next->next->next->next->next->next->name, "cont"); |
| assert_int_equal(grps->data->next->next->next->next->next->next->nodetype, LYS_CONTAINER); |
| assert_string_equal(grps->data->next->next->next->next->next->next->next->name, "choice"); |
| assert_int_equal(grps->data->next->next->next->next->next->next->next->nodetype, LYS_CHOICE); |
| assert_string_equal(grps->exts[0].name, "urn:example:extensions:c-define"); |
| assert_int_equal(grps->exts[0].insubstmt_index, 0); |
| assert_int_equal(grps->exts[0].insubstmt, LYEXT_SUBSTMT_SELF); |
| FREE_ARRAY(st->ctx, grps, lysp_grp_free); |
| grps = NULL; |
| |
| /* min subelems */ |
| data = ELEMENT_WRAPPER_START "<grouping name=\"grp-name\" />" ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &grp_meta, NULL, NULL), LY_SUCCESS); |
| assert_string_equal(grps->name, "grp-name"); |
| FREE_ARRAY(st->ctx, grps, lysp_grp_free); |
| grps = NULL; |
| |
| st->finished_correctly = true; |
| } |
| |
| static void |
| test_container_elem(void **state) |
| { |
| struct test_parser_yin_state *st = *state; |
| const char *data; |
| struct lysp_node *siblings = NULL; |
| struct tree_node_meta node_meta = {NULL, &siblings}; |
| struct lysp_node_container *parsed = NULL; |
| |
| /* max subelems */ |
| st->yin_ctx->parsed_mod->version = LYS_VERSION_1_1; |
| data = ELEMENT_WRAPPER_START |
| "<container name=\"cont-name\">" |
| "<anydata name=\"anyd\"/>" |
| "<anyxml name=\"anyx\"/>" |
| "<config value=\"true\"/>" |
| "<container name=\"subcont\"/>" |
| "<description><text>desc</text></description>" |
| "<grouping name=\"sub-grp\"/>" |
| "<if-feature name=\"iff\"/>" |
| "<leaf name=\"leaf\"> <type name=\"type\"/> </leaf>" |
| "<leaf-list name=\"llist\"> <type name=\"type\"/> </leaf-list>" |
| "<list name=\"list\"/>" |
| "<must condition=\"cond\"/>" |
| "<notification name=\"notf\"/>" |
| "<presence value=\"presence\"/>" |
| "<reference><text>ref</text></reference>" |
| "<status value=\"current\"/>" |
| "<typedef name=\"tpdf\"> <type name=\"type\"/> </typedef>" |
| "<uses name=\"uses-name\"/>" |
| "<when condition=\"when-cond\"/>" |
| "<action name=\"act\"/>" |
| "<choice name=\"choice\"/>" |
| EXT_SUBELEM |
| "</container>" |
| ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &node_meta, NULL, NULL), LY_SUCCESS); |
| parsed = (struct lysp_node_container *)siblings; |
| assert_string_equal(parsed->name, "cont-name"); |
| assert_null(parsed->parent); |
| assert_int_equal(parsed->nodetype, LYS_CONTAINER); |
| assert_true(parsed->flags & LYS_CONFIG_W); |
| assert_true(parsed->flags & LYS_STATUS_CURR); |
| assert_null(parsed->next); |
| assert_string_equal(parsed->dsc, "desc"); |
| assert_string_equal(parsed->ref, "ref"); |
| assert_string_equal(parsed->when->cond, "when-cond"); |
| assert_string_equal(parsed->iffeatures[0].str, "iff"); |
| assert_string_equal(parsed->musts->arg.str, "cond"); |
| assert_string_equal(parsed->presence, "presence"); |
| assert_string_equal(parsed->typedefs->name, "tpdf"); |
| assert_string_equal(parsed->groupings->name, "sub-grp"); |
| assert_string_equal(parsed->child->name, "anyd"); |
| assert_int_equal(parsed->child->nodetype, LYS_ANYDATA); |
| assert_string_equal(parsed->child->next->name, "anyx"); |
| assert_int_equal(parsed->child->next->nodetype, LYS_ANYXML); |
| assert_string_equal(parsed->child->next->next->name, "subcont"); |
| assert_int_equal(parsed->child->next->next->nodetype, LYS_CONTAINER); |
| assert_string_equal(parsed->child->next->next->next->name, "leaf"); |
| assert_int_equal(parsed->child->next->next->next->nodetype, LYS_LEAF); |
| assert_string_equal(parsed->child->next->next->next->next->name, "llist"); |
| assert_int_equal(parsed->child->next->next->next->next->nodetype, LYS_LEAFLIST); |
| assert_string_equal(parsed->child->next->next->next->next->next->name, "list"); |
| assert_int_equal(parsed->child->next->next->next->next->next->nodetype, LYS_LIST); |
| assert_string_equal(parsed->child->next->next->next->next->next->next->name, "uses-name"); |
| assert_int_equal(parsed->child->next->next->next->next->next->next->nodetype, LYS_USES); |
| assert_string_equal(parsed->child->next->next->next->next->next->next->next->name, "choice"); |
| assert_int_equal(parsed->child->next->next->next->next->next->next->next->nodetype, LYS_CHOICE); |
| assert_null(parsed->child->next->next->next->next->next->next->next->next); |
| assert_string_equal(parsed->notifs->name, "notf"); |
| assert_string_equal(parsed->actions->name, "act"); |
| assert_string_equal(parsed->exts[0].name, "urn:example:extensions:c-define"); |
| assert_int_equal(parsed->exts[0].insubstmt_index, 0); |
| assert_int_equal(parsed->exts[0].insubstmt, LYEXT_SUBSTMT_SELF); |
| lysp_node_free(st->ctx, siblings); |
| ly_set_erase(&st->yin_ctx->tpdfs_nodes, NULL); |
| siblings = NULL; |
| |
| /* min subelems */ |
| data = ELEMENT_WRAPPER_START "<container name=\"cont-name\" />" ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &node_meta, NULL, NULL), LY_SUCCESS); |
| parsed = (struct lysp_node_container *)siblings; |
| assert_string_equal(parsed->name, "cont-name"); |
| lysp_node_free(st->ctx, siblings); |
| siblings = NULL; |
| |
| st->finished_correctly = true; |
| } |
| |
| static void |
| test_case_elem(void **state) |
| { |
| struct test_parser_yin_state *st = *state; |
| const char *data; |
| struct lysp_node *siblings = NULL; |
| struct tree_node_meta node_meta = {NULL, &siblings}; |
| struct lysp_node_case *parsed = NULL; |
| |
| /* max subelems */ |
| st->yin_ctx->parsed_mod->version = LYS_VERSION_1_1; |
| data = ELEMENT_WRAPPER_START |
| "<case name=\"case-name\">" |
| "<anydata name=\"anyd\"/>" |
| "<anyxml name=\"anyx\"/>" |
| "<container name=\"subcont\"/>" |
| "<description><text>desc</text></description>" |
| "<if-feature name=\"iff\"/>" |
| "<leaf name=\"leaf\"> <type name=\"type\"/> </leaf>" |
| "<leaf-list name=\"llist\"> <type name=\"type\"/> </leaf-list>" |
| "<list name=\"list\"/>" |
| "<reference><text>ref</text></reference>" |
| "<status value=\"current\"/>" |
| "<uses name=\"uses-name\"/>" |
| "<when condition=\"when-cond\"/>" |
| "<choice name=\"choice\"/>" |
| EXT_SUBELEM |
| "</case>" |
| ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &node_meta, NULL, NULL), LY_SUCCESS); |
| parsed = (struct lysp_node_case *)siblings; |
| assert_string_equal(parsed->name, "case-name"); |
| assert_null(parsed->parent); |
| assert_int_equal(parsed->nodetype, LYS_CASE); |
| assert_true(parsed->flags & LYS_STATUS_CURR); |
| assert_null(parsed->next); |
| assert_string_equal(parsed->dsc, "desc"); |
| assert_string_equal(parsed->ref, "ref"); |
| assert_string_equal(parsed->when->cond, "when-cond"); |
| assert_string_equal(parsed->iffeatures[0].str, "iff"); |
| assert_string_equal(parsed->child->name, "anyd"); |
| assert_int_equal(parsed->child->nodetype, LYS_ANYDATA); |
| assert_string_equal(parsed->child->next->name, "anyx"); |
| assert_int_equal(parsed->child->next->nodetype, LYS_ANYXML); |
| assert_string_equal(parsed->child->next->next->name, "subcont"); |
| assert_int_equal(parsed->child->next->next->nodetype, LYS_CONTAINER); |
| assert_string_equal(parsed->child->next->next->next->name, "leaf"); |
| assert_int_equal(parsed->child->next->next->next->nodetype, LYS_LEAF); |
| assert_string_equal(parsed->child->next->next->next->next->name, "llist"); |
| assert_int_equal(parsed->child->next->next->next->next->nodetype, LYS_LEAFLIST); |
| assert_string_equal(parsed->child->next->next->next->next->next->name, "list"); |
| assert_int_equal(parsed->child->next->next->next->next->next->nodetype, LYS_LIST); |
| assert_string_equal(parsed->child->next->next->next->next->next->next->name, "uses-name"); |
| assert_int_equal(parsed->child->next->next->next->next->next->next->nodetype, LYS_USES); |
| assert_string_equal(parsed->child->next->next->next->next->next->next->next->name, "choice"); |
| assert_int_equal(parsed->child->next->next->next->next->next->next->next->nodetype, LYS_CHOICE); |
| assert_null(parsed->child->next->next->next->next->next->next->next->next); |
| assert_string_equal(parsed->exts[0].name, "urn:example:extensions:c-define"); |
| assert_int_equal(parsed->exts[0].insubstmt_index, 0); |
| assert_int_equal(parsed->exts[0].insubstmt, LYEXT_SUBSTMT_SELF); |
| lysp_node_free(st->ctx, siblings); |
| siblings = NULL; |
| |
| /* min subelems */ |
| data = ELEMENT_WRAPPER_START "<case name=\"case-name\" />" ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &node_meta, NULL, NULL), LY_SUCCESS); |
| parsed = (struct lysp_node_case *)siblings; |
| assert_string_equal(parsed->name, "case-name"); |
| lysp_node_free(st->ctx, siblings); |
| siblings = NULL; |
| |
| st->finished_correctly = true; |
| } |
| |
| static void |
| test_choice_elem(void **state) |
| { |
| struct test_parser_yin_state *st = *state; |
| const char *data; |
| struct lysp_node *siblings = NULL; |
| struct tree_node_meta node_meta = {NULL, &siblings}; |
| struct lysp_node_choice *parsed = NULL; |
| |
| /* max subelems */ |
| st->yin_ctx->parsed_mod->version = LYS_VERSION_1_1; |
| data = ELEMENT_WRAPPER_START |
| "<choice name=\"choice-name\">" |
| "<anydata name=\"anyd\"/>" |
| "<anyxml name=\"anyx\"/>" |
| "<case name=\"sub-case\"/>" |
| "<choice name=\"choice\"/>" |
| "<config value=\"true\"/>" |
| "<container name=\"subcont\"/>" |
| "<default value=\"def\"/>" |
| "<description><text>desc</text></description>" |
| "<if-feature name=\"iff\"/>" |
| "<leaf name=\"leaf\"> <type name=\"type\"/> </leaf>" |
| "<leaf-list name=\"llist\"> <type name=\"type\"/> </leaf-list>" |
| "<list name=\"list\"/>" |
| "<mandatory value=\"true\" />" |
| "<reference><text>ref</text></reference>" |
| "<status value=\"current\"/>" |
| "<when condition=\"when-cond\"/>" |
| EXT_SUBELEM |
| "</choice>" |
| ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &node_meta, NULL, NULL), LY_SUCCESS); |
| parsed = (struct lysp_node_choice *)siblings; |
| assert_string_equal(parsed->name, "choice-name"); |
| assert_null(parsed->parent); |
| assert_int_equal(parsed->nodetype, LYS_CHOICE); |
| assert_true(parsed->flags & LYS_CONFIG_W && parsed->flags & LYS_MAND_TRUE && parsed->flags & LYS_STATUS_CURR); |
| assert_null(parsed->next); |
| assert_string_equal(parsed->dsc, "desc"); |
| assert_string_equal(parsed->ref, "ref"); |
| assert_string_equal(parsed->when->cond, "when-cond"); |
| assert_string_equal(parsed->iffeatures[0].str, "iff"); |
| assert_string_equal(parsed->child->name, "anyd"); |
| assert_int_equal(parsed->child->nodetype, LYS_ANYDATA); |
| assert_string_equal(parsed->child->next->name, "anyx"); |
| assert_int_equal(parsed->child->next->nodetype, LYS_ANYXML); |
| assert_string_equal(parsed->child->next->next->name, "sub-case"); |
| assert_int_equal(parsed->child->next->next->nodetype, LYS_CASE); |
| assert_string_equal(parsed->child->next->next->next->name, "choice"); |
| assert_int_equal(parsed->child->next->next->next->nodetype, LYS_CHOICE); |
| assert_string_equal(parsed->child->next->next->next->next->name, "subcont"); |
| assert_int_equal(parsed->child->next->next->next->next->nodetype, LYS_CONTAINER); |
| assert_string_equal(parsed->child->next->next->next->next->next->name, "leaf"); |
| assert_int_equal(parsed->child->next->next->next->next->next->nodetype, LYS_LEAF); |
| assert_string_equal(parsed->child->next->next->next->next->next->next->name, "llist"); |
| assert_int_equal(parsed->child->next->next->next->next->next->next->nodetype, LYS_LEAFLIST); |
| assert_string_equal(parsed->child->next->next->next->next->next->next->next->name, "list"); |
| assert_int_equal(parsed->child->next->next->next->next->next->next->next->nodetype, LYS_LIST); |
| assert_null(parsed->child->next->next->next->next->next->next->next->next); |
| assert_string_equal(parsed->exts[0].name, "urn:example:extensions:c-define"); |
| assert_int_equal(parsed->exts[0].insubstmt_index, 0); |
| assert_int_equal(parsed->exts[0].insubstmt, LYEXT_SUBSTMT_SELF); |
| lysp_node_free(st->ctx, siblings); |
| siblings = NULL; |
| |
| /* min subelems */ |
| data = ELEMENT_WRAPPER_START "<choice name=\"choice-name\" />" ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &node_meta, NULL, NULL), LY_SUCCESS); |
| parsed = (struct lysp_node_choice *)siblings; |
| assert_string_equal(parsed->name, "choice-name"); |
| lysp_node_free(st->ctx, siblings); |
| siblings = NULL; |
| |
| st->finished_correctly = true; |
| } |
| |
| static void |
| test_inout_elem(void **state) |
| { |
| struct test_parser_yin_state *st = *state; |
| const char *data; |
| struct lysp_action_inout inout = {}; |
| struct inout_meta inout_meta = {NULL, &inout}; |
| |
| /* max subelements */ |
| st->yin_ctx->parsed_mod->version = LYS_VERSION_1_1; |
| data = ELEMENT_WRAPPER_START |
| "<input>" |
| "<anydata name=\"anyd\"/>" |
| "<anyxml name=\"anyx\"/>" |
| "<choice name=\"choice\"/>" |
| "<container name=\"subcont\"/>" |
| "<grouping name=\"sub-grp\"/>" |
| "<leaf name=\"leaf\"> <type name=\"type\"/> </leaf>" |
| "<leaf-list name=\"llist\"> <type name=\"type\"/> </leaf-list>" |
| "<list name=\"list\"/>" |
| "<must condition=\"cond\"/>" |
| "<typedef name=\"tpdf\"> <type name=\"type\"/> </typedef>" |
| "<uses name=\"uses-name\"/>" |
| EXT_SUBELEM |
| "</input>" |
| ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &inout_meta, NULL, NULL), LY_SUCCESS); |
| assert_null(inout.parent); |
| assert_int_equal(inout.nodetype, LYS_INPUT); |
| assert_string_equal(inout.musts->arg.str, "cond"); |
| assert_string_equal(inout.typedefs->name, "tpdf"); |
| assert_string_equal(inout.groupings->name, "sub-grp"); |
| assert_string_equal(inout.data->name, "anyd"); |
| assert_int_equal(inout.data->nodetype, LYS_ANYDATA); |
| assert_string_equal(inout.data->next->name, "anyx"); |
| assert_int_equal(inout.data->next->nodetype, LYS_ANYXML); |
| assert_string_equal(inout.data->next->next->name, "choice"); |
| assert_int_equal(inout.data->next->next->nodetype, LYS_CHOICE); |
| assert_string_equal(inout.data->next->next->next->name, "subcont"); |
| assert_int_equal(inout.data->next->next->next->nodetype, LYS_CONTAINER); |
| assert_string_equal(inout.data->next->next->next->next->name, "leaf"); |
| assert_int_equal(inout.data->next->next->next->next->nodetype, LYS_LEAF); |
| assert_string_equal(inout.data->next->next->next->next->next->name, "llist"); |
| assert_int_equal(inout.data->next->next->next->next->next->nodetype, LYS_LEAFLIST); |
| assert_string_equal(inout.data->next->next->next->next->next->next->name, "list"); |
| assert_int_equal(inout.data->next->next->next->next->next->next->nodetype, LYS_LIST); |
| assert_string_equal(inout.data->next->next->next->next->next->next->next->name, "uses-name"); |
| assert_int_equal(inout.data->next->next->next->next->next->next->next->nodetype, LYS_USES); |
| assert_null(inout.data->next->next->next->next->next->next->next->next); |
| assert_string_equal(inout.exts[0].name, "urn:example:extensions:c-define"); |
| assert_int_equal(inout.exts[0].insubstmt_index, 0); |
| assert_int_equal(inout.exts[0].insubstmt, LYEXT_SUBSTMT_SELF); |
| lysp_action_inout_free(st->ctx, &inout); |
| memset(&inout, 0, sizeof inout); |
| |
| /* max subelements */ |
| st->yin_ctx->parsed_mod->version = LYS_VERSION_1_1; |
| data = ELEMENT_WRAPPER_START |
| "<output>" |
| "<anydata name=\"anyd\"/>" |
| "<anyxml name=\"anyx\"/>" |
| "<choice name=\"choice\"/>" |
| "<container name=\"subcont\"/>" |
| "<grouping name=\"sub-grp\"/>" |
| "<leaf name=\"leaf\"> <type name=\"type\"/> </leaf>" |
| "<leaf-list name=\"llist\"> <type name=\"type\"/> </leaf-list>" |
| "<list name=\"list\"/>" |
| "<must condition=\"cond\"/>" |
| "<typedef name=\"tpdf\"> <type name=\"type\"/> </typedef>" |
| "<uses name=\"uses-name\"/>" |
| EXT_SUBELEM |
| "</output>" |
| ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &inout_meta, NULL, NULL), LY_SUCCESS); |
| assert_null(inout.parent); |
| assert_int_equal(inout.nodetype, LYS_OUTPUT); |
| assert_string_equal(inout.musts->arg.str, "cond"); |
| assert_string_equal(inout.typedefs->name, "tpdf"); |
| assert_string_equal(inout.groupings->name, "sub-grp"); |
| assert_string_equal(inout.data->name, "anyd"); |
| assert_int_equal(inout.data->nodetype, LYS_ANYDATA); |
| assert_string_equal(inout.data->next->name, "anyx"); |
| assert_int_equal(inout.data->next->nodetype, LYS_ANYXML); |
| assert_string_equal(inout.data->next->next->name, "choice"); |
| assert_int_equal(inout.data->next->next->nodetype, LYS_CHOICE); |
| assert_string_equal(inout.data->next->next->next->name, "subcont"); |
| assert_int_equal(inout.data->next->next->next->nodetype, LYS_CONTAINER); |
| assert_string_equal(inout.data->next->next->next->next->name, "leaf"); |
| assert_int_equal(inout.data->next->next->next->next->nodetype, LYS_LEAF); |
| assert_string_equal(inout.data->next->next->next->next->next->name, "llist"); |
| assert_int_equal(inout.data->next->next->next->next->next->nodetype, LYS_LEAFLIST); |
| assert_string_equal(inout.data->next->next->next->next->next->next->name, "list"); |
| assert_int_equal(inout.data->next->next->next->next->next->next->nodetype, LYS_LIST); |
| assert_string_equal(inout.data->next->next->next->next->next->next->next->name, "uses-name"); |
| assert_int_equal(inout.data->next->next->next->next->next->next->next->nodetype, LYS_USES); |
| assert_null(inout.data->next->next->next->next->next->next->next->next); |
| assert_string_equal(inout.exts[0].name, "urn:example:extensions:c-define"); |
| assert_int_equal(inout.exts[0].insubstmt_index, 0); |
| assert_int_equal(inout.exts[0].insubstmt, LYEXT_SUBSTMT_SELF); |
| lysp_action_inout_free(st->ctx, &inout); |
| memset(&inout, 0, sizeof inout); |
| |
| /* min subelems */ |
| data = ELEMENT_WRAPPER_START "<input><leaf name=\"l\"><type name=\"empty\"/></leaf></input>" ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &inout_meta, NULL, NULL), LY_SUCCESS); |
| lysp_action_inout_free(st->ctx, &inout); |
| memset(&inout, 0, sizeof inout); |
| |
| data = ELEMENT_WRAPPER_START "<output><leaf name=\"l\"><type name=\"empty\"/></leaf></output>" ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &inout_meta, NULL, NULL), LY_SUCCESS); |
| lysp_action_inout_free(st->ctx, &inout); |
| memset(&inout, 0, sizeof inout); |
| |
| /* invalid combinations */ |
| data = ELEMENT_WRAPPER_START "<input name=\"test\"/>" ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &inout_meta, NULL, NULL), LY_EVALID); |
| logbuf_assert("Unexpected attribute \"name\" of \"input\" element. Line number 1."); |
| memset(&inout, 0, sizeof inout); |
| |
| st->finished_correctly = true; |
| } |
| |
| static void |
| test_action_elem(void **state) |
| { |
| struct test_parser_yin_state *st = *state; |
| const char *data; |
| struct lysp_action *actions = NULL; |
| struct tree_node_meta act_meta = {NULL, (struct lysp_node **)&actions}; |
| |
| /* max subelems */ |
| st->yin_ctx->parsed_mod->version = LYS_VERSION_1_1; |
| data = ELEMENT_WRAPPER_START |
| "<action name=\"act\">" |
| "<description><text>desc</text></description>" |
| "<grouping name=\"grouping\"/>" |
| "<if-feature name=\"iff\"/>" |
| "<input><uses name=\"uses-name\"/></input>" |
| "<output><must condition=\"cond\"/><leaf name=\"l\"><type name=\"type\"/></leaf></output>" |
| "<reference><text>ref</text></reference>" |
| "<status value=\"deprecated\"/>" |
| "<typedef name=\"tpdf\"> <type name=\"type\"/> </typedef>" |
| EXT_SUBELEM |
| "</action>" |
| ELEMENT_WRAPPER_END; |
| /* there must be parent for action */ |
| act_meta.parent = (void*)1; |
| assert_int_equal(test_element_helper(st, data, &act_meta, NULL, NULL), LY_SUCCESS); |
| act_meta.parent = NULL; |
| assert_non_null(actions->parent); |
| assert_int_equal(actions->nodetype, LYS_ACTION); |
| assert_true(actions->flags & LYS_STATUS_DEPRC); |
| assert_string_equal(actions->name, "act"); |
| assert_string_equal(actions->dsc, "desc"); |
| assert_string_equal(actions->ref, "ref"); |
| assert_string_equal(actions->iffeatures[0].str, "iff"); |
| assert_string_equal(actions->typedefs->name, "tpdf"); |
| assert_string_equal(actions->groupings->name, "grouping"); |
| assert_string_equal(actions->input.data->name, "uses-name"); |
| assert_string_equal(actions->output.musts->arg.str, "cond"); |
| assert_string_equal(actions->exts[0].name, "urn:example:extensions:c-define"); |
| assert_int_equal(actions->exts[0].insubstmt_index, 0); |
| assert_int_equal(actions->exts[0].insubstmt, LYEXT_SUBSTMT_SELF); |
| FREE_ARRAY(st->ctx, actions, lysp_action_free) |
| actions = NULL; |
| |
| st->yin_ctx->parsed_mod->version = LYS_VERSION_1_1; |
| data = ELEMENT_WRAPPER_START |
| "<rpc name=\"act\">" |
| "<description><text>desc</text></description>" |
| "<grouping name=\"grouping\"/>" |
| "<if-feature name=\"iff\"/>" |
| "<input><uses name=\"uses-name\"/></input>" |
| "<output><must condition=\"cond\"/><leaf name=\"l\"><type name=\"type\"/></leaf></output>" |
| "<reference><text>ref</text></reference>" |
| "<status value=\"deprecated\"/>" |
| "<typedef name=\"tpdf\"> <type name=\"type\"/> </typedef>" |
| EXT_SUBELEM |
| "</rpc>" |
| ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &act_meta, NULL, NULL), LY_SUCCESS); |
| assert_null(actions->parent); |
| assert_int_equal(actions->nodetype, LYS_RPC); |
| assert_true(actions->flags & LYS_STATUS_DEPRC); |
| assert_string_equal(actions->name, "act"); |
| assert_string_equal(actions->dsc, "desc"); |
| assert_string_equal(actions->ref, "ref"); |
| assert_string_equal(actions->iffeatures[0].str, "iff"); |
| assert_string_equal(actions->typedefs->name, "tpdf"); |
| assert_string_equal(actions->groupings->name, "grouping"); |
| assert_string_equal(actions->input.data->name, "uses-name"); |
| assert_string_equal(actions->output.musts->arg.str, "cond"); |
| assert_string_equal(actions->exts[0].name, "urn:example:extensions:c-define"); |
| assert_int_equal(actions->exts[0].insubstmt_index, 0); |
| assert_int_equal(actions->exts[0].insubstmt, LYEXT_SUBSTMT_SELF); |
| FREE_ARRAY(st->ctx, actions, lysp_action_free) |
| actions = NULL; |
| |
| /* min subelems */ |
| data = ELEMENT_WRAPPER_START "<action name=\"act\" />" ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &act_meta, NULL, NULL), LY_SUCCESS); |
| assert_string_equal(actions->name, "act"); |
| FREE_ARRAY(st->ctx, actions, lysp_action_free) |
| actions = NULL; |
| |
| st->finished_correctly = true; |
| } |
| |
| static void |
| test_augment_elem(void **state) |
| { |
| struct test_parser_yin_state *st = *state; |
| const char *data; |
| struct lysp_augment *augments = NULL; |
| struct tree_node_meta aug_meta = {NULL, (struct lysp_node **)&augments}; |
| |
| st->yin_ctx->parsed_mod->version = LYS_VERSION_1_1; |
| data = ELEMENT_WRAPPER_START |
| "<augment target-node=\"target\">" |
| "<action name=\"action\"/>" |
| "<anydata name=\"anyd\"/>" |
| "<anyxml name=\"anyx\"/>" |
| "<case name=\"case\"/>" |
| "<choice name=\"choice\"/>" |
| "<container name=\"subcont\"/>" |
| "<description><text>desc</text></description>" |
| "<if-feature name=\"iff\"/>" |
| "<leaf name=\"leaf\"> <type name=\"type\"/> </leaf>" |
| "<leaf-list name=\"llist\"> <type name=\"type\"/> </leaf-list>" |
| "<list name=\"list\"/>" |
| "<notification name=\"notif\"/>" |
| "<reference><text>ref</text></reference>" |
| "<status value=\"current\"/>" |
| "<uses name=\"uses\"/>" |
| "<when condition=\"when-cond\"/>" |
| EXT_SUBELEM |
| "</augment>" |
| ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &aug_meta, NULL, NULL), LY_SUCCESS); |
| assert_string_equal(augments->nodeid, "target"); |
| assert_null(augments->parent); |
| assert_int_equal(augments->nodetype, LYS_AUGMENT); |
| assert_true(augments->flags & LYS_STATUS_CURR); |
| assert_string_equal(augments->dsc, "desc"); |
| assert_string_equal(augments->ref, "ref"); |
| assert_string_equal(augments->when->cond, "when-cond"); |
| assert_string_equal(augments->iffeatures[0].str, "iff"); |
| assert_string_equal(augments->child->name, "anyd"); |
| assert_int_equal(augments->child->nodetype, LYS_ANYDATA); |
| assert_string_equal(augments->child->next->name, "anyx"); |
| assert_int_equal(augments->child->next->nodetype, LYS_ANYXML); |
| assert_string_equal(augments->child->next->next->name, "case"); |
| assert_int_equal(augments->child->next->next->nodetype, LYS_CASE); |
| assert_string_equal(augments->child->next->next->next->name, "choice"); |
| assert_int_equal(augments->child->next->next->next->nodetype, LYS_CHOICE); |
| assert_string_equal(augments->child->next->next->next->next->name, "subcont"); |
| assert_int_equal(augments->child->next->next->next->next->nodetype, LYS_CONTAINER); |
| assert_string_equal(augments->child->next->next->next->next->next->name, "leaf"); |
| assert_int_equal(augments->child->next->next->next->next->next->nodetype, LYS_LEAF); |
| assert_string_equal(augments->child->next->next->next->next->next->next->name, "llist"); |
| assert_int_equal(augments->child->next->next->next->next->next->next->nodetype, LYS_LEAFLIST); |
| assert_string_equal(augments->child->next->next->next->next->next->next->next->name, "list"); |
| assert_int_equal(augments->child->next->next->next->next->next->next->next->nodetype, LYS_LIST); |
| assert_string_equal(augments->child->next->next->next->next->next->next->next->next->name, "uses"); |
| assert_int_equal(augments->child->next->next->next->next->next->next->next->next->nodetype, LYS_USES); |
| assert_null(augments->child->next->next->next->next->next->next->next->next->next); |
| assert_string_equal(augments->actions->name, "action"); |
| assert_string_equal(augments->notifs->name, "notif"); |
| assert_string_equal(augments->exts[0].name, "urn:example:extensions:c-define"); |
| assert_int_equal(augments->exts[0].insubstmt_index, 0); |
| assert_int_equal(augments->exts[0].insubstmt, LYEXT_SUBSTMT_SELF); |
| FREE_ARRAY(st->ctx, augments, lysp_augment_free) |
| augments = NULL; |
| |
| data = ELEMENT_WRAPPER_START "<augment target-node=\"target\" />" ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &aug_meta, NULL, NULL), LY_SUCCESS); |
| assert_string_equal(augments->nodeid, "target"); |
| FREE_ARRAY(st->ctx, augments, lysp_augment_free) |
| augments = NULL; |
| |
| st->finished_correctly = true; |
| } |
| |
| static void |
| test_deviate_elem(void **state) |
| { |
| struct test_parser_yin_state *st = *state; |
| const char *data; |
| struct lysp_deviate *deviates = NULL; |
| struct lysp_deviate_add *d_add; |
| struct lysp_deviate_rpl *d_rpl; |
| struct lysp_deviate_del *d_del; |
| |
| /* all valid arguments with min subelems */ |
| data = ELEMENT_WRAPPER_START "<deviate value=\"not-supported\" />" ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &deviates, NULL, NULL), LY_SUCCESS); |
| assert_int_equal(deviates->mod, LYS_DEV_NOT_SUPPORTED); |
| lysp_deviate_free(st->ctx, deviates); |
| free(deviates); |
| deviates = NULL; |
| |
| data = ELEMENT_WRAPPER_START "<deviate value=\"add\" />" ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &deviates, NULL, NULL), LY_SUCCESS); |
| assert_int_equal(deviates->mod, LYS_DEV_ADD); |
| lysp_deviate_free(st->ctx, deviates); |
| free(deviates); |
| deviates = NULL; |
| |
| data = ELEMENT_WRAPPER_START "<deviate value=\"replace\" />" ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &deviates, NULL, NULL), LY_SUCCESS); |
| assert_int_equal(deviates->mod, LYS_DEV_REPLACE); |
| lysp_deviate_free(st->ctx, deviates); |
| free(deviates); |
| deviates = NULL; |
| |
| data = ELEMENT_WRAPPER_START "<deviate value=\"delete\" />" ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &deviates, NULL, NULL), LY_SUCCESS); |
| assert_int_equal(deviates->mod, LYS_DEV_DELETE); |
| lysp_deviate_free(st->ctx, deviates); |
| free(deviates); |
| deviates = NULL; |
| |
| /* max subelems and valid arguments */ |
| data = ELEMENT_WRAPPER_START |
| "<deviate value=\"not-supported\">" |
| EXT_SUBELEM |
| "</deviate>" |
| ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &deviates, NULL, NULL), LY_SUCCESS); |
| assert_int_equal(deviates->mod, LYS_DEV_NOT_SUPPORTED); |
| assert_string_equal(deviates->exts[0].name, "urn:example:extensions:c-define"); |
| assert_int_equal(deviates->exts[0].insubstmt_index, 0); |
| assert_int_equal(deviates->exts[0].insubstmt, LYEXT_SUBSTMT_SELF); |
| lysp_deviate_free(st->ctx, deviates); |
| free(deviates); |
| deviates = NULL; |
| |
| data = ELEMENT_WRAPPER_START |
| "<deviate value=\"add\">" |
| "<units name=\"units\"/>" |
| "<must condition=\"cond\"/>" |
| "<unique tag=\"utag\"/>" |
| "<default value=\"def\"/>" |
| "<config value=\"true\"/>" |
| "<mandatory value=\"true\"/>" |
| "<min-elements value=\"5\"/>" |
| "<max-elements value=\"15\"/>" |
| EXT_SUBELEM |
| "</deviate>" |
| ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &deviates, NULL, NULL), LY_SUCCESS); |
| d_add = (struct lysp_deviate_add *)deviates; |
| assert_int_equal(d_add->mod, LYS_DEV_ADD); |
| assert_null(d_add->next); |
| assert_string_equal(d_add->units, "units"); |
| assert_string_equal(d_add->musts->arg.str, "cond"); |
| assert_string_equal(d_add->uniques[0].str, "utag"); |
| assert_string_equal(d_add->dflts[0].str, "def"); |
| assert_true(d_add->flags & LYS_MAND_TRUE && d_add->flags & LYS_CONFIG_W); |
| assert_int_equal(d_add->min, 5); |
| assert_int_equal(d_add->max, 15); |
| assert_string_equal(deviates->exts[0].name, "urn:example:extensions:c-define"); |
| assert_int_equal(deviates->exts[0].insubstmt_index, 0); |
| assert_int_equal(deviates->exts[0].insubstmt, LYEXT_SUBSTMT_SELF); |
| lysp_deviate_free(st->ctx, deviates); |
| free(deviates); |
| deviates = NULL; |
| |
| data = ELEMENT_WRAPPER_START |
| "<deviate value=\"replace\">" |
| "<type name=\"newtype\"/>" |
| "<units name=\"uni\"/>" |
| "<default value=\"def\"/>" |
| "<config value=\"true\"/>" |
| "<mandatory value=\"true\"/>" |
| "<min-elements value=\"5\"/>" |
| "<max-elements value=\"15\"/>" |
| EXT_SUBELEM |
| "</deviate>" |
| ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &deviates, NULL, NULL), LY_SUCCESS); |
| d_rpl = (struct lysp_deviate_rpl *)deviates; |
| assert_int_equal(d_rpl->mod, LYS_DEV_REPLACE); |
| assert_null(d_rpl->next); |
| assert_string_equal(d_rpl->type->name, "newtype"); |
| assert_string_equal(d_rpl->units, "uni"); |
| assert_string_equal(d_rpl->dflt.str, "def"); |
| assert_true(d_rpl->flags & LYS_MAND_TRUE && d_rpl->flags & LYS_CONFIG_W); |
| assert_int_equal(d_rpl->min, 5); |
| assert_int_equal(d_rpl->max, 15); |
| assert_string_equal(deviates->exts[0].name, "urn:example:extensions:c-define"); |
| assert_int_equal(deviates->exts[0].insubstmt_index, 0); |
| assert_int_equal(deviates->exts[0].insubstmt, LYEXT_SUBSTMT_SELF); |
| lysp_deviate_free(st->ctx, deviates); |
| free(deviates); |
| deviates = NULL; |
| |
| data = ELEMENT_WRAPPER_START |
| "<deviate value=\"delete\">" |
| "<units name=\"u\"/>" |
| "<must condition=\"c\"/>" |
| "<unique tag=\"tag\"/>" |
| "<default value=\"default\"/>" |
| EXT_SUBELEM |
| "</deviate>" |
| ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &deviates, NULL, NULL), LY_SUCCESS); |
| d_del = (struct lysp_deviate_del *)deviates; |
| assert_int_equal(d_del->mod, LYS_DEV_DELETE); |
| assert_null(d_del->next); |
| assert_string_equal(d_del->units, "u"); |
| assert_string_equal(d_del->musts->arg.str, "c"); |
| assert_string_equal(d_del->uniques[0].str, "tag"); |
| assert_string_equal(d_del->dflts[0].str, "default"); |
| assert_string_equal(deviates->exts[0].name, "urn:example:extensions:c-define"); |
| assert_int_equal(deviates->exts[0].insubstmt_index, 0); |
| assert_int_equal(deviates->exts[0].insubstmt, LYEXT_SUBSTMT_SELF); |
| lysp_deviate_free(st->ctx, deviates); |
| free(deviates); |
| deviates = NULL; |
| |
| /* invalid arguments */ |
| data = ELEMENT_WRAPPER_START "<deviate value=\"\" />" ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &deviates, NULL, NULL), LY_EVALID); |
| logbuf_assert("Invalid value \"\" of \"value\" attribute in \"deviate\" element. Valid values are \"not-supported\", \"add\", \"replace\" and \"delete\". Line number 1."); |
| deviates = NULL; |
| |
| data = ELEMENT_WRAPPER_START "<deviate value=\"invalid\" />" ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &deviates, NULL, NULL), LY_EVALID); |
| logbuf_assert("Invalid value \"invalid\" of \"value\" attribute in \"deviate\" element. Valid values are \"not-supported\", \"add\", \"replace\" and \"delete\". Line number 1."); |
| deviates = NULL; |
| |
| data = ELEMENT_WRAPPER_START "<deviate value=\"ad\" />" ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &deviates, NULL, NULL), LY_EVALID); |
| logbuf_assert("Invalid value \"ad\" of \"value\" attribute in \"deviate\" element. Valid values are \"not-supported\", \"add\", \"replace\" and \"delete\". Line number 1."); |
| deviates = NULL; |
| |
| data = ELEMENT_WRAPPER_START "<deviate value=\"adds\" />" ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &deviates, NULL, NULL), LY_EVALID); |
| logbuf_assert("Invalid value \"adds\" of \"value\" attribute in \"deviate\" element. Valid values are \"not-supported\", \"add\", \"replace\" and \"delete\". Line number 1."); |
| deviates = NULL; |
| |
| data = ELEMENT_WRAPPER_START |
| "<deviate value=\"not-supported\">" |
| "<must condition=\"c\"/>" |
| "</deviate>" |
| ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &deviates, NULL, NULL), LY_EVALID); |
| logbuf_assert("Deviate of this type doesn't allow \"must\" as it's sub-element. Line number 1."); |
| |
| st->finished_correctly = true; |
| } |
| |
| static void |
| test_deviation_elem(void **state) |
| { |
| struct test_parser_yin_state *st = *state; |
| const char *data; |
| struct lysp_deviation *deviations = NULL; |
| |
| /* min subelems */ |
| data = ELEMENT_WRAPPER_START |
| "<deviation target-node=\"target\">" |
| "<deviate value=\"not-supported\"/>" |
| "</deviation>" |
| ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &deviations, NULL, NULL), LY_SUCCESS); |
| assert_string_equal(deviations->nodeid, "target"); |
| assert_int_equal(deviations->deviates->mod, LYS_DEV_NOT_SUPPORTED); |
| FREE_ARRAY(st->ctx, deviations, lysp_deviation_free); |
| deviations = NULL; |
| |
| /* max subelems */ |
| data = ELEMENT_WRAPPER_START |
| "<deviation target-node=\"target\">" |
| "<reference><text>ref</text></reference>" |
| "<description><text>desc</text></description>" |
| "<deviate value=\"add\"/>" |
| EXT_SUBELEM |
| "</deviation>" |
| ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &deviations, NULL, NULL), LY_SUCCESS); |
| assert_string_equal(deviations->nodeid, "target"); |
| assert_int_equal(deviations->deviates->mod, LYS_DEV_ADD); |
| assert_string_equal(deviations->ref, "ref"); |
| assert_string_equal(deviations->dsc, "desc"); |
| assert_string_equal(deviations->exts[0].name, "urn:example:extensions:c-define"); |
| assert_int_equal(deviations->exts[0].insubstmt_index, 0); |
| assert_int_equal(deviations->exts[0].insubstmt, LYEXT_SUBSTMT_SELF); |
| FREE_ARRAY(st->ctx, deviations, lysp_deviation_free); |
| deviations = NULL; |
| |
| /* invalid */ |
| data = ELEMENT_WRAPPER_START "<deviation target-node=\"target\"/>" ELEMENT_WRAPPER_END; |
| assert_int_equal(test_element_helper(st, data, &deviations, NULL, NULL), LY_EVALID); |
| FREE_ARRAY(st->ctx, deviations, lysp_deviation_free); |
| deviations = NULL; |
| logbuf_assert("Missing mandatory sub-element \"deviate\" of \"deviation\" element. Line number 1."); |
| /* TODO */ |
| st->finished_correctly = true; |
| } |
| |
| static struct lysp_module * |
| mod_renew(struct lys_yin_parser_ctx *ctx) |
| { |
| struct ly_ctx *ly_ctx = ctx->parsed_mod->mod->ctx; |
| |
| lys_module_free(ctx->parsed_mod->mod, NULL); |
| ctx->parsed_mod = calloc(1, sizeof *ctx->parsed_mod); |
| ctx->parsed_mod->mod = calloc(1, sizeof *ctx->parsed_mod->mod); |
| ctx->parsed_mod->mod->parsed = ctx->parsed_mod; |
| ctx->parsed_mod->mod->ctx = ly_ctx; |
| |
| return ctx->parsed_mod; |
| } |
| |
| static void |
| test_module_elem(void **state) |
| { |
| struct test_parser_yin_state *st = *state; |
| const char *data; |
| struct lysp_module *lysp_mod = mod_renew(st->yin_ctx); |
| |
| /* max subelems */ |
| data = "<module xmlns=\"urn:ietf:params:xml:ns:yang:yin:1\" name=\"mod\">\n" |
| "<yang-version value=\"1.1\"/>\n" |
| "<namespace uri=\"ns\"/>\n" |
| "<prefix value=\"pref\"/>\n" |
| "<include module=\"b-mod\"/>\n" |
| "<import module=\"a-mod\"><prefix value=\"imp-pref\"/></import>\n" |
| "<organization><text>org</text></organization>\n" |
| "<contact><text>contact</text></contact>\n" |
| "<description><text>desc</text></description>" |
| "<reference><text>ref</text></reference>\n" |
| "<revision date=\"2019-02-02\"/>\n" |
| "<anydata name=\"anyd\"/>\n" |
| "<anyxml name=\"anyx\"/>\n" |
| "<choice name=\"choice\"/>\n" |
| "<container name=\"cont\"/>\n" |
| "<leaf name=\"leaf\"> <type name=\"type\"/> </leaf>\n" |
| "<leaf-list name=\"llist\"> <type name=\"type\"/> </leaf-list>\n" |
| "<list name=\"sub-list\"/>\n" |
| "<uses name=\"uses-name\"/>\n" |
| "<augment target-node=\"target\"/>\n" |
| "<deviation target-node=\"target\">""<deviate value=\"not-supported\"/>""</deviation>\n" |
| "<extension name=\"ext\"/>\n" |
| "<feature name=\"feature\"/>\n" |
| "<grouping name=\"grp\"/>\n" |
| "<identity name=\"ident-name\"/>\n" |
| "<notification name=\"notf\"/>\n" |
| "<rpc name=\"rpc-name\"/>\n" |
| "<typedef name=\"tpdf\"> <type name=\"type\"/> </typedef>\n" |
| EXT_SUBELEM"\n" |
| "</module>\n"; |
| assert_int_equal(ly_in_new_memory(data, &st->in), LY_SUCCESS); |
| assert_int_equal(lyxml_ctx_new(st->ctx, st->in, &st->yin_ctx->xmlctx), LY_SUCCESS); |
| |
| assert_int_equal(yin_parse_mod(st->yin_ctx, lysp_mod), LY_SUCCESS); |
| assert_string_equal(lysp_mod->mod->name, "mod"); |
| assert_string_equal(lysp_mod->revs, "2019-02-02"); |
| assert_string_equal(lysp_mod->mod->ns, "ns"); |
| assert_string_equal(lysp_mod->mod->prefix, "pref"); |
| assert_null(lysp_mod->mod->filepath); |
| assert_string_equal(lysp_mod->mod->org, "org"); |
| assert_string_equal(lysp_mod->mod->contact, "contact"); |
| assert_string_equal(lysp_mod->mod->dsc, "desc"); |
| assert_string_equal(lysp_mod->mod->ref, "ref"); |
| assert_int_equal(lysp_mod->version, LYS_VERSION_1_1); |
| assert_string_equal(lysp_mod->imports->name, "a-mod"); |
| assert_string_equal(lysp_mod->includes->name, "b-mod"); |
| assert_string_equal(lysp_mod->extensions->name, "ext"); |
| assert_string_equal(lysp_mod->features->name, "feature"); |
| assert_string_equal(lysp_mod->identities->name, "ident-name"); |
| assert_string_equal(lysp_mod->typedefs->name, "tpdf"); |
| assert_string_equal(lysp_mod->groupings->name, "grp"); |
| assert_string_equal(lysp_mod->data->name, "anyd"); |
| assert_int_equal(lysp_mod->data->nodetype, LYS_ANYDATA); |
| assert_string_equal(lysp_mod->data->next->name, "anyx"); |
| assert_int_equal(lysp_mod->data->next->nodetype, LYS_ANYXML); |
| assert_string_equal(lysp_mod->data->next->next->name, "choice"); |
| assert_int_equal(lysp_mod->data->next->next->nodetype, LYS_CHOICE); |
| assert_string_equal(lysp_mod->data->next->next->next->name, "cont"); |
| assert_int_equal(lysp_mod->data->next->next->next->nodetype, LYS_CONTAINER); |
| assert_string_equal(lysp_mod->data->next->next->next->next->name, "leaf"); |
| assert_int_equal(lysp_mod->data->next->next->next->next->nodetype, LYS_LEAF); |
| assert_string_equal(lysp_mod->data->next->next->next->next->next->name, "llist"); |
| assert_int_equal(lysp_mod->data->next->next->next->next->next->nodetype, LYS_LEAFLIST); |
| assert_string_equal(lysp_mod->data->next->next->next->next->next->next->name, "sub-list"); |
| assert_int_equal(lysp_mod->data->next->next->next->next->next->next->nodetype, LYS_LIST); |
| assert_string_equal(lysp_mod->data->next->next->next->next->next->next->next->name, "uses-name"); |
| assert_int_equal(lysp_mod->data->next->next->next->next->next->next->next->nodetype, LYS_USES); |
| assert_null(lysp_mod->data->next->next->next->next->next->next->next->next); |
| assert_string_equal(lysp_mod->augments->nodeid, "target"); |
| assert_string_equal(lysp_mod->rpcs->name, "rpc-name"); |
| assert_string_equal(lysp_mod->notifs->name, "notf"); |
| assert_string_equal(lysp_mod->deviations->nodeid, "target"); |
| assert_string_equal(lysp_mod->exts[0].name, "urn:example:extensions:c-define"); |
| assert_int_equal(lysp_mod->exts[0].insubstmt_index, 0); |
| assert_int_equal(lysp_mod->exts[0].insubstmt, LYEXT_SUBSTMT_SELF); |
| |
| /* min subelems */ |
| ly_in_free(st->in, 0); |
| lyxml_ctx_free(st->yin_ctx->xmlctx); |
| lysp_mod = mod_renew(st->yin_ctx); |
| data = "<module xmlns=\"urn:ietf:params:xml:ns:yang:yin:1\" name=\"mod\">" |
| "<namespace uri=\"ns\"/>" |
| "<prefix value=\"pref\"/>" |
| "<yang-version value=\"1.1\"/>" |
| "</module>"; |
| assert_int_equal(ly_in_new_memory(data, &st->in), LY_SUCCESS); |
| assert_int_equal(lyxml_ctx_new(st->ctx, st->in, &st->yin_ctx->xmlctx), LY_SUCCESS); |
| assert_int_equal(yin_parse_mod(st->yin_ctx, lysp_mod), LY_SUCCESS); |
| assert_string_equal(lysp_mod->mod->name, "mod"); |
| |
| /* incorrect subelem order */ |
| ly_in_free(st->in, 0); |
| lyxml_ctx_free(st->yin_ctx->xmlctx); |
| lysp_mod = mod_renew(st->yin_ctx); |
| data = "<module xmlns=\"urn:ietf:params:xml:ns:yang:yin:1\" name=\"mod\">" |
| "<feature name=\"feature\"/>\n" |
| "<namespace uri=\"ns\"/>" |
| "<prefix value=\"pref\"/>" |
| "<yang-version value=\"1.1\"/>" |
| "</module>"; |
| assert_int_equal(ly_in_new_memory(data, &st->in), LY_SUCCESS); |
| assert_int_equal(lyxml_ctx_new(st->ctx, st->in, &st->yin_ctx->xmlctx), LY_SUCCESS); |
| assert_int_equal(yin_parse_mod(st->yin_ctx, lysp_mod), LY_EVALID); |
| logbuf_assert("Invalid order of module\'s sub-elements \"namespace\" can\'t appear after \"feature\". Line number 2."); |
| |
| st->finished_correctly = true; |
| } |
| |
| static struct lysp_submodule * |
| submod_renew(struct lys_yin_parser_ctx *ctx, const char *belongs_to) |
| { |
| struct ly_ctx *ly_ctx = ctx->parsed_mod->mod->ctx; |
| |
| lys_module_free(ctx->parsed_mod->mod, NULL); |
| ctx->parsed_mod = calloc(1, sizeof(struct lysp_submodule)); |
| ctx->parsed_mod->mod = calloc(1, sizeof *ctx->parsed_mod->mod); |
| lydict_insert(ly_ctx, belongs_to, 0, &ctx->parsed_mod->mod->name); |
| ctx->parsed_mod->mod->parsed = ctx->parsed_mod; |
| ctx->parsed_mod->mod->ctx = ly_ctx; |
| |
| return (struct lysp_submodule *)ctx->parsed_mod; |
| } |
| |
| static void |
| test_submodule_elem(void **state) |
| { |
| struct test_parser_yin_state *st = *state; |
| const char *data; |
| struct lysp_submodule *lysp_submod = submod_renew(st->yin_ctx, "module-name"); |
| |
| /* max subelements */ |
| data = "<submodule xmlns=\"urn:ietf:params:xml:ns:yang:yin:1\" name=\"mod\">\n" |
| "<yang-version value=\"1.1\"/>\n" |
| "<belongs-to module=\"module-name\"><prefix value=\"pref\"/></belongs-to>" |
| "<include module=\"b-mod\"/>\n" |
| "<import module=\"a-mod\"><prefix value=\"imp-pref\"/></import>\n" |
| "<organization><text>org</text></organization>\n" |
| "<contact><text>contact</text></contact>\n" |
| "<description><text>desc</text></description>" |
| "<reference><text>ref</text></reference>\n" |
| "<revision date=\"2019-02-02\"/>\n" |
| "<anydata name=\"anyd\"/>\n" |
| "<anyxml name=\"anyx\"/>\n" |
| "<choice name=\"choice\"/>\n" |
| "<container name=\"cont\"/>\n" |
| "<leaf name=\"leaf\"> <type name=\"type\"/> </leaf>\n" |
| "<leaf-list name=\"llist\"> <type name=\"type\"/> </leaf-list>\n" |
| "<list name=\"sub-list\"/>\n" |
| "<uses name=\"uses-name\"/>\n" |
| "<augment target-node=\"target\"/>\n" |
| "<deviation target-node=\"target\">""<deviate value=\"not-supported\"/>""</deviation>\n" |
| "<extension name=\"ext\"/>\n" |
| "<feature name=\"feature\"/>\n" |
| "<grouping name=\"grp\"/>\n" |
| "<identity name=\"ident-name\"/>\n" |
| "<notification name=\"notf\"/>\n" |
| "<rpc name=\"rpc-name\"/>\n" |
| "<typedef name=\"tpdf\"> <type name=\"type\"/> </typedef>\n" |
| EXT_SUBELEM"\n" |
| "</submodule>\n"; |
| assert_int_equal(ly_in_new_memory(data, &st->in), LY_SUCCESS); |
| assert_int_equal(lyxml_ctx_new(st->ctx, st->in, &st->yin_ctx->xmlctx), LY_SUCCESS); |
| |
| assert_int_equal(yin_parse_submod(st->yin_ctx, lysp_submod), LY_SUCCESS); |
| assert_string_equal(lysp_submod->name, "mod"); |
| assert_string_equal(lysp_submod->revs, "2019-02-02"); |
| assert_string_equal(lysp_submod->prefix, "pref"); |
| assert_null(lysp_submod->filepath); |
| assert_string_equal(lysp_submod->org, "org"); |
| assert_string_equal(lysp_submod->contact, "contact"); |
| assert_string_equal(lysp_submod->dsc, "desc"); |
| assert_string_equal(lysp_submod->ref, "ref"); |
| assert_int_equal(lysp_submod->version, LYS_VERSION_1_1); |
| assert_string_equal(lysp_submod->imports->name, "a-mod"); |
| assert_string_equal(lysp_submod->includes->name, "b-mod"); |
| assert_string_equal(lysp_submod->extensions->name, "ext"); |
| assert_string_equal(lysp_submod->features->name, "feature"); |
| assert_string_equal(lysp_submod->identities->name, "ident-name"); |
| assert_string_equal(lysp_submod->typedefs->name, "tpdf"); |
| assert_string_equal(lysp_submod->groupings->name, "grp"); |
| assert_string_equal(lysp_submod->data->name, "anyd"); |
| assert_int_equal(lysp_submod->data->nodetype, LYS_ANYDATA); |
| assert_string_equal(lysp_submod->data->next->name, "anyx"); |
| assert_int_equal(lysp_submod->data->next->nodetype, LYS_ANYXML); |
| assert_string_equal(lysp_submod->data->next->next->name, "choice"); |
| assert_int_equal(lysp_submod->data->next->next->nodetype, LYS_CHOICE); |
| assert_string_equal(lysp_submod->data->next->next->next->name, "cont"); |
| assert_int_equal(lysp_submod->data->next->next->next->nodetype, LYS_CONTAINER); |
| assert_string_equal(lysp_submod->data->next->next->next->next->name, "leaf"); |
| assert_int_equal(lysp_submod->data->next->next->next->next->nodetype, LYS_LEAF); |
| assert_string_equal(lysp_submod->data->next->next->next->next->next->name, "llist"); |
| assert_int_equal(lysp_submod->data->next->next->next->next->next->nodetype, LYS_LEAFLIST); |
| assert_string_equal(lysp_submod->data->next->next->next->next->next->next->name, "sub-list"); |
| assert_int_equal(lysp_submod->data->next->next->next->next->next->next->nodetype, LYS_LIST); |
| assert_string_equal(lysp_submod->data->next->next->next->next->next->next->next->name, "uses-name"); |
| assert_int_equal(lysp_submod->data->next->next->next->next->next->next->next->nodetype, LYS_USES); |
| assert_null(lysp_submod->data->next->next->next->next->next->next->next->next); |
| assert_string_equal(lysp_submod->augments->nodeid, "target"); |
| assert_string_equal(lysp_submod->rpcs->name, "rpc-name"); |
| assert_string_equal(lysp_submod->notifs->name, "notf"); |
| assert_string_equal(lysp_submod->deviations->nodeid, "target"); |
| assert_string_equal(lysp_submod->exts[0].name, "urn:example:extensions:c-define"); |
| assert_int_equal(lysp_submod->exts[0].insubstmt_index, 0); |
| assert_int_equal(lysp_submod->exts[0].insubstmt, LYEXT_SUBSTMT_SELF); |
| |
| /* min subelemnts */ |
| ly_in_free(st->in, 0); |
| lyxml_ctx_free(st->yin_ctx->xmlctx); |
| lysp_submod = submod_renew(st->yin_ctx, "module-name"); |
| data = "<submodule xmlns=\"urn:ietf:params:xml:ns:yang:yin:1\" name=\"submod\">" |
| "<yang-version value=\"1\"/>" |
| "<belongs-to module=\"module-name\"><prefix value=\"pref\"/></belongs-to>" |
| "</submodule>"; |
| assert_int_equal(ly_in_new_memory(data, &st->in), LY_SUCCESS); |
| assert_int_equal(lyxml_ctx_new(st->ctx, st->in, &st->yin_ctx->xmlctx), LY_SUCCESS); |
| assert_int_equal(yin_parse_submod(st->yin_ctx, lysp_submod), LY_SUCCESS); |
| assert_string_equal(lysp_submod->prefix, "pref"); |
| assert_int_equal(lysp_submod->version, LYS_VERSION_1_0); |
| |
| /* incorrect subelem order */ |
| ly_in_free(st->in, 0); |
| lyxml_ctx_free(st->yin_ctx->xmlctx); |
| lysp_submod = submod_renew(st->yin_ctx, "module-name"); |
| data = "<submodule xmlns=\"urn:ietf:params:xml:ns:yang:yin:1\" name=\"submod\">" |
| "<yang-version value=\"1\"/>" |
| "<reference><text>ref</text></reference>\n" |
| "<belongs-to module=\"module-name\"><prefix value=\"pref\"/></belongs-to>" |
| "</submodule>"; |
| assert_int_equal(ly_in_new_memory(data, &st->in), LY_SUCCESS); |
| assert_int_equal(lyxml_ctx_new(st->ctx, st->in, &st->yin_ctx->xmlctx), LY_SUCCESS); |
| assert_int_equal(yin_parse_submod(st->yin_ctx, lysp_submod), LY_EVALID); |
| logbuf_assert("Invalid order of submodule's sub-elements \"belongs-to\" can't appear after \"reference\". Line number 2."); |
| |
| st->finished_correctly = true; |
| } |
| |
| static void |
| test_yin_parse_module(void **state) |
| { |
| struct test_parser_yin_state *st = *state; |
| const char *data; |
| struct lys_module *mod; |
| struct lys_yin_parser_ctx *yin_ctx = NULL; |
| struct ly_in *in = NULL; |
| |
| mod = calloc(1, sizeof *mod); |
| mod->ctx = st->ctx; |
| data = "<module xmlns=\"urn:ietf:params:xml:ns:yang:yin:1\" xmlns:md=\"urn:ietf:params:xml:ns:yang:ietf-yang-metadata\" name=\"a\"> \n" |
| "<yang-version value=\"1.1\"/>\n" |
| "<namespace uri=\"urn:tests:extensions:metadata:a\"/>\n" |
| "<prefix value=\"a\"/>\n" |
| "<import module=\"ietf-yang-metadata\">\n" |
| "<prefix value=\"md\"/>\n" |
| "</import>\n" |
| "<feature name=\"f\"/>\n" |
| "<md:annotation name=\"x\">\n" |
| "<description>\n" |
| "<text>test</text>\n" |
| "</description>\n" |
| "<reference>\n" |
| "<text>test</text>\n" |
| "</reference>\n" |
| "<if-feature name=\"f\"/>\n" |
| "<status value=\"current\"/>\n" |
| "<type name=\"uint8\"/>\n" |
| "<units name=\"meters\"/>\n" |
| "</md:annotation>\n" |
| "</module>\n"; |
| assert_int_equal(ly_in_new_memory(data, &in), LY_SUCCESS); |
| assert_int_equal(yin_parse_module(&yin_ctx, in, mod), LY_SUCCESS); |
| assert_null(mod->parsed->exts->child->next->child); |
| assert_string_equal(mod->parsed->exts->child->next->arg, "test"); |
| lys_module_free(mod, NULL); |
| yin_parser_ctx_free(yin_ctx); |
| ly_in_free(in, 0); |
| mod = NULL; |
| yin_ctx = NULL; |
| |
| mod = calloc(1, sizeof *mod); |
| mod->ctx = st->ctx; |
| data = "<module name=\"example-foo\"" |
| "xmlns=\"urn:ietf:params:xml:ns:yang:yin:1\"" |
| "xmlns:foo=\"urn:example:foo\"" |
| "xmlns:myext=\"urn:example:extensions\">\n" |
| |
| "<yang-version value=\"1\"/>\n" |
| |
| "<namespace uri=\"urn:example:foo\"/>\n" |
| "<prefix value=\"foo\"/>\n" |
| |
| "<import module=\"example-extensions\">\n" |
| "<prefix value=\"myext\"/>\n" |
| "</import>\n" |
| |
| "<list name=\"interface\">\n" |
| "<key value=\"name\"/>\n" |
| "<leaf name=\"name\">\n" |
| "<type name=\"string\"/>\n" |
| "</leaf>\n" |
| "<leaf name=\"mtu\">\n" |
| "<type name=\"uint32\"/>\n" |
| "<description>\n" |
| "<text>The MTU of the interface.</text>\n" |
| "</description>\n" |
| "<myext:c-define name=\"MY_MTU\"/>\n" |
| "</leaf>\n" |
| "</list>\n" |
| "</module>\n"; |
| assert_int_equal(ly_in_new_memory(data, &in), LY_SUCCESS); |
| assert_int_equal(yin_parse_module(&yin_ctx, in, mod), LY_SUCCESS); |
| lys_module_free(mod, NULL); |
| yin_parser_ctx_free(yin_ctx); |
| ly_in_free(in, 0); |
| mod = NULL; |
| yin_ctx = NULL; |
| |
| mod = calloc(1, sizeof *mod); |
| mod->ctx = st->ctx; |
| data = "<module name=\"example-foo\" xmlns=\"urn:ietf:params:xml:ns:yang:yin:1\">\n" |
| "<yang-version value=\"1\"/>\n" |
| "<namespace uri=\"urn:example:foo\"/>\n" |
| "<prefix value=\"foo\"/>\n" |
| "</module>\n"; |
| assert_int_equal(ly_in_new_memory(data, &in), LY_SUCCESS); |
| assert_int_equal(yin_parse_module(&yin_ctx, in, mod), LY_SUCCESS); |
| lys_module_free(mod, NULL); |
| yin_parser_ctx_free(yin_ctx); |
| ly_in_free(in, 0); |
| mod = NULL; |
| yin_ctx = NULL; |
| |
| |
| mod = calloc(1, sizeof *mod); |
| mod->ctx = st->ctx; |
| data = "<submodule name=\"example-foo\" xmlns=\"urn:ietf:params:xml:ns:yang:yin:1\">" |
| "</submodule>\n"; |
| assert_int_equal(ly_in_new_memory(data, &in), LY_SUCCESS); |
| assert_int_equal(yin_parse_module(&yin_ctx, in, mod), LY_EINVAL); |
| logbuf_assert("Input data contains submodule which cannot be parsed directly without its main module."); |
| lys_module_free(mod, NULL); |
| yin_parser_ctx_free(yin_ctx); |
| ly_in_free(in, 0); |
| |
| mod = calloc(1, sizeof *mod); |
| mod->ctx = st->ctx; |
| data = "<module name=\"example-foo\" xmlns=\"urn:ietf:params:xml:ns:yang:yin:1\">\n" |
| "<yang-version value=\"1\"/>\n" |
| "<namespace uri=\"urn:example:foo\"/>\n" |
| "<prefix value=\"foo\"/>\n" |
| "</module>" |
| "<module>"; |
| assert_int_equal(ly_in_new_memory(data, &in), LY_SUCCESS); |
| assert_int_equal(yin_parse_module(&yin_ctx, in, mod), LY_EVALID); |
| logbuf_assert("Trailing garbage \"<module>\" after module, expected end-of-input. Line number 5."); |
| lys_module_free(mod, NULL); |
| yin_parser_ctx_free(yin_ctx); |
| ly_in_free(in, 0); |
| mod = NULL; |
| yin_ctx = NULL; |
| |
| st->finished_correctly = true; |
| } |
| |
| static void |
| test_yin_parse_submodule(void **state) |
| { |
| struct test_parser_yin_state *st = *state; |
| const char *data; |
| struct lys_yin_parser_ctx *yin_ctx = NULL; |
| struct lysp_submodule *submod = NULL; |
| struct ly_in *in; |
| |
| lydict_insert(st->ctx, "a", 0, &st->yin_ctx->parsed_mod->mod->name); |
| |
| data = "<?xml version=\"1.0\" encoding=\"UTF-8\"?>" |
| "<submodule name=\"asub\"" |
| "xmlns=\"urn:ietf:params:xml:ns:yang:yin:1\"" |
| "xmlns:a=\"urn:a\">" |
| "<yang-version value=\"1\"/>\n" |
| "<belongs-to module=\"a\">" |
| "<prefix value=\"a_pref\"/>" |
| "</belongs-to>" |
| "<include module=\"atop\"/>" |
| "<feature name=\"fox\"/>" |
| "<notification name=\"bar-notif\">" |
| "<if-feature name=\"bar\"/>" |
| "</notification>" |
| "<notification name=\"fox-notif\">" |
| "<if-feature name=\"fox\"/>" |
| "</notification>" |
| "<augment target-node=\"/a_pref:top\">" |
| "<if-feature name=\"bar\"/>" |
| "<container name=\"bar-sub\"/>" |
| "</augment>" |
| "<augment target-node=\"/top\">" |
| "<container name=\"bar-sub2\"/>" |
| "</augment>" |
| "</submodule>"; |
| assert_int_equal(ly_in_new_memory(data, &in), LY_SUCCESS); |
| assert_int_equal(yin_parse_submodule(&yin_ctx, st->ctx, (struct lys_parser_ctx *)st->yin_ctx, in, &submod), LY_SUCCESS); |
| lysp_module_free((struct lysp_module *)submod); |
| yin_parser_ctx_free(yin_ctx); |
| ly_in_free(in, 0); |
| yin_ctx = NULL; |
| submod = NULL; |
| |
| data = "<?xml version=\"1.0\" encoding=\"UTF-8\"?>" |
| "<submodule name=\"asub\" xmlns=\"urn:ietf:params:xml:ns:yang:yin:1\">" |
| "<yang-version value=\"1\"/>\n" |
| "<belongs-to module=\"a\">" |
| "<prefix value=\"a_pref\"/>" |
| "</belongs-to>" |
| "</submodule>"; |
| assert_int_equal(ly_in_new_memory(data, &in), LY_SUCCESS); |
| assert_int_equal(yin_parse_submodule(&yin_ctx, st->ctx, (struct lys_parser_ctx *)st->yin_ctx, in, &submod), LY_SUCCESS); |
| lysp_module_free((struct lysp_module *)submod); |
| yin_parser_ctx_free(yin_ctx); |
| ly_in_free(in, 0); |
| yin_ctx = NULL; |
| submod = NULL; |
| |
| data = "<?xml version=\"1.0\" encoding=\"UTF-8\"?>" |
| "<module name=\"inval\" xmlns=\"urn:ietf:params:xml:ns:yang:yin:1\">" |
| "</module>"; |
| assert_int_equal(ly_in_new_memory(data, &in), LY_SUCCESS); |
| assert_int_equal(yin_parse_submodule(&yin_ctx, st->ctx, (struct lys_parser_ctx *)st->yin_ctx, in, &submod), LY_EINVAL); |
| logbuf_assert("Input data contains module in situation when a submodule is expected."); |
| lysp_module_free((struct lysp_module *)submod); |
| yin_parser_ctx_free(yin_ctx); |
| ly_in_free(in, 0); |
| yin_ctx = NULL; |
| submod = NULL; |
| |
| data = "<?xml version=\"1.0\" encoding=\"UTF-8\"?>" |
| "<submodule name=\"asub\" xmlns=\"urn:ietf:params:xml:ns:yang:yin:1\">" |
| "<yang-version value=\"1\"/>\n" |
| "<belongs-to module=\"a\">" |
| "<prefix value=\"a_pref\"/>" |
| "</belongs-to>" |
| "</submodule>" |
| "<submodule name=\"asub\" xmlns=\"urn:ietf:params:xml:ns:yang:yin:1\">" |
| "<yang-version value=\"1\"/>\n" |
| "<belongs-to module=\"a\">" |
| "<prefix value=\"a_pref\"/>" |
| "</belongs-to>" |
| "</submodule>"; |
| assert_int_equal(ly_in_new_memory(data, &in), LY_SUCCESS); |
| assert_int_equal(yin_parse_submodule(&yin_ctx, st->ctx, (struct lys_parser_ctx *)st->yin_ctx, in, &submod), LY_EVALID); |
| logbuf_assert("Trailing garbage \"<submodule name...\" after submodule, expected end-of-input. Line number 2."); |
| lysp_module_free((struct lysp_module *)submod); |
| yin_parser_ctx_free(yin_ctx); |
| ly_in_free(in, 0); |
| yin_ctx = NULL; |
| submod = NULL; |
| |
| st->finished_correctly = true; |
| } |
| |
| int |
| main(void) |
| { |
| |
| const struct CMUnitTest tests[] = { |
| cmocka_unit_test_setup_teardown(test_yin_match_keyword, setup_f, teardown_f), |
| cmocka_unit_test_setup_teardown(test_yin_parse_element_generic, setup_f, teardown_f), |
| cmocka_unit_test_setup_teardown(test_yin_parse_extension_instance, setup_f, teardown_f), |
| cmocka_unit_test_setup_teardown(test_yin_parse_content, setup_f, teardown_f), |
| cmocka_unit_test_setup_teardown(test_validate_value, setup_f, teardown_f), |
| |
| cmocka_unit_test(test_yin_match_argument_name), |
| cmocka_unit_test_setup_teardown(test_enum_elem, setup_element_test, teardown_f), |
| cmocka_unit_test_setup_teardown(test_bit_elem, setup_element_test, teardown_f), |
| cmocka_unit_test_setup_teardown(test_meta_elem, setup_element_test, teardown_f), |
| cmocka_unit_test_setup_teardown(test_import_elem, setup_element_test, teardown_f), |
| cmocka_unit_test_setup_teardown(test_status_elem, setup_element_test, teardown_f), |
| cmocka_unit_test_setup_teardown(test_ext_elem, setup_element_test, teardown_f), |
| cmocka_unit_test_setup_teardown(test_yin_element_elem, setup_element_test, teardown_f), |
| cmocka_unit_test_setup_teardown(test_yangversion_elem, setup_element_test, teardown_f), |
| cmocka_unit_test_setup_teardown(test_mandatory_elem, setup_element_test, teardown_f), |
| cmocka_unit_test_setup_teardown(test_argument_elem, setup_element_test, teardown_f), |
| cmocka_unit_test_setup_teardown(test_base_elem, setup_element_test, teardown_f), |
| cmocka_unit_test_setup_teardown(test_belongsto_elem, setup_element_test, teardown_f), |
| cmocka_unit_test_setup_teardown(test_config_elem, setup_element_test, teardown_f), |
| cmocka_unit_test_setup_teardown(test_default_elem, setup_element_test, teardown_f), |
| cmocka_unit_test_setup_teardown(test_err_app_tag_elem, setup_element_test, teardown_f), |
| cmocka_unit_test_setup_teardown(test_err_msg_elem, setup_element_test, teardown_f), |
| cmocka_unit_test_setup_teardown(test_fracdigits_elem, setup_element_test, teardown_f), |
| cmocka_unit_test_setup_teardown(test_iffeature_elem, setup_element_test, teardown_f), |
| cmocka_unit_test_setup_teardown(test_length_elem, setup_element_test, teardown_f), |
| cmocka_unit_test_setup_teardown(test_modifier_elem, setup_element_test, teardown_f), |
| cmocka_unit_test_setup_teardown(test_namespace_elem, setup_element_test, teardown_f), |
| cmocka_unit_test_setup_teardown(test_pattern_elem, setup_element_test, teardown_f), |
| cmocka_unit_test_setup_teardown(test_value_position_elem, setup_element_test, teardown_f), |
| cmocka_unit_test_setup_teardown(test_prefix_elem, setup_element_test, teardown_f), |
| cmocka_unit_test_setup_teardown(test_range_elem, setup_element_test, teardown_f), |
| cmocka_unit_test_setup_teardown(test_reqinstance_elem, setup_element_test, teardown_f), |
| cmocka_unit_test_setup_teardown(test_revision_date_elem, setup_element_test, teardown_f), |
| cmocka_unit_test_setup_teardown(test_unique_elem, setup_element_test, teardown_f), |
| cmocka_unit_test_setup_teardown(test_units_elem, setup_element_test, teardown_f), |
| cmocka_unit_test_setup_teardown(test_when_elem, setup_element_test, teardown_f), |
| cmocka_unit_test_setup_teardown(test_yin_text_value_elem, setup_element_test, teardown_f), |
| cmocka_unit_test_setup_teardown(test_type_elem, setup_element_test, teardown_f), |
| cmocka_unit_test_setup_teardown(test_max_elems_elem, setup_element_test, teardown_f), |
| cmocka_unit_test_setup_teardown(test_min_elems_elem, setup_element_test, teardown_f), |
| cmocka_unit_test_setup_teardown(test_ordby_elem, setup_element_test, teardown_f), |
| cmocka_unit_test_setup_teardown(test_any_elem, setup_element_test, teardown_f), |
| cmocka_unit_test_setup_teardown(test_leaf_elem, setup_element_test, teardown_f), |
| cmocka_unit_test_setup_teardown(test_leaf_list_elem, setup_element_test, teardown_f), |
| cmocka_unit_test_setup_teardown(test_presence_elem, setup_element_test, teardown_f), |
| cmocka_unit_test_setup_teardown(test_key_elem, setup_element_test, teardown_f), |
| cmocka_unit_test_setup_teardown(test_typedef_elem, setup_element_test, teardown_f), |
| cmocka_unit_test_setup_teardown(test_refine_elem, setup_element_test, teardown_f), |
| cmocka_unit_test_setup_teardown(test_uses_elem, setup_element_test, teardown_f), |
| cmocka_unit_test_setup_teardown(test_revision_elem, setup_element_test, teardown_f), |
| cmocka_unit_test_setup_teardown(test_include_elem, setup_element_test, teardown_f), |
| cmocka_unit_test_setup_teardown(test_list_elem, setup_element_test, teardown_f), |
| cmocka_unit_test_setup_teardown(test_notification_elem, setup_element_test, teardown_f), |
| cmocka_unit_test_setup_teardown(test_grouping_elem, setup_element_test, teardown_f), |
| cmocka_unit_test_setup_teardown(test_container_elem, setup_element_test, teardown_f), |
| cmocka_unit_test_setup_teardown(test_case_elem, setup_element_test, teardown_f), |
| cmocka_unit_test_setup_teardown(test_choice_elem, setup_element_test, teardown_f), |
| cmocka_unit_test_setup_teardown(test_inout_elem, setup_element_test, teardown_f), |
| cmocka_unit_test_setup_teardown(test_action_elem, setup_element_test, teardown_f), |
| cmocka_unit_test_setup_teardown(test_augment_elem, setup_element_test, teardown_f), |
| cmocka_unit_test_setup_teardown(test_deviate_elem, setup_element_test, teardown_f), |
| cmocka_unit_test_setup_teardown(test_deviation_elem, setup_element_test, teardown_f), |
| cmocka_unit_test_setup_teardown(test_module_elem, setup_element_test, teardown_f), |
| cmocka_unit_test_setup_teardown(test_submodule_elem, setup_element_test, teardown_f), |
| |
| cmocka_unit_test_setup_teardown(test_yin_parse_module, setup_f, teardown_f), |
| cmocka_unit_test_setup_teardown(test_yin_parse_submodule, setup_f, teardown_f), |
| }; |
| |
| return cmocka_run_group_tests(tests, setup_ly_ctx, destroy_ly_ctx); |
| } |