Skip to main content

Hornet: An embedded DSL for Logic Programming in Python.

Project description

Hornet

Horn clauses via Expression Trees — a Prolog-like embedded DSL for Python ≥ 3.13.

Hornet lets you write logic programs directly in Python. Instead of parsing Prolog strings, it hijacks Python's operator overloading and __call__ syntax to build expression trees, which a resolution engine then solves via unification and backtracking.


Installation

pip install hornet-dsl

Requires Python 3.13+. Dependencies: toolz, immutables.


Core Concepts

Terms

Hornet's term algebra mirrors Prolog's:

Hornet Prolog equivalent
Variable('X') / symbols.X X (logic variable)
Atom('foo') / symbols.foo foo (atom)
symbols.foo(X, Y) foo(X, Y) (compound term)
[1, 2, 3] / promote([1,2,3]) [1,2,3] (list)
[H | T] via BitOr [H|T] (cons cell)

Import symbols dynamically from hornet.symbols. Names starting with uppercase become Variables; lowercase become Atoms; _ is the anonymous wildcard.

from hornet.symbols import X, Y, parent, mortal, human

Facts and Rules

Facts and rules are built with .when():

db.tell(
    parent('socrates', 'sophroniscus'),   # fact
    human('socrates'),                     # fact
    mortal(X).when(human(X)),             # rule: mortal(X) :- human(X).
)

Queries

from hornet import database
from hornet.symbols import X, mortal

db = database()
db.tell(human('socrates'))
db.tell(mortal(X).when(human(X)))

for subst in db.ask(mortal(X)):
    print(subst[X])   # → socrates

db.ask() returns an iterable of substitutions. Each substitution maps variables to their bound values.


Built-in Predicates

Hornet ships a standard library of predicates pre-loaded in every database:

Predicate Description
equal(X, Y) Unification (X = Y)
unequal(X, Y) Negation of unification
let(R, Expr) Arithmetic evaluation (R is Expr)
arithmetic_equal(X, Y) Arithmetic equality
smaller(A, B) / greater(A, B) Numeric comparison
call(G) Call a goal term
once(G) Call G, commit to first solution
findall(O, G, L) Collect all solutions
member(X, L) List membership
append(A, B, C) List concatenation
reverse(L, R) List reversal
select(X, L, R) Select element from list
length(L, N) List length
maplist(G, L) Apply goal to each list element
is_var, nonvar, is_atom, is_int, … Type checks
write, writeln, nl I/O
cut Prolog cut (!)
fail Always fails
true Always succeeds
repeat Succeeds infinitely
throw(E) Raise an exception
ifelse(T, Y, N) Soft-cut conditional
phrase(G, L) DCG query

Arithmetic

Arithmetic expressions are built using Python's operators on symbolic terms and evaluated lazily by let:

from hornet.symbols import X, Y, R, let, arithmetic_equal

# R is X * Y + 1
db.ask(let(R, X * Y + 1))

# supported: + - * / // % ** ~ & | ^ << >>

Definite Clause Grammars (DCGs)

Hornet supports DCG notation via DCG() and DCGs():

from hornet import DCGs, database
from hornet.symbols import S, NP, VP, noun, verb, det, phrase

db = database()
db.tell(*DCGs(
    S.when(NP, VP),
    NP.when(det, noun),
    VP.when(verb),
    det.when(['the']),
    noun.when(['cat']),
    verb.when(['sleeps']),
))

for subst in db.ask(phrase(S, ['the', 'cat', 'sleeps'])):
    print('parsed!')

DCG rules are automatically expanded to difference lists. The inline(goal) escape hatch lets you embed regular Prolog goals inside a DCG body.


Extending with Python

Register native Python predicates using the @predicate decorator. This is how ifelse/3 is implemented internally:

from hornet import database, predicate
from hornet.clauses import Database, Subst
from hornet.combinators import Step, if_then_else
from hornet.clauses import resolve
from hornet.symbols import T, Y, N

db = database()

@db.tell
@predicate(ifelse(T, Y, N))
def _(db: Database, subst: Subst) -> Step[Database]:
    return if_then_else(
        resolve(subst[T]),
        resolve(subst[Y]),
        resolve(subst[N]),
    )(db, subst.map)

Architecture

Hornet is built on two main layers:

Term algebra (hornet.terms): Python expressions construct expression trees rather than computing values. Operator overloading (+, *, |, **, …) and __call__ produce nested Symbolic structures — Functor, Atom, Variable, Cons, Operator subclasses — which represent both data and goals. promote() lifts Python primitives (integers, strings, lists) into this algebra transparently.

Resolution engine (hornet.combinators): A triple-barrelled continuation monad drives search. Every goal is a function (ctx, subst) → Step, where a Step takes three continuations — success (emit a substitution and continue), failure (backtrack), and prune (implement cut). The combinators then, choice, prunable, neg, and if_then_else compose goals; trampoline() drives the whole thing iteratively to avoid stack overflow.


Examples

The examples/ directory includes:

  • append.py — list splitting via backtracking
  • queens.py — N-queens constraint solver
  • fizzbuzz.py — FizzBuzz via DCGs
  • symdiff.py — symbolic differentiation and simplification
  • parsing.py — natural language parsing with a German grammar
  • turing.py — a Turing machine interpreter
  • hanoi.py — Towers of Hanoi with Turtle graphics

License

MIT. See LICENSE.md.

Project details


Download files

Download the file for your platform. If you're not sure which to choose, learn more about installing packages.

Source Distribution

hornet_dsl-0.4.1a5.tar.gz (35.2 kB view details)

Uploaded Source

Built Distribution

If you're not sure about the file name format, learn more about wheel file names.

hornet_dsl-0.4.1a5-py3-none-any.whl (21.5 kB view details)

Uploaded Python 3

File details

Details for the file hornet_dsl-0.4.1a5.tar.gz.

File metadata

  • Download URL: hornet_dsl-0.4.1a5.tar.gz
  • Upload date:
  • Size: 35.2 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? No
  • Uploaded via: pdm/2.26.6 CPython/3.12.11 Linux/6.18.12-329.current

File hashes

Hashes for hornet_dsl-0.4.1a5.tar.gz
Algorithm Hash digest
SHA256 c8e7aaf6de1e199835b65f8deb05df7969c7f1f6cc8caf5ddd9c42b1bc4d68d6
MD5 241224adf5a9f0268dd3c2183a21b31d
BLAKE2b-256 b60e8739c5e906e7824e9c00e24875a52951b2a861b3d378a57e4580d8b26007

See more details on using hashes here.

File details

Details for the file hornet_dsl-0.4.1a5-py3-none-any.whl.

File metadata

  • Download URL: hornet_dsl-0.4.1a5-py3-none-any.whl
  • Upload date:
  • Size: 21.5 kB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? No
  • Uploaded via: pdm/2.26.6 CPython/3.12.11 Linux/6.18.12-329.current

File hashes

Hashes for hornet_dsl-0.4.1a5-py3-none-any.whl
Algorithm Hash digest
SHA256 68f22a08071dd59af94385af9328d4456ace22760ee2405b397a5aa74b284d8d
MD5 63a4a0ae67f3c1a6c80aef7f87ed793f
BLAKE2b-256 dbe3d1d1de421779524cefc96efc73eea0c40e987a668980103c9243849f821b

See more details on using hashes here.

Supported by

AWS Cloud computing and Security Sponsor Datadog Monitoring Depot Continuous Integration Fastly CDN Google Download Analytics Pingdom Monitoring Sentry Error logging StatusPage Status page