Class 10 - wrapping external compiled libraries
There are several approaches to wrap external libraries in Python:
- ctypes module from standard library,
- cffi,
- cython,
- nanobind.
- SIP used by PyQt and WxWidgets,
- f2py used by NumPy to wrap Fortran,
- 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:
- PEP 508 – Dependency specification for Python Software Packages
- PEP 517 – A build-system independent format for source trees
- PEP 518 – Specifying Minimum Build System Requirements for Python Projects
- PEP 660 – Editable installs for pyproject.toml based builds (wheel based)
Other references: