# -*- coding: utf-8 -*- # Copyright (C) 2006, 2011-2015 Anthony Towns # Andreas Barth # Fabio Tranchitella # 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; either version 2 of the License, or # (at your option) any later version. # 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. import re EXCUSES_LABELS = { "PASS": 'Pass', "FAIL": 'Failed', "ALWAYSFAIL": 'Always failed', "REGRESSION": 'Regression', "RUNNING": 'Test in progress', "RUNNING-ALWAYSFAIL": 'Test in progress (always failed)', } class Excuse(object): """Excuse class This class represents an update excuse, which is a detailed explanation of why a package can or cannot be updated in the testing distribution from a newer package in another distribution (like for example unstable). The main purpose of the excuses is to be written in an HTML file which will be published over HTTP. The maintainers will be able to parse it manually or automatically to find the explanation of why their packages have been updated or not. """ ## @var reemail # Regular expression for removing the email address reemail = re.compile(r" *<.*?>") def __init__(self, name): """Class constructor This method initializes the excuse with the specified name and the default values. """ self.name = name self.ver = ("-", "-") self.maint = None self.urgency = None self.daysold = None self.mindays = None self.section = None self._is_valid = False self._dontinvalidate = False self.forced = False self.run_autopkgtest = False self.distribution = "ubuntu" self.invalid_deps = [] self.deps = {} self.sane_deps = [] self.break_deps = [] self.bugs = [] self.newbugs = set() self.oldbugs = set() self.reason = {} self.htmlline = [] # type (e. g. "autopkgtest") -> package (e. g. "foo 2-1") -> arch -> # ['PASS'|'ALWAYSFAIL'|'REGRESSION'|'RUNNING'|'RUNNING-ALWAYSFAIL', log_url, history_url] self.tests = {} def sortkey(self): if self.daysold == None: return (-1, self.name) return (self.daysold, self.name) @property def is_valid(self): return self._is_valid @is_valid.setter def is_valid(self, value): self._is_valid = value @property def dontinvalidate(self): return self._dontinvalidate @dontinvalidate.setter def dontinvalidate(self, value): self._dontinvalidate = value def set_vers(self, tver, uver): """Set the testing and unstable versions""" if tver: self.ver = (tver, self.ver[1]) if uver: self.ver = (self.ver[0], uver) def set_maint(self, maint): """Set the package maintainer's name""" self.maint = self.reemail.sub("", maint) def set_section(self, section): """Set the section of the package""" self.section = section def set_urgency(self, date): """Set the urgency of upload of the package""" self.urgency = date def set_distribution(self, distribution): """Set the distribution name""" self.distribution = distribution def add_dep(self, name, arch): """Add a dependency""" if name not in self.deps: self.deps[name]=[] self.deps[name].append(arch) def add_sane_dep(self, name): """Add a sane dependency""" if name not in self.sane_deps: self.sane_deps.append(name) def add_break_dep(self, name, arch): """Add a break dependency""" if (name, arch) not in self.break_deps: self.break_deps.append( (name, arch) ) def invalidate_dep(self, name): """Invalidate dependency""" if name not in self.invalid_deps: self.invalid_deps.append(name) def setdaysold(self, daysold, mindays): """Set the number of days from the upload and the minimum number of days for the update""" self.daysold = daysold self.mindays = mindays def force(self): """Add force hint""" self.forced = True def addhtml(self, note): """Add a note in HTML""" self.htmlline.append(note) def html(self): """Render the excuse in HTML""" lp_pkg = "https://launchpad.net/%s/+source/%s" % (self.distribution, self.name.split("/")[0]) if self.ver[0] == "-": lp_old = self.ver[0] else: lp_old = "%s" % ( lp_pkg, self.ver[0], self.ver[0]) if self.ver[1] == "-": lp_new = self.ver[1] else: lp_new = "%s" % ( lp_pkg, self.ver[1], self.ver[1]) res = ( "%s (%s to %s)\n
    \n" % (self.name, self.name, lp_pkg, self.name, lp_old, lp_new)) if self.maint: res = res + "
  • Maintainer: %s\n" % (self.maint) if self.section and self.section.find("/") > -1: res = res + "
  • Section: %s\n" % (self.section) if self.daysold != None: if self.mindays == 0: res = res + ("
  • %d days old\n" % self.daysold) elif self.daysold < self.mindays: res = res + ("
  • Too young, only %d of %d days old\n" % (self.daysold, self.mindays)) else: res = res + ("
  • %d days old (needed %d days)\n" % (self.daysold, self.mindays)) for testtype in sorted(self.tests): for pkg in sorted(self.tests[testtype]): archmsg = [] for arch in sorted(self.tests[testtype][pkg]): status, log_url, history_url, artifact_url, retry_url = self.tests[testtype][pkg][arch] label = EXCUSES_LABELS[status] if history_url: message = '%s' % (history_url, arch) else: message = arch message += ': %s' % (log_url, label) if retry_url: message += ' ' % retry_url if artifact_url: message += ' [artifacts]' % artifact_url archmsg.append(message) res = res + ("
  • %s for %s: %s
  • \n" % (testtype, pkg, ', '.join(archmsg))) for x in self.htmlline: res = res + "
  • " + x + "\n" lastdep = "" for x in sorted(self.deps, key=lambda x: x.split('/')[0]): dep = x.split('/')[0] if dep == lastdep: continue lastdep = dep if x in self.invalid_deps: res = res + "
  • Depends: %s %s (not considered)\n" % (self.name, dep, dep) else: res = res + "
  • Depends: %s %s\n" % (self.name, dep, dep) for (n,a) in self.break_deps: if n not in self.deps: res += "
  • Ignoring %s depends: %s\n" % (a, n, n) if self.is_valid: res += "
  • Valid candidate\n" res = res + "
\n" return res def setbugs(self, oldbugs, newbugs): """"Set the list of old and new bugs""" self.newbugs.update(newbugs) self.oldbugs.update(oldbugs) def addreason(self, reason): """"adding reason""" self.reason[reason] = 1 def addtest(self, type_, package, arch, state, log_url, history_url=None, artifact_url=None, retry_url=None): """Add test result""" self.tests.setdefault(type_, {}).setdefault(package, {})[arch] = [state, log_url, history_url, artifact_url, retry_url] # TODO merge with html() def text(self): """Render the excuse in text""" res = [] res.append("%s (%s to %s)" % \ (self.name, self.ver[0], self.ver[1])) if self.maint: maint = self.maint res.append("Maintainer: %s" % maint) if self.section and self.section.find("/") > -1: res.append("Section: %s" % (self.section)) if self.daysold != None: if self.mindays == 0: res.append("%d days old" % self.daysold) elif self.daysold < self.mindays: res.append(("Too young, only %d of %d days old" % (self.daysold, self.mindays))) else: res.append(("%d days old (needed %d days)" % (self.daysold, self.mindays))) for x in self.htmlline: res.append("" + x + "") lastdep = "" for x in sorted(self.deps, key=lambda x: x.split('/')[0]): dep = x.split('/')[0] if dep == lastdep: continue lastdep = dep if x in self.invalid_deps: res.append("Depends: %s %s (not considered)" % (self.name, dep)) else: res.append("Depends: %s %s" % (self.name, dep)) for (n,a) in self.break_deps: if n not in self.deps: res.append("Ignoring %s depends: %s" % (a, n)) if self.is_valid: res.append("Valid candidate") return res def excusedata(self): """Render the excuse in as key-value data""" excusedata = {} excusedata["excuses"] = self.text() excusedata["source"] = self.name excusedata["old-version"] = self.ver[0] excusedata["new-version"] = self.ver[1] excusedata["age"] = self.daysold excusedata["age-needed"] = self.mindays excusedata["new-bugs"] = sorted(self.newbugs) excusedata["old-bugs"] = sorted(self.oldbugs) if self.forced: excusedata["forced-reason"] = list(self.reason.keys()) excusedata["reason"] = [] else: excusedata["reason"] = list(self.reason.keys()) excusedata["is-candidate"] = self.is_valid excusedata["tests"] = self.tests return excusedata