Skip to content

Instantly share code, notes, and snippets.

@mgedmin
Created November 28, 2013 09:16
Show Gist options
  • Save mgedmin/7689218 to your computer and use it in GitHub Desktop.
Save mgedmin/7689218 to your computer and use it in GitHub Desktop.
Automate PPA package copies, version 2.
#!/usr/bin/python
"""Copy published precise PPA packages to quantal raring saucy.
Typical usage:
- build a gtimelog package for precise
- dput ppa:gtimelog-dev gtimelog_0.8.1-0ppa0_source.changes
- wait for it to be built
- run ppa-gtimelog-copy-packages
Based on https://gist.github.com/mgedmin/6306694
"""
import functools
import logging
import optparse
import sys
import time
from collections import defaultdict
from launchpadlib.launchpad import Launchpad
#
# Hardcoded configuration
#
PPA_OWNER = 'gtimelog-dev'
PPA_NAME = 'ppa'
PACKAGE_WHITELIST = ['gtimelog']
SOURCE_SERIES = 'precise'
TARGET_SERIESES = ['saucy', 'raring', 'quantal', 'trusty']
POCKET = 'Release'
APP_NAME = 'ppa-gtimelog-copy-packages'
#
# Logging
#
log = logging.getLogger(APP_NAME)
STARTUP_TIME = LAST_LOG_TIME = time.time()
REQUESTS = LAST_REQUESTS = 0
class DebugFormatter(logging.Formatter):
def format(self, record):
global LAST_LOG_TIME, LAST_REQUESTS
msg = super(DebugFormatter, self).format(record)
if msg.startswith(" "): # continuation of previous message
return msg
now = time.time()
elapsed = now - STARTUP_TIME
delta = now - LAST_LOG_TIME
LAST_LOG_TIME = now
delta_requests = REQUESTS - LAST_REQUESTS
LAST_REQUESTS = REQUESTS
return "\n%.3fs (%+.3fs) [%d/+%d] %s" % (elapsed, delta, REQUESTS, delta_requests, msg)
def enable_http_debugging():
import httplib2
httplib2.debuglevel = 1
def install_request_counter():
import httplib2
orig = httplib2.Http.request
@functools.wraps(orig)
def wrapper(*args, **kw):
global REQUESTS
REQUESTS += 1
return orig(*args, **kw)
httplib2.Http.request = wrapper
def set_up_logging(level=logging.INFO):
handler = logging.StreamHandler(sys.stdout)
if level == logging.DEBUG:
handler.setFormatter(DebugFormatter())
log.addHandler(handler)
log.setLevel(level)
#
# Caching decorators
#
class once(object):
"""A @property that is computed only once per instance.
Also known as @reify in Pyramid and @Lazy in Zope.
"""
def __init__(self, fn):
self.fn = fn
def __get__(self, obj, type=None):
value = self.fn(obj)
# Put the value in the instance __dict__. Since the once
# descriptor has no __set__ and is therefore not a data
# descriptor, instance __dict__ will take precedence on
# subsequent attribute accesses.
setattr(obj, self.fn.__name__, value)
return value
def cache(fn):
"""Trivial memoization decorator."""
cache = fn.cache = {}
@functools.wraps(fn)
def inner(*args):
# The way we use the cache is suboptimal for methods: it is
# shared between all instances, but the instance is part of
# the cache lookup key, negating any advantage of the sharing.
# Luckily in this script there's only one instance.
try:
return cache[args]
except KeyError:
value = cache[args] = fn(*args)
return value
except TypeError:
raise TypeError('%s argument types preclude caching: %s' %
(fn.__name__, repr(args)))
return inner
#
# Launchpad API wrapper with heavy caching
#
class LaunchpadWrapper(object):
application_name = 'ppa-gtimelog-copy-packages'
launchpad_instance = 'production'
ppa_owner = PPA_OWNER
ppa_name = PPA_NAME
def __init__(self):
self.queue = defaultdict(set)
# Trivial caching wrappers for Launchpad API
@once
def lp(self):
log.debug("Logging in...")
# cost: 2 HTTP requests
return Launchpad.login_with(self.application_name, self.launchpad_instance)
@once
def owner(self):
lp = self.lp # ensure correct ordering of debug messages
log.debug("Getting the owner...")
# cost: 1 HTTP request
return lp.people[self.ppa_owner]
@once
def ppa(self):
owner = self.owner # ensure correct ordering of debug messages
log.debug("Getting the PPA...")
# cost: 1 HTTP request
return owner.getPPAByName(name=self.ppa_name)
@cache
def get_series(self, name):
ppa = self.ppa # ensure correct ordering of debug messages
log.debug("Locating the series: %s...", name)
# cost: 1 HTTP request
return ppa.distribution.getSeries(name_or_version=name)
@cache
def get_published_sources(self, series_name):
series = self.get_series(series_name)
log.debug("Listing source packages for %s...", series_name)
# cost: 1 HTTP request
return self.ppa.getPublishedSources(distro_series=series)
@cache
def get_builds_for_source(self, source):
log.debug("Listing %s builds for %s %s...",
source.distro_series_link.rpartition('/')[-1],
source.source_package_name,
source.source_package_version)
# cost: 1 HTTP request, plus another one for accessing the list
return source.getBuilds()
# Package availability: for a certain package name + version
# - source package is available but not published
# - source package is published, but binary package is not built
# - binary package is built but not published
# - binary package is published
# And then there are also superseded, deleted and obsolete packages.
# Figuring out binary package status costs extra HTTP requests.
# We're dealing with arch: any packages only, so there's only one
# relevant binary package. It should be trivial to extend this to
# arch: all.
# A package can be copied when it has a published binary package
# in the source series.
# A package should be copied when it can be copied, and it doesn't
# exist in the target series.
@cache
def get_source_packages(self, series_name, package_names=None):
"""Return {package_name: {version: 'Status', ...}, ...}"""
res = defaultdict(dict)
for source in self.get_published_sources(series_name):
name = source.source_package_name
if package_names is not None and name not in package_names:
continue
version = source.source_package_version
res[name][version] = source
return res
def get_source_for(self, name, version, series_name):
sources = self.get_source_packages(series_name)
return sources.get(name, {}).get(version)
def is_missing(self, name, version, series_name):
return self.get_source_for(name, version, series_name) is None
def get_builds_for(self, name, version, series_name):
source = self.get_source_for(name, version, series_name)
if not source:
return None
return self.get_builds_for_source(source)
def has_published_binaries(self, name, version, series_name):
builds = self.get_builds_for(name, version, series_name)
# XXX: this is probably somewhat bogus
# e.g. it doesn't check that the binary has been published
return not builds or builds[0].buildstate == u'Successfully built'
@cache
def get_usable_sources(self, package_names, series_name):
res = []
for source in self.get_published_sources(series_name):
name = source.source_package_name
if name not in package_names:
continue
version = source.source_package_version
if source.status in ('Superseded', 'Deleted', 'Obsolete'):
log.info("%s %s is %s in %s", name, version,
source.status.lower(), series_name)
continue
if source.status != ('Published'):
log.warning("%s %s is %s in %s", name, version,
source.status.lower(), series_name)
continue
res.append((name, version))
return res
def queue_copy(self, name, source_series, target_series, pocket):
self.queue[source_series, target_series, pocket].add(name)
def perform_queued_copies(self):
first = True
for (source_series, target_series, pocket), names in self.queue.items():
if not names:
continue
if first:
log.info("")
first = False
log.info("Copying %s to %s", ', '.join(sorted(names)),
target_series)
self.ppa.syncSources(from_archive=self.ppa,
to_series=target_series,
to_pocket=pocket,
include_binaries=True,
source_names=sorted(names))
def main():
parser = optparse.OptionParser('usage: %prog [options]',
description="copy ppa:%s/%s packages from %s to %s"
% (PPA_OWNER, PPA_NAME, SOURCE_SERIES, ', '.join(TARGET_SERIESES)))
parser.add_option('-v', '--verbose', action='count')
parser.add_option('-q', '--quiet', action='store_true')
parser.add_option('--debug', action='store_true')
opts, args = parser.parse_args()
if opts.debug:
enable_http_debugging()
install_request_counter()
set_up_logging(logging.DEBUG)
elif opts.verbose > 1:
install_request_counter()
set_up_logging(logging.DEBUG)
elif opts.verbose:
set_up_logging(logging.INFO)
else:
set_up_logging(logging.WARNING)
lp = LaunchpadWrapper()
for (name, version) in lp.get_usable_sources(tuple(PACKAGE_WHITELIST),
SOURCE_SERIES):
mentioned = False
notices = []
for target_series_name in TARGET_SERIESES:
source = lp.get_source_for(name, version, target_series_name)
if source is None:
mentioned = True
log.info("%s %s missing from %s", name, version,
target_series_name)
if lp.has_published_binaries(name, version,
SOURCE_SERIES):
lp.queue_copy(name, SOURCE_SERIES,
target_series_name, POCKET)
else:
builds = lp.get_builds_for(name, version, SOURCE_SERIES)
if builds:
log.info(" but it isn't built yet (state: %s) - %s",
builds[0].buildstate, builds[0].web_link)
elif source.status != 'Published':
notices.append(" but it is %s in %s" %
(source.status.lower(), target_series_name))
elif not lp.has_published_binaries(name, version, target_series_name):
builds = lp.get_builds_for(name, version, target_series_name)
if builds:
notices.append(" but it isn't built yet for %s (state: %s) - %s" %
(target_series_name, builds[0].buildstate,
builds[0].web_link))
if not mentioned or notices:
log.info("%s %s", name, version)
for notice in notices:
log.info(notice)
lp.perform_queued_copies()
log.debug("All done")
if __name__ == '__main__':
main()
@mgedmin
Copy link
Author

mgedmin commented Sep 5, 2019

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment