Compiling Python from Source Code
Finding Version-Specific Source Code
The source code for all of the released Python versions can be found on the python source releases web-page on the python.org website. For example, if one wants use Python 3.9, follow the aforelinked web-page and select the latest 3.9.x version. Navigate to the section containing the files and look for the XZ compressed source tarball.
Preliminary Setup
Throughout this guide it is assumed that the Python executables will be
stored in ~/dev/pyexes
. Either create these directories now:
cd ~
mkdir dev
mkdir dev/pyexes
Or, choose an alternative directory and modify the subsequent commands accordingly.
Note, in order to avoid system issues, the executables stored in this directory should never be installed to the system (i.e. added to the system path) unless one understands the intended (and unintended) effects of doing so.
Download and Extract the Source Code
Create a directory for the specific Python version to be installed:
mkdir dev/pyexes/py39
Create a temporary directory to store all the files that are downloaded and created during the configuration and build process, and then change into it:
mkdir dev/pyexes/py39/tempfiles
cd dev/pyexes/py39/tempfiles
Download the source code for the particular Python version:
curl https://www.python.org/ftp/python/3.9.16/Python-3.9.16.tar.xz --output Python-3.9.16.tar.xz
Extract the .tar.xz
file:
tar -xJvf Python-3.9.16.tar.xz
Configure
Configure the build process by running configure
:
Python-3.9.16/configure --prefix=/home/<username>/dev/pyexes/py39 --with-pydebug --enable-optimizations --with-lto
More information on the configure
flags for the build process
can be found on the
configure python
documentation web-page on the
python.org website. Alternatively, one
can run Python-3.9.16/configure --help
).
Of particular import is the --prefix
option which lets one set
the location where the build will be installed. If one doesn't set this, the
default install location will be in /usr/local
which will
conflict with any system Python versions already installed. The path set
here must be an absolute directory name (i.e. instead of ~
one
must specify /home/<username>
).
The --enable-optimizations
and --with-lto
flags
enables Profile Guide Optimization (PGO) and Link Time Optimization (LTO)
which is recommended for best performance (note that these features were
introduced in Python version 3.6, so these flags will not work if compiling
a Python version earlier than 3.6.
Debug Builds
Setting the --with-pydebug
flag creates a "pydebug" build. It
turns on various extra sanity checks which help catch common issues. It is
recommended to always develop under a debug build of CPython (unless one is
taking performance measurements). However, this may cause compatibility
issues with some Python package wheels if using Python versions prior to
3.8. It is recommended to read the
Python Builds and Compatible Package Wheels
article to understand how the configuration and build process may influence
which wheels can be used to install packages.
Build
Build the Python executable:
make -j32
Change the number of cores passed to the -j
flag to match those
available on the system. Alternatively, if the version of Make being used
supports it, use -j
with the number omitted and Make will not
limit the number of simultaneous steps.
The build process (compilation) will take anywhere from 2-10 minutes
depending on the computer's hardware. Once it has completed, scroll up and
there should be a line stating that the
Python build finished successfully!
, below this exclamation, if
any optional modules were not able to be added they will be listed (e.g.
_bz2
, _hashlib
etc.). The reason these modules
failed to be added is because they rely on certain development headers being
supplied by the associated libraries (e.g. the zlib
library for
the compression module). If one requires the functionality offered by the
optional modules that were not added, then install the associated
development libraries for each module using the system's package manager. A
web-search will have to be carried out to find the names of the exact
packages to be installed. More information can be found under the
build dependencies
section of the
setup and building
web-page on the
python.org website.
If one doesn't need to install any additional packages to attain some of the functionality offered by the optional modules that were not added, skip to the Install section.
Once the packages are installed, rerun the configure
command,
followed by make
:
Python-3.9.16/configure --prefix=/home/<username>/dev/pyexes/py39 --with-pydebug --enable-optimizations --with-lto
make -j32
Install
After running make
, a working build will be created in the
directory from which the command was run. This means that Python can be run
from this directory and the interpreter will realise where it is being run
from and use the files found in this working directory. Hence, there is
technically no need to install
the built copy of Python.
However, to keep things tidy, one can install (i.e. copy) the necessary
files from the working build to the location specified by the
--prefix
flag earlier:
make install
Note that if the --prefix
flag wasn't specified earlier when
setting up the build configuration via the configure
command,
then one will have to run the above command with elevated privileges (e.g.
sudo make install
) as it will install to
/usr/local
by default which is a location that requires
elevated privileges to write to.
After installing, there may be a warning that the pip3
and
pip3.9
scripts were installed to
~/dev/pyexes/py39/bin
which is not on the system path. This can
be ignored if one does not intend to use this Python version for
system-related actions.
Python has now been successfully installed. All that remains is to remove leftover files.
Cleaning Temporary Files
Navigate up and out of the tempfiles
directory:
cd ..
Remove the tempfiles
directory which contains all the files
that are no longer needed (i.e. source code, configuration files and build
files):
rm -rf tempfiles
Now only four directories should remain in the
--prefix
location specified earlier: bin
,
include
, lib
and share
. The most
important one is the python3.9
executable located in the
bin
directory.
Test the Created Executable
Test that the Python executable was created successfully by printing its version:
bin/python3.9 --version # Python 3.9.16
Using Different Python Versions
To use different Python versions at any time, simply call the
version-specific Python executable located in the respective
bin
folder. For example:
~/dev/pyexes/py39/bin/python3.9
However, if one has multiple Python versions, the use of virtual environments is recommended. See the Python Virtual Environments guide for more information.