A tool for running in-source unittests for Anwer Set Programming (ASP)
Project description
asp-selftest
A unit testing framework for Answer Set Programming (ASP) that enables in-source test definitions and execution.
Overview
asp-selftest extends the Clingo ASP solver with integrated testing capabilities, allowing developers to write and execute unit tests directly within their logic programs. Tests are defined using standard ASP syntax and executed in isolated contexts to ensure reliability and maintainability.
Quick Start
Installation
pip install asp-selftest
Basic Usage
clingo+ examples/edges.lp --run-asp-tests
Core Concepts
In-Source Unit Testing
Tests are embedded directly in ASP source files using #program directives. Consider the following example from nodes.lp:
% Implicit 'base' program
% Infer nodes from given edges.
node(A) :- edge(A, _).
node(B) :- edge(_, B).
% Verify that at least one edge exists.
cannot("at least one edge") :- not { edge(_, _) } > 0.
#program test_edge_leads_to_nodes(base).
% Test data: a simple graph with one edge.
edge(x, y).
% Assertions: verify expected node inference.
cannot("node x") :- not node(x).
cannot("node y") :- not node(y).
cannot("node z") :- not node(z). % This assertion will fail
The cannot Predicate
The framework uses cannot predicates as inverted assertions. This design leverages ASP's constraint mechanism to avoid optimization issues that would affect traditional positive assertions.
Execution Example:
$ clingo+ nodes.lp --run-asp-tests
...
Reading from nodes.lp
Testing nodes.lp
test_edge_leads_to_nodes(base)
...
AssertionError: cannot("node z")
File nodes.lp, line 11, in test_edge_leads_to_nodes(base). Model follows.
edge(x,y)
node(x)
node(y)
The test fails because node(z) does not exist in the model. To correct this assertion:
cannot("node z") :- node(z).
Validating Base Programs
After unit tests pass, the framework validates the base program. If prerequisites are missing, appropriate errors are reported:
$ clingo+ nodes.lp --run-asp-tests
...
AssertionError: cannot("at least one edge")
File nodes.lp, line ?, in base. Model follows.
<empty model>
Adding the required data file resolves the issue:
$ clingo+ nodes.lp edges.lp --run-asp-tests
...
Testing nodes.lp
test_edge_leads_to_nodes(base)
Testing edges.lp
Testing base
base
Solving...
Answer: 1 (Time: 0.003s)
edge(a,b) node(b) node(a)
SATISFIABLE
Test Dependencies
Tests and their dependencies are specified using #program directives. Test names must begin with the test_ prefix. Formal parameters declare dependencies on other program units:
#program unit_A.
#program test_unit_A(base, unit_A).
The implicit base program[^guide] must be explicitly referenced when required as a dependency. Actual arguments to test programs are not defined.
[^guide]: Potassco User Guide §3.1.2
Test Scoping
The framework enforces strict test isolation:
- Tests in each file execute within the context of that file only
- When file A includes file B:
- Tests in B execute with only B's logic loaded
- Tests in A execute with both A's and B's logic loaded
This scoping ensures that tests remain independent and do not interfere with each other.
Error Reporting
The framework provides clear, actionable error messages for syntax and semantic errors:
$ clingo+ logic.lp
...
Traceback (most recent call last):
...
File "logic.lp", line 2
1 node(A) :- edge(A, _).
2 node(B) :- edge(_, A).
^ 'B' is unsafe
^^^^^^^^^^^^^^^^^^^^^^^^ unsafe variables in: node(B):-[#inc_base];edge(#Anon0,A).
Understanding cannot Predicates
The framework uses cannot predicates rather than positive assertions for technical reasons related to ASP's optimization behavior. Traditional positive assertions can be optimized away by the solver, requiring complex idioms to prevent this. The cannot approach leverages ASP's constraint mechanism[^guide] for more reliable testing.
Technical Background
In ASP, constraints are headless rules that must always evaluate to false. When a constraint becomes true, the runtime considers the model invalid. The natural reading of a constraint is: "it cannot be the case that..."
By using cannot as a predicate head rather than a constraint, the framework allows these predicates to appear in models when they become true. The test runner then inspects the model and raises errors for any cannot predicates present.
Example without test execution:
$ clingo+ logic.lp
clingo+ version 5.8.0
Reading from logic.lp
Solving...
Answer: 1 (Time: 0.001s)
cannot("at least one edge")
SATISFIABLE
The cannot predicate appears in the model, which the test runner would flag as a failure. This approach provides a straightforward testing mechanism: if you can write ASP constraints, you can write cannot assertions.
Architecture
Plugin System
asp-selftest is built on a flexible plugin architecture that enables modular extension and customization of the testing framework. The plugin system uses a functional composition pattern where each plugin wraps the next in a processing chain, allowing for clean separation of concerns and easy extensibility.
The core plugin chain includes:
- clingo_main_plugin: Provides CLI integration and argument handling
- stdin_to_tempfile_plugin: Manages input from stdin by converting it to temporary files
- clingo_syntaxerror_plugin: Enhances error messages with rich formatting and context
- clingo_sequencer_plugin: Orchestrates the standard Clingo workflow (Load → Ground → Solve)
- testrunner_plugin: Discovers and executes tests, enforcing isolation and dependency management
- clingo_reify_plugin: Provides ASP reification support for advanced meta-programming
- clingo_defaults_plugin: Configures default behaviors and settings
Each plugin receives the next plugin in the chain as its first argument and can intercept, modify, or enhance the processing pipeline. This architecture allows developers to extend the framework with custom plugins for specialized testing scenarios or integration with other tools.
Project Status
asp-selftest is actively maintained and used in a production environment. The framework has been successfully deployed for formal specification of railway interlocking systems, comprising 35 files, over 100 tests, and more than 600 assertions.
The project was presented at Declarative Amsterdam in November 2024.
Installation and Usage
Installation
pip install asp-selftest
Running ASP Tests
clingo+ <file.lp> --run-asp-tests
Running Python Tests
The framework includes support for in-source Python tests:
clingo+ --run-python-tests
Requirements
- Python 3.13 or higher
- Clingo 5.8.0 or higher
License
This project is licensed under the GNU General Public License v3.0. See the LICENSE file for details.
Contributing
Contributions are welcome. Please ensure that all tests pass before submitting pull requests.
Repository
This project has been migrated from GitHub to Codeberg.
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 asp_selftest-0.1.6.tar.gz.
File metadata
- Download URL: asp_selftest-0.1.6.tar.gz
- Upload date:
- Size: 46.8 kB
- Tags: Source
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.1.0 CPython/3.13.5
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
7dca5b96cbb1c621038032aad3ede91916ff1a5f2fa4fe58b4a2300ca37f10d9
|
|
| MD5 |
26617b816014fdc35aaf0133a9e494a5
|
|
| BLAKE2b-256 |
41a42ed65a5deda428751c338ee19652b977d96cb813e3a09caac475e5c2cbf1
|
File details
Details for the file asp_selftest-0.1.6-py3-none-any.whl.
File metadata
- Download URL: asp_selftest-0.1.6-py3-none-any.whl
- Upload date:
- Size: 52.4 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.1.0 CPython/3.13.5
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
22d986500418f96c36a5267991e4218aba90e3a733cb2d78906bfdca59f6db1c
|
|
| MD5 |
11c77d19ef4f948b7de511c245679f02
|
|
| BLAKE2b-256 |
ec318813081ace913a0c30ad1fe1ea46d4bed606f2838fcc48b88a4cc3dd47e9
|