You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
223 lines
11 KiB
223 lines
11 KiB
# This is a backward-compatible suffix used in all pyproject-rpm-macros directories |
|
# For the main Python it's empty, for all others it's "-3.X" |
|
%_pyproject_files_pkgversion %{expr:"%{python3_pkgversion}" != "3" ? "-%{python3_pkgversion}" : ""} |
|
|
|
# This is a directory where wheels are stored and installed from, absolute |
|
%_pyproject_wheeldir %{_builddir}%{?buildsubdir:/%{buildsubdir}}/pyproject-wheeldir%{_pyproject_files_pkgversion} |
|
|
|
# This is a directory used as TMPDIR, where pip copies sources to and builds from, relative to PWD |
|
# For proper debugsource packages, we create TMPDIR within PWD |
|
# See https://github.com/pypa/pip/issues/7555#issuecomment-595180864 |
|
# |
|
# This will be used in debugsource package paths (applies to extension modules only) |
|
# NB: pytest collects tests from here if not hidden |
|
# https://docs.pytest.org/en/latest/reference.html#confval-norecursedirs |
|
%_pyproject_builddir %{_builddir}%{?buildsubdir:/%{buildsubdir}}/.pyproject-builddir%{_pyproject_files_pkgversion} |
|
|
|
# We prefix all created files with this value to make them unique |
|
# Ideally, we would put them into %%{buildsubdir}, but that value changes during the spec |
|
# The used value is similar to the one used to define the default %%buildroot |
|
%_pyproject_files_prefix %{name}-%{version}-%{release}.%{_arch}%{_pyproject_files_pkgversion} |
|
|
|
%pyproject_files %{_builddir}/%{_pyproject_files_prefix}-pyproject-files |
|
%_pyproject_modules %{_builddir}/%{_pyproject_files_prefix}-pyproject-modules |
|
%_pyproject_ghost_distinfo %{_builddir}/%{_pyproject_files_prefix}-pyproject-ghost-distinfo |
|
%_pyproject_record %{_builddir}/%{_pyproject_files_prefix}-pyproject-record |
|
%_pyproject_buildrequires %{_builddir}/%{_pyproject_files_prefix}-pyproject-buildrequires |
|
|
|
# Avoid leaking %%{_pyproject_builddir} to pytest collection |
|
# https://bugzilla.redhat.com/show_bug.cgi?id=1935212 |
|
# The value is read and used by the %%pytest and %%tox macros: |
|
%_set_pytest_addopts %global __pytest_addopts --ignore=%{_pyproject_builddir} |
|
|
|
%pyproject_wheel(C:) %{expand:\\\ |
|
%_set_pytest_addopts |
|
mkdir -p "%{_pyproject_builddir}" |
|
CFLAGS="${CFLAGS:-${RPM_OPT_FLAGS}}" LDFLAGS="${LDFLAGS:-${RPM_LD_FLAGS}}" TMPDIR="%{_pyproject_builddir}" \\\ |
|
%{__python3} -Bs %{_rpmconfigdir}/redhat/pyproject_wheel.py %{?**} %{_pyproject_wheeldir} |
|
} |
|
|
|
|
|
%pyproject_build_lib %{!?__pyproject_build_lib_warned:%{warn:The %%{pyproject_build_lib} macro is deprecated. |
|
It only works with setuptools and is not build-backend-agnostic. |
|
The macro is not scheduled for removal, but there is a possibility of incompatibilities with future versions of setuptools. |
|
As a replacement for the macro for the setuptools backend on Fedora 37+, you can use $PWD/build/lib for pure Python packages, |
|
or $PWD/build/lib.%%{python3_platform}-cpython-%%{python3_version_nodots} for packages with extension modules. |
|
Other build backends and older distributions may need different paths. |
|
See https://lists.fedoraproject.org/archives/list/python-devel@lists.fedoraproject.org/thread/HMLOPAU3RZLXD4BOJHTIPKI3I4U6U7OE/ for details. |
|
}%global __pyproject_build_lib_warned 1}%{expand:\\\ |
|
$( |
|
pyproject_build_lib=() |
|
if [ -d build/lib.%{python3_platform}-cpython-%{python3_version_nodots} ]; then |
|
pyproject_build_lib+=( "${PWD}/build/lib.%{python3_platform}-cpython-%{python3_version_nodots}" ) |
|
fi |
|
if [ -d build/lib.%{python3_platform}-%{python3_version} ]; then |
|
pyproject_build_lib+=( "${PWD}/build/lib.%{python3_platform}-%{python3_version}" ) |
|
fi |
|
if [ -d build/lib ]; then |
|
pyproject_build_lib+=( "${PWD}/build/lib" ) |
|
fi |
|
for directory in $(find "%{_pyproject_builddir}" -type d -wholename "%{_pyproject_builddir}/pip-req-build-*/build/lib.%{python3_platform}-%{python3_version}" 2>/dev/null); do |
|
pyproject_build_lib+=( "${directory}" ) |
|
done |
|
for directory in $(find "%{_pyproject_builddir}" -type d -wholename "%{_pyproject_builddir}/pip-req-build-*/build/lib" 2>/dev/null); do |
|
pyproject_build_lib+=( "${directory}" ) |
|
done |
|
echo $(IFS=:; echo "${pyproject_build_lib[*]}") |
|
)} |
|
|
|
|
|
%pyproject_install() %{expand:\\\ |
|
specifier=$(ls %{_pyproject_wheeldir}/*.whl | xargs basename --multiple | sed -E 's/([^-]+)-([^-]+)-.+\\\.whl/\\\1==\\\2/') |
|
if [ -z $specifier ]; then |
|
echo 'ERROR: %%%%pyproject_install found no wheel in %%%%{_pyproject_wheeldir} %{_pyproject_wheeldir}' >&2 |
|
exit 1 |
|
fi |
|
TMPDIR="%{_pyproject_builddir}" %{__python3} -m pip install --root %{buildroot} --prefix %{_prefix} --no-deps --disable-pip-version-check --progress-bar off --verbose --ignore-installed --no-warn-script-location --no-index --no-cache-dir --find-links %{_pyproject_wheeldir} $specifier |
|
if [ -d %{buildroot}%{_bindir} ]; then |
|
%py3_shebang_fix %{buildroot}%{_bindir}/* |
|
rm -rfv %{buildroot}%{_bindir}/__pycache__ |
|
fi |
|
rm -f %{_pyproject_ghost_distinfo} |
|
site_dirs=() |
|
# Process %%{python3_sitelib} if exists |
|
if [ -d %{buildroot}%{python3_sitelib} ]; then |
|
site_dirs+=( "%{python3_sitelib}" ) |
|
fi |
|
# Process %%{python3_sitearch} if exists and does not equal to %%{python3_sitelib} |
|
if [ %{buildroot}%{python3_sitearch} != %{buildroot}%{python3_sitelib} ] && [ -d %{buildroot}%{python3_sitearch} ]; then |
|
site_dirs+=( "%{python3_sitearch}" ) |
|
fi |
|
# Process all *.dist-info dirs in sitelib/sitearch |
|
for site_dir in ${site_dirs[@]}; do |
|
for distinfo in %{buildroot}$site_dir/*.dist-info; do |
|
echo "%ghost ${distinfo#%{buildroot}}" >> %{_pyproject_ghost_distinfo} |
|
sed -i 's/pip/rpm/' ${distinfo}/INSTALLER |
|
PYTHONPATH=%{_rpmconfigdir}/redhat \\ |
|
%{__python3} -B %{_rpmconfigdir}/redhat/pyproject_preprocess_record.py \\ |
|
--buildroot %{buildroot} --record ${distinfo}/RECORD --output %{_pyproject_record} |
|
rm -fv ${distinfo}/RECORD |
|
rm -fv ${distinfo}/REQUESTED |
|
done |
|
done |
|
lines=$(wc -l %{_pyproject_ghost_distinfo} | cut -f1 -d" ") |
|
if [ $lines -ne 1 ]; then |
|
echo -e "\\n\\nWARNING: %%%%pyproject_extras_subpkg won't work without explicit -i or -F, found $lines dist-info directories.\\n\\n" >&2 |
|
rm %{_pyproject_ghost_distinfo} # any attempt to use this will fail |
|
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 |
|
%pyproject_extras_subpkg(n:i:f:F) %{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. |
|
# See this thread http://lists.rpm.org/pipermail/rpm-list/2021-June/002048.html |
|
# Since RPM 4.19, 2 signs are needed instead. 4.18.90+ is a pre-release of RPM 4.19. |
|
# On the CI, we build tests/escape_percentages.spec to verify the assumptions. |
|
%pyproject_save_files(lL) %{expand:\\\ |
|
%{expr:v"0%{?rpmversion}" >= v"4.18.90" ? "RPM_PERCENTAGES_COUNT=2" : "RPM_PERCENTAGES_COUNT=8" } \\ |
|
%{__python3} %{_rpmconfigdir}/redhat/pyproject_save_files.py \\ |
|
--output-files "%{pyproject_files}" \\ |
|
--output-modules "%{_pyproject_modules}" \\ |
|
--buildroot "%{buildroot}" \\ |
|
--sitelib "%{python3_sitelib}" \\ |
|
--sitearch "%{python3_sitearch}" \\ |
|
--python-version "%{python3_version}" \\ |
|
--pyproject-record "%{_pyproject_record}" \\ |
|
--prefix "%{_prefix}" \\ |
|
%{**} |
|
} |
|
|
|
# -t - Process only top-level modules |
|
# -e - Exclude the module names matching given glob, may be used repeatedly |
|
%pyproject_check_import(e:t) %{expand:\\\ |
|
if [ ! -f "%{_pyproject_modules}" ]; then |
|
echo 'ERROR: %%%%pyproject_check_import only works when %%%%pyproject_save_files is used' >&2 |
|
exit 1 |
|
fi |
|
%py3_check_import -f "%{_pyproject_modules}" %{?**} |
|
} |
|
|
|
|
|
%_pyproject_check_import_allow_no_modules(e:t) \ |
|
if [ -z "$(cat %{_pyproject_modules})" ]; then\ |
|
echo "No modules to check found, exiting check"\ |
|
else\ |
|
%pyproject_check_import %{?**}\ |
|
fi |
|
|
|
|
|
%default_toxenv py%{python3_version_nodots} |
|
%toxenv %{default_toxenv} |
|
|
|
|
|
# Note: Keep the options in sync with this macro from macros.aaa-pyproject-srpm |
|
%pyproject_buildrequires(rRxtNwe:C:) %{expand:\\\ |
|
%_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 |
|
# see https://bugzilla.redhat.com/show_bug.cgi?id=2097535 |
|
%{?_package_note_flags:%_generate_package_note_file} |
|
%{-R: |
|
%{-r:%{error:The -R and -r options are mutually exclusive}} |
|
%{-x:%{error:The -R and -x options are mutually exclusive}} |
|
%{-e:%{error:The -R and -e options are mutually exclusive}} |
|
%{-t:%{error:The -R and -t options are mutually exclusive}} |
|
%{-w:%{error:The -R and -w options are mutually exclusive}} |
|
} |
|
%{-N: |
|
%{-r:%{error:The -N and -r options are mutually exclusive}} |
|
%{-x:%{error:The -N and -x options are mutually exclusive}} |
|
%{-e:%{error:The -N and -e options are mutually exclusive}} |
|
%{-t:%{error:The -N and -t options are mutually exclusive}} |
|
%{-w:%{error:The -N and -w options are mutually exclusive}} |
|
%{-C:%{error:The -N and -C options are mutually exclusive}} |
|
} |
|
%{-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 'python%{python3_pkgversion}-devel' |
|
echo 'python%{python3_pkgversion}dist(pip) >= 19' |
|
echo 'python%{python3_pkgversion}dist(packaging)' |
|
%{!-N:if [ -f pyproject.toml ]; then |
|
%["%{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" |
|
]" |
|
] |
|
elif [ -f setup.py ]; then |
|
# 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(wheel)' |
|
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 |
|
exit 1 |
|
fi} |
|
# setuptools assumes no pre-existing dist-info |
|
rm -rfv *.dist-info/ >&2 |
|
if [ -f %{__python3} ]; then |
|
mkdir -p "%{_pyproject_builddir}" |
|
echo -n > %{_pyproject_buildrequires} |
|
CFLAGS="${CFLAGS:-${RPM_OPT_FLAGS}}" LDFLAGS="${LDFLAGS:-${RPM_LD_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 |
|
cat %{_pyproject_buildrequires} |
|
fi |
|
# Incomplete .dist-info dir might confuse importlib.metadata |
|
rm -rfv *.dist-info/ >&2 |
|
} |
|
|
|
|
|
%tox(e:) %{expand:\\\ |
|
TOX_TESTENV_PASSENV="${TOX_TESTENV_PASSENV:-*}" \\ |
|
%{?py3_test_envvars}%{?!py3_test_envvars:PYTHONDONTWRITEBYTECODE=1 \\ |
|
PATH="%{buildroot}%{_bindir}:$PATH" \\ |
|
PYTHONPATH="${PYTHONPATH:-%{buildroot}%{python3_sitearch}:%{buildroot}%{python3_sitelib}}" \\ |
|
%{?__pytest_addopts:PYTEST_ADDOPTS="${PYTEST_ADDOPTS:-} %{__pytest_addopts}"}} \\ |
|
HOSTNAME="rpmbuild" \\ |
|
%{__python3} -m tox --current-env -q --recreate -e "%{-e:%{-e*}}%{!-e:%{toxenv}}" %{?*} |
|
}
|
|
|