| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124 |
- from __future__ import annotations
- import os.path
- import re
- from wheel.cli import WheelError
- from wheel.wheelfile import WheelFile
- DIST_INFO_RE = re.compile(r"^(?P<namever>(?P<name>.+?)-(?P<ver>\d.*?))\.dist-info$")
- BUILD_NUM_RE = re.compile(rb"Build: (\d\w*)$")
- def pack(directory: str, dest_dir: str, build_number: str | None):
- """Repack a previously unpacked wheel directory into a new wheel file.
- The .dist-info/WHEEL file must contain one or more tags so that the target
- wheel file name can be determined.
- :param directory: The unpacked wheel directory
- :param dest_dir: Destination directory (defaults to the current directory)
- """
- # Find the .dist-info directory
- dist_info_dirs = [
- fn
- for fn in os.listdir(directory)
- if os.path.isdir(os.path.join(directory, fn)) and DIST_INFO_RE.match(fn)
- ]
- if len(dist_info_dirs) > 1:
- raise WheelError(f"Multiple .dist-info directories found in {directory}")
- elif not dist_info_dirs:
- raise WheelError(f"No .dist-info directories found in {directory}")
- # Determine the target wheel filename
- dist_info_dir = dist_info_dirs[0]
- name_version = DIST_INFO_RE.match(dist_info_dir).group("namever")
- # Read the tags and the existing build number from .dist-info/WHEEL
- existing_build_number = None
- wheel_file_path = os.path.join(directory, dist_info_dir, "WHEEL")
- with open(wheel_file_path, "rb") as f:
- tags, existing_build_number = read_tags(f.read())
- if not tags:
- raise WheelError(
- "No tags present in {}/WHEEL; cannot determine target wheel "
- "filename".format(dist_info_dir)
- )
- # Set the wheel file name and add/replace/remove the Build tag in .dist-info/WHEEL
- build_number = build_number if build_number is not None else existing_build_number
- if build_number is not None:
- if build_number:
- name_version += "-" + build_number
- if build_number != existing_build_number:
- with open(wheel_file_path, "rb+") as f:
- wheel_file_content = f.read()
- wheel_file_content = set_build_number(wheel_file_content, build_number)
- f.seek(0)
- f.truncate()
- f.write(wheel_file_content)
- # Reassemble the tags for the wheel file
- tagline = compute_tagline(tags)
- # Repack the wheel
- wheel_path = os.path.join(dest_dir, f"{name_version}-{tagline}.whl")
- with WheelFile(wheel_path, "w") as wf:
- print(f"Repacking wheel as {wheel_path}...", end="", flush=True)
- wf.write_files(directory)
- print("OK")
- def read_tags(input_str: bytes) -> tuple[list[str], str | None]:
- """Read tags from a string.
- :param input_str: A string containing one or more tags, separated by spaces
- :return: A list of tags and a list of build tags
- """
- tags = []
- existing_build_number = None
- for line in input_str.splitlines():
- if line.startswith(b"Tag: "):
- tags.append(line.split(b" ")[1].rstrip().decode("ascii"))
- elif line.startswith(b"Build: "):
- existing_build_number = line.split(b" ")[1].rstrip().decode("ascii")
- return tags, existing_build_number
- def set_build_number(wheel_file_content: bytes, build_number: str | None) -> bytes:
- """Compute a build tag and add/replace/remove as necessary.
- :param wheel_file_content: The contents of .dist-info/WHEEL
- :param build_number: The build tags present in .dist-info/WHEEL
- :return: The (modified) contents of .dist-info/WHEEL
- """
- replacement = (
- ("Build: %s\r\n" % build_number).encode("ascii") if build_number else b""
- )
- wheel_file_content, num_replaced = BUILD_NUM_RE.subn(
- replacement, wheel_file_content
- )
- if not num_replaced:
- wheel_file_content += replacement
- return wheel_file_content
- def compute_tagline(tags: list[str]) -> str:
- """Compute a tagline from a list of tags.
- :param tags: A list of tags
- :return: A tagline
- """
- impls = sorted({tag.split("-")[0] for tag in tags})
- abivers = sorted({tag.split("-")[1] for tag in tags})
- platforms = sorted({tag.split("-")[2] for tag in tags})
- return "-".join([".".join(impls), ".".join(abivers), ".".join(platforms)])
|