Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(1030)

Side by Side Diff: git_cl.py

Issue 10537117: Add support for new repo topology, with submodule-specific merge commits (Closed) Base URL: svn://svn.chromium.org/chrome/trunk/tools/depot_tools/
Patch Set: Created 8 years, 6 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View unified diff | Download patch | Annotate | Revision Log
« no previous file with comments | « no previous file | tests/git_cl_test.py » ('j') | tests/git_cl_test.py » ('J')
Toggle Intra-line Diffs ('i') | Expand Comments ('e') | Collapse Comments ('c') | Show Comments Hide Comments ('s')
OLDNEW
1 #!/usr/bin/env python 1 #!/usr/bin/env python
2 # Copyright (c) 2012 The Chromium Authors. All rights reserved. 2 # Copyright (c) 2012 The Chromium Authors. All rights reserved.
3 # Use of this source code is governed by a BSD-style license that can be 3 # Use of this source code is governed by a BSD-style license that can be
4 # found in the LICENSE file. 4 # found in the LICENSE file.
5 5
6 # Copyright (C) 2008 Evan Martin <martine@danga.com> 6 # Copyright (C) 2008 Evan Martin <martine@danga.com>
7 7
8 """A git-command for integrating reviews on Rietveld.""" 8 """A git-command for integrating reviews on Rietveld."""
9 9
10 import json 10 import json
(...skipping 51 matching lines...) Expand 10 before | Expand all | Expand 10 after
62 return e.stdout 62 return e.stdout
63 63
64 64
65 def RunGit(args, **kwargs): 65 def RunGit(args, **kwargs):
66 """Returns stdout.""" 66 """Returns stdout."""
67 return RunCommand(['git'] + args, **kwargs) 67 return RunCommand(['git'] + args, **kwargs)
68 68
69 69
70 def RunGitWithCode(args): 70 def RunGitWithCode(args):
71 """Returns return code and stdout.""" 71 """Returns return code and stdout."""
72 out, code = subprocess2.communicate(['git'] + args, stdout=subprocess2.PIPE) 72 try:
73 return code, out[0] 73 out, code = subprocess2.communicate(['git'] + args, stdout=subprocess2.PIPE)
74 return code, out[0]
75 except ValueError, e:
76 return 1, ''
M-A Ruel 2012/06/12 16:27:36 Can you add a comment when it's happening?
szager1 2012/06/13 13:54:46 When the subprocess fails, communicate() returns N
74 77
75 78
76 def usage(more): 79 def usage(more):
77 def hook(fn): 80 def hook(fn):
78 fn.usage_more = more 81 fn.usage_more = more
79 return fn 82 return fn
80 return hook 83 return hook
81 84
82 85
83 def ask_for_data(prompt): 86 def ask_for_data(prompt):
(...skipping 1044 matching lines...) Expand 10 before | Expand all | Expand 10 after
1128 if 'GIT_EXTERNAL_DIFF' in env: 1131 if 'GIT_EXTERNAL_DIFF' in env:
1129 del env['GIT_EXTERNAL_DIFF'] 1132 del env['GIT_EXTERNAL_DIFF']
1130 subprocess2.call( 1133 subprocess2.call(
1131 ['git', 'diff', '--no-ext-diff', '--stat', '-M'] + args, env=env) 1134 ['git', 'diff', '--no-ext-diff', '--stat', '-M'] + args, env=env)
1132 1135
1133 if settings.GetIsGerrit(): 1136 if settings.GetIsGerrit():
1134 return GerritUpload(options, args, cl) 1137 return GerritUpload(options, args, cl)
1135 return RietveldUpload(options, args, cl) 1138 return RietveldUpload(options, args, cl)
1136 1139
1137 1140
1141 def IsSubmoduleMergeCommit(ref):
1142 # When submodules are added to the repo, we expect there to be a single
1143 # non-git-svn merge commit at remote HEAD with a signature comment.
1144 pattern = '^SVN changes up to revision [0-9]*$'
1145 cmd = ['rev-list', '--merges', '--grep="%s"' % pattern, '%s^!' % ref]
1146 return RunGit(cmd) != ''
1147
1148
1138 def SendUpstream(parser, args, cmd): 1149 def SendUpstream(parser, args, cmd):
1139 """Common code for CmdPush and CmdDCommit 1150 """Common code for CmdPush and CmdDCommit
1140 1151
1141 Squashed commit into a single. 1152 Squashed commit into a single.
1142 Updates changelog with metadata (e.g. pointer to review). 1153 Updates changelog with metadata (e.g. pointer to review).
1143 Pushes/dcommits the code upstream. 1154 Pushes/dcommits the code upstream.
1144 Updates review and closes. 1155 Updates review and closes.
1145 """ 1156 """
1146 parser.add_option('--bypass-hooks', action='store_true', dest='bypass_hooks', 1157 parser.add_option('--bypass-hooks', action='store_true', dest='bypass_hooks',
1147 help='bypass upload presubmit hook') 1158 help='bypass upload presubmit hook')
(...skipping 11 matching lines...) Expand all
1159 if not args or cmd == 'push': 1170 if not args or cmd == 'push':
1160 # Default to merging against our best guess of the upstream branch. 1171 # Default to merging against our best guess of the upstream branch.
1161 args = [cl.GetUpstreamBranch()] 1172 args = [cl.GetUpstreamBranch()]
1162 1173
1163 if options.contributor: 1174 if options.contributor:
1164 if not re.match('^.*\s<\S+@\S+>$', options.contributor): 1175 if not re.match('^.*\s<\S+@\S+>$', options.contributor):
1165 print "Please provide contibutor as 'First Last <email@example.com>'" 1176 print "Please provide contibutor as 'First Last <email@example.com>'"
1166 return 1 1177 return 1
1167 1178
1168 base_branch = args[0] 1179 base_branch = args[0]
1180 base_has_submodules = IsSubmoduleMergeCommit(base_branch)
1169 1181
1170 # Make sure index is up-to-date before running diff-index. 1182 # Make sure index is up-to-date before running diff-index.
1171 RunGit(['update-index', '--refresh', '-q'], error_ok=True) 1183 RunGit(['update-index', '--refresh', '-q'], error_ok=True)
1172 if RunGit(['diff-index', 'HEAD']): 1184 if RunGit(['diff-index', 'HEAD']):
1173 print 'Cannot %s with a dirty tree. You must commit locally first.' % cmd 1185 print 'Cannot %s with a dirty tree. You must commit locally first.' % cmd
1174 return 1 1186 return 1
1175 1187
1176 # This rev-list syntax means "show all commits not in my branch that 1188 # This rev-list syntax means "show all commits not in my branch that
1177 # are in base_branch". 1189 # are in base_branch".
1178 upstream_commits = RunGit(['rev-list', '^' + cl.GetBranchRef(), 1190 upstream_commits = RunGit(['rev-list', '^' + cl.GetBranchRef(),
1179 base_branch]).splitlines() 1191 base_branch]).splitlines()
1180 if upstream_commits: 1192 if upstream_commits:
1181 print ('Base branch "%s" has %d commits ' 1193 print ('Base branch "%s" has %d commits '
1182 'not in this branch.' % (base_branch, len(upstream_commits))) 1194 'not in this branch.' % (base_branch, len(upstream_commits)))
1183 print 'Run "git merge %s" before attempting to %s.' % (base_branch, cmd) 1195 print 'Run "git merge %s" before attempting to %s.' % (base_branch, cmd)
1184 return 1 1196 return 1
1185 1197
1198 # This is the revision `svn dcommit` will commit on top of.
1199 svn_head = RunGit(['log', '--grep=^git-svn-id:', '-1',
1200 '--pretty=format:%H'])
1201
1186 if cmd == 'dcommit': 1202 if cmd == 'dcommit':
1187 # This is the revision `svn dcommit` will commit on top of. 1203 # If the base_head is a submodule merge commit, the second parent of the
M-A Ruel 2012/06/12 16:27:36 I still think it should be the primary parent (and
szager1 2012/06/13 13:54:46 Done.
1188 svn_head = RunGit(['log', '--grep=^git-svn-id:', '-1', 1204 # base_head should be a git-svn commit, which is what we're interested in.
1189 '--pretty=format:%H']) 1205 base_svn_head = base_branch
1190 extra_commits = RunGit(['rev-list', '^' + svn_head, base_branch]) 1206 if base_has_submodules:
1207 base_svn_head += '^2'
1208
1209 extra_commits = RunGit(['rev-list', '^' + svn_head, base_svn_head])
1191 if extra_commits: 1210 if extra_commits:
1192 print ('This branch has %d additional commits not upstreamed yet.' 1211 print ('This branch has %d additional commits not upstreamed yet.'
1193 % len(extra_commits.splitlines())) 1212 % len(extra_commits.splitlines()))
1194 print ('Upstream "%s" or rebase this branch on top of the upstream trunk ' 1213 print ('Upstream "%s" or rebase this branch on top of the upstream trunk '
1195 'before attempting to %s.' % (base_branch, cmd)) 1214 'before attempting to %s.' % (base_branch, cmd))
1196 return 1 1215 return 1
1197 1216
1198 if not options.bypass_hooks: 1217 if not options.bypass_hooks:
1199 author = None 1218 author = None
1200 if options.contributor: 1219 if options.contributor:
(...skipping 38 matching lines...) Expand 10 before | Expand all | Expand 10 after
1239 1258
1240 if options.contributor: 1259 if options.contributor:
1241 description += "\nPatch from %s." % options.contributor 1260 description += "\nPatch from %s." % options.contributor
1242 print 'Description:', repr(description) 1261 print 'Description:', repr(description)
1243 1262
1244 branches = [base_branch, cl.GetBranchRef()] 1263 branches = [base_branch, cl.GetBranchRef()]
1245 if not options.force: 1264 if not options.force:
1246 subprocess2.call(['git', 'diff', '--stat'] + branches) 1265 subprocess2.call(['git', 'diff', '--stat'] + branches)
1247 ask_for_data('About to commit; enter to confirm.') 1266 ask_for_data('About to commit; enter to confirm.')
1248 1267
1249 # We want to squash all this branch's commits into one commit with the 1268 # We want to squash all this branch's commits into one commit with the proper
1250 # proper description. 1269 # description. We do this by doing a "reset --soft" to the base branch (which
1251 # We do this by doing a "reset --soft" to the base branch (which keeps 1270 # keeps the working copy the same), then dcommitting that. If origin/master
1252 # the working copy the same), then dcommitting that. 1271 # has a submodule merge commit, we'll also need to cherry-pick the squashed
1272 # commit onto a branch based on the git-svn head.
1253 MERGE_BRANCH = 'git-cl-commit' 1273 MERGE_BRANCH = 'git-cl-commit'
1254 # Delete the merge branch if it already exists. 1274 CHERRY_PICK_BRANCH = 'git-cl-cherry-pick'
1255 if RunGitWithCode(['show-ref', '--quiet', '--verify', 1275 # Delete the branches if they exist
cmp 2012/06/12 16:05:46 nit: append a period to make this a first class se
szager1 2012/06/13 13:54:46 Done.
1256 'refs/heads/' + MERGE_BRANCH])[0] == 0: 1276 for branch in [MERGE_BRANCH, CHERRY_PICK_BRANCH]:
1257 RunGit(['branch', '-D', MERGE_BRANCH]) 1277 cmd = ['show-ref', '--quiet', '--verify', 'refs/heads/%s' % branch]
1278 result = RunGitWithCode(cmd)
1279 if result[0] == 0:
1280 RunGit(['branch', '-D', branch])
1258 1281
1259 # We might be in a directory that's present in this branch but not in the 1282 # We might be in a directory that's present in this branch but not in the
1260 # trunk. Move up to the top of the tree so that git commands that expect a 1283 # trunk. Move up to the top of the tree so that git commands that expect a
1261 # valid CWD won't fail after we check out the merge branch. 1284 # valid CWD won't fail after we check out the merge branch.
1262 rel_base_path = RunGit(['rev-parse', '--show-cdup']).strip() 1285 rel_base_path = RunGit(['rev-parse', '--show-cdup']).strip()
1263 if rel_base_path: 1286 if rel_base_path:
1264 os.chdir(rel_base_path) 1287 os.chdir(rel_base_path)
1265 1288
1266 # Stuff our change into the merge branch. 1289 # Stuff our change into the merge branch.
1267 # We wrap in a try...finally block so if anything goes wrong, 1290 # We wrap in a try...finally block so if anything goes wrong,
1268 # we clean up the branches. 1291 # we clean up the branches.
1269 retcode = -1 1292 retcode = -1
1270 try: 1293 try:
1271 RunGit(['checkout', '-q', '-b', MERGE_BRANCH]) 1294 RunGit(['checkout', '-q', '-b', MERGE_BRANCH])
1272 RunGit(['reset', '--soft', base_branch]) 1295 RunGit(['reset', '--soft', base_branch])
1273 if options.contributor: 1296 if options.contributor:
1274 RunGit(['commit', '--author', options.contributor, '-m', description]) 1297 RunGit(['commit', '--author', options.contributor, '-m', description])
1275 else: 1298 else:
1276 RunGit(['commit', '-m', description]) 1299 RunGit(['commit', '-m', description])
1300 if base_has_submodules:
1301 cherry_pick_commit = RunGit(['rev-list', 'HEAD^!']).rstrip()
1302 RunGit(['branch', CHERRY_PICK_BRANCH, svn_head])
1303 RunGit(['checkout', CHERRY_PICK_BRANCH])
1304 RunGit(['cherry-pick', cherry_pick_commit])
1277 if cmd == 'push': 1305 if cmd == 'push':
1278 # push the merge branch. 1306 # push the merge branch.
1279 remote, branch = cl.FetchUpstreamTuple() 1307 remote, branch = cl.FetchUpstreamTuple()
1280 retcode, output = RunGitWithCode( 1308 retcode, output = RunGitWithCode(
1281 ['push', '--porcelain', remote, 'HEAD:%s' % branch]) 1309 ['push', '--porcelain', remote, 'HEAD:%s' % branch])
1282 logging.debug(output) 1310 logging.debug(output)
1283 else: 1311 else:
1284 # dcommit the merge branch. 1312 # dcommit the merge branch.
1285 retcode, output = RunGitWithCode(['svn', 'dcommit', 1313 retcode, output = RunGitWithCode(['svn', 'dcommit',
1286 '--no-rebase', '--rmdir']) 1314 '--no-rebase', '--rmdir'])
1287 finally: 1315 finally:
1288 # And then swap back to the original branch and clean up. 1316 # And then swap back to the original branch and clean up.
1289 RunGit(['checkout', '-q', cl.GetBranch()]) 1317 RunGit(['checkout', '-q', cl.GetBranch()])
1290 RunGit(['branch', '-D', MERGE_BRANCH]) 1318 RunGit(['branch', '-D', MERGE_BRANCH])
1319 if base_has_submodules:
1320 RunGit(['branch', '-D', CHERRY_PICK_BRANCH])
1291 1321
1292 if cl.GetIssue(): 1322 if cl.GetIssue():
1293 if cmd == 'dcommit' and 'Committed r' in output: 1323 if cmd == 'dcommit' and 'Committed r' in output:
1294 revision = re.match('.*?\nCommitted r(\\d+)', output, re.DOTALL).group(1) 1324 revision = re.match('.*?\nCommitted r(\\d+)', output, re.DOTALL).group(1)
1295 elif cmd == 'push' and retcode == 0: 1325 elif cmd == 'push' and retcode == 0:
1296 match = (re.match(r'.*?([a-f0-9]{7})\.\.([a-f0-9]{7})$', l) 1326 match = (re.match(r'.*?([a-f0-9]{7})\.\.([a-f0-9]{7})$', l)
1297 for l in output.splitlines(False)) 1327 for l in output.splitlines(False))
1298 match = filter(None, match) 1328 match = filter(None, match)
1299 if len(match) != 1: 1329 if len(match) != 1:
1300 DieWithError("Couldn't parse ouput to extract the committed hash:\n%s" % 1330 DieWithError("Couldn't parse ouput to extract the committed hash:\n%s" %
(...skipping 266 matching lines...) Expand 10 before | Expand all | Expand 10 after
1567 'and retry or visit go/isgaeup.\n%s') % (e.code, str(e))) 1597 'and retry or visit go/isgaeup.\n%s') % (e.code, str(e)))
1568 1598
1569 # Not a known command. Default to help. 1599 # Not a known command. Default to help.
1570 GenUsage(parser, 'help') 1600 GenUsage(parser, 'help')
1571 return CMDhelp(parser, argv) 1601 return CMDhelp(parser, argv)
1572 1602
1573 1603
1574 if __name__ == '__main__': 1604 if __name__ == '__main__':
1575 fix_encoding.fix_encoding() 1605 fix_encoding.fix_encoding()
1576 sys.exit(main(sys.argv[1:])) 1606 sys.exit(main(sys.argv[1:]))
OLDNEW
« no previous file with comments | « no previous file | tests/git_cl_test.py » ('j') | tests/git_cl_test.py » ('J')

Powered by Google App Engine
This is Rietveld 408576698