Skip to content
Snippets Groups Projects
Unverified Commit dbbe6b5b authored by metamben's avatar metamben Committed by GitHub
Browse files

Prevent closing parens with non-matching paren type (#23690)

* Prevent closing parens with non-matching paren type.
* Properly consume early `]]`.
parent dcf306e5
No related branches found
No related tags found
No related merge requests found
......@@ -82,40 +82,38 @@
(params/->Optional parsed))
(s/defn ^:private parse-tokens* :- [(s/one [ParsedToken] "parsed tokens") (s/one [StringOrToken] "remaining tokens")]
[tokens :- [StringOrToken], level :- s/Int]
[tokens :- [StringOrToken], optional-level :- s/Int, param-level :- s/Int]
(loop [acc [], [token & more] tokens]
(condp = token
nil
(if (pos? level)
(if (or (pos? optional-level) (pos? param-level))
(throw (ex-info (tru "Invalid query: found '[[' or '{{' with no matching ']]' or '}}'")
{:type qp.error-type/invalid-query}))
{:type qp.error-type/invalid-query}))
[acc nil])
:optional-begin
(let [[parsed more] (parse-tokens* more (inc level))]
(let [[parsed more] (parse-tokens* more (inc optional-level) param-level)]
(recur (conj acc (apply optional parsed)) more))
:param-begin
(let [[parsed more] (parse-tokens* more (inc level))]
(let [[parsed more] (parse-tokens* more optional-level (inc param-level))]
(recur (conj acc (apply param parsed)) more))
:optional-end
(if (pos? level)
(if (pos? optional-level)
[acc more]
[(conj acc "]]" more)])
(recur (conj acc "]]") more))
:param-end
(if (pos? level)
(if (pos? param-level)
[acc more]
[(conj acc "}}") more])
(recur (conj acc "}}") more))
(recur (conj acc token) more))))
(s/defn ^:private parse-tokens :- [ParsedToken]
[tokens :- [StringOrToken]]
(let [[parsed remaining] (parse-tokens* tokens 0)
parsed (concat parsed (when (seq remaining)
(parse-tokens remaining)))]
(let [parsed (first (parse-tokens* tokens 0 0))]
;; now loop over everything in `parsed`, and if we see 2 strings next to each other put them back together
;; e.g. [:token "x" "}}"] -> [:token "x}}"]
(loop [acc [], last (first parsed), [x & more] (rest parsed)]
......
......@@ -76,7 +76,11 @@
"JSON queries that contain non-param fragments like '}}'"
{"{x: {y: \"{{param}}\"}}" ["{x: {y: \"" (param "param") "\"}}"]
"{$match: {{{date}}, field: 1}}}" ["{$match: {" (param "date") ", field: 1}}}"]}}]
"{$match: {{{date}}, field: 1}}}" ["{$match: {" (param "date") ", field: 1}}}"]}
"Queries that contain non-param fragments like '}}'"
{"select ']]' from t [[where x = {{foo}}]]" ["select ']]' from t " (optional "where x = " (param "foo"))]
"select '}}' from t [[where x = {{foo}}]]" ["select '}}' from t " (optional "where x = " (param "foo"))]}}]
(testing group
(doseq [[s expected] s->expected]
(is (= expected
......@@ -87,7 +91,9 @@
(doseq [invalid ["select * from foo [[where bar = {{baz}} "
"select * from foo [[where bar = {{baz]]"
"select * from foo {{bar}} {{baz"
"select * from foo [[clause 1 {{bar}}]] [[clause 2"]]
"select * from foo [[clause 1 {{bar}}]] [[clause 2"
"select * from foo where bar = {{baz]]"
"select * from foo [[where bar = {{baz}}}}"]]
(is (thrown? clojure.lang.ExceptionInfo
(params.parse/parse invalid))
(format "Parsing %s should throw an exception" (pr-str invalid))))))
0% Loading or .
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment