-
Notifications
You must be signed in to change notification settings - Fork 1
/
parse.clj
76 lines (70 loc) · 2.68 KB
/
parse.clj
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
; Copyright 2024 Vivid Inc. and/or its affiliates.
;
; Licensed under the Apache License, Version 2.0 (the "License")
; you may not use this file except in compliance with the License.
; You may obtain a copy of the License at
;
; https://www.apache.org/licenses/LICENSE-2.0
;
; Unless required by applicable law or agreed to in writing, software
; distributed under the License is distributed on an "AS IS" BASIS,
; WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
; See the License for the specific language governing permissions and
; limitations under the License.
(ns ^:internal-api vivid.art.parse
(:require
[clojure.spec.alpha :as s]
[clojure.string :as str]
[farolero.core :as farolero]
[instaparse.core :as insta]
[vivid.art.specs])
(:import
(java.util.regex Pattern)))
(defn make-grammar ; TODO Enforce delimiter rules.
[delimiters]
(let [q #(Pattern/quote %)
alts (str/join " | " (map name (keys delimiters)))
terminals (str/join
(interleave (map (fn [[k v]] (format "%s = '%s'"
(name k) v))
delimiters)
(repeat "\n")))
non-terms (str/join "|"
(map q (vals delimiters)))]
(str "s = (" alts " | content)*\n"
terminals
"content = #'(?s)(?:(?!" non-terms ").)*'")))
(def ^:const tree-transformation
{;; Strip the grammar starting rule from the token stream
:s (fn [& xs] xs)
;; Template tokens appear as namespaced keywords
:begin-eval (fn [_] :vivid.art/begin-eval)
:begin-forms (fn [_] :vivid.art/begin-forms)
:end-eval (fn [_] :vivid.art/end-eval)
:end-forms (fn [_] :vivid.art/end-forms)
;; Inline string content
:content str})
(defn confirm-parse-output
"Either raise a condition on parse error, or allow
parse-result to pass through."
[parse-result]
(if (insta/failure? parse-result)
(farolero/signal :vivid.art/parse-error {:instaparse (insta/get-failure parse-result)})
parse-result))
(defn parse
"Tokenize a template string into a sequence of tokens suitable for parsing.
Each lexeme is any of:
- Template content, as a string.
- Template delimiter, as a Clojure keyword."
[^String template-str
delimiters]
(let [grammar (make-grammar delimiters)
parser (insta/parser grammar)]
(->> template-str
(insta/parse parser)
(confirm-parse-output)
(insta/transform tree-transformation))))
(s/fdef parse
:args (s/cat :template-str string?
:delimiters :vivid.art/delimiters)
:ret seq?)