]> TLD Linux GIT Repositories - tld-builder.git/blob - TLD_Builder/install.py
- more python3 fixes, dropped python2 support
[tld-builder.git] / TLD_Builder / install.py
1 # vi: encoding=utf-8 ts=8 sts=4 sw=4 et
2
3 import re, os
4 import string
5 import sys
6 from io import StringIO
7 import chroot
8 import util
9 import log
10
11 hold = [
12     'dev',
13     'poldek',
14     'rpm-build',
15     'mksh',
16     'coreutils',
17     'util-linux'
18 ]
19
20 def close_killset(killset):
21     k = killset.keys()
22     if len(k) == 0:
23         return True
24     rx = re.compile(r'^.* marks (?P<name>[^\s]+?)-[^-]+-[^-]+\s.*$')
25     errors = ""
26     for p in k:
27         if p in hold:
28             del killset[p]
29             errors += "cannot remove %s because it's crucial\n" % p
30         else:
31             f = chroot.popen("poldek --noask --test --test --erase %s" % p, user = "root")
32             crucial = 0
33             e = []
34             for l in f:
35                 m = rx.search(l)
36                 if m:
37                     pkg = m.group('name')
38                     if pkg in hold:
39                         errors += "cannot remove %s because it's required " \
40                                   "by %s, that is crucial\n" % (p, pkg)
41                         crucial = 1
42                     e.append(pkg)
43             f.close()
44             if crucial:
45                 del killset[p]
46             else:
47                 for p in e:
48                     killset[p] = 2
49     return errors
50
51 def upgrade_from_batch(r, b):
52     f = chroot.popen("rpm --test -F %s 2>&1" % ' '.join(b.files), user = "root")
53     killset = {}
54     rx = re.compile(r' \(installed\) (?P<name>[^\s]+)-[^-]+-[^-]+$')
55     for l in f:
56         m = rx.search(l)
57         if m: killset[m.group('name')] = 1
58     f.close()
59     if len(killset) != 0:
60         err = close_killset(killset)
61         if err != "":
62             util.append_to(b.logfile, err)
63             log.notice("cannot upgrade rpms")
64             return False
65         k = ' '.join(killset.keys())
66         if True:
67             b.log_line("upgrade requires removal of %s" % k)
68             res = chroot.run("rpm -e %s" % k, logfile = b.logfile, user = "root")
69             if res != 0:
70                 b.log_line("package removal failed")
71                 return False
72             else:
73                 b.log_line("packages removed sucessfuly")
74         else:
75             b.log_line("upgrade would need removal of %s" % k)
76             return False
77     b.log_line("upgrading packages")
78     logbuf = StringIO()
79     res = chroot.run("rpm -Fvh %s" % ' '.join(b.files), user = "root", logfile = b.logfile)
80     if res != 0:
81         b.log_line("package upgrade failed")
82         logbuf.close()
83         return False
84     logbuf.close()
85     return True
86
87 def uninstall(conflicting, b):
88     b.log_line("uninstalling conflicting packages")
89     err = close_killset(conflicting)
90     if err != "":
91         util.append_to(b.logfile, err)
92         b.log_line("error: conflicting packages uninstallation failed")
93         return False
94     else:
95         for k in conflicting.keys():
96             b.log_line("removing %s" % k)
97             res = chroot.run("poldek --noask --erase %s" % k, logfile = b.logfile, user = "root")
98             if res != 0:
99                 b.log_line("package %s removal failed" % k)
100     return True
101
102 def is_rpmorg():
103     f = chroot.popen("rpm --version 2>&1")
104     v = re.compile(r'(RPM version|rpm \(RPM\)) (?P<major>\d)\.(?P<minor>\d+)(\.\d+)?')
105     for l in f:
106         m = v.search(l)
107         if m:
108             major = int(m.group('major'))
109             minor = int(m.group('minor'))
110             if major == 4 and minor > 5:
111                 f.close()
112                 return True
113     f.close()
114     return False
115
116 def uninstall_self_conflict(b):
117     b.log_line("checking BuildConflict-ing packages")
118     if is_rpmorg():
119         rpmcommand = "rpmbuild --nobuild -br"
120     else:
121         rpmcommand = "rpmbuild -bp --nobuild --short-circuit --define 'prep exit 0'"
122     f = chroot.popen("set -e; TMPDIR=%(tmpdir)s %(rpmcommand)s %(rpmdefs)s %(topdir)s/%(spec)s 2>&1" % {
123         'rpmcommand': rpmcommand,
124         'tmpdir': b.tmpdir(),
125         'rpmdefs' : b.rpmbuild_opts(),
126         'topdir' : b.get_topdir(),
127         'spec': b.spec,
128     })
129     # java-sun >= 1.5 conflicts with soprano-2.1.67-1.src
130     # java-sun conflicts with soprano-2.1.67-1.src
131     rx = re.compile(r"\s+(?P<name>[\w-]+)\s+.*conflicts with [^\s]+-[^-]+-[^-]+\.src($| .*)")
132     conflicting = {}
133     for l in f:
134         m = rx.search(l)
135         if m:
136             b.log_line("rpmbuild: %s" % l.rstrip())
137             conflicting[m.group('name')] = 1
138     f.close()
139     if len(conflicting) and not uninstall(conflicting, b):
140         return False
141     b.log_line("no BuildConflicts found")
142     return True
143
144 def install_br(r, b):
145     def get_missing_br(r, b):
146         # ignore internal rpm dependencies, see lib/rpmns.c for list
147         ignore_br = re.compile(r'^\s*(rpmlib|cpuinfo|getconf|uname|soname|user|group|mounted|diskspace|digest|gnupg|macro|envvar|running|sanitycheck|vcheck|signature|verify|exists|executable|readable|writable)\(.*')
148
149         tmpdir = b.tmpdir()
150         if is_rpmorg():
151             rpmcommand = "rpmbuild --nobuild -br"
152         else:
153             rpmcommand = "rpmbuild --nobuild"
154         cmd = "set -e; TMPDIR=%(tmpdir)s %(rpmcommand)s %(rpmdefs)s %(topdir)s/%(spec)s 2>&1" % {
155             'rpmcommand': rpmcommand,
156             'tmpdir': tmpdir,
157             'topdir' : b.get_topdir(),
158             'rpmdefs' : b.rpmbuild_opts(),
159             'spec': b.spec,
160         }
161         f = chroot.popen(cmd)
162         rx = re.compile(r"^\s*(?P<name>[^\s]+) .*is needed by")
163         needed = {}
164         b.log_line("checking BR")
165         for l in f:
166             b.log_line("rpm: %s" % l.rstrip())
167             m = rx.search(l)
168             if m and not ignore_br.match(l):
169                 needed[m.group('name')] = 1
170         f.close()
171         return needed
172
173     needed = get_missing_br(r, b);
174
175     if len(needed) == 0:
176         b.log_line("no BR needed")
177         return True
178
179     nbr = ""
180     for bre in needed.keys():
181         nbr = nbr + " " + re.escape(bre)
182     br = nbr.strip()
183     b.log_line("updating poldek cache...")
184     chroot.run("poldek --up --upa", user = "root", logfile = b.logfile)
185     # check conflicts in BRed packages
186     b.log_line("checking conflicting packages in BRed packages")
187     f = chroot.popen("poldek --test --test --noask --caplookup -Q -v %s --upgrade %s" % (b.ignores(), br), user = "root")
188     # phonon-devel-4.3.1-1.i686 conflicts with qt4-phonon-devel-4.5.0-6.i686
189     # jdbc-stdext >= 2.0 is required by installed java-struts-1.3.10-1.noarch
190     # jmx is needed by (installed) java-commons-modeler-2.0-1.noarch
191     rx = re.compile(r".*(conflicts with|is required by|is needed by)( installed| \(installed\)|) (?P<name>[^\s]+)-[^-]+-[^-]+($| .*)")
192     conflicting = {}
193     for l in f:
194         b.log_line("poldek: %s" % l.rstrip())
195         m = rx.search(l)
196         if m: conflicting[m.group('name')] = 1
197     f.close()
198     if len(conflicting) == 0:
199         b.log_line("no conflicts found")
200     else:
201         if not uninstall(conflicting, b):
202             return False
203
204     # recheck BuildRequires since above uninstallation could remove some required deps
205     needed = get_missing_br(r, b);
206
207     if len(needed) == 0:
208         b.log_line("no BR needed")
209         return True
210
211     nbr = ""
212     for bre in needed.keys():
213         nbr = nbr + " " + re.escape(bre)
214     br = nbr.strip()
215
216     b.log_line("installing BR: %s" % br)
217     res = chroot.run("set -x; poldek --noask --caplookup -Q -v %s --upgrade %s" % (b.ignores(), br),
218             user = "root",
219             logfile = b.logfile)
220     if res != 0:
221         b.log_line("error: BR installation failed")
222         return False
223     return True