Skip to main content

Build and publish crates with pyo3, rust-cpython and cffi bindings as well as rust binaries as python packages

Project description

Maturin

formerly pyo3-pack

Actions Status FreeBSD Crates.io PyPI Maturin User Guide Chat on Gitter

Build and publish crates with pyo3, rust-cpython and cffi bindings as well as rust binaries as python packages.

This project is meant as a zero configuration replacement for setuptools-rust and milksnake. It supports building wheels for python 3.5+ on windows, linux, mac and freebsd, can upload them to pypi and has basic pypy support.

Check out the User Guide!

Usage

You can either download binaries from the latest release or install it with pip:

pip install maturin

There are four main commands:

  • maturin new creates a new cargo project with maturin configured.
  • maturin publish builds the crate into python packages and publishes them to pypi.
  • maturin build builds the wheels and stores them in a folder (target/wheels by default), but doesn't upload them. It's possible to upload those with twine or maturin upload.
  • maturin develop builds the crate and installs it as a python module directly in the current virtualenv. Note that while maturin develop is faster, it doesn't support all the feature that running pip install after maturin build supports.

pyo3 and rust-cpython bindings are automatically detected, for cffi or binaries you need to pass -b cffi or -b bin. maturin doesn't need extra configuration files and doesn't clash with an existing setuptools-rust or milksnake configuration. You can even integrate it with testing tools such as tox. There are examples for the different bindings in the test-crates folder.

The name of the package will be the name of the cargo project, i.e. the name field in the [package] section of Cargo.toml. The name of the module, which you are using when importing, will be the name value in the [lib] section (which defaults to the name of the package). For binaries, it's simply the name of the binary generated by cargo.

Python packaging basics

Python packages come in two formats: A built form called wheel and source distributions (sdist), both of which are archives. A wheel can be compatible with any python version, interpreter (cpython and pypy, mainly), operating system and hardware architecture (for pure python wheels), can be limited to a specific platform and architecture (e.g. when using ctypes or cffi) or to a specific python interpreter and version on a specific architecture and operating system (e.g. with pyo3 and rust-cpython).

When using pip install on a package, pip tries to find a matching wheel and install that. If it doesn't find one, it downloads the source distribution and builds a wheel for the current platform, which requires the right compilers to be installed. Installing a wheel is much faster than installing a source distribution as building wheels is generally slow.

When you publish a package to be installable with pip install, you upload it to pypi, the official package repository. For testing, you can use test pypi instead, which you can use with pip install --index-url https://test.pypi.org/simple/. Note that for publishing for linux, you need to use the manylinux docker container, while for publishing from your repository you can use the messense/maturin-action github action.

pyo3 and rust-cpython

For pyo3 and rust-cpython, maturin can only build packages for installed python versions. On linux and mac, all python versions in PATH are used. If you don't set your own interpreters with -i, a heuristic is used to search for python installations. On windows all versions from the python launcher (which is installed by default by the python.org installer) and all conda environments except base are used. You can check which versions are picked up with the list-python subcommand.

pyo3 will set the used python interpreter in the environment variable PYTHON_SYS_EXECUTABLE, which can be used from custom build scripts. Maturin can build and upload wheels for pypy with pyo3, even though only pypy3.7-7.3 on linux is tested.

Cffi

Cffi wheels are compatible with all python versions including pypy. If cffi isn't installed and python is running inside a virtualenv, maturin will install it, otherwise you have to install it yourself (pip install cffi).

maturin uses cbindgen to generate a header file, which can be customized by configuring cbindgen through a cbindgen.toml file inside your project root. Alternatively you can use a build script that writes a header file to $PROJECT_ROOT/target/header.h.

Based on the header file maturin generates a module which exports an ffi and a lib object.

Example of a custom build script
use cbindgen;
use std::env;
use std::path::Path;

fn main() {
    let crate_dir = env::var("CARGO_MANIFEST_DIR").unwrap();

    let bindings = cbindgen::Builder::new()
        .with_no_includes()
        .with_language(cbindgen::Language::C)
        .with_crate(crate_dir)
        .generate()
        .unwrap();
    bindings.write_to_file(Path::new("target").join("header.h"));
}

Mixed rust/python projects

To create a mixed rust/python project, create a folder with your module name (i.e. lib.name in Cargo.toml) next to your Cargo.toml and add your python sources there:

my-project
├── Cargo.toml
├── my_project
│   ├── __init__.py
│   └── bar.py
├── pyproject.toml
├── Readme.md
└── src
    └── lib.rs

You can specify a different python source directory in Cargo.toml by setting package.metadata.maturin.python-source, for example

[package.metadata.maturin]
python-source = "python"

then the project structure would look like this:

my-project
├── Cargo.toml
├── python
│   └── my_project
│       ├── __init__.py
│       └── bar.py
├── pyproject.toml
├── Readme.md
└── src
    └── lib.rs

maturin will add the native extension as a module in your python folder. When using develop, maturin will copy the native library and for cffi also the glue code to your python folder. You should add those files to your gitignore.

With cffi you can do from .my_project import lib and then use lib.my_native_function, with pyo3/rust-cpython you can directly from .my_project import my_native_function.

Example layout with pyo3 after maturin develop:

my-project
├── Cargo.toml
├── my_project
│   ├── __init__.py
│   ├── bar.py
│   └── my_project.cpython-36m-x86_64-linux-gnu.so
├── Readme.md
└── src
    └── lib.rs

Python metadata

maturin supports PEP 621, you can specify python package metadata in pyproject.toml. maturin merges metadata from Cargo.toml and pyproject.toml, pyproject.toml take precedence over Cargo.toml.

To specify python dependencies, add a list dependencies in a [project] section in the pyproject.toml. This list is equivalent to install_requires in setuptools:

[project]
name = "my-project"
dependencies = ["flask~=1.1.0", "toml==0.10.0"]

Pip allows adding so called console scripts, which are shell commands that execute some function in you program. You can add console scripts in a section [project.scripts]. The keys are the script names while the values are the path to the function in the format some.module.path:class.function, where the class part is optional. The function is called with no arguments. Example:

[project.scripts]
get_42 = "my_project:DummyClass.get_42"

You can also specify trove classifiers in your Cargo.toml under project.classifiers:

[project]
name = "my-project"
classifiers = ["Programming Language :: Python"]

Source distribution

maturin supports building through pyproject.toml. To use it, create a pyproject.toml next to your Cargo.toml with the following content:

[build-system]
requires = ["maturin>=0.12,<0.13"]
build-backend = "maturin"

If a pyproject.toml with a [build-system] entry is present, maturin will build a source distribution of your package, unless --no-sdist is specified. The source distribution will contain the same files as cargo package. To only build a source distribution, pass --interpreter without any values.

You can then e.g. install your package with pip install .. With pip install . -v you can see the output of cargo and maturin.

You can use the options compatibility, skip-auditwheel, bindings, strip, cargo-extra-args and rustc-extra-args under [tool.maturin] the same way you would when running maturin directly. The bindings key is required for cffi and bin projects as those can't be automatically detected. Currently, all builds are in release mode (see this thread for details).

For a non-manylinux build with cffi bindings you could use the following:

[build-system]
requires = ["maturin>=0.12,<0.13"]
build-backend = "maturin"

[tool.maturin]
bindings = "cffi"
compatibility = "linux"

manylinux option is also accepted as an alias of compatibility for backward compatibility with old version of maturin.

To include arbitrary files in the sdist for use during compilation specify sdist-include as an array of globs:

[tool.maturin]
sdist-include = ["path/**/*"]

There's a maturin sdist command for only building a source distribution as workaround for pypa/pip#6041.

Manylinux and auditwheel

For portability reasons, native python modules on linux must only dynamically link a set of very few libraries which are installed basically everywhere, hence the name manylinux. The pypa offers special docker images and a tool called auditwheel to ensure compliance with the manylinux rules. If you want to publish widely usable wheels for linux pypi, you need to use a manylinux docker image.

The Rust compiler since version 1.47 requires at least glibc 2.11, so you need to use at least manylinux2010. For publishing, we recommend enforcing the same manylinux version as the image with the manylinux flag, e.g. use --manylinux 2014 if you are building in quay.io/pypa/manylinux2014_x86_64. The messense/maturin-action github action already takes care of this if you set e.g. manylinux: 2014.

maturin contains a reimplementation of auditwheel automatically checks the generated library and gives the wheel the proper. If your system's glibc is too new or you link other shared libraries, it will assign the linux tag. You can also manually disable those checks and directly use native linux target with --manylinux off.

For full manylinux compliance you need to compile in a CentOS docker container. The pyo3/maturin image is based on the manylinux2010 image, and passes arguments to the maturin binary. You can use it like this:

docker run --rm -v $(pwd):/io ghcr.io/pyo3/maturin build --release  # or other maturin arguments

Note that this image is very basic and only contains python, maturin and stable rust. If you need additional tools, you can run commands inside the manylinux container. See konstin/complex-manylinux-maturin-docker for a small educational example or nanoporetech/fast-ctc-decode for a real world setup.

maturin itself is manylinux compliant when compiled for the musl target.

Code

The main part is the maturin library, which is completely documented and should be well integrable. The accompanying main.rs takes care username and password for the pypi upload and otherwise calls into the library.

The sysconfig folder contains the output of python -m sysconfig for different python versions and platform, which is helpful during development.

You need to install cffi and virtualenv (pip install cffi virtualenv) to run the tests.

There are some optional hacks that can speed up the tests (over 80s to 17s on my machine).

  1. By running cargo build --release --manifest-path test-crates/cargo-mock/Cargo.toml you can activate a cargo cache avoiding to rebuild the pyo3 test crates with every python version.
  2. Delete target/test-cache to clear the cache (e.g. after changing a test crate) or remove test-crates/cargo-mock/target/release/cargo to deactivate it.
  3. By running the tests with the faster-tests feature, binaries are stripped and wheels are only stored and not compressed.

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

maturin-0.12.15.tar.gz (148.2 kB view details)

Uploaded Source

Built Distributions

maturin-0.12.15-py3-none-win_arm64.whl (4.4 MB view details)

Uploaded Python 3 Windows ARM64

maturin-0.12.15-py3-none-win_amd64.whl (5.1 MB view details)

Uploaded Python 3 Windows x86-64

maturin-0.12.15-py3-none-win32.whl (4.6 MB view details)

Uploaded Python 3 Windows x86

maturin-0.12.15-py3-none-musllinux_1_1_x86_64.whl (8.3 MB view details)

Uploaded Python 3 musllinux: musl 1.1+ x86-64

maturin-0.12.15-py3-none-musllinux_1_1_ppc64le.whl (7.3 MB view details)

Uploaded Python 3 musllinux: musl 1.1+ ppc64le

maturin-0.12.15-py3-none-musllinux_1_1_i686.whl (8.4 MB view details)

Uploaded Python 3 musllinux: musl 1.1+ i686

maturin-0.12.15-py3-none-musllinux_1_1_armv7l.whl (8.0 MB view details)

Uploaded Python 3 musllinux: musl 1.1+ ARMv7l

maturin-0.12.15-py3-none-musllinux_1_1_aarch64.whl (7.9 MB view details)

Uploaded Python 3 musllinux: musl 1.1+ ARM64

maturin-0.12.15-py3-none-manylinux_2_17_s390x.manylinux2014_s390x.whl (8.6 MB view details)

Uploaded Python 3 manylinux: glibc 2.17+ s390x

maturin-0.12.15-py3-none-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl (7.3 MB view details)

Uploaded Python 3 manylinux: glibc 2.17+ ppc64le

maturin-0.12.15-py3-none-manylinux_2_17_armv7l.manylinux2014_armv7l.whl (8.0 MB view details)

Uploaded Python 3 manylinux: glibc 2.17+ ARMv7l

maturin-0.12.15-py3-none-manylinux_2_17_aarch64.manylinux2014_aarch64.whl (7.9 MB view details)

Uploaded Python 3 manylinux: glibc 2.17+ ARM64

maturin-0.12.15-py3-none-manylinux_2_12_i686.manylinux2010_i686.whl (8.4 MB view details)

Uploaded Python 3 manylinux: glibc 2.12+ i686

maturin-0.12.15-py3-none-manylinux_2_5_x86_64.manylinux1_x86_64.whl (8.3 MB view details)

Uploaded Python 3 manylinux: glibc 2.5+ x86-64

maturin-0.12.15-py3-none-macosx_10_9_x86_64.macosx_11_0_arm64.macosx_10_9_universal2.whl (10.9 MB view details)

Uploaded Python 3 macOS 10.9+ universal2 (ARM64, x86-64) macOS 10.9+ x86-64 macOS 11.0+ ARM64

maturin-0.12.15-py3-none-macosx_10_7_x86_64.whl (5.7 MB view details)

Uploaded Python 3 macOS 10.7+ x86-64

File details

Details for the file maturin-0.12.15.tar.gz.

File metadata

  • Download URL: maturin-0.12.15.tar.gz
  • Upload date:
  • Size: 148.2 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? No
  • Uploaded via: maturin/0.12.14

File hashes

Hashes for maturin-0.12.15.tar.gz
Algorithm Hash digest
SHA256 2ca6222ae70276ba2f2d0c7296804bf62db87fec73198f274f8c1729a3f6c8e5
MD5 4b054d6d3c5d66aa75f4713c7d9dd02d
BLAKE2b-256 304db8a32e0bd711f7905fa8f10487be7dbf9992c5b5b3bfa8734abf2487d967

See more details on using hashes here.

File details

Details for the file maturin-0.12.15-py3-none-win_arm64.whl.

File metadata

File hashes

Hashes for maturin-0.12.15-py3-none-win_arm64.whl
Algorithm Hash digest
SHA256 da6b9568e14696856863b21f90c764a748085d942e2b856d6b3a3b9859aeb559
MD5 2b42598a7e75870651ddd6b05c326809
BLAKE2b-256 6ce417fa452e0c789244375b3b9c5b799dea42d9d1fe559670611ad8e9d6a01a

See more details on using hashes here.

File details

Details for the file maturin-0.12.15-py3-none-win_amd64.whl.

File metadata

File hashes

Hashes for maturin-0.12.15-py3-none-win_amd64.whl
Algorithm Hash digest
SHA256 18e853765bd0c96dc974f2514b85a819edb435cae7bf36e8caeb27aa0bb1116d
MD5 d9c476b9acd3290917794647f50c95ef
BLAKE2b-256 f61dd7be3f7aa2f15e26830f76edb1f18a8cb46b3765f99508f0242e7f1dbde0

See more details on using hashes here.

File details

Details for the file maturin-0.12.15-py3-none-win32.whl.

File metadata

  • Download URL: maturin-0.12.15-py3-none-win32.whl
  • Upload date:
  • Size: 4.6 MB
  • Tags: Python 3, Windows x86
  • Uploaded using Trusted Publishing? No
  • Uploaded via: maturin/0.12.14

File hashes

Hashes for maturin-0.12.15-py3-none-win32.whl
Algorithm Hash digest
SHA256 da0aa84b002d846ee1fb74a3d045a036ec9b224c8eea7dfc13f15a74488e1f61
MD5 a3d6b2a92ac558344850004ecaff998b
BLAKE2b-256 b66e39cc637b29d3ae47dfca87a1a6e6aa58cdc562749db5d000109e339ab051

See more details on using hashes here.

File details

Details for the file maturin-0.12.15-py3-none-musllinux_1_1_x86_64.whl.

File metadata

File hashes

Hashes for maturin-0.12.15-py3-none-musllinux_1_1_x86_64.whl
Algorithm Hash digest
SHA256 5e0ab3251d759aad04778afbf25c8dd0fdcd300c9d97e660518def54c94e83ea
MD5 50fa8a5369634f6ebd892f57865073d8
BLAKE2b-256 06961a67502b51ae13908db6dfcd45aec2e844caf8921b32dbc7d511a12323c6

See more details on using hashes here.

File details

Details for the file maturin-0.12.15-py3-none-musllinux_1_1_ppc64le.whl.

File metadata

File hashes

Hashes for maturin-0.12.15-py3-none-musllinux_1_1_ppc64le.whl
Algorithm Hash digest
SHA256 f719d75b028f2a5f1b873ddc710c121b4b2bcb404faf18df8af5e2e924636d98
MD5 7a00d66d843d05b988e3ddd3bf4d2671
BLAKE2b-256 a3c1f6579914a065c72547da1d799288b298167265fd7c2d2c752b31c05b2fa0

See more details on using hashes here.

File details

Details for the file maturin-0.12.15-py3-none-musllinux_1_1_i686.whl.

File metadata

File hashes

Hashes for maturin-0.12.15-py3-none-musllinux_1_1_i686.whl
Algorithm Hash digest
SHA256 ad8921ca470c5a6dd2e5412c791b1dfc4e22a8fa8198d5b244e1bcae52c10d43
MD5 0e27db9bca36eba4370718466c040c1d
BLAKE2b-256 5ce9bc465af334d393fb4919b75799219e1ae593e491cfedbaccb8cf03064e84

See more details on using hashes here.

File details

Details for the file maturin-0.12.15-py3-none-musllinux_1_1_armv7l.whl.

File metadata

File hashes

Hashes for maturin-0.12.15-py3-none-musllinux_1_1_armv7l.whl
Algorithm Hash digest
SHA256 072b7e0099ff4f6772565fed8fcd14c45d6943cef8c43fc1214fec41eebcbed7
MD5 08825ed7619e60c1cb183909a058bcaa
BLAKE2b-256 96d1b53cf141ced495bb84fcc933a82fc1faa4a387b32c2f9f971bc81b9d935e

See more details on using hashes here.

File details

Details for the file maturin-0.12.15-py3-none-musllinux_1_1_aarch64.whl.

File metadata

File hashes

Hashes for maturin-0.12.15-py3-none-musllinux_1_1_aarch64.whl
Algorithm Hash digest
SHA256 a58f7be0b6a79c0f72dd088e962fdde397032d46e64bd4c368821282a2139265
MD5 095c1441879fc3f340b3492a8817337e
BLAKE2b-256 7033042ff90d1e93197d4a8588865f9312bf7186883b86e9f1d43c844807e850

See more details on using hashes here.

File details

Details for the file maturin-0.12.15-py3-none-manylinux_2_17_s390x.manylinux2014_s390x.whl.

File metadata

File hashes

Hashes for maturin-0.12.15-py3-none-manylinux_2_17_s390x.manylinux2014_s390x.whl
Algorithm Hash digest
SHA256 39e43173f6d2165f09369f85d772387c14f23790b2f17a37a53eb1086c53b838
MD5 70d5c676144b22959fd8105386cface3
BLAKE2b-256 8641e6560063e9e2bd2ef1e2fd18d763a5b2eeb1d5585f3b5b8ebbd4fdc90fda

See more details on using hashes here.

File details

Details for the file maturin-0.12.15-py3-none-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl.

File metadata

File hashes

Hashes for maturin-0.12.15-py3-none-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl
Algorithm Hash digest
SHA256 052703cc6834a91ade9dc2065a59d3a545d2d887a62a4346f5ba1b9344a316ab
MD5 0e1b7e8acefd61522b18fc82a03453e8
BLAKE2b-256 59be98d0e9d9ab6773acbfd3804980025fd290f2e01dafd57ee92ee69531d7ca

See more details on using hashes here.

File details

Details for the file maturin-0.12.15-py3-none-manylinux_2_17_armv7l.manylinux2014_armv7l.whl.

File metadata

File hashes

Hashes for maturin-0.12.15-py3-none-manylinux_2_17_armv7l.manylinux2014_armv7l.whl
Algorithm Hash digest
SHA256 87546abc68166c8ff9aaba94edb8f16d2d40bcb70e3da7687105bbbf3eee02bd
MD5 849151c8fda1b806d07c5caa06cf7635
BLAKE2b-256 c66f26b04d63a1a245dfda27c54c096bb26aae5b35a58e24e8b0b770d109aa5a

See more details on using hashes here.

File details

Details for the file maturin-0.12.15-py3-none-manylinux_2_17_aarch64.manylinux2014_aarch64.whl.

File metadata

File hashes

Hashes for maturin-0.12.15-py3-none-manylinux_2_17_aarch64.manylinux2014_aarch64.whl
Algorithm Hash digest
SHA256 0215a75ef14a982128d2a66ece5471171648f0477662970846082e107a7ec8fe
MD5 7330c8c947d237e152b6392fe46b725c
BLAKE2b-256 844b80b84cd1f840676ae7a0fa0d4c42d1e16ff46a73dfecb2f696f4e1748736

See more details on using hashes here.

File details

Details for the file maturin-0.12.15-py3-none-manylinux_2_12_i686.manylinux2010_i686.whl.

File metadata

File hashes

Hashes for maturin-0.12.15-py3-none-manylinux_2_12_i686.manylinux2010_i686.whl
Algorithm Hash digest
SHA256 cb8c6c87179154b6dc101dc64a1b06a2519727042275b042e908018f5eb34c1f
MD5 fed22e933eaa829a7447d43670dee408
BLAKE2b-256 2e6465ddd48aa25b4d12e50c75fb91e06eeff44a87eaa3541a1c6263f496ede9

See more details on using hashes here.

File details

Details for the file maturin-0.12.15-py3-none-manylinux_2_5_x86_64.manylinux1_x86_64.whl.

File metadata

File hashes

Hashes for maturin-0.12.15-py3-none-manylinux_2_5_x86_64.manylinux1_x86_64.whl
Algorithm Hash digest
SHA256 31527eeebb3c6a14cd856cef29ce8d52de650603e09e1a7ca41251e4e6404114
MD5 03806059942103e781b1f29abb04b1f7
BLAKE2b-256 279ae8ff7e60b9eaaa104b1574cd6207239978f27f4b5acd6c6bd4981821df32

See more details on using hashes here.

File details

Details for the file maturin-0.12.15-py3-none-macosx_10_9_x86_64.macosx_11_0_arm64.macosx_10_9_universal2.whl.

File metadata

File hashes

Hashes for maturin-0.12.15-py3-none-macosx_10_9_x86_64.macosx_11_0_arm64.macosx_10_9_universal2.whl
Algorithm Hash digest
SHA256 a16dc4b55d2fd3a5d215ae162c9686f2ebe571342fc8c77b6bca6ebd7ec38de8
MD5 c5e95b00145adea5d3d684b90725909a
BLAKE2b-256 13a0aeee181d24fd6d44c25094c386b598c2c4384e10f45e324e02a404509ec7

See more details on using hashes here.

File details

Details for the file maturin-0.12.15-py3-none-macosx_10_7_x86_64.whl.

File metadata

File hashes

Hashes for maturin-0.12.15-py3-none-macosx_10_7_x86_64.whl
Algorithm Hash digest
SHA256 9a7350b3eb11668ee7469d44b629e1b70af107fb2295d1cb11fd82370f1a687c
MD5 c76f5219df414151208c9c893208f5d5
BLAKE2b-256 f58bee660c925c036a4eb7680e71afaf8f332ca9aeb3fcd8bdd251ec81b627c6

See more details on using hashes here.

Supported by

AWS AWS Cloud computing and Security Sponsor Datadog Datadog Monitoring Fastly Fastly CDN Google Google Download Analytics Microsoft Microsoft PSF Sponsor Pingdom Pingdom Monitoring Sentry Sentry Error logging StatusPage StatusPage Status page