diff options
| -rw-r--r-- | bitbake/lib/bb/fetch2/__init__.py | 3 | ||||
| -rw-r--r-- | bitbake/lib/bb/fetch2/perforce.py | 270 |
2 files changed, 150 insertions, 123 deletions
diff --git a/bitbake/lib/bb/fetch2/__init__.py b/bitbake/lib/bb/fetch2/__init__.py index a63498a88b..b6fcaaad55 100644 --- a/bitbake/lib/bb/fetch2/__init__.py +++ b/bitbake/lib/bb/fetch2/__init__.py | |||
| @@ -803,7 +803,8 @@ def runfetchcmd(cmd, d, quiet=False, cleanup=None): | |||
| 803 | 'GIT_SMART_HTTP', | 803 | 'GIT_SMART_HTTP', |
| 804 | 'SSH_AUTH_SOCK', 'SSH_AGENT_PID', | 804 | 'SSH_AUTH_SOCK', 'SSH_AGENT_PID', |
| 805 | 'SOCKS5_USER', 'SOCKS5_PASSWD', | 805 | 'SOCKS5_USER', 'SOCKS5_PASSWD', |
| 806 | 'DBUS_SESSION_BUS_ADDRESS'] | 806 | 'DBUS_SESSION_BUS_ADDRESS', |
| 807 | 'P4CONFIG'] | ||
| 807 | 808 | ||
| 808 | if not cleanup: | 809 | if not cleanup: |
| 809 | cleanup = [] | 810 | cleanup = [] |
diff --git a/bitbake/lib/bb/fetch2/perforce.py b/bitbake/lib/bb/fetch2/perforce.py index ce3cda2670..b8169f2cc9 100644 --- a/bitbake/lib/bb/fetch2/perforce.py +++ b/bitbake/lib/bb/fetch2/perforce.py | |||
| @@ -1,14 +1,12 @@ | |||
| 1 | # ex:ts=4:sw=4:sts=4:et | 1 | # ex:ts=4:sw=4:sts=4:et |
| 2 | # -*- tab-width: 4; c-basic-offset: 4; indent-tabs-mode: nil -*- | 2 | # -*- tab-width: 4; c-basic-offset: 4; indent-tabs-mode: nil -*- |
| 3 | """ | 3 | """ |
| 4 | BitBake 'Fetch' implementations | 4 | BitBake 'Fetch' implementation for perforce |
| 5 | |||
| 6 | Classes for obtaining upstream sources for the | ||
| 7 | BitBake build tools. | ||
| 8 | 5 | ||
| 9 | """ | 6 | """ |
| 10 | 7 | ||
| 11 | # Copyright (C) 2003, 2004 Chris Larson | 8 | # Copyright (C) 2003, 2004 Chris Larson |
| 9 | # Copyright (C) 2016 Kodak Alaris, Inc. | ||
| 12 | # | 10 | # |
| 13 | # This program is free software; you can redistribute it and/or modify | 11 | # This program is free software; you can redistribute it and/or modify |
| 14 | # it under the terms of the GNU General Public License version 2 as | 12 | # it under the terms of the GNU General Public License version 2 as |
| @@ -26,7 +24,6 @@ BitBake build tools. | |||
| 26 | # Based on functions from the base bb module, Copyright 2003 Holger Schurig | 24 | # Based on functions from the base bb module, Copyright 2003 Holger Schurig |
| 27 | 25 | ||
| 28 | import os | 26 | import os |
| 29 | import subprocess | ||
| 30 | import logging | 27 | import logging |
| 31 | import bb | 28 | import bb |
| 32 | from bb import data | 29 | from bb import data |
| @@ -36,151 +33,180 @@ from bb.fetch2 import logger | |||
| 36 | from bb.fetch2 import runfetchcmd | 33 | from bb.fetch2 import runfetchcmd |
| 37 | 34 | ||
| 38 | class Perforce(FetchMethod): | 35 | class Perforce(FetchMethod): |
| 36 | """ Class to fetch from perforce repositories """ | ||
| 39 | def supports(self, ud, d): | 37 | def supports(self, ud, d): |
| 38 | """ Check to see if a given url can be fetched with perforce. """ | ||
| 40 | return ud.type in ['p4'] | 39 | return ud.type in ['p4'] |
| 41 | 40 | ||
| 42 | def doparse(url, d): | 41 | def urldata_init(self, ud, d): |
| 43 | parm = {} | 42 | """ |
| 44 | path = url.split("://")[1] | 43 | Initialize perforce specific variables within url data. If P4CONFIG is |
| 45 | delim = path.find("@"); | 44 | provided by the env, use it. If P4PORT is specified by the recipe, use |
| 45 | its values, which may override the settings in P4CONFIG. | ||
| 46 | """ | ||
| 47 | ud.basecmd = d.getVar('FETCHCMD_p4', True) | ||
| 48 | if not ud.basecmd: | ||
| 49 | ud.basecmd = "/usr/bin/env p4" | ||
| 50 | |||
| 51 | ud.dldir = d.getVar('P4DIR', True) | ||
| 52 | if not ud.dldir: | ||
| 53 | ud.dldir = '%s/%s' % (d.getVar('DL_DIR', True), 'p4') | ||
| 54 | |||
| 55 | path = ud.url.split('://')[1] | ||
| 56 | path = path.split(';')[0] | ||
| 57 | delim = path.find('@'); | ||
| 46 | if delim != -1: | 58 | if delim != -1: |
| 47 | (user, pswd, host, port) = path.split('@')[0].split(":") | 59 | (ud.user, ud.pswd) = path.split('@')[0].split(':') |
| 48 | path = path.split('@')[1] | 60 | ud.path = path.split('@')[1] |
| 49 | else: | 61 | else: |
| 50 | (host, port) = d.getVar('P4PORT', False).split(':') | 62 | ud.path = path |
| 51 | user = "" | ||
| 52 | pswd = "" | ||
| 53 | |||
| 54 | if path.find(";") != -1: | ||
| 55 | keys=[] | ||
| 56 | values=[] | ||
| 57 | plist = path.split(';') | ||
| 58 | for item in plist: | ||
| 59 | if item.count('='): | ||
| 60 | (key, value) = item.split('=') | ||
| 61 | keys.append(key) | ||
| 62 | values.append(value) | ||
| 63 | |||
| 64 | parm = dict(list(zip(keys, values))) | ||
| 65 | path = "//" + path.split(';')[0] | ||
| 66 | host += ":%s" % (port) | ||
| 67 | parm["cset"] = Perforce.getcset(d, path, host, user, pswd, parm) | ||
| 68 | |||
| 69 | return host, path, user, pswd, parm | ||
| 70 | doparse = staticmethod(doparse) | ||
| 71 | |||
| 72 | def getcset(d, depot, host, user, pswd, parm): | ||
| 73 | p4opt = "" | ||
| 74 | if "cset" in parm: | ||
| 75 | return parm["cset"]; | ||
| 76 | if user: | ||
| 77 | p4opt += " -u %s" % (user) | ||
| 78 | if pswd: | ||
| 79 | p4opt += " -P %s" % (pswd) | ||
| 80 | if host: | ||
| 81 | p4opt += " -p %s" % (host) | ||
| 82 | |||
| 83 | p4date = d.getVar("P4DATE", True) | ||
| 84 | if "revision" in parm: | ||
| 85 | depot += "#%s" % (parm["revision"]) | ||
| 86 | elif "label" in parm: | ||
| 87 | depot += "@%s" % (parm["label"]) | ||
| 88 | elif p4date: | ||
| 89 | depot += "@%s" % (p4date) | ||
| 90 | |||
| 91 | p4cmd = d.getVar('FETCHCMD_p4', True) or "p4" | ||
| 92 | logger.debug(1, "Running %s%s changes -m 1 %s", p4cmd, p4opt, depot) | ||
| 93 | p4file, errors = bb.process.run("%s%s changes -m 1 %s" % (p4cmd, p4opt, depot)) | ||
| 94 | cset = p4file.strip() | ||
| 95 | logger.debug(1, "READ %s", cset) | ||
| 96 | if not cset: | ||
| 97 | return -1 | ||
| 98 | |||
| 99 | return cset.split(' ')[1] | ||
| 100 | getcset = staticmethod(getcset) | ||
| 101 | 63 | ||
| 102 | def urldata_init(self, ud, d): | 64 | ud.usingp4config = False |
| 103 | (host, path, user, pswd, parm) = Perforce.doparse(ud.url, d) | 65 | p4port = d.getVar('P4PORT', True) |
| 104 | 66 | ||
| 105 | base_path = path.replace('/...', '') | 67 | if p4port: |
| 106 | base_path = self._strip_leading_slashes(base_path) | 68 | logger.debug(1, 'Using recipe provided P4PORT: %s' % p4port) |
| 107 | 69 | ud.host = p4port | |
| 108 | if "label" in parm: | 70 | else: |
| 109 | version = parm["label"] | 71 | logger.debug(1, 'Trying to use P4CONFIG to automatically set P4PORT...') |
| 72 | ud.usingp4config = True | ||
| 73 | p4cmd = '%s info | grep "Server address"' % ud.basecmd | ||
| 74 | bb.fetch2.check_network_access(d, p4cmd) | ||
| 75 | ud.host = runfetchcmd(p4cmd, d, True) | ||
| 76 | ud.host = ud.host.split(': ')[1].strip() | ||
| 77 | logger.debug(1, 'Determined P4PORT to be: %s' % ud.host) | ||
| 78 | if not ud.host: | ||
| 79 | raise FetchError('Could not determine P4PORT from P4CONFIG') | ||
| 80 | |||
| 81 | if ud.path.find('/...') >= 0: | ||
| 82 | ud.pathisdir = True | ||
| 110 | else: | 83 | else: |
| 111 | version = Perforce.getcset(d, path, host, user, pswd, parm) | 84 | ud.pathisdir = False |
| 112 | 85 | ||
| 113 | ud.localfile = data.expand('%s+%s+%s.tar.gz' % (host, base_path.replace('/', '.'), version), d) | 86 | cleanedpath = ud.path.replace('/...', '').replace('/', '.') |
| 87 | cleanedhost = ud.host.replace(':', '.') | ||
| 88 | ud.pkgdir = os.path.join(ud.dldir, cleanedhost, cleanedpath) | ||
| 114 | 89 | ||
| 115 | def download(self, ud, d): | 90 | ud.setup_revisons(d) |
| 91 | |||
| 92 | ud.localfile = data.expand('%s_%s_%s.tar.gz' % (cleanedhost, cleanedpath, ud.revision), d) | ||
| 93 | |||
| 94 | def _buildp4command(self, ud, d, command, depot_filename=None): | ||
| 116 | """ | 95 | """ |
| 117 | Fetch urls | 96 | Build a p4 commandline. Valid commands are "changes", "print", and |
| 97 | "files". depot_filename is the full path to the file in the depot | ||
| 98 | including the trailing '#rev' value. | ||
| 118 | """ | 99 | """ |
| 100 | p4opt = "" | ||
| 101 | |||
| 102 | if ud.user: | ||
| 103 | p4opt += ' -u "%s"' % (ud.user) | ||
| 119 | 104 | ||
| 120 | (host, depot, user, pswd, parm) = Perforce.doparse(ud.url, d) | 105 | if ud.pswd: |
| 106 | p4opt += ' -P "%s"' % (ud.pswd) | ||
| 121 | 107 | ||
| 122 | if depot.find('/...') != -1: | 108 | if ud.host and not ud.usingp4config: |
| 123 | path = depot[:depot.find('/...')] | 109 | p4opt += ' -p %s' % (ud.host) |
| 110 | |||
| 111 | if hasattr(ud, 'revision') and ud.revision: | ||
| 112 | pathnrev = '%s@%s' % (ud.path, ud.revision) | ||
| 113 | else: | ||
| 114 | pathnrev = '%s' % (ud.path) | ||
| 115 | |||
| 116 | if depot_filename: | ||
| 117 | if ud.pathisdir: # Remove leading path to obtain filename | ||
| 118 | filename = depot_filename[len(ud.path)-1:] | ||
| 119 | else: | ||
| 120 | filename = depot_filename[depot_filename.rfind('/'):] | ||
| 121 | filename = filename[:filename.find('#')] # Remove trailing '#rev' | ||
| 122 | |||
| 123 | if command == 'changes': | ||
| 124 | p4cmd = '%s%s changes -m 1 //%s' % (ud.basecmd, p4opt, pathnrev) | ||
| 125 | elif command == 'print': | ||
| 126 | if depot_filename != None: | ||
| 127 | p4cmd = '%s%s print -o "p4/%s" "%s"' % (ud.basecmd, p4opt, filename, depot_filename) | ||
| 128 | else: | ||
| 129 | raise FetchError('No depot file name provided to p4 %s' % command, ud.url) | ||
| 130 | elif command == 'files': | ||
| 131 | p4cmd = '%s%s files //%s' % (ud.basecmd, p4opt, pathnrev) | ||
| 124 | else: | 132 | else: |
| 125 | path = depot[:depot.rfind('/')] | 133 | raise FetchError('Invalid p4 command %s' % command, ud.url) |
| 126 | 134 | ||
| 127 | module = parm.get('module', os.path.basename(path)) | 135 | return p4cmd |
| 128 | 136 | ||
| 129 | # Get the p4 command | 137 | def _p4listfiles(self, ud, d): |
| 130 | p4opt = "" | 138 | """ |
| 131 | if user: | 139 | Return a list of the file names which are present in the depot using the |
| 132 | p4opt += " -u %s" % (user) | 140 | 'p4 files' command, including trailing '#rev' file revision indicator |
| 141 | """ | ||
| 142 | p4cmd = self._buildp4command(ud, d, 'files') | ||
| 143 | bb.fetch2.check_network_access(d, p4cmd) | ||
| 144 | p4fileslist = runfetchcmd(p4cmd, d, True) | ||
| 145 | p4fileslist = [f.rstrip() for f in p4fileslist.splitlines()] | ||
| 146 | |||
| 147 | if not p4fileslist: | ||
| 148 | raise FetchError('Unable to fetch listing of p4 files from %s@%s' % (ud.host, ud.path)) | ||
| 149 | |||
| 150 | count = 0 | ||
| 151 | filelist = [] | ||
| 133 | 152 | ||
| 134 | if pswd: | 153 | for filename in p4fileslist: |
| 135 | p4opt += " -P %s" % (pswd) | 154 | item = filename.split(' - ') |
| 155 | lastaction = item[1].split() | ||
| 156 | logger.debug(1, 'File: %s Last Action: %s' % (item[0], lastaction[0])) | ||
| 157 | if lastaction[0] == 'delete': | ||
| 158 | continue | ||
| 159 | filelist.append(item[0]) | ||
| 136 | 160 | ||
| 137 | if host: | 161 | return filelist |
| 138 | p4opt += " -p %s" % (host) | ||
| 139 | 162 | ||
| 140 | p4cmd = d.getVar('FETCHCMD_p4', True) or "p4" | 163 | def download(self, ud, d): |
| 164 | """ Get the list of files, fetch each one """ | ||
| 165 | filelist = self._p4listfiles(ud, d) | ||
| 166 | if not filelist: | ||
| 167 | raise FetchError('No files found in depot %s@%s' % (ud.host, ud.path)) | ||
| 141 | 168 | ||
| 142 | # create temp directory | 169 | bb.utils.remove(ud.pkgdir, True) |
| 143 | logger.debug(2, "Fetch: creating temporary directory") | 170 | bb.utils.mkdirhier(ud.pkgdir) |
| 144 | bb.utils.mkdirhier(d.expand('${WORKDIR}')) | 171 | os.chdir(ud.pkgdir) |
| 145 | mktemp = d.getVar("FETCHCMD_p4mktemp", True) or d.expand("mktemp -d -q '${WORKDIR}/oep4.XXXXXX'") | ||
| 146 | tmpfile, errors = bb.process.run(mktemp) | ||
| 147 | tmpfile = tmpfile.strip() | ||
| 148 | if not tmpfile: | ||
| 149 | raise FetchError("Fetch: unable to create temporary directory.. make sure 'mktemp' is in the PATH.", ud.url) | ||
| 150 | 172 | ||
| 151 | if "label" in parm: | 173 | for afile in filelist: |
| 152 | depot = "%s@%s" % (depot, parm["label"]) | 174 | p4fetchcmd = self._buildp4command(ud, d, 'print', afile) |
| 153 | else: | 175 | bb.fetch2.check_network_access(d, p4fetchcmd) |
| 154 | cset = Perforce.getcset(d, depot, host, user, pswd, parm) | 176 | runfetchcmd(p4fetchcmd, d) |
| 155 | depot = "%s@%s" % (depot, cset) | ||
| 156 | 177 | ||
| 157 | os.chdir(tmpfile) | 178 | os.chdir(ud.pkgdir) |
| 158 | logger.info("Fetch " + ud.url) | 179 | runfetchcmd('tar -czf %s p4' % (ud.localpath), d, cleanup = [ud.localpath]) |
| 159 | logger.info("%s%s files %s", p4cmd, p4opt, depot) | ||
| 160 | p4file, errors = bb.process.run("%s%s files %s" % (p4cmd, p4opt, depot)) | ||
| 161 | p4file = [f.rstrip() for f in p4file.splitlines()] | ||
| 162 | 180 | ||
| 163 | if not p4file: | 181 | def clean(self, ud, d): |
| 164 | raise FetchError("Fetch: unable to get the P4 files from %s" % depot, ud.url) | 182 | """ Cleanup p4 specific files and dirs""" |
| 183 | bb.utils.remove(ud.localpath) | ||
| 184 | bb.utils.remove(ud.pkgdir, True) | ||
| 165 | 185 | ||
| 166 | count = 0 | 186 | def supports_srcrev(self): |
| 187 | return True | ||
| 167 | 188 | ||
| 168 | for file in p4file: | 189 | def _revision_key(self, ud, d, name): |
| 169 | list = file.split() | 190 | """ Return a unique key for the url """ |
| 191 | return 'p4:%s' % ud.pkgdir | ||
| 170 | 192 | ||
| 171 | if list[2] == "delete": | 193 | def _latest_revision(self, ud, d, name): |
| 172 | continue | 194 | """ Return the latest upstream scm revision number """ |
| 195 | p4cmd = self._buildp4command(ud, d, "changes") | ||
| 196 | bb.fetch2.check_network_access(d, p4cmd) | ||
| 197 | tip = runfetchcmd(p4cmd, d, True) | ||
| 198 | |||
| 199 | if not tip: | ||
| 200 | raise FetchError('Could not determine the latest perforce changelist') | ||
| 173 | 201 | ||
| 174 | dest = list[0][len(path)+1:] | 202 | tipcset = tip.split(' ')[1] |
| 175 | where = dest.find("#") | 203 | logger.debug(1, 'p4 tip found to be changelist %s' % tipcset) |
| 204 | return tipcset | ||
| 176 | 205 | ||
| 177 | subprocess.call("%s%s print -o %s/%s %s" % (p4cmd, p4opt, module, dest[:where], list[0]), shell=True) | 206 | def sortable_revision(self, ud, d, name): |
| 178 | count = count + 1 | 207 | """ Return a sortable revision number """ |
| 208 | return False, self._build_revision(ud, d) | ||
| 179 | 209 | ||
| 180 | if count == 0: | 210 | def _build_revision(self, ud, d): |
| 181 | logger.error() | 211 | return ud.revision |
| 182 | raise FetchError("Fetch: No files gathered from the P4 fetch", ud.url) | ||
| 183 | 212 | ||
| 184 | runfetchcmd("tar -czf %s %s" % (ud.localpath, module), d, cleanup = [ud.localpath]) | ||
| 185 | # cleanup | ||
| 186 | bb.utils.prunedir(tmpfile) | ||
