This project formalises a semantics for L4, a legal DSL for the law, and implements a rule engine execution pipeline, along with various language binding libraries for interacting with L4 and the pipeline. Among these is a browser JS ESM library which is used to implement an IDE that parses, transpiles and executes L4, as well as visualise execution traces completely in the browser.
Try out our IDE and rule engine here!
For context, this evolved from an older pipeline involving logical-english-client which utilised Logical English to transpile L4 to Prolog and generate execution traces.
More precisely, this project contains:
-
A denotational semantics for L4 constitutive rules, formalised as an equational theory mapping L4 rules into Horn clauses in the Prolog term algebra (ie. Prolog's concrete syntax), implemented as a parser and transpiler in Clojure / Clojurescript.
Note that we use such Horn clauses as our intermediate representation because for one, they are syntactically close to L4, and are convenient to work with. Moreover, Prolog / Datalog Horn clauses have well-understood semantics, and are widely adopted as the standard format for (and hence highly interoperable with) many rule engines and static analysis tools (including SMT based solvers like Z3). See this section for more details.
-
A SWI-Prolog based rule engine runtime, accompanied by custom Prolog predicates for executing transpiled L4 specifications, and obtaining execution traces.
Note that SWI-Prolog was chosen for convenience (details on why that is here), and that one can easily build a runtime for our Prolog intermediate representation using Z3 for instance. Currently, integrating other such backends is planned but not worked on yet.
-
Language binding libraries that allow other languages to interact with L4 and its execution pipeline:
Language Library status Example usage Clojure / JVM In progress JVM main.clj Clojurescript / ESM in browser ✔️ index.js Clojurescript / CommonJS on NodeJS ✔️ node_example_usage.js Python ✔️ example_usage.py Note that all these bind to the same Clojure + SWI-Prolog code base under the hood so that they have the exact same functionality, behaviour and semantics. See here for more details.
-
An in-browser IDE powered by CodeMirror and the browser ESM library to parse, transpile, execute L4 and visualise execution traces (via guifier) completely in the browser.
See this section for more details about the semantics and pipeline implemented in this project.
- Java
- Clojure
- Babashka
- pnpm
- SWI-Prolog
This project is developed with:
- JDK LTS 21
- Nodejs LTS 20
- SWI-Prolog 9.3.7
# Install npm dependencies.
pnpm install
# Install SWI-Prolog runtime dependencies.
pnpm install:swipl-deps
# Pre-compile our SWI-Prolog runtime library to qlf files.
pnpm build:swipl-qlfs
-
Make sure to setup the project first.
-
Run the following command to start a local dev server in the
public
directory:pnpm start
-
Navigate to http://localhost:8000 in your browser.
The webpage contains some instructions on how to use the IDE.
Run the following command:
pnpm build:all
L4's denotational semantics is given as an equational theory from
the term algebra (ie concrete syntax) of L4 to that of Prolog.
This is primarily documented and implemented by the
l4-rule->prolog-rule
function in
l4_to_prolog.cljc.
-
We use Meander to implement a term rewriting system which orients the equational theory from left to right.
Each rewrite rule has an accompanying comment above it which describes the equation it implements.
-
We rewrite L4's concrete syntax directly into that of Prolog, so that Prolog's concrete syntax is essentially our intermediate representation. We do not have any fancy data types or separate form of abstract syntax in our transpilation pipeline.
-
Prolog syntax makes for an extremely convenient intermediate representation to work with because:
-
The concrete syntax of L4's constitutive rules is a variation of Prolog-style Horn clauses, containing additional syntactic sugar.
-
Prolog is homoiconic, with a concrete syntax that is very close to s-expressions.
-
The transpiler and most of the rest of the project is implemented in Clojure / Clojurescript which is a lisp, and hence convenient for manipulating s-expressions.
-
As mentioned in the overview, such an intermediate representation is a rather standard one adopted by many other downstream formalisms and tools that are based on Horn clauses. These tools utilise relatively similar semantics (which as mentioned, are well-understood), and can readily consume our intermediate representation.
They include:
In other words, adopting Prolog syntax as our intermediate representation keeps our parser and transpiler really lean (ie. under 300 lines of code), and also allows us to equip L4 with standard, well-understood Horn clause based semantics, and facilitates the integration of a wide variety of tools and their functionality (which we hope to achieve eventually).
-
-
We transform nested function applications into predicative syntax, eg:
MIN (SUM 0 1 (PRODUCT 1 2)) 3 < MINUS 2 1
gets expanded to something like:
product_list([1, 2], Var__0), sum_list(([0, 1, Var__0]), Var__1), min_list(([Var__1, 3]), Var__2), minus_list(([2, 1]), Var__3), lt(Var__2, Var__3)
The idea is that during the recursive transformation of a term, whenever we find a nested function application, we:
- Capture its context in a continuation.
- Generate a fresh variable of the form
Var__N
. - Throw the fresh variable to the continuation.
- Lift the function application from its nested context up to the top most term.
- Convert the function application into a predicate application, using the fresh variable as the output variable.
Note that such rules are:
- Formalised with the help of a standard big-step semantics with first-class continuations.
- Implemented using Meander's $ macro to conveniently manipulate nested terms and their contexts (captured as continuations).
Check out the implementation and comments in the code for more details!
We implement a rule engine runtime in SWI-Prolog, along with some custom Prolog predicates, in order to execute L4 specifications that have been transpiled to our intermediate representation, ie the concrete syntax of Prolog.
As mentioned previously, we could have built a runtime using any other Horn clause based formalism, but for now, we chose Prolog, and SWI-Prolog in particular, as it is extremely convenient for the following reasons:
-
Prolog is a backward-chaining rule engine for Horn clauses (or at least it can be used as one, because it uses SLD-resolution), and has powerful meta-programming functionality.
This means that to implement a backward chaining rule engine in Prolog with custom behaviour, one can simply utilise meta-programming to customise the existing behaviour of the Prolog interpreter to suit their needs, rather than have to implement everything like the backward chaining and unification mechanism themselves.
For instance, we can easily implement meta-predicates that utilise SWI-Prolog's meta-programming APIs to hook into the Prolog interpreter and log execution traces. This can also be used to pass callbacks into the interpreter for an interactive Q&A expert system. Without meta-programming, we would have to implement a whole new interpreter ourselves just to get such functionality.
-
SWI-Prolog has good library support, including constraint solving and date libraries.
This lets us easily solve planning problems on top of just executing programs, and integrate date logic reasoning.
-
SWI-Prolog has language bindings to various languages like JS (including browser JS via WASM), Python and Java.
This lets us, without much additional effort, recycle the same codebase and runtime to implement JS, Python and Java libraries that integrate L4 with them, and even execute in the browser. In addition, this ensures that the execution behaviour and semantics across all these libraries is exactly the same.
We provide libraries for Clojure, Java, JS and Python with language bindings to L4, which expose programmatic APIs for these languages to interop with L4 and call various parts of the Clojure parser → Clojure transpiler → SWI Prolog rule engine pipeline, and interop with L4.
This is easy because:
-
Our parser and transpiler are written entirely in Clojure, which compiles to and interops bidirectionally with both JVM bytecode and JS.
This means that just by writing our parser → transpiler pipeline in Clojure, we get to integrate it with Java and JS for free.
-
JsPyBridge offers almost seamless bidirectional interop between JS and Python.
This lets us integrate our Clojure parser → Clojure transpiler with Python as well, for free, because we can interop Clojure ↔ JS ↔ Python
-
SWI-Prolog has good bindings and bidirectional interop with JS via WASM, Python and Java.
Using this, we wrote some glue code in Clojure and Python to connect the output of the Clojure transpiler to the SWI Prolog rule engine to complete the whole pipeline.