The ClojureScript Programming Language, bootstrapped into JavaScript
npm install clojurescript-nodejsClojureScript bootstrapped into JavaScript
==========================================
Making the ClojureScript language usable from nodejs
Command Line Tool Example
-------------------------
``bash`
$ echo '(ns cljs.user (:require [clojure.string :as str])) (println (str/reverse "hello"))' > a.cljs
$ runcljs a.cljs
olleh
JavaScript API Example
----------------------
`javascript
var cljs = require('clojurescript-nodejs');
var code = [
'(ns cljs.user',
' (:require [clojure.string :as str]))',
'(println (str/reverse "hello"))'
].join('\n');
/ Evaluate ClojureScript programs /
cljs.eval(code); // prints olleh
/* Compile ClojureSCript programs
* prints:
* goog.provide('cljs.user');
* goog.require('cljs.core');
* goog.require('clojure.string');
* cljs.core.println.call(null,clojure.string.reverse.call(null,"hello"));
*/
process.stdout.write(cljs.compile(code));
`
REPL
----
This has somewhat limited functionality.
You can evaluate expressions and define functions and variables, but you
can't require additional namespaces.
`clojure`
$ cljsrepl
> (defn square [x] (* x x))
#'global.user/square
> (square 12)
144
The Problem
-----------
I want to write a program in ClojureScript.
I want to break it up into multiple files, and include functions from the
standard library.
I don't want to install java.
I don't want to bootstrap my own compiler.
I just want to write some ClojureScript. Is that too much to ask?
The Solution
------------
This is the ClojureScript compiler
bootstrapped into JavaScript. It can be used to compile non-trivial programs
that require functions defined in external files including the standard library.
This package includes:
* a bootstrapped compiler (ie. a JavaScript library that can
compile/evaluate ClojureScript programs)
* the ClojureScript standard library
* a script (runcljs) for running cljs files
* a REPL
* the source files used to generate a thin wrapper around the compiler and
supply it with a means of resolving required files
* a script for bootstrapping the compiler from the java-based ClojureScript
compiler and generating a fresh standard library
* unit tests
All the functionality except for bootstrapping can be used without java
installed.
API
---
`javascript`
eval(code, libraryPaths=[], isExpression=false, filename=null)
Evaluates ClojureScript from a string.
Returns the value of the last expression.
code the string to evaluate
libraryPaths an array of library paths can optionally be provided.
These should be paths to directories containing ClojureScript
files that will be searched when require statements are
encountered. For more information, see
Require Name Resolution.
isExpression a boolean determining whether to treat the string as
an expression or an entire program
filename filename displayed in error messages
`javascript`
compile(code, libraryPaths=[], isExpression=false, filename=null)
Compiles a ClojureScript program into JavaScript, returning the result as a string.
Arguments have the same meaning as those of eval.
`javascript`
evalfile(path, libraryPaths=[], isExpression=false)
Evaluates a ClojureScript program in a file.
Returns the value of the last expression.
path path to the file to be evaluated
All other arguments are the same as those of eval.
`javascript`
compilefile(path, libraryPaths=[], isExpression=false)
Compiles a ClojureScript program from a file into JavaScript, returning the result as a string.
Arguments have the same meaning as those of evalfile.
Require Name Resolution
-----------------------
Programs evaluated or compiled with this tool may import code from other files.
`clojure
(ns hello.world
(:require [clojure.string]
[foo.bar.baz]))
(println (clojure.string/reverse "Hello, World!"))
(foo.bar.baz/my-fn 1 2 3)
`
The above ClojureScript program requires code from 2 external files.
- clojure.string is part of the ClojureScript standard libraryfoo.bar.baz
- is a user-provided file containing (at least) a function called "my-fn"
clojure.string and foo.bar.baz are namespaces.:require
This tool implements a policy for converting namespaces like these into paths to files
which are loaded by the runtime. This policy is based on observing the directory structure
of the standard library and the namespaces used within it. statements get compiledgoog.require
into calls to from
Google's Closure Library.
Short on documentation on exactly how namespaces are resolved into file paths, I resorted
to reverse engineering. This may be incorrect according to some spec. If someone knows about
such a spec, please contact me and I'll fix this tool!
This will explain how namespaces are convert to the path to a ClojureScript,
Clojure, or JavaScirpt file.
- Start with a namespace (such as foo.bar.baz). It must contain at least 2foo/bar/baz
parts (this is a requirement of the cloSure library).
- Replace periods with path files separators (e.g. "/") and convert to lowercase. foo/bar/baz.EXT
- Search the library paths for the files andfoo/bar/baz/baz.EXT
where EXT is one of .cljs, .cljc or .js. Thefoo/bar/baz.cljs
spec does specify that the extensions should be searched for in this order.
Thus, the list of paths to search for is
, foo/bar/baz/baz.cljs,foo/bar/baz.cljc
, foo/bar/baz/baz.cljc,foo/bar/baz.js
, foo/bar/baz/baz.js.runcljs
- The process searches for these files relative to a number of library paths.
Library paths are supplied as arguments to the compilation/evaluation
functions. The tool uses the directory of the file being evaluatedruncljs
as a library path. Additional library paths can be supplied to with-p
the argument. The last library path to be searched is the standardcljs
library. This tool has the location of the standard library baked into it.
It expects a directory called in the same directory as the core.jslib
file. These are located inside the directory in this repo. cljs`
contains the standard library and is searched last to allow other libraries to
override the standard library's namespaces.
For some examples of files including external files, consult the test directory or
the standard library.
Bootstrapping
-------------
The script bootstrap/bootstrap.sh is used for generating a javascript file
containing the clojure compiler (lib/core.js), and the directory structure
containing the standard library (lib/cljs). When run (with no arguments), it:
* downloads a .jar containing the java-based ClojureScript compiler
* generates a build config file
* compiles the file at src/clojurescript/core.cljs to javascript, bundled with
libraries it includes, storing the result in lib/core.js
* compiles src/clojurescript/core.cljs a second time, configured to spit out the
stand library as separate files. The standard library is then copied to
lib/cljs.
This is the only part of this tool that requires java be installed. Running this
script is only necessary after changing src/clojurescript/core.cljs.
Todo
----
Get the generated compiler into such a state that it can recompile
src/clojurescript/core.cljs without needing java at all.