mirror of
https://gerrit.googlesource.com/git-repo
synced 2025-06-30 20:17:08 +00:00
Compare commits
10 Commits
Author | SHA1 | Date | |
---|---|---|---|
6623b21e10 | |||
ca8c32cd7a | |||
f0a9a1a30e | |||
879a9a5cf0 | |||
ff6929dde8 | |||
1c85f4e43b | |||
719965af35 | |||
5732e47ebb | |||
f3fdf823cf | |||
a1bfd2cd72 |
@ -22,6 +22,7 @@ following DTD:
|
|||||||
<!DOCTYPE manifest [
|
<!DOCTYPE manifest [
|
||||||
<!ELEMENT manifest (remote*,
|
<!ELEMENT manifest (remote*,
|
||||||
default?,
|
default?,
|
||||||
|
manifest-server?,
|
||||||
remove-project*,
|
remove-project*,
|
||||||
project*)>
|
project*)>
|
||||||
|
|
||||||
@ -33,6 +34,9 @@ following DTD:
|
|||||||
<!ELEMENT default (EMPTY)>
|
<!ELEMENT default (EMPTY)>
|
||||||
<!ATTLIST default remote IDREF #IMPLIED>
|
<!ATTLIST default remote IDREF #IMPLIED>
|
||||||
<!ATTLIST default revision CDATA #IMPLIED>
|
<!ATTLIST default revision CDATA #IMPLIED>
|
||||||
|
|
||||||
|
<!ELEMENT manifest-server (EMPTY)>
|
||||||
|
<!ATTLIST url CDATA #REQUIRED>
|
||||||
|
|
||||||
<!ELEMENT project (EMPTY)>
|
<!ELEMENT project (EMPTY)>
|
||||||
<!ATTLIST project name CDATA #REQUIRED>
|
<!ATTLIST project name CDATA #REQUIRED>
|
||||||
@ -89,6 +93,27 @@ Attribute `revision`: Name of a Git branch (e.g. `master` or
|
|||||||
revision attribute will use this revision.
|
revision attribute will use this revision.
|
||||||
|
|
||||||
|
|
||||||
|
Element manifest-server
|
||||||
|
-----------------------
|
||||||
|
|
||||||
|
At most one manifest-server may be specified. The url attribute
|
||||||
|
is used to specify the URL of a manifest server, which is an
|
||||||
|
XML RPC service that will return a manifest in which each project
|
||||||
|
is pegged to a known good revision for the current branch and
|
||||||
|
target.
|
||||||
|
|
||||||
|
The manifest server should implement:
|
||||||
|
|
||||||
|
GetApprovedManifest(branch, target)
|
||||||
|
|
||||||
|
The target to use is defined by environment variables TARGET_PRODUCT
|
||||||
|
and TARGET_BUILD_VARIANT. These variables are used to create a string
|
||||||
|
of the form $TARGET_PRODUCT-$TARGET_BUILD_VARIANT, e.g. passion-userdebug.
|
||||||
|
If one of those variables or both are not present, the program will call
|
||||||
|
GetApprovedManifest without the target paramater and the manifest server
|
||||||
|
should choose a reasonable default target.
|
||||||
|
|
||||||
|
|
||||||
Element project
|
Element project
|
||||||
---------------
|
---------------
|
||||||
|
|
||||||
|
@ -17,6 +17,7 @@ import os
|
|||||||
import sys
|
import sys
|
||||||
import subprocess
|
import subprocess
|
||||||
import tempfile
|
import tempfile
|
||||||
|
from signal import SIGTERM
|
||||||
from error import GitError
|
from error import GitError
|
||||||
from trace import REPO_TRACE, IsTrace, Trace
|
from trace import REPO_TRACE, IsTrace, Trace
|
||||||
|
|
||||||
@ -29,8 +30,9 @@ LAST_CWD = None
|
|||||||
|
|
||||||
_ssh_proxy_path = None
|
_ssh_proxy_path = None
|
||||||
_ssh_sock_path = None
|
_ssh_sock_path = None
|
||||||
|
_ssh_clients = []
|
||||||
|
|
||||||
def _ssh_sock(create=True):
|
def ssh_sock(create=True):
|
||||||
global _ssh_sock_path
|
global _ssh_sock_path
|
||||||
if _ssh_sock_path is None:
|
if _ssh_sock_path is None:
|
||||||
if not create:
|
if not create:
|
||||||
@ -51,6 +53,24 @@ def _ssh_proxy():
|
|||||||
'git_ssh')
|
'git_ssh')
|
||||||
return _ssh_proxy_path
|
return _ssh_proxy_path
|
||||||
|
|
||||||
|
def _add_ssh_client(p):
|
||||||
|
_ssh_clients.append(p)
|
||||||
|
|
||||||
|
def _remove_ssh_client(p):
|
||||||
|
try:
|
||||||
|
_ssh_clients.remove(p)
|
||||||
|
except ValueError:
|
||||||
|
pass
|
||||||
|
|
||||||
|
def terminate_ssh_clients():
|
||||||
|
global _ssh_clients
|
||||||
|
for p in _ssh_clients:
|
||||||
|
try:
|
||||||
|
os.kill(p.pid, SIGTERM)
|
||||||
|
p.wait()
|
||||||
|
except OSError:
|
||||||
|
pass
|
||||||
|
_ssh_clients = []
|
||||||
|
|
||||||
class _GitCall(object):
|
class _GitCall(object):
|
||||||
def version(self):
|
def version(self):
|
||||||
@ -119,7 +139,7 @@ class GitCommand(object):
|
|||||||
if disable_editor:
|
if disable_editor:
|
||||||
env['GIT_EDITOR'] = ':'
|
env['GIT_EDITOR'] = ':'
|
||||||
if ssh_proxy:
|
if ssh_proxy:
|
||||||
env['REPO_SSH_SOCK'] = _ssh_sock()
|
env['REPO_SSH_SOCK'] = ssh_sock()
|
||||||
env['GIT_SSH'] = _ssh_proxy()
|
env['GIT_SSH'] = _ssh_proxy()
|
||||||
|
|
||||||
if project:
|
if project:
|
||||||
@ -188,6 +208,9 @@ class GitCommand(object):
|
|||||||
except Exception, e:
|
except Exception, e:
|
||||||
raise GitError('%s: %s' % (command[1], e))
|
raise GitError('%s: %s' % (command[1], e))
|
||||||
|
|
||||||
|
if ssh_proxy:
|
||||||
|
_add_ssh_client(p)
|
||||||
|
|
||||||
self.process = p
|
self.process = p
|
||||||
self.stdin = p.stdin
|
self.stdin = p.stdin
|
||||||
|
|
||||||
@ -210,4 +233,8 @@ class GitCommand(object):
|
|||||||
else:
|
else:
|
||||||
p.stderr = None
|
p.stderr = None
|
||||||
|
|
||||||
return self.process.wait()
|
try:
|
||||||
|
rc = p.wait()
|
||||||
|
finally:
|
||||||
|
_remove_ssh_client(p)
|
||||||
|
return rc
|
||||||
|
@ -23,7 +23,10 @@ from signal import SIGTERM
|
|||||||
from urllib2 import urlopen, HTTPError
|
from urllib2 import urlopen, HTTPError
|
||||||
from error import GitError, UploadError
|
from error import GitError, UploadError
|
||||||
from trace import Trace
|
from trace import Trace
|
||||||
from git_command import GitCommand, _ssh_sock
|
|
||||||
|
from git_command import GitCommand
|
||||||
|
from git_command import ssh_sock
|
||||||
|
from git_command import terminate_ssh_clients
|
||||||
|
|
||||||
R_HEADS = 'refs/heads/'
|
R_HEADS = 'refs/heads/'
|
||||||
R_TAGS = 'refs/tags/'
|
R_TAGS = 'refs/tags/'
|
||||||
@ -371,7 +374,7 @@ def _open_ssh(host, port):
|
|||||||
return False
|
return False
|
||||||
|
|
||||||
command = ['ssh',
|
command = ['ssh',
|
||||||
'-o','ControlPath %s' % _ssh_sock(),
|
'-o','ControlPath %s' % ssh_sock(),
|
||||||
'-p',str(port),
|
'-p',str(port),
|
||||||
'-M',
|
'-M',
|
||||||
'-N',
|
'-N',
|
||||||
@ -391,6 +394,8 @@ def _open_ssh(host, port):
|
|||||||
return True
|
return True
|
||||||
|
|
||||||
def close_ssh():
|
def close_ssh():
|
||||||
|
terminate_ssh_clients()
|
||||||
|
|
||||||
for key,p in _ssh_cache.iteritems():
|
for key,p in _ssh_cache.iteritems():
|
||||||
try:
|
try:
|
||||||
os.kill(p.pid, SIGTERM)
|
os.kill(p.pid, SIGTERM)
|
||||||
@ -399,7 +404,7 @@ def close_ssh():
|
|||||||
pass
|
pass
|
||||||
_ssh_cache.clear()
|
_ssh_cache.clear()
|
||||||
|
|
||||||
d = _ssh_sock(create=False)
|
d = ssh_sock(create=False)
|
||||||
if d:
|
if d:
|
||||||
try:
|
try:
|
||||||
os.rmdir(os.path.dirname(d))
|
os.rmdir(os.path.dirname(d))
|
||||||
|
@ -65,8 +65,8 @@ class XmlManifest(object):
|
|||||||
|
|
||||||
self._Unload()
|
self._Unload()
|
||||||
|
|
||||||
def Link(self, name):
|
def Override(self, name):
|
||||||
"""Update the repo metadata to use a different manifest.
|
"""Use a different manifest, just for the current instantiation.
|
||||||
"""
|
"""
|
||||||
path = os.path.join(self.manifestProject.worktree, name)
|
path = os.path.join(self.manifestProject.worktree, name)
|
||||||
if not os.path.isfile(path):
|
if not os.path.isfile(path):
|
||||||
@ -80,6 +80,11 @@ class XmlManifest(object):
|
|||||||
finally:
|
finally:
|
||||||
self.manifestFile = old
|
self.manifestFile = old
|
||||||
|
|
||||||
|
def Link(self, name):
|
||||||
|
"""Update the repo metadata to use a different manifest.
|
||||||
|
"""
|
||||||
|
self.Override(name)
|
||||||
|
|
||||||
try:
|
try:
|
||||||
if os.path.exists(self.manifestFile):
|
if os.path.exists(self.manifestFile):
|
||||||
os.remove(self.manifestFile)
|
os.remove(self.manifestFile)
|
||||||
@ -123,6 +128,12 @@ class XmlManifest(object):
|
|||||||
root.appendChild(e)
|
root.appendChild(e)
|
||||||
root.appendChild(doc.createTextNode(''))
|
root.appendChild(doc.createTextNode(''))
|
||||||
|
|
||||||
|
if self._manifest_server:
|
||||||
|
e = doc.createElement('manifest-server')
|
||||||
|
e.setAttribute('url', self._manifest_server)
|
||||||
|
root.appendChild(e)
|
||||||
|
root.appendChild(doc.createTextNode(''))
|
||||||
|
|
||||||
sort_projects = list(self.projects.keys())
|
sort_projects = list(self.projects.keys())
|
||||||
sort_projects.sort()
|
sort_projects.sort()
|
||||||
|
|
||||||
@ -168,6 +179,11 @@ class XmlManifest(object):
|
|||||||
self._Load()
|
self._Load()
|
||||||
return self._default
|
return self._default
|
||||||
|
|
||||||
|
@property
|
||||||
|
def manifest_server(self):
|
||||||
|
self._Load()
|
||||||
|
return self._manifest_server
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def IsMirror(self):
|
def IsMirror(self):
|
||||||
return self.manifestProject.config.GetBoolean('repo.mirror')
|
return self.manifestProject.config.GetBoolean('repo.mirror')
|
||||||
@ -178,6 +194,7 @@ class XmlManifest(object):
|
|||||||
self._remotes = {}
|
self._remotes = {}
|
||||||
self._default = None
|
self._default = None
|
||||||
self.branch = None
|
self.branch = None
|
||||||
|
self._manifest_server = None
|
||||||
|
|
||||||
def _Load(self):
|
def _Load(self):
|
||||||
if not self._loaded:
|
if not self._loaded:
|
||||||
@ -246,6 +263,15 @@ class XmlManifest(object):
|
|||||||
if self._default is None:
|
if self._default is None:
|
||||||
self._default = _Default()
|
self._default = _Default()
|
||||||
|
|
||||||
|
for node in config.childNodes:
|
||||||
|
if node.nodeName == 'manifest-server':
|
||||||
|
url = self._reqatt(node, 'url')
|
||||||
|
if self._manifest_server is not None:
|
||||||
|
raise ManifestParseError, \
|
||||||
|
'duplicate manifest-server in %s' % \
|
||||||
|
(self.manifestFile)
|
||||||
|
self._manifest_server = url
|
||||||
|
|
||||||
for node in config.childNodes:
|
for node in config.childNodes:
|
||||||
if node.nodeName == 'project':
|
if node.nodeName == 'project':
|
||||||
project = self._ParseProject(node)
|
project = self._ParseProject(node)
|
||||||
@ -315,7 +341,7 @@ class XmlManifest(object):
|
|||||||
def _ParseProject(self, node):
|
def _ParseProject(self, node):
|
||||||
"""
|
"""
|
||||||
reads a <project> element from the manifest file
|
reads a <project> element from the manifest file
|
||||||
"""
|
"""
|
||||||
name = self._reqatt(node, 'name')
|
name = self._reqatt(node, 'name')
|
||||||
|
|
||||||
remote = self._get_remote(node)
|
remote = self._get_remote(node)
|
||||||
|
@ -136,7 +136,7 @@ is shown, then the branch appears in all projects.
|
|||||||
hdr('%c%c %-*s' % (current, published, width, name))
|
hdr('%c%c %-*s' % (current, published, width, name))
|
||||||
out.write(' |')
|
out.write(' |')
|
||||||
|
|
||||||
if in_cnt < project_cnt and (in_cnt == 1):
|
if in_cnt < project_cnt:
|
||||||
fmt = out.write
|
fmt = out.write
|
||||||
paths = []
|
paths = []
|
||||||
if in_cnt < project_cnt - in_cnt:
|
if in_cnt < project_cnt - in_cnt:
|
||||||
@ -150,15 +150,17 @@ is shown, then the branch appears in all projects.
|
|||||||
for b in i.projects:
|
for b in i.projects:
|
||||||
have.add(b.project)
|
have.add(b.project)
|
||||||
for p in projects:
|
for p in projects:
|
||||||
paths.append(p.relpath)
|
if not p in have:
|
||||||
|
paths.append(p.relpath)
|
||||||
|
|
||||||
s = ' %s %s' % (type, ', '.join(paths))
|
s = ' %s %s' % (type, ', '.join(paths))
|
||||||
if width + 7 + len(s) < 80:
|
if width + 7 + len(s) < 80:
|
||||||
fmt(s)
|
fmt(s)
|
||||||
else:
|
else:
|
||||||
out.nl()
|
fmt(' %s:' % type)
|
||||||
fmt(' %s:' % type)
|
|
||||||
for p in paths:
|
for p in paths:
|
||||||
out.nl()
|
out.nl()
|
||||||
fmt(' %s' % p)
|
fmt(width*' ' + ' %s' % p)
|
||||||
|
else:
|
||||||
|
out.write(' in all projects')
|
||||||
out.nl()
|
out.nl()
|
||||||
|
33
subcmds/smartsync.py
Normal file
33
subcmds/smartsync.py
Normal file
@ -0,0 +1,33 @@
|
|||||||
|
#
|
||||||
|
# Copyright (C) 2010 The Android Open Source Project
|
||||||
|
#
|
||||||
|
# Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
# you may not use this file except in compliance with the License.
|
||||||
|
# You may obtain a copy of the License at
|
||||||
|
#
|
||||||
|
# http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
#
|
||||||
|
# Unless required by applicable law or agreed to in writing, software
|
||||||
|
# distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
# See the License for the specific language governing permissions and
|
||||||
|
# limitations under the License.
|
||||||
|
|
||||||
|
from sync import Sync
|
||||||
|
|
||||||
|
class Smartsync(Sync):
|
||||||
|
common = True
|
||||||
|
helpSummary = "Update working tree to the latest known good revision"
|
||||||
|
helpUsage = """
|
||||||
|
%prog [<project>...]
|
||||||
|
"""
|
||||||
|
helpDescription = """
|
||||||
|
The '%prog' command is a shortcut for sync -s.
|
||||||
|
"""
|
||||||
|
|
||||||
|
def _Options(self, p):
|
||||||
|
Sync._Options(self, p, show_smart=False)
|
||||||
|
|
||||||
|
def Execute(self, opt, args):
|
||||||
|
opt.smart_sync = True
|
||||||
|
Sync.Execute(self, opt, args)
|
112
subcmds/sync.py
112
subcmds/sync.py
@ -17,11 +17,14 @@ from optparse import SUPPRESS_HELP
|
|||||||
import os
|
import os
|
||||||
import re
|
import re
|
||||||
import shutil
|
import shutil
|
||||||
|
import socket
|
||||||
import subprocess
|
import subprocess
|
||||||
import sys
|
import sys
|
||||||
import time
|
import time
|
||||||
|
import xmlrpclib
|
||||||
|
|
||||||
from git_command import GIT
|
from git_command import GIT
|
||||||
|
from git_refs import R_HEADS
|
||||||
from project import HEAD
|
from project import HEAD
|
||||||
from project import Project
|
from project import Project
|
||||||
from project import RemoteSpec
|
from project import RemoteSpec
|
||||||
@ -57,6 +60,10 @@ back to the manifest revision. This option is especially helpful
|
|||||||
if the project is currently on a topic branch, but the manifest
|
if the project is currently on a topic branch, but the manifest
|
||||||
revision is temporarily needed.
|
revision is temporarily needed.
|
||||||
|
|
||||||
|
The -s/--smart-sync option can be used to sync to a known good
|
||||||
|
build as specified by the manifest-server element in the current
|
||||||
|
manifest.
|
||||||
|
|
||||||
SSH Connections
|
SSH Connections
|
||||||
---------------
|
---------------
|
||||||
|
|
||||||
@ -87,7 +94,7 @@ later is required to fix a server side protocol bug.
|
|||||||
|
|
||||||
"""
|
"""
|
||||||
|
|
||||||
def _Options(self, p):
|
def _Options(self, p, show_smart=True):
|
||||||
p.add_option('-l','--local-only',
|
p.add_option('-l','--local-only',
|
||||||
dest='local_only', action='store_true',
|
dest='local_only', action='store_true',
|
||||||
help="only update working tree, don't fetch")
|
help="only update working tree, don't fetch")
|
||||||
@ -97,6 +104,10 @@ later is required to fix a server side protocol bug.
|
|||||||
p.add_option('-d','--detach',
|
p.add_option('-d','--detach',
|
||||||
dest='detach_head', action='store_true',
|
dest='detach_head', action='store_true',
|
||||||
help='detach projects back to manifest revision')
|
help='detach projects back to manifest revision')
|
||||||
|
if show_smart:
|
||||||
|
p.add_option('-s', '--smart-sync',
|
||||||
|
dest='smart_sync', action='store_true',
|
||||||
|
help='smart sync using manifest from a known good build')
|
||||||
|
|
||||||
g = p.add_option_group('repo Version options')
|
g = p.add_option_group('repo Version options')
|
||||||
g.add_option('--no-repo-verify',
|
g.add_option('--no-repo-verify',
|
||||||
@ -138,32 +149,36 @@ later is required to fix a server side protocol bug.
|
|||||||
if not path:
|
if not path:
|
||||||
continue
|
continue
|
||||||
if path not in new_project_paths:
|
if path not in new_project_paths:
|
||||||
project = Project(
|
"""If the path has already been deleted, we don't need to do it
|
||||||
manifest = self.manifest,
|
"""
|
||||||
name = path,
|
if os.path.exists(self.manifest.topdir + '/' + path):
|
||||||
remote = RemoteSpec('origin'),
|
project = Project(
|
||||||
gitdir = os.path.join(self.manifest.topdir,
|
manifest = self.manifest,
|
||||||
path, '.git'),
|
name = path,
|
||||||
worktree = os.path.join(self.manifest.topdir, path),
|
remote = RemoteSpec('origin'),
|
||||||
relpath = path,
|
gitdir = os.path.join(self.manifest.topdir,
|
||||||
revisionExpr = 'HEAD',
|
path, '.git'),
|
||||||
revisionId = None)
|
worktree = os.path.join(self.manifest.topdir, path),
|
||||||
if project.IsDirty():
|
relpath = path,
|
||||||
print >>sys.stderr, 'error: Cannot remove project "%s": \
|
revisionExpr = 'HEAD',
|
||||||
|
revisionId = None)
|
||||||
|
|
||||||
|
if project.IsDirty():
|
||||||
|
print >>sys.stderr, 'error: Cannot remove project "%s": \
|
||||||
uncommitted changes are present' % project.relpath
|
uncommitted changes are present' % project.relpath
|
||||||
print >>sys.stderr, ' commit changes, then run sync again'
|
print >>sys.stderr, ' commit changes, then run sync again'
|
||||||
return -1
|
return -1
|
||||||
else:
|
else:
|
||||||
print >>sys.stderr, 'Deleting obsolete path %s' % project.worktree
|
print >>sys.stderr, 'Deleting obsolete path %s' % project.worktree
|
||||||
shutil.rmtree(project.worktree)
|
shutil.rmtree(project.worktree)
|
||||||
# Try deleting parent subdirs if they are empty
|
# Try deleting parent subdirs if they are empty
|
||||||
dir = os.path.dirname(project.worktree)
|
dir = os.path.dirname(project.worktree)
|
||||||
while dir != self.manifest.topdir:
|
while dir != self.manifest.topdir:
|
||||||
try:
|
try:
|
||||||
os.rmdir(dir)
|
os.rmdir(dir)
|
||||||
except OSError:
|
except OSError:
|
||||||
break
|
break
|
||||||
dir = os.path.dirname(dir)
|
dir = os.path.dirname(dir)
|
||||||
|
|
||||||
new_project_paths.sort()
|
new_project_paths.sort()
|
||||||
fd = open(file_path, 'w')
|
fd = open(file_path, 'w')
|
||||||
@ -182,6 +197,51 @@ uncommitted changes are present' % project.relpath
|
|||||||
print >>sys.stderr, 'error: cannot combine -n and -l'
|
print >>sys.stderr, 'error: cannot combine -n and -l'
|
||||||
sys.exit(1)
|
sys.exit(1)
|
||||||
|
|
||||||
|
if opt.smart_sync:
|
||||||
|
if not self.manifest.manifest_server:
|
||||||
|
print >>sys.stderr, \
|
||||||
|
'error: cannot smart sync: no manifest server defined in manifest'
|
||||||
|
sys.exit(1)
|
||||||
|
try:
|
||||||
|
server = xmlrpclib.Server(self.manifest.manifest_server)
|
||||||
|
p = self.manifest.manifestProject
|
||||||
|
b = p.GetBranch(p.CurrentBranch)
|
||||||
|
branch = b.merge
|
||||||
|
if branch.startswith(R_HEADS):
|
||||||
|
branch = branch[len(R_HEADS):]
|
||||||
|
|
||||||
|
env = dict(os.environ)
|
||||||
|
if (env.has_key('TARGET_PRODUCT') and
|
||||||
|
env.has_key('TARGET_BUILD_VARIANT')):
|
||||||
|
target = '%s-%s' % (env['TARGET_PRODUCT'],
|
||||||
|
env['TARGET_BUILD_VARIANT'])
|
||||||
|
[success, manifest_str] = server.GetApprovedManifest(branch, target)
|
||||||
|
else:
|
||||||
|
[success, manifest_str] = server.GetApprovedManifest(branch)
|
||||||
|
|
||||||
|
if success:
|
||||||
|
manifest_name = "smart_sync_override.xml"
|
||||||
|
manifest_path = os.path.join(self.manifest.manifestProject.worktree,
|
||||||
|
manifest_name)
|
||||||
|
try:
|
||||||
|
f = open(manifest_path, 'w')
|
||||||
|
try:
|
||||||
|
f.write(manifest_str)
|
||||||
|
finally:
|
||||||
|
f.close()
|
||||||
|
except IOError:
|
||||||
|
print >>sys.stderr, 'error: cannot write manifest to %s' % \
|
||||||
|
manifest_path
|
||||||
|
sys.exit(1)
|
||||||
|
self.manifest.Override(manifest_name)
|
||||||
|
else:
|
||||||
|
print >>sys.stderr, 'error: %s' % manifest_str
|
||||||
|
sys.exit(1)
|
||||||
|
except socket.error:
|
||||||
|
print >>sys.stderr, 'error: cannot connect to manifest server %s' % (
|
||||||
|
self.manifest.manifest_server)
|
||||||
|
sys.exit(1)
|
||||||
|
|
||||||
rp = self.manifest.repoProject
|
rp = self.manifest.repoProject
|
||||||
rp.PreSync()
|
rp.PreSync()
|
||||||
|
|
||||||
|
@ -20,6 +20,17 @@ from command import InteractiveCommand
|
|||||||
from editor import Editor
|
from editor import Editor
|
||||||
from error import UploadError
|
from error import UploadError
|
||||||
|
|
||||||
|
UNUSUAL_COMMIT_THRESHOLD = 5
|
||||||
|
|
||||||
|
def _ConfirmManyUploads(multiple_branches=False):
|
||||||
|
if multiple_branches:
|
||||||
|
print "ATTENTION: One or more branches has an unusually high number of commits."
|
||||||
|
else:
|
||||||
|
print "ATTENTION: You are uploading an unusually high number of commits."
|
||||||
|
print "YOU PROBABLY DO NOT MEAN TO DO THIS. (Did you rebase across branches?)"
|
||||||
|
answer = raw_input("If you are sure you intend to do this, type 'yes': ").strip()
|
||||||
|
return answer == "yes"
|
||||||
|
|
||||||
def _die(fmt, *args):
|
def _die(fmt, *args):
|
||||||
msg = fmt % args
|
msg = fmt % args
|
||||||
print >>sys.stderr, 'error: %s' % msg
|
print >>sys.stderr, 'error: %s' % msg
|
||||||
@ -128,6 +139,10 @@ Gerrit Code Review: http://code.google.com/p/gerrit/
|
|||||||
answer = sys.stdin.readline().strip()
|
answer = sys.stdin.readline().strip()
|
||||||
answer = answer in ('y', 'Y', 'yes', '1', 'true', 't')
|
answer = answer in ('y', 'Y', 'yes', '1', 'true', 't')
|
||||||
|
|
||||||
|
if answer:
|
||||||
|
if len(branch.commits) > UNUSUAL_COMMIT_THRESHOLD:
|
||||||
|
answer = _ConfirmManyUploads()
|
||||||
|
|
||||||
if answer:
|
if answer:
|
||||||
self._UploadAndReport([branch], people)
|
self._UploadAndReport([branch], people)
|
||||||
else:
|
else:
|
||||||
@ -192,6 +207,16 @@ Gerrit Code Review: http://code.google.com/p/gerrit/
|
|||||||
todo.append(branch)
|
todo.append(branch)
|
||||||
if not todo:
|
if not todo:
|
||||||
_die("nothing uncommented for upload")
|
_die("nothing uncommented for upload")
|
||||||
|
|
||||||
|
many_commits = False
|
||||||
|
for branch in todo:
|
||||||
|
if len(branch.commits) > UNUSUAL_COMMIT_THRESHOLD:
|
||||||
|
many_commits = True
|
||||||
|
break
|
||||||
|
if many_commits:
|
||||||
|
if not _ConfirmManyUploads(multiple_branches=True):
|
||||||
|
_die("upload aborted by user")
|
||||||
|
|
||||||
self._UploadAndReport(todo, people)
|
self._UploadAndReport(todo, people)
|
||||||
|
|
||||||
def _FindGerritChange(self, branch):
|
def _FindGerritChange(self, branch):
|
||||||
@ -258,6 +283,10 @@ Gerrit Code Review: http://code.google.com/p/gerrit/
|
|||||||
print >>sys.stderr, " use 'repo upload' without --replace"
|
print >>sys.stderr, " use 'repo upload' without --replace"
|
||||||
sys.exit(1)
|
sys.exit(1)
|
||||||
|
|
||||||
|
if len(branch.commits) > UNUSUAL_COMMIT_THRESHOLD:
|
||||||
|
if not _ConfirmManyUploads(multiple_branches=True):
|
||||||
|
_die("upload aborted by user")
|
||||||
|
|
||||||
branch.replace_changes = to_replace
|
branch.replace_changes = to_replace
|
||||||
self._UploadAndReport([branch], people)
|
self._UploadAndReport([branch], people)
|
||||||
|
|
||||||
|
Reference in New Issue
Block a user