D7net
Home
Console
Upload
information
Create File
Create Folder
About
Tools
:
/
opt
/
saltstack
/
salt
/
lib
/
python3.10
/
site-packages
/
salt
/
utils
/
pkg
/
Filename :
rpm.py
back
Copy
""" Common functions for working with RPM packages """ import collections import datetime import logging import platform import subprocess import salt.utils.path import salt.utils.stringutils log = logging.getLogger(__name__) # These arches compiled from the rpmUtils.arch python module source ARCHES_64 = ("x86_64", "athlon", "amd64", "ia32e", "ia64", "geode") ARCHES_32 = ("i386", "i486", "i586", "i686") ARCHES_PPC = ("ppc", "ppc64", "ppc64le", "ppc64iseries", "ppc64pseries") ARCHES_S390 = ("s390", "s390x") ARCHES_SPARC = ("sparc", "sparcv8", "sparcv9", "sparcv9v", "sparc64", "sparc64v") ARCHES_ALPHA = ( "alpha", "alphaev4", "alphaev45", "alphaev5", "alphaev56", "alphapca56", "alphaev6", "alphaev67", "alphaev68", "alphaev7", ) ARCHES_ARM_32 = ( "armv5tel", "armv5tejl", "armv6l", "armv6hl", "armv7l", "armv7hl", "armv7hnl", ) ARCHES_ARM_64 = ("aarch64",) ARCHES_SH = ("sh3", "sh4", "sh4a") ARCHES = ( ARCHES_64 + ARCHES_32 + ARCHES_PPC + ARCHES_S390 + ARCHES_ALPHA + ARCHES_ARM_32 + ARCHES_ARM_64 + ARCHES_SH ) # EPOCHNUM can't be used until RHEL5 is EOL as it is not present QUERYFORMAT = "%{NAME}_|-%{EPOCH}_|-%{VERSION}_|-%{RELEASE}_|-%{ARCH}_|-%{REPOID}_|-%{INSTALLTIME}" def get_osarch(): """ Get the os architecture using rpm --eval """ if salt.utils.path.which("rpm"): ret = subprocess.Popen( ["rpm", "--eval", "%{_host_cpu}"], close_fds=True, stdout=subprocess.PIPE, stderr=subprocess.PIPE, ).communicate()[0] else: ret = "".join([x for x in platform.uname()[-2:] if x][-1:]) return salt.utils.stringutils.to_str(ret).strip() or "unknown" def check_32(arch, osarch=None): """ Returns True if both the OS arch and the passed arch are x86 or ARM 32-bit """ if osarch is None: osarch = get_osarch() return all(x in ARCHES_32 for x in (osarch, arch)) or all( x in ARCHES_ARM_32 for x in (osarch, arch) ) def pkginfo(name, version, arch, repoid, install_date=None, install_date_time_t=None): """ Build and return a pkginfo namedtuple """ pkginfo_tuple = collections.namedtuple( "PkgInfo", ("name", "version", "arch", "repoid", "install_date", "install_date_time_t"), ) return pkginfo_tuple(name, version, arch, repoid, install_date, install_date_time_t) def resolve_name(name, arch, osarch=None): """ Resolve the package name and arch into a unique name referred to by salt. For example, on a 64-bit OS, a 32-bit package will be pkgname.i386. """ if osarch is None: osarch = get_osarch() if not check_32(arch, osarch) and arch not in (osarch, "noarch"): name += ".{}".format(arch) return name def parse_pkginfo(line, osarch=None): """ A small helper to parse an rpm/repoquery command's output. Returns a pkginfo namedtuple. """ try: name, epoch, version, release, arch, repoid, install_time = line.split("_|-") # Handle unpack errors (should never happen with the queryformat we are # using, but can't hurt to be careful). except ValueError: return None name = resolve_name(name, arch, osarch) if release: version += "-{}".format(release) if epoch not in ("(none)", "0"): version = ":".join((epoch, version)) if install_time not in ("(none)", "0"): install_date = ( datetime.datetime.utcfromtimestamp(int(install_time)).isoformat() + "Z" ) install_date_time_t = int(install_time) else: install_date = None install_date_time_t = None return pkginfo(name, version, arch, repoid, install_date, install_date_time_t) def combine_comments(comments): """ Given a list of comments, strings, a single comment or a single string, return a single string of text containing all of the comments, prepending the '#' and joining with newlines as necessary. """ if not isinstance(comments, list): comments = [comments] ret = [] for comment in comments: if not isinstance(comment, str): comment = str(comment) # Normalize for any spaces (or lack thereof) after the # ret.append("# {}\n".format(comment.lstrip("#").lstrip())) return "".join(ret) def version_to_evr(verstring): """ Split the package version string into epoch, version and release. Return this as tuple. The epoch is always not empty. The version and the release can be an empty string if such a component could not be found in the version string. "2:1.0-1.2" => ('2', '1.0', '1.2) "1.0" => ('0', '1.0', '') "" => ('0', '', '') """ if verstring in [None, ""]: return "0", "", "" idx_e = verstring.find(":") if idx_e != -1: try: epoch = str(int(verstring[:idx_e])) except ValueError: # look, garbage in the epoch field, how fun, kill it epoch = "0" # this is our fallback, deal else: epoch = "0" idx_r = verstring.find("-") if idx_r != -1: version = verstring[idx_e + 1 : idx_r] release = verstring[idx_r + 1 :] else: version = verstring[idx_e + 1 :] release = "" return epoch, version, release