2016-01-21 01:23:03 -05:00
|
|
|
PEP: 513
|
|
|
|
Title: A Platform Tag for Portable Linux Built Distributions
|
|
|
|
Version: $Revision$
|
|
|
|
Last-Modified: $Date$
|
|
|
|
Author: Robert T. McGibbon <rmcgibbo@gmail.com>, Nathaniel J. Smith <njs@pobox.com>
|
|
|
|
BDFL-Delegate: Nick Coghlan <ncoghlan@gmail.com>
|
2016-01-21 01:46:49 -05:00
|
|
|
Discussions-To: Distutils SIG <distutils-sig@python.org>
|
2021-06-26 12:11:00 -04:00
|
|
|
Status: Superseded
|
2016-01-21 01:23:03 -05:00
|
|
|
Type: Informational
|
|
|
|
Content-Type: text/x-rst
|
|
|
|
Created: 19-Jan-2016
|
2016-01-30 02:10:13 -05:00
|
|
|
Post-History: 19-Jan-2016, 25-Jan-2016, 29-Jan-2016
|
2021-06-26 12:11:00 -04:00
|
|
|
Superseded-By: 600
|
2016-01-30 03:59:52 -05:00
|
|
|
Resolution: https://mail.python.org/pipermail/distutils-sig/2016-January/028211.html
|
2016-01-21 01:23:03 -05:00
|
|
|
|
|
|
|
|
|
|
|
Abstract
|
|
|
|
========
|
|
|
|
|
|
|
|
This PEP proposes the creation of a new platform tag for Python package built
|
2016-01-30 15:51:23 -05:00
|
|
|
distributions, such as wheels, called ``manylinux1_{x86_64,i686}`` with
|
2016-01-27 06:37:05 -05:00
|
|
|
external dependencies limited to a standardized, restricted subset of
|
2016-01-21 01:23:03 -05:00
|
|
|
the Linux kernel and core userspace ABI. It proposes that PyPI support
|
2016-01-27 06:37:05 -05:00
|
|
|
uploading and distributing wheels with this platform tag, and that ``pip``
|
2016-01-21 01:23:03 -05:00
|
|
|
support downloading and installing these packages on compatible platforms.
|
|
|
|
|
|
|
|
|
|
|
|
Rationale
|
|
|
|
=========
|
|
|
|
|
|
|
|
Currently, distribution of binary Python extensions for Windows and OS X is
|
2022-01-21 06:03:51 -05:00
|
|
|
straightforward. Developers and packagers build wheels (:pep:`427`, :pep:`491`),
|
|
|
|
which are
|
2016-01-27 06:37:05 -05:00
|
|
|
assigned platform tags such as ``win32`` or ``macosx_10_6_intel``, and upload
|
|
|
|
these wheels to PyPI. Users can download and install these wheels using tools
|
|
|
|
such as ``pip``.
|
2016-01-21 01:23:03 -05:00
|
|
|
|
|
|
|
For Linux, the situation is much more delicate. In general, compiled Python
|
|
|
|
extension modules built on one Linux distribution will not work on other Linux
|
2016-01-27 06:37:05 -05:00
|
|
|
distributions, or even on different machines running the same Linux
|
|
|
|
distribution with different system libraries installed.
|
2016-01-21 01:23:03 -05:00
|
|
|
|
2022-01-21 06:03:51 -05:00
|
|
|
Build tools using :pep:`425` platform tags do not track information about the
|
2016-01-21 01:23:03 -05:00
|
|
|
particular Linux distribution or installed system libraries, and instead assign
|
2016-01-30 15:51:23 -05:00
|
|
|
all wheels the too-vague ``linux_i686`` or ``linux_x86_64`` tags. Because of
|
2016-01-21 01:23:03 -05:00
|
|
|
this ambiguity, there is no expectation that ``linux``-tagged built
|
|
|
|
distributions compiled on one machine will work properly on another, and for
|
|
|
|
this reason, PyPI has not permitted the uploading of wheels for Linux.
|
|
|
|
|
|
|
|
It would be ideal if wheel packages could be compiled that would work on *any*
|
|
|
|
linux system. But, because of the incredible diversity of Linux systems -- from
|
|
|
|
PCs to Android to embedded systems with custom libcs -- this cannot
|
|
|
|
be guaranteed in general.
|
|
|
|
|
|
|
|
Instead, we define a standard subset of the kernel+core userspace ABI that,
|
|
|
|
in practice, is compatible enough that packages conforming to this standard
|
|
|
|
will work on *many* linux systems, including essentially all of the desktop
|
|
|
|
and server distributions in common use. We know this because there are
|
|
|
|
companies who have been distributing such widely-portable pre-compiled Python
|
2016-01-27 06:37:05 -05:00
|
|
|
extension modules for Linux -- e.g. Enthought with Canopy [4]_ and Continuum
|
|
|
|
Analytics with Anaconda [5]_.
|
2016-01-21 01:23:03 -05:00
|
|
|
|
2016-07-11 11:14:08 -04:00
|
|
|
Building on the compatibility lessons learned from these companies, we thus
|
2016-01-21 01:23:03 -05:00
|
|
|
define a baseline ``manylinux1`` platform tag for use by binary Python
|
|
|
|
wheels, and introduce the implementation of preliminary tools to aid in the
|
|
|
|
construction of these ``manylinux1`` wheels.
|
|
|
|
|
|
|
|
|
|
|
|
Key Causes of Inter-Linux Binary Incompatibility
|
|
|
|
================================================
|
|
|
|
|
|
|
|
To properly define a standard that will guarantee that wheel packages meeting
|
|
|
|
this specification will operate on *many* linux platforms, it is necessary to
|
|
|
|
understand the root causes which often prevent portability of pre-compiled
|
|
|
|
binaries on Linux. The two key causes are dependencies on shared libraries
|
|
|
|
which are not present on users' systems, and dependencies on particular
|
|
|
|
versions of certain core libraries like ``glibc``.
|
|
|
|
|
|
|
|
|
|
|
|
External Shared Libraries
|
|
|
|
-------------------------
|
|
|
|
|
|
|
|
Most desktop and server linux distributions come with a system package manager
|
|
|
|
(examples include ``APT`` on Debian-based systems, ``yum`` on
|
|
|
|
``RPM``-based systems, and ``pacman`` on Arch linux) that manages, among other
|
|
|
|
responsibilities, the installation of shared libraries installed to system
|
|
|
|
directories such as ``/usr/lib``. Most non-trivial Python extensions will depend
|
|
|
|
on one or more of these shared libraries, and thus function properly only on
|
|
|
|
systems where the user has the proper libraries (and the proper
|
|
|
|
versions thereof), either installed using their package manager, or installed
|
|
|
|
manually by setting certain environment variables such as ``LD_LIBRARY_PATH``
|
|
|
|
to notify the runtime linker of the location of the depended-upon shared
|
|
|
|
libraries.
|
|
|
|
|
|
|
|
|
|
|
|
Versioning of Core Shared Libraries
|
|
|
|
-----------------------------------
|
|
|
|
|
2016-01-27 06:37:05 -05:00
|
|
|
Even if the developers a Python extension module wish to use no
|
2016-01-21 01:23:03 -05:00
|
|
|
external shared libraries, the modules will generally have a dynamic runtime
|
|
|
|
dependency on the GNU C library, ``glibc``. While it is possible, statically
|
2016-01-27 06:37:05 -05:00
|
|
|
linking ``glibc`` is usually a bad idea because certain important C functions
|
|
|
|
like ``dlopen()`` cannot be called from code that statically links ``glibc``. A
|
|
|
|
runtime shared library dependency on a system-provided ``glibc`` is unavoidable
|
|
|
|
in practice.
|
2016-01-21 01:23:03 -05:00
|
|
|
|
|
|
|
The maintainers of the GNU C library follow a strict symbol versioning scheme
|
|
|
|
for backward compatibility. This ensures that binaries compiled against an older
|
|
|
|
version of ``glibc`` can run on systems that have a newer ``glibc``. The
|
|
|
|
opposite is generally not true -- binaries compiled on newer Linux
|
2016-01-27 06:37:05 -05:00
|
|
|
distributions tend to rely upon versioned functions in ``glibc`` that are not
|
2016-01-21 01:23:03 -05:00
|
|
|
available on older systems.
|
|
|
|
|
2016-01-27 06:37:05 -05:00
|
|
|
This generally prevents wheels compiled on the latest Linux distributions
|
|
|
|
from being portable.
|
2016-01-21 01:23:03 -05:00
|
|
|
|
|
|
|
|
|
|
|
The ``manylinux1`` policy
|
|
|
|
=========================
|
|
|
|
|
|
|
|
For these reasons, to achieve broad portability, Python wheels
|
|
|
|
|
2016-01-27 06:37:05 -05:00
|
|
|
* should depend only on an extremely limited set of external shared
|
|
|
|
libraries; and
|
|
|
|
* should depend only on "old" symbol versions in those external shared
|
|
|
|
libraries; and
|
|
|
|
* should depend only on a widely-compatible kernel ABI.
|
2016-01-21 01:23:03 -05:00
|
|
|
|
2016-01-27 06:37:05 -05:00
|
|
|
To be eligible for the ``manylinux1`` platform tag, a Python wheel must
|
|
|
|
therefore both (a) contain binary executables and compiled code that links
|
2016-02-10 00:54:46 -05:00
|
|
|
*only* to libraries with SONAMEs
|
2016-01-27 06:37:05 -05:00
|
|
|
included in the following list: ::
|
2016-01-21 01:23:03 -05:00
|
|
|
|
|
|
|
libpanelw.so.5
|
|
|
|
libncursesw.so.5
|
|
|
|
libgcc_s.so.1
|
|
|
|
libstdc++.so.6
|
|
|
|
libm.so.6
|
|
|
|
libdl.so.2
|
|
|
|
librt.so.1
|
|
|
|
libc.so.6
|
|
|
|
libnsl.so.1
|
|
|
|
libutil.so.1
|
|
|
|
libpthread.so.0
|
2017-08-28 15:16:11 -04:00
|
|
|
libresolv.so.2
|
2016-01-21 01:23:03 -05:00
|
|
|
libX11.so.6
|
|
|
|
libXext.so.6
|
|
|
|
libXrender.so.1
|
|
|
|
libICE.so.6
|
|
|
|
libSM.so.6
|
|
|
|
libGL.so.1
|
|
|
|
libgobject-2.0.so.0
|
|
|
|
libgthread-2.0.so.0
|
|
|
|
libglib-2.0.so.0
|
|
|
|
|
2016-01-30 03:45:54 -05:00
|
|
|
and, (b) work on a stock CentOS 5.11 [6]_ system that contains the system
|
2016-02-10 00:54:46 -05:00
|
|
|
package manager's provided versions of these libraries.
|
2016-01-27 06:37:05 -05:00
|
|
|
|
2019-07-18 18:29:30 -04:00
|
|
|
``libcrypt.so.1`` was retrospectively removed from the whitelist after
|
|
|
|
Fedora 30 was released with ``libcrypt.so.2`` instead.
|
|
|
|
|
2016-01-30 15:51:23 -05:00
|
|
|
Because CentOS 5 is only available for x86_64 and i686 architectures,
|
2016-01-27 06:37:05 -05:00
|
|
|
these are the only architectures currently supported by the ``manylinux1``
|
|
|
|
policy.
|
|
|
|
|
2016-01-21 01:23:03 -05:00
|
|
|
On Debian-based systems, these libraries are provided by the packages ::
|
|
|
|
|
|
|
|
libncurses5 libgcc1 libstdc++6 libc6 libx11-6 libxext6
|
|
|
|
libxrender1 libice6 libsm6 libgl1-mesa-glx libglib2.0-0
|
|
|
|
|
|
|
|
On RPM-based systems, these libraries are provided by the packages ::
|
|
|
|
|
|
|
|
ncurses libgcc libstdc++ glibc libXext libXrender
|
|
|
|
libICE libSM mesa-libGL glib2
|
|
|
|
|
|
|
|
This list was compiled by checking the external shared library dependencies of
|
2016-01-27 06:37:05 -05:00
|
|
|
the Canopy [4]_ and Anaconda [5]_ distributions, which both include a wide array
|
2016-01-21 01:23:03 -05:00
|
|
|
of the most popular Python modules and have been confirmed in practice to work
|
|
|
|
across a wide swath of Linux systems in the wild.
|
|
|
|
|
2016-01-27 06:37:05 -05:00
|
|
|
Many of the permitted system libraries listed above use symbol versioning
|
|
|
|
schemes for backward compatibility. The latest symbol versions provided with
|
|
|
|
the CentOS 5.11 versions of these libraries are: ::
|
|
|
|
|
|
|
|
GLIBC_2.5
|
|
|
|
CXXABI_3.4.8
|
|
|
|
GLIBCXX_3.4.9
|
|
|
|
GCC_4.2.0
|
|
|
|
|
|
|
|
Therefore, as a consequence of requirement (b), any wheel that depends on
|
|
|
|
versioned symbols from the above shared libraries may depend only on symbols
|
|
|
|
with the following versions: ::
|
2016-01-21 01:23:03 -05:00
|
|
|
|
|
|
|
GLIBC <= 2.5
|
|
|
|
CXXABI <= 3.4.8
|
|
|
|
GLIBCXX <= 3.4.9
|
|
|
|
GCC <= 4.2.0
|
|
|
|
|
2016-01-27 06:37:05 -05:00
|
|
|
These recommendations are the outcome of the relevant discussions in January
|
|
|
|
2016 [7]_, [8]_.
|
|
|
|
|
|
|
|
Note that in our recommendations below, we do not suggest that ``pip``
|
|
|
|
or PyPI should attempt to check for and enforce the details of this
|
|
|
|
policy (just as they don't check for and enforce the details of
|
|
|
|
existing platform tags like ``win32``). The text above is provided (a)
|
|
|
|
as advice to package builders, and (b) as a method for allocating
|
|
|
|
blame if a given wheel doesn't work on some system: if it satisfies
|
|
|
|
the policy above, then this is a bug in the spec or the installation
|
|
|
|
tool; if it does not satisfy the policy above, then it's a bug in the
|
|
|
|
wheel. One useful consequence of this approach is that it leaves open
|
|
|
|
the possibility of further updates and tweaks as we gain more
|
|
|
|
experience, e.g., we could have a "manylinux 1.1" policy which targets
|
|
|
|
the same systems and uses the same ``manylinux1`` platform tag (and
|
|
|
|
thus requires no further changes to ``pip`` or PyPI), but that adjusts
|
|
|
|
the list above to remove libraries that have turned out to be
|
|
|
|
problematic or add libraries that have turned out to be safe.
|
|
|
|
|
|
|
|
|
2016-02-10 00:54:46 -05:00
|
|
|
libpythonX.Y.so.1
|
|
|
|
-----------------
|
|
|
|
|
|
|
|
Note that ``libpythonX.Y.so.1`` is *not* on the list of libraries that
|
|
|
|
a ``manylinux1`` extension is allowed to link to. Explicitly linking
|
|
|
|
to ``libpythonX.Y.so.1`` is unnecessary in almost all cases: the way
|
|
|
|
ELF linking works, extension modules that are loaded into the
|
|
|
|
interpreter automatically get access to all of the interpreter's
|
|
|
|
symbols, regardless of whether or not the extension itself is
|
|
|
|
explicitly linked against libpython. Furthermore, explicit linking to
|
|
|
|
libpython creates problems in the common configuration where Python is
|
|
|
|
not built with ``--enable-shared``. In particular, on Debian and
|
|
|
|
Ubuntu systems, ``apt install pythonX.Y`` does not even install
|
|
|
|
``libpythonX.Y.so.1``, meaning that any wheel that *did* depend on
|
|
|
|
``libpythonX.Y.so.1`` could fail to import.
|
|
|
|
|
|
|
|
There is one situation where extensions that are linked in this way
|
|
|
|
can fail to work: if a host program (e.g., ``apache2``) uses
|
|
|
|
``dlopen()`` to load a module (e.g., ``mod_wsgi``) that embeds the
|
|
|
|
CPython interpreter, and the host program does *not* pass the
|
|
|
|
``RTLD_GLOBAL`` flag to ``dlopen()``, then the embedded CPython will
|
|
|
|
be unable to load any extension modules that do not themselves link
|
|
|
|
explicitly to ``libpythonX.Y.so.1``. Fortunately, ``apache2`` *does*
|
|
|
|
set the ``RTLD_GLOBAL`` flag, as do all the other programs that
|
|
|
|
embed-CPython-via-a-dlopened-plugin that we could locate, so this does
|
|
|
|
not seem to be a serious problem in practice. The incompatibility with
|
|
|
|
Debian/Ubuntu is more of an issue than the theoretical incompatibility
|
|
|
|
with a rather obscure corner case.
|
|
|
|
|
|
|
|
This is a rather complex and subtle issue that extends beyond
|
|
|
|
the scope of ``manylinux1``; for more discussion see: [9]_, [10]_,
|
|
|
|
[11]_.
|
|
|
|
|
2016-12-25 14:19:18 -05:00
|
|
|
|
2016-02-10 00:54:46 -05:00
|
|
|
UCS-2 vs UCS-4 builds
|
|
|
|
---------------------
|
|
|
|
|
|
|
|
All versions of CPython 2.x, plus CPython 3.0-3.2 inclusive, can be
|
|
|
|
built in two ABI-incompatible modes: builds using the
|
|
|
|
``--enable-unicode=ucs2`` configure flag store Unicode data in UCS-2
|
|
|
|
(or really UTF-16) format, while builds using the
|
|
|
|
``--enable-unicode=ucs4`` configure flag store Unicode data in
|
|
|
|
UCS-4. (CPython 3.3 and greater use a different storage method that
|
|
|
|
always supports UCS-4.) If we want to make sure ``ucs2`` wheels don't
|
|
|
|
get installed into ``ucs4`` CPythons and vice-versa, then something
|
|
|
|
must be done.
|
|
|
|
|
|
|
|
An earlier version of this PEP included a requirement that
|
|
|
|
``manylinux1`` wheels targeting these older CPython versions should
|
|
|
|
always use the ``ucs4`` ABI. But then, in between the PEP's initial
|
|
|
|
acceptance and its implementation, ``pip`` and ``wheel`` gained
|
|
|
|
first-class support for tracking and checking this aspect of ABI
|
|
|
|
compatibility for the relevant CPython versions, which is a better
|
|
|
|
solution. So we now allow the ``manylinux1`` platform tags to be used
|
|
|
|
in combination with any ABI tag. However, to maintain compatibility it
|
|
|
|
is crucial to ensure that all ``manylinux1`` wheels include a
|
|
|
|
non-trivial abi tag. For example, a wheel built against a ``ucs4``
|
|
|
|
CPython might have a name like::
|
|
|
|
|
|
|
|
PKG-VERSION-cp27-cp27mu-manylinux1_x86_64.whl
|
|
|
|
^^^^^^ Good!
|
|
|
|
|
|
|
|
While a wheel built against the ``ucs2`` ABI might have a name like::
|
|
|
|
|
|
|
|
PKG-VERSION-cp27-cp27m-manylinux1_x86_64.whl
|
|
|
|
^^^^^ Okay!
|
|
|
|
|
|
|
|
But you should never have a wheel with a name like::
|
|
|
|
|
|
|
|
PKG-VERSION-cp27-none-manylinux1_x86_64.whl
|
|
|
|
^^^^ BAD! Don't do this!
|
|
|
|
|
2016-12-25 14:19:18 -05:00
|
|
|
This wheel claims to be simultaneously compatible with *both* ucs2 and
|
|
|
|
ucs4 builds, which is bad.
|
|
|
|
|
2016-02-10 00:54:46 -05:00
|
|
|
We note for information that the ``ucs4`` ABI appears to be much more
|
|
|
|
widespread among Linux CPython distributors.
|
|
|
|
|
|
|
|
|
2016-12-25 14:19:18 -05:00
|
|
|
fpectl builds vs. no fpectl builds
|
|
|
|
----------------------------------
|
|
|
|
|
|
|
|
All extant versions of CPython can be built either with or without the
|
|
|
|
``--with-fpectl`` flag to ``configure``. It turns out that this
|
|
|
|
changes the CPython ABI: extensions that are built against a
|
|
|
|
no-``fpectl`` CPython are always compatible with yes-``fpectl``
|
|
|
|
CPython, but the reverse is not necessarily true. (Symptom: errors at
|
|
|
|
import time complaining about ``undefined symbol: PyFPE_jbuf``.) See:
|
|
|
|
[16]_.
|
|
|
|
|
|
|
|
For maximum compatibility, therefore, the CPython used to build
|
|
|
|
manylinux1 wheels must be compiled *without* the ``--with-fpectl``
|
|
|
|
flag, and manylinux1 extensions must not reference the symbol
|
|
|
|
``PyFPE_jbuf``.
|
|
|
|
|
|
|
|
|
2016-01-27 06:37:05 -05:00
|
|
|
Compilation of Compliant Wheels
|
|
|
|
===============================
|
|
|
|
|
|
|
|
The way glibc, libgcc, and libstdc++ manage their symbol versioning
|
|
|
|
means that in practice, the compiler toolchains that most developers
|
|
|
|
use to do their daily work are incapable of building
|
2016-07-11 11:14:08 -04:00
|
|
|
``manylinux1``-compliant wheels. Therefore, we do not attempt to change
|
2016-01-27 06:37:05 -05:00
|
|
|
the default behavior of ``pip wheel`` / ``bdist_wheel``: they will
|
|
|
|
continue to generate regular ``linux_*`` platform tags, and developers
|
|
|
|
who wish to use them to generate ``manylinux1``-tagged wheels will
|
|
|
|
have to change the tag as a second post-processing step.
|
2016-01-21 01:23:03 -05:00
|
|
|
|
|
|
|
To support the compilation of wheels meeting the ``manylinux1`` standard, we
|
|
|
|
provide initial drafts of two tools.
|
|
|
|
|
2016-01-27 06:37:05 -05:00
|
|
|
|
|
|
|
Docker Image
|
|
|
|
------------
|
|
|
|
|
|
|
|
The first tool is a Docker image based on CentOS 5.11, which is recommended as
|
|
|
|
an easy to use self-contained build box for compiling ``manylinux1`` wheels
|
2016-02-10 00:54:46 -05:00
|
|
|
[12]_. Compiling on a more recently-released linux distribution will generally
|
2016-01-21 01:23:03 -05:00
|
|
|
introduce dependencies on too-new versioned symbols. The image comes with a
|
|
|
|
full compiler suite installed (``gcc``, ``g++``, and ``gfortran`` 4.8.2) as
|
2016-01-27 06:37:05 -05:00
|
|
|
well as the latest releases of Python and ``pip``.
|
|
|
|
|
|
|
|
Auditwheel
|
|
|
|
----------
|
|
|
|
|
2016-02-10 00:54:46 -05:00
|
|
|
The second tool is a command line executable called ``auditwheel`` [13]_ that
|
2016-01-27 06:37:05 -05:00
|
|
|
may aid in package maintainers in dealing with third-party external
|
|
|
|
dependencies.
|
|
|
|
|
|
|
|
There are at least three methods for building wheels that use third-party
|
|
|
|
external libraries in a way that meets the above policy.
|
|
|
|
|
|
|
|
1. The third-party libraries can be statically linked.
|
|
|
|
2. The third-party shared libraries can be distributed in
|
|
|
|
separate packages on PyPI which are depended upon by the wheel.
|
|
|
|
3. The third-party shared libraries can be bundled inside the wheel
|
|
|
|
libraries, linked with a relative path.
|
|
|
|
|
|
|
|
All of these are valid option which may be effectively used by different
|
|
|
|
packages and communities. Statically linking generally requires
|
|
|
|
package-specific modifications to the build system, and distributing
|
|
|
|
third-party dependencies on PyPI may require some coordination of the
|
|
|
|
community of users of the package.
|
|
|
|
|
|
|
|
As an often-automatic alternative to these options, we introduce ``auditwheel``.
|
|
|
|
The tool inspects all of the ELF files inside a wheel to check for
|
|
|
|
dependencies on versioned symbols or external shared libraries, and verifies
|
|
|
|
conformance with the ``manylinux1`` policy. This includes the ability to add
|
|
|
|
the new platform tag to conforming wheels. More importantly, ``auditwheel`` has
|
|
|
|
the ability to automatically modify wheels that depend on external shared
|
|
|
|
libraries by copying those shared libraries from the system into the wheel
|
|
|
|
itself, and modifying the appropriate ``RPATH`` entries such that these
|
|
|
|
libraries will be picked up at runtime. This accomplishes a similar result as
|
|
|
|
if the libraries had been statically linked without requiring changes to the
|
|
|
|
build system. Packagers are advised that bundling, like static linking, may
|
|
|
|
implicate copyright concerns.
|
|
|
|
|
|
|
|
|
|
|
|
Bundled Wheels on Linux
|
|
|
|
=======================
|
2016-01-21 01:23:03 -05:00
|
|
|
|
2016-01-27 06:37:05 -05:00
|
|
|
While we acknowledge many approaches for dealing with third-party library
|
|
|
|
dependencies within ``manylinux1`` wheels, we recognize that the ``manylinux1``
|
|
|
|
policy encourages bundling external dependencies, a practice
|
|
|
|
which runs counter to the package management policies of many linux
|
2016-02-10 00:54:46 -05:00
|
|
|
distributions' system package managers [14]_, [15]_. The primary purpose of
|
2016-01-27 06:37:05 -05:00
|
|
|
this is cross-distro compatibility. Furthermore, ``manylinux1`` wheels on PyPI
|
|
|
|
occupy a different niche than the Python packages available through the
|
|
|
|
system package manager.
|
|
|
|
|
|
|
|
The decision in this PEP to encourage departure from general Linux distribution
|
|
|
|
unbundling policies is informed by the following concerns:
|
|
|
|
|
|
|
|
1. In these days of automated continuous integration and deployment
|
|
|
|
pipelines, publishing new versions and updating dependencies is easier
|
|
|
|
than it was when those policies were defined.
|
|
|
|
2. ``pip`` users remain free to use the ``"--no-binary"`` option if they want
|
|
|
|
to force local builds rather than using pre-built wheel files.
|
|
|
|
3. The popularity of modern container based deployment and "immutable
|
|
|
|
infrastructure" models involve substantial bundling at the application
|
|
|
|
layer anyway.
|
|
|
|
4. Distribution of bundled wheels through PyPI is currently the norm for
|
|
|
|
Windows and OS X.
|
|
|
|
5. This PEP doesn't rule out the idea of offering more targeted binaries for
|
|
|
|
particular Linux distributions in the future.
|
|
|
|
|
|
|
|
The model described in this PEP is most ideally suited for cross-platform
|
|
|
|
Python packages, because it means they can reuse much of the
|
|
|
|
work that they're already doing to make static Windows and OS X wheels. We
|
|
|
|
recognize that it is less optimal for Linux-specific packages that might
|
|
|
|
prefer to interact more closely with Linux's unique package management
|
|
|
|
functionality and only care about targeting a small set of particular distos.
|
2016-01-21 01:23:03 -05:00
|
|
|
|
|
|
|
|
2016-01-27 06:37:05 -05:00
|
|
|
Security Implications
|
|
|
|
---------------------
|
|
|
|
|
|
|
|
One of the advantages of dependencies on centralized libraries in Linux is
|
|
|
|
that bugfixes and security updates can be deployed system-wide, and
|
|
|
|
applications which depend on these libraries will automatically feel the
|
|
|
|
effects of these patches when the underlying libraries are updated. This can
|
|
|
|
be particularly important for security updates in packages engaged in
|
|
|
|
communication across the network or cryptography.
|
|
|
|
|
|
|
|
``manylinux1`` wheels distributed through PyPI that bundle security-critical
|
|
|
|
libraries like OpenSSL will thus assume responsibility for prompt updates in
|
|
|
|
response disclosed vulnerabilities and patches. This closely parallels the
|
|
|
|
security implications of the distribution of binary wheels on Windows that,
|
|
|
|
because the platform lacks a system package manager, generally bundle their
|
|
|
|
dependencies. In particular, because it lacks a stable ABI, OpenSSL cannot be
|
|
|
|
included in the ``manylinux1`` profile.
|
|
|
|
|
2016-01-21 01:23:03 -05:00
|
|
|
|
|
|
|
|
|
|
|
Platform Detection for Installers
|
|
|
|
=================================
|
|
|
|
|
2016-01-27 06:37:05 -05:00
|
|
|
Above, we defined what it means for a *wheel* to be
|
|
|
|
``manylinux1``-compatible. Here we discuss what it means for a *Python
|
|
|
|
installation* to be ``manylinux1``-compatible. In particular, this is
|
|
|
|
important for tools like ``pip`` to know when deciding whether or not
|
|
|
|
they should consider ``manylinux1``-tagged wheels for installation.
|
|
|
|
|
|
|
|
Because the ``manylinux1`` profile is already known to work for the
|
|
|
|
many thousands of users of popular commercial Python distributions, we
|
|
|
|
suggest that installation tools should error on the side of assuming
|
|
|
|
that a system *is* compatible, unless there is specific reason to
|
|
|
|
think otherwise.
|
2016-01-21 01:23:03 -05:00
|
|
|
|
2016-01-30 03:45:54 -05:00
|
|
|
We know of four main sources of potential incompatibility that are
|
|
|
|
likely to arise in practice:
|
2016-01-21 01:23:03 -05:00
|
|
|
|
2016-01-30 02:10:13 -05:00
|
|
|
* Eventually, in the future, there may exist distributions that break
|
|
|
|
compatibility with this profile (e.g., if one of the libraries in
|
|
|
|
the profile changes its ABI in a backwards-incompatible way)
|
2016-01-21 01:23:03 -05:00
|
|
|
* A linux distribution that is too old (e.g. RHEL 4)
|
2016-01-27 06:37:05 -05:00
|
|
|
* A linux distribution that does not use ``glibc`` (e.g. Alpine Linux, which is
|
|
|
|
based on musl ``libc``, or Android)
|
2016-01-21 01:23:03 -05:00
|
|
|
|
2016-02-10 00:54:46 -05:00
|
|
|
To address these we propose a two-pronged
|
2016-01-30 03:45:54 -05:00
|
|
|
approach. To handle potential future incompatibilities, we standardize
|
|
|
|
a mechanism for a Python distributor to signal that a particular
|
|
|
|
Python install definitely is or is not compatible with ``manylinux1``:
|
|
|
|
this is done by installing a module named ``_manylinux``, and setting
|
|
|
|
its ``manylinux1_compatible`` attribute. We do not propose adding any
|
|
|
|
such module to the standard library -- this is merely a well-known
|
|
|
|
name by which distributors and installation tools can
|
|
|
|
rendezvous. However, if a distributor does add this module, *they
|
|
|
|
should add it to the standard library* rather than to a
|
|
|
|
``site-packages/`` directory, because the standard library is
|
2016-01-30 02:10:13 -05:00
|
|
|
inherited by virtualenvs (which we want), and ``site-packages/`` in
|
|
|
|
general is not.
|
|
|
|
|
2016-01-30 03:45:54 -05:00
|
|
|
Then, to handle the last two cases for existing Python
|
2016-01-30 02:10:13 -05:00
|
|
|
distributions, we suggest a simple and reliable method to check for
|
|
|
|
the presence and version of ``glibc`` (basically using it as a "clock"
|
|
|
|
for the overall age of the distribution).
|
|
|
|
|
|
|
|
Specifically, the algorithm we propose is::
|
|
|
|
|
|
|
|
def is_manylinux1_compatible():
|
2016-01-30 15:51:23 -05:00
|
|
|
# Only Linux, and only x86-64 / i686
|
2016-01-30 02:10:13 -05:00
|
|
|
from distutils.util import get_platform
|
2016-01-30 15:51:23 -05:00
|
|
|
if get_platform() not in ["linux-x86_64", "linux-i686"]:
|
2016-01-30 02:10:13 -05:00
|
|
|
return False
|
|
|
|
|
|
|
|
# Check for presence of _manylinux module
|
|
|
|
try:
|
|
|
|
import _manylinux
|
|
|
|
return bool(_manylinux.manylinux1_compatible)
|
|
|
|
except (ImportError, AttributeError):
|
|
|
|
# Fall through to heuristic check below
|
|
|
|
pass
|
|
|
|
|
|
|
|
# Check glibc version. CentOS 5 uses glibc 2.5.
|
|
|
|
return have_compatible_glibc(2, 5)
|
2016-01-21 01:23:03 -05:00
|
|
|
|
2016-01-30 02:10:13 -05:00
|
|
|
def have_compatible_glibc(major, minimum_minor):
|
2016-01-21 01:23:03 -05:00
|
|
|
import ctypes
|
|
|
|
|
|
|
|
process_namespace = ctypes.CDLL(None)
|
|
|
|
try:
|
|
|
|
gnu_get_libc_version = process_namespace.gnu_get_libc_version
|
|
|
|
except AttributeError:
|
2016-01-30 02:10:13 -05:00
|
|
|
# Symbol doesn't exist -> therefore, we are not linked to
|
|
|
|
# glibc.
|
2016-01-21 01:23:03 -05:00
|
|
|
return False
|
|
|
|
|
2016-01-30 02:10:13 -05:00
|
|
|
# Call gnu_get_libc_version, which returns a string like "2.5".
|
2016-01-21 01:23:03 -05:00
|
|
|
gnu_get_libc_version.restype = ctypes.c_char_p
|
|
|
|
version_str = gnu_get_libc_version()
|
|
|
|
# py2 / py3 compatibility:
|
|
|
|
if not isinstance(version_str, str):
|
|
|
|
version_str = version_str.decode("ascii")
|
|
|
|
|
2016-01-30 02:10:13 -05:00
|
|
|
# Parse string and check against requested version.
|
2016-01-21 01:23:03 -05:00
|
|
|
version = [int(piece) for piece in version_str.split(".")]
|
|
|
|
assert len(version) == 2
|
|
|
|
if major != version[0]:
|
|
|
|
return False
|
|
|
|
if minimum_minor > version[1]:
|
|
|
|
return False
|
|
|
|
return True
|
|
|
|
|
2016-01-30 02:10:13 -05:00
|
|
|
**Rejected alternatives:** We also considered using a configuration
|
|
|
|
file, e.g. ``/etc/python/compatibility.cfg``. The problem with this is
|
|
|
|
that a single filesystem might contain many different interpreter
|
|
|
|
environments, each with their own ABI profile -- the ``manylinux1``
|
|
|
|
compatibility of a system-installed x86_64 CPython might not tell us
|
2016-01-30 15:51:23 -05:00
|
|
|
much about the ``manylinux1`` compatibility of a user-installed i686
|
2016-01-30 02:10:13 -05:00
|
|
|
PyPy. Locating this configuration information within the Python
|
|
|
|
environment itself ensures that it remains attached to the correct
|
|
|
|
binary, and dramatically simplifies lookup code.
|
|
|
|
|
|
|
|
We also considered using a more elaborate structure, like a list of
|
|
|
|
all platform tags that should be considered compatible, together with
|
|
|
|
their preference ordering, for example: ``_binary_compat.compatible =
|
|
|
|
["manylinux1_x86_64", "centos5_x86_64", "linux_x86_64"]``. However,
|
|
|
|
this introduces several complications. For example, we want to be able
|
|
|
|
to distinguish between the state of "doesn't support ``manylinux1``"
|
|
|
|
(or eventually ``manylinux2``, etc.) versus "doesn't specify either
|
|
|
|
way whether it supports ``manylinux1``", which is not entirely obvious
|
|
|
|
in the above representation; and, it's not at all clear what features
|
|
|
|
are really needed vis a vis preference ordering given that right now
|
|
|
|
the only possible platform tags are ``manylinux1`` and ``linux``. So
|
|
|
|
we're deferring a more complete solution here for a separate PEP, when
|
|
|
|
/ if Linux gets more platform tags.
|
|
|
|
|
|
|
|
For the library compatibility check, we also considered much more
|
|
|
|
elaborate checks (e.g. checking the kernel version, searching for and
|
|
|
|
checking the versions of all the individual libraries listed in the
|
|
|
|
``manylinux1`` profile, etc.), but ultimately decided that this would
|
|
|
|
be more likely to introduce confusing bugs than actually help the
|
|
|
|
user. (For example: different distributions vary in where they
|
|
|
|
actually put these libraries, and if our checking code failed to use
|
|
|
|
the correct path search then it could easily return incorrect
|
|
|
|
answers.)
|
2016-01-21 01:23:03 -05:00
|
|
|
|
|
|
|
|
|
|
|
|
2016-01-27 06:37:05 -05:00
|
|
|
PyPI Support
|
|
|
|
============
|
2016-01-21 01:23:03 -05:00
|
|
|
|
2016-01-27 06:37:05 -05:00
|
|
|
PyPI should permit wheels containing the ``manylinux1`` platform tag to be
|
|
|
|
uploaded. PyPI should not attempt to formally verify that wheels containing
|
|
|
|
the ``manylinux1`` platform tag adhere to the ``manylinux1`` policy described
|
|
|
|
in this document. This verification tasks should be left to other tools, like
|
|
|
|
``auditwheel``, that are developed separately.
|
2016-01-21 01:23:03 -05:00
|
|
|
|
|
|
|
|
|
|
|
Rejected Alternatives
|
|
|
|
=====================
|
|
|
|
|
|
|
|
One alternative would be to provide separate platform tags for each Linux
|
|
|
|
distribution (and each version thereof), e.g. ``RHEL6``, ``ubuntu14_10``,
|
|
|
|
``debian_jessie``, etc. Nothing in this proposal rules out the possibility of
|
|
|
|
adding such platform tags in the future, or of further extensions to wheel
|
|
|
|
metadata that would allow wheels to declare dependencies on external
|
|
|
|
system-installed packages. However, such extensions would require substantially
|
|
|
|
more work than this proposal, and still might not be appreciated by package
|
|
|
|
developers who would prefer not to have to maintain multiple build environments
|
|
|
|
and build multiple wheels in order to cover all the common Linux distributions.
|
2016-07-11 11:14:08 -04:00
|
|
|
Therefore, we consider such proposals to be out-of-scope for this PEP.
|
2016-01-21 01:23:03 -05:00
|
|
|
|
|
|
|
|
2016-01-27 06:37:05 -05:00
|
|
|
Future updates
|
|
|
|
==============
|
|
|
|
|
|
|
|
We anticipate that at some point in the future there will be a
|
|
|
|
``manylinux2`` specifying a more modern baseline environment (perhaps
|
|
|
|
based on CentOS 6), and someday a ``manylinux3`` and so forth, but we
|
|
|
|
defer specifying these until we have more experience with the initial
|
|
|
|
``manylinux1`` proposal.
|
|
|
|
|
|
|
|
|
2016-01-21 01:23:03 -05:00
|
|
|
References
|
|
|
|
==========
|
|
|
|
|
2016-01-27 06:37:05 -05:00
|
|
|
.. [4] Enthought Canopy Python Distribution
|
2016-01-21 01:23:03 -05:00
|
|
|
(https://store.enthought.com/downloads/)
|
2016-01-27 06:37:05 -05:00
|
|
|
.. [5] Continuum Analytics Anaconda Python Distribution
|
2016-01-21 01:23:03 -05:00
|
|
|
(https://www.continuum.io/downloads)
|
2016-01-27 06:37:05 -05:00
|
|
|
.. [6] CentOS 5.11 Release Notes
|
|
|
|
(https://wiki.centos.org/Manuals/ReleaseNotes/CentOS5.11)
|
|
|
|
.. [7] manylinux-discuss mailing list discussion
|
|
|
|
(https://groups.google.com/forum/#!topic/manylinux-discuss/-4l3rrjfr9U)
|
|
|
|
.. [8] distutils-sig discussion
|
|
|
|
(https://mail.python.org/pipermail/distutils-sig/2016-January/027997.html)
|
2016-02-10 00:54:46 -05:00
|
|
|
.. [9] distutils-sig discussion
|
|
|
|
(https://mail.python.org/pipermail/distutils-sig/2016-February/028275.html)
|
|
|
|
.. [10] github issue discussion
|
|
|
|
(https://github.com/pypa/manylinux/issues/30)
|
|
|
|
.. [11] python bug tracker discussion
|
|
|
|
(https://bugs.python.org/issue21536)
|
|
|
|
.. [12] manylinux1 docker images
|
|
|
|
(Source: https://github.com/pypa/manylinux;
|
|
|
|
x86-64: https://quay.io/repository/pypa/manylinux1_x86_64;
|
|
|
|
x86-32: https://quay.io/repository/pypa/manylinux1_i686)
|
|
|
|
.. [13] auditwheel tool
|
2016-01-21 01:23:03 -05:00
|
|
|
(https://pypi.python.org/pypi/auditwheel)
|
2016-02-10 00:54:46 -05:00
|
|
|
.. [14] Fedora Bundled Software Policy
|
2016-01-27 06:37:05 -05:00
|
|
|
(https://fedoraproject.org/wiki/Bundled_Software_policy)
|
2016-02-10 00:54:46 -05:00
|
|
|
.. [15] Debian Policy Manual -- 4.13: Convenience copies of code
|
2016-01-27 06:37:05 -05:00
|
|
|
(https://www.debian.org/doc/debian-policy/ch-source.html#s-embeddedfiles)
|
2016-12-25 14:19:18 -05:00
|
|
|
.. [16] numpy bug report:
|
|
|
|
https://github.com/numpy/numpy/issues/8415#issuecomment-269095235
|
2016-01-27 06:37:05 -05:00
|
|
|
|
2016-01-21 01:23:03 -05:00
|
|
|
|
|
|
|
Copyright
|
|
|
|
=========
|
|
|
|
|
|
|
|
This document has been placed into the public domain.
|
|
|
|
|
|
|
|
..
|
|
|
|
Local Variables:
|
|
|
|
mode: indented-text
|
|
|
|
indent-tabs-mode: nil
|
|
|
|
sentence-end-double-space: t
|
|
|
|
fill-column: 70
|
|
|
|
coding: utf-8
|
|
|
|
End:
|