[calm - Cygwin server-side packaging maintenance script] branch master, updated. 20220704-36-g516c780
Jon TURNEY
jturney@sourceware.org
Mon Dec 5 11:55:10 GMT 2022
https://sourceware.org/git/gitweb.cgi?p=cygwin-apps/calm.git;h=516c78090f3668c4c4ed131c51f9354e8b9400d0
commit 516c78090f3668c4c4ed131c51f9354e8b9400d0
Author: Jon Turney <jon.turney@dronecode.org.uk>
Date: Mon Dec 5 11:50:18 2022 +0000
Make mksetupini continue to work for legacy x86 architecture
https://sourceware.org/git/gitweb.cgi?p=cygwin-apps/calm.git;h=3e0eb64ae31b39a67111c0401e8cda7d0c43a482
commit 3e0eb64ae31b39a67111c0401e8cda7d0c43a482
Author: Jon Turney <jon.turney@dronecode.org.uk>
Date: Mon Dec 5 11:41:24 2022 +0000
Clean up some obsolete tools which are only useful when we have multiple arches
https://sourceware.org/git/gitweb.cgi?p=cygwin-apps/calm.git;h=babffd1ee53003c589de0e1374e531c4380561e9
commit babffd1ee53003c589de0e1374e531c4380561e9
Author: Jon Turney <jon.turney@dronecode.org.uk>
Date: Sun Dec 4 22:30:54 2022 +0000
Allow correct keep-superseded-test: key name in hint
FIx a thinko in 3282d5c9, where the key we allowed in the hint parser
and the key we checked for in package expiry are different.
Fixes: 3282d5c9
Diff:
---
calm/compare-arches | 129 -----------------------------
calm/dedupsrc.py | 214 ------------------------------------------------
calm/find-duplicates.py | 176 ---------------------------------------
calm/hint.py | 2 +-
calm/mksetupini.py | 2 +-
5 files changed, 2 insertions(+), 521 deletions(-)
diff --git a/calm/compare-arches b/calm/compare-arches
deleted file mode 100755
index 4e2dd19..0000000
--- a/calm/compare-arches
+++ /dev/null
@@ -1,129 +0,0 @@
-#!/usr/bin/env python3
-#
-# Copyright (c) 2015 Jon Turney
-#
-# Permission is hereby granted, free of charge, to any person obtaining a copy
-# of this software and associated documentation files (the "Software"), to deal
-# in the Software without restriction, including without limitation the rights
-# to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
-# copies of the Software, and to permit persons to whom the Software is
-# furnished to do so, subject to the following conditions:
-#
-# The above copyright notice and this permission notice shall be included in
-# all copies or substantial portions of the Software.
-#
-# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
-# IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
-# FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
-# AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
-# LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
-# OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
-# THE SOFTWARE.
-#
-
-#
-# compare-arch - compare package sets between architectures
-#
-
-import argparse
-import logging
-import os
-import sys
-
-import common_constants
-import maintainers
-import package
-
-
-#
-#
-#
-
-def main(args):
- packages = {}
- pset = {}
- union = set()
-
- for arch in common_constants.ARCHES:
- # build package list
- packages[arch] = package.read_packages(args.rel_area, arch)
-
- # make a set of the package list
- pset[arch] = frozenset(packages[arch].keys())
-
- # make the union of all the package lists
- union = union.union(pset[arch])
-
- # XXX: could filter out obsolete here
-
- for p in sorted(union):
- exists = {}
- obs = {}
-
- for arch in common_constants.ARCHES:
- exists[arch] = False
- obs[arch] = True
- if p in packages[arch]:
- exists[arch] = True
- if '_obsolete' in packages[arch][p].hints.get('category', ''):
- obs[arch] = True
- else:
- obs[arch] = False
-
- if p.startswith('cygwin32') or p.startswith('cygwin64') or p.endswith('-debuginfo'):
- continue
-
- # packages which exist in all arches
- if all(exists.values()):
-
- # report packages which are obsolete only in some arch
- if any(obs.values()) and not all(obs.values()):
- print("%s is in all arches, but obsolete only in %s" % (p, [a for a in obs if obs[a]]))
-
- continue
-
- # ignore packages which are obsolete in all the arches they exist in
- if all(obs.values()):
- continue
-
-
-# print(p, exists, obs)
- print("%s is only in arch %s" % (p, [a for a in exists if exists[a]]))
-
- # are there any packages which have a maintainer, but don't exist?
- mlist = maintainers.read(args, getattr(args, 'orphanmaint', None))
- all_packages = maintainers.all_packages(mlist)
-
- for p in sorted(all_packages):
- if p not in union:
- logging.warning("package '%s' has a maintainer, but doesn't exist in any architecture" % (p))
-
- # find the set of packages which aren't in an arch
-# for arch in common_constants.ARCHES:
-# diff = union.difference(pset[arch])
-# print("only in %s" % arch)
-# print(sorted(diff))
-
-
-#
-#
-#
-
-if __name__ == "__main__":
- homedir_default = common_constants.HOMEDIR
- pkglist_default = common_constants.PKGMAINT
- relarea_default = common_constants.FTP
-
- parser = argparse.ArgumentParser(description='Compare arch package sets')
- parser.add_argument('--homedir', action='store', metavar='DIR', help="maintainer home directory (default: " + homedir_default + ")", default=homedir_default)
- parser.add_argument('--pkglist', action='store', metavar='FILE', help="package maintainer list (default: " + pkglist_default + ")", default=pkglist_default)
- parser.add_argument('--releasearea', action='store', metavar='DIR', help="release directory (default: " + relarea_default + ")", default=relarea_default, dest='rel_area')
- parser.add_argument('-v', '--verbose', action='count', dest='verbose', help='verbose output')
- (args) = parser.parse_args()
-
- if args.verbose:
- logging.getLogger().setLevel(logging.INFO)
-
- logging.basicConfig(format=os.path.basename(sys.argv[0]) + ': %(message)s')
-
- main(args)
diff --git a/calm/dedupsrc.py b/calm/dedupsrc.py
deleted file mode 100644
index 2c71d52..0000000
--- a/calm/dedupsrc.py
+++ /dev/null
@@ -1,214 +0,0 @@
-#!/usr/bin/env python3
-#
-# Copyright (c) 2017 Jon Turney
-#
-# Permission is hereby granted, free of charge, to any person obtaining a copy
-# of this software and associated documentation files (the "Software"), to deal
-# in the Software without restriction, including without limitation the rights
-# to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
-# copies of the Software, and to permit persons to whom the Software is
-# furnished to do so, subject to the following conditions:
-#
-# The above copyright notice and this permission notice shall be included in
-# all copies or substantial portions of the Software.
-#
-# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
-# IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
-# FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
-# AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
-# LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
-# OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
-# THE SOFTWARE.
-#
-
-#
-# Move a given source archive to src/ (assuming it is indentical in x86/ and
-# x86_64/) and adjust hints appropriately.
-# (XXX: could probably be extended to move to noarch/ if not source, as well)
-#
-
-import argparse
-import copy
-import os
-import re
-import sys
-
-from . import common_constants
-from . import hint
-from . import utils
-
-binary_only_hints = ['requires', 'depends', 'obsoletes', 'external-source']
-
-#
-#
-#
-
-
-def hint_file_write(fn, hints):
- with open(fn, 'w') as f:
- for k, v in hints.items():
- print("%s: %s" % (k, v), file=f)
-
-
-#
-#
-#
-
-def invent_sdesc(path, vr):
- for (dirpath, _subdirs, files) in os.walk(path):
- # debuginfo packages never have a good sdesc
- if 'debuginfo' in dirpath:
- continue
-
- # but just pick the sdesc from first sub-package which has one ...
- for f in files:
- if re.match('^.*-' + re.escape(vr) + '.hint$', f):
- hints = hint.hint_file_parse(os.path.join(dirpath, f), hint.pvr)
- if 'sdesc' in hints:
- sdesc = hints['sdesc']
-
- # ... which doesn't contain 'Obsoleted'
- if 'Obsoleted' in sdesc:
- continue
-
- # remove anything inside parentheses at the end of quoted
- # sdesc
- sdesc = re.sub(r'"(.*)"', r'\1', sdesc)
- sdesc = re.sub(r'(\(.*?\))$', '', sdesc)
- sdesc = sdesc.strip()
- sdesc = '"' + sdesc + '"'
-
- return sdesc
-
- return None
-#
-#
-#
-
-
-def dedup(archive, relarea):
- # split path and filename
- (path, filename) = os.path.split(archive)
-
- # parse tarfile name
- match = re.match(r'^(.+?)-(\d.*)-src\.tar' +
- common_constants.PACKAGE_COMPRESSIONS_RE + r'$', filename)
-
- if not match:
- print('tarfile name %s does not meet expectations' % (filename))
- sys.exit(1)
-
- p = match.group(1)
- vr = match.group(2)
- ext = match.group(3)
-
- # compute filenames
- to_filename = p + '-src-' + vr + '.tar.' + ext
- hint_filename = p + '-' + vr + '.hint'
- to_hint_filename = p + '-src-' + vr + '.hint'
-
- # read hints for both arches
- hints = {}
- for arch in ['x86', 'x86_64']:
- hint_pathname = os.path.join(relarea, arch, path, hint_filename)
-
- if not os.path.exists(hint_pathname):
- print('%s not found' % (hint_pathname))
- return 1
-
- hints[arch] = hint.hint_file_parse(hint_pathname, hint.pvr)
-
- # remove hints which only have meaning for binary packages
- #
- # (requires: tends to have libgcc1 more often on x86, so otherwise this
- # would cause spurious differences between hints to be reported)
- for h in binary_only_hints:
- if h in hints[arch]:
- del hints[arch][h]
-
- if hints['x86'] != hints['x86_64']:
- print('hints for %s-%s differ between arches' % (p, vr))
- return 1
-
- if ('skip' in hints['x86']) and (len(hints['x86']) == 1):
- print('hints for %s-%s is skip: only' % (p, vr))
- hints['x86']['category'] = ''
- # if hint only contains skip:, try to come up with a plausible sdesc
- sdesc = invent_sdesc(os.path.join(relarea, 'x86', path), vr)
- if sdesc:
- print('suggested sdesc is %s' % (sdesc))
- hints['x86']['sdesc'] = sdesc
-
- if 'sdesc' not in hints['x86']:
- print('hints for %s-%s has no sdesc:' % (p, vr))
- return 1
-
- # ensure target directory exists
- utils.makedirs(os.path.join(relarea, 'src', path, p + '-src'))
-
- # write .hint file for new -src package
- src_hints = copy.copy(hints['x86'])
-
- if 'source' not in src_hints['sdesc']:
- sdesc = re.sub(r'"(.*)"', r'\1', src_hints['sdesc'])
- sdesc += ' (source code)'
- src_hints['sdesc'] = '"' + sdesc + '"'
-
- if 'Source' not in src_hints['category']:
- src_hints['category'] = src_hints['category'] + ' Source'
-
- if 'parse-warnings' in src_hints:
- del src_hints['parse-warnings']
-
- to_hint_pathname = os.path.join(relarea, 'src', path, p + '-src', to_hint_filename)
- print('writing %s' % (to_hint_pathname))
- hint_file_write(to_hint_pathname, src_hints)
-
- # move the src files to src/
- for arch in ['x86', 'x86_64']:
- print('%s -> %s' % (os.path.join(relarea, arch, path, filename), os.path.join(relarea, 'src', path, p + '-src', to_filename)))
- os.rename(os.path.join(relarea, arch, path, filename), os.path.join(relarea, 'src', path, p + '-src', to_filename))
-
- # adjust external-source in .hint for all subpackages
- for arch in ['x86', 'x86_64']:
- for (dirpath, _subdirs, files) in os.walk(os.path.join(relarea, arch, path)):
- subpkg = os.path.basename(dirpath)
- filename = subpkg + '-' + vr + '.hint'
- if filename in files:
- hint_pathname = os.path.join(dirpath, filename)
- hints = hint.hint_file_parse(hint_pathname, hint.pvr)
- if 'parse-warnings' in hints:
- del hints['parse-warnings']
- if ('skip' in hints):
- # p was source only, so no package remains
- print('removing %s' % (hint_pathname))
- os.remove(hint_pathname)
- elif ('external-source' not in hints) or (hints['external-source'] == p):
- hints['external-source'] = p + '-src'
- print('writing %s' % (hint_pathname))
- hint_file_write(hint_pathname, hints)
-
- return 0
-
-#
-#
-#
-
-
-def main():
- relarea_default = common_constants.FTP
-
- parser = argparse.ArgumentParser(description='Source package deduplicator')
- parser.add_argument('archive', metavar='ARCHIVE', nargs=1, help="source archive to deduplicate")
- parser.add_argument('--releasearea', action='store', metavar='DIR', help="release directory (default: " + relarea_default + ")", default=relarea_default, dest='rel_area')
- (args) = parser.parse_args()
-
- return dedup(args.archive[0], args.rel_area)
-
-
-#
-#
-#
-
-if __name__ == "__main__":
- sys.exit(main())
diff --git a/calm/find-duplicates.py b/calm/find-duplicates.py
deleted file mode 100644
index 59d8012..0000000
--- a/calm/find-duplicates.py
+++ /dev/null
@@ -1,176 +0,0 @@
-#!/usr/bin/env python3
-#
-# Copyright (c) 2017 Jon Turney
-#
-# Permission is hereby granted, free of charge, to any person obtaining a copy
-# of this software and associated documentation files (the "Software"), to deal
-# in the Software without restriction, including without limitation the rights
-# to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
-# copies of the Software, and to permit persons to whom the Software is
-# furnished to do so, subject to the following conditions:
-#
-# The above copyright notice and this permission notice shall be included in
-# all copies or substantial portions of the Software.
-#
-# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
-# IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
-# FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
-# AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
-# LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
-# OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
-# THE SOFTWARE.
-#
-
-import argparse
-import hashlib
-import os
-import re
-import sys
-import tarfile
-
-import xtarfile
-
-from . import common_constants
-
-#
-# look for archives which are duplicated between x86 and x86_64
-# (these should probably be moved to noarch or src)
-#
-
-#
-# helper function to compute sha512 for a particular file
-# (block_size should be some multiple of sha512 block size which can be
-# efficiently read)
-#
-
-
-def sha512_file(f, block_size=256 * 128):
- sha512 = hashlib.sha512()
-
- for chunk in iter(lambda: f.read(block_size), b''):
- sha512.update(chunk)
-
- return sha512.hexdigest()
-
-#
-#
-#
-
-
-class TarMemberInfo:
- def __init__(self, info, sha512):
- self.info = info
- self.sha512 = sha512
-
-
-def read_tar(f):
- result = {}
-
- try:
- with xtarfile.open(f, mode='r') as t:
- for m in t:
- if m.isfile():
- f = t.extractfile(m)
- sha512 = sha512_file(f)
- else:
- sha512 = None
- result[m.name] = TarMemberInfo(m, sha512)
- except tarfile.ReadError:
- # if we can't read the tar archive, we should never consider it to have
- # the same contents as another tar archive...
- result[f] = None
-
- return result
-
-#
-#
-#
-
-
-def compare_archives(f1, f2):
- # for speed, first check that archives are of the same size
- if os.path.getsize(f1) != os.path.getsize(f2):
- return 'different archive size'
-
- # if they are both compressed empty files (rather than compressed empty tar
- # archives), they are the same
- if os.path.getsize(f1) <= 32:
- return None
-
- t1 = read_tar(f1)
- t2 = read_tar(f2)
-
- if t1.keys() != t2.keys():
- return 'different member lists'
-
- for m in t1:
- # compare size of member
- if t1[m].info.size != t2[m].info.size:
- return 'different size for member %s' % m
-
- # compare type of member
- if t1[m].info.type != t2[m].info.type:
- return 'different type for member %s' % m
-
- # for files, compare hash of file content
- if t1[m].info.isfile():
- if t1[m].sha512 != t2[m].sha512:
- return 'different hash for member %s' % m
- # for links, compare target
- elif t1[m].info.islnk() or t1[m].info.issym():
- if t1[m].info.linkname != t2[m].info.linkname:
- return 'different linkname for member %s' % m
-
- # permitted differences: mtime, mode, owner uid/gid
-
- return None
-
-#
-#
-#
-
-
-def find_duplicates(args):
- basedir = os.path.join(args.rel_area, common_constants.ARCHES[0], 'release')
-
- for (dirpath, _subdirs, files) in os.walk(basedir):
- relpath = os.path.relpath(dirpath, basedir)
- otherdir = os.path.join(args.rel_area, common_constants.ARCHES[1], 'release', relpath)
-
- for f in files:
- # not an archive
- if not re.match(r'^.*\.tar' + common_constants.PACKAGE_COMPRESSIONS_RE + r'$', f):
- continue
-
- f1 = os.path.join(dirpath, f)
- f2 = os.path.join(otherdir, f)
-
- if os.path.exists(f2):
- difference = compare_archives(f1, f2)
- if difference is None:
- print(os.path.join('release', relpath, f))
- elif args.verbose:
- print('%s: %s' % (os.path.join('release', relpath, f), difference))
-
-#
-#
-#
-
-
-def main():
- relarea_default = common_constants.FTP
-
- parser = argparse.ArgumentParser(description='Source package deduplicator')
- parser.add_argument('--releasearea', action='store', metavar='DIR', help="release directory (default: " + relarea_default + ")", default=relarea_default, dest='rel_area')
- parser.add_argument('-v', '--verbose', action='count', dest='verbose', help='verbose output')
- (args) = parser.parse_args()
-
- return find_duplicates(args)
-
-
-#
-#
-#
-
-if __name__ == "__main__":
- sys.exit(main())
diff --git a/calm/hint.py b/calm/hint.py
index b3e9a70..ddfcf74 100755
--- a/calm/hint.py
+++ b/calm/hint.py
@@ -90,7 +90,7 @@ hintkeys[override] = {
'keep-count': 'val',
'keep-count-test': 'val',
'keep-days': 'val',
- 'keep-superseded-versions': 'noval',
+ 'keep-superseded-test': 'noval',
'disable-check': 'val',
'replace-versions': 'val',
}
diff --git a/calm/mksetupini.py b/calm/mksetupini.py
index d788559..170311b 100755
--- a/calm/mksetupini.py
+++ b/calm/mksetupini.py
@@ -131,7 +131,7 @@ def main():
disable_check_choices = choiceList(['missing-curr', 'missing-depended-package', 'missing-obsoleted-package', 'missing-required-package', 'curr-most-recent'])
parser = argparse.ArgumentParser(description='Make setup.ini')
- parser.add_argument('--arch', action='store', required=True, choices=common_constants.ARCHES)
+ parser.add_argument('--arch', action='store', required=True, choices=common_constants.ARCHES + common_constants.ARCHIVED_ARCHES)
parser.add_argument('--disable-check', action=flatten_append, help='checks to disable', type=disable_check_choices, default=[], metavar=disable_check_choices.help())
parser.add_argument('--inifile', '-u', action='store', help='output filename', required=True)
parser.add_argument('--okmissing', action='append', help='superseded by --disable-check', choices=['curr', 'depended-package', 'obsoleted-package', 'required-package'])
More information about the Cygwin-apps-cvs
mailing list