diff options
author | Vsevolod Stakhov <vsevolod@highsecure.ru> | 2018-11-12 16:36:57 +0000 |
---|---|---|
committer | Vsevolod Stakhov <vsevolod@highsecure.ru> | 2018-11-12 16:36:57 +0000 |
commit | 557217a588880afd9a52d2a617470c7fe716f451 (patch) | |
tree | 7ab8e048b011327c02f0e3807f1b6481a4b99dd7 | |
parent | e0f8a1fc93fb5c012cf3453f8397fb14db10eb8e (diff) | |
download | rspamd-557217a588880afd9a52d2a617470c7fe716f451.tar.gz rspamd-557217a588880afd9a52d2a617470c7fe716f451.zip |
[Fix] Do not allow to merge an object with an array (or vice versa)
Issue: #2613
-rw-r--r-- | contrib/libucl/ucl_parser.c | 36 |
1 files changed, 33 insertions, 3 deletions
diff --git a/contrib/libucl/ucl_parser.c b/contrib/libucl/ucl_parser.c index c4b390407..574e240ab 100644 --- a/contrib/libucl/ucl_parser.c +++ b/contrib/libucl/ucl_parser.c @@ -89,6 +89,7 @@ ucl_set_err (struct ucl_parser *parser, int code, const char *str, UT_string **e } parser->err_code = code; + parser->state = UCL_STATE_ERROR; } static void @@ -633,14 +634,26 @@ ucl_parser_add_container (ucl_object_t *obj, struct ucl_parser *parser, bool is_array, uint32_t level, bool has_obrace) { struct ucl_stack *st; + bool need_free = false; if (!is_array) { if (obj == NULL) { obj = ucl_object_new_full (UCL_OBJECT, parser->chunks->priority); + need_free = true; } else { + if (obj->type == UCL_ARRAY) { + /* Bad combination for merge: array and object */ + ucl_set_err (parser, UCL_EMERGE, + "cannot merge an array with an object", + &parser->err); + + return NULL; + } + obj->type = UCL_OBJECT; } + if (obj->value.ov == NULL) { obj->value.ov = ucl_hash_create (parser->flags & UCL_PARSER_KEY_LOWERCASE); } @@ -649,8 +662,18 @@ ucl_parser_add_container (ucl_object_t *obj, struct ucl_parser *parser, else { if (obj == NULL) { obj = ucl_object_new_full (UCL_ARRAY, parser->chunks->priority); + need_free = true; } else { + if (obj->type == UCL_OBJECT) { + /* Bad combination for merge: array and object */ + ucl_set_err (parser, UCL_EMERGE, + "cannot merge an object with an array", + &parser->err); + + return NULL; + } + obj->type = UCL_ARRAY; } parser->state = UCL_STATE_VALUE; @@ -661,7 +684,10 @@ ucl_parser_add_container (ucl_object_t *obj, struct ucl_parser *parser, if (st == NULL) { ucl_set_err (parser, UCL_EINTERNAL, "cannot allocate memory for an object", &parser->err); - ucl_object_unref (obj); + if (need_free) { + ucl_object_unref (obj); + } + return NULL; } @@ -671,7 +697,10 @@ ucl_parser_add_container (ucl_object_t *obj, struct ucl_parser *parser, ucl_set_err (parser, UCL_ENESTED, "objects are nesting too deep (over 65535 limit)", &parser->err); - ucl_object_unref (obj); + if (need_free) { + ucl_object_unref (obj); + } + return NULL; } @@ -2404,6 +2433,7 @@ ucl_state_machine (struct ucl_parser *parser) parser->state = UCL_STATE_ERROR; return false; } + if (end_of_object) { p = chunk->pos; parser->state = UCL_STATE_AFTER_VALUE; @@ -2610,7 +2640,7 @@ ucl_state_machine (struct ucl_parser *parser) } } - if (parser->stack != NULL) { + if (parser->stack != NULL && parser->state != UCL_STATE_ERROR) { struct ucl_stack *st; bool has_error = false; |