mirror of
https://git.launchpad.net/ubuntu-dev-tools
synced 2025-05-05 14:01:33 +00:00
Snapshot and rmadison support
This commit is contained in:
parent
3013ee034a
commit
a2da2da87b
@ -100,7 +100,7 @@ def main():
|
|||||||
oldpkg.unpack()
|
oldpkg.unpack()
|
||||||
|
|
||||||
cmd = ['debdiff', oldpkg.dsc_name, newpkg.dsc_name]
|
cmd = ['debdiff', oldpkg.dsc_name, newpkg.dsc_name]
|
||||||
difffn = newdsc[:-3] + 'debdiff'
|
difffn = newpkg.dsc_name[:-3] + 'debdiff'
|
||||||
Logger.command(cmd + ['> %s' % difffn])
|
Logger.command(cmd + ['> %s' % difffn])
|
||||||
debdiff_file = open(difffn, 'w')
|
debdiff_file = open(difffn, 'w')
|
||||||
if subprocess.call(cmd, stdout=debdiff_file) > 2:
|
if subprocess.call(cmd, stdout=debdiff_file) > 2:
|
||||||
|
@ -39,7 +39,10 @@ import debian.debian_support
|
|||||||
|
|
||||||
from ubuntutools.config import UDTConfig
|
from ubuntutools.config import UDTConfig
|
||||||
from ubuntutools.logger import Logger
|
from ubuntutools.logger import Logger
|
||||||
from ubuntutools.lp.lpapicache import Launchpad, Distribution
|
from ubuntutools.lp.lpapicache import (Launchpad, Distribution,
|
||||||
|
SourcePackagePublishingHistory)
|
||||||
|
from ubuntutools.requestsync.mail import (SourcePackagePublishingHistory
|
||||||
|
as rmadison_SPPH)
|
||||||
|
|
||||||
class DownloadError(Exception):
|
class DownloadError(Exception):
|
||||||
"Unable to pull a source package"
|
"Unable to pull a source package"
|
||||||
@ -137,7 +140,7 @@ class SourcePackage(object):
|
|||||||
version=self.version.full_version,
|
version=self.version.full_version,
|
||||||
exact_match=True,
|
exact_match=True,
|
||||||
))
|
))
|
||||||
self._spph = spph[0]
|
self._spph = SourcePackagePublishingHistory(spph[0])
|
||||||
return self._spph
|
return self._spph
|
||||||
|
|
||||||
@property
|
@property
|
||||||
@ -145,7 +148,7 @@ class SourcePackage(object):
|
|||||||
"Cached archive component, in available"
|
"Cached archive component, in available"
|
||||||
if not self._component:
|
if not self._component:
|
||||||
Logger.debug('Determining component from Launchpad')
|
Logger.debug('Determining component from Launchpad')
|
||||||
self._component = self.lp_spph.component_name
|
self._component = self.lp_spph.getComponent()
|
||||||
return self._component
|
return self._component
|
||||||
|
|
||||||
@property
|
@property
|
||||||
@ -202,10 +205,17 @@ class SourcePackage(object):
|
|||||||
if (parsed.scheme != 'file'
|
if (parsed.scheme != 'file'
|
||||||
or os.path.realpath(os.path.dirname(parsed.path))
|
or os.path.realpath(os.path.dirname(parsed.path))
|
||||||
!= os.path.realpath(self.workdir)):
|
!= os.path.realpath(self.workdir)):
|
||||||
self._download_file(self._dsc_source)
|
if not self._download_file(self._dsc_source, self.dsc_name):
|
||||||
|
raise DownloadError('dsc not found')
|
||||||
else:
|
else:
|
||||||
self._download_file(self._lp_url(self.dsc_name))
|
if not self._download_file(self._lp_url(self.dsc_name),
|
||||||
|
self.dsc_name):
|
||||||
|
raise DownloadError('dsc not found')
|
||||||
|
self._check_dsc()
|
||||||
|
|
||||||
|
def _check_dsc(self):
|
||||||
|
"Check that the dsc matches what we are expecting"
|
||||||
|
assert os.path.exists(self.dsc_name)
|
||||||
self._dsc_fetched = True
|
self._dsc_fetched = True
|
||||||
|
|
||||||
assert self.source == self.dsc['Source']
|
assert self.source == self.dsc['Source']
|
||||||
@ -214,9 +224,11 @@ class SourcePackage(object):
|
|||||||
assert self.version.debian_revision == version.debian_revision
|
assert self.version.debian_revision == version.debian_revision
|
||||||
self.version = version
|
self.version = version
|
||||||
|
|
||||||
def _download_file(self, url):
|
def _download_file(self, url, filename):
|
||||||
"Download url to workdir"
|
"Download url to filename in workdir."
|
||||||
filename = os.path.basename(url)
|
logurl = url
|
||||||
|
if os.path.basename(url) != filename:
|
||||||
|
logurl += ' -> ' + filename
|
||||||
pathname = os.path.join(self.workdir, filename)
|
pathname = os.path.join(self.workdir, filename)
|
||||||
if self.dsc and not url.endswith('.dsc'):
|
if self.dsc and not url.endswith('.dsc'):
|
||||||
if self.dsc.verify_file(pathname):
|
if self.dsc.verify_file(pathname):
|
||||||
@ -226,12 +238,16 @@ class SourcePackage(object):
|
|||||||
if entry['name'] == filename]
|
if entry['name'] == filename]
|
||||||
assert len(size) == 1
|
assert len(size) == 1
|
||||||
size = int(size[0])
|
size = int(size[0])
|
||||||
Logger.normal('Downloading %s (%0.3f MiB)', url,
|
Logger.normal('Downloading %s (%0.3f MiB)', logurl,
|
||||||
size / 1024.0 / 1024)
|
size / 1024.0 / 1024)
|
||||||
else:
|
else:
|
||||||
Logger.normal('Downloading %s', url)
|
Logger.normal('Downloading %s', logurl)
|
||||||
|
|
||||||
|
try:
|
||||||
in_ = urllib2.urlopen(url)
|
in_ = urllib2.urlopen(url)
|
||||||
|
except urllib2.HTTPError, e:
|
||||||
|
return False
|
||||||
|
|
||||||
out = open(pathname, 'wb')
|
out = open(pathname, 'wb')
|
||||||
while True:
|
while True:
|
||||||
block = in_.read(10240)
|
block = in_.read(10240)
|
||||||
@ -255,9 +271,10 @@ class SourcePackage(object):
|
|||||||
if self.dsc is None:
|
if self.dsc is None:
|
||||||
self.pull_dsc()
|
self.pull_dsc()
|
||||||
for entry in self.dsc['Files']:
|
for entry in self.dsc['Files']:
|
||||||
for url in self._source_urls(entry['name']):
|
name = entry['name']
|
||||||
|
for url in self._source_urls(name):
|
||||||
try:
|
try:
|
||||||
if self._download_file(url):
|
if self._download_file(url, name):
|
||||||
break
|
break
|
||||||
except urllib2.HTTPError, e:
|
except urllib2.HTTPError, e:
|
||||||
Logger.normal('HTTP Error %i: %s', e.code, str(e))
|
Logger.normal('HTTP Error %i: %s', e.code, str(e))
|
||||||
@ -279,72 +296,75 @@ class SourcePackage(object):
|
|||||||
class DebianSourcePackage(SourcePackage):
|
class DebianSourcePackage(SourcePackage):
|
||||||
"Download / unpack a Debian source package"
|
"Download / unpack a Debian source package"
|
||||||
distribution = 'debian'
|
distribution = 'debian'
|
||||||
# TODO: Security support
|
|
||||||
# TODO: snapshot support
|
|
||||||
# TODO: Madison component fallback
|
|
||||||
# TODO: GPG verification fallback
|
|
||||||
|
|
||||||
class UbuntuSourcePackage(SourcePackage):
|
def __init__(self, *args, **kwargs):
|
||||||
"Download / unpack an Ubuntu source package"
|
super(DebianSourcePackage, self).__init__(*args, **kwargs)
|
||||||
distribution = 'ubuntu'
|
self.masters.append(UDTConfig.defaults['DEBSEC_MIRROR'])
|
||||||
|
# Cached values:
|
||||||
|
self._snapshot_list = None
|
||||||
|
|
||||||
# TODO: Delete everything after this point.
|
# Overridden methods:
|
||||||
def pull_source_pkg(archives, mirrors, component, package, version, workdir='.',
|
@property
|
||||||
unpack=False):
|
def lp_spph(self):
|
||||||
"""Download a source package or die.
|
"Return the LP Source Package Publishing History entry"
|
||||||
archives may be a list or single item (in which case mirrors can be too)
|
if not self._spph:
|
||||||
mirrors should be a dict (keyed on archive) unless archives is single"""
|
try:
|
||||||
|
return super(DebianSourcePackage, self).lp_spph
|
||||||
|
except IndexError:
|
||||||
|
pass
|
||||||
|
|
||||||
if not isinstance(archives, (tuple, list)):
|
Logger.normal('Using rmadison for component determination')
|
||||||
if not isinstance(mirrors, dict):
|
p = subprocess.Popen(('rmadison', '-u', 'debian', self.source),
|
||||||
mirrors = {archives: mirrors}
|
stdout=subprocess.PIPE)
|
||||||
archives = [archives]
|
rmadison = p.communicate()[0]
|
||||||
assert all(x in ('DEBIAN', 'DEBSEC', 'UBUNTU') for x in archives)
|
comp = 'main'
|
||||||
|
for line in rmadison.strip().splitlines():
|
||||||
|
pkg, ver, dist, archs = [x.strip() for x in line.split('|')]
|
||||||
|
comp = 'main'
|
||||||
|
if '/' in dist:
|
||||||
|
dist, comp = dist.split('/')
|
||||||
|
if ver == self.version.full_version:
|
||||||
|
self._spph = rmadison_SPPH(pkg, ver, comp)
|
||||||
|
return self._spph
|
||||||
|
|
||||||
for archive in archives:
|
Logger.normal('Guessing component from most recent upload')
|
||||||
filename = try_pull_from_archive(archive, mirrors.get(archive),
|
self._spph = rmadison_SPPH(self.source, self.version.full_version,
|
||||||
component, package, version,
|
comp)
|
||||||
workdir, unpack)
|
return self._spph
|
||||||
if filename:
|
|
||||||
return filename
|
|
||||||
|
|
||||||
if 'DEBIAN' in archives or 'DEBSEC' in archives:
|
def _source_urls(self, name):
|
||||||
Logger.info('Trying snapshot.debian.org')
|
"Generator of sources for name"
|
||||||
filename = try_pull_from_snapshot(package, version, workdir, unpack)
|
it = super(DebianSourcePackage, self)._source_urls(name)
|
||||||
if filename:
|
while True:
|
||||||
return filename
|
try:
|
||||||
|
yield it.next()
|
||||||
|
except StopIteration:
|
||||||
|
break
|
||||||
|
if self._snapshot_list:
|
||||||
|
yield self._snapshot_url(name)
|
||||||
|
|
||||||
if 'UBUNTU' in archives:
|
def pull_dsc(self):
|
||||||
Logger.info('Trying Launchpad')
|
"Retrieve dscfile and parse"
|
||||||
filename = try_pull_from_lp(package, 'ubuntu', version, workdir, unpack)
|
try:
|
||||||
if filename:
|
super(DebianSourcePackage, self).pull_dsc()
|
||||||
return filename
|
return
|
||||||
|
except DownloadError:
|
||||||
|
pass
|
||||||
|
|
||||||
raise Exception('Unable to locate %s/%s %s' % (package, component, version))
|
# Not all Debian Source packages get imported to LP
|
||||||
|
# (or the importer could be lagging)
|
||||||
|
for url in self._source_urls(self.dsc_name):
|
||||||
|
if self._download_file(url, self.dsc_name):
|
||||||
|
break
|
||||||
|
else:
|
||||||
|
raise DownloadError('dsc could not be found anywhere')
|
||||||
|
self._check_dsc()
|
||||||
|
|
||||||
def try_pull_from_archive(archive, mirror, component, package, version,
|
# Local methods:
|
||||||
workdir='.', unpack=False):
|
@property
|
||||||
"""Download a source package from the specified source, return filename.
|
def snapshot_list(self):
|
||||||
Try mirror first, then master.
|
"Return a filename -> hash dictionary from snapshot.debian.org"
|
||||||
"""
|
if self._snapshot_list is None:
|
||||||
assert archive in ('DEBIAN', 'DEBSEC', 'UBUNTU')
|
|
||||||
urls = []
|
|
||||||
if mirror and mirror != UDTConfig.defaults[archive + '_MIRROR']:
|
|
||||||
urls.append(dsc_url(mirror, component, package, version))
|
|
||||||
urls.append(dsc_url(UDTConfig.defaults[archive + '_MIRROR'], component,
|
|
||||||
package, version))
|
|
||||||
|
|
||||||
for url in urls:
|
|
||||||
cmd = ('dget', '-u' + ('x' if unpack else 'd'), url)
|
|
||||||
Logger.command(cmd)
|
|
||||||
return_code = subprocess.call(cmd, cwd=workdir)
|
|
||||||
if return_code == 0:
|
|
||||||
return os.path.basename(url)
|
|
||||||
|
|
||||||
def try_pull_from_snapshot(package, version, workdir='.', unpack=False):
|
|
||||||
"""Download Debian source package version version from snapshot.debian.org.
|
|
||||||
Return filename.
|
|
||||||
"""
|
|
||||||
try:
|
try:
|
||||||
import json
|
import json
|
||||||
except ImportError:
|
except ImportError:
|
||||||
@ -355,73 +375,28 @@ def try_pull_from_snapshot(package, version, workdir='.', unpack=False):
|
|||||||
|
|
||||||
try:
|
try:
|
||||||
srcfiles = json.load(urllib2.urlopen(
|
srcfiles = json.load(urllib2.urlopen(
|
||||||
'http://snapshot.debian.org/mr/package/%s/%s/srcfiles'
|
'http://snapshot.debian.org'
|
||||||
% (package, version)))
|
'/mr/package/%s/%s/srcfiles?fileinfo=1'
|
||||||
|
% (self.source, self.version.full_version)))
|
||||||
except urllib2.HTTPError:
|
except urllib2.HTTPError:
|
||||||
Logger.error('Version %s of %s not found on snapshot.debian.org',
|
Logger.error('Version %s of %s not found on '
|
||||||
version, package)
|
'snapshot.debian.org',
|
||||||
return
|
self.version.full_version, self.source)
|
||||||
|
self._snapshot_list = False
|
||||||
|
return False
|
||||||
|
self._snapshot_list = dict((info[0]['name'], hash_)
|
||||||
|
for hash_, info
|
||||||
|
in srcfiles['fileinfo'].iteritems())
|
||||||
|
return self._snapshot_list
|
||||||
|
|
||||||
for hash_ in srcfiles['result']:
|
def _snapshot_url(self, name):
|
||||||
hash_ = hash_['hash']
|
"Return the snapshot.debian.org URL for name"
|
||||||
|
return os.path.join('http://snapshot.debian.org/file',
|
||||||
|
self.snapshot_list[name])
|
||||||
|
|
||||||
try:
|
# TODO: GPG verification fallback
|
||||||
info = json.load(urllib2.urlopen(
|
|
||||||
'http://snapshot.debian.org/mr/file/%s/info' % hash_))
|
|
||||||
except urllib2.URLError:
|
|
||||||
Logger.error('Unable to dowload info for hash.')
|
|
||||||
return
|
|
||||||
|
|
||||||
filename = info['result'][0]['name']
|
|
||||||
if '/' in filename:
|
|
||||||
Logger.error('Unacceptable file name: %s', filename)
|
|
||||||
return
|
|
||||||
pathname = os.path.join(workdir, filename)
|
|
||||||
|
|
||||||
if os.path.exists(pathname):
|
class UbuntuSourcePackage(SourcePackage):
|
||||||
source_file = open(pathname, 'r')
|
"Download / unpack an Ubuntu source package"
|
||||||
sha1 = hashlib.sha1()
|
distribution = 'ubuntu'
|
||||||
sha1.update(source_file.read())
|
|
||||||
source_file.close()
|
|
||||||
if sha1.hexdigest() == hash_:
|
|
||||||
Logger.normal('Using existing %s', filename)
|
|
||||||
continue
|
|
||||||
|
|
||||||
Logger.normal('Downloading: %s (%0.3f MiB)', filename,
|
|
||||||
info['result'][0]['size'] / 1024.0 / 1024)
|
|
||||||
try:
|
|
||||||
in_ = urllib2.urlopen('http://snapshot.debian.org/file/%s' % hash_)
|
|
||||||
out = open(pathname, 'w')
|
|
||||||
while True:
|
|
||||||
block = in_.read(10240)
|
|
||||||
if block == '':
|
|
||||||
break
|
|
||||||
out.write(block)
|
|
||||||
sys.stdout.write('.')
|
|
||||||
sys.stdout.flush()
|
|
||||||
sys.stdout.write('\n')
|
|
||||||
sys.stdout.flush()
|
|
||||||
out.close()
|
|
||||||
except urllib2.URLError:
|
|
||||||
Logger.error('Error downloading %s', filename)
|
|
||||||
return
|
|
||||||
|
|
||||||
filename = dsc_name(package, version)
|
|
||||||
if unpack:
|
|
||||||
cmd = ('dpkg-source', '--no-check', '-x', filename)
|
|
||||||
Logger.command(cmd)
|
|
||||||
subprocess.check_call(cmd)
|
|
||||||
return filename
|
|
||||||
|
|
||||||
def try_pull_from_lp(package, distro, version, workdir='.', unpack=False):
|
|
||||||
"""Try to download the specified version of a source package from Launchpad.
|
|
||||||
Return filename.
|
|
||||||
"""
|
|
||||||
filename = dsc_name(package, version)
|
|
||||||
url = ('https://launchpad.net/%s/+archive/primary/+files/%s'
|
|
||||||
% (distro, filename))
|
|
||||||
cmd = ('dget', '-u' + ('x' if unpack else 'd'), url)
|
|
||||||
Logger.command(cmd)
|
|
||||||
return_code = subprocess.call(cmd, cwd=workdir)
|
|
||||||
if return_code == 0:
|
|
||||||
return filename
|
|
||||||
|
Loading…
x
Reference in New Issue
Block a user