Scala SMT-LIB is a lightweight, no dependency, abstraction over the SMT-LIB standard for Satisfiability Modulo Theory (SMT) solvers. It lets you use a type safe Scala API to build SMT-LIB scripts and ship them to any SMT-LIB complient solver over a process interface.
Scala SMT-LIB provides the tools for parsing and printing the SMT-LIB syntax. It can help you if you need to communicate with a native SMT solver process via its text interface. You can also write a Scala wrapper around an SMT solver and use it from a friendlier API. You could even get crazier and use a pure Scala SMT solver that happens to implement the SMT-LIB api.
The library is still under active development and is evolving along with the needs of the projects using it. Essentially it means the API is going to change quite frequently until we are satisfied with how it feels. However, the code is actually quite robust, and comes with an extensive test suite. You should not fear relying on a current snapshot.
Scala SMT-LIB deals with the messy details of parsing and writing SMT-LIB. It provides a programmer friendly interface to simplify the development of tools that rely on SMT-LIB.
You may want to use Scala SMT-LIB if:
- You plan to write a tool that will use SMT-LIB as its input format, but do not wish to spend too much time figuring out the SMT-LIB standard. Then you can plug-in the parser component of Scala SMT-LIB in your tool and only deal with the Scala representation of the few SMT-LIB commands of your interest.
- You need to output a complex SMT-LIB encoding of some mathematical problems. We got you covered: You can programatically build a set of expressions using the Scala SMT-LIB abstract syntax tree, and use the printer components to get a valid SMT-LIB representation to pass along to another tool.
- You need to query an external black-box SMT solver, but the task of setting up a proper communication with this strange beast seems a bit too daunting? Scala SMT-LIB offers a module that abstracts SMTLIB-compliant solvers. You can program your tool against this simple high-level API. Scala SMT-LIB provides integration with Z3 and CVC4 out of the box, and you can add support for any other solver by implementing a relatively thin interface.
The project is built with sbt. To build the library, just type:
sbt package
It will produce a jar that you can add to the classpath of your own project.
If you are building your project using sbt, it is possible to setup a reference to this github repository in the build system to automatically get the most recent build. Here is an example of how to do it, you can pick any commit. If you are interested in this route, you should check the sbt official documentation.
Optionnaly, you can test Scala SMT-LIB in your environment by running the test suite. The tests are organized in unit and functional tests. Tu run the unit tests (very fast) you can type:
sbt test
All tests should pass. Please open an issue if any test is failing. The
functional tests are testing end-to-end flows of Scala SMT-LIB. They do take a
bit more time and require some setup in your environment. In particular, they
will try to use the SMT solvers z3
and/or cvc4
if they are available in
your PATH (the commands tried are exactly z3
and cvc4
). If present in the
PATH, Scala SMT-LIB will test its interpreter module directly against these SMT
solvers. You can run those tests with:
sbt fun:test
To construct a parser, you will need a java.io.Reader and a lexer:
val is = new java.io.FileReader("INPUT")
val lexer = new smtlib.lexer.Lexer(is)
val parser = new smtlib.parser.Parser(lexer)
The parser then provides a parseCommand
functions that will consume the input
until the end of command:
val script: List[Command] = {
var cmds = new ListBuffer[Command]
var cmd = parser.parseCommand
while(cmd != null)
cmds.append(cmd)
cmds.toList
}
parseCommand
returns null
when the end of file is reached.
You can decompose a command using pattern matching:
import smtlib.parser.Commands._
cmd match {
case Assert(term) => ???
case CheckSat() => ???
case Pop(1) => ???
}
If you want to generate a script to feed to a solver, you can build the AST explicitly, and use the pretty printers:
import smtlib.parser.Commands._
import smtlib.parser.theories.Ints._
val x = QualifiedIdentifier(SimpleIdentifier(SSymbol("x")))
val y = QualifiedIdentifier(SimpleIdentifier(SSymbol("y")))
val formula = Assert(LessThan(NumeralLit(0), Plus(x, y)))
smtlib.printer.RecursivePrinter.toString(formula) //(assert (< 0 (+ x y)))
The above is a little bit verbose due to the objective of supporting all of SMT-LIB. We are hoping to provide a nicer API in the future to build SMT-LIB scripts, at least in the common cases, but the AST will probably remain at the core of the library.
This section introduces the low level API of Scala SMT-LIB. In the future, it is expected to be wrapped by a higher-level API to perform most common operations. However, that API is expected to remain relatively stable and at the core of the library, so if that fits your needs you should be able to safely use it. It comes with a relatively extensive test suite to make sure it is working as expected.
###Lexing
The lexer
package implements low level
parsing of Tokens
, The
Lexer
lexes the input into
tokens. It reads lazily from a java.io.Reader
, on invocation of the nextToken
method:
class Lexer(reader: java.io.Reader) {
def nextToken: Token = { ... }
}
One call to nextToken
will only consume the input until the end of the
current token. It will read one character at a time from the input Reader
until it was able to unambiguously identify the end of the token. You can
provide a buffered input as a way to improve performance on some use cases.
nextToken
either returns a valid Token
, or null
if EOF is reached in the
input Reader
. It will signal errors with exceptions: a value of null
is not
an error and is expected to be eventually reached (unless the input never ends,
such as stdin
). The following two exceptions can be thrown by the lexer:
UnexpectedCharException
signals that an error in the input. It corresponds to a malformed token --- for example, a#ffff
(instead of#xffff
) would throw the exception on the firstf
. The exception indicates the position in the input.UnexpectedEOFException
occurs when the EOF is reached in the middle of an un-completed token. For example, a string literal which is not closed before the EOF.
###Parsing
Usually one does not need to work on a token by token basis, and is only
interested in fully formated SMT-LIB expressions. The
parser
provides the extraction of SMT-LIB
expressions. The Parser
consumes
tokens generated by the above lexer:
class Parser(lexer: smtlib.lexer.Lexer) {
def nextCommand: Command = { ... }
def nextTerm: Term = { ... }
}
It provides many nextX
methods (more than shown above) to parse any expressions
defined by the SMT-LIB standard grammar. The more common are the nextCommand
and the nextTerm
to parse respectively a command and a term-formula expression.
A Parser
uses exceptions to signal parsing error, they are defined in the
Parser
companion object. They
provide a bunch of information, including the token on which the parsing
failed, the expected tokens, and the position in the input (the position is an
attribute of the token and can be accessed through it).
A Command
is the root of a complex abstract syntax tree (AST) representing
the corresponding SMT-LIB command. It is defined
here. A few commands take as
parameter a Term
, whose AST is defined
here.
###Printing
The printer
helps with printing out SMT-LIB
complient commands. This means that the output of a printer can be send
directly to an SMT solver.
###Standard Theories
Finally the theories
module provides tree
builders to create theory-specific formulas. Each theory module provides
apply
and unapply
methods on various object to manipulate the Term
representing the actual theory expression.
The library is still under development and the API will likely go through quite a few changes. It was originally part of CafeSat and has been made standalone in order for the Leon project to rely on it. Hopefully, it can be useful to other people as well.
###Testing
In order to attain a decent level of quality, there is a relatively strict policy for testing. Testing is separated in two levels of testing: unit tests and integration tests. Unit test are dependency free and run entirely in memory. While integration tests will rely on things like file system and external solvers. Unit test are very fast and should be easy to run as part of a regular build cycle (you can run them after each compile), while integration tests are a bit slower and are meant to pass on each commit.
In SBT, the command sbt test
will run the unit test suite, while the command
sbt it:test
will run the integration test suite. During developement, it
should be fine to run in mode ~testQuick
.
See Changelog.