Grammars and parsers for various logic
Project description
Logic ASTs: Abstract Syntax Trees for Logical Specifications
A collection of grammars, parsers, and abstract syntax trees (ASTs) for various logical formalisms.
The goal is to serve as a reusable foundation for academics and developers building tools that require logical expression parsing and manipulation, eliminating the need to create new parsers for each application.
Supported Logics
The library implements complete support for four logical systems:
-
Propositional Logic (base): Classical Boolean logic with conjunction, disjunction, negation, implication, equivalence, and exclusive-or operators.
-
Linear Temporal Logic (ltl): Temporal extension adding operators for reasoning about sequences of states over time. Includes Next (X), Eventually (F), Always (G), and Until (U) operators with optional time constraints.
-
Spatio-Temporal Reach-Escape Logic (strel): Combines temporal and spatial reasoning for multi-agent and distributed systems. Adds spatial operators (Everywhere, Somewhere, Reach, Escape) with distance constraints.
-
Signal Temporal Logic with Graph Operators (stl_go): Extends temporal logic with graph-based operators for specifying properties over multi-agent communication networks. Includes incoming and outgoing edge quantifiers with weight and count constraints.
Installation
Install from PyPI:
pip install logic-asts
Or if you'd like the latest main branch:
pip install git+https://github.com/anand-bala/logic-asts.git
Quick Start
Parse logical expressions:
import logic_asts
# Propositional logic
prop = logic_asts.parse_expr("(p & q) | ~r", syntax="base")
# Linear temporal logic
ltl = logic_asts.parse_expr("G(request -> F response)", syntax="ltl")
# Spatio-temporal logic
strel = logic_asts.parse_expr("G everywhere[0,5] ~obstacle", syntax="strel")
# Graph-based temporal logic
stl_go = logic_asts.parse_expr("in^[0,1]{E}_{c}[1,n] consensus", syntax="stl_go")
Create expressions programmatically:
from logic_asts.base import Variable, And, Or, Not
from logic_asts.ltl import Eventually, TimeInterval
p = Variable("p")
q = Variable("q")
# (p & q) | ~p
formula = (p & q) | ~p
# F[0,10] (p & q)
temporal = Eventually(p & q, TimeInterval(0, 10))
Evaluate propositional formulas:
from logic_asts.base import simple_eval
p = Variable("p")
q = Variable("q")
formula = p & q
# Evaluate: p=true, q=true -> Result: true
result = simple_eval(formula, {"p", "q"})
# Evaluate: p=true, q=false -> Result: false
result = simple_eval(formula, {"p"})
Type-safe tree traversal
The most convenient way to walk an expression tree is expr.iter_subtree(),
but its return type is Iterator[Expr]. If you need mypy (or pyright) to know
the precise element type, reach for one of the patterns below.
Pattern 1 -- typed iterator (preferred)
When you already hold a typed expression, call the matching iterator directly:
from logic_asts import ltl_expr_iter, parse_expr
expr = parse_expr("G (p U q)", syntax="ltl") # LTLExpr[str]
for node in ltl_expr_iter(expr): # Iterator[LTLExpr[str]]
...
| Your type | Iterator to use |
|---|---|
BoolExpr[AP] / BaseExpr[AP] |
bool_expr_iter(expr) |
LTLExpr[AP] |
ltl_expr_iter(expr) |
STRELExpr[AP] |
strel_expr_iter(expr) |
STLGOExpr[AP] |
stlgo_expr_iter(expr) |
All four functions also validate that the subtree contains no out-of-dialect
nodes and raise TypeError at runtime if it does.
Pattern 2 -- type-guard then typed iterator
When the static type is just Expr (e.g. coming from an untyped API), narrow
it first:
from logic_asts import Expr, is_ltl_expr, ltl_expr_iter
def process(expr: Expr) -> None:
if is_ltl_expr(expr, str): # narrows to LTLExpr[str]
for node in ltl_expr_iter(expr): # Iterator[LTLExpr[str]]
...
Pattern 3 -- filter to a single node class
Use the kind= argument on iter_subtree to visit only one concrete class.
The full tree is still traversed, but only matching nodes are yielded:
from logic_asts import Variable, parse_expr
expr = parse_expr("G (p U q)", syntax="ltl")
for v in expr.iter_subtree(kind=Variable): # Iterator[Variable[Any]]
print(v.name)
Pattern 4 -- custom type subset with ExprVisitor
For arbitrary subsets of node types, construct an ExprVisitor directly.
It both validates the tree and yields a typed iterator:
from logic_asts import ExprVisitor, And, Or, Not, Variable
for node in ExprVisitor((And, Or, Not, Variable), expr):
# node: And | Or | Not | Variable[Any]
...
Contributing
Contributions are welcome. Please ensure all tests pass and documentation is updated for new features.
License
This project is licensed under the BSD 2-clause license.
Project details
Release history Release notifications | RSS feed
Download files
Download the file for your platform. If you're not sure which to choose, learn more about installing packages.
Source Distribution
Built Distribution
Filter files by name, interpreter, ABI, and platform.
If you're not sure about the file name format, learn more about wheel file names.
Copy a direct link to the current filters
File details
Details for the file logic_asts-1.5.0.tar.gz.
File metadata
- Download URL: logic_asts-1.5.0.tar.gz
- Upload date:
- Size: 82.5 kB
- Tags: Source
- Uploaded using Trusted Publishing? No
- Uploaded via: uv/0.11.6 {"installer":{"name":"uv","version":"0.11.6","subcommand":["publish"]},"python":null,"implementation":{"name":null,"version":null},"distro":{"name":"Fedora Linux","version":"42","id":"","libc":null},"system":{"name":null,"release":null},"cpu":null,"openssl_version":null,"setuptools_version":null,"rustc_version":null,"ci":null}
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
7e892d1a2cf4b2c29f6deadc01ecfb1c7799c9b03a4977cce6563da4b06ccfeb
|
|
| MD5 |
70630f1c97ed8f053bfed21099d4c1ec
|
|
| BLAKE2b-256 |
d572dbeb8e3ef57697f448ac682556746644b63f4b0cac3fb9581c43ac616934
|
File details
Details for the file logic_asts-1.5.0-py3-none-any.whl.
File metadata
- Download URL: logic_asts-1.5.0-py3-none-any.whl
- Upload date:
- Size: 31.8 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: uv/0.11.6 {"installer":{"name":"uv","version":"0.11.6","subcommand":["publish"]},"python":null,"implementation":{"name":null,"version":null},"distro":{"name":"Fedora Linux","version":"42","id":"","libc":null},"system":{"name":null,"release":null},"cpu":null,"openssl_version":null,"setuptools_version":null,"rustc_version":null,"ci":null}
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
a8a382c6164ddbea1a0fc0f7689827cd950a380eecc3011e60e2913ca92f2eec
|
|
| MD5 |
28f6adf15da2e99d475569bdefa5c3b0
|
|
| BLAKE2b-256 |
b8de192d04a03829bc6041bb0b84a862679450f110c74b57f76a5214d2e37b59
|