forked from Mirror/pmbootstrap
Also remove args from functions that don't actually need it to faciliate this. This does not attempt to fix any of the bugs with pkgrel_bump.
132 lines
4.7 KiB
Python
132 lines
4.7 KiB
Python
# Copyright 2023 Oliver Smith
|
|
# SPDX-License-Identifier: GPL-3.0-or-later
|
|
from pmb.core.arch import Arch
|
|
from pmb.helpers import logging
|
|
|
|
import pmb.helpers.file
|
|
import pmb.helpers.pmaports
|
|
import pmb.helpers.repo
|
|
import pmb.parse
|
|
import pmb.parse.apkindex
|
|
|
|
|
|
def package(pkgname: str, reason="", dry: bool = False) -> None:
|
|
"""Increase the pkgrel in the APKBUILD of a specific package.
|
|
|
|
:param pkgname: name of the package
|
|
:param reason: string to display as reason why it was increased
|
|
:param dry: don't modify the APKBUILD, just print the message
|
|
"""
|
|
# Current and new pkgrel
|
|
path = pmb.helpers.pmaports.find(pkgname) / "APKBUILD"
|
|
apkbuild = pmb.parse.apkbuild(path)
|
|
pkgrel = int(apkbuild["pkgrel"])
|
|
pkgrel_new = pkgrel + 1
|
|
|
|
# Display the message, bail out in dry mode
|
|
logging.info(
|
|
"Increase '"
|
|
+ pkgname
|
|
+ "' pkgrel ("
|
|
+ str(pkgrel)
|
|
+ " -> "
|
|
+ str(pkgrel_new)
|
|
+ ")"
|
|
+ reason
|
|
)
|
|
if dry:
|
|
return
|
|
|
|
# Increase
|
|
old = "\npkgrel=" + str(pkgrel) + "\n"
|
|
new = "\npkgrel=" + str(pkgrel_new) + "\n"
|
|
pmb.helpers.file.replace(path, old, new)
|
|
|
|
# Verify
|
|
pmb.parse.apkbuild.cache_clear()
|
|
apkbuild = pmb.parse.apkbuild(path)
|
|
if int(apkbuild["pkgrel"]) != pkgrel_new:
|
|
raise RuntimeError(
|
|
f"Failed to bump pkgrel for package '{pkgname}'."
|
|
" Make sure that there's a line with exactly the"
|
|
f" string '{old.strip()}' and nothing else in: {path}"
|
|
)
|
|
|
|
|
|
def auto_apkindex_package(arch, aport, apk, dry: bool = False) -> bool:
|
|
"""Bump the pkgrel of a specific package if it is outdated in the given APKINDEX.
|
|
|
|
:param arch: the architecture, e.g. "armhf"
|
|
:param aport: parsed APKBUILD of the binary package's origin:
|
|
{"pkgname": ..., "pkgver": ..., "pkgrel": ..., ...}
|
|
:param apk: information about the binary package from the APKINDEX:
|
|
{"version": ..., "depends": [...], ...}
|
|
:param dry: don't modify the APKBUILD, just print the message
|
|
:returns: True when there was an APKBUILD that needed to be changed.
|
|
"""
|
|
version_aport = aport["pkgver"] + "-r" + aport["pkgrel"]
|
|
version_apk = apk["version"]
|
|
pkgname = aport["pkgname"]
|
|
|
|
# Skip when aport version != binary package version
|
|
compare = pmb.parse.version.compare(version_aport, version_apk)
|
|
if compare == -1:
|
|
logging.warning(
|
|
f"{pkgname}: skipping, because the aport version {version_aport} is lower"
|
|
f" than the binary version {version_apk}"
|
|
)
|
|
return False
|
|
if compare == 1:
|
|
logging.verbose(
|
|
f"{pkgname}: skipping, because the aport version {version_aport} is higher"
|
|
f" than the binary version {version_apk}"
|
|
)
|
|
return False
|
|
|
|
# Find missing depends
|
|
depends = apk["depends"]
|
|
logging.verbose("{}: checking depends: {}".format(pkgname, ", ".join(depends)))
|
|
missing = []
|
|
for depend in depends:
|
|
if depend.startswith("!"):
|
|
# Ignore conflict-dependencies
|
|
continue
|
|
|
|
providers = pmb.parse.apkindex.providers(depend, arch, must_exist=False)
|
|
if providers == {}:
|
|
# We're only interested in missing depends starting with "so:"
|
|
# (which means dynamic libraries that the package was linked
|
|
# against) and packages for which no aport exists.
|
|
if depend.startswith("so:") or not pmb.helpers.pmaports.find_optional(depend):
|
|
missing.append(depend)
|
|
|
|
# Increase pkgrel
|
|
if len(missing):
|
|
package(pkgname, reason=", missing depend(s): " + ", ".join(missing), dry=dry)
|
|
return True
|
|
|
|
return False
|
|
|
|
|
|
def auto(dry=False) -> list[str]:
|
|
""":returns: list of aport names, where the pkgrel needed to be changed"""
|
|
ret = []
|
|
for arch in Arch.supported():
|
|
paths = pmb.helpers.repo.apkindex_files(arch, exclude_mirrors=["alpine"])
|
|
for path in paths:
|
|
logging.info(f"scan {path}")
|
|
index = pmb.parse.apkindex.parse(path, False)
|
|
for pkgname, apk in index.items():
|
|
origin = apk["origin"]
|
|
# Only increase once!
|
|
if origin in ret:
|
|
logging.verbose(f"{pkgname}: origin '{origin}' found again")
|
|
continue
|
|
aport_path = pmb.helpers.pmaports.find_optional(origin)
|
|
if not aport_path:
|
|
logging.warning(f"{pkgname}: origin '{origin}' aport not found")
|
|
continue
|
|
aport = pmb.parse.apkbuild(aport_path)
|
|
if auto_apkindex_package(arch, aport, apk, dry):
|
|
ret.append(pkgname)
|
|
return ret
|