Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

Generate python bindings, what methods/programs to use [closed]

I'm looking at using python (CPython) in my program to both allow user scripting in my environment and to allow me to use pyside, the qt bindings for c++ to create the GUI for my application. These can be effectively separated with the idea that the GUI python code can later be compiled away for speed (if that would be possible).

I'm very new to python and i'm really looking for the most efficient way to generate solid bindings with the absolute minimum of additional code to maintain as the bindings are likely to change often as the project evolves. I need it so that python classes extend c++ classes that have virtuals.

I have already looked into PyBindGen and it chokes too often on things in my library for it to be practically useful.

Any help/advice/links/workflows you recommend in this regard would be very helpful.

like image 651
ceorron Avatar asked Dec 21 '12 12:12

ceorron


People also ask

How do Python bindings work?

The approach Cython takes to creating Python bindings uses a Python-like language to define the bindings and then generates C or C++ code that can be compiled into the module. There are several methods for building Python bindings with Cython . The most common one is to use setup from distutils .

What is pybind11?

pybind11 is a lightweight header-only library that exposes C++ types in Python and vice versa, mainly to create Python bindings of existing C++ code. Its goals and syntax are similar to the excellent Boost.

What is Cppyy?

cppyy is an automatic, run-time, Python-C++ bindings generator, for calling C++ from Python and Python from C++.


1 Answers

There are only two projects I know to have automatic binding generators for C++. The first one is SWIG. As some other answer has already said, it is a little bit old style, but it works. The second one is Boost.Python - by itself, it does not generate the bindings automatically, but you can use Boost.Pyste to do it for you. It requires GCC-XML to parse your original source code and write the Boost.Python bindings. Both options do support virtual methods in C++ that can be overloaded from Python.

That said, I must complement that, normally, when you are binding, you don't blindly bind everything you have in C++ into Python - if you do it like this you won't get a very pythonic feeling from the python side. Instead, you design how you'd like your library to be used in Python, in the most pythonic possible way, and then you walk back and see how to patch it through to your C++ code using one of the possible binding libraries. For example, instead of dealing with std::vector's, you'd prefer your library calls handle Python lists or iterables. If your C++ library receives a std::map, you'd like that is handled with Python dictionaries. If it is an array, maybe a numpy.ndarray would be more convenient. And so on...

That said, you can still design your bindings so that your maintenance is minimized.

Here is a list of other Python/C++ wrapping, in case you decide to look around a bit further:

  1. SWIG - as you already know
  2. Boost.Python - this is what we normally use around here - pretty well structured
  3. Cython - very neat syntax close to Python - they claim to be much faster than Boost.Python
  4. SIP - not very much spread, but it is there
  5. pybind11 - Syntax similar to Boost.Python, compact implementation thanks to C++11.

These are now inactive:

  1. PyBindGen - claims to be the fastest, but inactive since may 21st, 2017, previous release on 2014 - currently maintained on github (https://github.com/gjcarneiro/pybindgen/releases)
  2. ECS:Python - inactive since december 6th, 2014 (v2.8), moved to github (https://github.com/MarcusTomlinson/ECS-Python)
  3. PyCXX - C++ facilities to make it easier to write Python extensions - inactive? Last release was v7.0.2 on 23rd april 2017
  4. CLIF - CLIF provides a common foundation for creating C++ wrapper generators for various languages - inactive? Not much activity on repo (only 14 commits)

For completeness, it is also possible to load compiled C code directly into Python without creating formal bindings. You can do this using FFI with any of these two Python modules:

  1. ctypes - This native to Python and requires no external module installation.
  2. cffi - This is a new package inspired by the equivalent design on the Lua JIT.
like image 193
André Anjos Avatar answered Sep 27 '22 20:09

André Anjos