A language that means business.
Lemma is a pure, declarative language for business rules. It flows like natural language and encodes pricing rules, tax calculations, eligibility criteria, contracts, policies and law. Stakeholders can read them, systems can evaluate them.
Rules in Lemma are transparent, deterministic, logically consistent, temporally bound and explainable. So with the same data for the same spec and the same effective point in time, you will get the same result. Lemma can also tell you why you got that result. Audits and tracing become trivial, even as time passes and rules change.
spec pricing 2026-01-01
data quantity: number
data is_vip: false
rule discount: 0%
unless quantity >= 10 then 10%
unless quantity >= 50 then 20%
unless is_vip then 25%
rule price: quantity * 20
rule discounted_price: price - price * discount
The last matching unless wins, mirroring how business rules, legal documents, and SOPs are written: "In principle X applies, unless Y, unless Z..."
Laws, policies, and business rules traditionally exist in natural language. While humans must understand these rules, we rely on systems to enforce them. Over time, organizations have built massive IT infrastructures to house these rules; however, as both the regulations and the systems evolve, they become harder to manage and the disconnect between them grows.
Lemma provides a single source of truth. Rules written in Lemma are human-readable, time-aware, and pure. Its logic engine guarantees deterministic and logically consistent outcomes through static analysis — invalid specs are rejected before evaluation ever runs. Furthermore, Lemma provides unrivaled auditability by explaining exactly how rules were applied for every evaluation.
This allows you to implement policy changes rapidly without compromising compliance. Lemma requires no database and maintains no state; by design, it is secure, able to run within existing applications and yes, it is blazingly fast.
Lemma aims to combine deterministic evaluation, transparent explanations, temporal versioning (rules that evolve on a timeline, separate from how you deploy code), registry-style sharing of specs, and interop (CLI, HTTP, WASM, MCP, and stable language bindings). The engine already exposes inversion (constraint-style “what inputs satisfy this outcome?”) via the Rust and Hex APIs; CLI and user docs for inversion are still evolving. Planned work includes tables as a first-class data type for data-driven rules and performance competitive with high performance programming languages.
AI models operate on approximations. The complexity of their neural networks makes tracing decisions ("explaining") practically impossible. While they excel at natural language, they are ill-suited for mathematics, strict protocols, or compliance.
Lemma provides certainty and transparency. Every result is exact, verifiable, and delivered in microseconds. Lemma offers seamless interoperability, allowing you to ground your AI systems in deterministic logic.
CLI (from crates.io/crates/lemma):
cargo install lemmaOr via npm:
npm install -g lemmaRust library (from crates.io/crates/lemma-engine):
cargo add lemma-engine --rename lemmaCreate shipping.lemma:
spec shipping
data money: quantity
-> unit eur 1.00
-> unit usd 1.19
-> decimals 2
-> minimum 0 eur
data weight: quantity
-> unit kilogram 1
-> unit gram 0.001
data is_express: true
data package_weight: 2.5 kilogram
rule express_fee: 0 eur
unless is_express then 4.99 eur
rule base_shipping: 5.99 eur
unless package_weight > 1 kilogram then 8.99 eur
unless package_weight > 5 kilogram then 15.99 eur
rule total_cost: base_shipping + express_fee
Run it:
$ lemma run shipping
┌───────────────┬───────────┐
│ base_shipping ┆ 8.99 eur │
├───────────────┼───────────┤
│ express_fee ┆ 4.99 eur │
├───────────────┼───────────┤
│ total_cost ┆ 13.98 eur │
└───────────────┴───────────┘
Override data from the command line:
lemma run shipping is_express=false package_weight=6.0Define custom types with units, constraints, and automatic conversion:
spec type_examples
data money: quantity
-> unit eur 1.00
-> unit usd 1.19
-> decimals 2
-> minimum 0
data status: text
-> option "active"
-> option "inactive"
data discount: ratio
-> minimum 0%
-> maximum 100%
Primitive types: boolean, number, quantity (with units; elapsed time via trait duration, calendar periods via trait calendar), text, date, time, ratio, and ranges (date range, time range, number range, quantity range, ratio range, plus named <type> range). Import stdlib with uses lemma units (units.duration, units.calendar, …).
Reference data and rules across specs:
spec employee
data years_service: 8
spec leave_policy
data base_leave_days: 25
data bonus_leave_days: 5
data senior_threshold: 5
spec leave_entitlement
uses employee
uses leave_policy
rule is_senior:
employee.years_service >= leave_policy.senior_threshold
rule annual_leave_days: leave_policy.base_leave_days
unless is_senior
then leave_policy.base_leave_days + leave_policy.bonus_leave_days
Multiple versions of a spec can coexist. The engine resolves the correct one based on a point in time:
spec pricing
data base_price: 20
data quantity: number
rule total: base_price * quantity
spec pricing 2025-01-01
data base_price: 25
data quantity: number
rule total: base_price * quantity
lemma run pricing --effective 2024-06-01 # uses base_price: 20
lemma run pricing --effective 2025-06-01 # uses base_price: 25When type constraints are not enough, veto blocks a rule entirely:
spec performance_review
data start_date: date
data review_date: date
data performance_score: number
-> minimum 0
-> maximum 100
rule bonus_percentage: 0%
unless performance_score >= 70 then 5%
unless performance_score >= 90 then 10%
unless review_date < start_date
then veto "Review date must be after start date"
A vetoed rule produces no result. See veto semantics.
Reference shared specs from a registry with @:
spec invoicing
uses fin: @lemma/std finance
data subtotal: 250 eur
data tax_rate: 21%
rule tax: subtotal * tax_rate
rule total: subtotal + tax
lemma fetch --all # fetch all @... dependencies
lemma fetch @lemma/std -f # force re-fetch if content changedlemma run pricing # evaluate all rules
lemma run pricing --rules=total,tax # specific rules only
lemma run pricing quantity=10 is_vip=true # override data
lemma run --interactive # interactive mode
lemma run pricing --effective 2025-01-01 # temporal query
lemma run pricing --json # JSON output
lemma run pricing -x # show reasoning
lemma schema pricing # spec schema
lemma list # list all specs
lemma format # format .lemma files
lemma fetch # fetch registry dependencieslemma server --prefix ./policies
# Evaluate via query parameters
curl "http://localhost:8012/pricing?quantity=10&is_member=true"
# Evaluate via JSON body
curl -X POST http://localhost:8012/pricing \
-H "Content-Type: application/json" \
-d '{"quantity": 10, "is_member": true}'
# Evaluate specific rules
curl "http://localhost:8012/pricing/discount,total?quantity=10"Routes: GET / (list specs), GET /openapi.json, GET /docs (interactive API docs), GET /health
Live-reload with --watch:
lemma server --prefix ./policies --watchAI assistants interact with Lemma specs via the Model Context Protocol:
lemma mcp # read-only
lemma mcp --admin # enable spec creationnpm install @lemmabase/lemma-engineimport { Lemma } from '@lemmabase/lemma-engine';
const engine = await Lemma();docker pull ghcr.io/lemma/lemma:latest
# Run a spec
docker run --rm -v "$(pwd):/specs" ghcr.io/lemma/lemma run shipping
# Deploy as HTTP API
docker run -d -p 8012:8012 -v "$(pwd):/specs" ghcr.io/lemma/lemma \
server /specs --host 0.0.0.0 --port 8012Supports linux/amd64 and linux/arm64.
- Language Guide -- specs, data, rules, types
- LLM guide (llms.txt) -- authoring Lemma from business logic
- Composing specs --
uses, temporal versions, pins - Reference -- operators, literals, syntax
- Veto Semantics -- when rules produce no value
- Examples -- example
.lemmafiles - CLI Reference -- all commands and flags
- Registry -- shared specs and
@references
Lemma is pre-1.0. The language and APIs are stable for most use cases, but breaking changes may occur between minor versions. Pin your dependency version and review the changelog before upgrading.
Contributions welcome! See contributing for setup and workflow.
From the repository root, run cargo precommit before opening a PR. It runs versions-verify, Hex mix precommit, VS Code npm precommit, fmt --check, Clippy, Nextest, WASM npm build.js + test.js, and cargo-deny (install cargo-nextest, cargo-deny, Elixir/Mix, Node.js, and wasm-pack first). SDK steps always run, not only when their directories change. When bumping the workspace release version, use cargo bump <version> and cargo verify so every mirrored copy stays aligned (see xtask/README.md).
Apache 2.0 -- see LICENSE for details.
GitHub -- Issues -- Documentation -- WASM