Churchyard (talk | contribs) m (Fix arm name) |
m (fix typing mistake) |
||
(22 intermediate revisions by 3 users not shown) | |||
Line 31: | Line 31: | ||
== Current status == | == Current status == | ||
[[Category: | [[Category:ChangeAcceptedF34]] | ||
<!-- When your change proposal page is completed and ready for review and announcement --> | <!-- When your change proposal page is completed and ready for review and announcement --> | ||
<!-- remove Category:ChangePageIncomplete and change it to Category:ChangeReadyForWrangler --> | <!-- remove Category:ChangePageIncomplete and change it to Category:ChangeReadyForWrangler --> | ||
Line 50: | Line 50: | ||
CLOSED as NEXTRELEASE -> change is completed and verified and will be delivered in next release under development | CLOSED as NEXTRELEASE -> change is completed and verified and will be delivered in next release under development | ||
--> | --> | ||
* FESCo issue: | * FESCo issue: [https://pagure.io/fesco/issue/2469 #2469] | ||
* Tracker bug: | * Tracker bug: [https://bugzilla.redhat.com/show_bug.cgi?id=1878859 #1878859] | ||
* Release notes tracker: | * Release notes tracker: [https://pagure.io/fedora-docs/release-notes/issue/554 #554] | ||
== Detailed Description == | == Detailed Description == | ||
Line 60: | Line 60: | ||
==== A Long Time Ago in the Fedora Galaxy ==== | ==== A Long Time Ago in the Fedora Galaxy ==== | ||
Many releases ago, when Fedora | Many releases ago, when Fedora wasn't being built for power and arm yet, the Python maintainers mapped the Python "platform triplet" or "multiarch name" to <code>%{_arch}-linux%{_gnu}</code>. This worked. For example, on x86_64, this is <code>x86_64-linux-gnu</code> on Fedora and this is consistent with the "platform triplet" used in filenames in upstream. | ||
==== The Phantom Technical Debt ==== | ==== The Phantom Technical Debt ==== | ||
Line 76: | Line 76: | ||
In the meantime, cross-Linux builds become a thing. The [https://www.python.org/dev/peps/pep-0513/ manylinux1 standard] was created in 2016, allowing to build Python wheels with compiled extension modules on one Linux platform and ship them to many. The first manylinux version only supported x86_64 and i686 and hence it was not impacted by Fedora's patching decisions. | In the meantime, cross-Linux builds become a thing. The [https://www.python.org/dev/peps/pep-0513/ manylinux1 standard] was created in 2016, allowing to build Python wheels with compiled extension modules on one Linux platform and ship them to many. The first manylinux version only supported x86_64 and i686 and hence it was not impacted by Fedora's patching decisions. | ||
The manylinux standard arguably made the upstream Python packaging ecosystem a much nicer | The manylinux standard arguably made the upstream Python packaging ecosystem a much nicer place. Installing packages with compiled extension modules was no longer such a pain. One could just run <code>pip install numpy</code> and not worry about a disturbing lack of a Fortran compiler. For that reason, manylinux wheels become widely adopted by the most popular projects. | ||
==== A New Architecture ==== | ==== A New Architecture ==== | ||
Line 84: | Line 84: | ||
==== Revenge of the Patches ==== | ==== Revenge of the Patches ==== | ||
We have discovered [https://github.com/pypa/manylinux/issues/687 a problem with the ppc64le manylinux2014 wheels]: The CentOS 7 manylinux2014 container images ship upstream Python without RHEL/CentOS/EPEL patches. When an extension module | We have discovered [https://github.com/pypa/manylinux/issues/687 a problem with the ppc64le manylinux2014 wheels]: The CentOS 7 manylinux2014 container images ship upstream Python without RHEL/CentOS/EPEL patches. When an extension module is built there, it is named with an upstream named suffix: <code>.cpython-XY(m)-powerpc64le-linux-gnu.so</code>. The wheel is installable on Fedora (with Fedora's patches), but the module won't (even be considered for) import, because Fedora's Pythons expect the extension suffix to be <code>.cpython-XY(m)-ppc64le-linux-gnu.so</code>. | ||
In theory, we have the same problem on armv7hl, but there are no manylinux2014 containers available for that platform, so there are no such wheels out there ( | In theory, we have the same problem on armv7hl, but there are no manylinux2014 containers available for that platform, so there are no such wheels out there yet (to our knowledge). | ||
The same problem also exists the other way around, albeit it's arguably less severe. It is possible to build manylinux wheels on (some version of) Fedora or EL (using the Python from | The same problem also exists the other way around, albeit it's arguably less severe. It is possible to build manylinux wheels on (some version of) Fedora or EL (using the Python from that distribution). However extension modules from such ppc64le wheels won't import on other Linux distributions. | ||
==== The Workaround Awakens ==== | ==== The Workaround Awakens ==== | ||
To allow importing extension modules from ppc64le manylinux wheels, we have [https://src.fedoraproject.org/rpms/python3.9/pull-request/24 patched Pythons (3.5+) in Fedora] to consider both "Fedora's" and upstream | To allow importing extension modules from ppc64le manylinux wheels, we have [https://src.fedoraproject.org/rpms/python3.9/pull-request/24 patched Pythons (3.5+) in Fedora] to consider both "Fedora's" and upstream extension suffixes when importing extension modules. This workarounds works well for users installing manylinux wheels on Fedora, but does not solve the problem when building the wheels on Fedora. | ||
=== The Change === | === The Change === | ||
Line 98: | Line 98: | ||
With this change proposal, we plan to [https://src.fedoraproject.org/rpms/python3.9/pull-request/28 switch to use the upstream architecture names] and keep the workaround to preserve backwards compatibility. When we do that the following will happen: | With this change proposal, we plan to [https://src.fedoraproject.org/rpms/python3.9/pull-request/28 switch to use the upstream architecture names] and keep the workaround to preserve backwards compatibility. When we do that the following will happen: | ||
# The Python standard library extension module suffixes will change to <code>.cpython-39-powerpc64le-linux-gnu.so</code> and <code>.cpython-39-arm-linux-gnueabihf.so</code>. Python will still import extension modules with the legacy suffixes <code>.cpython-39-ppc64le-linux-gnu.so</code> and <code>.cpython-39-arm-linux-gnu.so</code>. Other architectures not built by koji.fedoraproject.org will also be renamed, see [https://src.fedoraproject.org/rpms/python3.9/pull-request/28 the pull request for a complete regex]. This will happen for Python 3. | # The Python standard library extension module suffixes will change to <code>.cpython-39-powerpc64le-linux-gnu.so</code> and <code>.cpython-39-arm-linux-gnueabihf.so</code>. Python will still import extension modules with the legacy suffixes <code>.cpython-39-ppc64le-linux-gnu.so</code> and <code>.cpython-39-arm-linux-gnu.so</code>. Other architectures not built by koji.fedoraproject.org will also be renamed, see [https://src.fedoraproject.org/rpms/python3.9/pull-request/28#comment-0 the pull request for a complete regex]. This will happen for Python 3.7, 3.8, 3.5, 3.6 and 3.9. | ||
# The newly built Python packages with extension modules will also change the suffixes. Packages that assume the platform triplet is always <code>%{_arch}-linux%{_gnu}</code> (e.g. in the <code>%files</code> section) will need to be adapted (see the [[#New_Macros|New Macros]] section). A mix of legacy and upstream suffixes will co-exist and work together. | # The newly built Python packages with extension modules will also change the suffixes. Packages that assume the platform triplet is always <code>%{_arch}-linux%{_gnu}</code> (e.g. in the <code>%files</code> section) will need to be adapted (see the [[#New_Macros|New Macros]] section). A mix of legacy and upstream suffixes will co-exist and work together. | ||
# When safe, we will drop the workaround to support the legacy names. For example, when we initially package Python 3.10, it will be packaged without the workaround. On the other hand, older Python versions might never be able to drop it, because users will carry their own built extension modules from previous releases. | # When safe, we will drop the workaround to support the legacy names. For example, when we initially package Python 3.10, it will be packaged without the workaround. On the other hand, older Python versions might never be able to drop it, because users will carry their own built extension modules from previous releases. | ||
Line 119: | Line 119: | ||
* etc. | * etc. | ||
'''% | Beware that due to changes in Python, this macro only works with Python 3. For Python 3.4, the value is <code>.cpython-34m.so</code> on all architectures (similarly on older Python 3 versions, but we don't have such in Fedora or EPEL). | ||
'''%python3_platform_triplet''' | |||
Defined as: | Defined as: | ||
% | %python3_platform_triplet %(%{__python3} -Ic "import sysconfig; print(sysconfig.get_config_var('MULTIARCH'))") | ||
Values will be: | Values will be: | ||
Line 132: | Line 134: | ||
* etc. | * etc. | ||
Both macros will be backported to stable Fedoras and EPEL 7+ and will have the corresponding <code>%python_</code> variant | Beware that due to changes in Python, this macro only works with Python versions 3.6 and newer. | ||
Both macros will be backported to stable Fedoras and EPEL 7+ and will have the corresponding <code>%python_</code> variant. | |||
== Feedback == | == Feedback == | ||
Eighth_Doctor: it's the cleverest written change proposal I've seen in over a decade :) | |||
<!-- Summarize the feedback from the community and address why you chose not to accept proposed alternatives. This section is optional for all change proposals but is strongly suggested. Incorporating feedback here as it is raised gives FESCo a clearer view of your proposal and leaves a good record for the future. If you get no feedback, that is useful to note in this section as well. For innovative or possibly controversial ideas, consider collecting feedback before you file the change proposal. --> | <!-- Summarize the feedback from the community and address why you chose not to accept proposed alternatives. This section is optional for all change proposals but is strongly suggested. Incorporating feedback here as it is raised gives FESCo a clearer view of your proposal and leaves a good record for the future. If you get no feedback, that is useful to note in this section as well. For innovative or possibly controversial ideas, consider collecting feedback before you file the change proposal. --> | ||
Line 171: | Line 175: | ||
== Scope == | == Scope == | ||
* Proposal owners: | * Proposal owners: | ||
** Review and merge https://src.fedoraproject.org/rpms/python3.9/pull-request/28 | ** Review and merge https://src.fedoraproject.org/rpms/python3.9/pull-request/28 (done) | ||
** Add the new macros to all Fedoras and EPEL 7+ (done) and mention them in the guidelines | |||
** Monitor builds of Python packages with extension modules, fix issues if they arise. | ** Monitor builds of Python packages with extension modules, fix issues if they arise. | ||
** Backport to Python 3.5-3.8 | ** Backport to Python 3.5-3.8 (in progress) | ||
** Drop the workaround from Python 3.10 once packaged | ** Drop the workaround from Python 3.10 once packaged | ||
<!-- What work do the feature owners have to accomplish to complete the feature in time for release? Is it a large change affecting many parts of the distribution or is it a very isolated change? What are those changes?--> | <!-- What work do the feature owners have to accomplish to complete the feature in time for release? Is it a large change affecting many parts of the distribution or is it a very isolated change? What are those changes?--> | ||
* Other developers: Mostly nothing, adapt the <code>%files</code> section if needed <!-- REQUIRED FOR SYSTEM WIDE CHANGES --> | * Other developers: Mostly nothing, adapt the <code>%files</code> section if needed (done via PRs) <!-- REQUIRED FOR SYSTEM WIDE CHANGES --> | ||
<!-- What work do other developers have to accomplish to complete the feature in time for release? Is it a large change affecting many parts of the distribution or is it a very isolated change? What are those changes?--> | <!-- What work do other developers have to accomplish to complete the feature in time for release? Is it a large change affecting many parts of the distribution or is it a very isolated change? What are those changes?--> | ||
Line 184: | Line 189: | ||
The issue is required to be filed prior to feature submission, to ensure that someone is on board to do any process development work and testing and that all changes make it into the pipeline; a bullet point in a change is not sufficient communication --> | The issue is required to be filed prior to feature submission, to ensure that someone is on board to do any process development work and testing and that all changes make it into the pipeline; a bullet point in a change is not sufficient communication --> | ||
* Policies and guidelines: | * Policies and guidelines: the new macros will be documented in the Python packaging guidelines <!-- REQUIRED FOR SYSTEM WIDE CHANGES --> | ||
<!-- Do the packaging guidelines or other documents need to be updated for this feature? If so, does it need to happen before or after the implementation is done? If a FPC ticket exists, add a link here. --> | <!-- Do the packaging guidelines or other documents need to be updated for this feature? If so, does it need to happen before or after the implementation is done? If a FPC ticket exists, add a link here. --> | ||
Latest revision as of 21:39, 3 January 2021
🔗 Python Upstream Architecture Names
🔗 Summary
Use CPython upstream architecture naming in Fedora's Python ecosystem (mostly in filenames) instead of the previously patched Fedora names.
For example, have /usr/lib64/python3.9/lib-dynload/array.cpython-39-powerpc64le-linux-gnu.so
instead of /usr/lib64/python3.9/lib-dynload/array.cpython-39-ppc64le-linux-gnu.so
.
This makes packaging of Python itself a tad trickier, but it moves Fedora's Python closer to upstream and solves interoperability problems with ppc64le manylinux wheels.
The change has impact only on ppc64le and armv7hl (considering the architectures built by koji.fedoraproject.org).
Packages assuming the filenames always contain %{_arch}-linux%{_gnu}
will need to be adapted.
🔗 Owner
- Name: Miro Hrončok
- Name: Lumír Balhar
- Email: python-maint@redhat.com
🔗 Current status
- Targeted release: Fedora 34
- Last updated: 2021-01-03
- FESCo issue: #2469
- Tracker bug: #1878859
- Release notes tracker: #554
🔗 Detailed Description
🔗 The Saga
🔗 A Long Time Ago in the Fedora Galaxy
Many releases ago, when Fedora wasn't being built for power and arm yet, the Python maintainers mapped the Python "platform triplet" or "multiarch name" to %{_arch}-linux%{_gnu}
. This worked. For example, on x86_64, this is x86_64-linux-gnu
on Fedora and this is consistent with the "platform triplet" used in filenames in upstream.
🔗 The Phantom Technical Debt
Later around the year 2015, as more architectures were added, Python build scripts were patched to use "the Fedora's architecture names":
powerpc
was changed toppc
https://src.fedoraproject.org/rpms/python3/c/0efd3d31- the arm triplet was patched https://src.fedoraproject.org/rpms/python3/c/2f6352e7
- even the mips one https://src.fedoraproject.org/rpms/python3/c/4bc70e0c
At the time, that was a reasonable decision: the idea of cross-Linux builds was sci-fi, and Fedora was not trying to stay close to upstream as it is now (we had around 60 patches; today we're down to around 6).
🔗 Rise of the Manylinux Wheels
In the meantime, cross-Linux builds become a thing. The manylinux1 standard was created in 2016, allowing to build Python wheels with compiled extension modules on one Linux platform and ship them to many. The first manylinux version only supported x86_64 and i686 and hence it was not impacted by Fedora's patching decisions.
The manylinux standard arguably made the upstream Python packaging ecosystem a much nicer place. Installing packages with compiled extension modules was no longer such a pain. One could just run pip install numpy
and not worry about a disturbing lack of a Fortran compiler. For that reason, manylinux wheels become widely adopted by the most popular projects.
🔗 A New Architecture
With the third manylinux version -- manylinux2014 (created in 2019, named after the oldest Linux it supports -- CentOS 7), support for more architectures was introduced: x86_64, i686, aarch64, armv7l, ppc64, ppc64le, s390x. The adaption of new architectures is somehow slow, because the official manylinux2014 containers only currently (August 2020) exist for x86_64, i686, aarch64, ppc64le and s390x.
🔗 Revenge of the Patches
We have discovered a problem with the ppc64le manylinux2014 wheels: The CentOS 7 manylinux2014 container images ship upstream Python without RHEL/CentOS/EPEL patches. When an extension module is built there, it is named with an upstream named suffix: .cpython-XY(m)-powerpc64le-linux-gnu.so
. The wheel is installable on Fedora (with Fedora's patches), but the module won't (even be considered for) import, because Fedora's Pythons expect the extension suffix to be .cpython-XY(m)-ppc64le-linux-gnu.so
.
In theory, we have the same problem on armv7hl, but there are no manylinux2014 containers available for that platform, so there are no such wheels out there yet (to our knowledge).
The same problem also exists the other way around, albeit it's arguably less severe. It is possible to build manylinux wheels on (some version of) Fedora or EL (using the Python from that distribution). However extension modules from such ppc64le wheels won't import on other Linux distributions.
🔗 The Workaround Awakens
To allow importing extension modules from ppc64le manylinux wheels, we have patched Pythons (3.5+) in Fedora to consider both "Fedora's" and upstream extension suffixes when importing extension modules. This workarounds works well for users installing manylinux wheels on Fedora, but does not solve the problem when building the wheels on Fedora.
🔗 The Change
With this change proposal, we plan to switch to use the upstream architecture names and keep the workaround to preserve backwards compatibility. When we do that the following will happen:
- The Python standard library extension module suffixes will change to
.cpython-39-powerpc64le-linux-gnu.so
and.cpython-39-arm-linux-gnueabihf.so
. Python will still import extension modules with the legacy suffixes.cpython-39-ppc64le-linux-gnu.so
and.cpython-39-arm-linux-gnu.so
. Other architectures not built by koji.fedoraproject.org will also be renamed, see the pull request for a complete regex. This will happen for Python 3.7, 3.8, 3.5, 3.6 and 3.9. - The newly built Python packages with extension modules will also change the suffixes. Packages that assume the platform triplet is always
%{_arch}-linux%{_gnu}
(e.g. in the%files
section) will need to be adapted (see the New Macros section). A mix of legacy and upstream suffixes will co-exist and work together. - When safe, we will drop the workaround to support the legacy names. For example, when we initially package Python 3.10, it will be packaged without the workaround. On the other hand, older Python versions might never be able to drop it, because users will carry their own built extension modules from previous releases.
🔗 New Macros
For packagers' convenience we will add 2 new Python macros:
%python3_ext_suffix
Defined as:
%python3_ext_suffix %(%{__python3} -Ic "import sysconfig; print(sysconfig.get_config_var('EXT_SUFFIX'))")
Values will be:
.cpython-39-x86_64-linux-gnu.so
.cpython-39-powerpc64le-linux-gnu.so
on Fedora 34+ /.cpython-39-ppc64le-linux-gnu.so
on older Fedoras.cpython-39-arm-linux-gnueabihf.so
on Fedora 34+ /.cpython-39-arm-linux-gnu.so
on older Fedoras- etc.
Beware that due to changes in Python, this macro only works with Python 3. For Python 3.4, the value is .cpython-34m.so
on all architectures (similarly on older Python 3 versions, but we don't have such in Fedora or EPEL).
%python3_platform_triplet
Defined as:
%python3_platform_triplet %(%{__python3} -Ic "import sysconfig; print(sysconfig.get_config_var('MULTIARCH'))")
Values will be:
x86_64-linux-gnu
powerpc64le-linux-gnu
/ppc64le-linux-gnu
arm-linux-gnueabihf
/arm-linux-gnu
- etc.
Beware that due to changes in Python, this macro only works with Python versions 3.6 and newer.
Both macros will be backported to stable Fedoras and EPEL 7+ and will have the corresponding %python_
variant.
🔗 Feedback
Eighth_Doctor: it's the cleverest written change proposal I've seen in over a decade :)
🔗 Benefit to Fedora
Users of ppc64le and armv7hl Fedora (and future RHEL) will have a closer-to-upstream Python experience and will no longer suffer from compatibility issues when they install or build manylinux wheels. The upstream-downstream balance will be restored.
🔗 Scope
- Proposal owners:
- Review and merge https://src.fedoraproject.org/rpms/python3.9/pull-request/28 (done)
- Add the new macros to all Fedoras and EPEL 7+ (done) and mention them in the guidelines
- Monitor builds of Python packages with extension modules, fix issues if they arise.
- Backport to Python 3.5-3.8 (in progress)
- Drop the workaround from Python 3.10 once packaged
- Other developers: Mostly nothing, adapt the
%files
section if needed (done via PRs)
- Release engineering: a check of an impact with Release Engineering is not needed
- Policies and guidelines: the new macros will be documented in the Python packaging guidelines
- Trademark approval: not needed for this Change
- Alignment with Objectives: no
🔗 Upgrade/compatibility impact
No significant user visible upgrade/compatibility problem is anticipated. Filenames will be different, but the old filenames are still supported. Scripts that hardcode filename assumptions might break.
🔗 How To Test
On ppc64le, try to install a manylinux wheel and import from it. It should work on any Python ≥ 3.5. E.g.:
pip install simple-manylinux-demo python -c 'from dummyextension import extension'
On ppc64le, try to build a manylinux wheel and import from it on another Linux. It should work on any Python ≥ 3.5. E.g.:
pip wheel . # on some project with extension module auditwheel repair ...whl wormhole send ...whl # or any other way
On another ppc64le Linux (such as Debian or openSUSE):
wormhole receive ... pip install ...whl python -c 'from ... import ...'
You can also build a regular (non-manylinux) wheel on Fedora 33/32 and install and import it on Fedora 34. It should work. The other way around will most likely also work, unless Fedora 34 has an incompatible glibc update.
🔗 User Experience
Users of ppc64le and armv7hl Fedora (and future RHEL) will have a closer-to-upstream Python experience and will no longer suffer from compatibility issues when they install or build manylinux wheels.
🔗 Dependencies
No known dependencies. May the force be with us.
🔗 Contingency Plan
- Contingency mechanism: Revert the change and rebuild all affected packages.
- Contingency deadline: Soft before the mass rebuild, so we could leverage it for the revert-rebuilds. Hard before the beta freeze.
- Blocks release? No
- Blocks product? No
🔗 Documentation
This page is the documentation.