diff --git a/britney.py b/britney.py index 9b5ce6c..8ecf7c0 100755 --- a/britney.py +++ b/britney.py @@ -65,6 +65,9 @@ Other than source and binary packages, Britney loads the following data: * Blocks, which contains user-supplied blocks read from Launchpad bugs (see LPBlockBugPolicy). + * ExcuseBugs, which contains user-supplied update-excuse read from + Launchpad bugs (see LPExcuseBugsPolicy). + For a more detailed explanation about the format of these files, please read the documentation of the related methods. The exact meaning of them will be instead explained in the chapter "Excuses Generation". @@ -219,6 +222,7 @@ from britney2.policies.policy import (AgePolicy, from britney2.policies.autopkgtest import AutopkgtestPolicy from britney2.policies.sourceppa import SourcePPAPolicy from britney2.policies.email import EmailPolicy +from britney2.policies.lpexcusebugs import LPExcuseBugsPolicy from britney2.utils import (log_and_format_old_libraries, read_nuninst, write_nuninst, write_heidi, format_and_log_uninst, newly_uninst, @@ -526,6 +530,7 @@ class Britney(object): if getattr(self.options, 'check_buildd', 'no') == 'yes': self._policy_engine.add_policy(BuiltOnBuilddPolicy(self.options, self.suite_info)) self._policy_engine.add_policy(LPBlockBugPolicy(self.options, self.suite_info)) + self._policy_engine.add_policy(LPExcuseBugsPolicy(self.options, self.suite_info)) self._policy_engine.add_policy(SourcePPAPolicy(self.options, self.suite_info)) self._policy_engine.add_policy(LinuxPolicy(self.options, self.suite_info)) add_email_policy = getattr(self.options, 'email_enable', 'no') diff --git a/britney2/policies/lpexcusebugs.py b/britney2/policies/lpexcusebugs.py new file mode 100644 index 0000000..a9f2f5a --- /dev/null +++ b/britney2/policies/lpexcusebugs.py @@ -0,0 +1,78 @@ +import os +import time + +from collections import defaultdict + +from britney2 import SuiteClass +from britney2.policies.rest import Rest +from britney2.policies.policy import BasePolicy, PolicyVerdict + + +class LPExcuseBugsPolicy(BasePolicy): + """update-excuse Launchpad bug policy to link to a bug report, does not prevent migration + + This policy will read an user-supplied "ExcuseBugs" file from the unstable + directory (provided by an external script) with rows of the following + format: + + + + The dates are expressed as the number of seconds from the Unix epoch + (1970-01-01 00:00:00 UTC). + """ + + def __init__(self, options, suite_info): + super().__init__( + "update-excuse", + options, + suite_info, + {SuiteClass.PRIMARY_SOURCE_SUITE}, + ) + self.filename = os.path.join(options.unstable, "ExcuseBugs") + + def initialise(self, britney): + super().initialise(britney) + self.excuse_bugs = defaultdict(list) # srcpkg -> [(bug, date), ...] + + self.logger.info( + "Loading user-supplied excuse bug data from %s" % self.filename + ) + try: + for line in open(self.filename): + ln = line.split() + if len(ln) != 3: + self.logger.warning( + "ExcuseBugs, ignoring malformed line %s" % line, + ) + continue + try: + self.excuse_bugs[ln[0]].append((ln[1], int(ln[2]))) + except ValueError: + self.logger.error( + 'ExcuseBugs, unable to parse "%s"' % line + ) + except FileNotFoundError: + self.logger.info( + "ExcuseBugs, file %s not found, no bugs will be recorded", + self.filename, + ) + + def apply_src_policy_impl( + self, + excuse_bugs_info, + item, + source_data_tdist, + source_data_srcdist, + excuse, + ): + source_name = item.package + excuse_bug = self.excuse_bugs[source_name] + + for bug, date in excuse_bug: + excuse_bugs_info[bug] = date + excuse.addinfo( + 'Also see bug %s last updated on %s' + % (bug, bug, time.asctime(time.gmtime(date))) + ) + + return PolicyVerdict.PASS diff --git a/tests/test_lpexcusebugs.py b/tests/test_lpexcusebugs.py new file mode 100755 index 0000000..e6bce78 --- /dev/null +++ b/tests/test_lpexcusebugs.py @@ -0,0 +1,137 @@ +#!/usr/bin/python3 +# (C) 2020 Canonical Ltd. +# +# 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. + +from collections import defaultdict + +import calendar +import fileinput +import json +import os +import pprint +import sys +import time +import unittest +import yaml +from unittest.mock import DEFAULT, patch, call + +PROJECT_DIR = os.path.dirname(os.path.dirname(os.path.abspath(__file__))) +sys.path.insert(0, PROJECT_DIR) + +from britney2.policies.policy import PolicyVerdict +from britney2.policies.email import EmailPolicy, person_chooser, address_chooser + +from tests.test_sourceppa import FakeOptions +from tests import TestBase + + +class FakeItem: + package = "chromium-browser" + + +class FakeSourceData: + version = "55.0" + + +FakeItem, + + +class FakeExcuse: + is_valid = True + daysold = 0 + reason = [] + tentative_policy_verdict = PolicyVerdict.PASS + + +class ET(TestBase): + """ Test block bug policy """ + + def setUp(self): + super().setUp() + # disable ADT, not relevant for us + for line in fileinput.input(self.britney_conf, inplace=True): + if line.startswith("ADT_ENABLE"): + print("ADT_ENABLE = no") + elif line.startswith("MINDAYS_EMERGENCY"): + print("MINDAYS_EMERGENCY = 10") + elif not line.startswith("ADT_AMQP") and not line.startswith( + "ADT_SWIFT_URL" + ): + sys.stdout.write(line) + self.excuse_bugs_file = os.path.join(self.data.dirs[True], "ExcuseBugs") + self.sourceppa_cache = {} + + self.data.add("libc6", False) + + def do_test(self, unstable_add, expect_bugs): + """Run britney with some unstable packages and verify excuses. + + unstable_add is a list of (binpkgname, field_dict, [(bugno, timestamp)]) + + expect_bugs is a list of tuples (package, bug, timestamp) that is + checked against the bugs reported as being blocked during this + do_test run. + + Return (output, excuses_dict, excuses_html, bugs). + """ + for (pkg, fields, bugs) in unstable_add: + self.data.add(pkg, True, fields, True, None) + self.sourceppa_cache.setdefault(pkg, {}) + if fields["Version"] not in self.sourceppa_cache[pkg]: + self.sourceppa_cache[pkg][fields["Version"]] = "" + print("Writing to %s" % self.excuse_bugs_file) + with open(self.excuse_bugs_file, "w") as f: + for (bug, ts) in bugs: + f.write("%s %s %s" % (pkg, bug, ts)) + + # Set up sourceppa cache for testing + sourceppa_path = os.path.join(self.data.dirs[True], "SourcePPA") + with open(sourceppa_path, "w", encoding="utf-8") as sourceppa: + sourceppa.write(json.dumps(self.sourceppa_cache)) + + (excuses_yaml, excuses_html, out) = self.run_britney() + + bugs_blocked_by = [] + + # convert excuses to source indexed dict + excuses_dict = {} + for s in yaml.safe_load(excuses_yaml)["sources"]: + excuses_dict[s["source"]] = s + + if "SHOW_EXCUSES" in os.environ: + print("------- excuses -----") + pprint.pprint(excuses_dict, width=200) + if "SHOW_HTML" in os.environ: + print("------- excuses.html -----\n%s\n" % excuses_html) + if "SHOW_OUTPUT" in os.environ: + print("------- output -----\n%s\n" % out) + + self.assertNotIn("FIXME", out) + + self.assertDictEqual( + expect_bugs["libc6"], + { + k: v + for (k, v) in excuses_dict["libc6"]["policy_info"][ + "update-excuse" + ].items() + if k != "verdict" + }, + ) + + return (out, excuses_dict, excuses_html, bugs_blocked_by) + + def test_email_sent(self): + """Test that an email is sent through the SMTP server""" + unixtime = calendar.timegm(time.strptime("Thu May 28 16:38:19 2020")) + pkg = ("libc6", {"Version": "2"}, [("1", unixtime)]) + + self.do_test([pkg], {"libc6": {"1": unixtime}}) + + +if __name__ == "__main__": + unittest.main()