Using pip to install host packages with pyproject.toml-based (PEP 517) builds is problematic: * If build isolation is used, pip will create an isolated build environment, install any build dependencies for the requested package, then build the requested package. It does not appear currently possible to have pip install the build dependencies with hash-checking mode enabled[1]. * If build isolation is not used, any build dependencies must be installed in the build environment before invoking pip to build the requested package[2]. This would require creating a package dependency resolution system to install build dependencies, and any dependencies of dependencies, in the correct order. * It is very difficult to patch the packages installed by pip. This adds a new include file (python3-host-build.mk) with recipes to install host Python packages with pyproject.toml-based builds. This is backwards-compatible with packages that require running setup.py. Besides addressing the above issues (the OpenWrt build system already resolves dependencies between packages, checks all source downloads against known hashes, and supports patching packages), host packages also: * Capture package licensing and maintainer information * Enable uscan checking for package updates/CVEs * Are a known concept for OpenWrt packagers/developers The existing functionality of using host pip to install packages will remain for now, but should be considered deprecated and expected to be removed in the future. This also updates Py3Build/CheckHostPipVersionMatch for the case where the host-pip-requirements directory does not exist or is empty. [1]: https://pip.pypa.io/en/stable/user_guide/#changes-to-the-pip-dependency-resolver-in-20-3-2020 [2]: https://pip.pypa.io/en/stable/cli/pip_install/#cmdoption-no-build-isolation Signed-off-by: Jeffery To <jeffery.to@gmail.com>
249 lines
7.1 KiB
Makefile
249 lines
7.1 KiB
Makefile
#
|
|
# Copyright (C) 2007-2016 OpenWrt.org
|
|
#
|
|
# This is free software, licensed under the GNU General Public License v2.
|
|
# See /LICENSE for more information.
|
|
#
|
|
|
|
# Note: include this file after `include $(TOPDIR)/rules.mk in your package Makefile
|
|
|
|
python3_mk_path:=$(dir $(lastword $(MAKEFILE_LIST)))
|
|
include $(python3_mk_path)python3-host.mk
|
|
|
|
PYTHON3_DIR:=$(STAGING_DIR)/usr
|
|
PYTHON3_INC_DIR:=$(PYTHON3_DIR)/include/python$(PYTHON3_VERSION)
|
|
PYTHON3_LIB_DIR:=$(PYTHON3_DIR)/lib/python$(PYTHON3_VERSION)
|
|
|
|
PYTHON3_PKG_DIR:=/usr/lib/python$(PYTHON3_VERSION)/site-packages
|
|
|
|
PYTHON3:=python$(PYTHON3_VERSION)
|
|
|
|
PYTHON3PATH:=$(PYTHON3_LIB_DIR):$(STAGING_DIR)/$(PYTHON3_PKG_DIR):$(PKG_INSTALL_DIR)/$(PYTHON3_PKG_DIR)
|
|
|
|
-include $(PYTHON3_LIB_DIR)/config-$(PYTHON3_VERSION)/Makefile-vars
|
|
|
|
# These configure args are needed in detection of path to Python header files
|
|
# using autotools.
|
|
CONFIGURE_ARGS += \
|
|
_python_sysroot="$(STAGING_DIR)" \
|
|
_python_prefix="/usr" \
|
|
_python_exec_prefix="/usr"
|
|
|
|
PYTHON3_VARS = \
|
|
CC="$(TARGET_CC)" \
|
|
CCSHARED="$(TARGET_CC) $(FPIC)" \
|
|
CXX="$(TARGET_CXX)" \
|
|
LD="$(TARGET_CC)" \
|
|
LDSHARED="$(TARGET_CC) -shared" \
|
|
CFLAGS="$(TARGET_CFLAGS)" \
|
|
CPPFLAGS="$(TARGET_CPPFLAGS) -I$(PYTHON3_INC_DIR)" \
|
|
LDFLAGS="$(TARGET_LDFLAGS) -lpython$(PYTHON3_VERSION)" \
|
|
_PYTHON_HOST_PLATFORM="$(_PYTHON_HOST_PLATFORM)" \
|
|
__PYVENV_LAUNCHER__="/usr/bin/$(PYTHON3)" \
|
|
PYTHONPATH="$(PYTHON3PATH)" \
|
|
PYTHONDONTWRITEBYTECODE=1 \
|
|
_python_sysroot="$(STAGING_DIR)" \
|
|
_python_prefix="/usr" \
|
|
_python_exec_prefix="/usr"
|
|
|
|
# $(1) => directory of python script
|
|
# $(2) => python script and its arguments
|
|
# $(3) => additional variables
|
|
define Python3/Run
|
|
cd "$(if $(strip $(1)),$(strip $(1)),.)" && \
|
|
$(PYTHON3_VARS) \
|
|
$(3) \
|
|
$(HOST_PYTHON3_BIN) $(2)
|
|
endef
|
|
|
|
# $(1) => build subdir
|
|
# $(2) => additional arguments to setup.py
|
|
# $(3) => additional variables
|
|
define Python3/ModSetup
|
|
$(INSTALL_DIR) $(PKG_INSTALL_DIR)/$(PYTHON3_PKG_DIR)
|
|
$(call SetupPyShim,$(PKG_BUILD_DIR)/$(strip $(1)))
|
|
$(call Python3/Run, \
|
|
$(PKG_BUILD_DIR)/$(strip $(1)), \
|
|
setup.py $(2), \
|
|
$(3) PY_PKG_VERSION=$(PKG_VERSION))
|
|
endef
|
|
|
|
define Python3/FixShebang
|
|
$(SED) "1"'!'"b;s,^#"'!'".*python.*,#"'!'"/usr/bin/python3," -i --follow-symlinks $(1)
|
|
endef
|
|
|
|
# default max recursion is 10
|
|
PYTHON3_COMPILEALL_MAX_RECURSION_LEVEL:=20
|
|
|
|
# $(1) => directory of python source files to compile
|
|
#
|
|
# XXX [So that you won't goof as I did]
|
|
# Note: Yes, I tried to use the -O & -OO flags here.
|
|
# However the generated byte-codes were not portable.
|
|
# So, we just stuck to un-optimized byte-codes,
|
|
# which is still way better/faster than running
|
|
# Python sources all the time.
|
|
#
|
|
# Setting a fixed hash seed value is less secure than using
|
|
# random seed values, but is necessary for reproducible builds
|
|
# (for now).
|
|
#
|
|
# Should revisit this when https://bugs.python.org/issue37596
|
|
# (and other related reproducibility issues) are fixed.
|
|
define Python3/CompileAll
|
|
$(call Python3/Run,, \
|
|
-m compileall -r "$(PYTHON3_COMPILEALL_MAX_RECURSION_LEVEL)" -b -d '/' $(1),
|
|
$(if $(SOURCE_DATE_EPOCH),PYTHONHASHSEED="$(SOURCE_DATE_EPOCH)")
|
|
)
|
|
endef
|
|
|
|
# $(1) => target directory
|
|
define Python3/DeleteSourceFiles
|
|
$(FIND) $(1) -type f -name '*.py' -delete
|
|
endef
|
|
|
|
# $(1) => target directory
|
|
define Python3/DeleteNonSourceFiles
|
|
$(FIND) $(1) -not -type d -not -name '*.py' -delete
|
|
endef
|
|
|
|
# $(1) => target directory
|
|
define Python3/DeleteEmptyDirs
|
|
$(FIND) $(1) -mindepth 1 -empty -type d -not -path '$(1)/CONTROL' -not -path '$(1)/CONTROL/*' -delete
|
|
endef
|
|
|
|
|
|
# Py3Package
|
|
|
|
define Py3Package/filespec/Default
|
|
+|$(PYTHON3_PKG_DIR)
|
|
endef
|
|
|
|
# $(1) => package name
|
|
# $(2) => src directory
|
|
# $(3) => dest directory
|
|
define Py3Package/ProcessFilespec
|
|
$(eval $(call shexport,Py3Package/$(1)/filespec))
|
|
$(SHELL) $(python3_mk_path)python-package-install.sh \
|
|
"$(2)" "$(3)" "$$$$$(call shvar,Py3Package/$(1)/filespec)"
|
|
endef
|
|
|
|
define Py3Package
|
|
define Package/$(1)-src
|
|
$(call Package/$(1))
|
|
DEPENDS:=
|
|
CONFLICTS:=
|
|
PROVIDES:=
|
|
EXTRA_DEPENDS:=
|
|
TITLE+= (sources)
|
|
USERID:=
|
|
MENU:=
|
|
endef
|
|
|
|
define Package/$(1)-src/description
|
|
$$(call Package/$(1)/description)
|
|
|
|
This package contains the Python source files for $(1).
|
|
endef
|
|
|
|
define Package/$(1)-src/config
|
|
depends on PACKAGE_$(1)
|
|
endef
|
|
|
|
# Add default PyPackage filespec none defined
|
|
ifeq ($(origin Py3Package/$(1)/filespec),undefined)
|
|
Py3Package/$(1)/filespec=$$(Py3Package/filespec/Default)
|
|
endif
|
|
|
|
ifndef Py3Package/$(1)/install
|
|
define Py3Package/$(1)/install
|
|
if [ -d $(PKG_INSTALL_DIR)/usr/bin ]; then \
|
|
$(INSTALL_DIR) $$(1)/usr/bin ; \
|
|
$(CP) $(PKG_INSTALL_DIR)/usr/bin/* $$(1)/usr/bin/ ; \
|
|
fi
|
|
endef
|
|
endif
|
|
|
|
ifndef Package/$(1)/install
|
|
define Package/$(1)/install
|
|
$$(call Py3Package/$(1)/install,$$(1))
|
|
$$(call Py3Package/ProcessFilespec,$(1),$(PKG_INSTALL_DIR),$$(1))
|
|
$(FIND) $$(1) -name '*.exe' -delete
|
|
$$(call Python3/CompileAll,$$(1))
|
|
$$(call Python3/DeleteSourceFiles,$$(1))
|
|
$$(call Python3/DeleteEmptyDirs,$$(1))
|
|
if [ -d "$$(1)/usr/bin" ]; then \
|
|
$$(call Python3/FixShebang,$$(1)/usr/bin/*) ; \
|
|
fi
|
|
endef
|
|
|
|
define Package/$(1)-src/install
|
|
$$(call Py3Package/$(1)/install,$$(1))
|
|
$$(call Py3Package/ProcessFilespec,$(1),$(PKG_INSTALL_DIR),$$(1))
|
|
$$(call Python3/DeleteNonSourceFiles,$$(1))
|
|
$$(call Python3/DeleteEmptyDirs,$$(1))
|
|
endef
|
|
endif # Package/$(1)/install
|
|
endef
|
|
|
|
|
|
# Py3Build
|
|
|
|
PYTHON3_PKG_SETUP_DIR ?=
|
|
PYTHON3_PKG_SETUP_GLOBAL_ARGS ?=
|
|
PYTHON3_PKG_SETUP_ARGS ?= --single-version-externally-managed
|
|
PYTHON3_PKG_SETUP_VARS ?=
|
|
|
|
PYTHON3_PKG_HOST_PIP_INSTALL_ARGS = \
|
|
$(foreach req,$(HOST_PYTHON3_PACKAGE_BUILD_DEPENDS), \
|
|
--requirement \
|
|
$(if $(findstring /,$(req)),$(req),$(python3_mk_path)host-pip-requirements/$(req).txt) \
|
|
)
|
|
|
|
define Py3Build/FindStdlibDepends
|
|
$(SHELL) $(python3_mk_path)python3-find-stdlib-depends.sh -n "$(PKG_NAME)" "$(PKG_BUILD_DIR)";
|
|
endef
|
|
|
|
ifneq ($(strip $(PYPI_NAME)),)
|
|
define Py3Build/CheckHostPipVersionMatch
|
|
if [ -d "$(python3_mk_path)host-pip-requirements" ] && \
|
|
[ -n "$$$$($(FIND) $(python3_mk_path)host-pip-requirements -maxdepth 1 -mindepth 1 -name '*.txt' -print -quit 2>/dev/null)" ]; then \
|
|
if grep -q "$(PYPI_NAME)==" $(python3_mk_path)host-pip-requirements/*.txt ; then \
|
|
if ! grep -q "$(PYPI_NAME)==$(PKG_VERSION)" $(python3_mk_path)host-pip-requirements/*.txt ; then \
|
|
printf "\nPlease update version of $(PYPI_NAME) to $(PKG_VERSION) in 'host-pip-requirements'/\n\n" ; \
|
|
exit 1 ; \
|
|
fi \
|
|
fi \
|
|
fi
|
|
endef
|
|
endif
|
|
|
|
define Py3Build/InstallBuildDepends
|
|
$(if $(PYTHON3_PKG_HOST_PIP_INSTALL_ARGS), \
|
|
$(call HostPython3/PipInstall,$(PYTHON3_PKG_HOST_PIP_INSTALL_ARGS)) \
|
|
)
|
|
endef
|
|
|
|
define Py3Build/Compile/Default
|
|
$(call Py3Build/InstallBuildDepends)
|
|
$(call Python3/ModSetup, \
|
|
$(PYTHON3_PKG_SETUP_DIR), \
|
|
$(PYTHON3_PKG_SETUP_GLOBAL_ARGS) \
|
|
install --prefix="/usr" --root="$(PKG_INSTALL_DIR)" \
|
|
$(PYTHON3_PKG_SETUP_ARGS), \
|
|
$(PYTHON3_PKG_SETUP_VARS) \
|
|
)
|
|
endef
|
|
|
|
Py3Build/Configure=$(Py3Build/Configure/Default)
|
|
Py3Build/Compile=$(Py3Build/Compile/Default)
|
|
|
|
PYTHON3_PKG_BUILD ?= 1
|
|
|
|
ifeq ($(strip $(PYTHON3_PKG_BUILD)),1)
|
|
ifeq ($(PY3),stdlib)
|
|
Hooks/Configure/Post+=Py3Build/FindStdlibDepends
|
|
endif
|
|
Hooks/Configure/Post+=Py3Build/CheckHostPipVersionMatch
|
|
Build/Compile=$(Py3Build/Compile)
|
|
endif
|