upload: Support uploading to Gerrit over https://

If SSH is not available, Gerrit returns NOT_AVAILABLE to the /ssh_info
query made by repo upload. In this case fallback to the /p/$PROJECT URL
that Gerrit also exports and use that for uploads.

Change-Id: I1e3e39ab709ecc0a692614a41a42446426f39c08
This commit is contained in:
Shawn O. Pearce 2012-01-11 14:58:54 -08:00
parent 34fb20f67c
commit c9571423f8
2 changed files with 42 additions and 64 deletions

View File

@ -527,7 +527,7 @@ class Remote(object):
self.projectname = self._Get('projectname') self.projectname = self._Get('projectname')
self.fetch = map(lambda x: RefSpec.FromString(x), self.fetch = map(lambda x: RefSpec.FromString(x),
self._Get('fetch', all=True)) self._Get('fetch', all=True))
self._review_protocol = None self._review_url = None
def _InsteadOf(self): def _InsteadOf(self):
globCfg = GitConfig.ForUser() globCfg = GitConfig.ForUser()
@ -554,9 +554,8 @@ class Remote(object):
connectionUrl = self._InsteadOf() connectionUrl = self._InsteadOf()
return _preconnect(connectionUrl) return _preconnect(connectionUrl)
@property def ReviewUrl(self, userEmail):
def ReviewProtocol(self): if self._review_url is None:
if self._review_protocol is None:
if self.review is None: if self.review is None:
return None return None
@ -565,67 +564,47 @@ class Remote(object):
u = 'http://%s' % u u = 'http://%s' % u
if u.endswith('/Gerrit'): if u.endswith('/Gerrit'):
u = u[:len(u) - len('/Gerrit')] u = u[:len(u) - len('/Gerrit')]
if not u.endswith('/ssh_info'): if u.endswith('/ssh_info'):
u = u[:len(u) - len('/ssh_info')]
if not u.endswith('/'): if not u.endswith('/'):
u += '/' u += '/'
u += 'ssh_info' http_url = u
if u in REVIEW_CACHE: if u in REVIEW_CACHE:
info = REVIEW_CACHE[u] self._review_url = REVIEW_CACHE[u]
self._review_protocol = info[0]
self._review_host = info[1]
self._review_port = info[2]
elif 'REPO_HOST_PORT_INFO' in os.environ: elif 'REPO_HOST_PORT_INFO' in os.environ:
info = os.environ['REPO_HOST_PORT_INFO'] host, port = os.environ['REPO_HOST_PORT_INFO'].split()
self._review_protocol = 'ssh' self._review_url = self._SshReviewUrl(userEmail, host, port)
self._review_host = info.split(" ")[0] REVIEW_CACHE[u] = self._review_url
self._review_port = info.split(" ")[1]
REVIEW_CACHE[u] = (
self._review_protocol,
self._review_host,
self._review_port)
else: else:
try: try:
info = urllib2.urlopen(u).read() info_url = u + 'ssh_info'
if info == 'NOT_AVAILABLE': info = urllib2.urlopen(info_url).read()
raise UploadError('%s: SSH disabled' % self.review)
if '<' in info: if '<' in info:
# Assume the server gave us some sort of HTML # Assume the server gave us some sort of HTML
# response back, like maybe a login page. # response back, like maybe a login page.
# #
raise UploadError('%s: Cannot parse response' % u) raise UploadError('%s: Cannot parse response' % info_url)
self._review_protocol = 'ssh' if info == 'NOT_AVAILABLE':
self._review_host = info.split(" ")[0] # Assume HTTP if SSH is not enabled.
self._review_port = info.split(" ")[1] self._review_url = http_url + 'p/'
except urllib2.HTTPError, e:
if e.code == 404:
self._review_protocol = 'http-post'
self._review_host = None
self._review_port = None
else: else:
raise UploadError('Upload over SSH unavailable') host, port = info.split()
self._review_url = self._SshReviewUrl(userEmail, host, port)
except urllib2.HTTPError, e:
raise UploadError('%s: %s' % (self.review, str(e)))
except urllib2.URLError, e: except urllib2.URLError, e:
raise UploadError('%s: %s' % (self.review, str(e))) raise UploadError('%s: %s' % (self.review, str(e)))
REVIEW_CACHE[u] = ( REVIEW_CACHE[u] = self._review_url
self._review_protocol, return self._review_url + self.projectname
self._review_host,
self._review_port)
return self._review_protocol
def SshReviewUrl(self, userEmail): def _SshReviewUrl(self, userEmail, host, port):
if self.ReviewProtocol != 'ssh':
return None
username = self._config.GetString('review.%s.username' % self.review) username = self._config.GetString('review.%s.username' % self.review)
if username is None: if username is None:
username = userEmail.split("@")[0] username = userEmail.split('@')[0]
return 'ssh://%s@%s:%s/%s' % ( return 'ssh://%s@%s:%s/' % (username, host, port)
username,
self._review_host,
self._review_port,
self.projectname)
def ToLocal(self, rev): def ToLocal(self, rev):
"""Convert a remote revision string to something we have locally. """Convert a remote revision string to something we have locally.

View File

@ -866,32 +866,31 @@ class Project(object):
branch.remote.projectname = self.name branch.remote.projectname = self.name
branch.remote.Save() branch.remote.Save()
if branch.remote.ReviewProtocol == 'ssh': url = branch.remote.ReviewUrl(self.UserEmail)
if dest_branch.startswith(R_HEADS): if url is None:
dest_branch = dest_branch[len(R_HEADS):] raise UploadError('review not configured')
cmd = ['push']
if url.startswith('ssh://'):
rp = ['gerrit receive-pack'] rp = ['gerrit receive-pack']
for e in people[0]: for e in people[0]:
rp.append('--reviewer=%s' % sq(e)) rp.append('--reviewer=%s' % sq(e))
for e in people[1]: for e in people[1]:
rp.append('--cc=%s' % sq(e)) rp.append('--cc=%s' % sq(e))
cmd.append('--receive-pack=%s' % " ".join(rp))
cmd.append(url)
if dest_branch.startswith(R_HEADS):
dest_branch = dest_branch[len(R_HEADS):]
ref_spec = '%s:refs/for/%s' % (R_HEADS + branch.name, dest_branch) ref_spec = '%s:refs/for/%s' % (R_HEADS + branch.name, dest_branch)
if auto_topic: if auto_topic:
ref_spec = ref_spec + '/' + branch.name ref_spec = ref_spec + '/' + branch.name
cmd = ['push']
cmd.append('--receive-pack=%s' % " ".join(rp))
cmd.append(branch.remote.SshReviewUrl(self.UserEmail))
cmd.append(ref_spec) cmd.append(ref_spec)
if GitCommand(self, cmd, bare = True).Wait() != 0: if GitCommand(self, cmd, bare = True).Wait() != 0:
raise UploadError('Upload failed') raise UploadError('Upload failed')
else:
raise UploadError('Unsupported protocol %s' \
% branch.remote.review)
msg = "posted to %s for %s" % (branch.remote.review, dest_branch) msg = "posted to %s for %s" % (branch.remote.review, dest_branch)
self.bare_git.UpdateRef(R_PUB + branch.name, self.bare_git.UpdateRef(R_PUB + branch.name,
R_HEADS + branch.name, R_HEADS + branch.name,