mirror of
				https://github.com/lubuntu-team/ppa-britney.git
				synced 2025-10-24 21:24:03 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			301 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			Python
		
	
	
		
			Executable File
		
	
	
	
	
			
		
		
	
	
			301 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			Python
		
	
	
		
			Executable File
		
	
	
	
	
| #! /usr/bin/python
 | |
| 
 | |
| # Copyright (C) 2009, 2010, 2011, 2012  Canonical Ltd.
 | |
| 
 | |
| # This program is free software: you can redistribute it and/or modify
 | |
| # it under the terms of the GNU General Public License as published by
 | |
| # the Free Software Foundation; version 3 of the License.
 | |
| #
 | |
| # This program is distributed in the hope that it will be useful,
 | |
| # but WITHOUT ANY WARRANTY; without even the implied warranty of
 | |
| # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 | |
| # GNU General Public License for more details.
 | |
| #
 | |
| # You should have received a copy of the GNU General Public License
 | |
| # along with this program.  If not, see <http://www.gnu.org/licenses/>.
 | |
| 
 | |
| from __future__ import print_function
 | |
| 
 | |
| from collections import defaultdict
 | |
| import gzip
 | |
| import optparse
 | |
| import os
 | |
| import re
 | |
| import sys
 | |
| import tempfile
 | |
| 
 | |
| import apt_pkg
 | |
| 
 | |
| 
 | |
| default_base = '/home/ubuntu-archive/mirror/ubuntu'
 | |
| default_suite = 'disco'
 | |
| components = ('main', 'restricted', 'universe', 'multiverse')
 | |
| 
 | |
| # Cut-down RE from deb822.PkgRelation.
 | |
| re_dep = re.compile(r'^\s*([a-zA-Z0-9.+\-]{2,})')
 | |
| 
 | |
| re_kernel_image_di = re.compile(r'^kernel-image-(.+)-di')
 | |
| 
 | |
| 
 | |
| # Cheaper version of deb822.PkgRelation.parse_relations.
 | |
| def parse_relation_packages(raw):
 | |
|     for or_dep in raw.split(','):
 | |
|         for dep in or_dep.split('|'):
 | |
|             match = re_dep.match(dep.strip())
 | |
|             if match:
 | |
|                 yield match.group(1)
 | |
| 
 | |
| 
 | |
| def primary_arches(suite):
 | |
|     return ('amd64', 'i386')
 | |
| 
 | |
| 
 | |
| def ports_arches(suite):
 | |
|     if suite == 'lucid':
 | |
|         return ('armel', 'ia64', 'powerpc', 'sparc')
 | |
|     elif suite == 'precise':
 | |
|         return ('armel', 'armhf', 'powerpc')
 | |
|     elif suite in ('14.09', '14.09-factory'):
 | |
|         return ('armhf',)
 | |
|     elif suite in ('trusty', 'vivid', 'wily'):
 | |
|         return ('arm64', 'armhf', 'powerpc', 'ppc64el')
 | |
|     elif suite in ('xenial', 'yakkety'):
 | |
|         return ('arm64', 'armhf', 'powerpc', 'ppc64el', 's390x')
 | |
|     else:
 | |
|         return ('arm64', 'armhf', 'ppc64el', 's390x')
 | |
| 
 | |
| 
 | |
| def read_tag_file(path):
 | |
|     tmp = tempfile.NamedTemporaryFile(prefix='checkrdepends.', delete=False)
 | |
|     try:
 | |
|         compressed = gzip.open(path)
 | |
|         try:
 | |
|             tmp.write(compressed.read())
 | |
|         finally:
 | |
|             compressed.close()
 | |
|         tmp.close()
 | |
|         with open(tmp.name) as uncompressed:
 | |
|             tag_file = apt_pkg.TagFile(uncompressed)
 | |
|             prev_name = None
 | |
|             prev_stanza = None
 | |
|             for stanza in tag_file:
 | |
|                 try:
 | |
|                     name = stanza['package']
 | |
|                 except KeyError:
 | |
|                     continue
 | |
|                 if name != prev_name and prev_stanza is not None:
 | |
|                     yield prev_stanza
 | |
|                 prev_name = name
 | |
|                 prev_stanza = stanza
 | |
|             if prev_stanza is not None:
 | |
|                 yield prev_stanza
 | |
|     finally:
 | |
|         os.unlink(tmp.name)
 | |
| 
 | |
| 
 | |
| def read_sources(path):
 | |
|     ret = {
 | |
|         'binary': {},
 | |
|         'source': defaultdict(set),
 | |
|         'build_deps': defaultdict(set),
 | |
|     }
 | |
|     binary = ret['binary']
 | |
|     source = ret['source']
 | |
|     build_deps = ret['build_deps']
 | |
|     for stanza in read_tag_file(path):
 | |
|         if 'binary' not in stanza:
 | |
|             continue
 | |
|         name = stanza['package']
 | |
|         binpkgs = [b.rstrip(',') for b in stanza['binary'].split()]
 | |
|         binary[name] = binpkgs
 | |
|         for binpkg in binpkgs:
 | |
|             source[binpkg].add(stanza['package'])
 | |
|         for field in ('build-depends', 'build-depends-indep'):
 | |
|             if field not in stanza:
 | |
|                 continue
 | |
|             for depname in parse_relation_packages(stanza[field]):
 | |
|                 build_deps[depname].add(name)
 | |
|     return ret
 | |
| 
 | |
| 
 | |
| def read_packages(debs, path, sources, ignores=[], missing_ok=False):
 | |
|     ret = {'deps': defaultdict(dict)}
 | |
|     deps = ret['deps']
 | |
|     try:
 | |
|         for stanza in read_tag_file(path):
 | |
|             name = stanza['package']
 | |
|             for field in ('pre-depends', 'depends', 'recommends'):
 | |
|                 if field not in stanza:
 | |
|                     continue
 | |
|                 for depname in parse_relation_packages(stanza[field]):
 | |
|                     if depname not in debs:
 | |
|                         continue
 | |
|                     # skip dependencies that are built from the same source,
 | |
|                     # when we're doing a sourceful removal.
 | |
|                     if name in ignores:
 | |
|                         continue
 | |
|                     deps[depname][name] = (field, stanza['architecture'])
 | |
|     except IOError:
 | |
|         if not missing_ok:
 | |
|             raise
 | |
|     return ret
 | |
| 
 | |
| 
 | |
| def read_di(debs, path):
 | |
|     ret = set()
 | |
|     try:
 | |
|         with open(path) as manifest:
 | |
|             for line in manifest:
 | |
|                 udeb = line.split()[0]
 | |
|                 ret.add(udeb)
 | |
|                 match = re_kernel_image_di.match(udeb)
 | |
|                 if match:
 | |
|                     re_modules = re.compile(r'-modules-%s-di' % match.group(1))
 | |
|                     for pkg in debs:
 | |
|                         if re_modules.search(pkg):
 | |
|                             ret.add(pkg)
 | |
|     except IOError:
 | |
|         pass
 | |
|     return ret
 | |
| 
 | |
| 
 | |
| def pockets(opts):
 | |
|     if '-' in opts.suite:
 | |
|         return ('',)
 | |
|     else:
 | |
|         return ('', '-updates', '-security', '-backports')
 | |
| 
 | |
| 
 | |
| def render_dep(name, field, arch):
 | |
|     ret = name
 | |
|     if field == "recommends":
 | |
|         ret += " (r)"
 | |
|     if arch == "all":
 | |
|         ret += " [all]"
 | |
|     return ret
 | |
| 
 | |
| 
 | |
| def search(opts, pkgs):
 | |
|     for pocket in pockets(opts):
 | |
|         pocket_base = '%s/dists/%s%s' % (opts.archive_base, opts.suite, pocket)
 | |
|         if opts.arches:
 | |
|             arches = opts.arches
 | |
|         else:
 | |
|             arches = list(primary_arches(opts.suite))
 | |
|             if opts.ports:
 | |
|                 arches.extend(ports_arches(opts.suite))
 | |
| 
 | |
|         packages = defaultdict(dict)
 | |
|         sources = {}
 | |
|         for comp in components:
 | |
|             comp_base = '%s/%s' % (pocket_base, comp)
 | |
|             sources[comp] = read_sources('%s/source/Sources.gz' % comp_base)
 | |
| 
 | |
|         if opts.binary:
 | |
|             debs = pkgs
 | |
|             ignores = []
 | |
|         else:
 | |
|             debs = set()
 | |
|             for src in pkgs:
 | |
|                 for comp in components:
 | |
|                     if src in sources[comp]['binary']:
 | |
|                         debs.update(set(sources[comp]['binary'][src]))
 | |
|             ignores = debs = sorted(debs)
 | |
| 
 | |
|         # Now we have the source<->binary mapping, we can read Packages
 | |
|         # files but only bother to remember the dependencies we need.
 | |
|         for comp in components:
 | |
|             comp_base = '%s/%s' % (pocket_base, comp)
 | |
|             di_comp = '%s/debian-installer' % comp
 | |
|             di_comp_base = '%s/%s' % (pocket_base, di_comp)
 | |
| 
 | |
|             build_deps = sources[comp]['build_deps']
 | |
|             for deb in debs:
 | |
|                 if opts.directory is not None:
 | |
|                     out = open(os.path.join(opts.directory, deb), 'a')
 | |
|                 else:
 | |
|                     out = sys.stdout
 | |
| 
 | |
|                 # build dependencies
 | |
|                 if deb in build_deps:
 | |
|                     print("-- %s%s/%s build deps on %s:" %
 | |
|                           (opts.suite, pocket, comp, deb), file=out)
 | |
|                     for pkg in sorted(build_deps[deb]):
 | |
|                         print(pkg, file=out)
 | |
| 
 | |
|                 # binary dependencies
 | |
|                 for arch in arches:
 | |
|                     if arch not in packages[comp]:
 | |
|                         packages[comp][arch] = \
 | |
|                             read_packages(debs,
 | |
|                                           '%s/binary-%s/Packages.gz' %
 | |
|                                           (comp_base, arch),
 | |
|                                           sources[comp], ignores)
 | |
|                     if arch not in packages[di_comp]:
 | |
|                         packages[di_comp][arch] = \
 | |
|                             read_packages(debs,
 | |
|                                           '%s/binary-%s/Packages.gz' %
 | |
|                                           (di_comp_base, arch),
 | |
|                                           sources[comp], ignores,
 | |
|                                           missing_ok=True)
 | |
|                         if comp == 'main':
 | |
|                             di_images = \
 | |
|                                 read_di(debs,
 | |
|                                         '%s/installer-%s/current/images/'
 | |
|                                         'udeb.list' % (comp_base, arch))
 | |
|                             di_deps = packages[di_comp][arch]['deps']
 | |
|                             for udeb in di_images:
 | |
|                                 di_deps[udeb]['debian-installer-images'] = (
 | |
|                                     'depends', arch)
 | |
| 
 | |
|                     deps = packages[comp][arch]['deps']
 | |
|                     di_deps = packages[di_comp][arch]['deps']
 | |
|                     if deb in deps:
 | |
|                         print("-- %s%s/%s %s deps on %s:" %
 | |
|                               (opts.suite, pocket, comp, arch, deb), file=out)
 | |
|                         for pkg, (field, pkgarch) in sorted(deps[deb].items()):
 | |
|                             print(render_dep(pkg, field, pkgarch), file=out)
 | |
|                     if deb in di_deps:
 | |
|                         print("-- %s%s/%s %s deps on %s:" %
 | |
|                               (opts.suite, pocket, di_comp, arch, deb),
 | |
|                               file=out)
 | |
|                         for pkg, (field, pkgarch) in sorted(
 | |
|                                 di_deps[deb].items()):
 | |
|                             print(render_dep(pkg, field, pkgarch), file=out)
 | |
| 
 | |
|                 if opts.directory is not None:
 | |
|                     out.close()
 | |
| 
 | |
| 
 | |
| def main():
 | |
|     parser = optparse.OptionParser(usage='%prog [options] pkg [...]')
 | |
|     parser.add_option('-B', '--archive-base', dest='archive_base',
 | |
|                       help=('archive base directory (default: %s)' %
 | |
|                             default_base),
 | |
|                       default=default_base)
 | |
|     parser.add_option('-s', '--suite', dest='suite',
 | |
|                       help='suite to check (default: %s)' % default_suite,
 | |
|                       default=default_suite)
 | |
|     parser.add_option('-a', '--arch', dest='arches', action='append',
 | |
|                       help='check only this architecture '
 | |
|                            '(may be given multiple times)')
 | |
|     parser.add_option('-b', '--binary', dest='binary', action='store_true',
 | |
|                       help='treat arguments as binary packages, not source')
 | |
|     parser.add_option('--no-ports', dest='ports',
 | |
|                       default=True, action='store_false',
 | |
|                       help='skip ports architectures')
 | |
|     parser.add_option('-d', '--directory', dest='directory', metavar='DIR',
 | |
|                       help='output to directory DIR (one file per package) '
 | |
|                            'instead of standard output')
 | |
|     opts, args = parser.parse_args()
 | |
| 
 | |
|     if 'CHECKRDEPENDS_PROFILE' in os.environ:
 | |
|         import profile
 | |
|         profile.run('search(opts, args)')
 | |
|     else:
 | |
|         search(opts, args)
 | |
| 
 | |
| 
 | |
| if __name__ == '__main__':
 | |
|     main()
 |