2019-02-28 23:43:26 +00:00
|
|
|
#!/usr/bin/env python3
|
|
|
|
|
|
|
|
import argparse
|
|
|
|
import json
|
|
|
|
import os
|
2019-03-26 16:10:41 +00:00
|
|
|
import shutil
|
2019-02-28 23:43:26 +00:00
|
|
|
import urllib.request
|
2020-07-13 21:34:10 +00:00
|
|
|
import sys
|
2019-03-26 16:10:41 +00:00
|
|
|
from pathlib import Path
|
2020-07-13 21:34:10 +00:00
|
|
|
from typing import Dict, Tuple, Any, Optional
|
2019-02-28 23:43:26 +00:00
|
|
|
|
2021-01-11 07:59:22 +00:00
|
|
|
from .io import parse_package_config
|
|
|
|
|
2020-06-28 18:24:40 +00:00
|
|
|
PACKAGES_ROOT = Path(__file__).parent.parent / "packages"
|
2019-02-28 23:43:26 +00:00
|
|
|
|
2019-03-26 16:10:41 +00:00
|
|
|
|
2020-07-13 21:34:10 +00:00
|
|
|
def _extract_sdist(pypi_metadata: Dict[str, Any]) -> Dict:
|
|
|
|
"""Get sdist file path from the meta-data"""
|
|
|
|
sdist_extensions = tuple(
|
|
|
|
extension
|
|
|
|
for (name, extensions, description) in shutil.get_unpack_formats()
|
|
|
|
for extension in extensions
|
|
|
|
)
|
2019-03-26 16:10:41 +00:00
|
|
|
|
2020-07-13 21:34:10 +00:00
|
|
|
# The first one we can use. Usually a .tar.gz
|
|
|
|
for entry in pypi_metadata["urls"]:
|
|
|
|
if entry["filename"].endswith(sdist_extensions):
|
|
|
|
return entry
|
2019-03-26 16:10:41 +00:00
|
|
|
|
2020-07-13 21:34:10 +00:00
|
|
|
raise Exception(
|
|
|
|
"No sdist URL found for package %s (%s)"
|
2020-09-24 10:28:10 +00:00
|
|
|
% (
|
|
|
|
pypi_metadata["info"].get("name"),
|
|
|
|
pypi_metadata["info"].get("package_url"),
|
|
|
|
)
|
2020-07-13 21:34:10 +00:00
|
|
|
)
|
2019-03-26 16:10:41 +00:00
|
|
|
|
|
|
|
|
2021-03-31 19:11:41 +00:00
|
|
|
def _get_metadata(package: str, version: Optional[str] = None) -> Tuple[Dict, Dict]:
|
2020-07-13 21:34:10 +00:00
|
|
|
"""Download metadata for a package from PyPi"""
|
2021-03-31 19:11:41 +00:00
|
|
|
version = ("/" + version) if version is not None else ""
|
|
|
|
url = f"https://pypi.org/pypi/{package}{version}/json"
|
2019-03-26 16:10:41 +00:00
|
|
|
|
2020-07-13 21:34:10 +00:00
|
|
|
with urllib.request.urlopen(url) as fd:
|
|
|
|
pypi_metadata = json.load(fd)
|
2019-03-26 16:10:41 +00:00
|
|
|
|
2020-07-13 21:34:10 +00:00
|
|
|
sdist_metadata = _extract_sdist(pypi_metadata)
|
2019-03-26 16:10:41 +00:00
|
|
|
|
2020-07-13 21:34:10 +00:00
|
|
|
return sdist_metadata, pypi_metadata
|
2019-03-26 16:10:41 +00:00
|
|
|
|
2019-02-28 23:43:26 +00:00
|
|
|
|
2020-07-13 21:34:10 +00:00
|
|
|
def make_package(package: str, version: Optional[str] = None):
|
|
|
|
"""
|
|
|
|
Creates a template that will work for most pure Python packages,
|
|
|
|
but will have to be edited for more complex things.
|
|
|
|
"""
|
2019-02-28 23:43:26 +00:00
|
|
|
import yaml
|
|
|
|
|
2021-03-31 19:11:41 +00:00
|
|
|
sdist_metadata, pypi_metadata = _get_metadata(package, version)
|
2020-07-13 21:34:10 +00:00
|
|
|
url = sdist_metadata["url"]
|
|
|
|
sha256 = sdist_metadata["digests"]["sha256"]
|
|
|
|
version = pypi_metadata["info"]["version"]
|
2019-02-28 23:43:26 +00:00
|
|
|
|
|
|
|
yaml_content = {
|
2020-06-28 18:24:40 +00:00
|
|
|
"package": {"name": package, "version": version},
|
2020-07-13 21:34:10 +00:00
|
|
|
"source": {"url": url, "sha256": sha256},
|
2020-06-28 18:24:40 +00:00
|
|
|
"test": {"imports": [package]},
|
2019-02-28 23:43:26 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if not (PACKAGES_ROOT / package).is_dir():
|
|
|
|
os.makedirs(PACKAGES_ROOT / package)
|
2020-06-28 18:24:40 +00:00
|
|
|
with open(PACKAGES_ROOT / package / "meta.yaml", "w") as fd:
|
2019-02-28 23:43:26 +00:00
|
|
|
yaml.dump(yaml_content, fd, default_flow_style=False)
|
|
|
|
|
|
|
|
|
2020-07-13 21:34:10 +00:00
|
|
|
def update_package(package: str):
|
|
|
|
import yaml
|
|
|
|
|
|
|
|
meta_path = PACKAGES_ROOT / package / "meta.yaml"
|
2021-01-11 07:59:22 +00:00
|
|
|
yaml_content = parse_package_config(meta_path)
|
2020-07-13 21:34:10 +00:00
|
|
|
|
|
|
|
if "url" not in yaml_content["source"]:
|
|
|
|
print(f"Skipping: {package} is a local package!")
|
|
|
|
sys.exit(0)
|
|
|
|
|
|
|
|
if set(yaml_content.keys()).difference(
|
|
|
|
("package", "source", "test", "requirements")
|
|
|
|
):
|
|
|
|
print(
|
|
|
|
f"{package}: Only pure-python packages can be updated using this script. "
|
|
|
|
f"Aborting."
|
|
|
|
)
|
|
|
|
sys.exit(1)
|
|
|
|
|
|
|
|
sdist_metadata, pypi_metadata = _get_metadata(package)
|
|
|
|
pypi_ver = pypi_metadata["info"]["version"]
|
|
|
|
local_ver = yaml_content["package"]["version"]
|
|
|
|
if pypi_ver <= local_ver:
|
|
|
|
print(f"{package} already up to date. Local: {local_ver} PyPi: {pypi_ver}")
|
|
|
|
sys.exit(0)
|
|
|
|
print(f"Updating {package} from {local_ver} to {pypi_ver}.")
|
|
|
|
|
|
|
|
if "patches" in yaml_content["source"]:
|
|
|
|
import warnings
|
|
|
|
|
|
|
|
warnings.warn(
|
|
|
|
f"Pyodide applies patches to {package}. Update the "
|
|
|
|
"patches (if needed) to avoid build failing."
|
|
|
|
)
|
|
|
|
|
|
|
|
yaml_content["source"]["url"] = sdist_metadata["url"]
|
|
|
|
yaml_content["source"].pop("md5", None)
|
|
|
|
yaml_content["source"]["sha256"] = sdist_metadata["digests"]["sha256"]
|
|
|
|
yaml_content["package"]["version"] = pypi_metadata["info"]["version"]
|
|
|
|
with open(PACKAGES_ROOT / package / "meta.yaml", "w") as fd:
|
|
|
|
yaml.dump(yaml_content, fd, default_flow_style=False)
|
|
|
|
|
|
|
|
|
2019-02-28 23:43:26 +00:00
|
|
|
def make_parser(parser):
|
2020-06-28 18:24:40 +00:00
|
|
|
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
|
2020-06-28 18:24:40 +00:00
|
|
|
complex things.""".strip()
|
|
|
|
parser.add_argument("package", type=str, nargs=1, help="The package name on PyPI")
|
2020-07-13 21:34:10 +00:00
|
|
|
parser.add_argument("--update", action="store_true", help="Update existing package")
|
2019-02-28 23:43:26 +00:00
|
|
|
parser.add_argument(
|
2020-06-28 18:24:40 +00:00
|
|
|
"--version",
|
|
|
|
type=str,
|
|
|
|
default=None,
|
2020-05-19 21:38:41 +00:00
|
|
|
help="Package version string, "
|
2020-06-28 18:24:40 +00:00
|
|
|
"e.g. v1.2.1 (defaults to latest stable release)",
|
|
|
|
)
|
2019-02-28 23:43:26 +00:00
|
|
|
return parser
|
|
|
|
|
|
|
|
|
|
|
|
def main(args):
|
|
|
|
package = args.package[0]
|
2020-07-13 21:34:10 +00:00
|
|
|
if args.update:
|
2021-03-31 19:11:41 +00:00
|
|
|
update_package(package)
|
|
|
|
return
|
|
|
|
make_package(package, args.version)
|
2019-02-28 23:43:26 +00:00
|
|
|
|
|
|
|
|
2020-06-28 18:24:40 +00:00
|
|
|
if __name__ == "__main__":
|
2019-02-28 23:43:26 +00:00
|
|
|
parser = make_parser(argparse.ArgumentParser())
|
|
|
|
args = parser.parse_args()
|
|
|
|
main(args)
|