我正在尝试使用 Instaparse 建立语法。我经常发现这段代码在第一个断言失败,发出“空列表”:
(defn parse-it []
(let [parser (insta/parser ebnf)
res (insta/parses parser input)
_ (assert (seq res) (str "Empty list"))
choices (count res)
_ (assert (= choices 1))]
(first res)))
我总是解决问题,但这需要反复试验。有什么方法可以查明错误吗?
解决问题的一个示例是从input
上述代码中的文件中删除尾随空格。
编辑
根据 Stefan 的回答,我更改了代码:
(defn parse-it []
(let [my-parser (insta/parser ebnf)
xs (insta/parses my-parser input)
num-choices (count xs)
msg (cond
(zero? num-choices) "insta/parses might be able to be corrected"
(> num-choices 1) (str "insta/parses shows more than one way to parse: " num-choices)
(= 1 num-choices) "insta/parses shows one choice, so all good")
res (cond
;; Just fix there being more than one rather than show them all
;(> num-choices 1) xs
(= num-choices 1) (first xs)
(zero? num-choices) (insta/parse my-parser input))
_ (assert res (str "No result. Num of choices is: " num-choices))]
[msg res]))
上面的代码可以解决问题:总是得到一个准确的答案。对我来说,insta/parses
返回一个空列表后,insta/parse
需要调用以获取错误信息并不是那么明显。使用解析错误文档将产生比上面更好的代码。它显示了错误信息实际上是如何存在于元数据中的,以及如何检索它 - 这个问题的答案已经在文档中!