Skip to main content

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 the following logical systems:

  1. Propositional Logic (base): Classical Boolean logic with conjunction, disjunction, negation, implication, equivalence, and exclusive-or operators.

  2. 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.

  3. 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.

  4. 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.

  5. Sequential Extended Regular Expressions (sere): Regex-like patterns over Boolean state formulas. Supported operators:

    • Concatenation ;, fusion :, alternation |.
    • Length-matching intersection && and non-length-matching intersection &.
    • Repetition [*], [+], [*i], [*i..j], [*i..].
    • Fusion-iteration [:*], [:+], [:*i], [:*i..j], [:*i..].
    • Goto-repetition [->], [->i], [->i..j], [->i..].
    • Equal-count repetition [=], [=i], [=i..j], [=i..].
    • first_match(r) (SVA-derived).
    • SERE complement ~r (extension beyond Spot; not part of Spot's SERE grammar).

    Two deliberate divergences from Spot:

    • ~ denotes SERE complement (language complement), not Boolean negation. Use ! for Boolean negation; it is the sole Boolean negation glyph across every grammar in this package.
    • Goto and equal repetition ([->], [=]) accept arbitrary SERE operands here, not just Boolean formulas as in Spot.

    Not supported: delay operators ##i / ##[i..j], Spot's automatic simplification rules, and trace evaluation.

  6. Property Specification Logic (psl): LTL extended with the SERE-LTL binding operators {r}[]-> f, {r}<>-> f, {r}, {r}!, and !{r} (plus the sugar {r}[]=> f and {r}<>=> f). PSL reuses the full SERE grammar above, including the extensions beyond Spot, so any SERE node in {...} works. Spot-style automatic simplification rules and trace evaluation are not supported.

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")

# Sequential Extended Regular Expressions
sere = logic_asts.parse_expr("a ; b[+] ; c", syntax="sere")

# SERE with complement and goto-repetition (extensions beyond Spot)
sere_ext = logic_asts.parse_expr("first_match(~a) ; b[->3]", syntax="sere")

# PSL (LTL + SERE bindings)
psl = logic_asts.parse_expr("{a;b}[]-> F c", syntax="psl")

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)
SEREExpr[AP] sere_expr_iter(expr)
PSLExpr[AP] psl_expr_iter(expr)

All of these 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


Download files

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

Source Distribution

logic_asts-1.7.0.tar.gz (151.3 kB view details)

Uploaded Source

Built Distribution

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

logic_asts-1.7.0-py3-none-any.whl (47.0 kB view details)

Uploaded Python 3

File details

Details for the file logic_asts-1.7.0.tar.gz.

File metadata

  • Download URL: logic_asts-1.7.0.tar.gz
  • Upload date:
  • Size: 151.3 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? No
  • Uploaded via: uv/0.11.16 {"installer":{"name":"uv","version":"0.11.16","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

Hashes for logic_asts-1.7.0.tar.gz
Algorithm Hash digest
SHA256 9c3987c3b2eb428de53d69bd0ede7eb1069496578592475e3c33ac1638e90461
MD5 bf23d492d7882d2f521544837452bab1
BLAKE2b-256 3dcdf8af2f7f14d1c6241b05229a683ffb3a26222be7b840593ebcbc689d9c4f

See more details on using hashes here.

File details

Details for the file logic_asts-1.7.0-py3-none-any.whl.

File metadata

  • Download URL: logic_asts-1.7.0-py3-none-any.whl
  • Upload date:
  • Size: 47.0 kB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? No
  • Uploaded via: uv/0.11.16 {"installer":{"name":"uv","version":"0.11.16","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

Hashes for logic_asts-1.7.0-py3-none-any.whl
Algorithm Hash digest
SHA256 1ec27011cf042a238c9cdb213428c5b211211ff0caa14adb5a6ca38cf484cf4a
MD5 faaf85ebc55dc67412beb89e5d8f0b25
BLAKE2b-256 cc1825ad2de557db9fd6f54cbc76fa2597aad955666521253d1772fc433bd21c

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