pyodide/pyodide-build/pyodide_build/mkpkg.py

329 lines
9.7 KiB
Python
Raw Normal View History

2019-02-28 23:43:26 +00:00
#!/usr/bin/env python3
import argparse
import json
import os
import shutil
import subprocess
import sys
2022-02-21 22:27:03 +00:00
import urllib.error
import urllib.request
2021-07-12 11:54:01 +00:00
import warnings
2022-02-21 22:27:03 +00:00
from pathlib import Path
from typing import Any, Literal
2019-02-28 23:43:26 +00:00
from ruamel.yaml import YAML
2019-02-28 23:43:26 +00:00
2021-07-12 11:54:01 +00:00
class MkpkgFailedException(Exception):
pass
SDIST_EXTENSIONS = tuple(
extension
for (name, extensions, description) in shutil.get_unpack_formats()
for extension in extensions
)
def _find_sdist(pypi_metadata: dict[str, Any]) -> dict[str, Any] | None:
"""Get sdist file path from the metadata"""
# The first one we can use. Usually a .tar.gz
for entry in pypi_metadata["urls"]:
if entry["packagetype"] == "sdist" and entry["filename"].endswith(
SDIST_EXTENSIONS
):
return entry
return None
def _find_wheel(pypi_metadata: dict[str, Any]) -> dict[str, Any] | None:
"""Get wheel file path from the metadata"""
for entry in pypi_metadata["urls"]:
if entry["packagetype"] == "bdist_wheel" and entry["filename"].endswith(
"py3-none-any.whl"
):
return entry
return None
def _find_dist(
2022-02-20 22:13:37 +00:00
pypi_metadata: dict[str, Any], source_types=list[Literal["wheel", "sdist"]]
) -> dict[str, Any]:
"""Find a wheel or sdist, as appropriate.
source_types controls which types (wheel and/or sdist) are accepted and also
the priority order.
E.g., ["wheel", "sdist"] means accept either wheel or sdist but prefer wheel.
["sdist", "wheel"] means accept either wheel or sdist but prefer sdist.
"""
result = None
for source in source_types:
if source == "wheel":
result = _find_wheel(pypi_metadata)
if source == "sdist":
result = _find_sdist(pypi_metadata)
if result:
return result
types_str = " or ".join(source_types)
name = pypi_metadata["info"].get("name")
url = pypi_metadata["info"].get("package_url")
raise MkpkgFailedException(f"No {types_str} found for package {name} ({url})")
def _get_metadata(package: str, version: str | None = None) -> dict:
"""Download metadata for a package from PyPI"""
version = ("/" + version) if version is not None else ""
url = f"https://pypi.org/pypi/{package}{version}/json"
2021-07-12 11:54:01 +00:00
try:
with urllib.request.urlopen(url) as fd:
pypi_metadata = json.load(fd)
except urllib.error.HTTPError as e:
raise MkpkgFailedException(
f"Failed to load metadata for {package}{version} from "
f"https://pypi.org/pypi/{package}{version}/json: {e}"
2021-07-12 11:54:01 +00:00
)
2021-07-12 11:54:01 +00:00
return pypi_metadata
2019-02-28 23:43:26 +00:00
def run_prettier(meta_path):
subprocess.run(["npx", "prettier", "-w", meta_path])
def make_package(
packages_dir: Path,
package: str,
version: str | None = None,
source_fmt: Literal["wheel", "sdist"] | None = None,
):
"""
Creates a template that will work for most pure Python packages,
but will have to be edited for more complex things.
"""
print(f"Creating meta.yaml package for {package}")
2021-07-23 19:32:34 +00:00
yaml = YAML()
2019-02-28 23:43:26 +00:00
2021-07-12 11:54:01 +00:00
pypi_metadata = _get_metadata(package, version)
if source_fmt:
sources = [source_fmt]
else:
# Prefer wheel unless sdist is specifically requested.
sources = ["wheel", "sdist"]
dist_metadata = _find_dist(pypi_metadata, sources)
url = dist_metadata["url"]
sha256 = dist_metadata["digests"]["sha256"]
version = pypi_metadata["info"]["version"]
2019-02-28 23:43:26 +00:00
2021-07-23 19:32:34 +00:00
homepage = pypi_metadata["info"]["home_page"]
summary = pypi_metadata["info"]["summary"]
license = pypi_metadata["info"]["license"]
pypi = "https://pypi.org/project/" + package
2019-02-28 23:43:26 +00:00
yaml_content = {
"package": {"name": package, "version": version},
"source": {"url": url, "sha256": sha256},
"test": {"imports": [package]},
2021-07-23 19:32:34 +00:00
"about": {
"home": homepage,
"PyPI": pypi,
2021-07-23 19:32:34 +00:00
"summary": summary,
"license": license,
},
2019-02-28 23:43:26 +00:00
}
package_dir = packages_dir / package
package_dir.mkdir(parents=True, exist_ok=True)
meta_path = package_dir / "meta.yaml"
if meta_path.exists():
raise MkpkgFailedException(f"The package {package} already exists")
yaml.dump(yaml_content, meta_path)
try:
run_prettier(meta_path)
except FileNotFoundError:
warnings.warn("'npx' executable missing, output has not been prettified.")
success(f"Output written to {meta_path}")
2019-02-28 23:43:26 +00:00
# TODO: use rich for coloring outputs
2021-07-12 11:54:01 +00:00
class bcolors:
HEADER = "\033[95m"
OKBLUE = "\033[94m"
OKCYAN = "\033[96m"
OKGREEN = "\033[92m"
WARNING = "\033[93m"
FAIL = "\033[91m"
ENDC = "\033[0m"
BOLD = "\033[1m"
UNDERLINE = "\033[4m"
def abort(msg):
print(bcolors.FAIL + msg + bcolors.ENDC)
sys.exit(1)
def warn(msg):
warnings.warn(bcolors.WARNING + msg + bcolors.ENDC)
def success(msg):
print(bcolors.OKBLUE + msg + bcolors.ENDC)
def update_package(
root: Path,
package: str,
2022-04-08 17:20:29 +00:00
version: str | None = None,
update_patched: bool = True,
source_fmt: Literal["wheel", "sdist"] | None = None,
):
2021-07-12 11:54:01 +00:00
yaml = YAML()
meta_path = root / package / "meta.yaml"
if not meta_path.exists():
abort(f"{meta_path} does not exist")
yaml_content = yaml.load(meta_path.read_bytes())
if "url" not in yaml_content["source"]:
raise MkpkgFailedException(f"Skipping: {package} is a local package!")
2021-07-12 11:54:01 +00:00
build_info = yaml_content.get("build", {})
if build_info.get("library", False) or build_info.get("sharedlibrary", False):
raise MkpkgFailedException(f"Skipping: {package} is a library!")
if yaml_content["source"]["url"].endswith("whl"):
old_fmt = "wheel"
else:
old_fmt = "sdist"
2022-04-08 17:20:29 +00:00
pypi_metadata = _get_metadata(package, version)
pypi_ver = pypi_metadata["info"]["version"]
local_ver = yaml_content["package"]["version"]
already_up_to_date = pypi_ver <= local_ver and (
source_fmt is None or source_fmt == old_fmt
)
if already_up_to_date:
print(f"{package} already up to date. Local: {local_ver} PyPI: {pypi_ver}")
return
2021-07-12 11:54:01 +00:00
print(f"{package} is out of date: {local_ver} <= {pypi_ver}.")
if "patches" in yaml_content["source"]:
2021-07-12 11:54:01 +00:00
if update_patched:
warn(
f"Pyodide applies patches to {package}. Update the "
"patches (if needed) to avoid build failing."
)
else:
raise MkpkgFailedException(
f"Pyodide applies patches to {package}. Skipping update."
)
if source_fmt:
# require the type requested
sources = [source_fmt]
elif old_fmt == "wheel":
# prefer wheel to sdist
sources = ["wheel", "sdist"]
else:
# prefer sdist to wheel
sources = ["sdist", "wheel"]
dist_metadata = _find_dist(pypi_metadata, sources)
yaml_content["source"]["url"] = dist_metadata["url"]
yaml_content["source"].pop("md5", None)
yaml_content["source"]["sha256"] = dist_metadata["digests"]["sha256"]
yaml_content["package"]["version"] = pypi_metadata["info"]["version"]
yaml.dump(yaml_content, meta_path)
run_prettier(meta_path)
2021-07-12 11:54:01 +00:00
success(f"Updated {package} from {local_ver} to {pypi_ver}.")
2019-02-28 23:43:26 +00:00
def make_parser(parser):
parser.description = """
2019-02-28 23:43:26 +00:00
Make a new pyodide package. Creates a simple template that will work
2020-06-16 21:29:42 +00:00
for most pure Python packages, but will have to be edited for more
complex things.""".strip()
parser.add_argument("package", type=str, nargs=1, help="The package name on PyPI")
parser.add_argument("--update", action="store_true", help="Update existing package")
2021-07-12 11:54:01 +00:00
parser.add_argument(
"--update-if-not-patched",
action="store_true",
help="Update existing package if it has no patches",
)
parser.add_argument(
"--source-format",
help="Which source format is preferred. Options are wheel or sdist. "
"If none is provided, then either a wheel or an sdist will be used. "
"When updating a package, the type will be kept the same if possible.",
)
2019-02-28 23:43:26 +00:00
parser.add_argument(
"--version",
type=str,
default=None,
help="Package version string, "
"e.g. v1.2.1 (defaults to latest stable release)",
)
2019-02-28 23:43:26 +00:00
return parser
def main(args):
PYODIDE_ROOT = os.environ.get("PYODIDE_ROOT")
if PYODIDE_ROOT is None:
raise ValueError("PYODIDE_ROOT is not set")
PACKAGES_ROOT = Path(PYODIDE_ROOT) / "packages"
2021-07-12 11:54:01 +00:00
try:
package = args.package[0]
if args.update:
2022-04-08 17:20:29 +00:00
update_package(
PACKAGES_ROOT,
2022-04-08 17:20:29 +00:00
package,
args.version,
update_patched=True,
source_fmt=args.source_format,
)
2021-07-12 11:54:01 +00:00
return
if args.update_if_not_patched:
2022-04-08 17:20:29 +00:00
update_package(
PACKAGES_ROOT,
2022-04-08 17:20:29 +00:00
package,
args.version,
update_patched=False,
source_fmt=args.source_format,
)
2021-07-12 11:54:01 +00:00
return
make_package(
PACKAGES_ROOT, package, args.version, source_fmt=args.source_format
)
2021-07-12 11:54:01 +00:00
except MkpkgFailedException as e:
# This produces two types of error messages:
#
# When the request to get the pypi json fails, it produces a message like:
# "Failed to load metadata for libxslt from https://pypi.org/pypi/libxslt/json: HTTP Error 404: Not Found"
#
# If there is no sdist it prints an error message like:
# "No sdist URL found for package swiglpk (https://pypi.org/project/swiglpk/)"
abort(e.args[0])
2019-02-28 23:43:26 +00:00
if __name__ == "__main__":
2019-02-28 23:43:26 +00:00
parser = make_parser(argparse.ArgumentParser())
args = parser.parse_args()
main(args)