Skip to main content

just solve it - a command line tool to run multiple SMT (Satisfiability Modulo Theories) solvers in parallel

Project description

jsi

just solve it - a command-line utility to run a portfolio of SMT solvers in parallel

Screenshot of jsi running on an unsat division problem

Highlights

  • ๐Ÿ† acts as a "virtual best solver" by running multiple solvers in parallel and printing the result of the fastest solver to stdout
  • ๐Ÿ” discovers available solvers on the PATH at runtime
  • ๐Ÿ›ฃ๏ธ runs solvers in parallel and monitors their progress
  • ๐Ÿ“œ parses solver output to determine if the problem is sat, unsat, error, unknown, etc
  • โฐ can terminate solvers after a timeout
  • โธ๏ธ can be interrupted with Ctrl-C and remaining solvers will be killed
  • ๐ŸŽ runs with minimal startup time (<100ms), and also supports an experimental daemon mode with a rust client for extra low-latency (<10ms)
  • ๐Ÿ”ช reaps orphaned solver processes
  • ๐Ÿ–ฅ๏ธ supports macOS and Linux
  • ๐Ÿ supports Python 3.11+

Getting Started

We recommend using uv to install jsi.

# install jsi
uv tool install jsi

# run it
jsi --help

Verify the installation with one of the included examples:

jsi --full-run --timeout 2s examples/unsat-div.smt2

Features

๐Ÿงฐ Configuration

This is how jsi finds and runs solvers:

  • it first attempts to load custom solver definitions from ~/.jsi/solvers.json
  • if that file doesn't exist, it loads the default definitions from the installed package (see src/jsi/config/solvers.json)

Based on these definitions, jsi knows what executables to look for, whether a given solver is enabled, how to enable model generation, etc.

Then:

  • it looks up the solver cache in ~/.jsi/cache.json
  • if that file doesn't exist, it will scan the PATH and cache the results

It does this because loading cached paths is 4-5x faster than scanning the PATH.

๐Ÿ’ก Tip: ~/.jsi/cache.json can always be safely deleted, jsi will generate it again next time it runs. If you make changes to ~/.jsi/solvers.json (like adding a new solver), you should delete the cache file, otherwise jsi won't pick up the new solver.

๐ŸŽจ Rich Output

jsi uses rich to render nice colored output. However importing rich at startup adds about 30-40ms to jsi's startup time, so by default jsi only uses rich if it detects that its output is a tty.

If you want to minimize jsi's startup time, you can force it to use basic output by redirecting its stderr to a file: jsi ... 2> jsi.err

๐Ÿ“‹ Run a specific sequence of solvers

Sometimes it can be useful to run only a subset of available solvers, for instance when you already know the top 2-3 solvers for a given problem.

jsi supports a --sequence option that allows you to specify a sequence of solvers to run as a comma-separated list of solver names (as defined in your ~/.jsi/solvers.json file).

Screenshot of jsi running a sequence of solvers

๐Ÿ“Š CSV Output

In addition to the table output, jsi can also output results in CSV format, which is useful for further processing like generating graphs or importing into spreadsheets (especially in conjunction with the --full-run option).

$ jsi --full-run --sequence stp,cvc4,cvc5 --csv examples/unsat-div.smt2
stp returned unsat
cvc4 returned unsat
cvc5 returned unsat
unsat
; (result from stp)

                                   Results
โ”โ”โ”โ”โ”โ”โ”โ”โ”โ”ณโ”โ”โ”โ”โ”โ”โ”โ”โ”ณโ”โ”โ”โ”โ”โ”โ”ณโ”โ”โ”โ”โ”โ”โ”โ”โ”ณโ”โ”โ”โ”โ”โ”โ”โ”โ”โ”โ”โ”โ”โ”โ”โ”โ”โ”โ”โ”โ”โ”โ”โ”โ”โ”โ”โ”โ”โ”โ”โ”โ”โ”โ”ณโ”โ”โ”โ”โ”โ”โ”“
โ”ƒ solver โ”ƒ result โ”ƒ exit โ”ƒ   time โ”ƒ output file                      โ”ƒ size โ”ƒ
โ”กโ”โ”โ”โ”โ”โ”โ”โ”โ•‡โ”โ”โ”โ”โ”โ”โ”โ”โ•‡โ”โ”โ”โ”โ”โ”โ•‡โ”โ”โ”โ”โ”โ”โ”โ”โ•‡โ”โ”โ”โ”โ”โ”โ”โ”โ”โ”โ”โ”โ”โ”โ”โ”โ”โ”โ”โ”โ”โ”โ”โ”โ”โ”โ”โ”โ”โ”โ”โ”โ”โ”โ•‡โ”โ”โ”โ”โ”โ”โ”ฉ
โ”‚ stp    โ”‚ unsat  โ”‚    0 โ”‚  0.01s โ”‚ examples/unsat-div.smt2.stp.out  โ”‚ 6.0B โ”‚
โ”‚ cvc4   โ”‚ unsat  โ”‚    0 โ”‚  9.75s โ”‚ examples/unsat-div.smt2.cvc4.out โ”‚ 6.0B โ”‚
โ”‚ cvc5   โ”‚ unsat  โ”‚    0 โ”‚ 13.01s โ”‚ examples/unsat-div.smt2.cvc5.out โ”‚ 6.0B โ”‚
โ””โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”ดโ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”ดโ”€โ”€โ”€โ”€โ”€โ”€โ”ดโ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”ดโ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”ดโ”€โ”€โ”€โ”€โ”€โ”€โ”˜
writing results to: examples/unsat-div.smt2.csv

$ bat examples/unsat-div.smt2.csv

โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”ฌโ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€
       โ”‚ File: examples/unsat-div.smt2.csv
โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”ผโ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€
   1   โ”‚ solver,result,exit,time,output file,size
   2   โ”‚ stp,unsat,0,0.01s,examples/unsat-div.smt2.stp.out,6
   3   โ”‚ cvc4,unsat,0,9.75s,examples/unsat-div.smt2.cvc4.out,6
   4   โ”‚ cvc5,unsat,0,13.01s,examples/unsat-div.smt2.cvc5.out,6
๐Ÿ”ช Reaper mode

jsi makes a best effort to be resilient about crashes and avoid orphaned solver processes. In particular:

  • it spawns a reaper thread that checks if the jsi's parent process is still running, and if not, it will kill all solver subprocesses
  • it handles keyboard interrupts and SIGTERM
  • it can optionally spawn a reaper subprocess that monitors jsi's pid, and if it notices that jsi has died, it will kill any solver subprocesses
๐Ÿงช Experimental Daemon Mode

jsi can also run in daemon mode, where it will start a subprocess to handle requests. This mode is experimental and subject to change.

# start the daemon with
jsi --daemon

# or
python -m jsi.server

# tail server logs with
tail -f ~/.jsi/daemon/server.{err,out}

The daemon will listen for requests on a unix socket, and each request should be a single line containing the path to an smt2 file to solve.

You can then send requests to the daemon:

# directly with nc
$ echo -n $(pwd)/examples/easy-sat.smt2 | nc -U ~/.jsi/daemon/server.sock
sat
; (result from yices)

# with the included Python client
$ python -m jsi.client examples/easy-sat.smt2
sat
; (result from yices)

or for the lowest latency, use the included Rust client:

# build it
(cd jsi-client-rs && cargo build --release)

# install it
(cd jsi-client-rs && ln -s $(pwd)/target/release/jsif /usr/local/bin/jsif)

# use it
jsif examples/easy-sat.smt2

This benchmark shows why you might want to use the Rust client:

hyperfine --shell=none \
  "python -m jsi.client examples/easy-sat.smt2" \
  "jsif examples/easy-sat.smt2"

Benchmark 1: python -m jsi.client examples/easy-sat.smt2
  Time (mean ยฑ ฯƒ):     290.9 ms ยฑ   9.1 ms    [User: 75.7 ms, System: 18.9 ms]
  Range (min โ€ฆ max):   282.3 ms โ€ฆ 313.5 ms    10 runs

Benchmark 2: jsif examples/easy-sat.smt2
  Time (mean ยฑ ฯƒ):     196.7 ms ยฑ   4.3 ms    [User: 1.2 ms, System: 2.3 ms]
  Range (min โ€ฆ max):   190.9 ms โ€ฆ 207.2 ms    15 runs

Summary
  jsif examples/easy-sat.smt2 ran
    1.48 ยฑ 0.06 times faster than python -m jsi.client examples/easy-sat.smt2

โš ๏ธ Warning: the daemon mode is experimental and subject to change. Not all options are supported at this time (like --sequence, --csv, --timeout, etc).

Tips

Installing solvers

If you have no solver installed (or even only a single solver installed), jsi will not be particularly useful. It won't install any solvers for you, you need to install them yourself.

If you want to run a Docker image that is already pre-configured with multiple high-performance solvers, check out our solvers image.

Adding new solvers

To add a new solver, the process is roughly:

  • if you already have ~/.jsi/solvers.json, modify it directly
  • if you don't have ~/.jsi/solvers.json, copy src/jsi/config/solvers.json to ~/.jsi/solvers.json
  • add the new solver to the ~/.jsi/solvers.json file (you will need to fill the executable name, the args you want to always pass to the solver, and optionally the model generation option)
  • delete ~/.jsi/cache.json if it exists
  • on the next run, jsi will should pick up the new solver

If you wish to contribute a new solver definition to upstream jsi, please check out the CONTRIBUTING.md file for more information and submit a PR.

Running multiple versions of the same solver

Follow the instructions above to add the solver, but use different names for each version.

The only issue is that you will need to make sure the executable names are unique, so that jsi can tell them apart (e.g. yices-2.6.4 and yices-2.6.5).

Disabling a solver

To avoid running a particular solver:

  • temporarily, run with an explicit --sequence option that excludes the solver
  • permanently, edit ~/.jsi/solvers.json and set enabled to false

Contributing

See CONTRIBUTING.md

Acknowledgements

The setup for this project is based on postmodern-python.

Disclaimer

This code is being provided as is. No guarantee, representation or warranty is being made, express or implied, as to the safety or correctness of the code. It has not been audited and as such there can be no assurance it will work as intended, and users may experience delays, failures, errors, omissions or loss of transmitted information. Nothing in this repo should be construed as investment advice or legal advice for any particular facts or circumstances and is not meant to replace competent counsel. It is strongly advised for you to contact a reputable attorney in your jurisdiction for any questions or concerns with respect thereto. a16z is not liable for any use of the foregoing, and users should proceed with caution and use at their own risk. See a16z.com/disclosures for more info.

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

just_solve_it-0.1.0.tar.gz (29.6 kB view details)

Uploaded Source

Built Distribution

just_solve_it-0.1.0-py3-none-any.whl (26.9 kB view details)

Uploaded Python 3

File details

Details for the file just_solve_it-0.1.0.tar.gz.

File metadata

  • Download URL: just_solve_it-0.1.0.tar.gz
  • Upload date:
  • Size: 29.6 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/5.1.1 CPython/3.12.7

File hashes

Hashes for just_solve_it-0.1.0.tar.gz
Algorithm Hash digest
SHA256 1da56a666675556f9aa184539c0129e6b60a58239bc384d1de72caefda25dde8
MD5 81530c0e553a3726b080574d9b23db44
BLAKE2b-256 b72e3a0ea3d12cd1163296a50aeca5ed42eee06e67b520ad28c9df6b86c5b6c7

See more details on using hashes here.

Provenance

The following attestation bundles were made for just_solve_it-0.1.0.tar.gz:

Publisher: release.yml on a16z/jsi

Attestations:

File details

Details for the file just_solve_it-0.1.0-py3-none-any.whl.

File metadata

File hashes

Hashes for just_solve_it-0.1.0-py3-none-any.whl
Algorithm Hash digest
SHA256 997780615343de4018c025285bd5b1bdd4a66218006f440e785291fd3d920995
MD5 71ba61a5e97b5a18ba17d9960275e8b8
BLAKE2b-256 4d29cd30971c74329795adbb1cc5d96ee2389b6bd3441650f7dd2916e416e6aa

See more details on using hashes here.

Provenance

The following attestation bundles were made for just_solve_it-0.1.0-py3-none-any.whl:

Publisher: release.yml on a16z/jsi

Attestations:

Supported by

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