-
-
Notifications
You must be signed in to change notification settings - Fork 148
/
Copy pathcljs_SLASH_analyzer_SLASH_api.cljc
219 lines (194 loc) · 7.55 KB
/
cljs_SLASH_analyzer_SLASH_api.cljc
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
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
; Copyright (c) Rich Hickey. All rights reserved.
; The use and distribution terms for this software are covered by the
; Eclipse Public License 1.0 (http://opensource.org/licenses/eclipse-1.0.php)
; which can be found in the file epl-v10.html at the root of this distribution.
; By using this software in any fashion, you are agreeing to be bound by
; the terms of this license.
; You must not remove this notice, or any other, from this software.
(ns cljs.analyzer.api
"This is intended to be a stable api for those who need programmatic access
to the analyzer."
(:refer-clojure :exclude [all-ns ns-interns ns-resolve resolve find-ns
ns-publics remove-ns])
(:require [cljs.env :as env]
[cljs.analyzer :as ana]))
;; =============================================================================
;; Useful Utilities
(defn empty-state
"Creates an empty compilation state Atom<Map>."
[]
(env/default-compiler-env))
(defmacro with-state
"Run the body with the given compilation state Atom<Map>."
[state body]
`(env/with-compiler-env ~state
~@body))
(defn empty-env
"Creates an empty analysis environment."
[]
(ana/empty-env))
(defmacro no-warn
"Disable analyzer warnings for any analysis executed in body."
[& body]
(let [no-warnings (zipmap (keys ana/*cljs-warnings*) (repeat false))]
`(binding [ana/*cljs-warnings* ~no-warnings]
~@body)))
(defn warning-enabled?
"Test if the given warning-type is enabled."
[warning-type]
(ana/*cljs-warnings* warning-type))
(defn default-warning-handler
"The default warning handler.
Outputs the warning messages to *err*."
[warning-type env extra]
(ana/default-warning-handler warning-type env extra))
(defn get-options
"Return the compiler options from compiler state."
([] (get-options env/*compiler*))
([state]
(get @state :options)))
(defn get-js-index
"Return the currently computed Google Closure js dependency index from the
compiler state."
([] (get-options env/*compiler*))
([state]
(get @state :js-dependency-index)))
#?(:clj
(defn analyze
"Given an environment, a map containing {:locals (mapping of names to bindings), :context
(one of :statement, :expr, :return), :ns (a symbol naming the
compilation ns)}, and form, returns an expression object (a map
containing at least :form, :op and :env keys). If expr has any (immediately)
nested exprs, must have :children [exprs...] entry. This will
facilitate code walking without knowing the details of the op set."
([env form] (analyze env form nil))
([env form name] (analyze env form name nil))
([env form name opts]
(analyze
(if-not (nil? env/*compiler*)
env/*compiler*
(env/default-compiler-env opts))
env form name opts))
([state env form name opts]
(env/with-compiler-env state
(binding [ana/*cljs-warning-handlers* (:warning-handlers opts ana/*cljs-warning-handlers*)]
(ana/analyze env form name opts))))))
#?(:clj
(defn forms-seq
"Seq of Clojure/ClojureScript forms from rdr, a java.io.Reader. Optionally
accepts a filename argument which will be used in any emitted errors."
([rdr] (ana/forms-seq* rdr nil))
([rdr filename] (ana/forms-seq* rdr filename))))
#?(:clj
(defn parse-ns
"Helper for parsing only the essential namespace information from a
ClojureScript source file and returning a cljs.closure/IJavaScript compatible
map _not_ a namespace AST node.
By default does not load macros or perform any analysis of dependencies. If
opts parameter provided :analyze-deps and :load-macros keys their values will
be used for *analyze-deps* and *load-macros* bindings respectively. This
function does _not_ side-effect the ambient compilation environment unless
requested via opts where :restore is false."
([src] (parse-ns src nil nil))
([src opts] (parse-ns src nil opts))
([src dest opts]
(parse-ns
(if-not (nil? env/*compiler*)
env/*compiler*
(env/default-compiler-env opts))
src dest opts))
([state src dest opts]
(env/with-compiler-env state
(binding [ana/*cljs-warning-handlers* (:warning-handlers opts ana/*cljs-warning-handlers*)]
(ana/parse-ns src dest opts))))))
#?(:clj
(defn analyze-file
"Given a java.io.File, java.net.URL or a string identifying a resource on the
classpath attempt to analyze it.
This function side-effects the ambient compilation environment
`cljs.env/*compiler*` to aggregate analysis information. opts argument is
compiler options, if :cache-analysis true will cache analysis to
\":output-dir/some/ns/foo.cljs.cache.edn\". This function does not return a
meaningful value."
([f] (analyze-file f nil))
([f opts]
(analyze-file
(if-not (nil? env/*compiler*)
env/*compiler*
(env/default-compiler-env opts))
f opts))
([state f opts]
(env/with-compiler-env state
(binding [ana/*cljs-warning-handlers* (:warning-handlers opts ana/*cljs-warning-handlers*)]
(ana/analyze-file f opts))))))
;; =============================================================================
;; Main API
(defn resolve
"Given an analysis environment resolve a var. Analogous to
clojure.core/resolve"
[env sym]
{:pre [(map? env) (symbol? sym)]}
(try
(ana/resolve-var env sym
(ana/confirm-var-exists-throw))
(catch #?(:clj Exception :cljs :default) e
(ana/resolve-macro-var env sym))))
(defn all-ns
"Return all namespaces. Analagous to clojure.core/all-ns but
returns symbols identifying namespaces not Namespace instances."
([]
(all-ns env/*compiler*))
([state]
(keys (get @state ::ana/namespaces))))
(defn find-ns
"Given a namespace return the corresponding namespace analysis map. Analagous
to clojure.core/find-ns."
([sym]
(find-ns env/*compiler* sym))
([state sym]
{:pre [(symbol? sym)]}
(get-in @state [::ana/namespaces sym])))
(defn ns-interns
"Given a namespace return all the var analysis maps. Analagous to
clojure.core/ns-interns but returns var analysis maps not vars."
([ns]
(ns-interns env/*compiler* ns))
([state ns]
{:pre [(symbol? ns)]}
(merge
(get-in @state [::ana/namespaces ns :macros])
(get-in @state [::ana/namespaces ns :defs]))))
(defn ns-publics
"Given a namespace return all the public var analysis maps. Analagous to
clojure.core/ns-publics but returns var analysis maps not vars."
([ns]
(ns-publics env/*compiler* ns))
([state ns]
{:pre [(symbol? ns)]}
(->> (merge
(get-in @state [::ana/namespaces ns :macros])
(get-in @state [::ana/namespaces ns :defs]))
(remove (fn [[k v]] (:private v)))
(into {}))))
(defn ns-resolve
"Given a namespace and a symbol return the corresponding var analysis map.
Analagous to clojure.core/ns-resolve but returns var analysis map not Var."
([ns sym]
(ns-resolve env/*compiler* ns sym))
([state ns sym]
{:pre [(symbol? ns) (symbol? sym)]}
(get-in @state [::ana/namespaces ns :defs sym])))
(defn remove-ns
"Removes the namespace named by the symbol."
([ns]
(remove-ns env/*compiler* ns))
([state ns]
{:pre [(symbol? ns)]}
(swap! state update-in [::ana/namespaces] dissoc ns)))
(defmacro in-cljs-user
"Binds cljs.analyzer/*cljs-ns* to 'cljs.user and uses the given compilation
environment atom and runs body."
[env & body]
`(binding [cljs.analyzer/*cljs-ns* 'cljs.user]
(cljs.env/with-compiler-env ~env
~@body)))