/boot-cljs

Boot task to compile ClojureScript programs.

Primary LanguageClojureEclipse Public License 1.0EPL-1.0

boot-cljs

[adzerk/boot-cljs "0.0-3308-0"] ;; latest stable release

[adzerk/boot-cljs "1.7.48-SNAPSHOT"] ;; latest release

Boot task to compile ClojureScript applications.

  • Provides the cljs task–compiles ClojureScript to JavaScript.

  • Provides a mechanism by which multiple multiple JS applications can be compiled in the same project.

Try It

In a terminal do:

mkdir -p src/foop
echo -e '(ns foop.core)\n(.log js/console "hello world")' > src/foop/core.cljs
boot -s src -d adzerk/boot-cljs cljs

The compiled JavaScript will be written to target/main.js.

Usage

Add ClojureScript and boot-cljs to your build.boot dependencies and require the namespace:

(set-env! :dependencies '[[adzerk/boot-cljs "X.Y.Z" :scope "test"]])
(require '[adzerk.boot-cljs :refer :all])

You can see the options available on the command line:

boot cljs -h

Or in the REPL:

boot.user=> (doc cljs)

You can see debug level output from the task by setting boot's logging level:

boot -vv cljs

This will show the options being passed to the CLJS compiler.

Compilation Levels

The ClojureScript compiler uses the Google Closure compiler to generate optimized JavaScript when desired. There are three different Closure compilation levels: whitespace, simple, and advanced. You may specify the desired compilation level with the -O or --optimizations option:

boot cljs -O advanced

The default level is none, which bypasses the Closure compiler completely. This is handy for development work because it's very fast.

Same HTML for Development and Production

The HTML file will always need to have a <script> tag to load the compiled JavaScript:

<script type='text/javascript' src='main.js'></script>

Source Maps

Source maps associate locations in the compiled JavaScript file with the corresponding line and column in the ClojureScript source files. When source maps are enabled (the -s or --source-map option) the browser developer tools will refer to locations in the ClojureScript source rather than the compiled JavaScript.

boot cljs -s

You may need to enable source maps in your browser's developer tools settings.

Options

Option Description Task option .cljs.edn
optimization :none (default), :advanced ×
source-map Use source maps (default true for :none optimization) ×
id Selected .cljs.edn file ×
require Namespaces to require on load ×
init-fns Functions to call on load ×
compiler-options Cljs compiler options × ×

The cljs task normally does a good job of figuring out which options to pass to the compiler on its own. However, options can be provided via the -c or --compiler-options option, to provide specific options to the CLJS compiler:

boot cljs -c '{:target :nodejs}'

Incremental Builds

You can run boot such that it watches source files for changes and recompiles the JavaScript file as necessary:

boot watch cljs

You can also get audible notifications whenever the project is rebuilt:

boot watch speak cljs

Note: The watch and speak tasks are not part of boot-cljs–they're built-in tasks that come with boot.

Multiple Builds

The cljs task provides a way to specify application entry points at which it can point the CLJS compiler for compilation. These entry points are provided via files with the .cljs.edn extension.

These files have the following structure (e.g. js/index.cljs.edn):

{:require  [foo.bar baz.baf]
 :init-fns [foo.bar/init baz.baf/doit]
 :compiler-options {:target :nodejs}}

For each .cljs.edn file in the fileset, the cljs task will:

  • Create a CLJS namespace corresponding to the file's path, e.g. given the file foo/bar.cljs.edn it will create the foo.bar CLJS namespace. This namespace will :require any namespaces given in the :require key of the EDN, and add a do expression that calls any functions in :init-fns at the top level. These functions will be called with no arguments.

  • Configure compiler options according to :compiler-options key of the EDN, if there is one.

  • Configure the compiler to produce compiled JS at a location derived from the file's path, e.g. given the file foo/bar.cljs.edn the output JS file will be foo/bar.js.

  • Point the CLJS compiler at the generated namespace only. This "scopes" the compiler to that namespace plus any transitive dependencies via :require.

The example above would result in the following CLJS namespace, js/index.cljs:

(ns js.index
  (:require foo.bar baz.baf))

(do (foo.bar/init)
    (baz.baf/doit))

The result would be compiled to js/index.js. This is the JS script you'd add to the application's HTML file via a <script> tag.

Preamble and Externs Files

Jars with deps.cljs, like the ones provided by cljsjs can be used to supply Javascript libraries. If you need to use local js files you can manually create deps.cljs in your local project:

src/deps.cljs:

{:foreign-libs [{:file "bar.js"
                 :provides ["foo.bar"]}]
 :externs ["bar.ext.js"]}

src/bar.js:

function foo() {
  console.log("Hello world from local js");
}

src/bar.ext.js:

foo = function() {};

Examples

Create a new ClojureScript project, like so:

my-project
├── build.boot
├── html
│   └── index.html
└── src
    └── foop.cljs

And add the following contents to build.boot:

(set-env!
  :source-paths    #{"src"}
  :resource-paths    #{"html"}
  :dependencies '[[adzerk/boot-cljs "0.0-X-Y" :scope "test"]])

(require '[adzerk.boot-cljs :refer :all])

Then in a terminal:

boot cljs -s

The compiled JavaScript file will be target/main.js.

Compile with advanced optimizations and source maps:

boot cljs -sO advanced

Further Reading

For an example project with a local web server, CLJS REPL, and live-reload, check out boot-cljs-example!

Hacking

To build the boot-cljs JAR and install to your local Maven repository:

boot build-jar

To run the tests:

boot run-tests

To deploy a snapshot to Clojars:

boot build-jar push-snapshot

To deploy a release to Clojars:

boot build-jar push-release

License

Copyright © 2014 Adzerk
Copyright © 2015 Juho Teperi

Distributed under the Eclipse Public License either version 1.0 or (at your option) any later version.