@ -387,6 +387,8 @@ class Britney(object):
help = " do not build the non-installability status, use the cache from file " )
help = " do not build the non-installability status, use the cache from file " )
parser . add_option ( " " , " --print-uninst " , action = " store_true " , dest = " print_uninst " , default = False ,
parser . add_option ( " " , " --print-uninst " , action = " store_true " , dest = " print_uninst " , default = False ,
help = " just print a summary of uninstallable packages " )
help = " just print a summary of uninstallable packages " )
parser . add_option ( " " , " --components " , action = " store " , dest = " components " ,
help = " Sources/Packages are laid out by components listed (, sep) " )
( self . options , self . args ) = parser . parse_args ( )
( self . options , self . args ) = parser . parse_args ( )
# integrity checks
# integrity checks
@ -417,6 +419,11 @@ class Britney(object):
not getattr ( self . options , k . lower ( ) ) :
not getattr ( self . options , k . lower ( ) ) :
setattr ( self . options , k . lower ( ) , v )
setattr ( self . options , k . lower ( ) , v )
if getattr ( self . options , " components " , None ) :
self . options . components = [ s . strip ( ) for s in self . options . components . split ( " , " ) ]
else :
self . options . components = None
if not hasattr ( self . options , " heidi_delta_output " ) :
if not hasattr ( self . options , " heidi_delta_output " ) :
self . options . heidi_delta_output = self . options . heidi_output + " Delta "
self . options . heidi_delta_output = self . options . heidi_output + " Delta "
@ -543,19 +550,10 @@ class Britney(object):
# Data reading/writing methods
# Data reading/writing methods
# ----------------------------
# ----------------------------
def read_sources ( self , basedir , intern = sys . intern ) :
def _read_sources_file ( self , filename , sources = None , intern = sys . intern ) :
""" Read the list of source packages from the specified directory
if sources is None :
sources = { }
The source packages are read from the ` Sources ' file within the
directory specified as ` basedir ' parameter. Considering the
large amount of memory needed , not all the fields are loaded
in memory . The available fields are Version , Maintainer and Section .
The method returns a list where every item represents a source
package as a dictionary .
"""
sources = { }
filename = os . path . join ( basedir , " Sources " )
self . __log ( " Loading source packages from %s " % filename )
self . __log ( " Loading source packages from %s " % filename )
with open ( filename , encoding = ' utf-8 ' ) as f :
with open ( filename , encoding = ' utf-8 ' ) as f :
@ -583,38 +581,37 @@ class Britney(object):
]
]
return sources
return sources
def read_binaries ( self , basedir , distribution , arch , intern = sys . intern ) :
def read_sources ( self , basedir ) :
""" Read the list of binary packages from the specified directory
""" Read the list of source packages from the specified directory
The binary packages are read from the ` Packages_ $ { arch } ' files
within the directory specified as ` basedir ' parameter, replacing
$ { arch } with the value of the arch parameter . Considering the
large amount of memory needed , not all the fields are loaded
in memory . The available fields are Version , Source , Multi - Arch ,
Depends , Conflicts , Provides and Architecture .
After reading the packages , reverse dependencies are computed
and saved in the ` rdepends ' keys, and the `Provides ' field is
used to populate the virtual packages list .
The dependencies are parsed with the apt_pkg . parse_depends method ,
The source packages are read from the ` Sources ' file within the
and they are stored both as the format of its return value and
directory specified as ` basedir ' parameter. Considering the
text .
large amount of memory needed , not all the fields are loaded
in memory . The available fields are Version , Maintainer and Section .
The method returns a tuple . The first element is a list where
The method returns a list where every item represents a source
every item represents a binary package as a dictionary ; the second
package as a dictionary .
element is a dictionary which maps virtual packages to real
packages that provide them .
"""
"""
packages = { }
if self . options . components :
provides = defaultdict ( set )
sources = { }
sources = self . sources
for component in self . options . components :
all_binaries = self . all_binaries
filename = os . path . join ( basedir , component , " source " , " Sources " )
self . _read_sources_file ( filename , sources )
else :
filename = os . path . join ( basedir , " Sources " )
sources = self . _read_sources_file ( filename )
filename = os . path . join ( basedir , " Packages_ %s " % arch )
return sources
def _read_packages_file ( self , filename , arch , srcdist , packages = None , intern = sys . intern ) :
self . __log ( " Loading binary packages from %s " % filename )
self . __log ( " Loading binary packages from %s " % filename )
if packages is None :
packages = { }
all_binaries = self . all_binaries
with open ( filename , encoding = ' utf-8 ' ) as f :
with open ( filename , encoding = ' utf-8 ' ) as f :
Packages = apt_pkg . TagFile ( f )
Packages = apt_pkg . TagFile ( f )
get_field = Packages . section . get
get_field = Packages . section . get
@ -675,20 +672,19 @@ class Britney(object):
dpkg [ SOURCEVER ] = intern ( source [ source . find ( " ( " ) + 1 : source . find ( " ) " ) ] )
dpkg [ SOURCEVER ] = intern ( source [ source . find ( " ( " ) + 1 : source . find ( " ) " ) ] )
# if the source package is available in the distribution, then register this binary package
# if the source package is available in the distribution, then register this binary package
if dpkg [ SOURCE ] in s ou rces[ distribution] :
if dpkg [ SOURCE ] in s rcdist:
# There may be multiple versions of any arch:all packages
# There may be multiple versions of any arch:all packages
# (in unstable) if some architectures have out-of-date
# (in unstable) if some architectures have out-of-date
# binaries. We only want to include the package in the
# binaries. We only want to include the package in the
# source -> binary mapping once. It doesn't matter which
# source -> binary mapping once. It doesn't matter which
# of the versions we include as only the package name and
# of the versions we include as only the package name and
# architecture are recorded.
# architecture are recorded.
if pkg_id not in s ou rces[ distribution] [ dpkg [ SOURCE ] ] [ BINARIES ] :
if pkg_id not in s rcdist[ dpkg [ SOURCE ] ] [ BINARIES ] :
s ou rces[ distribution] [ dpkg [ SOURCE ] ] [ BINARIES ] . append ( pkg_id )
s rcdist[ dpkg [ SOURCE ] ] [ BINARIES ] . append ( pkg_id )
# if the source package doesn't exist, create a fake one
# if the source package doesn't exist, create a fake one
else :
else :
s ou rces[ distribution] [ dpkg [ SOURCE ] ] = [ dpkg [ SOURCEVER ] , ' faux ' , [ pkg_id ] , None , True ]
s rcdist[ dpkg [ SOURCE ] ] = [ dpkg [ SOURCEVER ] , ' faux ' , [ pkg_id ] , None , True ]
# register virtual packages and real packages that provide them
if dpkg [ PROVIDES ] :
if dpkg [ PROVIDES ] :
parts = apt_pkg . parse_depends ( dpkg [ PROVIDES ] , False )
parts = apt_pkg . parse_depends ( dpkg [ PROVIDES ] , False )
nprov = [ ]
nprov = [ ]
@ -706,7 +702,6 @@ class Britney(object):
provided = intern ( provided )
provided = intern ( provided )
provided_version = intern ( provided_version )
provided_version = intern ( provided_version )
part = ( provided , provided_version , intern ( op ) )
part = ( provided , provided_version , intern ( op ) )
provides [ provided ] . add ( ( pkg , provided_version ) )
nprov . append ( part )
nprov . append ( part )
dpkg [ PROVIDES ] = nprov
dpkg [ PROVIDES ] = nprov
else :
else :
@ -719,6 +714,66 @@ class Britney(object):
else :
else :
all_binaries [ pkg_id ] = dpkg
all_binaries [ pkg_id ] = dpkg
# add the resulting dictionary to the package list
packages [ pkg ] = dpkg
return packages
def read_binaries ( self , basedir , distribution , arch ) :
""" Read the list of binary packages from the specified directory
The binary packages are read from the ` Packages ' files for `arch ' .
If components are specified , the files
for each component are loaded according to the usual Debian mirror
layout .
If no components are specified , a single file named
` Packages_ $ { arch } ' is expected to be within the directory
specified as ` basedir ' parameter, replacing $ {arch} with the
value of the arch parameter .
Considering the
large amount of memory needed , not all the fields are loaded
in memory . The available fields are Version , Source , Multi - Arch ,
Depends , Conflicts , Provides and Architecture .
After reading the packages , reverse dependencies are computed
and saved in the ` rdepends ' keys, and the `Provides ' field is
used to populate the virtual packages list .
The dependencies are parsed with the apt_pkg . parse_depends method ,
and they are stored both as the format of its return value and
text .
The method returns a tuple . The first element is a list where
every item represents a binary package as a dictionary ; the second
element is a dictionary which maps virtual packages to real
packages that provide them .
"""
if self . options . components :
packages = { }
for component in self . options . components :
filename = os . path . join ( basedir ,
component , " binary- %s " % arch , " Packages " )
self . _read_packages_file ( filename , arch ,
self . sources [ distribution ] , packages )
else :
filename = os . path . join ( basedir , " Packages_ %s " % arch )
packages = self . _read_packages_file ( filename , arch ,
self . sources [ distribution ] )
# create provides
provides = defaultdict ( set )
for pkg , dpkg in packages . items ( ) :
# register virtual packages and real packages that provide
# them
for provided_pkg , provided_version , _ in dpkg [ PROVIDES ] :
provides [ provided_pkg ] . add ( ( pkg , provided_version ) )
# return a tuple with the list of real and virtual packages
# return a tuple with the list of real and virtual packages
return ( packages , provides )
return ( packages , provides )