Last summer I moved boodle1 from Figwheel to Figwheel Main.2 I was pretty
satisfied with this setup, but the rise of shadow-cljs in the Clojure community
has been tickling my curiosity over and over again. While configuring
shadow-cljs
I took the chance to set lein
aside and finally explore clj
and deps.edn
, another thing on my to-do list that has been waiting for me for
some time.
Let’s start with deps.edn
:
{:deps
{
;; … Clojure dependencies …
}
:paths ["src/clj" "resources"]
:aliases {:run {:main-opts ["-m" "boodle.core"]}
:test {:extra-paths ["test/clj"]
:extra-deps {lambdaisland/kaocha {:mvn/version "0.0-266"}}}
:outdated {:extra-deps {olical/depot {:mvn/version "1.4.0"}}
:main-opts ["-m" "depot.outdated.main"]}}}
One thing worth of notice here is :paths
. I included the resources
directory
in it because I serve it via compojure
/http-kit
and that’s where I put the
static assets used in my main index page. This solution works when I run
mount.core/start
via REPL and when I use the :run
alias from the command
line.
Speaking of aliases, they are invaluable. With :run
I can use clj -A:run
instead of lein run
, and with clj -A:outdated -a outdated
I can easily check
for outdated dependencies in my deps.edn
thanks to depot. Unit tests are now
handled by Kaocha, and as per its instructions I have a bin/kaocha
file which
picks up the :test
alias and runs all my tests.
I love the simplicity of deps.edn
, it feels like a de-cluttered project.clj
.
Configuring shadow-cljs was a bit more complicated. It took me a while to
understand how to properly integrate it with CIDER. However, the manual is great
and the fine people on Slack’s #shadow-cljs
(thanks Ryan Haywood and Thomas
Heller!) made it all nice and smooth.
This is my shadow-cljs.edn
:
{:source-paths ["src/cljs"]
:nrepl {:port 8777
:middleware [refactor-nrepl.middleware/wrap-refactor]}
:dependencies [
;; … ClojureScript dependencies …
]
:builds {:boodle {:target :browser
:output-dir "resources/public/js"
:asset-path "/js"
:modules {:main {:entries [boodle.core]}}}}}
Basic setup, straight from the manual pages. Notice that :output-dir
refers to
the same resources
directory configured in deps.edn
. This tells shadow-cljs
to place the build output exactly where I need.
The other important change is in the CLJS libraries I use. shadow-cljs does not
support CLJSJS, so I had to install pikaday
and moment
via yarn
.
$ yarn add pikaday moment
Since boodle
relies on re-frame
, I had to install three other libraries to
make shadow-cljs build my code happily:
$ yarn add react react-dom create-react-class
In the namespace where I use them, the :require
had to be changed:
(:require ;; … other requires …
["react-dom" :refer [findDOMNode]]
["pikaday" :as pikaday])
And the code had to be adapted:
(let [default-opts {:field (findDOMNode this)
;; … other options …
}
instance (pikaday. opts)]
;; … other ClojureScript code …
)
However, a project setup is not ready unless I can work on it via Emacs. These are the steps I made to have a working CLJ/CLJS development environment. When this issue3 will be fixed some of it may become unnecessary.
First, I added a couple of dependencies in shadow-cljs.edn
:
:dependencies [;; … other dependencies …
[cider/cider-nrepl "0.18.0"]
[refactor-nrepl "2.4.0"]]
Then I removed everything Figwheel Main related from the .dir-locals.el
file
located in the root directory of boodle
:
((nil
(cider-default-cljs-repl . shadow-select)
(cider-pprint-fn . zprint)
(cider-preferred-build-tool . clojure-cli)
(cider-known-endpoints . (("hathaway" "localhost" "8777")))
(cider-ns-refresh-after-fn . "mount.core/start")
(cider-ns-refresh-before-fn . "mount.core/stop"))
(emacs-lisp-mode
(flycheck-disabled-checkers . "emacs-lisp-checkdoc")))
I ran cider-jack-in-clj
to have a CLJ REPL and check I didn’t break something
on the Clojure side of boodle
.
From a terminal, I used shadow-cljs watch boodle
to have shadow-cljs
build
boodle
and offer me a nice nREPL server listening on port 8777
, the one
I specified in shadow-cljs.edn
.
Back in CIDER, cider-connect-cljs
brought up a REPL connected to shadow-cljs
nREPL server, and let me select the build I want it operating on (default is
dev
, I just typed in boodle
). Connecting to boodle
via the browser
finalized the connection between my code and the CLJS environment.
A great journey. Once again, boodle
proved fertile ground to understand the
Clojure ecosystem and play with it to learn something new. If you want more
details, the project is on my SourceHut.4
-
See: Moving to Figwheel Main. ↩︎