Skip to content

Commit 6826737

Browse files
committed
Add ClojureScript support
Increases minimum Clojure version to 1.7 for reader conditionals, and updates Instaparse to 1.4.8 for ClojureScript support.
1 parent 5b943e7 commit 6826737

File tree

6 files changed

+196
-61
lines changed

6 files changed

+196
-61
lines changed

.travis.yml

+10-5
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,11 @@
11
language: clojure
2-
lein: lein2
3-
script: lein2 with-profile dev:dev,1.4:dev,1.5:dev,1.6:dev,1.7 test
4-
jdk:
5-
- openjdk7
6-
- openjdk6
2+
script:
3+
- jdk_switcher use openjdk6
4+
- lein with-profile dev:dev,1.7:dev,1.8:dev,1.9 test
5+
- lein with-profile 1.7:1.8 doo node once
6+
- jdk_switcher use openjdk7
7+
- lein with-profile dev:dev,1.7:dev,1.8:dev,1.9 test
8+
- lein with-profile 1.7:1.8 doo node once
9+
- jdk_switcher use openjdk8
10+
- lein with-profile dev:dev,1.7:dev,1.8:dev,1.9 test
11+
- lein with-profile 1.7:1.8:1.9 doo node once

project.clj

+22-6
Original file line numberDiff line numberDiff line change
@@ -2,12 +2,28 @@
22
:description "A HTTP route matching library"
33
:url "https://github.com/weavejester/clout"
44
:license {:name "Eclipse Public License"
5-
:url "http://www.eclipse.org/legal/epl-v10.html"}
6-
:dependencies [[org.clojure/clojure "1.5.1"]
7-
[instaparse "1.4.0" :exclusions [org.clojure/clojure]]]
5+
:url "http://www.eclipse.org/legal/epl-v10.html"}
6+
:dependencies [[org.clojure/clojure "1.7.0"]
7+
[org.clojure/clojurescript "1.7.228" :scope "provided"]
8+
[instaparse "1.4.8" :exclusions [org.clojure/clojure]]]
9+
:plugins [[lein-doo "0.1.7"]
10+
[lein-cljsbuild "1.1.4"]]
11+
:cljsbuild {:builds
12+
{:test
13+
{:source-paths ["src" "test"]
14+
:compiler {:main clout.test-runner
15+
:output-dir "target/out"
16+
:output-to "target/test/advanced.js"
17+
:target :nodejs
18+
:optimizations :advanced}}}}
19+
:doo {:build "test"}
820
:profiles
9-
{:dev {:jvm-opts ^:replace []
21+
{:dev {:jvm-opts ^:replace []
1022
:dependencies [[ring/ring-mock "0.2.0"]
1123
[criterium "0.4.2"]]}
12-
:1.6 {:dependencies [[org.clojure/clojure "1.6.0"]]}
13-
:1.7 {:dependencies [[org.clojure/clojure "1.7.0-beta2"]]}})
24+
:1.7 {:dependencies [[org.clojure/clojure "1.7.0"]
25+
[org.clojure/clojurescript "1.7.228"]]}
26+
:1.8 {:dependencies [[org.clojure/clojure "1.8.0"]
27+
[org.clojure/clojurescript "1.8.51"]]}
28+
:1.9 {:dependencies [[org.clojure/clojure "1.9.0"]
29+
[org.clojure/clojurescript "1.9.946"]]}})

src/clout/core.clj src/clout/core.cljc

+53-41
Original file line numberDiff line numberDiff line change
@@ -2,24 +2,36 @@
22
"A small language for routing."
33
(:require [clojure.string :as string]
44
[clojure.set :as set]
5-
[instaparse.core :as insta]))
5+
[instaparse.core :as insta]
6+
#?(:cljs [clout.internal.regex :refer [re-unicode-letter]])))
67

78
(def ^:private re-chars (set "\\.*+|?()[]{}$^"))
9+
#?(:cljs (def ^:private re-char-escapes
10+
(into {} (for [c re-chars] [c (str "\\" c)]))))
811

912
(defn- re-escape [s]
10-
(string/escape s #(if (re-chars %) (str \\ %))))
11-
12-
(defn- re-groups* [^java.util.regex.Matcher matcher]
13-
(for [i (range (.groupCount matcher))]
14-
(.group matcher (int (inc i)))))
13+
(string/escape s #?(:clj #(if (re-chars %) (str \\ %)) :cljs re-char-escapes)))
14+
15+
(defn- re-match-groups [re s]
16+
#?(:clj
17+
(let [matcher (re-matcher re s)]
18+
(when (.matches matcher)
19+
(for [i (range (.groupCount matcher))]
20+
(.group matcher (int (inc i))))))
21+
:cljs
22+
(if (string? s)
23+
(let [matches (.exec re s)]
24+
(when (= (first matches) s)
25+
(rest matches)))
26+
(throw (js/TypeError. "re-match-groups must match against string.")))))
1527

1628
(defn- assoc-conj [m k v]
1729
(assoc m k
18-
(if-let [cur (get m k)]
19-
(if (vector? cur)
20-
(conj cur v)
21-
[cur v])
22-
v)))
30+
(if-let [cur (get m k)]
31+
(if (vector? cur)
32+
(conj cur v)
33+
[cur v])
34+
v)))
2335

2436
(defn- assoc-keys-with-groups [groups keys]
2537
(reduce (fn [m [k v]] (assoc-conj m k v))
@@ -46,26 +58,26 @@
4658
(route-matches [_ request]
4759
(let [path-info (if absolute?
4860
(request-url request)
49-
(path-info request))
50-
matcher (re-matcher re path-info)]
51-
(if (.matches matcher)
52-
(assoc-keys-with-groups (re-groups* matcher) keys))))
53-
Object
54-
(toString [_] source))
61+
(path-info request))]
62+
(let [groups (re-match-groups re path-info)]
63+
(when groups
64+
(assoc-keys-with-groups groups keys)))))
65+
#?@(:clj [Object
66+
(toString [_] source)]))
5567

5668
(def ^:private route-parser
5769
(insta/parser
58-
"route = (scheme / part) part*
59-
scheme = #'(https?:)?//'
60-
61-
<part> = literal | escaped | wildcard | param
62-
literal = #'(:[^\\p{L}_*{}\\\\]|[^:*{}\\\\])+'
63-
escaped = #'\\\\.'
64-
wildcard = '*'
65-
66-
param = key pattern?
67-
key = <':'> #'([\\p{L}_][\\p{L}_0-9-]*)'
68-
pattern = '{' (#'(?:[^{}\\\\]|\\\\.)+' | pattern)* '}'"
70+
(let [letter #?(:clj "\\p{L}" :cljs re-unicode-letter)]
71+
(str
72+
"route = (scheme / part) part*
73+
scheme = #'(https?:)?//'
74+
<part> = literal | escaped | wildcard | param
75+
literal = #'(:[^" letter "_*{}\\\\]|[^:*{}\\\\])+'
76+
escaped = #'\\\\.'
77+
wildcard = '*'
78+
param = key pattern?
79+
key = <':'> #'([" letter "_][" letter "_0-9-]*)'
80+
pattern = '{' (#'(?:[^{}\\\\]|\\\\.)+' | pattern)* '}'"))
6981
:no-slurp true))
7082

7183
(defn- parse [parser text]
@@ -92,7 +104,7 @@
92104

93105
(defn- route-regex [parse-tree regexs]
94106
(insta/transform
95-
{:route (comp re-pattern str)
107+
{:route (comp re-pattern #?(:cljs #(str "^" % "$")) str)
96108
:scheme #(if (= % "//") "https?://" %)
97109
:literal re-escape
98110
:escaped #(re-escape (subs % 1))
@@ -108,19 +120,19 @@
108120
(defn route-compile
109121
"Compile a route string for more efficient route matching."
110122
([path]
111-
(route-compile path {}))
123+
(route-compile path {}))
112124
([path regexs]
113-
(let [ast (parse route-parser path)
114-
ks (route-keys ast)]
115-
(assert (set/subset? (set (keys regexs)) (set ks))
116-
"unused keys in regular expression map")
117-
(CompiledRoute.
118-
path
119-
(route-regex ast regexs)
120-
(vec ks)
121-
(absolute-url? path)))))
122-
123-
(extend-type String
125+
(let [ast (parse route-parser path)
126+
ks (route-keys ast)]
127+
(assert (set/subset? (set (keys regexs)) (set ks))
128+
"unused keys in regular expression map")
129+
(CompiledRoute.
130+
path
131+
(route-regex ast regexs)
132+
(vec ks)
133+
(absolute-url? path)))))
134+
135+
(extend-type #?(:clj String :cljs string)
124136
Route
125137
(route-matches [route request]
126138
(route-matches (route-compile route) request)))

src/clout/internal/regex.cljs

+72
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,72 @@
1+
(ns ^:no-doc clout.internal.regex)
2+
3+
(def re-unicode-letter
4+
(str
5+
"A-Za-z\\xAA\\xB5\\xBA\\xC0-\\xD6\\xD8-\\xF6\\xF8-\\u02C1\\u02C6-\\u02D1"
6+
"\\u02E0-\\u02E4\\u02EC\\u02EE\\u0370-\\u0374\\u0376\\u0377\\u037A-\\u037D"
7+
"\\u0386\\u0388-\\u038A\\u038C\\u038E-\\u03A1\\u03A3-\\u03F5\\u03F7-\\u0481"
8+
"\\u048A-\\u0527\\u0531-\\u0556\\u0559\\u0561-\\u0587\\u05D0-\\u05EA\\u05F0"
9+
"-\\u05F2\\u0620-\\u064A\\u066E\\u066F\\u0671-\\u06D3\\u06D5\\u06E5\\u06E6"
10+
"\\u06EE\\u06EF\\u06FA-\\u06FC\\u06FF\\u0710\\u0712-\\u072F\\u074D-\\u07A5"
11+
"\\u07B1\\u07CA-\\u07EA\\u07F4\\u07F5\\u07FA\\u0800-\\u0815\\u081A\\u0824"
12+
"\\u0828\\u0840-\\u0858\\u08A0\\u08A2-\\u08AC\\u0904-\\u0939\\u093D\\u0950"
13+
"\\u0958-\\u0961\\u0971-\\u0977\\u0979-\\u097F\\u0985-\\u098C\\u098F\\u0990"
14+
"\\u0993-\\u09A8\\u09AA-\\u09B0\\u09B2\\u09B6-\\u09B9\\u09BD\\u09CE\\u09DC"
15+
"\\u09DD\\u09DF-\\u09E1\\u09F0\\u09F1\\u0A05-\\u0A0A\\u0A0F\\u0A10\\u0A13-"
16+
"\\u0A28\\u0A2A-\\u0A30\\u0A32\\u0A33\\u0A35\\u0A36\\u0A38\\u0A39\\u0A59-"
17+
"\\u0A5C\\u0A5E\\u0A72-\\u0A74\\u0A85-\\u0A8D\\u0A8F-\\u0A91\\u0A93-\\u0AA8"
18+
"\\u0AAA-\\u0AB0\\u0AB2\\u0AB3\\u0AB5-\\u0AB9\\u0ABD\\u0AD0\\u0AE0\\u0AE1"
19+
"\\u0B05-\\u0B0C\\u0B0F\\u0B10\\u0B13-\\u0B28\\u0B2A-\\u0B30\\u0B32\\u0B33"
20+
"\\u0B35-\\u0B39\\u0B3D\\u0B5C\\u0B5D\\u0B5F-\\u0B61\\u0B71\\u0B83\\u0B85-"
21+
"\\u0B8A\\u0B8E-\\u0B90\\u0B92-\\u0B95\\u0B99\\u0B9A\\u0B9C\\u0B9E\\u0B9F"
22+
"\\u0BA3\\u0BA4\\u0BA8-\\u0BAA\\u0BAE-\\u0BB9\\u0BD0\\u0C05-\\u0C0C\\u0C0E-"
23+
"\\u0C10\\u0C12-\\u0C28\\u0C2A-\\u0C33\\u0C35-\\u0C39\\u0C3D\\u0C58\\u0C59"
24+
"\\u0C60\\u0C61\\u0C85-\\u0C8C\\u0C8E-\\u0C90\\u0C92-\\u0CA8\\u0CAA-\\u0CB3"
25+
"\\u0CB5-\\u0CB9\\u0CBD\\u0CDE\\u0CE0\\u0CE1\\u0CF1\\u0CF2\\u0D05-\\u0D0C"
26+
"\\u0D0E-\\u0D10\\u0D12-\\u0D3A\\u0D3D\\u0D4E\\u0D60\\u0D61\\u0D7A-\\u0D7F"
27+
"\\u0D85-\\u0D96\\u0D9A-\\u0DB1\\u0DB3-\\u0DBB\\u0DBD\\u0DC0-\\u0DC6\\u0E01"
28+
"-\\u0E30\\u0E32\\u0E33\\u0E40-\\u0E46\\u0E81\\u0E82\\u0E84\\u0E87\\u0E88"
29+
"\\u0E8A\\u0E8D\\u0E94-\\u0E97\\u0E99-\\u0E9F\\u0EA1-\\u0EA3\\u0EA5\\u0EA7"
30+
"\\u0EAA\\u0EAB\\u0EAD-\\u0EB0\\u0EB2\\u0EB3\\u0EBD\\u0EC0-\\u0EC4\\u0EC6"
31+
"\\u0EDC-\\u0EDF\\u0F00\\u0F40-\\u0F47\\u0F49-\\u0F6C\\u0F88-\\u0F8C\\u1000"
32+
"-\\u102A\\u103F\\u1050-\\u1055\\u105A-\\u105D\\u1061\\u1065\\u1066\\u106E-"
33+
"\\u1070\\u1075-\\u1081\\u108E\\u10A0-\\u10C5\\u10C7\\u10CD\\u10D0-\\u10FA"
34+
"\\u10FC-\\u1248\\u124A-\\u124D\\u1250-\\u1256\\u1258\\u125A-\\u125D\\u1260"
35+
"-\\u1288\\u128A-\\u128D\\u1290-\\u12B0\\u12B2-\\u12B5\\u12B8-\\u12BE"
36+
"\\u12C0\\u12C2-\\u12C5\\u12C8-\\u12D6\\u12D8-\\u1310\\u1312-\\u1315\\u1318"
37+
"-\\u135A\\u1380-\\u138F\\u13A0-\\u13F4\\u1401-\\u166C\\u166F-\\u167F"
38+
"\\u1681-\\u169A\\u16A0-\\u16EA\\u1700-\\u170C\\u170E-\\u1711\\u1720-"
39+
"\\u1731\\u1740-\\u1751\\u1760-\\u176C\\u176E-\\u1770\\u1780-\\u17B3\\u17D7"
40+
"\\u17DC\\u1820-\\u1877\\u1880-\\u18A8\\u18AA\\u18B0-\\u18F5\\u1900-\\u191C"
41+
"\\u1950-\\u196D\\u1970-\\u1974\\u1980-\\u19AB\\u19C1-\\u19C7\\u1A00-"
42+
"\\u1A16\\u1A20-\\u1A54\\u1AA7\\u1B05-\\u1B33\\u1B45-\\u1B4B\\u1B83-\\u1BA0"
43+
"\\u1BAE\\u1BAF\\u1BBA-\\u1BE5\\u1C00-\\u1C23\\u1C4D-\\u1C4F\\u1C5A-\\u1C7D"
44+
"\\u1CE9-\\u1CEC\\u1CEE-\\u1CF1\\u1CF5\\u1CF6\\u1D00-\\u1DBF\\u1E00-\\u1F15"
45+
"\\u1F18-\\u1F1D\\u1F20-\\u1F45\\u1F48-\\u1F4D\\u1F50-\\u1F57\\u1F59\\u1F5B"
46+
"\\u1F5D\\u1F5F-\\u1F7D\\u1F80-\\u1FB4\\u1FB6-\\u1FBC\\u1FBE\\u1FC2-\\u1FC4"
47+
"\\u1FC6-\\u1FCC\\u1FD0-\\u1FD3\\u1FD6-\\u1FDB\\u1FE0-\\u1FEC\\u1FF2-"
48+
"\\u1FF4\\u1FF6-\\u1FFC\\u2071\\u207F\\u2090-\\u209C\\u2102\\u2107\\u210A-"
49+
"\\u2113\\u2115\\u2119-\\u211D\\u2124\\u2126\\u2128\\u212A-\\u212D\\u212F-"
50+
"\\u2139\\u213C-\\u213F\\u2145-\\u2149\\u214E\\u2183\\u2184\\u2C00-\\u2C2E"
51+
"\\u2C30-\\u2C5E\\u2C60-\\u2CE4\\u2CEB-\\u2CEE\\u2CF2\\u2CF3\\u2D00-"
52+
"\\u2D25\\u2D27\\u2D2D\\u2D30-\\u2D67\\u2D6F\\u2D80-\\u2D96\\u2DA0-\\u2DA6"
53+
"\\u2DA8-\\u2DAE\\u2DB0-\\u2DB6\\u2DB8-\\u2DBE\\u2DC0-\\u2DC6\\u2DC8-"
54+
"\\u2DCE\\u2DD0-\\u2DD6\\u2DD8-\\u2DDE\\u2E2F\\u3005\\u3006\\u3031-\\u3035"
55+
"\\u303B\\u303C\\u3041-\\u3096\\u309D-\\u309F\\u30A1-\\u30FA\\u30FC-"
56+
"\\u30FF\\u3105-\\u312D\\u3131-\\u318E\\u31A0-\\u31BA\\u31F0-\\u31FF"
57+
"\\u3400-\\u4DB5\\u4E00-\\u9FCC\\uA000-\\uA48C\\uA4D0-\\uA4FD\\uA500-"
58+
"\\uA60C\\uA610-\\uA61F\\uA62A\\uA62B\\uA640-\\uA66E\\uA67F-\\uA697\\uA6A0-"
59+
"\\uA6E5\\uA717-\\uA71F\\uA722-\\uA788\\uA78B-\\uA78E\\uA790-\\uA793"
60+
"\\uA7A0-\\uA7AA\\uA7F8-\\uA801\\uA803-\\uA805\\uA807-\\uA80A\\uA80C-"
61+
"\\uA822\\uA840-\\uA873\\uA882-\\uA8B3\\uA8F2-\\uA8F7\\uA8FB\\uA90A-"
62+
"\\uA925\\uA930-\\uA946\\uA960-\\uA97C\\uA984-\\uA9B2\\uA9CF\\uAA00-"
63+
"\\uAA28\\uAA40-\\uAA42\\uAA44-\\uAA4B\\uAA60-\\uAA76\\uAA7A\\uAA80-\\uAAAF"
64+
"\\uAAB1\\uAAB5\\uAAB6\\uAAB9-\\uAABD\\uAAC0\\uAAC2\\uAADB-\\uAADD\\uAAE0-"
65+
"\\uAAEA\\uAAF2-\\uAAF4\\uAB01-\\uAB06\\uAB09-\\uAB0E\\uAB11-\\uAB16"
66+
"\\uAB20-\\uAB26\\uAB28-\\uAB2E\\uABC0-\\uABE2\\uAC00-\\uD7A3\\uD7B0-"
67+
"\\uD7C6\\uD7CB-\\uD7FB\\uF900-\\uFA6D\\uFA70-\\uFAD9\\uFB00-\\uFB06\\uFB13"
68+
"-\\uFB17\\uFB1D\\uFB1F-\\uFB28\\uFB2A-\\uFB36\\uFB38-\\uFB3C\\uFB3E\\uFB40"
69+
"\\uFB41\\uFB43\\uFB44\\uFB46-\\uFBB1\\uFBD3-\\uFD3D\\uFD50-\\uFD8F\\uFD92-"
70+
"\\uFDC7\\uFDF0-\\uFDFB\\uFE70-\\uFE74\\uFE76-\\uFEFC\\uFF21-\\uFF3A\\uFF41"
71+
"-\\uFF5A\\uFF66-\\uFFBE\\uFFC2-\\uFFC7\\uFFCA-\\uFFCF\\uFFD2-\\uFFD7"
72+
"\\uFFDA-\\uFFDC"))

test/clout/core_test.clj test/clout/core_test.cljc

+34-9
Original file line numberDiff line numberDiff line change
@@ -1,9 +1,31 @@
11
(ns clout.core-test
2-
(:import [clojure.lang ExceptionInfo]
3-
[java.util.regex PatternSyntaxException])
4-
(:require [clojure.test :refer :all]
5-
[ring.mock.request :refer [request]]
6-
[clout.core :refer :all]))
2+
(:import #?@(:clj [[clojure.lang ExceptionInfo]
3+
[java.util.regex PatternSyntaxException]]
4+
:cljs [[goog Uri]]))
5+
(:require #?@(:clj [[clojure.test :refer :all]
6+
[ring.mock.request :refer [request]]]
7+
:cljs [[cljs.test :refer-macros [is are deftest testing use-fixtures]]])
8+
[clout.core :refer [route-matches route-compile]]))
9+
10+
#?(:cljs
11+
(defn request
12+
"Naive implementation of the Ring Mock Request in ClojureScript."
13+
[method uri]
14+
(let [uri (Uri. uri)
15+
host (if-not (empty? (.getDomain uri)) (.getDomain uri) "localhost")
16+
port (.getPort uri)
17+
scheme (.getScheme uri)
18+
path (js/encodeURI (.getPath uri))]
19+
{:server-port (or port 80)
20+
:server-name host
21+
:remote-addr "localhost"
22+
:uri (if (clojure.string/blank? path) "/" path)
23+
:query-string (.getQuery uri)
24+
:scheme (if-not (empty? scheme) (keyword scheme) :http)
25+
:request-method method
26+
:headers {"host" (if port
27+
(str host ":" port)
28+
host)}})))
729

830
(deftest fixed-path
931
(are [path] (route-matches path (request :get path))
@@ -112,16 +134,19 @@
112134
(assoc :path-info "/bar")))))
113135

114136
(deftest custom-matches
115-
(let [route (route-compile "/foo/:bar" {:bar #"\d+"})]
137+
(let [route (route-compile "/foo/:bar" {:bar #?(:clj #"\d+" :cljs "\\d+")})]
116138
(is (not (route-matches route (request :get "/foo/bar"))))
117139
(is (not (route-matches route (request :get "/foo/1x"))))
118140
(is (route-matches route (request :get "/foo/10")))))
119141

120142
(deftest unused-regex-keys
121-
(is (thrown? AssertionError (route-compile "/:foo" {:foa #"\d+"})))
122-
(is (thrown? AssertionError (route-compile "/:foo" {:foo #"\d+" :bar #".*"}))))
143+
(is (thrown? #?(:clj AssertionError :cljs js/Error)
144+
(route-compile "/:foo" {:foa #"\d+"})))
145+
(is (thrown? #?(:clj AssertionError :cljs js/Error)
146+
(route-compile "/:foo" {:foo #"\d+" :bar #".*"}))))
123147

124148
(deftest invalid-inline-patterns
125149
(is (thrown? ExceptionInfo (route-compile "/:foo{")))
126150
(is (thrown? ExceptionInfo (route-compile "/:foo{\\d{2}")))
127-
(is (thrown? PatternSyntaxException (route-compile "/:foo{[a-z}"))))
151+
(is (thrown? #?(:clj PatternSyntaxException :cljs js/Error)
152+
(route-compile "/:foo{[a-z}"))))

test/clout/test_runner.cljs

+5
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,5 @@
1+
(ns clout.test-runner
2+
(:require [doo.runner :refer-macros [doo-tests]]
3+
[clout.core-test]))
4+
5+
(doo-tests 'clout.core-test)

0 commit comments

Comments
 (0)