TL;DR: SPy is a subset/variant of Python specifically designed to be statically compilable while retaining a lot of the "useful" dynamic parts of Python.
It consists of:
- 
an interpreter (so that you can have the usual nice "development experience" that you have in Python) 
- 
a compiler (for speed) 
The documentation is very scarce at the moment, but the best source to understand the ideas behind SPy are probably the talks which Antonio Cuni gave:
Additional info can be found on:
- Antonio Cuni's blog
- A peek into a possible future of Python in the browser by Łukasz Langa.
At the moment, the only supported installation method for SPy is by doing an "editable install" of the Git repo checkout.
The most up-to-date version of the requirements and the installation steps is the GitHub action workflow.
Prerequisites:
- Python 3.12
Installation:
- 
Install the spypackage in editable mode:$ cd /path/to/spy/ $ pip install -e .[dev]
- 
Build the libspyruntime library:$ make -C spy/libspy
Run the test suite:
$ pytest
All the tests in spy/tests/compiler/ are executed in three modes:
- interp: run the SPy code via the interpreter
- doppler: perform redshift, then run the redshifted code via the interpreter
- C: generate C code, compile to WASM, then run it using- wasmtime
- 
Execute a program in interpreted mode: $ spy examples/hello.spy Hello world!
- 
Perform redshift and dump the generated source code: $ spy -r examples/hello.spy def main() -> void: print_str('Hello world!')
- 
Perform redshift and THEN execute the code: $ spy -r -x examples/hello.spy Hello world!
- 
Compile to executable: $ spy -c -t native examples/hello.spy $ ./examples/hello Hello world!
Moreover, there are more flags to stop the compilation pipeline and inspect the result at each phase.
The full compilation pipeline is:
- pyparse: source code -> generate Python AST
- parse: Python AST -> SPy AST
- symtable: Analyze the SPy AST and produce a symbol table for each scope
- redshift: SPy AST -> redshifted SPy AST
- cwrite: redshifted SPy AST -> C code
- compile: C code -> executable
Each step has a corresponding command line option which stops the compiler at that stage and dumps human-readable results.
Examples:
$ spy --pyparse examples/hello.spy
$ spy --parse examples/hello.spy
$ spy --symtable examples/hello.spy
$ spy --redshift examples/hello.spy
$ spy --cwrite examples/hello.spy
Moreover, the execute step performs the actual execution: it can happen
either after symtable (in "interp mode") or after redshift (in "doppler
mode").
(The following section should probably moved to the docs, once we have them)
The following is a simplified diagram which represent the main phases of the compilation pipeline:
graph TD
    SRC["*.spy source"]
    PYAST["CPython AST"]
    AST["SPy AST"]
    SYMAST["SPy AST + symtable"]
    SPyVM["SPyVM"]
    REDSHIFTED["Redshifted AST"]
    OUT["Output"]
    C["C Source (.c)"]
    EXE_NAT["Native exe"]
    EXE_WASI["WASI exe"]
    EXE_EM["Emscripten exe"]
    %% Core pipeline
    SRC -- pyparse --> PYAST -- parse --> AST -- ScopeAnalyzer --> SYMAST
    SYMAST -- import --> SPyVM -- execute --> OUT
    SPyVM -- redshift --> REDSHIFTED -- cwrite --> C
    C -- ninja --> EXE_NAT -- execute --> OUT
    C -- ninja --> EXE_WASI -- execute --> OUT
    C -- ninja --> EXE_EM -- execute --> OUT
    WASM is a target (either WASI or emscripten), but it's also a fundamental
building block of the interpreter.  The interpreter is currently written in
Python and runs on top of CPython, but it also needs to be able to call into
libspy (see below). This is achieved by compiling libspy to WASM and load
it into the Python interpreter using wasmtime.
So, depending on the execution mode, libspy is used in two very different
ways:
- 
interpreted: loaded in the python process via wasmtime. This is what happens for [interp]and[doppler]tests, and when you dospy hello.spy
- 
compiled: statically linked to the final executable. This is what happens for [C]tests and when you dospy --compile hello.spy.
libspy:
- 
spy/libspy/srcis a small runtime library written in C, which must be statically linked to any spy executable
- 
make -C spy/libspycreates alibspy.afor each supported target, which currently arenative,emscriptenandwasi
- 
spy/libspy/__init__.pycontains some support code to be able to load the WASM version of libspy in the interpreter.
the code in llwasm is just a thin wrapper over wasmtime to make it nicer
to interact with it.
The code in libspy/__init__.py uses llwasm to load libspy.wasm in the
interpreter. In particular, it implements the necessary "WASM imports" which
libspy uses to call back into the interpreter, for example to print debug
log messages, to trigger a panic and to turn WASM panics into SPyError
exceptions.
Normally, we execute SPy on top of CPython and we use wasmtime to load
libspy.wasm.
However, we can also run SPy on top of Pyodide: in that case, we are already
inside a WASM runtime engine (emscripten), so we don't need wasmtime.
The code in llwasm abstracts this difference away, and makes it possible to
transparently load libspy.wasm in either case.
If you want to contribute to SPy, be sure to review the contribution guidelines