Compare commits
No commits in common. "main" and "extra-files" have entirely different histories.
main
...
extra-file
@ -6,7 +6,6 @@ PYTHON_VERSIONS = [
|
|||||||
"3.9",
|
"3.9",
|
||||||
"3.10",
|
"3.10",
|
||||||
"3.11",
|
"3.11",
|
||||||
"3.12",
|
|
||||||
"latest",
|
"latest",
|
||||||
]
|
]
|
||||||
|
|
||||||
@ -72,9 +71,6 @@ def test_step(docker_tag, python_cmd="python"):
|
|||||||
"{} -V".format(python_cmd),
|
"{} -V".format(python_cmd),
|
||||||
"make clean-all test"
|
"make clean-all test"
|
||||||
],
|
],
|
||||||
"environment": {
|
|
||||||
"PIP_CACHE_DIR": ".pip-cache",
|
|
||||||
},
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -1,11 +1,11 @@
|
|||||||
---
|
---
|
||||||
repos:
|
repos:
|
||||||
- repo: https://github.com/psf/black
|
- repo: https://github.com/psf/black
|
||||||
rev: 24.4.2
|
rev: 22.3.0
|
||||||
hooks:
|
hooks:
|
||||||
- id: black
|
- id: black
|
||||||
- repo: https://github.com/pre-commit/pre-commit-hooks
|
- repo: https://github.com/pre-commit/pre-commit-hooks
|
||||||
rev: v4.6.0
|
rev: v4.1.0
|
||||||
hooks:
|
hooks:
|
||||||
- id: check-added-large-files
|
- id: check-added-large-files
|
||||||
- id: check-merge-conflict
|
- id: check-merge-conflict
|
||||||
@ -14,12 +14,12 @@ repos:
|
|||||||
- id: trailing-whitespace
|
- id: trailing-whitespace
|
||||||
- id: name-tests-test
|
- id: name-tests-test
|
||||||
exclude: tests/(common.py|util.py|(helpers|integration/factories)/(.+).py)
|
exclude: tests/(common.py|util.py|(helpers|integration/factories)/(.+).py)
|
||||||
- repo: https://github.com/pycqa/isort
|
- repo: https://github.com/asottile/reorder_python_imports
|
||||||
rev: 5.13.2
|
rev: v3.0.1
|
||||||
hooks:
|
hooks:
|
||||||
- id: isort
|
- id: reorder-python-imports
|
||||||
- repo: https://github.com/pre-commit/mirrors-mypy
|
- repo: https://github.com/pre-commit/mirrors-mypy
|
||||||
rev: v1.10.0
|
rev: v0.942
|
||||||
hooks:
|
hooks:
|
||||||
- id: mypy
|
- id: mypy
|
||||||
exclude: docs/
|
exclude: docs/
|
||||||
|
2
Makefile
2
Makefile
@ -38,7 +38,7 @@ build: $(ENV)/bin/hatch
|
|||||||
|
|
||||||
# Verify that the python version matches the git tag so we don't push bad shas
|
# Verify that the python version matches the git tag so we don't push bad shas
|
||||||
.PHONY: verify-tag-version
|
.PHONY: verify-tag-version
|
||||||
verify-tag-version: $(ENV)/bin/hatch
|
verify-tag-version:
|
||||||
$(eval TAG_NAME = $(shell [ -n "$(DRONE_TAG)" ] && echo $(DRONE_TAG) || git describe --tags --exact-match))
|
$(eval TAG_NAME = $(shell [ -n "$(DRONE_TAG)" ] && echo $(DRONE_TAG) || git describe --tags --exact-match))
|
||||||
test "v$(shell $(ENV)/bin/hatch version)" = "$(TAG_NAME)"
|
test "v$(shell $(ENV)/bin/hatch version)" = "$(TAG_NAME)"
|
||||||
|
|
||||||
|
@ -13,8 +13,6 @@
|
|||||||
# sys.path.insert(0, os.path.abspath('.'))
|
# sys.path.insert(0, os.path.abspath('.'))
|
||||||
# -- Project information -----------------------------------------------------
|
# -- Project information -----------------------------------------------------
|
||||||
|
|
||||||
from __future__ import annotations
|
|
||||||
|
|
||||||
project = "release-gitter"
|
project = "release-gitter"
|
||||||
copyright = "2021, iamthefij"
|
copyright = "2021, iamthefij"
|
||||||
author = "iamthefij"
|
author = "iamthefij"
|
||||||
|
@ -2,13 +2,13 @@
|
|||||||
This builder functions as a pseudo builder that instead downloads and installs a binary file using
|
This builder functions as a pseudo builder that instead downloads and installs a binary file using
|
||||||
release-gitter based on a pyproject.toml file. It's a total hack...
|
release-gitter based on a pyproject.toml file. It's a total hack...
|
||||||
"""
|
"""
|
||||||
|
|
||||||
from __future__ import annotations
|
from __future__ import annotations
|
||||||
|
|
||||||
from dataclasses import dataclass
|
from dataclasses import dataclass
|
||||||
from pathlib import Path
|
from pathlib import Path
|
||||||
from shutil import copy
|
from shutil import copy
|
||||||
from shutil import copytree
|
from shutil import copytree
|
||||||
|
from shutil import move
|
||||||
|
|
||||||
import toml
|
import toml
|
||||||
from wheel.wheelfile import WheelFile
|
from wheel.wheelfile import WheelFile
|
||||||
@ -17,9 +17,11 @@ import release_gitter as rg
|
|||||||
from release_gitter import removeprefix
|
from release_gitter import removeprefix
|
||||||
|
|
||||||
|
|
||||||
|
PACKAGE_NAME = "pseudo"
|
||||||
|
|
||||||
|
|
||||||
@dataclass
|
@dataclass
|
||||||
class Config:
|
class Config:
|
||||||
name: str
|
|
||||||
format: str
|
format: str
|
||||||
git_url: str
|
git_url: str
|
||||||
hostname: str
|
hostname: str
|
||||||
@ -37,52 +39,56 @@ class Config:
|
|||||||
include_extra_files: list[str] | None = None
|
include_extra_files: list[str] | None = None
|
||||||
|
|
||||||
|
|
||||||
def download(config: Config, wheel_scripts: Path) -> list[Path]:
|
def download(config: Config) -> list[Path]:
|
||||||
"""Download and extract files to the wheel_scripts directory"""
|
release = rg.fetch_release(
|
||||||
return rg.download_release(
|
rg.GitRemoteInfo(config.hostname, config.owner, config.repo), config.version
|
||||||
rg.GitRemoteInfo(config.hostname, config.owner, config.repo),
|
)
|
||||||
wheel_scripts,
|
asset = rg.match_asset(
|
||||||
|
release,
|
||||||
config.format,
|
config.format,
|
||||||
version=config.version,
|
version=config.version,
|
||||||
system_mapping=config.map_system,
|
system_mapping=config.map_system,
|
||||||
arch_mapping=config.map_arch,
|
arch_mapping=config.map_arch,
|
||||||
extract_files=config.extract_files,
|
|
||||||
pre_release=config.pre_release,
|
|
||||||
exec=config.exec,
|
|
||||||
)
|
)
|
||||||
|
|
||||||
|
files = rg.download_asset(asset, extract_files=config.extract_files)
|
||||||
|
|
||||||
|
# Optionally execute post command
|
||||||
|
if config.exec:
|
||||||
|
rg.check_call(config.exec, shell=True)
|
||||||
|
|
||||||
|
return files
|
||||||
|
|
||||||
|
|
||||||
def read_metadata() -> Config:
|
def read_metadata() -> Config:
|
||||||
"""Read configuration from pyproject.toml"""
|
config = toml.load("pyproject.toml").get("tool", {}).get("release-gitter")
|
||||||
pyproject = toml.load("pyproject.toml").get("tool", {}).get("release-gitter")
|
if not config:
|
||||||
if not pyproject:
|
|
||||||
raise ValueError("Must have configuration in [tool.release-gitter]")
|
raise ValueError("Must have configuration in [tool.release-gitter]")
|
||||||
|
|
||||||
git_url = pyproject.pop("git-url", None)
|
git_url = config.pop("git-url")
|
||||||
remote_info = rg.parse_git_url(git_url)
|
remote_info = rg.parse_git_remote(git_url)
|
||||||
|
|
||||||
config = Config(
|
args = Config(
|
||||||
name=pyproject.pop("name", remote_info.repo),
|
format=config.pop("format"),
|
||||||
format=pyproject.pop("format"),
|
|
||||||
git_url=git_url,
|
git_url=git_url,
|
||||||
hostname=pyproject.pop("hostname", remote_info.hostname),
|
hostname=config.pop("hostname", remote_info.hostname),
|
||||||
owner=pyproject.pop("owner", remote_info.owner),
|
owner=config.pop("owner", remote_info.owner),
|
||||||
repo=pyproject.pop("repo", remote_info.repo),
|
repo=config.pop("repo", remote_info.repo),
|
||||||
)
|
)
|
||||||
|
|
||||||
for key, value in pyproject.items():
|
for key, value in config.items():
|
||||||
setattr(config, str(key).replace("-", "_"), value)
|
setattr(args, str(key).replace("-", "_"), value)
|
||||||
|
|
||||||
if config.version is None:
|
if args.version is None:
|
||||||
config.version = rg.read_version(
|
args.version = rg.read_version(
|
||||||
config.version_git_tag,
|
args.version_git_tag,
|
||||||
not config.version_git_no_fetch,
|
not args.version_git_no_fetch,
|
||||||
)
|
)
|
||||||
|
|
||||||
if config.extract_all:
|
if args.extract_all:
|
||||||
config.extract_files = []
|
args.extract_files = []
|
||||||
|
|
||||||
return config
|
return args
|
||||||
|
|
||||||
|
|
||||||
class _PseudoBuildBackend:
|
class _PseudoBuildBackend:
|
||||||
@ -99,7 +105,7 @@ class _PseudoBuildBackend:
|
|||||||
version = removeprefix(metadata.version, "v") if metadata.version else "0.0.0"
|
version = removeprefix(metadata.version, "v") if metadata.version else "0.0.0"
|
||||||
|
|
||||||
# Returns distinfo dir?
|
# Returns distinfo dir?
|
||||||
dist_info = Path(metadata_directory) / f"{metadata.name}-{version}.dist-info"
|
dist_info = Path(metadata_directory) / f"{PACKAGE_NAME}-{version}.dist-info"
|
||||||
dist_info.mkdir()
|
dist_info.mkdir()
|
||||||
|
|
||||||
# Write metadata
|
# Write metadata
|
||||||
@ -108,7 +114,7 @@ class _PseudoBuildBackend:
|
|||||||
"\n".join(
|
"\n".join(
|
||||||
[
|
[
|
||||||
"Metadata-Version: 2.1",
|
"Metadata-Version: 2.1",
|
||||||
f"Name: {metadata.name}",
|
f"Name: {PACKAGE_NAME}",
|
||||||
f"Version: {version}",
|
f"Version: {version}",
|
||||||
]
|
]
|
||||||
)
|
)
|
||||||
@ -140,8 +146,6 @@ class _PseudoBuildBackend:
|
|||||||
def build_wheel(
|
def build_wheel(
|
||||||
self, wheel_directory, config_settings=None, metadata_directory=None
|
self, wheel_directory, config_settings=None, metadata_directory=None
|
||||||
):
|
):
|
||||||
if metadata_directory is None:
|
|
||||||
raise ValueError("Cannot build wheel without metadata_directory")
|
|
||||||
metadata_directory = Path(metadata_directory)
|
metadata_directory = Path(metadata_directory)
|
||||||
|
|
||||||
metadata = read_metadata()
|
metadata = read_metadata()
|
||||||
@ -150,13 +154,15 @@ class _PseudoBuildBackend:
|
|||||||
wheel_directory = Path(wheel_directory)
|
wheel_directory = Path(wheel_directory)
|
||||||
wheel_directory.mkdir(exist_ok=True)
|
wheel_directory.mkdir(exist_ok=True)
|
||||||
|
|
||||||
wheel_scripts = wheel_directory / f"{metadata.name}-{version}.data/scripts"
|
wheel_scripts = wheel_directory / f"{PACKAGE_NAME}-{version}.data/scripts"
|
||||||
wheel_scripts.mkdir(parents=True, exist_ok=True)
|
wheel_scripts.mkdir(parents=True, exist_ok=True)
|
||||||
|
|
||||||
copytree(metadata_directory, wheel_directory / metadata_directory.name)
|
copytree(metadata_directory, wheel_directory / metadata_directory.name)
|
||||||
|
|
||||||
metadata = read_metadata()
|
metadata = read_metadata()
|
||||||
download(metadata, wheel_scripts)
|
files = download(metadata)
|
||||||
|
for file in files:
|
||||||
|
move(file, wheel_scripts / file.name)
|
||||||
|
|
||||||
for file_name in metadata.include_extra_files or []:
|
for file_name in metadata.include_extra_files or []:
|
||||||
file = Path(file_name)
|
file = Path(file_name)
|
||||||
@ -169,11 +175,11 @@ class _PseudoBuildBackend:
|
|||||||
|
|
||||||
print(f"ls {wheel_directory}: {list(wheel_directory.rglob('*'))}")
|
print(f"ls {wheel_directory}: {list(wheel_directory.rglob('*'))}")
|
||||||
|
|
||||||
wheel_filename = f"{metadata.name}-{version}-py2.py3-none-any.whl"
|
wheel_filename = f"{PACKAGE_NAME}-{version}-py2.py3-none-any.whl"
|
||||||
with WheelFile(wheel_directory / wheel_filename, "w") as wf:
|
with WheelFile(wheel_directory / wheel_filename, "w") as wf:
|
||||||
print("Repacking wheel as {}...".format(wheel_filename), end="")
|
print("Repacking wheel as {}...".format(wheel_filename), end="")
|
||||||
# sys.stdout.flush()
|
# sys.stdout.flush()
|
||||||
wf.write_files(str(wheel_directory))
|
wf.write_files(wheel_directory)
|
||||||
|
|
||||||
return wheel_filename
|
return wheel_filename
|
||||||
|
|
||||||
|
@ -1,46 +0,0 @@
|
|||||||
from __future__ import annotations
|
|
||||||
|
|
||||||
import shutil
|
|
||||||
import subprocess
|
|
||||||
import venv
|
|
||||||
from pathlib import Path
|
|
||||||
from unittest import TestCase
|
|
||||||
|
|
||||||
ITEST_VENV_PATH = Path("venv-itest")
|
|
||||||
|
|
||||||
|
|
||||||
class TestPseudoBuilder(TestCase):
|
|
||||||
def setUp(self):
|
|
||||||
venv.create(
|
|
||||||
ITEST_VENV_PATH,
|
|
||||||
system_site_packages=False,
|
|
||||||
clear=True,
|
|
||||||
with_pip=True,
|
|
||||||
)
|
|
||||||
self.pip_install("-e", ".[builder]")
|
|
||||||
|
|
||||||
def tearDown(self):
|
|
||||||
shutil.rmtree(ITEST_VENV_PATH)
|
|
||||||
|
|
||||||
def pip_install(self, *args: str):
|
|
||||||
subprocess.run(
|
|
||||||
[str(ITEST_VENV_PATH.joinpath("bin", "pip")), "install", *args],
|
|
||||||
check=True,
|
|
||||||
)
|
|
||||||
|
|
||||||
def test_install_remote_package(self):
|
|
||||||
self.assertTrue(ITEST_VENV_PATH.exists())
|
|
||||||
self.assertTrue(ITEST_VENV_PATH.joinpath("bin", "python").exists())
|
|
||||||
self.assertTrue(ITEST_VENV_PATH.joinpath("bin", "pip").exists())
|
|
||||||
|
|
||||||
itest_packages = {
|
|
||||||
"stylua": "git+https://github.com/JohnnyMorganz/StyLua",
|
|
||||||
"selene": "git+https://github.com/amitds1997/selene",
|
|
||||||
}
|
|
||||||
|
|
||||||
for package, source in itest_packages.items():
|
|
||||||
self.pip_install("--no-index", "--no-build-isolation", source)
|
|
||||||
# Check if the package is installed
|
|
||||||
assert ITEST_VENV_PATH.joinpath("bin", package).exists()
|
|
||||||
# Check if the package has executable permissions
|
|
||||||
assert ITEST_VENV_PATH.joinpath("bin", package).stat().st_mode & 0o111
|
|
@ -48,12 +48,12 @@ dependencies = [
|
|||||||
[tool.hatch.envs.test.scripts]
|
[tool.hatch.envs.test.scripts]
|
||||||
run = [
|
run = [
|
||||||
"coverage erase",
|
"coverage erase",
|
||||||
"coverage run --source=release_gitter,pseudo_builder -m unittest discover -p '*_test.py'",
|
"coverage run --source=release_gitter -m unittest discover . *_test.py",
|
||||||
"coverage report -m # --fail-under 70",
|
"coverage report -m # --fail-under 70",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[tool.hatch.envs.test.matrix]]
|
[[tool.hatch.envs.test.matrix]]
|
||||||
python = ["3", "3.7", "3.8", "3.9", "3.10", "3.11", "3.12"]
|
python = ["3", "3.7", "3.8", "3.9", "3.10", "3.11"]
|
||||||
|
|
||||||
[tool.hatch.envs.lint]
|
[tool.hatch.envs.lint]
|
||||||
detached = true
|
detached = true
|
||||||
@ -62,8 +62,3 @@ dependencies = ["pre-commit"]
|
|||||||
[tool.hatch.envs.lint.scripts]
|
[tool.hatch.envs.lint.scripts]
|
||||||
all = "pre-commit run --all-files"
|
all = "pre-commit run --all-files"
|
||||||
install-hooks = "pre-commit install --install-hooks"
|
install-hooks = "pre-commit install --install-hooks"
|
||||||
|
|
||||||
[tool.isort]
|
|
||||||
add_imports = ["from __future__ import annotations"]
|
|
||||||
force_single_line = true
|
|
||||||
profile = "black"
|
|
||||||
|
@ -2,13 +2,10 @@
|
|||||||
from __future__ import annotations
|
from __future__ import annotations
|
||||||
|
|
||||||
import argparse
|
import argparse
|
||||||
import logging
|
|
||||||
import platform
|
import platform
|
||||||
import tempfile
|
|
||||||
from collections.abc import Sequence
|
from collections.abc import Sequence
|
||||||
from dataclasses import dataclass
|
from dataclasses import dataclass
|
||||||
from io import BytesIO
|
from io import BytesIO
|
||||||
from itertools import product
|
|
||||||
from mimetypes import guess_type
|
from mimetypes import guess_type
|
||||||
from pathlib import Path
|
from pathlib import Path
|
||||||
from subprocess import check_call
|
from subprocess import check_call
|
||||||
@ -16,16 +13,12 @@ from subprocess import check_output
|
|||||||
from tarfile import TarFile
|
from tarfile import TarFile
|
||||||
from tarfile import TarInfo
|
from tarfile import TarInfo
|
||||||
from typing import Any
|
from typing import Any
|
||||||
from typing import NamedTuple
|
|
||||||
from urllib.parse import urlparse
|
from urllib.parse import urlparse
|
||||||
from zipfile import ZipFile
|
from zipfile import ZipFile
|
||||||
|
|
||||||
import requests
|
import requests
|
||||||
|
|
||||||
__version__ = "3.0.3"
|
__version__ = "2.2.0"
|
||||||
|
|
||||||
|
|
||||||
logging.basicConfig(level=logging.WARNING)
|
|
||||||
|
|
||||||
|
|
||||||
class UnsupportedContentTypeError(ValueError):
|
class UnsupportedContentTypeError(ValueError):
|
||||||
@ -54,37 +47,6 @@ def removesuffix(s: str, suf: str) -> str:
|
|||||||
return s[: -len(suf)] if s and s.endswith(suf) else s
|
return s[: -len(suf)] if s and s.endswith(suf) else s
|
||||||
|
|
||||||
|
|
||||||
SYSTEM_SYNONYMS: list[list[str]] = [
|
|
||||||
["Darwin", "darwin", "MacOS", "macos", "macOS"],
|
|
||||||
["Windows", "windows", "win", "win32", "win64"],
|
|
||||||
["Linux", "linux"],
|
|
||||||
]
|
|
||||||
|
|
||||||
ARCH_SYNONYMS: list[list[str]] = [
|
|
||||||
["arm"],
|
|
||||||
["x86_64", "amd64", "AMD64"],
|
|
||||||
["arm64", "aarch64", "armv8b", "armv8l"],
|
|
||||||
["x86", "i386", "i686"],
|
|
||||||
]
|
|
||||||
|
|
||||||
|
|
||||||
def get_synonyms(value: str, thesaurus: list[list[str]]) -> list[str]:
|
|
||||||
"""Gets synonym list for a given value."""
|
|
||||||
results = [value]
|
|
||||||
|
|
||||||
for l in thesaurus:
|
|
||||||
if value in l:
|
|
||||||
results += l
|
|
||||||
|
|
||||||
return results
|
|
||||||
|
|
||||||
|
|
||||||
class MatchedValues(NamedTuple):
|
|
||||||
version: str
|
|
||||||
system: str
|
|
||||||
arch: str
|
|
||||||
|
|
||||||
|
|
||||||
@dataclass
|
@dataclass
|
||||||
class GitRemoteInfo:
|
class GitRemoteInfo:
|
||||||
"""Extracts information about a repository"""
|
"""Extracts information about a repository"""
|
||||||
@ -126,13 +88,13 @@ class GitRemoteInfo:
|
|||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
def read_git_remote() -> str:
|
def parse_git_remote(git_url: str | None = None) -> GitRemoteInfo:
|
||||||
"""Reads the git remote url from the origin"""
|
|
||||||
return check_output(["git", "remote", "get-url", "origin"]).decode("UTF-8").strip()
|
|
||||||
|
|
||||||
|
|
||||||
def parse_git_url(git_url: str) -> GitRemoteInfo:
|
|
||||||
"""Extract Github repo info from a git remote url"""
|
"""Extract Github repo info from a git remote url"""
|
||||||
|
if not git_url:
|
||||||
|
git_url = (
|
||||||
|
check_output(["git", "remote", "get-url", "origin"]).decode("UTF-8").strip()
|
||||||
|
)
|
||||||
|
|
||||||
# Normalize Github ssh url as a proper URL
|
# Normalize Github ssh url as a proper URL
|
||||||
if git_url.startswith("git@github.com:"):
|
if git_url.startswith("git@github.com:"):
|
||||||
git_ssh_parts = git_url.partition(":")
|
git_ssh_parts = git_url.partition(":")
|
||||||
@ -179,7 +141,6 @@ def read_git_tag(fetch: bool = True) -> str | None:
|
|||||||
def read_version(from_tags: bool = False, fetch: bool = False) -> str | None:
|
def read_version(from_tags: bool = False, fetch: bool = False) -> str | None:
|
||||||
"""Read version information from file or from git"""
|
"""Read version information from file or from git"""
|
||||||
if from_tags:
|
if from_tags:
|
||||||
logging.debug("Reading version from git tag")
|
|
||||||
return read_git_tag(fetch)
|
return read_git_tag(fetch)
|
||||||
|
|
||||||
matchers = {
|
matchers = {
|
||||||
@ -189,13 +150,10 @@ def read_version(from_tags: bool = False, fetch: bool = False) -> str | None:
|
|||||||
for name, extractor in matchers.items():
|
for name, extractor in matchers.items():
|
||||||
p = Path(name)
|
p = Path(name)
|
||||||
if p.exists():
|
if p.exists():
|
||||||
logging.debug(f"Reading version from {p}")
|
|
||||||
return extractor(p)
|
return extractor(p)
|
||||||
|
|
||||||
logging.warning(
|
# TODO: Log this out to stderr
|
||||||
"Unknown local project version. Didn't find any of %s", set(matchers.keys())
|
# raise ValueError(f"Unknown project type. Didn't find any of {matchers.keys()}")
|
||||||
)
|
|
||||||
|
|
||||||
return None
|
return None
|
||||||
|
|
||||||
|
|
||||||
@ -218,8 +176,6 @@ def fetch_release(
|
|||||||
|
|
||||||
# Return the latest if requested
|
# Return the latest if requested
|
||||||
if version is None or version == "latest":
|
if version is None or version == "latest":
|
||||||
logging.debug("Looking for latest release")
|
|
||||||
|
|
||||||
for release in result.json():
|
for release in result.json():
|
||||||
if release["prerelease"] and not pre_release:
|
if release["prerelease"] and not pre_release:
|
||||||
continue
|
continue
|
||||||
@ -229,8 +185,6 @@ def fetch_release(
|
|||||||
# Return matching version
|
# Return matching version
|
||||||
for release in result.json():
|
for release in result.json():
|
||||||
if release["tag_name"].endswith(version):
|
if release["tag_name"].endswith(version):
|
||||||
logging.debug(f"Found release {release['name']} matching version {version}")
|
|
||||||
|
|
||||||
return release
|
return release
|
||||||
|
|
||||||
raise ValueError(
|
raise ValueError(
|
||||||
@ -245,7 +199,7 @@ def match_asset(
|
|||||||
version: str | None = None,
|
version: str | None = None,
|
||||||
system_mapping: dict[str, str] | None = None,
|
system_mapping: dict[str, str] | None = None,
|
||||||
arch_mapping: dict[str, str] | None = None,
|
arch_mapping: dict[str, str] | None = None,
|
||||||
) -> tuple[dict[Any, Any], MatchedValues]:
|
) -> dict[Any, Any]:
|
||||||
"""Accepts a release and searches for an appropriate asset attached using
|
"""Accepts a release and searches for an appropriate asset attached using
|
||||||
a provided template and some alternative mappings for version, system, and machine info
|
a provided template and some alternative mappings for version, system, and machine info
|
||||||
|
|
||||||
@ -281,39 +235,37 @@ def match_asset(
|
|||||||
|
|
||||||
# This should never really happen
|
# This should never really happen
|
||||||
if version is None:
|
if version is None:
|
||||||
raise ValueError("No version provided or found in release name.")
|
if "{version}" in format:
|
||||||
|
raise ValueError(
|
||||||
|
"No version provided or found in release name but is in format"
|
||||||
|
)
|
||||||
|
else:
|
||||||
|
# This should never happen, but since version isn't used anywhere, we can make it an empty string
|
||||||
|
version = ""
|
||||||
|
|
||||||
system = platform.system()
|
system = platform.system()
|
||||||
if system_mapping:
|
if system_mapping:
|
||||||
systems = [system_mapping.get(system, system)]
|
system = system_mapping.get(system, system)
|
||||||
else:
|
|
||||||
systems = get_synonyms(system, SYSTEM_SYNONYMS)
|
|
||||||
|
|
||||||
arch = platform.machine()
|
arch = platform.machine()
|
||||||
if arch_mapping:
|
if arch_mapping:
|
||||||
archs = [arch_mapping.get(arch, arch)]
|
arch = arch_mapping.get(arch, arch)
|
||||||
else:
|
|
||||||
archs = get_synonyms(arch, ARCH_SYNONYMS)
|
|
||||||
|
|
||||||
expected_names = {
|
expected_names = {
|
||||||
format.format(
|
format.format(
|
||||||
version=version_opt,
|
version=normalized_version,
|
||||||
system=system_opt,
|
system=system,
|
||||||
arch=arch_opt,
|
arch=arch,
|
||||||
): MatchedValues(version=version_opt, system=system_opt, arch=arch_opt)
|
)
|
||||||
for version_opt, system_opt, arch_opt in product(
|
for normalized_version in (
|
||||||
(
|
version.lstrip("v"),
|
||||||
version.lstrip("v"),
|
"v" + version if not version.startswith("v") else version,
|
||||||
"v" + version if not version.startswith("v") else version,
|
|
||||||
),
|
|
||||||
systems,
|
|
||||||
archs,
|
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
for asset in release["assets"]:
|
for asset in release["assets"]:
|
||||||
if asset["name"] in expected_names:
|
if asset["name"] in expected_names:
|
||||||
return (asset, expected_names[asset["name"]])
|
return asset
|
||||||
|
|
||||||
raise ValueError(
|
raise ValueError(
|
||||||
f"Could not find asset named {expected_names} on release {release['name']}"
|
f"Could not find asset named {expected_names} on release {release['name']}"
|
||||||
@ -329,12 +281,8 @@ class PackageAdapter:
|
|||||||
"application/zip",
|
"application/zip",
|
||||||
"application/x-zip-compressed",
|
"application/x-zip-compressed",
|
||||||
):
|
):
|
||||||
logging.debug("Opening zip file from response content")
|
|
||||||
|
|
||||||
self._package = ZipFile(BytesIO(response.content))
|
self._package = ZipFile(BytesIO(response.content))
|
||||||
elif content_type == "application/x-tar":
|
elif content_type == "application/x-tar":
|
||||||
logging.debug("Opening tar file from response content")
|
|
||||||
|
|
||||||
self._package = TarFile(fileobj=response.raw)
|
self._package = TarFile(fileobj=response.raw)
|
||||||
elif content_type in (
|
elif content_type in (
|
||||||
"application/gzip",
|
"application/gzip",
|
||||||
@ -342,8 +290,6 @@ class PackageAdapter:
|
|||||||
"application/x-tar+xz",
|
"application/x-tar+xz",
|
||||||
"application/x-compressed-tar",
|
"application/x-compressed-tar",
|
||||||
):
|
):
|
||||||
logging.debug("Opening compressed tar file from response content")
|
|
||||||
|
|
||||||
self._package = TarFile.open(fileobj=BytesIO(response.content), mode="r:*")
|
self._package = TarFile.open(fileobj=BytesIO(response.content), mode="r:*")
|
||||||
else:
|
else:
|
||||||
raise UnsupportedContentTypeError(
|
raise UnsupportedContentTypeError(
|
||||||
@ -354,7 +300,6 @@ class PackageAdapter:
|
|||||||
"""Get list of all file names in package"""
|
"""Get list of all file names in package"""
|
||||||
if isinstance(self._package, ZipFile):
|
if isinstance(self._package, ZipFile):
|
||||||
return self._package.namelist()
|
return self._package.namelist()
|
||||||
|
|
||||||
if isinstance(self._package, TarFile):
|
if isinstance(self._package, TarFile):
|
||||||
return self._package.getnames()
|
return self._package.getnames()
|
||||||
|
|
||||||
@ -372,26 +317,19 @@ class PackageAdapter:
|
|||||||
If the `file_names` list is empty, all files will be extracted"""
|
If the `file_names` list is empty, all files will be extracted"""
|
||||||
if path is None:
|
if path is None:
|
||||||
path = Path.cwd()
|
path = Path.cwd()
|
||||||
|
|
||||||
if not members:
|
if not members:
|
||||||
logging.debug("Extracting all members to %s", path)
|
|
||||||
|
|
||||||
self._package.extractall(path=path)
|
self._package.extractall(path=path)
|
||||||
|
|
||||||
return self.get_names()
|
return self.get_names()
|
||||||
|
|
||||||
# TODO: Use walrus operator when dropping 3.7 support
|
|
||||||
missing_members = set(members) - set(self.get_names())
|
missing_members = set(members) - set(self.get_names())
|
||||||
if missing_members:
|
if missing_members:
|
||||||
raise ValueError(f"Missing members: {missing_members}")
|
raise ValueError(f"Missing members: {missing_members}")
|
||||||
|
|
||||||
logging.debug("Extracting members %s to %s", members, path)
|
|
||||||
|
|
||||||
if isinstance(self._package, ZipFile):
|
if isinstance(self._package, ZipFile):
|
||||||
self._package.extractall(path=path, members=members)
|
self._package.extractall(path=path, members=members)
|
||||||
if isinstance(self._package, TarFile):
|
if isinstance(self._package, TarFile):
|
||||||
self._package.extractall(
|
self._package.extractall(
|
||||||
path=path, members=(self._package.getmember(name) for name in members)
|
path=path, members=(TarInfo(name) for name in members)
|
||||||
)
|
)
|
||||||
|
|
||||||
return members
|
return members
|
||||||
@ -414,7 +352,7 @@ def get_asset_package(
|
|||||||
continue
|
continue
|
||||||
else:
|
else:
|
||||||
raise UnsupportedContentTypeError(
|
raise UnsupportedContentTypeError(
|
||||||
f"Cannot extract files from archive because we don't recognize the content types {possible_content_types}"
|
"Cannot extract files from archive because we don't recognize the content type"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
@ -441,10 +379,8 @@ def download_asset(
|
|||||||
result = requests.get(asset["browser_download_url"])
|
result = requests.get(asset["browser_download_url"])
|
||||||
|
|
||||||
if extract_files is not None:
|
if extract_files is not None:
|
||||||
logging.info("Extracting package %s", asset["name"])
|
|
||||||
package = get_asset_package(asset, result)
|
package = get_asset_package(asset, result)
|
||||||
extract_files = package.extractall(path=destination, members=extract_files)
|
extract_files = package.extractall(path=destination, members=extract_files)
|
||||||
|
|
||||||
return [destination / name for name in extract_files]
|
return [destination / name for name in extract_files]
|
||||||
|
|
||||||
file_name = destination / asset["name"]
|
file_name = destination / asset["name"]
|
||||||
@ -491,7 +427,6 @@ class MapAddAction(argparse.Action):
|
|||||||
|
|
||||||
|
|
||||||
def _parse_args(args: list[str] | None = None) -> argparse.Namespace:
|
def _parse_args(args: list[str] | None = None) -> argparse.Namespace:
|
||||||
logging.debug("Parsing arguments: %s", args)
|
|
||||||
parser = argparse.ArgumentParser()
|
parser = argparse.ArgumentParser()
|
||||||
parser.add_argument(
|
parser.add_argument(
|
||||||
"format",
|
"format",
|
||||||
@ -505,9 +440,7 @@ def _parse_args(args: list[str] | None = None) -> argparse.Namespace:
|
|||||||
default=Path.cwd(),
|
default=Path.cwd(),
|
||||||
help="Destination directory. Defaults to current directory",
|
help="Destination directory. Defaults to current directory",
|
||||||
)
|
)
|
||||||
parser.add_argument(
|
parser.add_argument("-v", action="store_true", help="verbose logging")
|
||||||
"-v", action="count", help="verbose or debug logging", default=0
|
|
||||||
)
|
|
||||||
parser.add_argument(
|
parser.add_argument(
|
||||||
"--hostname",
|
"--hostname",
|
||||||
help="Git repository hostname",
|
help="Git repository hostname",
|
||||||
@ -578,29 +511,12 @@ def _parse_args(args: list[str] | None = None) -> argparse.Namespace:
|
|||||||
action="store_true",
|
action="store_true",
|
||||||
help="Only print the URL and do not download",
|
help="Only print the URL and do not download",
|
||||||
)
|
)
|
||||||
parser.add_argument(
|
|
||||||
"--use-temp-dir",
|
|
||||||
action="store_true",
|
|
||||||
help="Use a temporary directory as the destination",
|
|
||||||
)
|
|
||||||
|
|
||||||
parsed_args = parser.parse_args(args)
|
parsed_args = parser.parse_args(args)
|
||||||
|
|
||||||
# Merge in fields from args and git remote
|
# Merge in fields from args and git remote
|
||||||
if not all((parsed_args.owner, parsed_args.repo, parsed_args.hostname)):
|
if not all((parsed_args.owner, parsed_args.repo, parsed_args.hostname)):
|
||||||
# Check to see if a git url was provided. If not, we use local directory git remote
|
remote_info = parse_git_remote(parsed_args.git_url)
|
||||||
if parsed_args.git_url is None:
|
|
||||||
parsed_args.git_url = read_git_remote()
|
|
||||||
|
|
||||||
# If using a local repo, try to determine version from project files
|
|
||||||
if parsed_args.version is None:
|
|
||||||
parsed_args.version = read_version(
|
|
||||||
parsed_args.version_git_tag,
|
|
||||||
not parsed_args.version_git_no_fetch,
|
|
||||||
)
|
|
||||||
|
|
||||||
# Get parts from git url
|
|
||||||
remote_info = parse_git_url(parsed_args.git_url)
|
|
||||||
|
|
||||||
def merge_field(a, b, field):
|
def merge_field(a, b, field):
|
||||||
value = getattr(a, field)
|
value = getattr(a, field)
|
||||||
@ -610,12 +526,15 @@ def _parse_args(args: list[str] | None = None) -> argparse.Namespace:
|
|||||||
for field in ("owner", "repo", "hostname"):
|
for field in ("owner", "repo", "hostname"):
|
||||||
merge_field(parsed_args, remote_info, field)
|
merge_field(parsed_args, remote_info, field)
|
||||||
|
|
||||||
|
if parsed_args.version is None:
|
||||||
|
parsed_args.version = read_version(
|
||||||
|
parsed_args.version_git_tag,
|
||||||
|
not parsed_args.version_git_no_fetch,
|
||||||
|
)
|
||||||
|
|
||||||
if parsed_args.extract_all:
|
if parsed_args.extract_all:
|
||||||
parsed_args.extract_files = []
|
parsed_args.extract_files = []
|
||||||
|
|
||||||
if parsed_args.use_temp_dir:
|
|
||||||
parsed_args.destination = Path(tempfile.mkdtemp())
|
|
||||||
|
|
||||||
return parsed_args
|
return parsed_args
|
||||||
|
|
||||||
|
|
||||||
@ -628,99 +547,55 @@ def download_release(
|
|||||||
arch_mapping: dict[str, str] | None = None,
|
arch_mapping: dict[str, str] | None = None,
|
||||||
extract_files: list[str] | None = None,
|
extract_files: list[str] | None = None,
|
||||||
pre_release=False,
|
pre_release=False,
|
||||||
exec: str | None = None,
|
|
||||||
) -> list[Path]:
|
) -> list[Path]:
|
||||||
"""Convenience method for fetching, downloading, and extracting a release
|
"""Convenience method for fetching, downloading and extracting a release"""
|
||||||
|
|
||||||
This is slightly different than running off the commandline, it will execute the shell script
|
|
||||||
from the destination directory, not the current working directory.
|
|
||||||
"""
|
|
||||||
release = fetch_release(
|
release = fetch_release(
|
||||||
remote_info,
|
remote_info,
|
||||||
version=version,
|
version=version,
|
||||||
pre_release=pre_release,
|
pre_release=pre_release,
|
||||||
)
|
)
|
||||||
asset, matched_values = match_asset(
|
asset = match_asset(
|
||||||
release,
|
release,
|
||||||
format,
|
format,
|
||||||
version=version,
|
version=version,
|
||||||
system_mapping=system_mapping,
|
system_mapping=system_mapping,
|
||||||
arch_mapping=arch_mapping,
|
arch_mapping=arch_mapping,
|
||||||
)
|
)
|
||||||
|
|
||||||
format_fields = dict(
|
|
||||||
asset_name=asset["name"],
|
|
||||||
**matched_values._asdict(),
|
|
||||||
)
|
|
||||||
|
|
||||||
formatted_files = (
|
|
||||||
[file.format(**format_fields) for file in extract_files]
|
|
||||||
if extract_files is not None
|
|
||||||
else None
|
|
||||||
)
|
|
||||||
|
|
||||||
files = download_asset(
|
files = download_asset(
|
||||||
asset,
|
asset,
|
||||||
extract_files=formatted_files,
|
extract_files=extract_files,
|
||||||
destination=destination,
|
destination=destination,
|
||||||
)
|
)
|
||||||
|
|
||||||
if exec:
|
|
||||||
check_call(
|
|
||||||
exec.format(asset["name"], **format_fields), shell=True, cwd=destination
|
|
||||||
)
|
|
||||||
|
|
||||||
return files
|
return files
|
||||||
|
|
||||||
|
|
||||||
def main():
|
def main():
|
||||||
args = _parse_args()
|
args = _parse_args()
|
||||||
|
|
||||||
logging.getLogger().setLevel(30 - 10 * args.v)
|
|
||||||
|
|
||||||
# Fetch the release
|
|
||||||
release = fetch_release(
|
release = fetch_release(
|
||||||
GitRemoteInfo(args.hostname, args.owner, args.repo),
|
GitRemoteInfo(args.hostname, args.owner, args.repo),
|
||||||
version=args.version,
|
version=args.version,
|
||||||
pre_release=args.prerelease,
|
pre_release=args.prerelease,
|
||||||
)
|
)
|
||||||
|
asset = match_asset(
|
||||||
logging.debug("Found release: %s", release["name"])
|
|
||||||
|
|
||||||
version = args.version or release["tag_name"]
|
|
||||||
|
|
||||||
logging.debug("Release version: %s", version)
|
|
||||||
|
|
||||||
# Find the asset to download using mapping rules
|
|
||||||
asset, matched_values = match_asset(
|
|
||||||
release,
|
release,
|
||||||
args.format,
|
args.format,
|
||||||
version=version,
|
version=args.version,
|
||||||
system_mapping=args.map_system,
|
system_mapping=args.map_system,
|
||||||
arch_mapping=args.map_arch,
|
arch_mapping=args.map_arch,
|
||||||
)
|
)
|
||||||
|
|
||||||
logging.info(f"Downloading {asset['name']} from release {release['name']}")
|
if args.v:
|
||||||
|
print(f"Downloading {asset['name']} from release {release['name']}")
|
||||||
|
|
||||||
if args.url_only:
|
if args.url_only:
|
||||||
print(asset["browser_download_url"])
|
print(asset["browser_download_url"])
|
||||||
return
|
return
|
||||||
|
|
||||||
format_fields = dict(
|
|
||||||
asset_name=asset["name"],
|
|
||||||
**matched_values._asdict(),
|
|
||||||
)
|
|
||||||
|
|
||||||
# Format files to extract with version info, as this is sometimes included
|
|
||||||
formatted_files = (
|
|
||||||
[file.format(**format_fields) for file in args.extract_files]
|
|
||||||
if args.extract_files is not None
|
|
||||||
else None
|
|
||||||
)
|
|
||||||
|
|
||||||
files = download_asset(
|
files = download_asset(
|
||||||
asset,
|
asset,
|
||||||
extract_files=formatted_files,
|
extract_files=args.extract_files,
|
||||||
destination=args.destination,
|
destination=args.destination,
|
||||||
)
|
)
|
||||||
|
|
||||||
@ -728,11 +603,7 @@ def main():
|
|||||||
|
|
||||||
# Optionally execute post command
|
# Optionally execute post command
|
||||||
if args.exec:
|
if args.exec:
|
||||||
check_call(
|
check_call(args.exec.format(asset["name"]), shell=True)
|
||||||
args.exec.format(asset["name"], **format_fields),
|
|
||||||
shell=True,
|
|
||||||
cwd=args.destination,
|
|
||||||
)
|
|
||||||
|
|
||||||
|
|
||||||
if __name__ == "__main__":
|
if __name__ == "__main__":
|
||||||
|
@ -1,12 +1,11 @@
|
|||||||
from __future__ import annotations
|
from __future__ import annotations
|
||||||
|
|
||||||
import unittest
|
import unittest
|
||||||
from itertools import chain
|
|
||||||
from itertools import product
|
|
||||||
from tarfile import TarFile
|
from tarfile import TarFile
|
||||||
from typing import Any
|
from typing import Any
|
||||||
from typing import Callable
|
from typing import Callable
|
||||||
from typing import NamedTuple
|
from typing import NamedTuple
|
||||||
|
from typing import Optional
|
||||||
from unittest.mock import MagicMock
|
from unittest.mock import MagicMock
|
||||||
from unittest.mock import mock_open
|
from unittest.mock import mock_open
|
||||||
from unittest.mock import patch
|
from unittest.mock import patch
|
||||||
@ -22,11 +21,10 @@ class TestExpression(NamedTuple):
|
|||||||
args: list[Any]
|
args: list[Any]
|
||||||
kwargs: dict[str, Any]
|
kwargs: dict[str, Any]
|
||||||
expected: Any
|
expected: Any
|
||||||
exception: type[Exception] | None = None
|
exception: Optional[type[Exception]] = None
|
||||||
msg: str | None = None
|
|
||||||
|
|
||||||
def run(self, f: Callable):
|
def run(self, f: Callable):
|
||||||
with self.t.subTest(msg=self.msg, f=f, args=self.args, kwargs=self.kwargs):
|
with self.t.subTest(f=f, args=self.args, kwargs=self.kwargs):
|
||||||
try:
|
try:
|
||||||
result = f(*self.args, **self.kwargs)
|
result = f(*self.args, **self.kwargs)
|
||||||
self.t.assertIsNone(
|
self.t.assertIsNone(
|
||||||
@ -82,7 +80,7 @@ class TestRemoteInfo(unittest.TestCase):
|
|||||||
release_gitter.InvalidRemoteError,
|
release_gitter.InvalidRemoteError,
|
||||||
),
|
),
|
||||||
):
|
):
|
||||||
test_case.run(release_gitter.parse_git_url)
|
test_case.run(release_gitter.parse_git_remote)
|
||||||
|
|
||||||
def test_generate_release_url(self):
|
def test_generate_release_url(self):
|
||||||
for subtest in (
|
for subtest in (
|
||||||
@ -199,284 +197,5 @@ class TestContentTypeDetection(unittest.TestCase):
|
|||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
def first_result(f):
|
|
||||||
def wrapper(*args, **kwargs):
|
|
||||||
return f(*args, **kwargs)[0]
|
|
||||||
|
|
||||||
return wrapper
|
|
||||||
|
|
||||||
|
|
||||||
class TestMatchAsset(unittest.TestCase):
|
|
||||||
def test_match_asset_versions(self, *_):
|
|
||||||
# Input variations:
|
|
||||||
# Case 1: Version provided with prefix
|
|
||||||
# Case 2: Version provided without prefix
|
|
||||||
# Case 3: No version provided, tag exists in release
|
|
||||||
# These should be impossible
|
|
||||||
# Case 4: No version provided, tag doesn't exist in release but not in template
|
|
||||||
# Case 5: No version provided, tag doesn't exist in release and is in template
|
|
||||||
|
|
||||||
# Release variations:
|
|
||||||
# Case 1: tag_name with version prefix
|
|
||||||
# Case 2: tag_name without version prefix
|
|
||||||
|
|
||||||
# File variations:
|
|
||||||
# Case 1: file name with version prefix
|
|
||||||
# Case 2: file name without version prefix
|
|
||||||
|
|
||||||
def new_expression(version: str | None, tag_name: str, file_name: str):
|
|
||||||
release = {"tag_name": tag_name, "assets": [{"name": file_name}]}
|
|
||||||
expected = {"name": file_name}
|
|
||||||
return TestExpression(
|
|
||||||
self, [release, "file-{version}.zip", version], {}, expected
|
|
||||||
)
|
|
||||||
|
|
||||||
happy_cases = [
|
|
||||||
new_expression(version, tag_name, file_name)
|
|
||||||
for version, tag_name, file_name in product(
|
|
||||||
("v1.0.0", "1.0.0", None),
|
|
||||||
("v1.0.0", "1.0.0"),
|
|
||||||
("file-v1.0.0.zip", "file-1.0.0.zip"),
|
|
||||||
)
|
|
||||||
]
|
|
||||||
for test_case in happy_cases:
|
|
||||||
test_case.run(first_result(release_gitter.match_asset))
|
|
||||||
|
|
||||||
def test_match_asset_systems(self, *_):
|
|
||||||
# Input variations:
|
|
||||||
# Case 1: System mapping provided
|
|
||||||
# Case 2: No system mapping provided
|
|
||||||
|
|
||||||
# Test: We want to show that default matching will work out of the box with some values for the current machine
|
|
||||||
# Test: We want to show that non-standard mappings will always work if provided manually
|
|
||||||
|
|
||||||
def run_with_context(actual_system: str, *args, **kwargs):
|
|
||||||
with patch("platform.system", return_value=actual_system):
|
|
||||||
return release_gitter.match_asset(*args, **kwargs)
|
|
||||||
|
|
||||||
def new_expression(
|
|
||||||
actual_system: str,
|
|
||||||
system_mapping: dict[str, str] | None,
|
|
||||||
file_name: str,
|
|
||||||
expected: dict[str, str],
|
|
||||||
exception: type[Exception] | None = None,
|
|
||||||
msg: str | None = None,
|
|
||||||
):
|
|
||||||
release = {
|
|
||||||
"name": "v1.0.0",
|
|
||||||
"tag_name": "v1.0.0",
|
|
||||||
"assets": [{"name": file_name}],
|
|
||||||
}
|
|
||||||
return TestExpression(
|
|
||||||
self,
|
|
||||||
[actual_system, release, "file-{system}.zip"],
|
|
||||||
{"system_mapping": system_mapping},
|
|
||||||
expected,
|
|
||||||
exception,
|
|
||||||
msg,
|
|
||||||
)
|
|
||||||
|
|
||||||
test_cases = chain(
|
|
||||||
[
|
|
||||||
new_expression(
|
|
||||||
"Earth",
|
|
||||||
None,
|
|
||||||
"file-Earth.zip",
|
|
||||||
{"name": "file-Earth.zip"},
|
|
||||||
msg="Current system always included as an exact match synonym",
|
|
||||||
),
|
|
||||||
new_expression(
|
|
||||||
"Linux",
|
|
||||||
{"Linux": "jumanji"},
|
|
||||||
"file-jumanji.zip",
|
|
||||||
{"name": "file-jumanji.zip"},
|
|
||||||
msg="Non-standard system mapping works",
|
|
||||||
),
|
|
||||||
new_expression(
|
|
||||||
"Linux",
|
|
||||||
{},
|
|
||||||
"file-darwin.zip",
|
|
||||||
{},
|
|
||||||
ValueError,
|
|
||||||
msg="No matching system",
|
|
||||||
),
|
|
||||||
],
|
|
||||||
# Test default mappings
|
|
||||||
(
|
|
||||||
new_expression(
|
|
||||||
actual_system,
|
|
||||||
None,
|
|
||||||
file_name,
|
|
||||||
{"name": file_name},
|
|
||||||
msg="Default Linux mappings",
|
|
||||||
)
|
|
||||||
for actual_system, file_name in product(
|
|
||||||
("Linux", "linux"),
|
|
||||||
("file-Linux.zip", "file-linux.zip"),
|
|
||||||
)
|
|
||||||
),
|
|
||||||
(
|
|
||||||
new_expression(
|
|
||||||
actual_system,
|
|
||||||
None,
|
|
||||||
file_name,
|
|
||||||
{"name": file_name},
|
|
||||||
msg="Default macOS mappings",
|
|
||||||
)
|
|
||||||
for actual_system, file_name in product(
|
|
||||||
("Darwin", "darwin", "MacOS", "macos", "macOS"),
|
|
||||||
(
|
|
||||||
"file-Darwin.zip",
|
|
||||||
"file-darwin.zip",
|
|
||||||
"file-MacOS.zip",
|
|
||||||
"file-macos.zip",
|
|
||||||
),
|
|
||||||
)
|
|
||||||
),
|
|
||||||
(
|
|
||||||
new_expression(
|
|
||||||
actual_system,
|
|
||||||
None,
|
|
||||||
file_name,
|
|
||||||
{"name": file_name},
|
|
||||||
msg="Default Windows mappings",
|
|
||||||
)
|
|
||||||
for actual_system, file_name in product(
|
|
||||||
("Windows", "windows", "win", "win32", "win64"),
|
|
||||||
(
|
|
||||||
"file-Windows.zip",
|
|
||||||
"file-windows.zip",
|
|
||||||
"file-win.zip",
|
|
||||||
"file-win32.zip",
|
|
||||||
"file-win64.zip",
|
|
||||||
),
|
|
||||||
)
|
|
||||||
),
|
|
||||||
)
|
|
||||||
for test_case in test_cases:
|
|
||||||
test_case.run(first_result(run_with_context))
|
|
||||||
|
|
||||||
def test_match_asset_archs(self, *_):
|
|
||||||
# Input variations:
|
|
||||||
# Case 1: Arch mapping provided
|
|
||||||
# Case 2: No arch mapping provided
|
|
||||||
|
|
||||||
# Test: We want to show that default matching will work out of the box with some values for the current machine
|
|
||||||
# Test: We want to show that non-standard mappings will always work if provided manually
|
|
||||||
|
|
||||||
def run_with_context(actual_arch: str, *args, **kwargs):
|
|
||||||
with patch("platform.machine", return_value=actual_arch):
|
|
||||||
return release_gitter.match_asset(*args, **kwargs)
|
|
||||||
|
|
||||||
def new_expression(
|
|
||||||
actual_arch: str,
|
|
||||||
arch_mapping: dict[str, str] | None,
|
|
||||||
file_name: str,
|
|
||||||
expected: dict[str, str],
|
|
||||||
exception: type[Exception] | None = None,
|
|
||||||
msg: str | None = None,
|
|
||||||
):
|
|
||||||
release = {
|
|
||||||
"name": "v1.0.0",
|
|
||||||
"tag_name": "v1.0.0",
|
|
||||||
"assets": [{"name": file_name}],
|
|
||||||
}
|
|
||||||
return TestExpression(
|
|
||||||
self,
|
|
||||||
[actual_arch, release, "file-{arch}.zip"],
|
|
||||||
{"arch_mapping": arch_mapping},
|
|
||||||
expected,
|
|
||||||
exception,
|
|
||||||
msg,
|
|
||||||
)
|
|
||||||
|
|
||||||
test_cases = chain(
|
|
||||||
[
|
|
||||||
new_expression(
|
|
||||||
"Earth",
|
|
||||||
None,
|
|
||||||
"file-Earth.zip",
|
|
||||||
{"name": "file-Earth.zip"},
|
|
||||||
msg="Current arch always included as an exact match synonym",
|
|
||||||
),
|
|
||||||
new_expression(
|
|
||||||
"x86_64",
|
|
||||||
{"x86_64": "jumanji"},
|
|
||||||
"file-jumanji.zip",
|
|
||||||
{"name": "file-jumanji.zip"},
|
|
||||||
msg="Non-standard arch mapping works",
|
|
||||||
),
|
|
||||||
new_expression(
|
|
||||||
"x86_64",
|
|
||||||
{},
|
|
||||||
"file-arm.zip",
|
|
||||||
{},
|
|
||||||
ValueError,
|
|
||||||
msg="No matching arch",
|
|
||||||
),
|
|
||||||
],
|
|
||||||
# Test default mappings
|
|
||||||
(
|
|
||||||
new_expression(
|
|
||||||
actual_arch,
|
|
||||||
None,
|
|
||||||
file_name,
|
|
||||||
{"name": file_name},
|
|
||||||
msg="Default arm mappings",
|
|
||||||
)
|
|
||||||
for actual_arch, file_name in product(
|
|
||||||
("arm",),
|
|
||||||
("file-arm.zip",),
|
|
||||||
)
|
|
||||||
),
|
|
||||||
(
|
|
||||||
new_expression(
|
|
||||||
actual_arch,
|
|
||||||
None,
|
|
||||||
file_name,
|
|
||||||
{"name": file_name},
|
|
||||||
msg="Default amd64 mappings",
|
|
||||||
)
|
|
||||||
for actual_arch, file_name in product(
|
|
||||||
("amd64", "x86_64", "AMD64"),
|
|
||||||
("file-amd64.zip", "file-x86_64.zip"),
|
|
||||||
)
|
|
||||||
),
|
|
||||||
(
|
|
||||||
new_expression(
|
|
||||||
actual_arch,
|
|
||||||
None,
|
|
||||||
file_name,
|
|
||||||
{"name": file_name},
|
|
||||||
msg="Default arm64 mappings",
|
|
||||||
)
|
|
||||||
for actual_arch, file_name in product(
|
|
||||||
("arm64", "aarch64", "armv8b", "armv8l"),
|
|
||||||
(
|
|
||||||
"file-arm64.zip",
|
|
||||||
"file-aarch64.zip",
|
|
||||||
"file-armv8b.zip",
|
|
||||||
"file-armv8l.zip",
|
|
||||||
),
|
|
||||||
)
|
|
||||||
),
|
|
||||||
(
|
|
||||||
new_expression(
|
|
||||||
actual_arch,
|
|
||||||
None,
|
|
||||||
file_name,
|
|
||||||
{"name": file_name},
|
|
||||||
msg="Default x86 mappings",
|
|
||||||
)
|
|
||||||
for actual_arch, file_name in product(
|
|
||||||
("x86", "i386", "i686"),
|
|
||||||
("file-x86.zip", "file-i386.zip", "file-i686.zip"),
|
|
||||||
)
|
|
||||||
),
|
|
||||||
)
|
|
||||||
for test_case in test_cases:
|
|
||||||
test_case.run(first_result(run_with_context))
|
|
||||||
|
|
||||||
|
|
||||||
if __name__ == "__main__":
|
if __name__ == "__main__":
|
||||||
unittest.main()
|
unittest.main()
|
||||||
|
Loading…
x
Reference in New Issue
Block a user