Skip to content

Commit dd5d862

Browse files
anmonteiroswannodette
authored andcommitted
CLJS-2061: Support ns :require for JS libs, allow strings along with symbols
The strategy employed by this patch is to defer processing JS modules until after we have found the sources and computed the "missing modules" in NS declarations. `cljs.analyzer/parse-ns` returns the extra `:missing-js-modules` key, which it infers by looking at string declarations in dependencies.
1 parent 76ffac3 commit dd5d862

File tree

9 files changed

+106
-25
lines changed

9 files changed

+106
-25
lines changed
Original file line numberDiff line numberDiff line change
@@ -0,0 +1 @@
1+
out/
+13
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,13 @@
1+
# ClojureScript string-based requires demo
2+
3+
Running:
4+
5+
1. At the root of the ClojureScript repo, run `./script/bootstrap`
6+
2. Switch into this directory: `cd samples/string-requires-npm-deps`
7+
3. Build the project:
8+
9+
``` shell
10+
$ java -cp `ls ../../lib/*.jar | paste -sd ":" -`:../../src/main/cljs:../../src/main/clojure:src clojure.main build.clj
11+
```
12+
13+
4. run the generated JavaScript with `node out/main.js`
+13
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,13 @@
1+
(require '[cljs.build.api :as b])
2+
3+
(b/build "src"
4+
{:output-dir "out"
5+
:output-to "out/main.js"
6+
:optimizations :none
7+
:verbose true
8+
:target :nodejs
9+
:compiler-stats true
10+
:main 'foo.core
11+
:npm-deps {:react "15.6.1"
12+
:react-dom "15.6.1"}
13+
:closure-warnings {:non-standard-jsdoc :off}})
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,3 @@
1+
{
2+
"name": "string-requires-npm-deps"
3+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,10 @@
1+
(ns foo.core
2+
(:require [react :refer [createElement]]
3+
["react-dom/server" :as rds :refer [renderToString]]
4+
"create-react-class"))
5+
6+
(enable-console-print!)
7+
8+
(println "resolves single exports" create-react-class)
9+
10+
(println (renderToString (createElement "div" nil "Hello World!")))

src/main/cljs/cljs/module_deps.js

+14-4
Original file line numberDiff line numberDiff line change
@@ -30,7 +30,7 @@ md.on('package', function (pkg) {
3030
};
3131

3232
if (pkg.name != null) {
33-
pkgJson.provided = [ pkg.name ];
33+
pkgJson.provides = [ pkg.name ];
3434
}
3535

3636
if (pkg.main != null) {
@@ -49,16 +49,26 @@ md.on('end', function() {
4949
for (var i = 0; i < pkgJsons.length; i++) {
5050
var pkgJson = pkgJsons[i];
5151

52-
if (deps_files[pkgJson.main] != null && pkgJson.provided != null) {
53-
deps_files[pkgJson.main].provides = pkgJson.provided;
52+
if (deps_files[pkgJson.main] != null && pkgJson.provides != null) {
53+
deps_files[pkgJson.main].provides = pkgJson.provides;
5454
}
5555

5656
deps_files[pkgJson.file] = { file: pkgJson.file };
5757
}
5858

5959
var values = [];
6060
for (var key in deps_files) {
61-
values.push(deps_files[key]);
61+
var dep = deps_files[key];
62+
63+
if (dep.provides == null && !/node_modules\/[^/]*?\/package.json$/.test(dep.file)) {
64+
var match = dep.file.match(/node_modules\/(.*)\.js(on)*$/)
65+
66+
if (match != null){
67+
dep.provides = [ match[1] ];
68+
}
69+
}
70+
71+
values.push(dep);
6272
}
6373

6474
process.stdout.write(JSON.stringify(values));

src/main/clojure/cljs/analyzer.cljc

+17-5
Original file line numberDiff line numberDiff line change
@@ -2106,13 +2106,13 @@
21062106
(str msg "; offending spec: " (pr-str spec)))
21072107

21082108
(defn basic-validate-ns-spec [env macros? spec]
2109-
(when-not (or (symbol? spec) (sequential? spec))
2109+
(when-not (or (symbol? spec) (string? spec) (sequential? spec))
21102110
(throw
21112111
(error env
21122112
(parse-ns-error-msg spec
21132113
"Only [lib.ns & options] and lib.ns specs supported in :require / :require-macros"))))
21142114
(when (sequential? spec)
2115-
(when-not (symbol? (first spec))
2115+
(when-not (or (symbol? (first spec)) (string? (first spec)))
21162116
(throw
21172117
(error env
21182118
(parse-ns-error-msg spec
@@ -2210,7 +2210,7 @@
22102210
(recur fs ret true)))))
22112211

22122212
(defn parse-require-spec [env macros? deps aliases spec]
2213-
(if (symbol? spec)
2213+
(if (or (symbol? spec) (string? spec))
22142214
(recur env macros? deps aliases [spec])
22152215
(do
22162216
(basic-validate-ns-spec env macros? spec)
@@ -2222,7 +2222,11 @@
22222222
[lib js-module-provides] (if-some [js-module-name (get-in @env/*compiler* [:js-module-index (str lib)])]
22232223
[(symbol js-module-name) lib]
22242224
[lib nil])
2225-
{alias :as referred :refer renamed :rename :or {alias lib}} (apply hash-map opts)
2225+
{alias :as referred :refer renamed :rename
2226+
:or {alias (if (string? lib)
2227+
(symbol (munge lib))
2228+
lib)}}
2229+
(apply hash-map opts)
22262230
referred-without-renamed (seq (remove (set (keys renamed)) referred))
22272231
[rk uk renk] (if macros? [:require-macros :use-macros :rename-macros] [:require :use :rename])]
22282232
(when-not (or (symbol? alias) (nil? alias))
@@ -3565,7 +3569,14 @@
35653569
(= "cljc" (util/ext src)))
35663570
'cljs.core$macros
35673571
ns-name)
3568-
deps (merge (:uses ast) (:requires ast))]
3572+
deps (merge (:uses ast) (:requires ast))
3573+
missing-js-modules (into #{}
3574+
(comp
3575+
(filter (fn [[k v]]
3576+
(and (or (string? k) (string? v))
3577+
(not (js-module-exists? k)))))
3578+
(map val))
3579+
deps)]
35693580
(merge
35703581
{:ns (or ns-name 'cljs.user)
35713582
:provides [ns-name]
@@ -3574,6 +3585,7 @@
35743585
(cond-> (conj (set (vals deps)) 'cljs.core)
35753586
(get-in @env/*compiler* [:options :emit-constants])
35763587
(conj constants-ns-sym)))
3588+
:missing-js-modules missing-js-modules
35773589
:file dest
35783590
:source-file (when rdr src)
35793591
:source-forms (when-not rdr src)

src/main/clojure/cljs/closure.clj

+23-16
Original file line numberDiff line numberDiff line change
@@ -1977,13 +1977,14 @@
19771977
#(ensure-cljs-base-module % opts)))
19781978

19791979
(defn add-implicit-options
1980-
[{:keys [optimizations output-dir npm-deps]
1980+
[{:keys [optimizations output-dir npm-deps missing-js-modules]
19811981
:or {optimizations :none
19821982
output-dir "out"}
19831983
:as opts}]
19841984
(let [opts (cond-> (update opts :foreign-libs
19851985
(fn [libs]
1986-
(into (index-node-modules npm-deps opts)
1986+
(into (index-node-modules
1987+
(into missing-js-modules (keys npm-deps)) opts)
19871988
(expand-libs libs))))
19881989
(:closure-defines opts)
19891990
(assoc :closure-defines
@@ -2003,7 +2004,7 @@
20032004
:optimizations optimizations
20042005
:output-dir output-dir
20052006
:ups-libs libs
2006-
:ups-foreign-libs (into (index-node-modules (compute-upstream-npm-deps opts) opts)
2007+
:ups-foreign-libs (into (index-node-modules (keys (compute-upstream-npm-deps opts)) opts)
20072008
(expand-libs foreign-libs))
20082009
:ups-externs externs
20092010
:emit-constants emit-constants
@@ -2134,15 +2135,15 @@
21342135
(into [] (distinct (mapcat #(node-module-deps % opts) entries)))))
21352136

21362137
(defn index-node-modules
2137-
([npm-deps]
2138+
([modules]
21382139
(index-node-modules
2139-
npm-deps
2140+
modules
21402141
(when env/*compiler*
21412142
(:options @env/*compiler*))))
2142-
([npm-deps opts]
2143+
([modules opts]
21432144
(let [node-modules (io/file "node_modules")]
2144-
(if (and (not (empty? npm-deps)) (.exists node-modules) (.isDirectory node-modules))
2145-
(let [modules (map name (keys npm-deps))
2145+
(if (and (not (empty? modules)) (.exists node-modules) (.isDirectory node-modules))
2146+
(let [modules (into #{} (map name) modules)
21462147
deps-file (io/file (str (util/output-directory opts) File/separator
21472148
"cljs$node_modules.js"))]
21482149
(util/mkdirs deps-file)
@@ -2252,15 +2253,21 @@
22522253
(env/with-compiler-env compiler-env
22532254
;; we want to warn about NPM dep conflicts before installing the modules
22542255
(check-npm-deps opts)
2256+
(maybe-install-node-deps! opts)
22552257
(let [compiler-stats (:compiler-stats opts)
22562258
static-fns? (or (and (= (:optimizations opts) :advanced)
2257-
(not (false? (:static-fns opts))))
2258-
(:static-fns opts)
2259-
ana/*cljs-static-fns*)
2260-
all-opts (-> opts
2261-
maybe-install-node-deps!
2262-
add-implicit-options
2263-
process-js-modules)]
2259+
(not (false? (:static-fns opts))))
2260+
(:static-fns opts)
2261+
ana/*cljs-static-fns*)
2262+
sources (-find-sources source opts)
2263+
missing-js-modules (into #{}
2264+
(comp
2265+
(map :missing-js-modules)
2266+
cat)
2267+
sources)
2268+
all-opts (-> (assoc opts :missing-js-modules missing-js-modules)
2269+
add-implicit-options
2270+
process-js-modules)]
22642271
(check-output-to opts)
22652272
(check-output-dir opts)
22662273
(check-source-map opts)
@@ -2275,7 +2282,7 @@
22752282
(assoc :target (:target opts))
22762283
(assoc :js-dependency-index (deps/js-dependency-index all-opts))
22772284
;; Save list of sources for cljs.analyzer/locate-src - Juho Teperi
2278-
(assoc :sources (-find-sources source all-opts))))
2285+
(assoc :sources sources)))
22792286
(binding [comp/*recompiled* (when-not (false? (:recompile-dependents opts))
22802287
(atom #{}))
22812288
ana/*cljs-static-fns* static-fns?

src/test/clojure/cljs/analyzer_tests.clj

+12
Original file line numberDiff line numberDiff line change
@@ -766,6 +766,18 @@
766766
(catch Exception _))
767767
(is (= ["cljs.core/aset, arguments must be an array, followed by numeric indices, followed by a value, got [object string number] instead (consider goog.object/set for object access)"] @ws)))))
768768

769+
(deftest cljs-2061
770+
(let [test-cenv (atom
771+
(merge @(e/default-compiler-env)
772+
{:js-module-index {"react" "module$src$test$cljs$react"}}))
773+
ast (e/with-compiler-env test-cenv
774+
(a/parse-ns
775+
'[(ns test.cljs-2061
776+
(:require ["react" :as react]
777+
["react-dom/server" :refer [renderToString]]))]))]
778+
(is (= (:missing-js-modules ast) #{"react-dom/server"}))
779+
(is (= (:requires ast) '#{cljs.core module$src$test$cljs$react "react-dom/server"}))))
780+
769781
(comment
770782
(binding [a/*cljs-ns* a/*cljs-ns*]
771783
(a/no-warn

0 commit comments

Comments
 (0)