Compare commits
1 Commits
Author | SHA1 | Date | |
---|---|---|---|
16234bda52 |
@ -79,8 +79,21 @@ using the `-R` flag:
|
|||||||
%generate_buildrequires
|
%generate_buildrequires
|
||||||
%pyproject_buildrequires -R
|
%pyproject_buildrequires -R
|
||||||
|
|
||||||
Alternatively, the runtime dependencies can be obtained by building the wheel and reading the metadata from the built wheel.
|
Alternatively, if the project specifies its dependencies in the pyproject.toml
|
||||||
This can be enabled by using the `-w` flag.
|
`[project]` table (as defined in [PEP 621](https://www.python.org/dev/peps/pep-0621/)),
|
||||||
|
the runtime dependencies can be obtained by reading that metadata.
|
||||||
|
|
||||||
|
This can be enabled by using the `-p` flag.
|
||||||
|
This flag supports reading both the runtime dependencies, and the selected extras
|
||||||
|
(see the `-x` flag described below).
|
||||||
|
|
||||||
|
Please note that not all build backends which use pyproject.toml support the
|
||||||
|
`[project]` table scheme.
|
||||||
|
For example, poetry-core (at least in 1.9.0) defines package metadata in the
|
||||||
|
custom `[tool.poetry]` table which is not supported by the `%pyproject_buildrequires` macro.
|
||||||
|
|
||||||
|
Finally, the runtime dependencies can be obtained by building the wheel and reading the metadata from the built wheel.
|
||||||
|
This can be enabled with the `-w` flag and cannot be combined with `-p`.
|
||||||
Support for building wheels with `%pyproject_buildrequires -w` is **provisional** and the behavior might change.
|
Support for building wheels with `%pyproject_buildrequires -w` is **provisional** and the behavior might change.
|
||||||
Please subscribe to Fedora's [python-devel list] if you use the option.
|
Please subscribe to Fedora's [python-devel list] if you use the option.
|
||||||
|
|
||||||
@ -111,6 +124,14 @@ For example, if upstream suggests installing test dependencies with
|
|||||||
%generate_buildrequires
|
%generate_buildrequires
|
||||||
%pyproject_buildrequires -x testing
|
%pyproject_buildrequires -x testing
|
||||||
|
|
||||||
|
For projects that specify test requirements using [PEP 735] dependency groups,
|
||||||
|
these can be added using the `-g` flag.
|
||||||
|
Multiple groups can be supplied by repeating the flag or as a comma separated list.
|
||||||
|
For example, if upstream uses a dependency group called `tests`, the test deps would be generated by:
|
||||||
|
|
||||||
|
%generate_buildrequires
|
||||||
|
%pyproject_buildrequires -g tests
|
||||||
|
|
||||||
For projects that specify test requirements in their [tox] configuration,
|
For projects that specify test requirements in their [tox] configuration,
|
||||||
these can be added using the `-t` flag (default tox environment)
|
these can be added using the `-t` flag (default tox environment)
|
||||||
or the `-e` flag followed by the tox environment.
|
or the `-e` flag followed by the tox environment.
|
||||||
@ -134,20 +155,26 @@ The `-e` option redefines `%{toxenv}` for further reuse.
|
|||||||
Use `%{default_toxenv}` to get the default value.
|
Use `%{default_toxenv}` to get the default value.
|
||||||
|
|
||||||
The `-t`/`-e` option uses [tox-current-env]'s `--print-deps-to-file` behind the scenes.
|
The `-t`/`-e` option uses [tox-current-env]'s `--print-deps-to-file` behind the scenes.
|
||||||
|
It generates dependencies listed directly in `deps`,
|
||||||
|
dependencies defined through `extras`,
|
||||||
|
and on tox 4.22+ also dependencies defined through `dependency_groups`.
|
||||||
|
|
||||||
If your package specifies some tox plugins in `tox.requires`,
|
If your package specifies some tox plugins in `tox.requires`,
|
||||||
such plugins will be BuildRequired as well.
|
such plugins will be BuildRequired as well.
|
||||||
Not all plugins are guaranteed to play well with [tox-current-env],
|
Not all plugins are guaranteed to play well with [tox-current-env],
|
||||||
in worst case, patch/sed the requirement out from the tox configuration.
|
in worst case, patch/sed the requirement out from the tox configuration.
|
||||||
|
|
||||||
Note that neither `-x` or `-t` can be used with `-R`,
|
Note that neither `-x` or `-t` can be used with `-R` or `-N`,
|
||||||
because runtime dependencies are always required for testing.
|
because runtime dependencies are always required for testing.
|
||||||
You can only use those options if the build backend supports the [prepare-metadata-for-build-wheel hook],
|
You can only use those options if the build backend supports the [prepare-metadata-for-build-wheel hook],
|
||||||
or together with `-w`.
|
or together with `-p` or `-w`.
|
||||||
|
However, using `-g` with `-R` or `-N` is supported because dependency groups don't need to be used for testing
|
||||||
|
and can be obtained by reading `pyproject.toml` only.
|
||||||
|
|
||||||
[tox]: https://tox.readthedocs.io/
|
[tox]: https://tox.readthedocs.io/
|
||||||
[tox-current-env]: https://github.com/fedora-python/tox-current-env/
|
[tox-current-env]: https://github.com/fedora-python/tox-current-env/
|
||||||
[prepare-metadata-for-build-wheel hook]: https://www.python.org/dev/peps/pep-0517/#prepare-metadata-for-build-wheel
|
[prepare-metadata-for-build-wheel hook]: https://www.python.org/dev/peps/pep-0517/#prepare-metadata-for-build-wheel
|
||||||
|
[python-devel list]: https://lists.fedoraproject.org/archives/list/python-devel@lists.fedoraproject.org/
|
||||||
|
|
||||||
Additionally to generated requirements you can supply multiple file names to `%pyproject_buildrequires` macro.
|
Additionally to generated requirements you can supply multiple file names to `%pyproject_buildrequires` macro.
|
||||||
Dependencies will be loaded from them:
|
Dependencies will be loaded from them:
|
||||||
@ -157,7 +184,7 @@ Dependencies will be loaded from them:
|
|||||||
For packages not using build system you can use `-N` to entirely skip automatical
|
For packages not using build system you can use `-N` to entirely skip automatical
|
||||||
generation of requirements and install requirements only from manually specified files.
|
generation of requirements and install requirements only from manually specified files.
|
||||||
`-N` option implies `-R` and cannot be used in combination with other options mentioned above
|
`-N` option implies `-R` and cannot be used in combination with other options mentioned above
|
||||||
(`-w`, `-e`, `-t`, `-x`).
|
(`-w`, `-e`, `-t`, `-x`, `-p`).
|
||||||
|
|
||||||
The `%pyproject_buildrequires` macro also accepts the `-r` flag for backward compatibility;
|
The `%pyproject_buildrequires` macro also accepts the `-r` flag for backward compatibility;
|
||||||
it means "include runtime dependencies" which has been the default since version 0-53.
|
it means "include runtime dependencies" which has been the default since version 0-53.
|
||||||
@ -287,7 +314,7 @@ However, in Fedora packages, always list executables explicitly to avoid uninten
|
|||||||
`%pyproject_save_files` can automatically mark license files with `%license` macro
|
`%pyproject_save_files` can automatically mark license files with `%license` macro
|
||||||
and language (`*.mo`) files with `%lang` macro and appropriate language code.
|
and language (`*.mo`) files with `%lang` macro and appropriate language code.
|
||||||
Only license files declared via [PEP 639] `License-File` field are detected.
|
Only license files declared via [PEP 639] `License-File` field are detected.
|
||||||
[PEP 639] is still a draft and can be changed in the future.
|
[PEP 639] is still provisional and can be changed in the future.
|
||||||
It is possible to use the `-l` flag to declare that a missing license should
|
It is possible to use the `-l` flag to declare that a missing license should
|
||||||
terminate the build or `-L` (the default) to explicitly disable this check.
|
terminate the build or `-L` (the default) to explicitly disable this check.
|
||||||
Packagers are encouraged to use the `-l` flag when the `%license` file is not manually listed in `%files`
|
Packagers are encouraged to use the `-l` flag when the `%license` file is not manually listed in `%files`
|
||||||
@ -385,6 +412,78 @@ These arguments are still required:
|
|||||||
Multiple subpackages are generated when multiple names are provided.
|
Multiple subpackages are generated when multiple names are provided.
|
||||||
|
|
||||||
|
|
||||||
|
Provisional: Declarative Buildsystem (RPM 4.20+)
|
||||||
|
------------------------------------------------
|
||||||
|
|
||||||
|
It is possible to reduce some of the spec boilerplate by using the provided
|
||||||
|
pyproject [declarative buildsystem].
|
||||||
|
This option is only available with RPM 4.20+ (e.g. in Fedora 41+).
|
||||||
|
The declarative buildsystem is **provisional** and the behavior might change.
|
||||||
|
Please subscribe to Fedora's [python-devel list] if you use the feature.
|
||||||
|
|
||||||
|
To enable the pyproject declarative buildsystem, use the following:
|
||||||
|
|
||||||
|
BuildSystem: pyproject
|
||||||
|
BuildOption(install): <options for %%pyproject_save_files>
|
||||||
|
|
||||||
|
That way, RPM will automatically fill-in the `%prep`, `%generate_buildrequires`,
|
||||||
|
`%build`, `%install`, and `%check` sections the following defaults:
|
||||||
|
|
||||||
|
%prep
|
||||||
|
%autosetup -p1 -C
|
||||||
|
|
||||||
|
%generate_buildrequires
|
||||||
|
%pyproject_buildrequires
|
||||||
|
|
||||||
|
%build
|
||||||
|
%pyproject_wheel
|
||||||
|
|
||||||
|
%install
|
||||||
|
%pyproject_install
|
||||||
|
%pyproject_save_files <options from BuildOption(install)>
|
||||||
|
|
||||||
|
%check
|
||||||
|
%pyproject_check_import
|
||||||
|
|
||||||
|
To pass options to the individual macros, use `BuildOption` (see the [documentation of declarative buildsystems][declarative buildsystem]).
|
||||||
|
|
||||||
|
# pass options for %%pyproject_save_files (mandatory when not overriding %%install)
|
||||||
|
BuildOption(install): -l _module +auto
|
||||||
|
|
||||||
|
# replace the default options for %%autosetup
|
||||||
|
BuildOption(prep): -S git_am -C
|
||||||
|
|
||||||
|
# pass options to %%pyproject_buildrequires
|
||||||
|
BuildOption(generate_buildrequires): docs-requirements.txt -t
|
||||||
|
|
||||||
|
# pass options to %%pyproject_wheel
|
||||||
|
BuildOption(build): -C--global-option=--no-cython-compile
|
||||||
|
|
||||||
|
# pass options to %%pyproject_check_import
|
||||||
|
BuildOption(check): -e '*.test*'
|
||||||
|
|
||||||
|
Alternatively, you can supply your own sections to override the automatic ones:
|
||||||
|
|
||||||
|
BuildOption(generate_buildrequires): -w
|
||||||
|
...
|
||||||
|
%build
|
||||||
|
# do nothing, the wheel was built in %%generate_buildrequires
|
||||||
|
|
||||||
|
You can append to end of the automatic sections:
|
||||||
|
|
||||||
|
%check -a
|
||||||
|
# run %%pytest after %%pyproject_check_import
|
||||||
|
%pytest
|
||||||
|
|
||||||
|
Or prepend to the beginning of them:
|
||||||
|
|
||||||
|
%prep -p
|
||||||
|
# run %%gpgverify before %%autosetup
|
||||||
|
%gpgverify -k2 -s1 -d0
|
||||||
|
|
||||||
|
[declarative buildsystem]: https://rpm-software-management.github.io/rpm/manual/buildsystem.html
|
||||||
|
|
||||||
|
|
||||||
Limitations
|
Limitations
|
||||||
-----------
|
-----------
|
||||||
|
|
||||||
@ -434,6 +533,7 @@ so be prepared for problems.
|
|||||||
[PEP 517]: https://www.python.org/dev/peps/pep-0517/
|
[PEP 517]: https://www.python.org/dev/peps/pep-0517/
|
||||||
[PEP 518]: https://www.python.org/dev/peps/pep-0518/
|
[PEP 518]: https://www.python.org/dev/peps/pep-0518/
|
||||||
[PEP 639]: https://www.python.org/dev/peps/pep-0639/
|
[PEP 639]: https://www.python.org/dev/peps/pep-0639/
|
||||||
|
[PEP 735]: https://www.python.org/dev/peps/pep-0735/
|
||||||
[pip's documentation]: https://pip.pypa.io/en/stable/cli/pip_install/#vcs-support
|
[pip's documentation]: https://pip.pypa.io/en/stable/cli/pip_install/#vcs-support
|
||||||
|
|
||||||
|
|
||||||
|
@ -4,4 +4,14 @@
|
|||||||
# this macro will cause the package with the real macro to be installed.
|
# this macro will cause the package with the real macro to be installed.
|
||||||
# When macros.pyproject is installed, it overrides this macro.
|
# When macros.pyproject is installed, it overrides this macro.
|
||||||
# Note: This needs to maintain the same set of options as the real macro.
|
# Note: This needs to maintain the same set of options as the real macro.
|
||||||
%pyproject_buildrequires(rRxtNwe:C:) echo 'pyproject-rpm-macros' && exit 0
|
%pyproject_buildrequires(rRxtNwpe:g:C:) echo 'pyproject-rpm-macros' && exit 0
|
||||||
|
|
||||||
|
|
||||||
|
# Declarative buildsystem, requires RPM 4.20+ to work
|
||||||
|
# https://rpm-software-management.github.io/rpm/manual/buildsystem.html
|
||||||
|
# This is the minimal implementation to be in the srpm package,
|
||||||
|
# as required even before the BuildRequires are installed
|
||||||
|
%buildsystem_pyproject_conf() %nil
|
||||||
|
%buildsystem_pyproject_generate_buildrequires() %pyproject_buildrequires %*
|
||||||
|
%buildsystem_pyproject_build() %nil
|
||||||
|
%buildsystem_pyproject_install() %nil
|
||||||
|
@ -25,6 +25,11 @@
|
|||||||
%_pyproject_record %{_builddir}/%{_pyproject_files_prefix}-pyproject-record
|
%_pyproject_record %{_builddir}/%{_pyproject_files_prefix}-pyproject-record
|
||||||
%_pyproject_buildrequires %{_builddir}/%{_pyproject_files_prefix}-pyproject-buildrequires
|
%_pyproject_buildrequires %{_builddir}/%{_pyproject_files_prefix}-pyproject-buildrequires
|
||||||
|
|
||||||
|
# Internal macro, takes %%set_build_flags and strips all the exports
|
||||||
|
# TODO: Make such a list an actual source of %%set_build_flags (in redhat-rpm-config)
|
||||||
|
# Cannot use %%gsub directly to preserve EL 9 compatibility
|
||||||
|
%_pyproject_build_flags %{lua:local exports = rpm.expand('%{set_build_flags} ;'); print((exports:gsub('%s*;+%s+export%s+[%u_]+%s*;+%s*', ' ')))}
|
||||||
|
|
||||||
# Avoid leaking %%{_pyproject_builddir} to pytest collection
|
# Avoid leaking %%{_pyproject_builddir} to pytest collection
|
||||||
# https://bugzilla.redhat.com/show_bug.cgi?id=1935212
|
# https://bugzilla.redhat.com/show_bug.cgi?id=1935212
|
||||||
# The value is read and used by the %%pytest and %%tox macros:
|
# The value is read and used by the %%pytest and %%tox macros:
|
||||||
@ -33,7 +38,8 @@
|
|||||||
%pyproject_wheel(C:) %{expand:\\\
|
%pyproject_wheel(C:) %{expand:\\\
|
||||||
%_set_pytest_addopts
|
%_set_pytest_addopts
|
||||||
mkdir -p "%{_pyproject_builddir}"
|
mkdir -p "%{_pyproject_builddir}"
|
||||||
CFLAGS="${CFLAGS:-${RPM_OPT_FLAGS}}" LDFLAGS="${LDFLAGS:-${RPM_LD_FLAGS}}" TMPDIR="%{_pyproject_builddir}" \\\
|
%{_pyproject_build_flags} \\\
|
||||||
|
TMPDIR="%{_pyproject_builddir}" \\\
|
||||||
%{__python3} -Bs %{_rpmconfigdir}/redhat/pyproject_wheel.py %{?**} %{_pyproject_wheeldir}
|
%{__python3} -Bs %{_rpmconfigdir}/redhat/pyproject_wheel.py %{?**} %{_pyproject_wheeldir}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -109,15 +115,15 @@ fi
|
|||||||
|
|
||||||
|
|
||||||
# Note: the three times nested questionmarked -i -f -F pattern means: If none of those options was used -- in that case, we inject our own -f
|
# Note: the three times nested questionmarked -i -f -F pattern means: If none of those options was used -- in that case, we inject our own -f
|
||||||
%pyproject_extras_subpkg(n:i:f:F) %{expand:%{?python_extras_subpkg:%{python_extras_subpkg%{?!-i:%{?!-f:%{?!-F: -f %{_pyproject_ghost_distinfo}}}} %**}}}
|
%pyproject_extras_subpkg(n:i:f:FaA) %{expand:%{?python_extras_subpkg:%{python_extras_subpkg%{?!-i:%{?!-f:%{?!-F: -f %{_pyproject_ghost_distinfo}}}} %**}}}
|
||||||
|
|
||||||
|
|
||||||
# Escaping an actual percentage sign in path by 8 signs has been verified in RPM 4.16 and 4.17.
|
# Escaping shell-globs, percentage signs and spaces was reworked in RPM 4.19+
|
||||||
# See this thread http://lists.rpm.org/pipermail/rpm-list/2021-June/002048.html
|
# https://github.com/rpm-software-management/rpm/issues/1749#issuecomment-1020420616
|
||||||
# Since RPM 4.19, 2 signs are needed instead. 4.18.90+ is a pre-release of RPM 4.19.
|
# Since we support both ways, we pass either 4.19 or 4.18 to the script, so it knows which one to use
|
||||||
# On the CI, we build tests/escape_percentages.spec to verify the assumptions.
|
# Rather than passing the actual version, we let RPM compare the versions, as it is easier done here than in Python
|
||||||
%pyproject_save_files(lL) %{expand:\\\
|
%pyproject_save_files(lL) %{expand:\\\
|
||||||
%{expr:v"0%{?rpmversion}" >= v"4.18.90" ? "RPM_PERCENTAGES_COUNT=2" : "RPM_PERCENTAGES_COUNT=8" } \\
|
%{expr:v"0%{?rpmversion}" >= v"4.18.90" ? "RPM_FILES_ESCAPE=4.19" : "RPM_FILES_ESCAPE=4.18" } \\
|
||||||
%{__python3} %{_rpmconfigdir}/redhat/pyproject_save_files.py \\
|
%{__python3} %{_rpmconfigdir}/redhat/pyproject_save_files.py \\
|
||||||
--output-files "%{pyproject_files}" \\
|
--output-files "%{pyproject_files}" \\
|
||||||
--output-modules "%{_pyproject_modules}" \\
|
--output-modules "%{_pyproject_modules}" \\
|
||||||
@ -152,13 +158,17 @@ fi
|
|||||||
%default_toxenv py%{python3_version_nodots}
|
%default_toxenv py%{python3_version_nodots}
|
||||||
%toxenv %{default_toxenv}
|
%toxenv %{default_toxenv}
|
||||||
|
|
||||||
|
%_pyproject_tomlidep %["%{python3_pkgversion}" == "3"\
|
||||||
|
? "echo '(python%{python3_pkgversion}dist(tomli) if python%{python3_pkgversion}-devel < 3.11)'"\
|
||||||
|
: "%[v"%{python3_pkgversion}" < v"3.11"\
|
||||||
|
? "echo 'python%{python3_pkgversion}dist(tomli)'"\
|
||||||
|
: "true # will use tomllib, echo nothing"\
|
||||||
|
]"\
|
||||||
|
]
|
||||||
|
|
||||||
# Note: Keep the options in sync with this macro from macros.aaa-pyproject-srpm
|
# Note: Keep the options in sync with this macro from macros.aaa-pyproject-srpm
|
||||||
%pyproject_buildrequires(rRxtNwe:C:) %{expand:\\\
|
%pyproject_buildrequires(rRxtNwpe:g:C:) %{expand:\\\
|
||||||
%_set_pytest_addopts
|
%_set_pytest_addopts
|
||||||
# The _auto_set_build_flags feature does not do this in %%generate_buildrequires section,
|
|
||||||
# but we want to get an environment consistent with %%build:
|
|
||||||
%{?_auto_set_build_flags:%set_build_flags}
|
|
||||||
# The default flags expect the package note file to exist
|
# The default flags expect the package note file to exist
|
||||||
# see https://bugzilla.redhat.com/show_bug.cgi?id=2097535
|
# see https://bugzilla.redhat.com/show_bug.cgi?id=2097535
|
||||||
%{?_package_note_flags:%_generate_package_note_file}
|
%{?_package_note_flags:%_generate_package_note_file}
|
||||||
@ -168,6 +178,7 @@ fi
|
|||||||
%{-e:%{error:The -R and -e options are mutually exclusive}}
|
%{-e:%{error:The -R and -e options are mutually exclusive}}
|
||||||
%{-t:%{error:The -R and -t options are mutually exclusive}}
|
%{-t:%{error:The -R and -t options are mutually exclusive}}
|
||||||
%{-w:%{error:The -R and -w options are mutually exclusive}}
|
%{-w:%{error:The -R and -w options are mutually exclusive}}
|
||||||
|
%{-p:%{error:The -R and -p options are mutually exclusive}}
|
||||||
}
|
}
|
||||||
%{-N:
|
%{-N:
|
||||||
%{-r:%{error:The -N and -r options are mutually exclusive}}
|
%{-r:%{error:The -N and -r options are mutually exclusive}}
|
||||||
@ -175,25 +186,25 @@ fi
|
|||||||
%{-e:%{error:The -N and -e options are mutually exclusive}}
|
%{-e:%{error:The -N and -e options are mutually exclusive}}
|
||||||
%{-t:%{error:The -N and -t options are mutually exclusive}}
|
%{-t:%{error:The -N and -t options are mutually exclusive}}
|
||||||
%{-w:%{error:The -N and -w options are mutually exclusive}}
|
%{-w:%{error:The -N and -w options are mutually exclusive}}
|
||||||
|
%{-p:%{error:The -N and -p options are mutually exclusive}}
|
||||||
%{-C:%{error:The -N and -C options are mutually exclusive}}
|
%{-C:%{error:The -N and -C options are mutually exclusive}}
|
||||||
|
%{-g:if [ -f pyproject.toml ]; then
|
||||||
|
%_pyproject_tomlidep
|
||||||
|
fi}
|
||||||
|
}
|
||||||
|
%{-w:
|
||||||
|
%{-p:%{error:The -w and -p options are mutually exclusive}}
|
||||||
}
|
}
|
||||||
%{-e:%{expand:%global toxenv %(%{__python3} -s %{_rpmconfigdir}/redhat/pyproject_construct_toxenv.py %{?**})}}
|
%{-e:%{expand:%global toxenv %(%{__python3} -s %{_rpmconfigdir}/redhat/pyproject_construct_toxenv.py %{?**})}}
|
||||||
echo 'pyproject-rpm-macros' # first stdout line matches the implementation in macros.aaa-pyproject-srpm
|
echo 'pyproject-rpm-macros' # first stdout line matches the implementation in macros.aaa-pyproject-srpm
|
||||||
echo 'python%{python3_pkgversion}-devel'
|
echo 'python%{python3_pkgversion}-devel'
|
||||||
echo 'python%{python3_pkgversion}dist(pip) >= 19'
|
|
||||||
echo 'python%{python3_pkgversion}dist(packaging)'
|
echo 'python%{python3_pkgversion}dist(packaging)'
|
||||||
%{!-N:if [ -f pyproject.toml ]; then
|
%{!-N:echo 'python%{python3_pkgversion}dist(pip) >= 19'
|
||||||
%["%{python3_pkgversion}" == "3"
|
if [ -f pyproject.toml ]; then
|
||||||
? "echo '(python%{python3_pkgversion}dist(tomli) if python%{python3_pkgversion}-devel < 3.11)'"
|
%_pyproject_tomlidep
|
||||||
: "%[v"%{python3_pkgversion}" < v"3.11"
|
|
||||||
? "echo 'python%{python3_pkgversion}dist(tomli)'"
|
|
||||||
: "true # will use tomllib, echo nothing"
|
|
||||||
]"
|
|
||||||
]
|
|
||||||
elif [ -f setup.py ]; then
|
elif [ -f setup.py ]; then
|
||||||
# Note: If the default requirements change, also change them in the script!
|
# Note: If the default requirements change, also change them in the script!
|
||||||
echo 'python%{python3_pkgversion}dist(setuptools) >= 40.8'
|
echo 'python%{python3_pkgversion}dist(setuptools) >= 40.8'
|
||||||
echo 'python%{python3_pkgversion}dist(wheel)'
|
|
||||||
else
|
else
|
||||||
echo 'ERROR: Neither pyproject.toml nor setup.py found, consider using %%%%pyproject_buildrequires -N <requirements-file> if this is not a Python package.' >&2
|
echo 'ERROR: Neither pyproject.toml nor setup.py found, consider using %%%%pyproject_buildrequires -N <requirements-file> if this is not a Python package.' >&2
|
||||||
exit 1
|
exit 1
|
||||||
@ -203,7 +214,8 @@ rm -rfv *.dist-info/ >&2
|
|||||||
if [ -f %{__python3} ]; then
|
if [ -f %{__python3} ]; then
|
||||||
mkdir -p "%{_pyproject_builddir}"
|
mkdir -p "%{_pyproject_builddir}"
|
||||||
echo -n > %{_pyproject_buildrequires}
|
echo -n > %{_pyproject_buildrequires}
|
||||||
CFLAGS="${CFLAGS:-${RPM_OPT_FLAGS}}" LDFLAGS="${LDFLAGS:-${RPM_LD_FLAGS}}" TMPDIR="%{_pyproject_builddir}" \\\
|
%{_pyproject_build_flags} \\\
|
||||||
|
TMPDIR="%{_pyproject_builddir}" \\\
|
||||||
RPM_TOXENV="%{toxenv}" HOSTNAME="rpmbuild" %{__python3} -Bs %{_rpmconfigdir}/redhat/pyproject_buildrequires.py %{?!_python_no_extras_requires:--generate-extras} --python3_pkgversion %{python3_pkgversion} --wheeldir %{_pyproject_wheeldir} --output %{_pyproject_buildrequires} %{?**} >&2
|
RPM_TOXENV="%{toxenv}" HOSTNAME="rpmbuild" %{__python3} -Bs %{_rpmconfigdir}/redhat/pyproject_buildrequires.py %{?!_python_no_extras_requires:--generate-extras} --python3_pkgversion %{python3_pkgversion} --wheeldir %{_pyproject_wheeldir} --output %{_pyproject_buildrequires} %{?**} >&2
|
||||||
cat %{_pyproject_buildrequires}
|
cat %{_pyproject_buildrequires}
|
||||||
fi
|
fi
|
||||||
@ -221,3 +233,13 @@ PYTHONPATH="${PYTHONPATH:-%{buildroot}%{python3_sitearch}:%{buildroot}%{python3_
|
|||||||
HOSTNAME="rpmbuild" \\
|
HOSTNAME="rpmbuild" \\
|
||||||
%{__python3} -m tox --current-env -q --recreate -e "%{-e:%{-e*}}%{!-e:%{toxenv}}" %{?*}
|
%{__python3} -m tox --current-env -q --recreate -e "%{-e:%{-e*}}%{!-e:%{toxenv}}" %{?*}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
# Declarative buildsystem, requires RPM 4.20+ to work
|
||||||
|
# https://rpm-software-management.github.io/rpm/manual/buildsystem.html
|
||||||
|
%buildsystem_pyproject_conf() %nil
|
||||||
|
%buildsystem_pyproject_generate_buildrequires() %pyproject_buildrequires %*
|
||||||
|
%buildsystem_pyproject_build() %pyproject_wheel %*
|
||||||
|
%buildsystem_pyproject_install() %["%{shrink:%*}" == "" ? "%{error:BuildOption(install) is mandatory with pyproject BuildSystem.}" : "%pyproject_install \
|
||||||
|
%pyproject_save_files %*"]
|
||||||
|
%buildsystem_pyproject_check() %pyproject_check_import %*
|
||||||
|
@ -10,6 +10,7 @@ import subprocess
|
|||||||
import re
|
import re
|
||||||
import tempfile
|
import tempfile
|
||||||
import email.parser
|
import email.parser
|
||||||
|
import functools
|
||||||
import pathlib
|
import pathlib
|
||||||
import zipfile
|
import zipfile
|
||||||
|
|
||||||
@ -34,6 +35,7 @@ def print_err(*args, **kwargs):
|
|||||||
|
|
||||||
|
|
||||||
try:
|
try:
|
||||||
|
from packaging.markers import Marker
|
||||||
from packaging.requirements import Requirement, InvalidRequirement
|
from packaging.requirements import Requirement, InvalidRequirement
|
||||||
from packaging.utils import canonicalize_name
|
from packaging.utils import canonicalize_name
|
||||||
except ImportError as e:
|
except ImportError as e:
|
||||||
@ -99,18 +101,23 @@ class Requirements:
|
|||||||
return True
|
return True
|
||||||
return False
|
return False
|
||||||
|
|
||||||
def add(self, requirement_str, *, package_name=None, source=None):
|
def add(self, requirement, *, package_name=None, source=None, extra=None):
|
||||||
"""Output a Python-style requirement string as RPM dep"""
|
"""Output a Python-style requirement string as RPM dep"""
|
||||||
|
|
||||||
|
requirement_str = str(requirement)
|
||||||
print_err(f'Handling {requirement_str} from {source}')
|
print_err(f'Handling {requirement_str} from {source}')
|
||||||
|
|
||||||
try:
|
# requirements read initially from the metadata are strings
|
||||||
requirement = Requirement(requirement_str)
|
# further on we work with them as Requirement instances
|
||||||
except InvalidRequirement:
|
if not isinstance(requirement, Requirement):
|
||||||
hint = guess_reason_for_invalid_requirement(requirement_str)
|
try:
|
||||||
message = f'Requirement {requirement_str!r} from {source} is invalid.'
|
requirement = Requirement(requirement)
|
||||||
if hint:
|
except InvalidRequirement:
|
||||||
message += f' Hint: {hint}'
|
hint = guess_reason_for_invalid_requirement(requirement)
|
||||||
raise ValueError(message)
|
message = f'Requirement {requirement!r} from {source} is invalid.'
|
||||||
|
if hint:
|
||||||
|
message += f' Hint: {hint}'
|
||||||
|
raise ValueError(message)
|
||||||
|
|
||||||
if requirement.url:
|
if requirement.url:
|
||||||
print_err(
|
print_err(
|
||||||
@ -118,10 +125,17 @@ class Requirements:
|
|||||||
)
|
)
|
||||||
|
|
||||||
name = canonicalize_name(requirement.name)
|
name = canonicalize_name(requirement.name)
|
||||||
|
|
||||||
|
if extra is not None:
|
||||||
|
extra_str = f'extra == "{extra}"'
|
||||||
|
if requirement.marker is not None:
|
||||||
|
extra_str = f'({requirement.marker}) and {extra_str}'
|
||||||
|
requirement.marker = Marker(extra_str)
|
||||||
|
|
||||||
if (requirement.marker is not None and
|
if (requirement.marker is not None and
|
||||||
not self.evaluate_all_environments(requirement)):
|
not self.evaluate_all_environments(requirement)):
|
||||||
print_err(f'Ignoring alien requirement:', requirement_str)
|
print_err(f'Ignoring alien requirement:', requirement_str)
|
||||||
self.ignored_alien_requirements.append(requirement_str)
|
self.ignored_alien_requirements.append(requirement)
|
||||||
return
|
return
|
||||||
|
|
||||||
# Handle self-referencing requirements
|
# Handle self-referencing requirements
|
||||||
@ -215,7 +229,8 @@ def toml_load(opened_binary_file):
|
|||||||
return tomllib.load(opened_binary_file)
|
return tomllib.load(opened_binary_file)
|
||||||
|
|
||||||
|
|
||||||
def get_backend(requirements):
|
@functools.cache
|
||||||
|
def load_pyproject():
|
||||||
try:
|
try:
|
||||||
f = open('pyproject.toml', 'rb')
|
f = open('pyproject.toml', 'rb')
|
||||||
except FileNotFoundError:
|
except FileNotFoundError:
|
||||||
@ -223,6 +238,11 @@ def get_backend(requirements):
|
|||||||
else:
|
else:
|
||||||
with f:
|
with f:
|
||||||
pyproject_data = toml_load(f)
|
pyproject_data = toml_load(f)
|
||||||
|
return pyproject_data
|
||||||
|
|
||||||
|
|
||||||
|
def get_backend(requirements):
|
||||||
|
pyproject_data = load_pyproject()
|
||||||
|
|
||||||
buildsystem_data = pyproject_data.get('build-system', {})
|
buildsystem_data = pyproject_data.get('build-system', {})
|
||||||
requirements.extend(
|
requirements.extend(
|
||||||
@ -248,7 +268,6 @@ def get_backend(requirements):
|
|||||||
# with pyproject.toml without a specified build backend.
|
# with pyproject.toml without a specified build backend.
|
||||||
# If the default requirements change, also change them in the macro!
|
# If the default requirements change, also change them in the macro!
|
||||||
requirements.add('setuptools >= 40.8', source='default build backend')
|
requirements.add('setuptools >= 40.8', source='default build backend')
|
||||||
requirements.add('wheel', source='default build backend')
|
|
||||||
|
|
||||||
requirements.check(source='build backend')
|
requirements.check(source='build backend')
|
||||||
|
|
||||||
@ -302,7 +321,9 @@ def generate_run_requirements_hook(backend, requirements):
|
|||||||
raise ValueError(
|
raise ValueError(
|
||||||
'The build backend cannot provide build metadata '
|
'The build backend cannot provide build metadata '
|
||||||
'(incl. runtime requirements) before build. '
|
'(incl. runtime requirements) before build. '
|
||||||
'Use the provisional -w flag to build the wheel and parse the metadata from it, '
|
'If the dependencies are specified in the pyproject.toml [project] '
|
||||||
|
'table, you can use the -p flag to read them.'
|
||||||
|
'Alternatively, use the provisional -w flag to build the wheel and parse the metadata from it, '
|
||||||
'or use the -R flag not to generate runtime dependencies.'
|
'or use the -R flag not to generate runtime dependencies.'
|
||||||
)
|
)
|
||||||
dir_basename = prepare_metadata('.', config_settings=requirements.config_settings)
|
dir_basename = prepare_metadata('.', config_settings=requirements.config_settings)
|
||||||
@ -360,8 +381,35 @@ def generate_run_requirements_wheel(backend, requirements, wheeldir):
|
|||||||
raise RuntimeError('Could not find *.dist-info/METADATA in built wheel.')
|
raise RuntimeError('Could not find *.dist-info/METADATA in built wheel.')
|
||||||
|
|
||||||
|
|
||||||
def generate_run_requirements(backend, requirements, *, build_wheel, wheeldir):
|
def generate_run_requirements_pyproject(requirements):
|
||||||
if build_wheel:
|
pyproject_data = load_pyproject()
|
||||||
|
|
||||||
|
if not (project_table := pyproject_data.get('project', {})):
|
||||||
|
raise ValueError('Could not find the [project] table in pyproject.toml.')
|
||||||
|
|
||||||
|
dynamic_fields = project_table.get('dynamic', [])
|
||||||
|
if 'dependencies' in dynamic_fields or 'optional-dependencies' in dynamic_fields:
|
||||||
|
raise ValueError('Could not read the dependencies or optional-dependencies '
|
||||||
|
'from the [project] table in pyproject.toml, as the field is dynamic.')
|
||||||
|
|
||||||
|
dependencies = project_table.get('dependencies', [])
|
||||||
|
name = project_table.get('name')
|
||||||
|
requirements.extend(dependencies,
|
||||||
|
package_name=name,
|
||||||
|
source=f'pyproject.toml generated metadata: [dependencies] ({name})')
|
||||||
|
|
||||||
|
optional_dependencies = project_table.get('optional-dependencies', {})
|
||||||
|
for extra, dependencies in optional_dependencies.items():
|
||||||
|
requirements.extend(dependencies,
|
||||||
|
package_name=name,
|
||||||
|
source=f'pyproject.toml generated metadata: [optional-dependencies] {extra} ({name})',
|
||||||
|
extra=extra)
|
||||||
|
|
||||||
|
|
||||||
|
def generate_run_requirements(backend, requirements, *, build_wheel, read_pyproject_dependencies, wheeldir):
|
||||||
|
if read_pyproject_dependencies:
|
||||||
|
generate_run_requirements_pyproject(requirements)
|
||||||
|
elif build_wheel:
|
||||||
generate_run_requirements_wheel(backend, requirements, wheeldir)
|
generate_run_requirements_wheel(backend, requirements, wheeldir)
|
||||||
else:
|
else:
|
||||||
generate_run_requirements_hook(backend, requirements)
|
generate_run_requirements_hook(backend, requirements)
|
||||||
@ -411,6 +459,103 @@ def generate_tox_requirements(toxenv, requirements):
|
|||||||
source=f'tox --print-deps-only: {toxenv}')
|
source=f'tox --print-deps-only: {toxenv}')
|
||||||
|
|
||||||
|
|
||||||
|
def tox_dependency_groups(toxenv):
|
||||||
|
# We call this command separately instead of folding it into the previous one
|
||||||
|
# becasue --print-dependency-groups-to only works with tox 4.22+ and tox-current-env 0.0.14+.
|
||||||
|
# We handle failure gracefully: upstreams using dependency_groups should require tox >= 4.22.
|
||||||
|
toxenv = ','.join(toxenv)
|
||||||
|
with tempfile.NamedTemporaryFile('r') as groups:
|
||||||
|
r = subprocess.run(
|
||||||
|
[sys.executable, '-m', 'tox',
|
||||||
|
'--print-dependency-groups-to', groups.name,
|
||||||
|
'-q', '-e', toxenv],
|
||||||
|
check=False,
|
||||||
|
encoding='utf-8',
|
||||||
|
stdout=subprocess.PIPE,
|
||||||
|
stderr=subprocess.STDOUT,
|
||||||
|
)
|
||||||
|
if r.returncode == 0:
|
||||||
|
if r.stdout:
|
||||||
|
print_err(r.stdout, end='')
|
||||||
|
if output := groups.read().strip():
|
||||||
|
return output.splitlines()
|
||||||
|
return []
|
||||||
|
|
||||||
|
|
||||||
|
def generate_dependency_groups(requested_groups, requirements):
|
||||||
|
"""Adapted from https://peps.python.org/pep-0735/#reference-implementation (public domain)"""
|
||||||
|
from collections import defaultdict
|
||||||
|
|
||||||
|
def _normalize_name(name: str) -> str:
|
||||||
|
return re.sub(r"[-_.]+", "-", name).lower()
|
||||||
|
|
||||||
|
def _normalize_group_names(dependency_groups: dict) -> dict:
|
||||||
|
original_names = defaultdict(list)
|
||||||
|
normalized_groups = {}
|
||||||
|
|
||||||
|
for group_name, value in dependency_groups.items():
|
||||||
|
normed_group_name = _normalize_name(group_name)
|
||||||
|
original_names[normed_group_name].append(group_name)
|
||||||
|
normalized_groups[normed_group_name] = value
|
||||||
|
|
||||||
|
errors = []
|
||||||
|
for normed_name, names in original_names.items():
|
||||||
|
if len(names) > 1:
|
||||||
|
errors.append(f"{normed_name} ({', '.join(names)})")
|
||||||
|
if errors:
|
||||||
|
raise ValueError(f"Duplicate dependency group names: {', '.join(errors)}")
|
||||||
|
|
||||||
|
return normalized_groups
|
||||||
|
|
||||||
|
def _resolve_dependency_group(
|
||||||
|
dependency_groups: dict, group: str, past_groups: tuple[str, ...] = ()
|
||||||
|
) -> list[str]:
|
||||||
|
if group in past_groups:
|
||||||
|
raise ValueError(f"Cyclic dependency group include: {group} -> {past_groups}")
|
||||||
|
|
||||||
|
if group not in dependency_groups:
|
||||||
|
raise LookupError(f"Dependency group '{group}' not found")
|
||||||
|
|
||||||
|
raw_group = dependency_groups[group]
|
||||||
|
if not isinstance(raw_group, list):
|
||||||
|
raise ValueError(f"Dependency group '{group}' is not a list")
|
||||||
|
|
||||||
|
realized_group = []
|
||||||
|
for item in raw_group:
|
||||||
|
if isinstance(item, str):
|
||||||
|
realized_group.append(item)
|
||||||
|
elif isinstance(item, dict):
|
||||||
|
if tuple(item.keys()) != ("include-group",):
|
||||||
|
raise ValueError(f"Invalid dependency group item: {item}")
|
||||||
|
|
||||||
|
include_group = _normalize_name(next(iter(item.values())))
|
||||||
|
realized_group.extend(
|
||||||
|
_resolve_dependency_group(
|
||||||
|
dependency_groups, include_group, past_groups + (group,)
|
||||||
|
)
|
||||||
|
)
|
||||||
|
else:
|
||||||
|
raise ValueError(f"Invalid dependency group item: {item}")
|
||||||
|
|
||||||
|
return realized_group
|
||||||
|
|
||||||
|
def resolve(dependency_groups: dict, group: str) -> list[str]:
|
||||||
|
if not isinstance(dependency_groups, dict):
|
||||||
|
raise TypeError("Dependency Groups table is not a dict")
|
||||||
|
return _resolve_dependency_group(dependency_groups, _normalize_name(group))
|
||||||
|
|
||||||
|
pyproject_data = load_pyproject()
|
||||||
|
dependency_groups_raw = pyproject_data.get("dependency-groups", {})
|
||||||
|
dependency_groups = _normalize_group_names(dependency_groups_raw)
|
||||||
|
|
||||||
|
for group_names in requested_groups:
|
||||||
|
for group_name in group_names.split(","):
|
||||||
|
requirements.extend(
|
||||||
|
resolve(dependency_groups, group_name),
|
||||||
|
source=f"Dependency group {group_name}",
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
def python3dist(name, op=None, version=None, python3_pkgversion="3"):
|
def python3dist(name, op=None, version=None, python3_pkgversion="3"):
|
||||||
prefix = f"python{python3_pkgversion}dist"
|
prefix = f"python{python3_pkgversion}dist"
|
||||||
|
|
||||||
@ -423,9 +568,10 @@ def python3dist(name, op=None, version=None, python3_pkgversion="3"):
|
|||||||
|
|
||||||
|
|
||||||
def generate_requires(
|
def generate_requires(
|
||||||
*, include_runtime=False, build_wheel=False, wheeldir=None, toxenv=None, extras=None,
|
*, include_runtime=False, build_wheel=False, wheeldir=None, toxenv=None, extras=None, dependency_groups=None,
|
||||||
get_installed_version=importlib.metadata.version, # for dep injection
|
get_installed_version=importlib.metadata.version, # for dep injection
|
||||||
generate_extras=False, python3_pkgversion="3", requirement_files=None, use_build_system=True,
|
generate_extras=False, python3_pkgversion="3", requirement_files=None, use_build_system=True,
|
||||||
|
read_pyproject_dependencies=False,
|
||||||
output, config_settings=None,
|
output, config_settings=None,
|
||||||
):
|
):
|
||||||
"""Generate the BuildRequires for the project in the current directory
|
"""Generate the BuildRequires for the project in the current directory
|
||||||
@ -441,9 +587,10 @@ def generate_requires(
|
|||||||
config_settings=config_settings,
|
config_settings=config_settings,
|
||||||
)
|
)
|
||||||
|
|
||||||
|
dependency_groups = dependency_groups or []
|
||||||
try:
|
try:
|
||||||
if (include_runtime or toxenv) and not use_build_system:
|
if (include_runtime or toxenv or read_pyproject_dependencies) and not use_build_system:
|
||||||
raise ValueError('-N option cannot be used in combination with -r, -e, -t, -x options')
|
raise ValueError('-N option cannot be used in combination with -r, -e, -t, -x, -p options')
|
||||||
if requirement_files:
|
if requirement_files:
|
||||||
for req_file in requirement_files:
|
for req_file in requirement_files:
|
||||||
requirements.extend(
|
requirements.extend(
|
||||||
@ -457,8 +604,12 @@ def generate_requires(
|
|||||||
if toxenv:
|
if toxenv:
|
||||||
include_runtime = True
|
include_runtime = True
|
||||||
generate_tox_requirements(toxenv, requirements)
|
generate_tox_requirements(toxenv, requirements)
|
||||||
|
dependency_groups.extend(tox_dependency_groups(toxenv))
|
||||||
|
if dependency_groups:
|
||||||
|
generate_dependency_groups(dependency_groups, requirements)
|
||||||
if include_runtime:
|
if include_runtime:
|
||||||
generate_run_requirements(backend, requirements, build_wheel=build_wheel, wheeldir=wheeldir)
|
generate_run_requirements(backend, requirements, build_wheel=build_wheel,
|
||||||
|
read_pyproject_dependencies=read_pyproject_dependencies, wheeldir=wheeldir)
|
||||||
except EndPass:
|
except EndPass:
|
||||||
return
|
return
|
||||||
finally:
|
finally:
|
||||||
@ -485,7 +636,7 @@ def main(argv):
|
|||||||
help=argparse.SUPPRESS,
|
help=argparse.SUPPRESS,
|
||||||
)
|
)
|
||||||
parser.add_argument(
|
parser.add_argument(
|
||||||
'-p', '--python3_pkgversion', metavar='PYTHON3_PKGVERSION',
|
'--python3_pkgversion', metavar='PYTHON3_PKGVERSION',
|
||||||
default="3", help=argparse.SUPPRESS,
|
default="3", help=argparse.SUPPRESS,
|
||||||
)
|
)
|
||||||
parser.add_argument(
|
parser.add_argument(
|
||||||
@ -500,6 +651,11 @@ def main(argv):
|
|||||||
help='comma separated list of "extras" for runtime requirements '
|
help='comma separated list of "extras" for runtime requirements '
|
||||||
'(e.g. -x testing,feature-x) (implies --runtime, can be repeated)',
|
'(e.g. -x testing,feature-x) (implies --runtime, can be repeated)',
|
||||||
)
|
)
|
||||||
|
parser.add_argument(
|
||||||
|
'-g', '--dependency-groups', metavar='GROUPS', action='append',
|
||||||
|
help='comma separated list of dependency groups (PEP 735) for requirements '
|
||||||
|
'(e.g. -g tests,docs) (can be repeated)',
|
||||||
|
)
|
||||||
parser.add_argument(
|
parser.add_argument(
|
||||||
'-t', '--tox', action='store_true',
|
'-t', '--tox', action='store_true',
|
||||||
help=('generate test tequirements from tox environment '
|
help=('generate test tequirements from tox environment '
|
||||||
@ -515,6 +671,11 @@ def main(argv):
|
|||||||
help=('Generate run-time requirements by building the wheel '
|
help=('Generate run-time requirements by building the wheel '
|
||||||
'(useful for build backends without the prepare_metadata_for_build_wheel hook)'),
|
'(useful for build backends without the prepare_metadata_for_build_wheel hook)'),
|
||||||
)
|
)
|
||||||
|
parser.add_argument(
|
||||||
|
'-p', '--read-pyproject-dependencies', action='store_true', default=False,
|
||||||
|
help=('Generate dependencies from [project] table of pyproject.toml '
|
||||||
|
'instead of calling prepare_metadata_for_build_wheel hook)'),
|
||||||
|
)
|
||||||
parser.add_argument(
|
parser.add_argument(
|
||||||
'-R', '--no-runtime', action='store_false', dest='runtime',
|
'-R', '--no-runtime', action='store_false', dest='runtime',
|
||||||
help="Don't generate run-time requirements (implied by -N)",
|
help="Don't generate run-time requirements (implied by -N)",
|
||||||
@ -563,10 +724,12 @@ def main(argv):
|
|||||||
wheeldir=args.wheeldir,
|
wheeldir=args.wheeldir,
|
||||||
toxenv=args.toxenv,
|
toxenv=args.toxenv,
|
||||||
extras=args.extras,
|
extras=args.extras,
|
||||||
|
dependency_groups=args.dependency_groups,
|
||||||
generate_extras=args.generate_extras,
|
generate_extras=args.generate_extras,
|
||||||
python3_pkgversion=args.python3_pkgversion,
|
python3_pkgversion=args.python3_pkgversion,
|
||||||
requirement_files=args.requirement_files,
|
requirement_files=args.requirement_files,
|
||||||
use_build_system=args.use_build_system,
|
use_build_system=args.use_build_system,
|
||||||
|
read_pyproject_dependencies=args.read_pyproject_dependencies,
|
||||||
output=args.output,
|
output=args.output,
|
||||||
config_settings=parse_config_settings_args(args.config_settings),
|
config_settings=parse_config_settings_args(args.config_settings),
|
||||||
)
|
)
|
||||||
|
File diff suppressed because it is too large
Load Diff
@ -2,6 +2,7 @@ import argparse
|
|||||||
import fnmatch
|
import fnmatch
|
||||||
import json
|
import json
|
||||||
import os
|
import os
|
||||||
|
import re
|
||||||
|
|
||||||
from collections import defaultdict
|
from collections import defaultdict
|
||||||
from keyword import iskeyword
|
from keyword import iskeyword
|
||||||
@ -11,9 +12,15 @@ from importlib.metadata import Distribution
|
|||||||
|
|
||||||
# From RPM's build/files.c strtokWithQuotes delim argument
|
# From RPM's build/files.c strtokWithQuotes delim argument
|
||||||
RPM_FILES_DELIMETERS = ' \n\t'
|
RPM_FILES_DELIMETERS = ' \n\t'
|
||||||
|
RPM_GLOB_SYMBOLS = '[]{}*?!'
|
||||||
|
# Combined for escape_rpm_path_4_19()
|
||||||
|
RPM_SPECIAL_SYMBOLS = RPM_FILES_DELIMETERS + RPM_GLOB_SYMBOLS + '"' + "\\"
|
||||||
|
RPM_ESCAPE_REGEX = re.compile(f"([{re.escape(RPM_SPECIAL_SYMBOLS)}])")
|
||||||
|
|
||||||
# See the comment in the macro that wraps this script
|
# See the comment in the macro that wraps this script
|
||||||
RPM_PERCENTAGES_COUNT = int(os.getenv('RPM_PERCENTAGES_COUNT', '2'))
|
RPM_FILES_ESCAPE = os.getenv('RPM_FILES_ESCAPE', '4.19')
|
||||||
|
|
||||||
|
PYCACHED_SUFFIX = '{,.opt-?}.pyc'
|
||||||
|
|
||||||
# RPM hardcodes the lists of manpage extensions and directories,
|
# RPM hardcodes the lists of manpage extensions and directories,
|
||||||
# so we have to maintain separate ones :(
|
# so we have to maintain separate ones :(
|
||||||
@ -118,8 +125,9 @@ def pycached(script, python_version):
|
|||||||
"""
|
"""
|
||||||
assert script.suffix == ".py"
|
assert script.suffix == ".py"
|
||||||
pyver = "".join(python_version.split(".")[:2])
|
pyver = "".join(python_version.split(".")[:2])
|
||||||
pycname = f"{script.stem}.cpython-{pyver}{{,.opt-?}}.pyc"
|
pycname = f"{script.stem}.cpython-{pyver}{PYCACHED_SUFFIX}"
|
||||||
pyc = pycache_dir(script) / pycname
|
pyc = pycache_dir(script) / pycname
|
||||||
|
pyc.glob_suffix_len = len(PYCACHED_SUFFIX)
|
||||||
return [script, pyc]
|
return [script, pyc]
|
||||||
|
|
||||||
|
|
||||||
@ -212,10 +220,12 @@ def normalize_manpage_filename(prefix, path):
|
|||||||
if fnmatch.fnmatch(str(path.parent), mandir) and path.name != "dir":
|
if fnmatch.fnmatch(str(path.parent), mandir) and path.name != "dir":
|
||||||
# "abc.1.gz2" -> "abc.1*"
|
# "abc.1.gz2" -> "abc.1*"
|
||||||
if path.suffix[1:] in MANPAGE_EXTENSIONS:
|
if path.suffix[1:] in MANPAGE_EXTENSIONS:
|
||||||
return BuildrootPath(path.parent / (path.stem + "*"))
|
path = BuildrootPath(path.parent / (path.stem + "*"))
|
||||||
# "abc.1 -> abc.1*"
|
# "abc.1 -> abc.1*"
|
||||||
else:
|
else:
|
||||||
return BuildrootPath(path.parent / (path.name + "*"))
|
path = BuildrootPath(path.parent / (path.name + "*"))
|
||||||
|
path.glob_suffix_len = 1
|
||||||
|
return path
|
||||||
else:
|
else:
|
||||||
return path
|
return path
|
||||||
|
|
||||||
@ -424,60 +434,139 @@ def classify_paths(
|
|||||||
return paths
|
return paths
|
||||||
|
|
||||||
|
|
||||||
def escape_rpm_path(path):
|
def escape_rpm_path_4_19(path):
|
||||||
|
r"""
|
||||||
|
Escape special characters in string-paths or BuildrootPaths, RPM >= 4.19
|
||||||
|
|
||||||
|
E.g. a space in path otherwise makes RPM think it's multiple paths,
|
||||||
|
unless we escape it.
|
||||||
|
Or a literal % symbol in path might be expanded as a macro if not escaped by %%.
|
||||||
|
|
||||||
|
See https://github.com/rpm-software-management/rpm/pull/2103
|
||||||
|
and https://github.com/rpm-software-management/rpm/pull/2206
|
||||||
|
|
||||||
|
If the path ends with a glob produced by our other functions,
|
||||||
|
we cannot escape that part.
|
||||||
|
The BuildrootPath.glob_suffix_len attribute is used to indicate such globs.
|
||||||
|
When such suffix exists, it is not escaped.
|
||||||
|
|
||||||
|
|
||||||
|
Examples:
|
||||||
|
|
||||||
|
>>> escape_rpm_path_4_19(BuildrootPath('/usr/lib/python3.9/site-packages/setuptools'))
|
||||||
|
'/usr/lib/python3.9/site-packages/setuptools'
|
||||||
|
|
||||||
|
>>> escape_rpm_path_4_19('/usr/lib/python3.9/site-packages/setuptools/script (dev).tmpl')
|
||||||
|
'/usr/lib/python3.9/site-packages/setuptools/script\\ (dev).tmpl'
|
||||||
|
|
||||||
|
>>> escape_rpm_path_4_19('/usr/share/data/100%valid.path')
|
||||||
|
'/usr/share/data/100%%valid.path'
|
||||||
|
|
||||||
|
>>> escape_rpm_path_4_19('/usr/share/data/100 % valid.path')
|
||||||
|
'/usr/share/data/100\\ %%\\ valid.path'
|
||||||
|
|
||||||
|
>>> escape_rpm_path_4_19('/usr/share/data/1000 %% valid.path')
|
||||||
|
'/usr/share/data/1000\\ %%%%\\ valid.path'
|
||||||
|
|
||||||
|
>>> escape_rpm_path_4_19('/usr/share/data/spaces and "quotes" and ?')
|
||||||
|
'/usr/share/data/spaces\\ and\\ \\"quotes\\"\\ and\\ \\?'
|
||||||
|
|
||||||
|
>>> escape_rpm_path_4_19('/usr/share/data/spaces and [square brackets]')
|
||||||
|
'/usr/share/data/spaces\\ and\\ \\[square\\ brackets\\]'
|
||||||
|
|
||||||
|
>>> path = BuildrootPath('/whatever/__pycache__/bar.cpython-38{,.opt-?}.pyc')
|
||||||
|
>>> path.glob_suffix_len = len('{,.opt-?}.pyc')
|
||||||
|
>>> escape_rpm_path_4_19(path)
|
||||||
|
'/whatever/__pycache__/bar.cpython-38{,.opt-?}.pyc'
|
||||||
|
|
||||||
|
>>> path = BuildrootPath('/spa ces/__pycache__/bar.cpython-38{,.opt-?}.pyc')
|
||||||
|
>>> path.glob_suffix_len = len('{,.opt-?}.pyc')
|
||||||
|
>>> escape_rpm_path_4_19(path)
|
||||||
|
'/spa\\ ces/__pycache__/bar.cpython-38{,.opt-?}.pyc'
|
||||||
|
|
||||||
|
>>> path = BuildrootPath('/usr/man/man5/ipykernel.5*')
|
||||||
|
>>> path.glob_suffix_len = 1
|
||||||
|
>>> escape_rpm_path_4_19(path)
|
||||||
|
'/usr/man/man5/ipykernel.5*'
|
||||||
"""
|
"""
|
||||||
Escape special characters in string-paths or BuildrootPaths
|
glob_suffix_len = getattr(path, "glob_suffix_len", 0)
|
||||||
|
suffix = ""
|
||||||
|
path = str(path)
|
||||||
|
if glob_suffix_len:
|
||||||
|
suffix = path[-glob_suffix_len:]
|
||||||
|
path = path[:-glob_suffix_len]
|
||||||
|
if "%" in path:
|
||||||
|
path = path.replace("%", "%%")
|
||||||
|
# Prepend all matched/special characters (\1) with a backslash (escaped, hence \\):
|
||||||
|
return RPM_ESCAPE_REGEX.sub(r'\\\1', path) + suffix
|
||||||
|
|
||||||
|
|
||||||
|
def escape_rpm_path_4_18(path):
|
||||||
|
"""
|
||||||
|
Escape special characters in string-paths or BuildrootPaths, RPM < 4.19
|
||||||
|
|
||||||
E.g. a space in path otherwise makes RPM think it's multiple paths,
|
E.g. a space in path otherwise makes RPM think it's multiple paths,
|
||||||
unless we put it in "quotes".
|
unless we put it in "quotes".
|
||||||
Or a literal % symbol in path might be expanded as a macro if not escaped.
|
Or a literal % symbol in path might be expanded as a macro if not escaped.
|
||||||
|
|
||||||
Due to limitations in RPM,
|
Due to limitations in RPM < 4.19,
|
||||||
some paths with spaces and other special characters are not supported.
|
some paths with spaces and other special characters are not supported.
|
||||||
|
|
||||||
|
See this thread http://lists.rpm.org/pipermail/rpm-list/2021-June/002048.html
|
||||||
|
|
||||||
Examples:
|
Examples:
|
||||||
|
|
||||||
>>> escape_rpm_path(BuildrootPath('/usr/lib/python3.9/site-packages/setuptools'))
|
>>> escape_rpm_path_4_18(BuildrootPath('/usr/lib/python3.9/site-packages/setuptools'))
|
||||||
'/usr/lib/python3.9/site-packages/setuptools'
|
'/usr/lib/python3.9/site-packages/setuptools'
|
||||||
|
|
||||||
>>> escape_rpm_path('/usr/lib/python3.9/site-packages/setuptools/script (dev).tmpl')
|
>>> escape_rpm_path_4_18('/usr/lib/python3.9/site-packages/setuptools/script (dev).tmpl')
|
||||||
'"/usr/lib/python3.9/site-packages/setuptools/script (dev).tmpl"'
|
'"/usr/lib/python3.9/site-packages/setuptools/script (dev).tmpl"'
|
||||||
|
|
||||||
>>> escape_rpm_path('/usr/share/data/100%valid.path')
|
>>> escape_rpm_path_4_18('/usr/share/data/100%valid.path')
|
||||||
'/usr/share/data/100%%valid.path'
|
'/usr/share/data/100%%%%%%%%valid.path'
|
||||||
|
|
||||||
>>> escape_rpm_path('/usr/share/data/100 % valid.path')
|
>>> escape_rpm_path_4_18('/usr/share/data/100 % valid.path')
|
||||||
'"/usr/share/data/100 %% valid.path"'
|
'"/usr/share/data/100 %%%%%%%% valid.path"'
|
||||||
|
|
||||||
>>> escape_rpm_path('/usr/share/data/1000 %% valid.path')
|
>>> escape_rpm_path_4_18('/usr/share/data/1000 %% valid.path')
|
||||||
'"/usr/share/data/1000 %%%% valid.path"'
|
'"/usr/share/data/1000 %%%%%%%%%%%%%%%% valid.path"'
|
||||||
|
|
||||||
>>> escape_rpm_path('/usr/share/data/spaces and "quotes"')
|
>>> escape_rpm_path_4_18('/usr/share/data/spaces and "quotes"')
|
||||||
Traceback (most recent call last):
|
Traceback (most recent call last):
|
||||||
...
|
...
|
||||||
NotImplementedError: ...
|
NotImplementedError: ...
|
||||||
|
|
||||||
>>> escape_rpm_path('/usr/share/data/spaces and [square brackets]')
|
>>> escape_rpm_path_4_18('/usr/share/data/spaces and [square brackets]')
|
||||||
Traceback (most recent call last):
|
Traceback (most recent call last):
|
||||||
...
|
...
|
||||||
NotImplementedError: ...
|
NotImplementedError: ...
|
||||||
"""
|
"""
|
||||||
orig_path = path = str(path)
|
orig_path = path = str(path)
|
||||||
if "%" in path:
|
if "%" in path:
|
||||||
path = path.replace("%", "%" * RPM_PERCENTAGES_COUNT)
|
# Escaping an actual percentage sign in path by 8 signs
|
||||||
|
# has been verified in RPM 4.16 and 4.17:
|
||||||
|
path = path.replace("%", "%" * 8)
|
||||||
if any(symbol in path for symbol in RPM_FILES_DELIMETERS):
|
if any(symbol in path for symbol in RPM_FILES_DELIMETERS):
|
||||||
if '"' in path:
|
if '"' in path:
|
||||||
# As far as we know, RPM cannot list such file individually
|
# As far as we know, RPM < 4.19 cannot list such file individually
|
||||||
# See this thread http://lists.rpm.org/pipermail/rpm-list/2021-June/002048.html
|
# See this thread http://lists.rpm.org/pipermail/rpm-list/2021-June/002048.html
|
||||||
raise NotImplementedError(f'" symbol in path with spaces is not supported by %pyproject_save_files: {orig_path!r}')
|
raise NotImplementedError(f'" symbol in path with spaces is not supported by %pyproject_save_files on RPM < 4.19: {orig_path!r}')
|
||||||
if "[" in path or "]" in path:
|
if "[" in path or "]" in path:
|
||||||
# See https://bugzilla.redhat.com/show_bug.cgi?id=1990879
|
# See https://bugzilla.redhat.com/show_bug.cgi?id=1990879
|
||||||
# and https://github.com/rpm-software-management/rpm/issues/1749
|
# and https://github.com/rpm-software-management/rpm/issues/1749
|
||||||
raise NotImplementedError(f'[ or ] symbol in path with spaces is not supported by %pyproject_save_files: {orig_path!r}')
|
raise NotImplementedError(f'[ or ] symbol in path with spaces is not supported by %pyproject_save_files on RPM < 4.19: {orig_path!r}')
|
||||||
return f'"{path}"'
|
return f'"{path}"'
|
||||||
return path
|
return path
|
||||||
|
|
||||||
|
|
||||||
|
if RPM_FILES_ESCAPE == "4.19":
|
||||||
|
escape_rpm_path = escape_rpm_path_4_19
|
||||||
|
elif RPM_FILES_ESCAPE == "4.18":
|
||||||
|
escape_rpm_path = escape_rpm_path_4_18
|
||||||
|
else:
|
||||||
|
raise RuntimeError("RPM_FILES_ESCAPE must be 4.18 or 4.19")
|
||||||
|
|
||||||
|
|
||||||
def generate_file_list(paths_dict, module_globs, include_others=False):
|
def generate_file_list(paths_dict, module_globs, include_others=False):
|
||||||
"""
|
"""
|
||||||
This function takes the classified paths_dict and turns it into lines
|
This function takes the classified paths_dict and turns it into lines
|
||||||
|
@ -457,7 +457,7 @@ classified:
|
|||||||
- /usr/lib/python3.7/site-packages/comic2pdf-3.1.0.dist-info/top_level.txt
|
- /usr/lib/python3.7/site-packages/comic2pdf-3.1.0.dist-info/top_level.txt
|
||||||
- /usr/lib/python3.7/site-packages/comic2pdf-3.1.0.dist-info/zip-safe
|
- /usr/lib/python3.7/site-packages/comic2pdf-3.1.0.dist-info/zip-safe
|
||||||
licenses: []
|
licenses: []
|
||||||
modules: []
|
modules: {}
|
||||||
other:
|
other:
|
||||||
files:
|
files:
|
||||||
- /usr/bin/comic2pdf.py
|
- /usr/bin/comic2pdf.py
|
||||||
|
@ -6,16 +6,36 @@ import pytest
|
|||||||
import setuptools
|
import setuptools
|
||||||
import yaml
|
import yaml
|
||||||
|
|
||||||
from pyproject_buildrequires import generate_requires
|
from pyproject_buildrequires import generate_requires, load_pyproject
|
||||||
|
|
||||||
SETUPTOOLS_VERSION = packaging.version.parse(setuptools.__version__)
|
SETUPTOOLS_VERSION = packaging.version.parse(setuptools.__version__)
|
||||||
SETUPTOOLS_60 = SETUPTOOLS_VERSION >= packaging.version.parse('60')
|
SETUPTOOLS_60 = SETUPTOOLS_VERSION >= packaging.version.parse('60')
|
||||||
|
|
||||||
|
try:
|
||||||
|
import tox
|
||||||
|
except ImportError:
|
||||||
|
TOX_4_22 = False
|
||||||
|
else:
|
||||||
|
TOX_VERSION = packaging.version.parse(tox.__version__)
|
||||||
|
TOX_4_22 = TOX_VERSION >= packaging.version.parse('4.22')
|
||||||
|
|
||||||
testcases = {}
|
testcases = {}
|
||||||
with Path(__file__).parent.joinpath('pyproject_buildrequires_testcases.yaml').open() as f:
|
with Path(__file__).parent.joinpath('pyproject_buildrequires_testcases.yaml').open() as f:
|
||||||
testcases = yaml.safe_load(f)
|
testcases = yaml.safe_load(f)
|
||||||
|
|
||||||
|
|
||||||
|
@pytest.fixture(autouse=True)
|
||||||
|
def clear_pyproject_data():
|
||||||
|
"""
|
||||||
|
Clear pyproject data before each test.
|
||||||
|
In reality we build one RPM package at a time, so we can keep the once-loaded
|
||||||
|
pyproject.toml contents.
|
||||||
|
When testing, the cached data would leak the once-loaded data to all the
|
||||||
|
following test cases.
|
||||||
|
"""
|
||||||
|
load_pyproject.cache_clear()
|
||||||
|
|
||||||
|
|
||||||
@pytest.mark.parametrize('case_name', testcases)
|
@pytest.mark.parametrize('case_name', testcases)
|
||||||
def test_data(case_name, capfd, tmp_path, monkeypatch):
|
def test_data(case_name, capfd, tmp_path, monkeypatch):
|
||||||
case = testcases[case_name]
|
case = testcases[case_name]
|
||||||
@ -51,6 +71,7 @@ def test_data(case_name, capfd, tmp_path, monkeypatch):
|
|||||||
requirement_files = case.get('requirement_files', [])
|
requirement_files = case.get('requirement_files', [])
|
||||||
requirement_files = [open(f) for f in requirement_files]
|
requirement_files = [open(f) for f in requirement_files]
|
||||||
use_build_system = case.get('use_build_system', True)
|
use_build_system = case.get('use_build_system', True)
|
||||||
|
read_pyproject_dependencies = case.get('read_pyproject_dependencies', False)
|
||||||
try:
|
try:
|
||||||
generate_requires(
|
generate_requires(
|
||||||
get_installed_version=get_installed_version,
|
get_installed_version=get_installed_version,
|
||||||
@ -58,10 +79,12 @@ def test_data(case_name, capfd, tmp_path, monkeypatch):
|
|||||||
build_wheel=case.get('build_wheel', False),
|
build_wheel=case.get('build_wheel', False),
|
||||||
wheeldir=str(wheeldir),
|
wheeldir=str(wheeldir),
|
||||||
extras=case.get('extras', []),
|
extras=case.get('extras', []),
|
||||||
|
dependency_groups=case.get('dependency_groups', []),
|
||||||
toxenv=case.get('toxenv', None),
|
toxenv=case.get('toxenv', None),
|
||||||
generate_extras=case.get('generate_extras', False),
|
generate_extras=case.get('generate_extras', False),
|
||||||
requirement_files=requirement_files,
|
requirement_files=requirement_files,
|
||||||
use_build_system=use_build_system,
|
use_build_system=use_build_system,
|
||||||
|
read_pyproject_dependencies=read_pyproject_dependencies,
|
||||||
output=output,
|
output=output,
|
||||||
config_settings=case.get('config_settings'),
|
config_settings=case.get('config_settings'),
|
||||||
)
|
)
|
||||||
|
@ -25,6 +25,21 @@ TEST_RECORDS = yaml_data["records"]
|
|||||||
TEST_METADATAS = yaml_data["metadata"]
|
TEST_METADATAS = yaml_data["metadata"]
|
||||||
|
|
||||||
|
|
||||||
|
# insert glob_suffix_len for .pyc files and man pages globs
|
||||||
|
for paths_dict in EXPECTED_DICT.values():
|
||||||
|
for modules in paths_dict["modules"].values():
|
||||||
|
for module in modules:
|
||||||
|
for idx, file in enumerate(module["files"]):
|
||||||
|
if file.endswith(".pyc"):
|
||||||
|
module["files"][idx] = BuildrootPath(file)
|
||||||
|
module["files"][idx].glob_suffix_len = len("{,.opt-?}.pyc")
|
||||||
|
if "other" in paths_dict and "files" in paths_dict["other"]:
|
||||||
|
for idx, file in enumerate(paths_dict["other"]["files"]):
|
||||||
|
if file.endswith("*"):
|
||||||
|
paths_dict["other"]["files"][idx] = BuildrootPath(file)
|
||||||
|
paths_dict["other"]["files"][idx].glob_suffix_len = len("*")
|
||||||
|
|
||||||
|
|
||||||
@pytest.fixture
|
@pytest.fixture
|
||||||
def tldr_root(tmp_path):
|
def tldr_root(tmp_path):
|
||||||
prepare_pyproject_record(tmp_path, package="tldr")
|
prepare_pyproject_record(tmp_path, package="tldr")
|
||||||
|
@ -1,5 +1,6 @@
|
|||||||
Name: pyproject-rpm-macros
|
Name: pyproject-rpm-macros
|
||||||
Summary: RPM macros for PEP 517 Python packages
|
Summary: RPM macros for PEP 517 Python packages
|
||||||
|
# SPDX
|
||||||
License: MIT
|
License: MIT
|
||||||
|
|
||||||
%bcond tests 1
|
%bcond tests 1
|
||||||
@ -13,7 +14,7 @@ License: MIT
|
|||||||
# Increment Y and reset Z when new macros or features are added
|
# Increment Y and reset Z when new macros or features are added
|
||||||
# Increment Z when this is a bugfix or a cosmetic change
|
# Increment Z when this is a bugfix or a cosmetic change
|
||||||
# Dropping support for EOL Fedoras is *not* considered a breaking change
|
# Dropping support for EOL Fedoras is *not* considered a breaking change
|
||||||
Version: 1.12.0
|
Version: 1.16.2
|
||||||
Release: 1%{?dist}
|
Release: 1%{?dist}
|
||||||
|
|
||||||
# Macro files
|
# Macro files
|
||||||
@ -95,7 +96,8 @@ Requires: /usr/bin/sed
|
|||||||
# This package requires the %%generate_buildrequires functionality.
|
# This package requires the %%generate_buildrequires functionality.
|
||||||
# It has been introduced in RPM 4.15 (4.14.90 is the alpha of 4.15).
|
# It has been introduced in RPM 4.15 (4.14.90 is the alpha of 4.15).
|
||||||
# What we need is rpmlib(DynamicBuildRequires), but that is impossible to (Build)Require.
|
# What we need is rpmlib(DynamicBuildRequires), but that is impossible to (Build)Require.
|
||||||
Requires: (rpm-build >= 4.14.90 if rpm-build)
|
# Also, we need to avoid 4.19.90..4.19.91-7 due to rhbz#2284187
|
||||||
|
Requires: ((rpm-build >= 4.14.90 with (rpm-build < 4.19.90 or rpm-build >= 4.19.91-8)) if rpm-build)
|
||||||
BuildRequires: rpm-build >= 4.14.90
|
BuildRequires: rpm-build >= 4.14.90
|
||||||
|
|
||||||
%description
|
%description
|
||||||
@ -194,6 +196,47 @@ export HOSTNAME="rpmbuild" # to speedup tox in network-less mock, see rhbz#1856
|
|||||||
|
|
||||||
|
|
||||||
%changelog
|
%changelog
|
||||||
|
* Wed Nov 13 2024 Miro Hrončok <mhroncok@redhat.com> - 1.16.2-1
|
||||||
|
- Fix one remaining test for setuptools 70+
|
||||||
|
|
||||||
|
* Thu Nov 07 2024 Miro Hrončok <miro@hroncok.cz> - 1.16.1-1
|
||||||
|
- Support for setuptools 70+
|
||||||
|
- wheel is no longer generated as a dependency of the default build system
|
||||||
|
|
||||||
|
* Mon Nov 04 2024 Miro Hrončok <mhroncok@redhat.com> - 1.16.0-1
|
||||||
|
- %%pyproject_buildrequires: Add support for dependency groups (PEP 735), via the -g flag
|
||||||
|
- This is implied when used tox testenvs depend on dependency groups (requires tox 4.22+)
|
||||||
|
- Fixes: rhbz#2318849
|
||||||
|
|
||||||
|
* Thu Oct 03 2024 Karolina Surma <ksurma@redhat.com> - 1.15.1-1
|
||||||
|
- Fix handling of self-referencing extras when reading pyproject.toml
|
||||||
|
|
||||||
|
* Tue Sep 17 2024 Python Maint <python-maint@redhat.com> - 1.15.0-1
|
||||||
|
- Add a possibility to read runtime requirements from pyproject.toml [project] table
|
||||||
|
- Fixes: rhbz#2261939
|
||||||
|
- Don't generate a dependency on pip when %%pyproject_buildrequires -N is used
|
||||||
|
- Fixes: rhbz#2294510
|
||||||
|
- Even when %%_auto_set_build_flags is disabled, set all compiler flags when building wheels
|
||||||
|
- Fixes: rhbz#2293616
|
||||||
|
|
||||||
|
* Tue Jul 23 2024 Miro Hrončok <mhroncok@redhat.com> - 1.14.0-1
|
||||||
|
- Add a provisional RPM Declarative Buildsystem (RPM 4.20+)
|
||||||
|
|
||||||
|
* Fri Jul 19 2024 Fedora Release Engineering <releng@fedoraproject.org> - 1.13.0-2
|
||||||
|
- Rebuilt for https://fedoraproject.org/wiki/Fedora_41_Mass_Rebuild
|
||||||
|
|
||||||
|
* Tue Jul 02 2024 Miro Hrončok <mhroncok@redhat.com> - 1.13.0-1
|
||||||
|
- Properly escape weird characters from paths in %%{pyproject_files} (RPM 4.19+ only)
|
||||||
|
- Revert the temporary workaround for RPM 4.20 alpha 2 leaking \x1f (unit separators)
|
||||||
|
- Fixes: rhbz#1990879
|
||||||
|
|
||||||
|
* Tue Jun 25 2024 Cristian Le <fedora@lecris.me> - 1.12.2-1
|
||||||
|
- %%pyproject_extras_subpkg: Allow passing -a or -A to %%python_extras_subpkg
|
||||||
|
|
||||||
|
* Tue Jun 04 2024 Miro Hrončok <mhroncok@redhat.com> - 1.12.1-1
|
||||||
|
- Add a temporary workaround for RPM 4.20 alpha 2 leaking \x1f (unit separators)
|
||||||
|
- Related: rhbz#2284187
|
||||||
|
|
||||||
* Fri Jan 26 2024 Miro Hrončok <miro@hroncok.cz> - 1.12.0-1
|
* Fri Jan 26 2024 Miro Hrončok <miro@hroncok.cz> - 1.12.0-1
|
||||||
- Namespace pyproject-rpm-macros generated text files with %%{python3_pkgversion}
|
- Namespace pyproject-rpm-macros generated text files with %%{python3_pkgversion}
|
||||||
- That way, a single-spec can be used to build packages for multiple Python versions
|
- That way, a single-spec can be used to build packages for multiple Python versions
|
||||||
|
Loading…
Reference in New Issue
Block a user