title: Clojure Developer Tooling for Speed and Productivity in 2024
So youβll show me your cool workflows! (help me get better!)
I need to figure out β¦
- How to run my application
- How to run all the tests in my project
- How to build the artifacts of my app/lib
- How to deploy these artifacts so others can use them
The Javascript ecosystem, for all the abuse we hurl at them, has done an incredible job. The templating that people have access to out of the box is something we should all aspire to!
https://github.com/seancorfield/deps-new
clojure -Tnew lib :name me.vedang/depsnew_lib
tree depsnew_lib
- All the boilerplate!
- Clear Instructions in the README
- Aliases:
clojure -X:run-x
,clojure -X:run-m
Run your appclojure -T:build test
Run testsclojure -T:build ci
Build artifactsclojure -T:build deploy
Deploy artifacts
AND YOU SHOULD!
(looking at you, framework authors!)
- Does your favourite framework have a deps-new template? Add it!
- Do you find yourself copying the same code everywhere? Try and extract it into a template!
- What is deps-new? github/deps-new/README
- Templates provided by the Community deps-new/README#templates
- How to write your own template? practical.li/create-deps-new-template-for-clojure-cli-projects
- How do I connect to a REPL?
- How do I setup logging?
- β¦
https://github.com/babashka/neil
- neil can add aliases for you!
- neil can do so much more!
neil add cider
:cider ;; added by neil {:extra-deps {cider/cider-nrepl {:mvn/version "0.47.0"} djblue/portal {:mvn/version "0.52.2"} mx.cider/tools.deps.enrich-classpath {:mvn/version "1.19.0"} nrepl/nrepl {:mvn/version "1.1.1"} refactor-nrepl/refactor-nrepl {:mvn/version "3.10.0"}} :main-opts ["-m" "nrepl.cmdline" "--middleware" "[cider.nrepl/cider-middleware,refactor-nrepl.middleware/wrap-refactor,portal.nrepl/wrap-portal]"]}
neil dep search next.jdbc
neil dep add :lib com.github.seancorfield/next.jdbc
neil dep add :lib nextjournal/clerk :alias :cider
- Make it easy to add aliases to neil!
- I would like to add to existing aliases instead of creating new ones!
- Using neil michielborkent/new-clojure-project-quickstart.html
- All the options github/babashka/neil/README.md
- My neil script, with aliases I use github/vedang/neil
clojure -M:test:logs-dev:cider
Wait, whatβs the logging alias?
Logging is hard!
So once I figured it out, I made a template!
neil add logs-dev
:logs-dev ;; added by neil {:extra-deps {me.vedang/logger {:local/root "logger"}} :jvm-opts ["-Dclojure.tools.logging.factory=clojure.tools.logging.impl/log4j2-factory" "-Dorg.eclipse.jetty.util.log.class=org.eclipse.jetty.util.log.Slf4jLog" "-Dlog4j2.contextSelector=org.apache.logging.log4j.core.async.AsyncLoggerContextSelector" "-Dlog4j2.configurationFile=logger/log4j2-dev.xml" ;; Change logging.level to one of TRACE, DEBUG, INFO, WARN, ERROR ;; depending on requirement during development "-Dlogging.level=DEBUG"]}
clojure -Sdeps '{:deps {io.github.vedang/clj-logging {:git/sha "e009d366c827705f513ef9018ffd920a49ce19da"}}}' -Tnew create :template me.vedang/logger :name me.vedang/logger
tree logger
7 directories, 6 files
http://pedestal.io/pedestal/0.7/reference/logging.html
(in-ns 'me.vedang.depsnew-lib)
(require '[me.vedang.logger.interface :as log])
(log/info :function :welcome
:action :starting
:args {:greeting "Hello World"})
{ "instant": { "epochSecond": 1711161317, "nanoOfSecond": 787564000 }, "thread": "nREPL-session-d6c4a5db-e57c-4efe-906b-7e5f1d0ad6e6", "level": "INFO", "loggerName": "me.vedang.depsnew-lib", "message": "{\"function\":\"welcome\",\"action\":\"starting\",\"args\":{\"greeting\":\"Hello World\"},\"line\":3}", "endOfBatch": true, "loggerFqcn": "org.apache.logging.slf4j.Log4jLogger", "contextMap": {}, "threadId": 41, "threadPriority": 5 }
Prefer emitting logs as JSON events. This lets us push them to Observability tools easily in production.
Clojure is REPL-first! Where are my notebooks?
Weβve been writing notebooks and refactoring them into usable code-bases all our lives. So why do we not have beautiful looking notebooks like the Python people?
https://github.com/nextjournal/clerk
Write your comments like you write Markdown, done!
;;; # π Sunday Morning Fun With Our Group Meditation Data
;; First, we need to parse the data from the Whatsapp Group. Whatsapp
;; provides a `export-chat` feature that we can use, but there are
;; some things we need to fix first:
;; ## Whatsapp uses a ridiculous format for the date when the message was sent
^{:nextjournal.clerk/visibility {:code :show :result :hide}}
(def whatsapp-export-date-format
"Frankly, this is a ridiculous format that no one should use."
"MM/dd/yy, hh:mm a")
;; The problem isn't so much the format, as the fact that they make it
;; hard to parse. For example, they won't export the date as
;; `"05/01/23, 03:31 pm"`. They will export it as `"5/1/23, 3:31 PM"`.
;; Every part of this is meant to fail your parser. π€·π½ββοΈ
What Clerk notebooks look like:
- https://recife.pfeodrippe.com/notebooks/recife/notebook/slow_start.html
- https://book.clerk.vision/#vega-lite
- Talk about Separedit!
https://github.com/stepchowfun/tagref
(require 'clojure.math)
;; [tag:polynomial_nonzero] This function never returns zero.
(defn polynomial [x]
(+ (clojure.math/pow x 2) 1))
;;; in some other file
(defn inverse-polynomial [x]
;; This is safe due to [ref:polynomial_nonzero].
(/ 1 (polynomial x)))
tagref list-tags | awk '{print $1}'
- Using Clerk book.clerk.vision
- Using Separedit github/twlz0ne/separedit.el
- Using Tagref github/stepchowfun/tagref
- Jack Rusher on the power of interactive notebooks (among other things): Stop Writing Dead Programs
https://github.com/clj-kondo/clj-kondo https://github.com/weavejester/cljfmt https://github.com/kkinnear/zprint
I do automatic Linting and Formatting on the code base!
Export clj-kondo config for libraries you use! (create ~.clj-kondo~ directory at root first)
clj-kondo --lint "$(clojure -A:dev:test:cider:build -Spath)" --copy-configs --skip-lint
(clojure-lsp
will do this automatically)
tree .clj-kondo
12 directories, 6 files
Commit your .clj-kondo
folder!
(For Emacsen) Use Apheleia!
Runs code-formatter on the buffer βat the right timeβ with minimal disturbance.
Add clj-kondo
configuration for your favourite libraries!
- clj-kondo Documentation. github/clj-kondo/clj-kondo
- cljfmt Documentation. github/weavejester/cljfmt
- The zprint Reference. github/kkinnear/zprint/doc/reference.md
- Apheleia for Emacs. github/radian-software/apheleia
After all, this is where we are all day!
IF we can make the experience even 1% better, isnβt that something amazing?
neil add cider-storm
:cider-storm ;; added by neil {:classpath-overrides ;; we need to disable the official compiler and use ClojureStorm {org.clojure/clojure nil} :extra-deps {cider/cider-nrepl {:mvn/version "0.47.0"} com.github.flow-storm/clojure {:mvn/version "1.11.2"} com.github.flow-storm/flow-storm-dbg {:mvn/version "3.13.1"} djblue/portal {:mvn/version "0.52.2"} mx.cider/tools.deps.enrich-classpath {:mvn/version "1.19.0"} nrepl/nrepl {:mvn/version "1.1.1"} org.openjfx/javafx-controls {:mvn/version "23-ea+3"} org.openjfx/javafx-base {:mvn/version "23-ea+3"} org.openjfx/javafx-graphics {:mvn/version "23-ea+3"} org.openjfx/javafx-swing {:mvn/version "23-ea+3"} refactor-nrepl/refactor-nrepl {:mvn/version "3.10.0"}} :main-opts ["-m" "nrepl.cmdline" "--middleware" "[flow-storm.nrepl.middleware/wrap-flow-storm,cider.nrepl/cider-middleware,refactor-nrepl.middleware/wrap-refactor,portal.nrepl/wrap-portal]"] :jvm-opts ["-Dclojure.storm.instrumentEnable=true" "-Dclojure.storm.instrumentOnlyPrefixes=me.vedang."]}
(in-ns 'me.vedang.depsnew-lib)
(defn foo [n]
(->> (range n)
(filter odd?)
(partition-all 2)
(map second)
(drop 10)
(reduce +)))
- Flowstorm website
- @jpmonetta showing a full walk-through Reifying execution, the interactive programming missing piece
Smaller teams, clear ownership!
- Easier to test
- Clear contracts (API)
- Independent deployments
- No version hell
- Easy refactoring across multiple services (in theory)
- Easy collaboration and on-boarding (in theory)
- Contracts are unwritten
- Testing becomes messy (specifically, test-suite time)
- Refactoring, Collaboration, On-boarding Hell
Eventually, teams start building conventions and tooling and start refactoring.
The problem is, itβs too late by now.
https://github.com/polyfy/polylith
Polylith is a set of conventions and tooling that has thought through all these problems!
poly create workspace name:polylith top-ns:me.vedang :commit
tree -L 3
6 directories, 4 files
Addendum: you can also create a workspace in an existing repo:
poly create workspace top-ns:me.vedang
projects/<name> : Interface with hosting platform
Letβs think through an example:
- Static Site : projects/weblog_static (Artifact: Public Folder, Deployment: Github Pages)
- Dynamic Site : projects/weblog_dynamic (Artifact: Uberjar, Deployment: Dockerfile, fly.toml)
bases/<name> : Interface with the external world
Continuing with our example:
- Static Site: bases/server_static
- Dynamic Site: bases/server_dynamic (server! routes)
This folder contains code that interacts with the external world
Think: CLI tool, Server, Lambda function, Worker
components/<name> : Interface with a module of business logic.
Continuing with our example:
- components/content: The markdown files I want to serve as posts
- components/render : Code to convert MD files to HTML
- components/readers: Logged-in visitors on my dynamic website
This folder contains re-usable, modular code that implements specific functionality.
projects/weblog_static -> [bases/server_static components/content components/render]
projects/weblog_dynamic -> [bases/server_dynamic components/content components/render components/readers]
Code outside a component can only refer to functions in the componentβs interface
namespace.
During development, you have access to the entire code-base! But there are rules around how you can access any code outside your own brick.
(so I donβt need to know how you implement the code, Iβm only concerned with the interface of your component)
Interface functions pass-through the arguments to appropriate internal functions.
- Where does the Dockerfile go?
projects/weblog_dynamic
- Where do I put end-to-end tests?
projects/weblog_dynamic/test
- Where do I define the routes and the main class?
bases/server_dynamic/core.clj
- Which functions do I use to render HTML pages?
components/render/interface.clj
- Which functions do I write contract tests for?
components/render/interface.clj
- Which functions do I document thoroughly?
components/render/interface.clj
- Where do I hook observability?
components/render/interface.clj
,bases/server_dynamic/core.clj
- Where does the actual rendering functionality live?
render/posts.clj
,render/tags.clj
,render/index.clj
β¦
- Enables running the right tests
- Checks to ensure component code is used properly everywhere
- Upgrades libraries across all your bricks
β¦ and much more
- Learn Polylith Conventions polylith.gitbook.io
- Use
poly
to enforce conventions: cljdoc.org/polylith/clj-poly - Sean Corfieldβs series on monorepos: corfield.org/deps-edn-monorepo
- CI/CD! (Use
babashka
!) - Observability (
pedestal
! oriapetos
+clj-otel
) - Web Development (
martian
makes talking to APIs a breeze!)
βββββββ β β βββββ βββββββ β βββ β βββ ββ ββ β βββ β β βββ β ββ ββββββ β βββ β Vedang Manerikar βββββββ βββββ βββ βββββββ βββββββββββββββ ββ β β β - @vedang on πfosstodon, π₯οΈgithub, π¦twitter βββββββββ β β ββββββββ ββ - https://www.salher.ai βββββββ ββ βββ βββββββββ + We can help with your problems! β βββββ βββββ ββββββββ ββ + (design, product, engineering) β βββββββββββ ββββββββ βββββββ βββ βββ β β βββββ THANK YOU! QUESTIONS? β βββ β βββ β βββββββββββ β βββ β βββββ ββββ ββββ β βββββββ β β ββββββ