bendudson / cl4py

Common Lisp for Python

Geek Repo:Geek Repo

Github PK Tool:Github PK Tool

cl4py - Common Lisp for Python

The library cl4py (pronounce as clappy) allows Python programs to call Common Lisp libraries.

Motivation

You are a Python programmer, but you want access to some of the powerful features of Lisp, for example to compile code at run time? Or you want to use some awesome Lisp libraries? Or you are a Lisp programmer and want to show your work to your Python friends. In all these cases, cl4py is here to help you.

Tutorial

You can start any number of Lisp subprocesses within Python, like this:

>>> import cl4py
>>> lisp = cl4py.Lisp()

Of course, this requires you have some Lisp installed. If not, use something like apt install sbcl, pacman -S sbcl or brew install sbcl to correct this deficiency. Once you have a running Lisp process, you can execute Lisp code on it:

>>> lisp.eval("(+ 2 3)")
5

>>> add = lisp.eval("(function +)")
>>> add(1, 2, 3, 4)
10

>>> div = lisp.eval("(function /)")
>>> div(2, 4)
Fraction(1, 2)

Some Lisp data structures have no direct equivalent in Python, most notably, cons cells. The cl4py module provides a suitable Cons class and converts List conses to instances of cl4py.Cons.

>>> lisp.eval("(cons 1 2)")
Cons(1, 2)

>>> lst = lisp.eval("(cons 1 (cons 2 nil))")
List(1, 2)
>>> lst.car
1
>>> lst.cdr
List(2) # an abbreviation for Cons(2, None)

# conversion works vice versa, too:
>>> lisp.eval(cl4py.List('+', 2, 9))
11

# cl4py Conses are iterable, too!
>>> list(lst)
[1, 2]
>>> sum(lst)
3

For convenience, cl4py will implicitly convert Python tuples to Lisp lists and interpret Python strings as Lisp tokens.

>>> lisp.eval(('+', 2, 3))
5

>>> lst = lisp.eval(("loop", "repeat", 3, collect, 5))
List(5, 5, 5)

>>> lisp.eval("(cdddr #1=(1 . #1#))")
DottedList(1, ...)

It soon becomes clumsy to use eval to look up individual Lisp functions by name. Instead, it is possible to convert entire Lisp packages to Python modules, like this:

>>> cl = lisp.find_package('CL')
>>> cl.oppd(5)
True

>>> cl.cons(5, None)
List(5)

>>> cl.remove(5, [1, 5, 2, 7, 5, 9])
[1, 2, 3, 4]

# Higher-order functions work, too!
>>> cl.mapcar(cl.constantly(4), (1, 2, 3))
List(4, 4, 4)

# Of course, circular objects of all kinds are supported.
>>> twos = cl.cons(2,2)
>>> twos.cdr = twos
>>> cl.mapcar('+', (1, 2, 3, 4), twos)
List(3, 4, 5, 6)

Python strings are not treated as Lisp strings, but read in as Lisp tokens. This means that in order to actually send a string to Lisp, it must be wrapped into a cl4py.String, like this:

>>> lisp.eval(cl4py.String("foo"))
String("foo")

Frequently Asked Problems

Why does my Lisp subprocess complain about Package QL does not exist.

By default, cl4py starts a Lisp subprocess with sbcl --script. This means, that the Lisp process will ignore any user initialization files, including the Quicklisp setup.

One possible solution is to explicitly load Quicklisp from cl4py:

>>> lisp = cl4py.Lisp()
>>> lisp.eval('(load "~/quicklisp/setup.lisp")')

Related Projects

  • burgled-batteries - A bridge between Python and Lisp. The goal is that Lisp programs can use Python libraries, which is in some sense the opposite of cl4py. Furthermore it relies on the less portable mechanism of FFI calls.
  • CLAUDE - An earlier attempt to access Lisp libraries from Python. The key difference is that cl4py does not run Lisp directly in the host process. This makes cl4py more portable, but complicates the exchange of data.
  • cl-python - A much heavier solution than cl4py --- let's simply implement Python in Lisp! An amazing project. However, cl-python cannot access foreign libraries, e.g., NumPy. And people are probably hesitant to migrate away from CPython.
  • Hy - Python, but with Lisp syntax. This project is certainly a great way to get started with Lisp. It allows you to study the advantages of Lisp's seemingly weird syntax, without leaving the comfortable Python ecosystem. Once you understand the advantages of Lisp, you will doubly appreciate cl4py for your projects.

About

Common Lisp for Python

License:MIT License


Languages

Language:Python 82.3%Language:Common Lisp 17.7%