Building from source#
Warning
If you are building the latest development version of Pyodide from the main
branch, please make sure to follow the build instructions from the dev
version of the documentation at
pyodide.org/en/latest/
Pyodide can be built from sources on different platforms,
on Linux it is easiest using the Pyodide Docker image. This approach works with any native operating system as long as Docker is installed. You can also build on your native Linux OS if the correct build prerequisites are installed.
on MacOS it is recommended to install dependencies via conda-forge or using Homebrew, particularly with the M1 ARM CPU. Building with Docker is possible but very slow.
It is not possible to build on Windows, but you can use Windows Subsystem for Linux to create a Linux build environment.
Build instructions#
The Pyodide repository has a git submodule called pyodide-build. Make sure to
do a recursive clone:
git clone --recursive https://github.com/pyodide/pyodide
or if you have already cloned the Pyodide repository without the --recursive
flag, you may initialize the submodule with:
git submodule update --init
If you change git branches, make sure you update pyodide-build with
git submodule update.
Using Docker#
We provide a Debian-based x86_64 Docker image
(pyodide/pyodide-env) on
Docker Hub with the dependencies already installed to make it easier to build
Pyodide.
Note
These Docker images are also available from the Github packages at
github.com/orgs/pyodide/packages.
Install Docker
From a git checkout of Pyodide, run
./run_dockerRun
maketo build.
Note
If make is failing with Permission denied, you can run ./run_docker with root permissions: ./run_docker --root
Note
You can control the resources allocated to the build by setting the env
vars EMSDK_NUM_CORE, EMCC_CORES and PYODIDE_JOBS (the default for each is
4).
If running make deterministically stops at some point,
increasing the maximum RAM usage available to the docker container might help.
(The RAM available to the container is different from the physical RAM capacity of the machine.)
Ideally,
at least 3 GB of RAM should be available to the docker container to build
Pyodide smoothly. These settings can be changed via Docker preferences (see
here).
You can edit the files in the shared pyodide source folder on your host
machine (outside of Docker), and then repeatedly run make inside the Docker
environment to test your changes.
Using the “Docker” dev container#
We provide a dev container configuration that is equivalent to the use of
./run_docker script. It can be used in Visual Studio Code and
on GitHub Codespaces.
When prompted, select “Docker”.
Using the “Conda” dev container#
We provide another dev container configuration that corresponds to the “Linux with conda” method described below. When Visual Studio Code or GitHub Codespaces prompts for the dev container configuration, select “Conda”.
Using make#
Make sure the prerequisites for emsdk are installed. Pyodide will build a custom, patched version of emsdk, so there is no need to build it yourself prior.
You need Python 3.13.2 to run the build scripts. To make sure that the correct Python is used during the build it is recommended to use a virtual environment or a conda environment.
To build on Linux, you need:
A working native compiler toolchain, enough to build CPython.
CMake (required to install Emscripten)
You would need a working native compiler toolchain, enough to build CPython, for example,
apt install build-essentialon Debian based systems.Conda which can be installed from MiniForge
Then install the required Python version and other build dependencies in a separate conda environment,
conda env create -f environment.yml
conda activate pyodide-env
You would need,
System libraries in the root directory:
xcode-select --installConda which can be installed using Miniforge (both for Intel and M1 CPU)
Then install the required Python version and other build dependencies in a separate conda environment,
conda env create -f environment.yml
conda activate pyodide-env
To build on macOS with Homebrew, you need:
System command line tools:
xcode-select --installHomebrew for installing dependencies
Pyodide requires CMake3 but Homebrew installs CMake4. Use pip for cmake
pip install cmake==3.31brew install coreutils cmake autoconf automake libtool libffi ccache wget rustup-initInitialize Rust (needed for building test-rust-* packages):
rustup-initInstall the GNU patch and GNU sed (
brew install gpatch gnu-sed) and re-defining them temporarily aspatchandsed.
Note
If you encounter issues with the requirements, it is useful to check the exact list in the Dockerfile which is tested in the CI.
You can install the Python dependencies from the requirement file at the root of Pyodide folder:
pip install -r requirements.txt
After installing the build prerequisites, run from the command line:
make
Building a full Pyodide distribution#
By running make, you build a minimal Pyodide distribution that includes only
the core packages. Pyodide contains a large number of packages, and building
all of them takes a long time. If you want to build a full distribution, or make a custom
distribution with a subset of packages, there are two options:
Use pre-built packages from the
pyodide-recipesrepository.
Packages included in the Pyodide distribution are built separately and stored in
the pyodide-recipes repository.
Setting ENABLE_PREBUILT_PACKAGES environment variable to 1 will download
and install pre-built packages from the pyodide-recipes repository.
ENABLE_PREBUILT_PACKAGES=1 make
Build a subset of packages from the
pyodide-recipesrepository.
If you want to build a subset of packages, you can clone the
pyodide-recipes repository and build the packages you need.
git clone https://github.com/pyodide/pyodide-recipes
# Build the packages you need
pyodide build-recipes "numpy, scipy, toolz, attrs" --recipe-dir pyodide-recipes/packages --install
Dependencies of the listed packages will be built automatically as well. The
package names must match the folder names in pyodide-recipes/packages exactly; in particular
they are case-sensitive.
There are also some meta-packages that can be used to build a subset of packages:
“tag:min-scipy-stack”: includes the “core” meta-package as well as some core packages from the Scientific Python stack and their dependencies: “numpy”, “scipy”, “pandas”, “matplotlib”, “scikit-learn”, “joblib”, “pytest”. This option is non exhaustive and is mainly intended to make build faster while testing a diverse set of scientific packages.
“tag:library”: includes all shared and static libraries.
“tag:shared_library”: includes all shared libraries. Added in conjunction with the “library” tag.
“tag:static_library”: includes all static libraries. Added in conjunction with the “library” tag.
“tag:pyodide-test”: includes all packages built to test the Pyodide build system’s behaviour.
“tag:rust”: includes all packages that require a Rust toolchain to build, and their dependencies.
“*” builds all packages
You can exclude a package by prefixing it with “!”.
micropip is always automatically included.
Environment variables#
The following environment variables additionally impact the build:
PYODIDE_JOBS: the-joption passed to theemmake makecommand when applicable for parallel compilation. Default: 3.PYODIDE_BASE_URL: Base URL where Pyodide packages are deployed. It must end with a trailing/. Default:./to load Pyodide packages from the same base URL path as wherepyodide.jsis located. Example:https://cdn.jsdelivr.net/pyodide/v0.29.0/full/EXTRA_CFLAGS: Add extra compilation flags.EXTRA_LDFLAGS: Add extra linker flags.
Setting EXTRA_CFLAGS="-D DEBUG_F" provides detailed diagnostic information
whenever error branches are taken inside the Pyodide core code. These error
messages are frequently helpful even when the problem is a fatal configuration
problem and Pyodide cannot even be initialized. These error branches occur also
in correctly working code, but they are relatively uncommon so in practice the
amount of noise generated isn’t too large. The shorthand make debug
automatically sets this flag.
In certain cases, setting EXTRA_LDFLAGS="-s ASSERTIONS=1 or ASSERTIONS=2 can
also be helpful, but this slows down the linking and the runtime speed of
Pyodide a lot and generates a large amount of noise in the console.