A small Common Lisp implementation. Call lisp and python from one another.
Project description
Python's Lisp
Common Lisp programming language
The best python lisp implementation!
Implemented entirely in python 3.9 and above.
Embed lisp in python code and python in lisp code; or interact with lisp in the included repl.
Lisp Documentation
Goals: Implement a substantial subset of the features of common lisp.
Features
- Full macro system: defmacro, macroexpand, and a runtime library written in Lisp
- Complete lambda lists: &optional, &rest, &key, and &aux for functions and macros
- Closures, tail call optimization, and fully implemented continuations
- Condition system: handler-case, signal, make-condition
- Multiple return values: values, multiple-value-bind
- Embed Lisp in Python or call Python from Lisp
- Full interactive REPL with persistent history and session logging
Design Features
- Implemented from scratch with no external dependencies
- Extensible: add new primitives or Lisp definitions without modifying the interpreter
- Complete lexical analyzer and LL(1) recursive descent parser
- Full test suite
A Taste of the Language
Closures and higher-order functions:
; make-adder returns a closure over n
(defun make-adder (n)
(lambda (x) (+ x n)))
(setf add10 (make-adder 10))
(mapcar add10 '(1 2 3 4 5)) ;==> (11 12 13 14 15)
; filter and transform in one pipeline
(mapcar (lambda (x) (* x x))
(remove-if 'oddp '(1 2 3 4 5 6))) ;==> (4 16 36)
Macros that write code:
; swap! expands to a let at compile time -- no helper function needed
(defmacro swap! (a b)
`(let ((tmp ,a))
(setf ,a ,b)
(setf ,b tmp)))
(setf x 1 y 100)
(swap! x y)
x ;==> 100
y ;==> 1
Structs with generated constructors, predicates, and accessors:
(defstruct point x y)
(setf p (make-point :x 3 :y 4))
(point-p p) ;==> T
(point-x p) ;==> 3
(setf (point-x p) 10)
(list (point-x p) (point-y p)) ;==> (10 4)
CLI USAGE
Run the repl.
> python3 -m pythonslisp
Execute a lisp source file.
> python3 -m pythonslisp <lispSourceFile.lisp>
For various information requests there are.
> python3 -m pythonslisp (-h|--help|-v|--version)
Note that the repl provides access to two separate online documentation systems. There is the listener command help system accessible by typing ']help' in the repl. Then there is the Lisp online help system (LOHS) accessible by evaluating the sexpression '(help)'.
These two help systems document access to different things. The listener command help system provides documentation for commands recognized by the listener from inside the repl to control the listener in various ways such as working with session logs. The LOHS provides access to documentation for all lisp callables (primitives, functions and macros) as well as to various help topics of interest to the lisp programmer.
The LOHS is dynamic. As the user defines new functions and macros their documentation becomes available in the help system. Specifically the documentation system will display a "function header" which includes the function name and its lambda list (formal parameter list). This will be followed by any text in the documentation string coded by the lisp programmer.
If you want a lisp startup script to run whenever the interpreter is initialized/reinitialized you can add .pythonslisp_rc to your home directory.
API: Using Lisp as a Package
The easiest way to use the package is to import and instantiate an Interpreter. The constructor loads the full runtime automatically. Use the interpreter's eval() functions to call into Lisp. Everything is persistent across calls so you can mix and match which eval function you use at any given time. To reset the interpreter to a clean state, simply instantiate a new one.
The example below defines a fibonacci function in Lisp, then calls it from Python and prints the results.
from pythonslisp.Interpreter import Interpreter
from pythonslisp.AST import prettyPrintSExpr
interp = Interpreter( )
interp.rawEval( '''(defun fibo (num)
(if (< num 2)
1
(+ (fibo (- num 1))
(fibo (- num 2)))))''' )
def fibo_wrapper( nth ):
return interp.rawEval(f'(fibo {nth})')
print( fibo_wrapper(10) )
print( prettyPrintSExpr(fibo_wrapper(15)) )
If the lisp code returns an int,float,fraction,string,list or dict then it just returns the python versions of those things. Calls to rawEval will return the AST of an sexpression. A subsequent call to prettyPrintSExpr(ast) will convert the complex structure to a python string representation.
The interpreter's eval functions can handle any number of lisp expressions in the string argument. The return value is always the result of evaluating the last expression in the string.
Calling python from lisp is just as easy. Just call the python lisp primitive and pass it some python code in a lisp string.
# Simply call python
interp.rawEval( '(python "3 + 4")' )
# Passing values from lisp
interp.rawEval( '(setf num 3)' )
interp.rawEval( '(python (string num " + 4"))' )
Primarily your program will interact with pythonslisp.AST, pythonslisp.Interpreter, and pythonslisp.Parser. Methods prefixed by _ are private implementation details subject to change; all other methods and attributes form the public interface.
eval() functions in the interpreter that include the label 'instrumented' are designed for testing the performance of the interpreter. These special versions of eval() return a tuple of three values: return value, parse time in seconds, evaluation time in seconds. They are used by the listener's repl to report performance characteristics during interactive sessions.
To extend or modify the interpreter, evaluate '(help "MODIFYING-DOC")' in the REPL.
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 pythonslisp-0.39.44.tar.gz.
File metadata
- Download URL: pythonslisp-0.39.44.tar.gz
- Upload date:
- Size: 112.8 kB
- Tags: Source
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.2.0 CPython/3.9.0
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
722f9f01752a66421d1ddfd1221ba69bd510bfcb34217f31658c6a0e4e7d5bb5
|
|
| MD5 |
740df59845732a8b5fd0e4efab917dbf
|
|
| BLAKE2b-256 |
074ccdd169c612c71aaf29f25591d27ecf84fd2918fa99722deea4ebd2ce4491
|
File details
Details for the file pythonslisp-0.39.44-py3-none-any.whl.
File metadata
- Download URL: pythonslisp-0.39.44-py3-none-any.whl
- Upload date:
- Size: 127.6 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.2.0 CPython/3.9.0
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
03159359b1f26664959dbbf0f50f7ed4256352913555b0e1544708aec7ceb698
|
|
| MD5 |
2c9b74774348fe85b3e21db89a9b1cd0
|
|
| BLAKE2b-256 |
b95aa2385ca6861aa513671a626bc088d0a0e74b2fcd6f971574ed788178c07e
|