pyodide/pyodide-build/pyodide_build/mkpkg.py

355 lines
10 KiB
Python
Executable File

#!/usr/bin/env python3
import argparse
import json
import os
import shutil
import subprocess
import sys
import urllib.error
import urllib.request
import warnings
from pathlib import Path
from typing import Any, Literal, TypedDict
from ruamel.yaml import YAML
class URLDict(TypedDict):
comment_text: str
digests: dict[str, Any]
downloads: int
filename: str
has_sig: bool
md5_digest: str
packagetype: str
python_version: str
requires_python: str
size: int
upload_time: str
upload_time_iso_8601: str
url: str
yanked: bool
yanked_reason: str | None
class MetadataDict(TypedDict):
info: dict[str, Any]
last_serial: int
releases: dict[str, list[dict[str, Any]]]
urls: list[URLDict]
vulnerabilities: list[Any]
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: MetadataDict) -> URLDict | 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: MetadataDict) -> URLDict | 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(
pypi_metadata: MetadataDict, source_types: list[Literal["wheel", "sdist"]]
) -> URLDict:
"""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) -> MetadataDict:
"""Download metadata for a package from PyPI"""
version = ("/" + version) if version is not None else ""
url = f"https://pypi.org/pypi/{package}{version}/json"
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}"
)
return pypi_metadata
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,
) -> 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}")
yaml = YAML()
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"]
homepage = pypi_metadata["info"]["home_page"]
summary = pypi_metadata["info"]["summary"]
license = pypi_metadata["info"]["license"]
pypi = "https://pypi.org/project/" + package
yaml_content = {
"package": {"name": package, "version": version},
"source": {"url": url, "sha256": sha256},
"test": {"imports": [package]},
"about": {
"home": homepage,
"PyPI": pypi,
"summary": summary,
"license": license,
},
}
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}")
# TODO: use rich for coloring outputs
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,
version: str | None = None,
update_patched: bool = True,
source_fmt: Literal["wheel", "sdist"] | None = None,
) -> None:
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!")
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"
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
print(f"{package} is out of date: {local_ver} <= {pypi_ver}.")
if "patches" in yaml_content["source"]:
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)
success(f"Updated {package} from {local_ver} to {pypi_ver}.")
def make_parser(parser):
parser.description = """
Make a new pyodide package. Creates a simple template that will work
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")
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.",
)
parser.add_argument(
"--version",
type=str,
default=None,
help="Package version string, "
"e.g. v1.2.1 (defaults to latest stable release)",
)
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"
try:
package = args.package[0]
if args.update:
update_package(
PACKAGES_ROOT,
package,
args.version,
update_patched=True,
source_fmt=args.source_format,
)
return
if args.update_if_not_patched:
update_package(
PACKAGES_ROOT,
package,
args.version,
update_patched=False,
source_fmt=args.source_format,
)
return
make_package(
PACKAGES_ROOT, package, args.version, source_fmt=args.source_format
)
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])
if __name__ == "__main__":
parser = make_parser(argparse.ArgumentParser())
args = parser.parse_args()
main(args)