[rtems-source-builder commit] sb: Add --rsb-file options to %source and %patch to set a file name.
Chris Johns
chrisj at rtems.org
Fri Apr 15 02:00:14 UTC 2016
Module: rtems-source-builder
Branch: master
Commit: 383f7e66906213b4f0cc5bd464789bfff731ffee
Changeset: http://git.rtems.org/rtems-source-builder/commit/?id=383f7e66906213b4f0cc5bd464789bfff731ffee
Author: Chris Johns <chrisj at rtems.org>
Date: Fri Apr 15 11:44:27 2016 +1000
sb: Add --rsb-file options to %source and %patch to set a file name.
Override the automatic file name of a downloaded file and use the file
name provided by the option. This is useful if the URL has no meanful
file that can be automatically extracted from the URL.
---
source-builder/sb/build.py | 51 ++++++++++++++++++---
source-builder/sb/download.py | 100 ++++++++++++++++++++++++------------------
2 files changed, 103 insertions(+), 48 deletions(-)
diff --git a/source-builder/sb/build.py b/source-builder/sb/build.py
index 15d569d..b995e6b 100644
--- a/source-builder/sb/build.py
+++ b/source-builder/sb/build.py
@@ -209,8 +209,33 @@ class build:
if sm is None:
raise error.internal('source macro not found: %s in %s (%s)' % \
(s, name, _map))
- url = self.config.expand(sm[2])
- src = download.parse_url(url, '_sourcedir', self.config, self.opts)
+ opts = []
+ url = []
+ for sp in sm[2].split():
+ if len(url) == 0 and sp[0] == '-':
+ opts += [sp]
+ else:
+ url += [sp]
+ if len(url) == 0:
+ raise error.general('source URL not found: %s' % (' '.join(args)))
+ #
+ # Look for --rsb-file as an option we use as a local file name.
+ # This can be used if a URL has no reasonable file name the
+ # download URL parser can figure out.
+ #
+ file_override = None
+ if len(opts) > 0:
+ for o in opts:
+ if o.startswith('--rsb-file'):
+ os_ = o.split('=')
+ if len(os_) != 2:
+ raise error.general('invalid --rsb-file option: %s' % (' '.join(args)))
+ if os_[0] != '--rsb-file':
+ raise error.general('invalid --rsb-file option: %s' % (' '.join(args)))
+ file_override = os_[1]
+ opts = [o for o in opts if not o.startswith('--rsb-')]
+ url = self.config.expand(' '.join(url))
+ src = download.parse_url(url, '_sourcedir', self.config, self.opts, file_override)
download.get_file(src['url'], src['local'], self.opts, self.config)
if 'symlink' in src:
sname = name.replace('-', '_')
@@ -303,6 +328,22 @@ class build:
url += [pp]
if len(url) == 0:
raise error.general('patch URL not found: %s' % (' '.join(args)))
+ #
+ # Look for --rsb-file as an option we use as a local file name.
+ # This can be used if a URL has no reasonable file name the
+ # download URL parser can figure out.
+ #
+ file_override = None
+ if len(opts) > 0:
+ for o in opts:
+ if o.startswith('--rsb-file'):
+ os_ = o.split('=')
+ if len(os_) != 2:
+ raise error.general('invalid --rsb-file option: %s' % (' '.join(args)))
+ if os_[0] != '--rsb-file':
+ raise error.general('invalid --rsb-file option: %s' % (' '.join(args)))
+ file_override = os_[1]
+ opts = [o for o in opts if not o.startswith('--rsb-')]
if len(opts) == 0:
opts = default_opts
else:
@@ -312,12 +353,10 @@ class build:
#
# Parse the URL first in the source builder's patch directory.
#
- patch = download.parse_url(url, '_patchdir', self.config, self.opts)
+ patch = download.parse_url(url, '_patchdir', self.config, self.opts, file_override)
#
- # If not in the source builder package check the source directory.
+ # Download the patch
#
- if not path.exists(patch['local']):
- patch = download.parse_url(url, '_patchdir', self.config, self.opts)
download.get_file(patch['url'], patch['local'], self.opts, self.config)
if 'compressed' in patch:
patch['script'] = patch['compressed'] + ' ' + patch['local']
diff --git a/source-builder/sb/download.py b/source-builder/sb/download.py
index 1bd132e..32d801b 100644
--- a/source-builder/sb/download.py
+++ b/source-builder/sb/download.py
@@ -130,57 +130,62 @@ def _hash_check(file_, absfile, macros, remove = True):
def _local_path(source, pathkey, config):
for p in config.define(pathkey).split(':'):
- local = path.join(path.abspath(p), source['file'])
+ local_prefix = path.abspath(p)
+ local = path.join(local_prefix, source['file'])
if source['local'] is None:
- source['local_prefix'] = path.abspath(p)
+ source['local_prefix'] = local_prefix
source['local'] = local
if path.exists(local):
- source['local_prefix'] = path.abspath(p)
+ source['local_prefix'] = local_prefix
source['local'] = local
_hash_check(source['file'], local, config.macros)
break
def _http_parser(source, pathkey, config, opts):
#
- # Hack for gitweb.cgi patch downloads. We rewrite the various fields.
+ # If the file has not been overrided attempt to recover a possible file name.
#
- if 'gitweb.cgi' in source['url']:
- url = source['url']
- if '?' not in url:
- raise error.general('invalid gitweb.cgi request: %s' % (url))
- req = url.split('?')[1]
- if len(req) == 0:
- raise error.general('invalid gitweb.cgi request: %s' % (url))
+ if 'file-override' not in source['options']:
#
- # The gitweb.cgi request should have:
- # p=<what>
- # a=patch
- # h=<hash>
- # so extract the p and h parts to make the local name.
+ # Hack for gitweb.cgi patch downloads. We rewrite the various fields.
#
- p = None
- a = None
- h = None
- for r in req.split(';'):
- if '=' not in r:
- raise error.general('invalid gitweb.cgi path: %s' % (url))
- rs = r.split('=')
- if rs[0] == 'p':
- p = rs[1].replace('.', '-')
- elif rs[0] == 'a':
- a = rs[1]
- elif rs[0] == 'h':
- h = rs[1]
- if p is None or h is None:
- raise error.general('gitweb.cgi path missing p or h: %s' % (url))
- source['file'] = '%s-%s.patch' % (p, h)
- #
- # Check the source file name for any extra request query data and remove if
- # found. Some hosts do not like file names containing them.
- #
- if '?' in source['file']:
- qmark = source['file'].find('?')
- source['file'] = source['file'][:qmark]
+ if 'gitweb.cgi' in source['url']:
+ url = source['url']
+ if '?' not in url:
+ raise error.general('invalid gitweb.cgi request: %s' % (url))
+ req = url.split('?')[1]
+ if len(req) == 0:
+ raise error.general('invalid gitweb.cgi request: %s' % (url))
+ #
+ # The gitweb.cgi request should have:
+ # p=<what>
+ # a=patch
+ # h=<hash>
+ # so extract the p and h parts to make the local name.
+ #
+ p = None
+ a = None
+ h = None
+ for r in req.split(';'):
+ if '=' not in r:
+ raise error.general('invalid gitweb.cgi path: %s' % (url))
+ rs = r.split('=')
+ if rs[0] == 'p':
+ p = rs[1].replace('.', '-')
+ elif rs[0] == 'a':
+ a = rs[1]
+ elif rs[0] == 'h':
+ h = rs[1]
+ if p is None or h is None:
+ raise error.general('gitweb.cgi path missing p or h: %s' % (url))
+ source['file'] = '%s-%s.patch' % (p, h)
+ #
+ # Check the source file name for any extra request query data and remove if
+ # found. Some hosts do not like file names containing them.
+ #
+ if '?' in source['file']:
+ qmark = source['file'].find('?')
+ source['file'] = source['file'][:qmark]
#
# Check local path
#
@@ -296,17 +301,27 @@ parsers = { 'http': _http_parser,
'cvs': _cvs_parser,
'file': _file_parser }
-def parse_url(url, pathkey, config, opts):
+def parse_url(url, pathkey, config, opts, file_override = None):
#
# Split the source up into the parts we need.
#
source = {}
source['url'] = url
+ source['options'] = []
colon = url.find(':')
if url[colon + 1:colon + 3] != '//':
raise error.general('malforned URL (no protocol prefix): %s' % (url))
source['path'] = url[:colon + 3] + path.dirname(url[colon + 3:])
- source['file'] = path.basename(url)
+ if file_override is None:
+ source['file'] = path.basename(url)
+ else:
+ bad_chars = [c for c in ['/', '\\', '?', '*'] if c in file_override]
+ if len(bad_chars) > 0:
+ raise error.general('bad characters in file name: %s' % (file_override))
+
+ log.output('download: file-override: %s' % (file_override))
+ source['file'] = file_override
+ source['options'] += ['file-override']
source['name'], source['ext'] = path.splitext(source['file'])
if source['name'].endswith('.tar'):
source['name'] = source['name'][:-4]
@@ -332,7 +347,7 @@ def _http_downloader(url, local, config, opts):
if url.startswith('https://api.github.com'):
url = urllib_parse.urljoin(url, config.expand('tarball/%{version}'))
dst = os.path.relpath(path.host(local))
- log.output('download: %s -> %s' % (url, dst))
+ log.output('download: (full) %s -> %s' % (url, dst))
log.notice('download: %s -> %s' % (_sensible_url(url, len(dst)), dst))
failed = False
if _do_download(opts):
@@ -358,6 +373,7 @@ def _http_downloader(url, local, config, opts):
_ssl_context = ssl._create_unverified_context()
_in = urllib_request.urlopen(_req, context = _ssl_context)
except:
+ log.output('download: no ssl context')
_ssl_context = None
if _ssl_context is None:
_in = urllib_request.urlopen(_req)
More information about the vc
mailing list