mirror of
				https://git.launchpad.net/~ubuntu-release/britney/+git/britney2-ubuntu
				synced 2025-10-22 12:24:02 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			325 lines
		
	
	
		
			12 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
			
		
		
	
	
			325 lines
		
	
	
		
			12 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
| import os
 | |
| import re
 | |
| import json
 | |
| import math
 | |
| import socket
 | |
| import smtplib
 | |
| 
 | |
| from urllib.error import HTTPError, URLError
 | |
| from urllib.parse import unquote
 | |
| from collections import defaultdict
 | |
| 
 | |
| from britney2 import SuiteClass
 | |
| from britney2.policies.rest import Rest
 | |
| from britney2.policies.policy import BasePolicy, PolicyVerdict
 | |
| 
 | |
| 
 | |
| # Recurring emails should never be more than this many days apart
 | |
| MAX_INTERVAL = 30
 | |
| 
 | |
| API_PREFIX = "https://api.launchpad.net/1.0/"
 | |
| USER = API_PREFIX + "~"
 | |
| 
 | |
| # Don't send emails to these bots
 | |
| BOTS = {
 | |
|     USER + "ci-train-bot",
 | |
|     USER + "bileto-bot",
 | |
|     USER + "ubuntu-archive-robot",
 | |
|     USER + "katie",
 | |
| }
 | |
| 
 | |
| MESSAGE = """From: Ubuntu Release Team <noreply+proposed-migration@ubuntu.com>
 | |
| To: {recipients}
 | |
| X-Proposed-Migration: notice
 | |
| Subject: [proposed-migration] {source_name} {version} stuck in {series}-proposed for {age} day{plural}.
 | |
| 
 | |
| Hi,
 | |
| 
 | |
| {source_name} {version} needs attention.
 | |
| 
 | |
| It has been stuck in {series}-proposed for {age} day{plural}.
 | |
| 
 | |
| You either sponsored or uploaded this package, please investigate why it hasn't been approved for migration.
 | |
| 
 | |
| http://people.canonical.com/~ubuntu-archive/proposed-migration/{series}/update_excuses.html#{source_name}
 | |
| 
 | |
| https://wiki.ubuntu.com/ProposedMigration
 | |
| 
 | |
| If you have any questions about this email, please ask them in #ubuntu-release channel on libera.chat.
 | |
| 
 | |
| Regards, Ubuntu Release Team.
 | |
| """
 | |
| 
 | |
| 
 | |
| def person_chooser(source):
 | |
|     """Assign blame for the current source package."""
 | |
|     people = (
 | |
|         {
 | |
|             source["package_signer_link"],
 | |
|             source["sponsor_link"],
 | |
|             source["creator_link"],
 | |
|         }
 | |
|         - {None}
 | |
|         - BOTS
 | |
|     )
 | |
|     # some bots (e.g. bileto) generate uploads that are otherwise manual. We
 | |
|     # want to email the people that the bot was acting on behalf of.
 | |
|     bot = source["package_signer_link"] in BOTS
 | |
|     # direct uploads
 | |
|     regular = not source["creator_link"] and not source["sponsor_link"]
 | |
|     if bot or regular:
 | |
|         people.add(source["package_creator_link"])
 | |
|     return people
 | |
| 
 | |
| 
 | |
| def address_chooser(addresses):
 | |
|     """Prefer @ubuntu and @canonical addresses."""
 | |
|     first = ""
 | |
|     canonical = ""
 | |
|     for address in addresses:
 | |
|         if address.endswith("@ubuntu.com"):
 | |
|             return address
 | |
|         if address.endswith("@canonical.com"):
 | |
|             canonical = address
 | |
|         if not first:
 | |
|             first = address
 | |
|     return canonical or first
 | |
| 
 | |
| 
 | |
| class EmailPolicy(BasePolicy, Rest):
 | |
|     """Send an email when a package has been rejected."""
 | |
| 
 | |
|     def __init__(self, options, suite_info, dry_run=False):
 | |
|         super().__init__(
 | |
|             "email", options, suite_info, {SuiteClass.PRIMARY_SOURCE_SUITE}
 | |
|         )
 | |
|         self.filename = os.path.join(options.unstable, "EmailCache")
 | |
|         # Maps lp username -> email address
 | |
|         self.addresses = {}
 | |
|         # Dict of dicts; maps pkg name -> pkg version -> boolean
 | |
|         self.emails_by_pkg = defaultdict(dict)
 | |
|         # self.cache contains self.emails_by_pkg from previous run
 | |
|         self.cache = {}
 | |
|         self.dry_run = dry_run
 | |
|         self.email_host = getattr(self.options, "email_host", "localhost")
 | |
|         self.logger.info(
 | |
|             "EmailPolicy: will send emails to: %s", self.email_host
 | |
|         )
 | |
| 
 | |
|     def initialise(self, britney):
 | |
|         """Load cached source ppa data"""
 | |
|         super().initialise(britney)
 | |
| 
 | |
|         if os.path.exists(self.filename):
 | |
|             with open(self.filename, encoding="utf-8") as data:
 | |
|                 self.cache = json.load(data)
 | |
|             self.logger.info("Loaded cached email data from %s" % self.filename)
 | |
|         tmp = self.filename + ".new"
 | |
|         if os.path.exists(tmp):
 | |
|             # if we find a record on disk of emails sent from an incomplete
 | |
|             # britney run, merge them in now.
 | |
|             with open(tmp, encoding="utf-8") as data:
 | |
|                 self.cache.update(json.load(data))
 | |
|             self._save_progress(self.cache)
 | |
|             self.save_state()
 | |
| 
 | |
|     def _scrape_gpg_emails(self, person):
 | |
|         """Find email addresses from one person's GPG keys."""
 | |
|         if person in self.addresses:
 | |
|             return self.addresses[person]
 | |
|         addresses = []
 | |
|         try:
 | |
|             gpg = self.query_lp_rest_api(person + "/gpg_keys", {})
 | |
|             for key in gpg["entries"]:
 | |
|                 details = self.query_rest_api(
 | |
|                     "http://keyserver.ubuntu.com/pks/lookup",
 | |
|                     {
 | |
|                         "op": "index",
 | |
|                         "search": "0x" + key["fingerprint"],
 | |
|                         "exact": "on",
 | |
|                         "options": "mr",
 | |
|                     },
 | |
|                 )
 | |
|                 for line in details.splitlines():
 | |
|                     parts = line.split(":")
 | |
|                     if parts[0] == "info":
 | |
|                         if int(parts[1]) != 1 or int(parts[2]) > 1:
 | |
|                             break
 | |
|                     if parts[0] == "uid":
 | |
|                         flags = parts[4]
 | |
|                         if "e" in flags or "r" in flags:
 | |
|                             continue
 | |
|                         uid = unquote(parts[1])
 | |
|                         match = re.match(r"^.*<(.+@.+)>$", uid)
 | |
|                         if match:
 | |
|                             addresses.append(match.group(1))
 | |
|             address = self.addresses[person] = address_chooser(addresses)
 | |
|             if not address:
 | |
|                 return None
 | |
|             return address
 | |
|         except HTTPError as e:
 | |
|             if e.code != 410:  # suspended user
 | |
|                 raise
 | |
|             self.logger.info(
 | |
|                 "Ignoring person %s as suspended in Launchpad" % person
 | |
|             )
 | |
|             return None
 | |
| 
 | |
|     def scrape_gpg_emails(self, people):
 | |
|         """Find email addresses from GPG keys."""
 | |
|         emails = [self._scrape_gpg_emails(person) for person in (people or [])]
 | |
|         if not emails:
 | |
|             return
 | |
|         return [email for email in emails if email is not None]
 | |
| 
 | |
|     def lp_get_emails(self, pkg, version):
 | |
|         """Ask LP who uploaded this package."""
 | |
|         try:
 | |
|             data = self.query_lp_rest_api(
 | |
|                 "%s/+archive/primary" % self.options.distribution,
 | |
|                 {
 | |
|                     "ws.op": "getPublishedSources",
 | |
|                     "distro_series": "/%s/%s"
 | |
|                     % (self.options.distribution, self.options.series),
 | |
|                     "exact_match": "true",
 | |
|                     "order_by_date": "true",
 | |
|                     "pocket": "Proposed",
 | |
|                     "source_name": pkg,
 | |
|                     "version": version,
 | |
|                 },
 | |
|             )
 | |
|         except URLError as e:
 | |
|             self.logger.error("Error getting uploader from Launchpad for %s/%s: %s",
 | |
|                               pkg, version, e.reason)
 | |
|             return []
 | |
|         try:
 | |
|             source = next(reversed(data["entries"]))
 | |
|         # IndexError means no packages in -proposed matched this name/version,
 | |
|         # which is expected to happen when bileto runs britney.
 | |
|         except StopIteration:
 | |
|             self.logger.info(
 | |
|                 "Email getPublishedSources IndexError (%s %s)" % (pkg, version)
 | |
|             )
 | |
|             return []
 | |
|         return self.scrape_gpg_emails(person_chooser(source))
 | |
| 
 | |
|     def apply_src_policy_impl(
 | |
|         self, email_info, item, source_data_tdist, source_data_srcdist, excuse
 | |
|     ):
 | |
|         """Send email if package is rejected."""
 | |
|         source_name = item.package
 | |
|         max_age = 5 if excuse.is_valid else 1
 | |
|         series = self.options.series
 | |
|         version = source_data_srcdist.version
 | |
|         age = int(excuse.daysold) or 0
 | |
|         plural = "" if age == 1 else "s"
 | |
|         # an item is stuck if it's
 | |
|         # - old enough
 | |
|         # - not blocked
 | |
|         # - not temporarily rejected (e.g. by the autopkgtest policy when tests
 | |
|         #   are still running)
 | |
|         stuck = (
 | |
|             age >= max_age
 | |
|             and "block" not in excuse.reason
 | |
|             and excuse.tentative_policy_verdict
 | |
|             != PolicyVerdict.REJECTED_TEMPORARILY
 | |
|         )
 | |
|         if self.dry_run:
 | |
|             self.logger.info(
 | |
|                 "[email dry run] Considering: %s/%s: %s"
 | |
|                 % (source_name, version, "stuck" if stuck else "not stuck")
 | |
|             )
 | |
| 
 | |
|         if not stuck:
 | |
|             return PolicyVerdict.PASS
 | |
| 
 | |
|         cached = self.cache.get(source_name, {}).get(version)
 | |
|         try:
 | |
|             emails, last_sent = cached
 | |
|             # migration of older data
 | |
|             last_sent = int(last_sent)
 | |
|             # Find out whether we are due to send another email by calculating
 | |
|             # the most recent age at which we should have sent one.  A
 | |
|             # sequence of doubling intervals (0 + 1 = 1, 1 + 2 = 3, 3 + 4 = 7)
 | |
|             # is equivalent to 2^n-1, or 2^n + (max_age - 1) - 1.
 | |
|             # 2^(floor(log2(age))) straightforwardly calculates the most
 | |
|             # recent age at which we wanted to send an email.
 | |
|             last_due = int(
 | |
|                 math.pow(2, int(math.log(age + 2 - max_age, 2))) + max_age - 2
 | |
|             )
 | |
|             # Don't let the interval double without bounds.
 | |
|             if last_due - max_age >= MAX_INTERVAL:
 | |
|                 last_due = (
 | |
|                     int((age - max_age - MAX_INTERVAL) / MAX_INTERVAL)
 | |
|                     * MAX_INTERVAL
 | |
|                     + max_age
 | |
|                     + MAX_INTERVAL
 | |
|                 )
 | |
|             # And don't send emails before we've reached the minimum age
 | |
|             # threshold.
 | |
|             if last_due < max_age:
 | |
|                 last_due = max_age
 | |
| 
 | |
|         except TypeError:
 | |
|             # This exception happens when source_name, version never seen before
 | |
|             emails = []
 | |
|             last_sent = 0
 | |
|             last_due = max_age
 | |
|         if self.dry_run:
 | |
|             self.logger.info(
 | |
|                 "[email dry run] Age %d >= threshold %d: would email: %s"
 | |
|                 % (age, max_age, self.lp_get_emails(source_name, version))
 | |
|             )
 | |
|             # don't update the cache file in dry run mode; we'll see all output each time
 | |
|             return PolicyVerdict.PASS
 | |
|         if last_sent < last_due:
 | |
|             if not emails:
 | |
|                 emails = self.lp_get_emails(source_name, version)
 | |
|             if emails:
 | |
|                 recipients = ", ".join(emails)
 | |
|                 msg = MESSAGE.format(**locals())
 | |
|                 try:
 | |
|                     self.logger.info(
 | |
|                         "%s/%s stuck for %d days (email last sent at %d days old, "
 | |
|                         "threshold for sending %d days), emailing %s"
 | |
|                         % (
 | |
|                             source_name,
 | |
|                             version,
 | |
|                             age,
 | |
|                             last_sent,
 | |
|                             last_due,
 | |
|                             recipients,
 | |
|                         )
 | |
|                     )
 | |
|                     server = smtplib.SMTP(self.email_host)
 | |
|                     server.sendmail("noreply+proposed-migration@ubuntu.com", emails, msg)
 | |
|                     server.quit()
 | |
|                     # record the age at which the mail should have been sent
 | |
|                     last_sent = last_due
 | |
|                 except socket.error as err:
 | |
|                     self.logger.error(
 | |
|                         "Failed to send mail! Is SMTP server running?"
 | |
|                     )
 | |
|                     self.logger.error(err)
 | |
|         self.emails_by_pkg[source_name][version] = (emails, last_sent)
 | |
|         self._save_progress(self.emails_by_pkg)
 | |
|         return PolicyVerdict.PASS
 | |
| 
 | |
|     def _save_progress(self, my_data):
 | |
|         """Checkpoint after each sent mail"""
 | |
|         tmp = self.filename + ".new"
 | |
|         with open(tmp, "w", encoding="utf-8") as data:
 | |
|             json.dump(my_data, data)
 | |
|         return tmp
 | |
| 
 | |
|     def save_state(self, britney=None):
 | |
|         """Save email notification status of all pending packages"""
 | |
|         if not self.dry_run:
 | |
|             try:
 | |
|                 os.rename(self.filename + ".new", self.filename)
 | |
|             # if we haven't written any cache, don't clobber the old one
 | |
|             except FileNotFoundError:
 | |
|                 pass
 | |
|         if britney:
 | |
|             self.logger.info("Wrote email data to %s" % self.filename)
 |