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 konstin2/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 konstin2/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.9.tar.gz (142.1 kB view details)

Uploaded Source

Built Distributions

maturin-0.12.9-py3-none-win_arm64.whl (4.2 MB view details)

Uploaded Python 3 Windows ARM64

maturin-0.12.9-py3-none-win_amd64.whl (4.9 MB view details)

Uploaded Python 3 Windows x86-64

maturin-0.12.9-py3-none-win32.whl (4.5 MB view details)

Uploaded Python 3 Windows x86

maturin-0.12.9-py3-none-musllinux_1_1_x86_64.whl (8.1 MB view details)

Uploaded Python 3 musllinux: musl 1.1+ x86-64

maturin-0.12.9-py3-none-musllinux_1_1_ppc64le.whl (6.6 MB view details)

Uploaded Python 3 musllinux: musl 1.1+ ppc64le

maturin-0.12.9-py3-none-musllinux_1_1_i686.whl (8.3 MB view details)

Uploaded Python 3 musllinux: musl 1.1+ i686

maturin-0.12.9-py3-none-musllinux_1_1_armv7l.whl (7.9 MB view details)

Uploaded Python 3 musllinux: musl 1.1+ ARMv7l

maturin-0.12.9-py3-none-musllinux_1_1_aarch64.whl (7.7 MB view details)

Uploaded Python 3 musllinux: musl 1.1+ ARM64

maturin-0.12.9-py3-none-manylinux_2_17_s390x.manylinux2014_s390x.whl (8.0 MB view details)

Uploaded Python 3 manylinux: glibc 2.17+ s390x

maturin-0.12.9-py3-none-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl (6.6 MB view details)

Uploaded Python 3 manylinux: glibc 2.17+ ppc64le

maturin-0.12.9-py3-none-manylinux_2_17_armv7l.manylinux2014_armv7l.whl (7.9 MB view details)

Uploaded Python 3 manylinux: glibc 2.17+ ARMv7l

maturin-0.12.9-py3-none-manylinux_2_17_aarch64.manylinux2014_aarch64.whl (7.7 MB view details)

Uploaded Python 3 manylinux: glibc 2.17+ ARM64

maturin-0.12.9-py3-none-manylinux_2_12_i686.manylinux2010_i686.whl (8.3 MB view details)

Uploaded Python 3 manylinux: glibc 2.12+ i686

maturin-0.12.9-py3-none-manylinux_2_5_x86_64.manylinux1_x86_64.whl (8.1 MB view details)

Uploaded Python 3 manylinux: glibc 2.5+ x86-64

maturin-0.12.9-py3-none-macosx_10_9_x86_64.macosx_11_0_arm64.macosx_10_9_universal2.whl (10.6 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.9-py3-none-macosx_10_7_x86_64.whl (5.5 MB view details)

Uploaded Python 3 macOS 10.7+ x86-64

File details

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

File metadata

  • Download URL: maturin-0.12.9.tar.gz
  • Upload date:
  • Size: 142.1 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? No
  • Uploaded via: maturin/0.12.8

File hashes

Hashes for maturin-0.12.9.tar.gz
Algorithm Hash digest
SHA256 56e7a6a5c313c71c67fe6a4c12f85e6a16af76d1732c49ee2706166eff17a494
MD5 8877b608c60d7e1130dbfc59c2497b63
BLAKE2b-256 ee1b545d52dd0c1f075772544267ee1b6e60742f4ca8c1feec4d52738e93133d

See more details on using hashes here.

File details

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

File metadata

File hashes

Hashes for maturin-0.12.9-py3-none-win_arm64.whl
Algorithm Hash digest
SHA256 338ca1873bb383cd701299cf1c6adc418fed4b9c024a427b9ca20a9f234cf243
MD5 dabf421ca8aae3bef497f29cfaf92211
BLAKE2b-256 c047895d5080fead1c16b15448db435d76944369e220857da6e5c11cb3db98e5

See more details on using hashes here.

File details

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

File metadata

File hashes

Hashes for maturin-0.12.9-py3-none-win_amd64.whl
Algorithm Hash digest
SHA256 a393ce084f10dad9496a6f63e802a7672e333e29dde10fb9415e52cf9af7967d
MD5 137585847dbd9b6e50e6b2d3e07bb42a
BLAKE2b-256 6dfe3a2411b6ac314154300df87bf31ffc169cd11624416e05211bec9a64a067

See more details on using hashes here.

File details

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

File metadata

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

File hashes

Hashes for maturin-0.12.9-py3-none-win32.whl
Algorithm Hash digest
SHA256 469e19c7963514628d34e27e39899cbc0cdff0255d13820fdc82033634313778
MD5 e31f79adfd52e1624210ded747ebbde8
BLAKE2b-256 5c6e053e5b7122f3b080b3492194ace6b06040c6cb494ddef92e379c1b04ff51

See more details on using hashes here.

File details

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

File metadata

File hashes

Hashes for maturin-0.12.9-py3-none-musllinux_1_1_x86_64.whl
Algorithm Hash digest
SHA256 3cf28c95c1226544c3e097614b9c5f60386ba34da66f6df5fb93a1e25e210948
MD5 987b3a6291e1d9476979a420ea2fd83a
BLAKE2b-256 9b74ba12af4c77d171d34d75c0e984fafd0aee93f374adf2aff9d28d703280a6

See more details on using hashes here.

File details

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

File metadata

File hashes

Hashes for maturin-0.12.9-py3-none-musllinux_1_1_ppc64le.whl
Algorithm Hash digest
SHA256 7c94e9af18e8252412427a28ecbe6aa1fab07cc9719ac633c955254ba4984052
MD5 7e80cf539a95fd49249e6ffb4b525c8b
BLAKE2b-256 50efb46ab695ee9cd7ca920215d1280252e8518fdc4345697f0863a255e6beae

See more details on using hashes here.

File details

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

File metadata

File hashes

Hashes for maturin-0.12.9-py3-none-musllinux_1_1_i686.whl
Algorithm Hash digest
SHA256 5563045203c9e856788c457fada635edbee35950255b507631ed6f11484e9eb0
MD5 e4f01405f6d8c68cd9c958899e7dfe4b
BLAKE2b-256 35a1b089c56359c72784901ef2da67ac1c5034575f3571223e0fbd4d1429ba2b

See more details on using hashes here.

File details

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

File metadata

File hashes

Hashes for maturin-0.12.9-py3-none-musllinux_1_1_armv7l.whl
Algorithm Hash digest
SHA256 de0fb98d67101a19fe4729961d4c095db72b2e2d726089636d538fb989930756
MD5 c2adb57556b7d5d757d060a4c9c44330
BLAKE2b-256 570f68b11eaed96a661fee3e40f50316ab573ffc5da7c2eac265fc0b783aa7bd

See more details on using hashes here.

File details

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

File metadata

File hashes

Hashes for maturin-0.12.9-py3-none-musllinux_1_1_aarch64.whl
Algorithm Hash digest
SHA256 a2cea72cd1aa1c419d49a2546d162460a2731b7f72ff3ad60068463ea86dfec3
MD5 b09d66699771379724344d020a863f8e
BLAKE2b-256 83c9eec4fcdcf765d5332b8558e06110369f346a2c7c2656bf20e2490352cc89

See more details on using hashes here.

File details

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

File metadata

File hashes

Hashes for maturin-0.12.9-py3-none-manylinux_2_17_s390x.manylinux2014_s390x.whl
Algorithm Hash digest
SHA256 ff5022c595348d2a2234838484a607feb10396d468edf1fdb34f4a1d993b316d
MD5 b4beac2ceed526f7a9a972fbe77e73be
BLAKE2b-256 2202fef405f111cfc4fcf02f0d10292bb76ea6ce97e758869f14f4eb7c15d016

See more details on using hashes here.

File details

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

File metadata

File hashes

Hashes for maturin-0.12.9-py3-none-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl
Algorithm Hash digest
SHA256 3db0e2c46febc0562d4d0780a80847213ca45ce3d19c26863245239ac16cbdcc
MD5 3b724850a5627c7fc9c64f279ac21058
BLAKE2b-256 78f3c5f68462e4acf26d655e305b817700f1961198a0a8ada854cbe50d4f77a1

See more details on using hashes here.

File details

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

File metadata

File hashes

Hashes for maturin-0.12.9-py3-none-manylinux_2_17_armv7l.manylinux2014_armv7l.whl
Algorithm Hash digest
SHA256 386496c6b9af7c93ef704c1a8816a83ae2051e514c654b837975e44284595b36
MD5 afb2bfd4650751165f63869a942d8c1a
BLAKE2b-256 fef418a1c1bbbf4a7e491c69b435a8d2cf99a41d3ccdfb2bd8407069d71146ea

See more details on using hashes here.

File details

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

File metadata

File hashes

Hashes for maturin-0.12.9-py3-none-manylinux_2_17_aarch64.manylinux2014_aarch64.whl
Algorithm Hash digest
SHA256 e9859385b538d515e3b16822564589a97810d6fb5aa4a987dba819d92de69a83
MD5 53855015cb63b9334d9fc68dbd66221b
BLAKE2b-256 9df24aa4b2cd4032104388d93fc97a7bf67f54e8aa05581e56ef7c19ef2fdf74

See more details on using hashes here.

File details

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

File metadata

File hashes

Hashes for maturin-0.12.9-py3-none-manylinux_2_12_i686.manylinux2010_i686.whl
Algorithm Hash digest
SHA256 2f4608f3d1375c840ddb24be3ce964a792a1333fcdabc3a0d071725635ebe025
MD5 b1f7642c548987742e78152ed38e2434
BLAKE2b-256 c92bc5563dab0ac1f59994987930f811be2258c3ba53b05ae949cc4fcfd6d13a

See more details on using hashes here.

File details

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

File metadata

File hashes

Hashes for maturin-0.12.9-py3-none-manylinux_2_5_x86_64.manylinux1_x86_64.whl
Algorithm Hash digest
SHA256 4d498f2d34a7061db4ae4595d793c4e3b8fba41e3686b0bb3b0922c777911194
MD5 32d671912b981d53516f38e1f5736bb4
BLAKE2b-256 0987897e5fafaa9a34b64fda862b715f3f109c004ff2c672b6d3c21b5877526e

See more details on using hashes here.

File details

Details for the file maturin-0.12.9-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.9-py3-none-macosx_10_9_x86_64.macosx_11_0_arm64.macosx_10_9_universal2.whl
Algorithm Hash digest
SHA256 d5d68ec1019fd4bb4d04486b8db1c9dbee1ac7cfcd070e78f09c62f24632a280
MD5 87787c89f7a275471cc7013f78efc83c
BLAKE2b-256 9966b1334f6d033e99ed729351d4762f91e5dfe4cec7b46aa7eef96ac5d0a97c

See more details on using hashes here.

File details

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

File metadata

File hashes

Hashes for maturin-0.12.9-py3-none-macosx_10_7_x86_64.whl
Algorithm Hash digest
SHA256 798d7616f8a546cd26d81bafd24c8b6eccfca3dda362ffb2f8d0b4495a09d1b7
MD5 160a02a9d743ee835370cb5d6520cf7f
BLAKE2b-256 474154ef09c48a141b12eeacd3efec31353cfc858cf1fdeab33a1e4f72089d61

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