diff options
author | Kaz Kylheku <kaz@kylheku.com> | 2021-05-28 05:43:42 -0700 |
---|---|---|
committer | Kaz Kylheku <kaz@kylheku.com> | 2021-05-28 05:43:42 -0700 |
commit | c0c5e8836a89c8439a675bbd52d6fed134792477 (patch) | |
tree | 1ab3a2b9301c3d72fac44d1315d5b529d3b259ef /parser.y | |
parent | b2bedb3c0eae02198678732578db4fa0c0a89fd7 (diff) | |
download | txr-c0c5e8836a89c8439a675bbd52d6fed134792477.tar.gz txr-c0c5e8836a89c8439a675bbd52d6fed134792477.tar.bz2 txr-c0c5e8836a89c8439a675bbd52d6fed134792477.zip |
json: hash issues with quasiquoting.
There are two problems.
One is that in #J{~foo:"bar"} the foo expression is parsed
while the scanner is in Lisp mode and the : token is grabbed
by the parser as a lookahead token in the same mode. Thus it
comes back as a SYMTOK (the colon symbol).
We fix this by recognizing the colon symbol as synonym of
the colon character. But only if it is spelled ":" in
the syntax: we look at the lexeme.
The second problem is that even if we fix the above, ~foo
produces a sys:unquote form which is rejected because it is
not a string. We fix this in the most straightforward way:
by deleting the code which restricts keys to strings,
an extension I already thought about making anyway.
* parser.y (json_col): New non-terminal. This recognizes
either a SYMTOK or a ':' token, with the semantic restriction
that the SYMTOK's lexeme must be ":".
(json_vals): Use yybadtok instead of generic error.
(json_pairs): Do not require keys to be strings. Use json_col
instead of ':', and also yybadtok instead of generic error.
* y.tab.c.shipped: Updated.
Diffstat (limited to 'parser.y')
-rw-r--r-- | parser.y | 46 |
1 files changed, 25 insertions, 21 deletions
@@ -140,7 +140,7 @@ INLINE val expand_form_ver(val form, int ver) %type <val> line elems_opt elems clause_parts_h additional_parts_h %type <val> text texts elem var var_op modifiers %type <val> vector hash struct range tnode tree -%type <val> json json_val json_vals json_pairs +%type <val> json json_val json_vals json_pairs json_col %type <val> exprs exprs_opt n_exprs listacc i_expr i_dot_expr %type <val> n_expr n_exprs_opt n_dot_expr %type <val> list dwim meta compound @@ -996,13 +996,11 @@ json_vals : json_val { $$ = if3(parser->quasi_level > 0 && $$ = $1; } } | json_vals json_val { yyerr("missing comma in JSON array"); $$ = $1; } - | json_vals error { yyerr("bad element in JSON array"); + | json_vals error { yybadtok(yychar, lit("JSON array")); $$ = $1; } ; -json_pairs : json_val ':' json_val { if (!stringp($1)) - yyerr("non-string key in JSON hash"); - if (parser->quasi_level > 0 && +json_pairs : json_val json_col json_val { if (parser->quasi_level > 0 && (unquotes_occur($1, 0) || unquotes_occur($3, 0))) { $$ = cons(list($1, $3, nao), nil); } @@ -1010,24 +1008,30 @@ json_pairs : json_val ':' json_val { if (!stringp($1)) { $$ = make_hash(nil, nil, t); sethash($$, $1, $3); } } | json_pairs ',' - json_val ':' json_val { if (!stringp($3)) - yyerr("non-string key in JSON hash"); - if (consp($1)) - { $$ = cons(list($3, $5, nao), $1); } - else if (parser->quasi_level > 0 && - ((unquotes_occur($3, 0)) || - unquotes_occur($5, 0))) - { val pa = hash_pairs($1); - $$ = cons(list($3, $5, nao), pa); } - else - { sethash($1, $3, $5); - $$ = $1; } } - | json_val json_val { yyerr("missing colon in JSON hash"); } - | json_val ':' json_val - error { yyerr("missing comma in JSON hash"); } - | json_val error { yyerr("bad element in JSON hash"); } + json_val json_col json_val { if (consp($1)) + { $$ = cons(list($3, $5, nao), $1); } + else if (parser->quasi_level > 0 && + ((unquotes_occur($3, 0)) || + unquotes_occur($5, 0))) + { val pa = hash_pairs($1); + $$ = cons(list($3, $5, nao), pa); } + else + { sethash($1, $3, $5); + $$ = $1; } } + | json_val json_val { yyerr("missing colon in JSON hash"); } + | json_pairs json_val + error { yyerr("missing comma in JSON hash"); } + | json_val error { yybadtok(yychar, lit("JSON hash")); } ; +json_col : SYMTOK { if ($1[0] == ':' && $1[1] == 0) + { $$ = nil; } + else + { yybadtok(yychar, lit("JSON hash")); } } + | ':' { $$ = nil; } + ; + + list : '(' n_exprs ')' { $$ = rl($2, num($1)); } | '(' '.' n_exprs ')' { val a = car($3); val ur = uref_helper(parser, a); |