This page describes development practices for this codebase.
cljfmt is a common formatter used for Clojure, analogous to Prettier for other languages. While we do not format/indent consistently with cljfmt across the whole codebase, we recommend that you do so for code that you change/add. You can do so easily with the Calva extension in VSCode: It will (mostly) indent your code correctly as you type, and you can move your cursor to the start of the line(s) you've written and press Tab to auto-indent all Clojure forms nested under the one starting on the current line.
Most of our linters require babashka. Before running them, please install
https://github.com/babashka/babashka#installation. To invoke all the linters in
this section, run bb dev:lint.
To lint:
clojure -M:clj-kondo --lint src
We lint our Clojure(Script) code with https://github.com/clj-kondo/clj-kondo/. If you need to configure specific linters, see this documentation. Where possible, a global linting configuration is used and namespace specific configuration is avoided.
There are outstanding linting items that are currently ignored to allow linting the rest of the codebase in CI. These outstanding linting items should be addressed at some point:
TODO:lint#_:clj-kondo/ignore require a good understanding of the context to address as they usually involve something with a side effect or require changing multiple fns up the call stack.We use https://github.com/borkdude/carve to detect unused vars in our codebase.
To run this linter:
bb lint:carve
By default, the script runs in CI mode which prints unused vars if they are found. The script can be run in an interactive mode which prompts for keeping (ignoring) an unused var or removing it. Run this mode with:
bb lint:carve '{:interactive true}'
When a var is ignored, it is added to .carve/ignore. Please add a comment for
why a var is ignored to help others understand why it's unused.
Large vars have a lot of complexity and make it hard for the team to maintain and understand them. To run this linter:
bb lint:large-vars
To configure the linter, see the [:tasks/config :large-vars] path of bb.edn.
Documentation helps teams share their knowledge and enables more individuals to contribute to the codebase. Documenting our namespaces is a good first step to improving our documentation. To run this linter:
bb lint:ns-docstrings
To skip documenting a ns, use the common ^:no-doc metadata flag.
We use datascript's datalog to power our modeling and querying layer. Since datalog is concise, it is easy to write something invalid. To avoid typos and other preventable mistakes, we lint our queries and rules. Our queries are linted through clj-kondo and datalog-parser. clj-kondo will error if it detects an invalid query.
Our translations can be configured incorrectly. We can catch some of these mistakes as noted here.
We have unit and end to end tests.
To run end to end tests
yarn electron-watch
# in another shell
yarn e2e-test # or npx playwright test
If e2e failed after first running:
rm -rdf ~/.logseqrm -rdf <repo dir>/tmp/rm -rdf <appData dir>/Electron  (Reference: https://www.electronjs.org/de/docs/latest/api/app#appgetpathname)If e2e tests fail, they can be debugged by examining a trace dump with the playwright trace viewer. Locally this will get dumped into e2e-dump/. On CI the trace file will be under Artifacts at the bottom of a run page e.g. https://github.com/logseq/logseq/actions/runs/3574600322.
Our unit tests use the shadow-cljs test-runner. To run them:
yarn test
By convention, a namespace's tests are found at a corresponding namespace
of the same name with an added -test suffix. For example, tests
for frontend.db.model are found in frontend.db.model-test.
There are a couple different ways to develop with tests:
Tests can be selectively run on the commandline using our own test runner which provides the same test selection options as cognitect-labs/test runner. For this workflow:
clj -M:test watch test in one shell^:focus metadata flags to tests e.g. (deftest ^:focus test-name ...).node static/tests.js -i focus to only run those
tests. To run all tests except those tests run node static/tests.js -e focus.-r, run tests for frontend.util.page-property-test with node static/tests.js -r page-property.Multiple options can be specified to AND selections. For example, to run all frontend.util.page-property-test tests except for the focused one: node static/tests.js -r page-property -e focus
For help on more options, run node static/tests.js -h.
To run tests automatically on file save, run clojure -M:test watch test
--config-merge '{:autorun true}'. Specific namespace(s) can be auto run with
the :ns-regexp option e.g. clojure -M:test watch test --config-merge
'{:autorun true :ns-regexp "frontend.util.page-property-test"}'.
To write a test that uses a datascript db:
test-helper ns to create and
destroy test databases after each test.test-helper/load-test-files.test-helper/test-dbTo write a performance test:
Use frontend.util/with-time-number to get the time in ms.
Example:
(are [x timeout] (>= timeout (:time (util/with-time-number (block/normalize-block x true))))
  ... )
For examples of these tests, see frontend.db.query-dsl-test and frontend.db.model-test.
Async unit testing is well supported in ClojureScript. https://clojurescript.org/tools/testing#async-testing is a good guide for how to do this. We have a couple of test helpers that make testing async easier:
frontend.test.helper/deftest-async - deftest for async tests that ensures
uncaught exceptions don't abruptly end the test suite. If you don't use this
macro for async tests, you are expected to handle unexpected failures in your testfrontend.test.helper/with-reset - A version of with-redefs that works for
async contextsPlease refer to our accessibility guidelines.
For logging, we use https://github.com/lambdaisland/glogi. When in development, be sure to have enabled custom formatters in the desktop app and browser. Without this enabled, most of the log messages aren't readable.
We use both spec and malli for data validation and (and generation someday). malli has the advantage that its schema is data and can be used for additional purposes. See plugin-config for an example.
Specs should go under src/main/frontend/spec/ and be compatible with clojure
and clojurescript. See frontend.spec.storage for an example.
Malli schemas should go under src/main/frontend/schema/ and be compatible with clojure
and clojurescript. See frontend.schema.handler.plugin-config for an example.
By following these conventions, these should also be usable by babashka. This is helpful as it allows for third party tools to be written with logseq's data model.
There are some babashka tasks under nbb: which are useful for inspecting
database changes in realtime. See these
docs for more info.