Class 10 - wrapping external compiled libraries

There are several approaches to wrap external libraries in Python:

  1. ctypes module from standard library,
  2. cffi,
  3. cython,
  4. nanobind.
  5. SIP used by PyQt and WxWidgets,
  6. f2py used by NumPy to wrap Fortran,
  7. SWIG.

Build backends that support extension modules:

  • setuptools - simple C, C++ modules or modules that use custom code to compile external libraries,
  • meson-python - C, C++, Fortran, Rust and other languages supported by Meson,
  • scikit-build-core - C, C++, Fortran and other languages supported by CMake,
  • maturin - Rust.

We will use the WIGXJPF library as our example. A version of the library repackaged to use Meson to compile it is available here. The library was patched and tested to work with Clang and GCC under Mac OS Sequoia 15.1.1, GCC under Ubuntu 24.04.1 LTS inside Docker, and MSVC and GCC under Windows 10 with Visual Studio Community 2022. Under Windows 10, I used Windows Terminal with Powershell. You need to change the codepage of the terminal to Unicode, otherwise Meson will exit with a decoding error. This is done by executing chcp 65001 in the terminal. To verify manually if the project compiles, unpack it, enter the directory with meson.build file and run:

meson setup --buildtype release build_release
meson compile -C build_release

Afterwards, test the library by running test executables:

./build_release/csimple
./build_release/ccsimple

All the projects discussed below are available in class10.zip archive. You can extract them and then call ./archives/unarchive.sh or ./archives/unarchive.ps1 script to extract all of them. Manually, zip files can be unzipped as usual and git bundles are unpacked with git clone name.bundle target_directory. test.sh and test.ps1 scripts automatically compile and run an example script for each package. test.ps1 assumes you use uv to manage your Python installations and venvs. If you do not wish to install it, then you need to modify the script appropriately or simply use it as guide for manual use. The projects that do not use Meson or setuptools contain build.sh and build.ps1 scripts that compile the C code and call the Python module for testing.

We first use the ctypes module (pywigxjpf_ctypes) to call functions from a dynamic library. We compile the library and copy it to the same directory as the Python module. The function signatures need to be specified manually in code for Python to correctly call and interpret function results. We do not package the code.

It is somewhat more convenient to use CFFI project (pywigxjpf_ffi_dynamic) for the same purpose, since we can simply copy C function signatures from wigxjpf.h header file, tell CFFI where to find header files and the name of the library to link with, and call the functions from Python.

An alternative to dynamic linking, which forces us to keep track of the shared library and make sure that the operating system can find it, is to link the library statically to the extension module. Static linking incorporates the whole library in the extension module, which might lead to prohibitively large file sizes. Here and in many other cases this is not an issue and it makes dealing with extension modules simpler. We use CFFI to do static linking in pywigxjpf_ffi_static and pywigxjpf_ffi_static_setuptools projects. The former project is very similar to previous ones, ffibuilder was simply changed to link statically. In the setuptools variant we add our custom build backend that compiles wigxjpf with Meson and then hands off packaging to setuptools, which hands off compilation and linking of the extension module to CFFI. These projects work correctly under Linux and Mac OS, but they fail to compile under Windows 10, as CFFI produces incorrect compiler and linker invocations under this platform. This could be solved by providing suitable extra_compile_args and extra_link_args to CFFI, but that defeats part of the point of using CFFI in the first place. Instead we move on to using Meson to build and package the extension module.

In pywigxjpf_ffi_meson Meson drives the whole process but still uses CFFI. We use gen-cffi-src to execute our ffi_builder_static.py script, which produces the source code of Python extension module pywigxjpf_ffi.c but does not compile it. This is done by Meson because we use the source code as part of py.extension_module definition. We specify wigxjpf_dep as a dependency of the module, which tells Meson to find it and compile it. When we call python -m build Meson creates a source package and a binary wheel. Meson creates the source package by archiving all the files tracked by your version control system. This means that you have to use a VCS such as git or mercurial. On the other hand, the binary wheel contains only the files which Meson was told to “install”. This is why meson.build contains this fragment:

install_subdir(
  'src/pywigxjpf',
  install_dir: py.get_install_dir()
)

Without this, pure Python code would not get included.

In pywigxjpf_nanobind_meson we use nanobind to wrap wigxjpf. The source code of pywigxjpf.cpp is trivially simple and we only add some Python code to define the context manager. We do not specify nanobind as a build dependency in pyproject.toml, because this is handled by Meson.

References

PEPs related to the build process:

Other references: