1 # vi: encoding=utf-8 ts=8 sts=4 sw=4 et
3 import os, config, string, urllib, re
4 from common import fileexists, noarchcachedir
5 from baseftptree import BasePkg, BaseFtpTree
13 print "%d error(s) encountered... aborting" % errnum
32 os.rename(src, dst+'/'+src.split('/')[-1])
33 #print "mv: %s %s" % (src, dst+'/'+src.split('/')[-1])
36 def __init__(self, nvr, tree):
37 BasePkg.__init__(self, nvr, tree)
38 self.name=string.join(nvr.split('-')[:-2], '-')
39 self.version=nvr.split('-')[-2]
40 self.release=nvr.split('-')[-1]
41 self.marked4removal=False
42 self.marked4moving=False
46 def mark4moving(self):
47 if not self.marked4moving:
48 self.tree.marked4moving.append(self)
49 self.marked4moving=True
51 def mark4removal(self):
52 if not self.marked4removal:
53 self.tree.marked4removal.append(self)
54 self.marked4removal=True
57 self.errors.append(msg)
59 perror('%s %s' % (self.nvr, msg))
61 def warning(self, msg):
62 self.warnings.append(msg)
64 pwarning('%s %s' % (self.nvr, msg))
66 def load(self, content=None):
67 BasePkg.load(self, content)
68 if self.info.has_key('move'):
72 f=open(self.tree.basedir+'/SRPMS/.metadata/'+self.nvr+'.src.rpm.info', 'w')
73 for bid in self.build.keys():
74 f.write("info:build:%s:requester:%s\ninfo:build:%s:requester_email:%s\n" % (bid, self.build[bid].requester, bid, self.build[bid].requester_email))
75 for key in self.info.keys():
76 f.write("info:%s:%s\n" % (key, string.join(self.info[key], ':')))
77 for arch in self.files.keys():
78 for rpm in self.files[arch]:
79 f.write("file:%s:%s\n" % (arch, rpm))
82 for arch in self.files.keys():
83 for rpm in self.files[arch]:
84 rm(self.tree.basedir+'/'+arch+'/RPMS/'+rpm)
86 if fileexists(noarchcachedir+rpm+'.filelist'):
87 rm(noarchcachedir+rpm+'.filelist')
88 if fileexists(noarchcachedir+rpm+'.reqlist'):
89 rm(noarchcachedir+rpm+'.reqlist')
90 rm(self.tree.basedir+'/SRPMS/.metadata/'+self.nvr+'.src.rpm.info')
92 def move(self, dsttree):
93 if dsttree.has_key(self.nvr):
95 for arch in self.files.keys():
96 if arch in dsttree[self.nvr].files.keys():
97 pinfo("Arch %s for %s is already present in dest tree; removing from srctree" % (arch, self.nvr))
98 for rpm in self.files[arch]:
99 rm(self.tree.basedir+'/'+arch+'/RPMS/'+rpm)
102 dsttree[self.nvr].files[arch]=self.files[arch]
103 for rpm in self.files[arch]:
104 mv(self.tree.basedir+'/'+arch+'/RPMS/'+rpm, dsttree.basedir+'/'+arch+'/RPMS/')
106 for bid in self.build.keys():
107 dsttree[self.nvr].build[bid]=self.build[bid]
108 dsttree[self.nvr].writeinfo()
109 rm(self.tree.basedir+'/SRPMS/.metadata/'+self.nvr+'.src.rpm.info')
111 for arch in self.files.keys():
112 for rpm in self.files[arch]:
113 mv(self.tree.basedir+'/'+arch+'/RPMS/'+rpm, dsttree.basedir+'/'+arch+'/RPMS/')
114 mv(self.tree.basedir+'/SRPMS/.metadata/'+self.nvr+'.src.rpm.info', dsttree.basedir+'/SRPMS/.metadata/')
117 class FtpTree(BaseFtpTree):
118 def __init__(self, tree, loadall=False):
119 BaseFtpTree.__init__(self, tree)
121 self.marked4removal=[]
122 self.marked4moving=[]
124 self.__loadpkgnames()
126 for pkgname in self.pkgnames:
127 self.loadedpkgs[pkgname]=Pkg(pkgname, self)
129 self.do_checkbuild=True
131 def __getitem__(self, key):
132 if self.loadedpkgs.has_key(key):
133 return self.loadedpkgs[key]
134 elif key in self.pkgnames:
136 self.loadedpkgs[key]=pkg
141 def has_key(self, key):
142 if key in self.pkgnames:
150 def checktree(self, dsttree):
151 self.__checkbuild(self.loadedpkgs.values())
152 self.__checkarchs(dsttree, self.loadedpkgs.values())
154 def testmove(self, dsttree):
155 self.__checkbuild(self.marked4moving)
156 self.__checkarchs(dsttree, self.marked4moving)
158 def movepkgs(self, dsttree):
159 if self.do_checkbuild:
160 self.__checkbuild(self.marked4moving)
162 self.__checkarchs(dsttree, self.marked4moving)
164 self.__rmolderfromsrc()
165 self.__rmotherfromdst(dsttree)
167 for pkg in self.marked4moving:
170 def removepkgs(self):
171 if self.do_checkbuild:
172 self.__checkbuild(self.marked4removal)
174 for pkg in self.marked4removal:
177 def mark4removal(self, wannabepkgs):
178 self.__mark4something(wannabepkgs, Pkg.mark4removal)
180 def mark4moving(self, wannabepkgs):
181 self.__mark4something(wannabepkgs, Pkg.mark4moving)
184 # Internal functions below
186 def __loadpkgnames(self):
187 def checkfiletype(name):
188 if name[-13:]=='.src.rpm.info':
192 list=filter(checkfiletype, os.listdir(self.basedir+'/SRPMS/.metadata'))
193 self.pkgnames=map((lambda x: x[:-13]), list)
195 def __mark4something(self, wannabepkgs, markfunction):
196 def chopoffextension(pkg):
197 found=pkg.find('.src.rpm')
202 for wannabepkg in wannabepkgs:
203 pkgname=chopoffextension(wannabepkg)
204 if pkgname in self.pkgnames:
205 if not pkgname in self.loadedpkgs.keys():
206 self.loadedpkgs[pkgname]=Pkg(pkgname, self)
207 markfunction(self.loadedpkgs[pkgname])
209 perror('%s not found in source tree' % pkgname)
212 def __checkbuild(self, marked):
213 f=urllib.urlopen(config.builderqueue)
216 reid=re.compile(r'^.*id=(.*) pri.*$')
217 regb=re.compile(r'^group:.*$|builders:.*$', re.M)
218 for i in re.findall(regb, f.read()):
220 id=reid.sub(r'\1', i)
223 requests[id]=requests[id]+i
226 for bid in pkg.build.keys():
227 if requests.has_key(bid) and not requests[bid].find('?') == -1:
228 pkg.error("(buildid %s) building not finished" % bid)
230 def __checkarchs(self, dsttree, marked):
232 if len(pkg.files.keys()) <= 1:
233 pkg.error('has only src.rpm built')
235 otherpkgnames=self.__find_other_pkgs(pkg, dsttree)
236 if otherpkgnames: # check if we're not removing some archs
239 for somepkg in otherpkgnames:
240 curarchs.extend(Pkg(somepkg, dsttree).files.keys())
241 for arch in curarchs:
242 if arch not in pkg.files.keys():
243 missingarchs.append(arch)
245 pkg.error('moving would remove archs: %s' % missingarchs)
246 else: # warn if a package isn't built for all archs
247 if (config.separate_noarch and 'noarch' in pkg.files.keys() and
248 len(pkg.files.keys())==2):
250 elif len(pkg.files.keys()) != len(config.ftp_archs)+1:
252 for arch in config.ftp_archs:
253 if arch not in pkg.files.keys():
254 missingarchs.append(arch)
255 pkg.warning('not built for archs: %s' % missingarchs)
257 def __rmolderfromsrc(self):
258 for pkg in self.marked4moving:
259 olderpkgnames=self.__find_older_pkgs(pkg)
260 for i in olderpkgnames:
261 Pkg(i, self).remove()
263 def __rmotherfromdst(self, dsttree):
264 for pkg in self.marked4moving:
265 pkgnames=self.__find_other_pkgs(pkg, dsttree)
267 Pkg(i, dsttree).remove()
269 # Used more than once filter functions
271 def __find_other_pkgs(self, pkg, tree):
272 escapedpkgname=pkg.name.replace('.', '\.').replace('+', '\+')
273 ziewre=re.compile(escapedpkgname+'-[^-]*-[^-]*$')
274 def filter_other_pkgs(x):
275 if ziewre.match(x) and not x == pkg.nvr:
279 return filter(filter_other_pkgs, tree.pkgnames)
281 def __find_older_pkgs(self, pkg):
282 def filter_older_pkgs(x):
283 checking=x.split('-')
284 if checking[-2]<pkg.version:
286 elif checking[-2]==pkg.version:
287 if checking[-1]<pkg.release:
293 return filter(filter_older_pkgs, self.__find_other_pkgs(pkg, self))