A compact Notation3 (N3) reasoner in JavaScript.
|
Mission Eyeling aims to make knowledge itself computationally accountable, so every conclusion can be derived, checked, and explained. It does this by keeping reasoning close to explicit facts, rules, and proofs rather than hidden assumptions or opaque workflows. As a compact Notation3 reasoner for JavaScript, Eyeling is designed to fit into practical systems while remaining inspectable enough for researchers, engineers, and agents to understand why a result follows. The ambition is not just to process data, but to support a culture of verifiable knowledge: conclusions that can be exchanged, reproduced, challenged, and improved. |
Eyeling is characterized by:
=> rules derive new facts, while <= rules act as goal-directed definitions.@rdfjs/types.This README is the primary guide to using, extending, and maintaining Eyeling.
Eyeling is designed for people who want a small, inspectable reasoner that can run N3 rules in Node.js, the browser, tests, and RDF-oriented data pipelines.
Eyeling is a compact Notation3 reasoner implemented in JavaScript.
It accepts facts and rules written in N3-style syntax, computes the logical consequences of those rules, and emits newly derived results. It can be used as:
npx eyeling or the eyeling binary;require('eyeling');eyeling/browser;Eyeling is intentionally small and dependency-light. The source tree is organized as a miniature compiler and inference engine: lexer, parser, term model, rule normalization, built-ins, forward chaining, backward proving, printing, RDF-JS adapters, and CLI wiring.
Eyeling is not a database, a triple store, or a full web crawler. It is a reasoner. It reads a finite set of sources, reasons over them, and returns derived output. For persistent storage, indexing at scale, access control, or distributed querying, pair Eyeling with the appropriate storage and application layer.
echo '@prefix : <http://example.org/> .
:Socrates a :Man .
{ ?x a :Man } => { ?x a :Mortal } .' | npx eyeling
Expected output:
@prefix : <http://example.org/> .
:Socrates a :Mortal .
By default, the CLI prints newly derived triples, not the original input facts.
npm install eyeling
Use it from JavaScript:
const { reason } = require('eyeling');
const output = reason({}, `
@prefix : <http://example.org/> .
:Socrates a :Man .
{ ?x a :Man } => { ?x a :Mortal } .
`);
console.log(output);
node eyeling.js examples/socrates.n3
For proof output:
node eyeling.js --proof examples/socrates.n3
Or from JavaScript:
const { reason } = require('eyeling');
materialize(out, 1).
in(done).
out(X) :- in(X).
`);
console.log(output);
A fact is an RDF-like triple:
:Socrates a :Human .
:Human rdfs:subClassOf :Mortal .
Each triple has a subject, predicate, and object. Eyeling supports IRIs, prefixed names, literals, blank nodes, variables, lists, and quoted formulas in the supported N3 subset.
A forward rule uses =>:
{ ?s a ?class . ?class rdfs:subClassOf ?super . }
=>
{ ?s a ?super . } .
Read it as: if the body is provable, derive the head.
A backward rule uses <=:
{ ?x :moreInterestingThan ?y . }
<=
{ ?x math:greaterThan ?y . } .
Read it as: to prove the head, prove the body.
Backward rules are especially useful for derived predicates, reusable definitions, and built-in-backed computations that should not be materialized until needed.
Built-ins are predicates implemented by the engine. Examples include:
(2 3 5) math:sum ?total .
"Hello Eyeling" string:contains "Eye" .
(1 2 3) list:length ?n .
Built-ins are used in rule bodies to test conditions, bind variables, inspect formulas, format strings, work with lists, perform numeric operations, or dereference content.
Eyeling supports log:query as an output-selection mechanism. A program can derive a full closure internally and emit only the results selected by query-style rules.
For human-readable text output, use log:outputString:
@prefix : <http://example.org/> .
@prefix log: <http://www.w3.org/2000/10/swap/log#> .
:run :value "hello" .
{ :run :value ?text }
=>
{ :run log:outputString ?text } .
The CLI renders the log:outputString values directly:
hello
Eyeling supports a predicate-local performance directive for backward rules:
@prefix log: <http://www.w3.org/2000/10/swap/log#> .
:predicate log:memoize true .
A top-level log:memoize true triple is treated as an engine directive rather than an RDF fact. It asks the backward prover to reuse completed answers for that predicate during the current reasoning run. This is especially useful for recursive predicates with overlapping subgoals, such as the direct Fibonacci recurrence or Collatz trajectories where many starting values share suffixes:
:fibonacci log:memoize true .
:collatz log:memoize true .
Memoization is intentionally selective. It can slow down pure generators, predicates with mostly unique states, or small linear recursions because maintaining the table costs more than recomputation. Prefer it for dynamic-programming-shaped predicates that are called repeatedly with the same bound subject or object. The examples examples/fibonacci.n3, examples/collatz-1000.n3, and examples/fundamental-theorem-arithmetic.n3 show cases where the directive helps.
The CLI is exposed as eyeling and backed by bin/eyeling.cjs, which loads the bundled eyeling.js runtime.
eyeling [options] [file-or-url.n3|- ...]
When no file is given and stdin is piped, Eyeling reads from stdin. When multiple inputs are given, Eyeling parses each source separately, merges the ASTs, and reasons once over the merged document.
Run a local file:
eyeling examples/socrates.n3
Pipe a program from stdin:
cat examples/socrates.n3 | eyeling
Use explicit stdin:
eyeling - < examples/socrates.n3
Run facts and rules from separate files:
eyeling facts.n3 rules.n3
Print proof explanations:
eyeling --proof examples/socrates.n3
Print the parsed AST:
eyeling --ast examples/socrates.n3
Enable RDF/TriG compatibility mode:
eyeling --rdf data.trig rules.n3
Process an RDF Message Log one message at a time:
eyeling --rdf --stream-messages rules.n3 messages.trig
| Option | Meaning |
|---|---|
-a, --ast |
Print the parsed AST as JSON and exit. |
--builtin <module.js> |
Load a custom N3 built-in module. Repeatable. |
-d, --deterministic-skolem |
Make log:skolem stable across reasoning runs. |
-e, --enforce-https |
Rewrite http:// IRIs to https:// for log dereferencing built-ins. |
-h, --help |
Show help and exit. |
-p, --proof |
Enable proof explanations. |
-r, --rdf |
Enable RDF/TriG input and output compatibility. |
--stream-messages |
Process RDF Message Logs one message at a time under --rdf. |
--store <name> |
Use an optional persistent fact store. |
--store-clear |
Clear the named persistent store before the run. |
--store-path <dir> |
Use a Node.js persistent-store directory. |
-s, --super-restricted |
Disable all N3 built-ins except implication handling. |
-t, --stream |
Stream derived triples as soon as they are derived. |
-v, --version |
Print the package version and exit. |
The CLI has three important output modes:
--stream, print derived triples as they are found.log:query rules are present, derive the full closure, then print only query-selected triples.When log:outputString appears in the output set, Eyeling writes the string values directly to stdout. This is useful for examples that generate Markdown, reports, or concise verdicts.
A rule that derives false triggers Eyeling’s inference fuse and exits with code 65. JavaScript API calls expose the same code on thrown errors where applicable.
Import from the package root for Node.js:
const {
reason,
reasonStream,
runAsync,
reasonRdfJs,
rdfjs,
registerBuiltin,
unregisterBuiltin,
registerBuiltinModule,
loadBuiltinModule,
listBuiltinIris,
createFactStore,
INFERENCE_FUSE_EXIT_CODE,
} = require('eyeling');
reason(options, input)reason() is the simplest API. It runs the bundled reasoner in a child process and returns stdout as a string.
const { reason } = require('eyeling');
const out = reason({ proof: false }, `
@prefix : <http://example.org/> .
:a :p :b .
{ ?s :p ?o } => { ?s :q ?o } .
`);
console.log(out);
Useful options:
| Option | Description |
|---|---|
proof |
Include proof explanations when true. Defaults to false for API output. |
rdf |
Enable RDF/TriG compatibility mode. |
args |
Extra CLI-style arguments. |
maxBuffer |
Child-process output buffer limit. |
builtinModules |
Custom built-in module path or paths. |
store |
Optional persistent store name or options object; passed through to CLI --store. |
storePath |
Optional Node.js persistent store directory. |
storeClear |
Clear the named persistent store before the run. |
reason() accepts N3 text, supported RDF-JS input objects, AST bundles, and multi-source inputs.
Use sources when facts and rules should be parsed as separate documents and then merged:
const { reason } = require('eyeling');
const output = reason({}, {
sources: [
'@prefix : <http://example.org/> .\n:Socrates a :Man .\n',
'@prefix : <http://example.org/> .\n{ ?x a :Man } => { ?x a :Mortal } .\n',
],
});
Parsing sources separately prevents accidental blank-node label collisions across files.
reasonStream(input, options)reasonStream() runs in process and returns a structured result:
const { reasonStream } = require('eyeling');
const result = reasonStream(`
@prefix : <http://example.org/> .
:a :p :b .
{ ?s :p ?o } => { ?s :q ?o } .
`, {
includeInputFactsInClosure: false,
onDerived({ triple }) {
console.log('derived:', triple);
},
});
console.log(result.closureN3);
Result shape:
| Field | Meaning |
|---|---|
prefixes |
Prefix environment used for parsing and printing. |
facts |
Saturated closure as internal triples. |
derived |
Derived facts with explanation metadata. |
queryMode |
True when log:query output selection was used. |
queryTriples |
Query-selected output triples. |
queryDerived |
Query-selected derived facts with metadata. |
closureN3 |
Rendered closure or selected output as N3/TriG-compatible text. |
closureQuads |
RDF-JS quads when rdfjs: true is used. |
queryQuads |
RDF-JS query output quads when available. |
Useful options:
| Option | Description |
|---|---|
baseIri |
Base IRI for relative IRI resolution. |
proof |
Include proof explanations in closureN3. |
includeInputFactsInClosure |
Include original facts in closureN3. Defaults to true. |
onDerived |
Callback called for derived or query-selected output. |
enforceHttps |
Apply HTTPS rewriting for dereferencing built-ins. |
rdf |
Enable RDF/TriG compatibility mode. |
rdfjs |
Also emit RDF-JS quads where conversion is possible. |
dataFactory |
Custom RDF-JS DataFactory. |
skipUnsupportedRdfJs |
Skip N3-only terms when producing RDF-JS quads. |
builtinModules |
Register custom built-ins before reasoning. |
runAsync(input, options)runAsync() is the async execution API. Without a store option it keeps the same in-memory behavior as reasonStream(), but can also normalize async RDF-JS iterables before reasoning. With store, Eyeling opens a named persistent fact store, adds the new explicit facts, reuses facts already present in that store, reasons over the combined closure, and writes newly inferred facts back as inferred facts.
const { runAsync } = require('eyeling');
await runAsync(input); // memory store
await runAsync(input, {
store: 'my-dataset',
});
await runAsync(input, {
store: {
name: 'my-dataset',
clear: true,
path: './.eyeling-store', // Node.js path override
},
});
Named persistent stores are created automatically when first opened. Persistent stores use a term dictionary plus spo, pos, and osp triple indexes. Exact lookup and all subject/predicate/object bound-pattern scans are available through the FactStore API:
const { createFactStore, rdfjs } = require('eyeling');
const store = await createFactStore({ name: 'my-dataset' });
for await (const triple of store.match(null, rdfjs.namedNode('http://example.org/p'), null)) {
console.log(triple);
}
await store.close();
Node.js uses classic-level when it is installed and falls back to a small JSON-file key/value backend for dependency-free use and tests. Browser runtimes use IndexedDB through the same abstraction. The current synchronous reasonStream() path remains the default and does not open persistent storage.
CLI equivalents:
eyeling input.n3
# memory store
eyeling input.n3 --store my-dataset
# persistent store
eyeling input.n3 --store my-dataset --store-clear
# clear persistent store first
eyeling input.n3 --store my-dataset --store-path ./.eyeling-store
# Node.js path override
# Stream line-oriented RDF input into a store without reading one giant string.
eyeling --rdf big.nt --store my-dataset --store-path ./.eyeling-store
# Stream RDF Message Logs one message at a time and persist facts/inferences.
eyeling --rdf --stream-messages rules.n3 messages.trig --store my-dataset
reasonRdfJs(input, options)reasonRdfJs() returns an async iterable of derived RDF-JS quads:
const { reasonRdfJs } = require('eyeling');
for await (const quad of reasonRdfJs({
n3: `
@prefix : <http://example.org/> .
:a :p :b .
{ ?s :p ?o } => { ?s :q ?o } .
`,
})) {
console.log(quad.subject.value, quad.predicate.value, quad.object.value);
}
Use skipUnsupportedRdfJs: true when your rules may derive N3-only terms such as quoted formulas that cannot be represented as ordinary RDF-JS quads.
Use the browser entry point in browser or worker runtimes:
import eyeling, { reasonStream } from 'eyeling/browser';
const result = reasonStream(`
@prefix : <http://example.org/> .
:a :p :b .
{ ?s :p ?o } => { ?s :q ?o } .
`);
console.log(result.closureN3);
console.log(eyeling.version);
The browser entry loads dist/browser/eyeling.browser.js and exposes the API through globalThis.eyeling.
Eyeling includes a lightweight RDF-JS DataFactory and adapters for supported RDF-JS terms and quads. Its TypeScript declarations reuse the official @rdfjs/types interfaces, so Eyeling quads, terms, and data factories can be passed to other RDF-JS libraries without local type casts.
const { reasonStream, rdfjs } = require('eyeling');
const ex = 'http://example.org/';
const input = {
quads: [
rdfjs.quad(
rdfjs.namedNode(`${ex}Socrates`),
rdfjs.namedNode(`${ex}type`),
rdfjs.namedNode(`${ex}Man`),
),
],
n3: `
@prefix : <http://example.org/> .
{ ?x :type :Man } => { ?x :type :Mortal } .
`,
};
const result = reasonStream(input, { rdfjs: true });
console.log(result.closureQuads);
TypeScript users can use Eyeling directly with @rdfjs/types:
import type { DataFactory, Quad } from '@rdfjs/types';
import { rdfjs, reasonRdfJs } from 'eyeling';
const ex = 'http://example.org/';
const factory: DataFactory<Quad> = rdfjs;
const facts: Quad[] = [
factory.quad(
factory.namedNode(`${ex}Socrates`),
factory.namedNode(`${ex}type`),
factory.namedNode(`${ex}Man`),
),
];
for await (const quad of reasonRdfJs({
quads: facts,
n3: `
@prefix : <http://example.org/> .
{ ?x :type :Man } => { ?x :type :Mortal } .
`,
})) {
const derived: Quad = quad;
console.log(derived.subject.value, derived.predicate.value, derived.object.value);
}
The built-in rdfjs factory implements the standard RDF-JS constructors for named nodes, blank nodes, literals, default graph terms, variables, and quads, plus fromTerm() and fromQuad() clone helpers. @rdfjs/types is installed with Eyeling because the public declarations import it.
Supported RDF-JS input terms include named nodes, blank nodes, literals, variables, default graph terms, and default-graph quads. Named-graph input quads are rejected clearly unless handled through N3/TriG compatibility mode.
Use RDF-JS when you want Eyeling to sit inside a JavaScript RDF pipeline. Use raw N3 input when you need N3-only features such as quoted formulas or N3 rules represented directly in source text.
RDF compatibility mode is enabled with --rdf on the CLI or { rdf: true } in the API.
Use it when working with RDF/TriG-oriented syntax and RDF 1.2 constructs:
eyeling --rdf input.trig rules.n3
const result = reasonStream(input, { rdf: true });
In RDF mode, Eyeling accepts and serializes RDF-compatible forms such as:
PREFIX and BASE directives;RDF 1.2 triple terms require explicit RDF compatibility mode. This protects ordinary N3 users from accidentally mixing parser modes.
Eyeling supports RDF Message Logs, including parser-level message delimiters, under RDF compatibility mode.
A message log starts with a message version and separates messages with MESSAGE:
VERSION "1.2-messages"
PREFIX : <https://example.org/messages#>
:obs1 :value 21 .
MESSAGE
# Empty heartbeat message.
MESSAGE
:obs2 :value 22 .
Run the message log with rules:
eyeling --rdf rules.n3 messages.trig
For one-message-at-a-time processing:
eyeling --rdf --stream-messages rules.n3 messages.trig
--stream-messages can also be combined with --store to create/reuse a named store and persist each message’s explicit facts and inferred facts while keeping only one replay message in memory at a time:
eyeling --rdf --stream-messages rules.n3 messages.trig --store my-dataset --store-path ./.eyeling-store
Eyeling materializes a replay view under the eymsg: vocabulary:
@prefix eymsg: <https://eyereasoner.github.io/eyeling/vocab/message#> .
The replay view includes stream resources, ordered envelopes, offsets, payload kind, and payload graphs. Rules can inspect each payload graph with formula-aware built-ins such as log:includes, preserving message boundaries instead of treating all messages as one merged graph.
Important semantics:
See the included examples:
eyeling -r examples/rdf-messages.n3 examples/input/rdf-messages.trig
eyeling -r examples/rdf-message-flow.n3 examples/input/rdf-message-flow.trig
eyeling -r --stream-messages examples/rdf-message-flow.n3 examples/input/rdf-message-flow.trig
eyeling -r --stream-messages examples/alma-rdf-messages.n3 https://ugent-lib-opendata-prd.s3.ugent.be/alma-rdf/rdf-messages.20260404.nt
The Alma RDF Message Log example intentionally keeps the message log as a URL, because the source .nt file is larger than 9 GB.
Eyeling implements 104 public SWAP-style built-in predicates for the N3 engine across these namespaces. The authoritative machine-readable catalog is eyeling-builtins.ttl; this README lists the public names so users do not need to infer support from examples or internal dispatch code.
| Namespace | Count | Built-ins |
|---|---|---|
crypto: |
4 | sha, md5, sha256, sha512 |
math: |
26 | equalTo, notEqualTo, greaterThan, lessThan, notLessThan, notGreaterThan, sum, product, difference, quotient, integerQuotient, remainder, rounded, exponentiation, absoluteValue, acos, asin, atan, sin, cos, tan, sinh, cosh, tanh, degrees, negation |
time: |
8 | day, hour, minute, month, second, timeZone, year, localTime |
list: |
15 | append, first, rest, iterate, last, memberAt, remove, member, in, length, notMember, reverse, sort, map, firstRest |
rdf: |
2 | first, rest |
log: |
22 | equalTo, notEqualTo, conjunction, conclusion, content, semantics, semanticsOrError, parsedAsN3, rawType, dtlit, langlit, implies, impliedBy, query, includes, notIncludes, collectAllIn, forAllIn, skolem, uri, trace, outputString |
string: |
19 | concatenation, contains, containsIgnoringCase, endsWith, startsWith, equalIgnoringCase, notEqualIgnoringCase, greaterThan, lessThan, notGreaterThan, notLessThan, matches, notMatches, replace, scrape, format, length, charAt, setCharAt |
dt: |
8 | datatype, lexicalForm, language, validForDatatype, invalidForDatatype, sameValueAs, differentValueFrom, canonicalLiteral |
| Total | 104 |
The catalog marks each built-in with a coarse kind: ex:Test, ex:Function, ex:Relation, ex:Generator, ex:IO, ex:Meta, or ex:SideEffect.
Eyeling provides datatype built-ins in the namespace https://eyereasoner.github.io/eyeling/datatype#, usually used with the prefix dt:. They are intended for declarative datatype reasoning in N3 rule sets, including OWL 2 RL-style rules that need XSD value-space semantics without hard-coding OWL into the engine.
Supported operations include:
dt:datatype, dt:lexicalForm, and dt:language for literal inspection. dt:datatype returns the literal’s actual datatype IRI only, so RDF string literals return xsd:string; it does not return rdfs:Literal, which is a class of literals rather than a datatype IRI;dt:validForDatatype and dt:invalidForDatatype for lexical validity and datatype membership checks, either as ?literal dt:validForDatatype ?datatype tests or as tuple-to-boolean checks like (?literal ?datatype) dt:validForDatatype true;dt:sameValueAs and dt:differentValueFrom for value-space equality and inequality;dt:canonicalLiteral for canonical literal production.The built-ins use strict datatype lexical validation for these checks, including exact dateTime rollover/canonicalization such as 24:00:00 normalizing to the following day. They cover RDF language strings, rdf:PlainLiteral, rdf:XMLLiteral, rdfs:Literal, and the OWL 2 RL-relevant XSD set: xsd:string, xsd:normalizedString, xsd:token, xsd:language, xsd:Name, xsd:NCName, xsd:NMTOKEN, xsd:boolean, xsd:decimal, xsd:integer and its bounded integer subtypes, xsd:float, xsd:double, xsd:hexBinary, xsd:base64Binary, xsd:anyURI, xsd:dateTime, and xsd:dateTimeStamp. Lexical validation is intentionally strict for conformance: string-derived datatypes with whitespace-collapse facets must already be written in canonical collapsed lexical form, xsd:float and xsd:double enforce finite value ranges, xsd:anyURI rejects spaces, unsafe delimiters, and malformed percent escapes, and XML literals must be well-formed XML fragments.
Numeric built-ins support common XSD numeric literals. Integer-oriented operations use BigInt where possible, with safety limits to avoid accidental memory exhaustion. Date/time comparisons and timestamp arithmetic are supported for relevant operations.
Eyeling supports both native N3 list terms and materialized RDF collections. Anonymous rdf:first/rdf:rest collections can be materialized into list terms, while named list nodes keep their identity.
Formula-aware built-ins make Eyeling useful for meta-reasoning. log:includes, log:notIncludes, log:collectAllIn, and log:forAllIn prove goals inside formula scopes. log:query selects output triples, while log:outputString writes selected string values directly to stdout.
log:memoize is a top-level performance directive for backward predicates, not an RDF-producing fact and not a general-purpose built-in. Use it when a recursive predicate has overlapping subgoals whose completed answers are likely to be requested again.
log:semantics, log:content, and related built-ins may dereference sources. Use --enforce-https or { enforceHttps: true } in environments where HTTP-to-HTTPS rewriting is required.
Custom built-ins let applications extend Eyeling without modifying the core engine.
Create hello-builtin.js:
module.exports = ({ registerBuiltin, internLiteral, unifyTerm, terms }) => {
const { Var } = terms;
registerBuiltin('http://example.org/custom#hello', ({ goal, subst }) => {
const value = internLiteral('"world"');
if (goal.o instanceof Var) {
return [{ ...subst, [goal.o.name]: value }];
}
const next = unifyTerm(goal.o, value, subst);
return next === null ? [] : [next];
});
};
Use it:
eyeling --builtin ./hello-builtin.js program.n3
Program:
@prefix : <http://example.org/> .
@prefix cb: <http://example.org/custom#> .
{ :x cb:hello ?value }
=>
{ :x :value ?value } .
Expected derived output:
:x :value "world" .
const { registerBuiltin, reason } = require('eyeling');
registerBuiltin('http://example.org/custom#always', ({ subst }) => [subst]);
const out = reason({}, `
@prefix : <http://example.org/> .
@prefix cb: <http://example.org/custom#> .
{ :x cb:always true } => { :x :ok true } .
`);
registerBuiltinModule() accepts these shapes:
// Function form
module.exports = (api) => {
api.registerBuiltin('http://example.org/custom#p', handler);
};
// Object with register()
module.exports = {
register(api) {
api.registerBuiltin('http://example.org/custom#p', handler);
},
};
// Builtin map
module.exports = {
'http://example.org/custom#p': handler,
};
// Builtin map under builtins/default
module.exports = {
builtins: {
'http://example.org/custom#p': handler,
},
};
Handlers must return an array of substitution deltas. Return an empty array for failure and [subst] for success without new bindings.
Eyeling combines forward saturation with backward proving.
At a high level:
parse sources
↓
normalize terms, rules, and lists
↓
initialize fact set
↓
repeat until no new facts appear:
for each forward rule:
prove the rule body with the backward prover
for each solution:
instantiate and add the rule head
activate any newly derived rules
stop if false is derived
↓
render derived output, query-selected output, proof output, or strings
Forward rules are the outer control loop. They gradually saturate the fact set by adding ground consequences.
{ ?x :parent ?y }
=>
{ ?x :ancestor ?y } .
{ ?x :parent ?y . ?y :ancestor ?z }
=>
{ ?x :ancestor ?z } .
The backward prover solves rule bodies. It can match current facts, use backward rules, and invoke built-ins. This lets forward rules depend on predicates that are computed on demand.
{ ?x :interestingComparedWith ?y }
<=
{ ?x math:greaterThan ?y } .
{ 5 :interestingComparedWith 3 }
=>
{ :example :works true } .
For selected predicates, :predicate log:memoize true . enables tabling of completed backward answers. This preserves the normal declarative reading of the backward rules but can turn repeated recursive work into table lookups. The directive is removed from the data facts before reasoning, so it does not appear in normal output.
Eyeling treats top-level log:implies and log:impliedBy as rule forms and can activate derived implication facts as live rules during reasoning. This supports programs that derive rules as part of their logic.
Derived facts are indexed and deduplicated. Saturation stops when no rule can add a new fact. This avoids echoing already-known facts and keeps recursive programs such as transitive closure finite when the closure is finite.
If a rule derives false, Eyeling treats that as a reasoning failure and exits with INFERENCE_FUSE_EXIT_CODE, which is 65.
{ :policy :violated true } => false .
Use this pattern for integrity constraints, policy failures, and tests that should fail when an unwanted condition is provable.
Eyeling is organized as a set of small modules under lib/ plus packaging and browser glue.
input text / RDF-JS / AST
│
▼
lib/lexer.js tokenization and RDF compatibility normalization
│
▼
lib/parser.js N3/TriG-ish parser to internal AST
│
▼
lib/multisource.js source-level parsing and AST merging
│
▼
lib/prelude.js term model, triples, rules, prefixes, namespaces
│
▼
lib/engine.js forward chain, backward prover, rule activation
│
├── lib/builtins.js built-in predicates and custom registry
├── lib/deref.js dereferencing helpers
├── lib/skolem.js skolemization helpers
├── lib/time.js date/time helpers
└── lib/trace.js tracing support
│
▼
lib/printing.js / lib/explain.js
│
▼
CLI output, API result, proof document, RDF-JS quads, or browser result
| Path | Responsibility |
|---|---|
index.js |
Public Node package API. Wraps CLI bundle for reason() and exports in-process APIs. |
bin/eyeling.cjs |
Executable CLI shim. |
lib/entry.js |
Bundle entry that exposes public APIs and selected playground internals. |
lib/cli.js |
CLI argument handling, source loading, syntax errors, stream message mode. |
lib/engine.js |
Core reasoning engine, proof collection, stream APIs, RDF-JS output hooks. |
lib/store.js |
Optional async fact-store abstraction with memory and persistent backends. |
lib/builtins.js |
Built-in predicates, custom built-in registry, helper API. |
lib/lexer.js |
Lexer and compatibility normalization. |
lib/parser.js |
Parser for supported N3/RDF syntax. |
lib/prelude.js |
Core term classes, namespaces, triples, rules, prefix environment. |
lib/multisource.js |
Parse several documents independently and merge their ASTs. |
lib/rdfjs.js |
RDF-JS DataFactory and conversion adapters. |
lib/printing.js |
N3/TriG-compatible rendering. |
lib/explain.js |
Proof and explanation rendering. |
lib/deref.js |
Dereferencing support for log built-ins. |
lib/skolem.js |
Deterministic skolem term construction. |
lib/time.js |
Date/time parsing and formatting helpers. |
tools/bundle.js |
Builds eyeling.js and the browser bundle. |
Eyeling deliberately has a small public API:
reason() for simple Node use;reasonStream() for structured in-process reasoning;reasonRdfJs() for async RDF-JS output;rdfjs for a built-in data factory;INFERENCE_FUSE_EXIT_CODE for callers that need to distinguish logical failure from ordinary runtime failure.Everything else should be treated as internal unless explicitly documented.
.
├── README.md Project overview, user guide, and maintainer guide
├── LICENSE.md MIT license
├── package.json Package metadata, scripts, exports, engine range
├── index.js Node API entry
├── index.d.ts TypeScript declarations
├── eyeling.js Bundled Node runtime and CLI target
├── eyeling-builtins.ttl Built-in catalog in RDF
├── bin/ CLI executable shim
├── lib/ Source modules
├── dist/browser/ Browser bundle and ESM wrapper
├── docs/ Project documentation
├── examples/ N3 examples, RDF message inputs, and generated decks
├── spec/ RDF 1.2 parser test adapter
├── test/ API, built-in, store, example, package, playground, and stream tests
├── tools/ Build tooling
├── playground.html Browser playground
└── demo.html Simple browser demo
The package publishes the source modules, tests, examples, bundled runtime, browser bundle, declarations, README, license, and built-in catalog.
The repository contains more than two hundred N3 examples under examples/, RDF Message input files under examples/input/, and presentation-oriented Markdown decks under examples/deck/.
| Example | What it demonstrates |
|---|---|
examples/socrates.n3 |
Basic class inference. |
examples/backward.n3 |
Backward rule proving with a math built-in. |
examples/age.n3 |
Literal propagation. |
examples/family-cousins.n3 |
Multi-hop relational inference. |
examples/dijkstra.n3 |
Graph/path reasoning. |
examples/list-map.n3 |
List processing. |
examples/string-builtins-tests.n3 |
String built-ins. |
examples/math-builtins-tests.n3 |
Numeric built-ins. |
examples/rdf-messages.n3 |
RDF Message Log replay. |
examples/context-schema-audit.n3 |
Quoted-context schema validation with log:includes and list arity checks. |
These examples demonstrate log:memoize on predicates that benefit from reusing completed backward-rule answers:
| Example | Memoized predicate | Why it helps |
|---|---|---|
examples/fibonacci.n3 |
:fibonacci |
The direct recurrence repeatedly asks for the same smaller Fibonacci numbers. |
examples/collatz-1000.n3 |
:collatz |
Many starting values share Collatz suffix trajectories. |
examples/fundamental-theorem-arithmetic.n3 |
:smallestDivisorFrom, :factorSmallest |
Factorization and independent checks reuse trial-division and factor-list subgoals. |
npm run test:examples
Proof-only example checks:
npm run test:examples:proof
Many advanced examples use log:outputString to emit Markdown reports. This keeps the logical derivation and presentation in one N3 program. Run them from the CLI and redirect stdout when needed:
eyeling examples/rdf-message-flow.n3 examples/input/rdf-message-flow.trig > report.md
Package scripts are defined in package.json.
| Script | Purpose |
|---|---|
npm run build |
Rebuild eyeling.js and browser artifacts. |
npm run test:packlist |
Verify the package file list. |
npm run test:api |
Run API and stream-message API tests. |
npm run test:builtins |
Validate custom built-in contracts. |
npm run test:examples |
Run example corpus tests. |
npm run test:examples:proof |
Run proof-output checks for examples. |
npm run test:manifest |
Validate example/test manifest expectations. |
npm run test:playground |
Check playground serving headers. |
npm run test:package |
Verify package-level behavior. |
npm run test:store |
Verify memory and persistent fact-store matching. |
npm run test:rdf12 |
Run RDF 1.2 Turtle, N-Triples, N-Quads, and TriG syntax suites. |
npm test |
Build and run the full suite. |
Use the full test suite as the authoritative project check:
npm test
For quick API-level feedback during development:
npm run build
npm run test:api
npm run test:builtins
The tests exercise:
log:outputString rendering;npm install
npm run build
This regenerates:
eyeling.jsdist/browser/eyeling.browser.jsdist/browser/index.mjsMost logic lives in lib/. Prefer small, focused changes:
npm test before committing or publishing.lib/builtins.js or as an external custom built-in.test/builtins.test.js or behavior tests in test/api.test.js.eyeling-builtins.ttl.lib/lexer.js and/or lib/parser.js.lib/printing.js remains valid.A good example should include:
log:outputString only when human-readable report output is intended;The package metadata publishes Eyeling to npm with:
The repository includes GitHub workflows for pages, npm publishing, RDF 1.2 compliance, and releases.
Before versioning or publishing:
npm test
npm version patch # or minor / major
The preversion script runs the full test suite. The postversion script pushes the branch and tags.
Eyeling prints help when no positional input is provided and stdin is interactive. Provide a file, URL, -, or pipe data into stdin.
eyeling examples/socrates.n3
cat examples/socrates.n3 | eyeling
eyeling - < examples/socrates.n3
Enable RDF compatibility mode:
eyeling --rdf input.trig
or:
reasonStream(input, { rdf: true });
--stream-messages fails immediately--stream-messages requires RDF mode and cannot be combined with --ast, --stream, or proof output. It can be combined with --store.
Use:
eyeling --rdf --stream-messages rules.n3 messages.trig
Check the following:
--rdf?For debugging, try --proof or create a smaller reproduction with only the relevant facts and rule.
Default CLI output prints newly derived facts. Use reasonStream() with includeInputFactsInClosure: true when you need the complete closure including input facts.
Some N3 terms cannot be represented as ordinary RDF-JS quads. Use:
reasonStream(input, { rdfjs: true, skipUnsupportedRdfJs: true });
or keep the N3 rendering in closureN3.
A rule derived false. This is usually an integrity constraint or policy failure, not a parser error.
Use --enforce-https or { enforceHttps: true } when policy requires HTTPS. Also ensure the runtime has network access and that the remote source returns a supported RDF/N3-compatible content type.
Custom built-ins are code. Only load built-in modules you trust. In server environments, do not allow arbitrary users to provide --builtin paths or dynamically registered handlers.
log:semantics, log:content, and related built-ins may dereference IRIs. Treat this like network I/O:
--super-restricted for highly constrained execution.Recursive rules, generators, large joins, and high-cardinality facts can produce large closures. Eyeling includes duplicate suppression and safety caps for some operations, but application-level limits are still important for untrusted workloads.
Proof output can include source file labels and line references. Avoid exposing proof documents directly when source paths or input details are sensitive.
| Term | Meaning |
|---|---|
| AST | Abstract syntax tree produced by parsing N3/RDF input. |
| Backward chaining | Goal-directed proving: to prove a goal, prove supporting facts/rules/built-ins. |
| Built-in | Predicate implemented by JavaScript code rather than by input facts alone. |
| Closure | The set of facts available after reasoning reaches a fixpoint. |
| Derived fact | A fact added by a rule, not directly present as an input fact. |
| Fact | A triple asserted in the input or derived during reasoning. |
| Forward chaining | Saturation strategy that repeatedly applies rules to derive new facts. |
| Formula | A quoted graph-like N3 term that can be inspected by formula built-ins. |
| IRI | Internationalized Resource Identifier used to identify resources and predicates. |
| N3 | Notation3, an RDF-compatible notation with rules and formulas. |
| Prefix environment | Mapping from short prefixes such as : or math: to full IRI bases. |
| RDF-JS | JavaScript interface conventions for RDF terms, quads, and data factories. |
| RDF Message Log | Ordered record of RDF messages separated by message delimiters. |
| Skolemization | Replacing existential blank nodes with generated identifiers. |
| Substitution | Mapping from variables to terms during proof search. |
| Triple | Subject-predicate-object statement. |