]> CyberLeo.Net >> Repos - FreeBSD/releng/8.1.git/blob - release/scripts/package-split.py
Merge r209770 from stable/8:
[FreeBSD/releng/8.1.git] / release / scripts / package-split.py
1 #!/usr/local/bin/python
2 #
3 # This script generates a master INDEX file for the CD images built by the
4 # FreeBSD release engineers.  Each disc is given a list of desired packages.
5 # Dependencies of these packages are placed on either the same disc or an
6 # earlier disc.  The resulting master INDEX file is then written out.
7 #
8 # Usage: package-split.py <INDEX> <master INDEX>
9 #
10 # $FreeBSD$
11
12 import os
13 import sys
14
15 try:
16     arch = os.environ["PKG_ARCH"]
17 except:
18     arch = os.uname()[4]
19 print "Using arch %s" % (arch)
20
21 if 'PKG_VERBOSE' in os.environ:
22     verbose = 1
23 else:
24     verbose = 0
25
26 if 'PKG_DVD' in os.environ:
27     doing_dvd = 1
28 else:
29     doing_dvd = 0
30
31 # List of packages for disc1.
32 def disc1_packages():
33     pkgs = ['misc/freebsd-doc-bn', 
34             'misc/freebsd-doc-da',
35             'misc/freebsd-doc-de',
36             'misc/freebsd-doc-el',
37             'misc/freebsd-doc-en',
38             'misc/freebsd-doc-es',
39             'misc/freebsd-doc-fr',
40             'misc/freebsd-doc-hu',
41             'misc/freebsd-doc-it',
42             'misc/freebsd-doc-ja',
43             'misc/freebsd-doc-mn',
44             'misc/freebsd-doc-nl',
45             'misc/freebsd-doc-pl',
46             'misc/freebsd-doc-pt',
47             'misc/freebsd-doc-ru',
48             'misc/freebsd-doc-sr',
49             'misc/freebsd-doc-tr',
50             'misc/freebsd-doc-zh_cn',
51             'misc/freebsd-doc-zh_tw']
52
53     if doing_dvd:
54         pkgs.extend(['archivers/unzip',
55             'emulators/linux_base-f10',
56             'lang/perl5.10',
57             'net/mpd5',
58             'net/rsync',
59             'ports-mgmt/p5-FreeBSD-Portindex',
60             'ports-mgmt/portaudit',
61             'ports-mgmt/portmaster',
62             'ports-mgmt/portupgrade',
63             'shells/bash',
64             'shells/zsh',
65             'security/sudo',
66             'sysutils/screen',
67             'x11/gnome2',
68             'x11/kde4',
69             'x11/xorg'])
70     return pkgs
71
72 # The list of desired packages
73 def desired_packages():
74     disc1 = disc1_packages()
75     return [disc1]
76
77 # Suck the entire INDEX file into a two different dictionaries.  The first
78 # dictionary maps port names (origins) to package names.  The second
79 # dictionary maps a package name to a list of its dependent packages.
80 PACKAGE_COL=0
81 ORIGIN_COL=1
82 DEPENDS_COL=8
83
84 def load_index(index):
85     deps = {}
86     pkgs = {}
87     line_num = 1
88     for line in index:
89         fields = line.split('|')
90         name = fields[PACKAGE_COL]
91         if name in deps:
92             sys.stderr.write('%d: Duplicate package %s\n' % (line_num, name))
93             sys.exit(1)
94         origin = fields[ORIGIN_COL].replace('/usr/ports/', '', 1)
95         if origin in pkgs:
96             sys.stderr.write('%d: Duplicate port %s\n' % (line_num, origin))
97             sys.exit(1)
98         deps[name] = fields[DEPENDS_COL].split()
99         pkgs[origin] = name
100         line_num = line_num + 1
101     return (deps, pkgs)
102
103 # Layout the packages on the various CD images.  Here's how it works.  We walk
104 # each disc in the list of discs.  Within each disc we walk the list of ports.
105 # For each port, we add the package name to a dictionary with the value being
106 # the current disc number.  We also add all of the dependent packages.  If
107 # a package is already in the dictionary when we go to add it, we just leave
108 # the dictionary as it is.  This means that each package ends up on the first
109 # disc that either lists it or contains it as a dependency.
110 def layout_discs(discs, pkgs, deps):
111     disc_num = 1
112     layout = {}
113     for disc in discs:
114         for port in disc:
115             if port not in pkgs:
116                 sys.stderr.write('Disc %d: Unable to find package for %s\n' %
117                                  (disc_num, port))
118                 continue
119             pkg = pkgs[port]
120             pkg_list = [pkg] + deps[pkg]
121             for pkg in pkg_list:
122                 if pkg not in layout:
123                     if verbose:
124                         print "--> Adding %s to Disc %d" % (pkg, disc_num)
125                     layout[pkg] = disc_num
126         disc_num = disc_num + 1
127     return layout
128
129 # Generate a master INDEX file based on the generated layout.  The way this
130 # works is that for each INDEX line, we check to see if the package is in the
131 # layout.  If it is, we put that INDEX line into the master INDEX and append
132 # a new field with the disc number to the line.
133 def generate_index(index, layout, master_index):
134     for line in index:
135         pkg = line.split('|')[PACKAGE_COL]
136         if pkg in layout:
137             new_line = '%s|%d\n' % (line.splitlines()[0], layout[pkg])
138             master_index.write(new_line)
139
140 # Verify the command line arguments
141 if len(sys.argv) != 3:
142     sys.stderr.write('Invalid number of arguments\n')
143     sys.stderr.write('Usage: package-split.py <source INDEX> <master INDEX>\n')
144     sys.exit(1)
145
146 print "Loading %s..." % (sys.argv[1])
147 index = file(sys.argv[1])
148 (deps, pkgs) = load_index(index)
149 discs = desired_packages()
150 layout = layout_discs(discs, pkgs, deps)
151 index.seek(0)
152 print "Generating %s..." % (sys.argv[2])
153 master_index = file(sys.argv[2], 'w')
154 generate_index(index, layout, master_index)
155 index.close()
156 master_index.close()