From 7be6ff08e33dc87f967b91366a70218f4478a959 Mon Sep 17 00:00:00 2001 From: Jonas Thiem Date: Sat, 8 Jun 2019 22:01:36 +0200 Subject: [PATCH] pythonpackage can't return build requirements for wheels. Make sure the pythonpackage functions return an error when someone attempts to do so anyway --- pythonforandroid/pythonpackage.py | 49 ++++++++++++++++++++++++++++++- tests/test_pythonpackage_basic.py | 24 +++++++++++++-- 2 files changed, 69 insertions(+), 4 deletions(-) diff --git a/pythonforandroid/pythonpackage.py b/pythonforandroid/pythonpackage.py index 0ab5bf06c1..88b54994c5 100644 --- a/pythonforandroid/pythonpackage.py +++ b/pythonforandroid/pythonpackage.py @@ -97,6 +97,10 @@ def extract_metainfo_files_from_package( if not os.path.exists(output_folder) or os.path.isfile(output_folder): raise ValueError("output folder needs to be existing folder") + if debug: + print("extract_metainfo_files_from_package: extracting for " + + "package: " + str(package)) + # A temp folder for making a package copy in case it's a local folder, # because extracting metadata might modify files # (creating sdists/wheels...) @@ -418,6 +422,7 @@ def _extract_metainfo_files_from_package_unsafe( try: build_requires = [] metadata_path = None + if path_type != "wheel": # We need to process this first to get the metadata. @@ -447,7 +452,9 @@ def _extract_metainfo_files_from_package_unsafe( metadata = None with env: hooks = Pep517HookCaller(path, backend) - env.pip_install([transform_dep_for_pip(req) for req in build_requires]) + env.pip_install( + [transform_dep_for_pip(req) for req in build_requires] + ) reqs = hooks.get_requires_for_build_wheel({}) env.pip_install([transform_dep_for_pip(req) for req in reqs]) try: @@ -466,6 +473,15 @@ def _extract_metainfo_files_from_package_unsafe( "METADATA" ) + # Store type of metadata source. Can be "wheel", "source" for source + # distribution, and others get_package_as_folder() may support + # in the future. + with open(os.path.join(output_path, "metadata_source"), "w") as f: + try: + f.write(path_type) + except TypeError: # in python 2 path_type may be str/bytes: + f.write(path_type.decode("utf-8", "replace")) + # Copy the metadata file: shutil.copyfile(metadata_path, os.path.join(output_path, "METADATA")) finally: @@ -518,12 +534,23 @@ def _extract_info_from_package(dependency, - name - dependencies (a list of dependencies) """ + if debug: + print("_extract_info_from_package called with " + "extract_type={} include_build_requirements={}".format( + extract_type, include_build_requirements, + )) output_folder = tempfile.mkdtemp(prefix="pythonpackage-metafolder-") try: extract_metainfo_files_from_package( dependency, output_folder, debug=debug ) + # Extract the type of data source we used to get the metadata: + with open(os.path.join(output_folder, + "metadata_source"), "r") as f: + metadata_source_type = f.read().strip() + + # Extract main METADATA file: with open(os.path.join(output_folder, "METADATA"), "r", encoding="utf-8" ) as f: @@ -539,14 +566,34 @@ def _extract_info_from_package(dependency, raise ValueError("failed to obtain package name") return name elif extract_type == "dependencies": + # First, make sure we don't attempt to return build requirements + # for wheels since they usually come without pyproject.toml + # and we haven't implemented another way to get them: + if include_build_requirements and \ + metadata_source_type == "wheel": + if debug: + print("_extract_info_from_package: was called " + "with include_build_requirements=True on " + "package obtained as wheel, raising error...") + raise NotImplementedError( + "fetching build requirements for " + "wheels is not implemented" + ) + + # Get build requirements from pyproject.toml if requested: requirements = [] if os.path.exists(os.path.join(output_folder, 'pyproject.toml') ) and include_build_requirements: + # Read build system from pyproject.toml file: (PEP518) with open(os.path.join(output_folder, 'pyproject.toml')) as f: build_sys = pytoml.load(f)['build-system'] if "requires" in build_sys: requirements += build_sys["requires"] + elif include_build_requirements: + # For legacy packages with no pyproject.toml, we have to + # add setuptools as default build system. + requirements.append("setuptools") # Add requirements from metadata: requirements += [ diff --git a/tests/test_pythonpackage_basic.py b/tests/test_pythonpackage_basic.py index 8dfdd96e88..bed2ce9a36 100644 --- a/tests/test_pythonpackage_basic.py +++ b/tests/test_pythonpackage_basic.py @@ -43,6 +43,8 @@ def fake_metadata_extract(dep_name, output_folder, debug=False): Lorem Ipsum""" )) + with open(os.path.join(output_folder, "metadata_source"), "w") as f: + f.write(u"wheel") # since we have no pyproject.toml def test__extract_info_from_package(): @@ -87,9 +89,25 @@ def test_get_dep_names_of_package(): dep_names = get_dep_names_of_package("python-for-android") assert "colorama" in dep_names assert "setuptools" not in dep_names - dep_names = get_dep_names_of_package("python-for-android", - include_build_requirements=True) - assert "setuptools" in dep_names + try: + dep_names = get_dep_names_of_package( + "python-for-android", include_build_requirements=True, + verbose=True, + ) + except NotImplementedError as e: + # If python-for-android was fetched as wheel then build requirements + # cannot be obtained (since that is not implemented for wheels). + # Check for the correct error message: + assert "wheel" in str(e) + # (And yes it would be better to do a local test with something + # that is guaranteed to be a wheel and not remote on pypi, + # but that might require setting up a full local pypiserver. + # Not worth the test complexity for now, but if anyone has an + # idea in the future feel free to replace this subtest.) + else: + # We managed to obtain build requirements! + # Check setuptools is in here: + assert "setuptools" in dep_names # TEST 2 from local folder: assert "colorama" in get_dep_names_of_package(local_repo_folder())