OLD | NEW |
---|---|
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 Loading... | |
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 Loading... | |
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 Loading... | |
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 Loading... | |
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 Loading... | |
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:])) |
OLD | NEW |