1
0
mirror of https://github.com/openSUSE/osc.git synced 2025-01-04 05:46:16 +01:00
github.com_openSUSE_osc/osc/util/packagequery.py

192 lines
5.4 KiB
Python
Raw Normal View History

2022-09-09 11:46:28 +02:00
import sys
from .helper import decode_it
class PackageError(Exception):
"""base class for all package related errors"""
2022-09-12 15:40:48 +02:00
def __init__(self, fname, msg):
super().__init__()
self.fname = fname
self.msg = msg
2022-09-12 15:40:48 +02:00
class PackageQueries(dict):
"""Dict of package name keys and package query values. When assigning a
package query, to a name, the package is evaluated to see if it matches the
wanted architecture and if it has a greater version than the current value.
"""
2010-02-28 02:30:13 +01:00
2016-08-04 12:40:36 +02:00
# map debian and rpm arches to common obs arches
architectureMap = {'i386': ['i586', 'i686'], 'amd64': ['x86_64'], 'ppc64el': ['ppc64le'], 'armv6hl': ['armv6l'], 'armv7hl': ['armv7l']}
2010-02-28 02:30:13 +01:00
2010-02-27 20:11:15 +01:00
def __init__(self, wanted_architecture):
self.wanted_architecture = wanted_architecture
super().__init__()
2010-02-28 02:30:13 +01:00
def add(self, query):
"""Adds package query to dict if it is of the correct architecture and
is newer (has a greater version) than the currently assigned package.
2010-02-28 02:30:13 +01:00
:param query: a PackageQuery
"""
self.__setitem__(query.name(), query)
2010-02-28 02:30:13 +01:00
def __setitem__(self, name, query):
if decode_it(name) != decode_it(query.name()):
raise ValueError("key '%s' does not match "
"package query name '%s'" % (name, query.name()))
2010-02-28 02:30:13 +01:00
architecture = decode_it(query.arch())
2010-02-28 02:30:13 +01:00
if (architecture in [self.wanted_architecture, 'noarch', 'all', 'any']
or self.wanted_architecture in self.architectureMap.get(architecture,
2022-09-12 15:40:48 +02:00
[])):
2010-02-27 20:11:15 +01:00
current_query = self.get(name)
2010-02-28 02:30:13 +01:00
# if current query does not exist or is older than this new query
2010-02-27 20:11:15 +01:00
if current_query is None or current_query.vercmp(query) <= 0:
super().__setitem__(name, query)
2022-09-12 15:40:48 +02:00
2009-09-20 19:19:33 +02:00
class PackageQuery:
"""abstract base class for all package types"""
2022-09-12 15:40:48 +02:00
def read(self, all_tags=False, *extra_tags):
"""Returns a PackageQueryResult instance"""
raise NotImplementedError
# Hmmm... this should be a module function (inherting this stuff
# does not make much sense) (the same is true for the queryhdrmd5 method)
@staticmethod
def query(filename, all_tags=False, extra_rpmtags=(), extra_debtags=(), self_provides=True):
f = open(filename, 'rb')
magic = f.read(7)
f.seek(0)
extra_tags = ()
pkgquery = None
if magic[:4] == b'\xed\xab\xee\xdb':
from . import rpmquery
pkgquery = rpmquery.RpmQuery(f)
extra_tags = extra_rpmtags
elif magic == b'!<arch>':
from . import debquery
pkgquery = debquery.DebQuery(f)
extra_tags = extra_debtags
elif magic == b'<update':
f.close()
return None
elif magic[:5] == b'<?xml':
f.close()
return None
# arch tar ball compressed with gz, xz or zst
elif magic[:5] == b'\375\067zXZ' or magic[:2] == b'\037\213' or magic[:4] == b'(\xb5/\xfd':
from . import archquery
pkgquery = archquery.ArchQuery(f)
else:
raise PackageError(filename, 'unsupported package type. magic: \'%s\'' % magic)
pkgqueryresult = pkgquery.read(all_tags, self_provides, *extra_tags)
f.close()
return pkgqueryresult
@staticmethod
def queryhdrmd5(filename):
f = open(filename, 'rb')
magic = f.read(7)
f.seek(0)
if magic[:4] == b'\xed\xab\xee\xdb':
from . import rpmquery
f.close()
return rpmquery.RpmQuery.queryhdrmd5(filename)
return None
class PackageQueryResult:
"""abstract base class that represents the result of a package query"""
2022-09-12 15:40:48 +02:00
def name(self):
raise NotImplementedError
def version(self):
raise NotImplementedError
def release(self):
raise NotImplementedError
def epoch(self):
raise NotImplementedError
def arch(self):
raise NotImplementedError
def description(self):
raise NotImplementedError
2010-02-28 02:30:13 +01:00
def path(self):
raise NotImplementedError
2010-02-28 02:30:13 +01:00
def provides(self):
raise NotImplementedError
def requires(self):
raise NotImplementedError
def conflicts(self):
raise NotImplementedError
def obsoletes(self):
raise NotImplementedError
def recommends(self):
raise NotImplementedError
def suggests(self):
raise NotImplementedError
def supplements(self):
raise NotImplementedError
def enhances(self):
raise NotImplementedError
def gettag(self, tag):
raise NotImplementedError
2010-02-27 20:11:15 +01:00
def vercmp(self, pkgquery):
raise NotImplementedError
def canonname(self):
raise NotImplementedError
def evr(self):
evr = self.version()
if self.release():
evr += b"-" + self.release()
epoch = self.epoch()
if epoch is not None and epoch != 0:
2022-05-20 12:00:44 +02:00
evr = epoch + b":" + evr
return evr
def cmp(a, b):
return (a > b) - (a < b)
if __name__ == '__main__':
try:
pkgq = PackageQuery.query(sys.argv[1])
except PackageError as e:
print(e.msg)
sys.exit(2)
print(pkgq.name())
print(pkgq.version())
print(pkgq.release())
print(pkgq.description())
print('##########')
print('\n'.join(pkgq.provides()))
print('##########')
print('\n'.join(pkgq.requires()))