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 """Enables directory-specific presubmit checks to run at upload and/or commit. | 6 """Enables directory-specific presubmit checks to run at upload and/or commit. |
7 """ | 7 """ |
8 | 8 |
9 __version__ = '1.8.0' | 9 __version__ = '1.8.0' |
10 | 10 |
(...skipping 179 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
190 | 190 |
191 | 191 |
192 # Top level object so multiprocessing can pickle | 192 # Top level object so multiprocessing can pickle |
193 # Public access through OutputApi object. | 193 # Public access through OutputApi object. |
194 class _MailTextResult(_PresubmitResult): | 194 class _MailTextResult(_PresubmitResult): |
195 """A warning that should be included in the review request email.""" | 195 """A warning that should be included in the review request email.""" |
196 def __init__(self, *args, **kwargs): | 196 def __init__(self, *args, **kwargs): |
197 super(_MailTextResult, self).__init__() | 197 super(_MailTextResult, self).__init__() |
198 raise NotImplementedError() | 198 raise NotImplementedError() |
199 | 199 |
200 class GerritAccessor(object): | |
201 """Limited Gerrit functionality for canned presubmit checks to work. | |
202 | |
203 To avoid excessive Gerrit calls, caches the results. | |
204 """ | |
205 | |
206 def __init__(self, host): | |
207 self.host = host | |
208 self.cache = {} | |
209 | |
210 def _FetchChangeDetail(self, issue): | |
211 # Separate function to be easily mocked in tests. | |
212 return gerrit_util.GetChangeDetail( | |
213 self.host, str(issue), | |
214 ['ALL_REVISIONS', 'DETAILED_LABELS']) | |
215 | |
216 def GetChangeInfo(self, issue, force=False): | |
217 assert issue | |
218 cache_key = int(issue) | |
219 if force or cache_key not in self.cache: | |
220 self.cache[cache_key] = self._FetchChangeDetail(issue) | |
221 return self.cache[cache_key] | |
222 | |
223 def GetChangeDescription(self, issue, patchset=None, force=False): | |
Michael Achenbach
2016/04/28 07:17:30
Who's using force=True?
tandrii(chromium)
2016/04/28 19:31:50
nobody any more! Removed.
| |
224 """If patchset is none, fetches current patchset.""" | |
225 self.GetChangeInfo(issue, force=force) # Ensure details are in cache. | |
Michael Achenbach
2016/04/28 07:17:30
Why not:
info = self.GetChangeInfo(issue, force=fo
tandrii(chromium)
2016/04/28 19:31:50
good question. I thought modifying cache explicitl
| |
226 cache_key = int(issue) | |
227 | |
228 if patchset: | |
Michael Achenbach
2016/04/28 07:17:30
I assume gerrit patchset numbers are not starting
tandrii(chromium)
2016/04/28 19:31:51
yes, they don't.
| |
229 for rev, rev_info in self.cache[cache_key]['revisions'].iteritems(): | |
230 if str(rev_info['_number']) == str(patchset): | |
231 break | |
232 else: | |
233 raise Exception('patchset %s doesn\'t exist in issue %s' % ( | |
234 patchset, issue)) | |
235 else: | |
236 rev = self.cache[cache_key]['current_revision'] | |
237 rev_info = self.cache[cache_key]['revisions'][rev] | |
Michael Achenbach
2016/04/28 07:17:30
Can you explain of what each of the revisions is c
tandrii(chromium)
2016/04/28 19:31:50
rev_info is dict; rev, rev_info is a tuple, but th
Michael Achenbach
2016/04/29 08:10:18
No, just me being stupid. Got it now.
| |
238 | |
239 if 'real_description' not in rev_info: | |
240 rev_info['real_description'] = ( | |
241 gerrit_util.GetChangeDescriptionFromGitiles( | |
242 rev_info['fetch']['http']['url'], rev)) | |
243 return rev_info['real_description'] | |
244 | |
245 def GetChangeOwner(self, issue): | |
246 return self.GetChangeInfo(issue)['owner']['email'] | |
247 | |
248 def GetChangeReviewers(self, issue, approving_only=True): | |
249 # Gerrit has 'approved' sub-section, but it only lists 1 approver. | |
250 # So, if we look only for approvers, we have to look at all anyway. | |
251 # Also, assume LGTM means Code-Review label == 2. Other configurations | |
252 # aren't supported. | |
253 return [r['email'] | |
254 for r in self.GetChangeInfo(issue)['labels']['Code-Review']['all'] | |
Michael Achenbach
2016/04/28 07:17:30
Do you have a reference to a place that explains h
tandrii(chromium)
2016/04/28 19:31:50
https://gerrit-review.googlesource.com/Documentati
| |
255 if not approving_only or '2' == str(r.get('value', 0))] | |
256 | |
200 | 257 |
201 class OutputApi(object): | 258 class OutputApi(object): |
202 """An instance of OutputApi gets passed to presubmit scripts so that they | 259 """An instance of OutputApi gets passed to presubmit scripts so that they |
203 can output various types of results. | 260 can output various types of results. |
204 """ | 261 """ |
205 PresubmitResult = _PresubmitResult | 262 PresubmitResult = _PresubmitResult |
206 PresubmitAddReviewers = _PresubmitAddReviewers | 263 PresubmitAddReviewers = _PresubmitAddReviewers |
207 PresubmitError = _PresubmitError | 264 PresubmitError = _PresubmitError |
208 PresubmitPromptWarning = _PresubmitPromptWarning | 265 PresubmitPromptWarning = _PresubmitPromptWarning |
209 PresubmitNotifyResult = _PresubmitNotifyResult | 266 PresubmitNotifyResult = _PresubmitNotifyResult |
(...skipping 48 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
258 r".*\b[A-Z0-9_]{2,}$", | 315 r".*\b[A-Z0-9_]{2,}$", |
259 # SCM (can happen in dual SCM configuration). (Slightly over aggressive) | 316 # SCM (can happen in dual SCM configuration). (Slightly over aggressive) |
260 r"(|.*[\\\/])\.git[\\\/].*", | 317 r"(|.*[\\\/])\.git[\\\/].*", |
261 r"(|.*[\\\/])\.svn[\\\/].*", | 318 r"(|.*[\\\/])\.svn[\\\/].*", |
262 # There is no point in processing a patch file. | 319 # There is no point in processing a patch file. |
263 r".+\.diff$", | 320 r".+\.diff$", |
264 r".+\.patch$", | 321 r".+\.patch$", |
265 ) | 322 ) |
266 | 323 |
267 def __init__(self, change, presubmit_path, is_committing, | 324 def __init__(self, change, presubmit_path, is_committing, |
268 rietveld_obj, verbose, dry_run=None): | 325 rietveld_obj, verbose, dry_run=None, gerrit_obj=None): |
269 """Builds an InputApi object. | 326 """Builds an InputApi object. |
270 | 327 |
271 Args: | 328 Args: |
272 change: A presubmit.Change object. | 329 change: A presubmit.Change object. |
273 presubmit_path: The path to the presubmit script being processed. | 330 presubmit_path: The path to the presubmit script being processed. |
274 is_committing: True if the change is about to be committed. | 331 is_committing: True if the change is about to be committed. |
275 rietveld_obj: rietveld.Rietveld client object | 332 rietveld_obj: rietveld.Rietveld client object |
Michael Achenbach
2016/04/28 07:17:30
nit: Add to docu for consistency, though the rietv
tandrii(chromium)
2016/04/28 19:31:51
Done.
| |
276 """ | 333 """ |
277 # Version number of the presubmit_support script. | 334 # Version number of the presubmit_support script. |
278 self.version = [int(x) for x in __version__.split('.')] | 335 self.version = [int(x) for x in __version__.split('.')] |
279 self.change = change | 336 self.change = change |
280 self.is_committing = is_committing | 337 self.is_committing = is_committing |
281 self.rietveld = rietveld_obj | 338 self.rietveld = rietveld_obj |
282 self.dry_run = dry_run | 339 self.dry_run = dry_run |
340 self.gerrit = gerrit_obj | |
283 # TBD | 341 # TBD |
284 self.host_url = 'http://codereview.chromium.org' | 342 self.host_url = 'http://codereview.chromium.org' |
285 if self.rietveld: | 343 if self.rietveld: |
286 self.host_url = self.rietveld.url | 344 self.host_url = self.rietveld.url |
287 | 345 |
288 # We expose various modules and functions as attributes of the input_api | 346 # We expose various modules and functions as attributes of the input_api |
289 # so that presubmit scripts don't have to import them. | 347 # so that presubmit scripts don't have to import them. |
290 self.basename = os.path.basename | 348 self.basename = os.path.basename |
291 self.cPickle = cPickle | 349 self.cPickle = cPickle |
292 self.cpplint = cpplint | 350 self.cpplint = cpplint |
(...skipping 1049 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
1342 output_stream.write('** Post Upload Hook Messages **\n') | 1400 output_stream.write('** Post Upload Hook Messages **\n') |
1343 for result in results: | 1401 for result in results: |
1344 result.handle(output_stream) | 1402 result.handle(output_stream) |
1345 output_stream.write('\n') | 1403 output_stream.write('\n') |
1346 | 1404 |
1347 return results | 1405 return results |
1348 | 1406 |
1349 | 1407 |
1350 class PresubmitExecuter(object): | 1408 class PresubmitExecuter(object): |
1351 def __init__(self, change, committing, rietveld_obj, verbose, | 1409 def __init__(self, change, committing, rietveld_obj, verbose, |
1352 dry_run=None): | 1410 dry_run=None, gerrit_obj=None): |
Michael Achenbach
2016/04/28 07:17:30
nit: Maybe keep ( alignment? Or move all into a ne
tandrii(chromium)
2016/04/28 19:31:51
Done.
| |
1353 """ | 1411 """ |
1354 Args: | 1412 Args: |
1355 change: The Change object. | 1413 change: The Change object. |
1356 committing: True if 'gcl commit' is running, False if 'gcl upload' is. | 1414 committing: True if 'gcl commit' is running, False if 'gcl upload' is. |
1357 rietveld_obj: rietveld.Rietveld client object. | 1415 rietveld_obj: rietveld.Rietveld client object. |
Michael Achenbach
2016/04/28 07:17:30
nit: Add to docu for consistency, though the rietv
tandrii(chromium)
2016/04/28 19:31:51
Done.
| |
1358 """ | 1416 """ |
1359 self.change = change | 1417 self.change = change |
1360 self.committing = committing | 1418 self.committing = committing |
1361 self.rietveld = rietveld_obj | 1419 self.rietveld = rietveld_obj |
1420 self.gerrit = gerrit_obj | |
1362 self.verbose = verbose | 1421 self.verbose = verbose |
1363 self.dry_run = dry_run | 1422 self.dry_run = dry_run |
1364 | 1423 |
1365 def ExecPresubmitScript(self, script_text, presubmit_path): | 1424 def ExecPresubmitScript(self, script_text, presubmit_path): |
1366 """Executes a single presubmit script. | 1425 """Executes a single presubmit script. |
1367 | 1426 |
1368 Args: | 1427 Args: |
1369 script_text: The text of the presubmit script. | 1428 script_text: The text of the presubmit script. |
1370 presubmit_path: The path to the presubmit file (this will be reported via | 1429 presubmit_path: The path to the presubmit file (this will be reported via |
1371 input_api.PresubmitLocalPath()). | 1430 input_api.PresubmitLocalPath()). |
1372 | 1431 |
1373 Return: | 1432 Return: |
1374 A list of result objects, empty if no problems. | 1433 A list of result objects, empty if no problems. |
1375 """ | 1434 """ |
1376 | 1435 |
1377 # Change to the presubmit file's directory to support local imports. | 1436 # Change to the presubmit file's directory to support local imports. |
1378 main_path = os.getcwd() | 1437 main_path = os.getcwd() |
1379 os.chdir(os.path.dirname(presubmit_path)) | 1438 os.chdir(os.path.dirname(presubmit_path)) |
1380 | 1439 |
1381 # Load the presubmit script into context. | 1440 # Load the presubmit script into context. |
1382 input_api = InputApi(self.change, presubmit_path, self.committing, | 1441 input_api = InputApi(self.change, presubmit_path, self.committing, |
1383 self.rietveld, self.verbose, | 1442 self.rietveld, self.verbose, |
1384 dry_run=self.dry_run) | 1443 self.dry_run, self.gerrit) |
1385 context = {} | 1444 context = {} |
1386 try: | 1445 try: |
1387 exec script_text in context | 1446 exec script_text in context |
1388 except Exception, e: | 1447 except Exception, e: |
1389 raise PresubmitFailure('"%s" had an exception.\n%s' % (presubmit_path, e)) | 1448 raise PresubmitFailure('"%s" had an exception.\n%s' % (presubmit_path, e)) |
1390 | 1449 |
1391 # These function names must change if we make substantial changes to | 1450 # These function names must change if we make substantial changes to |
1392 # the presubmit API that are not backwards compatible. | 1451 # the presubmit API that are not backwards compatible. |
1393 if self.committing: | 1452 if self.committing: |
1394 function_name = 'CheckChangeOnCommit' | 1453 function_name = 'CheckChangeOnCommit' |
(...skipping 22 matching lines...) Expand all Loading... | |
1417 | 1476 |
1418 | 1477 |
1419 def DoPresubmitChecks(change, | 1478 def DoPresubmitChecks(change, |
1420 committing, | 1479 committing, |
1421 verbose, | 1480 verbose, |
1422 output_stream, | 1481 output_stream, |
1423 input_stream, | 1482 input_stream, |
1424 default_presubmit, | 1483 default_presubmit, |
1425 may_prompt, | 1484 may_prompt, |
1426 rietveld_obj, | 1485 rietveld_obj, |
1427 dry_run=None): | 1486 dry_run=None, |
1487 gerrit_obj=None): | |
Michael Achenbach
2016/04/28 07:17:30
Just some order nit: The dry_run feature is quite
tandrii(chromium)
2016/04/28 19:31:50
Done.
| |
1428 """Runs all presubmit checks that apply to the files in the change. | 1488 """Runs all presubmit checks that apply to the files in the change. |
1429 | 1489 |
1430 This finds all PRESUBMIT.py files in directories enclosing the files in the | 1490 This finds all PRESUBMIT.py files in directories enclosing the files in the |
1431 change (up to the repository root) and calls the relevant entrypoint function | 1491 change (up to the repository root) and calls the relevant entrypoint function |
1432 depending on whether the change is being committed or uploaded. | 1492 depending on whether the change is being committed or uploaded. |
1433 | 1493 |
1434 Prints errors, warnings and notifications. Prompts the user for warnings | 1494 Prints errors, warnings and notifications. Prompts the user for warnings |
1435 when needed. | 1495 when needed. |
1436 | 1496 |
1437 Args: | 1497 Args: |
1438 change: The Change object. | 1498 change: The Change object. |
1439 committing: True if 'gcl commit' is running, False if 'gcl upload' is. | 1499 committing: True if 'gcl commit' is running, False if 'gcl upload' is. |
1440 verbose: Prints debug info. | 1500 verbose: Prints debug info. |
1441 output_stream: A stream to write output from presubmit tests to. | 1501 output_stream: A stream to write output from presubmit tests to. |
1442 input_stream: A stream to read input from the user. | 1502 input_stream: A stream to read input from the user. |
1443 default_presubmit: A default presubmit script to execute in any case. | 1503 default_presubmit: A default presubmit script to execute in any case. |
1444 may_prompt: Enable (y/n) questions on warning or error. | 1504 may_prompt: Enable (y/n) questions on warning or error. |
1445 rietveld_obj: rietveld.Rietveld object. | 1505 rietveld_obj: rietveld.Rietveld object. |
1446 dry_run: if true, some Checks will be skipped. | 1506 dry_run: if true, some Checks will be skipped. |
1507 gerrit_obj: provides basic Gerrit codereview functionality. | |
1447 | 1508 |
1448 Warning: | 1509 Warning: |
1449 If may_prompt is true, output_stream SHOULD be sys.stdout and input_stream | 1510 If may_prompt is true, output_stream SHOULD be sys.stdout and input_stream |
1450 SHOULD be sys.stdin. | 1511 SHOULD be sys.stdin. |
1451 | 1512 |
1452 Return: | 1513 Return: |
1453 A PresubmitOutput object. Use output.should_continue() to figure out | 1514 A PresubmitOutput object. Use output.should_continue() to figure out |
1454 if there were errors or warnings and the caller should abort. | 1515 if there were errors or warnings and the caller should abort. |
1455 """ | 1516 """ |
1456 old_environ = os.environ | 1517 old_environ = os.environ |
1457 try: | 1518 try: |
1458 # Make sure python subprocesses won't generate .pyc files. | 1519 # Make sure python subprocesses won't generate .pyc files. |
1459 os.environ = os.environ.copy() | 1520 os.environ = os.environ.copy() |
1460 os.environ['PYTHONDONTWRITEBYTECODE'] = '1' | 1521 os.environ['PYTHONDONTWRITEBYTECODE'] = '1' |
1461 | 1522 |
1462 output = PresubmitOutput(input_stream, output_stream) | 1523 output = PresubmitOutput(input_stream, output_stream) |
1463 if committing: | 1524 if committing: |
1464 output.write("Running presubmit commit checks ...\n") | 1525 output.write("Running presubmit commit checks ...\n") |
1465 else: | 1526 else: |
1466 output.write("Running presubmit upload checks ...\n") | 1527 output.write("Running presubmit upload checks ...\n") |
1467 start_time = time.time() | 1528 start_time = time.time() |
1468 presubmit_files = ListRelevantPresubmitFiles( | 1529 presubmit_files = ListRelevantPresubmitFiles( |
1469 change.AbsoluteLocalPaths(True), change.RepositoryRoot()) | 1530 change.AbsoluteLocalPaths(True), change.RepositoryRoot()) |
1470 if not presubmit_files and verbose: | 1531 if not presubmit_files and verbose: |
1471 output.write("Warning, no PRESUBMIT.py found.\n") | 1532 output.write("Warning, no PRESUBMIT.py found.\n") |
1472 results = [] | 1533 results = [] |
1473 executer = PresubmitExecuter(change, committing, rietveld_obj, verbose, | 1534 executer = PresubmitExecuter(change, committing, rietveld_obj, verbose, |
1474 dry_run=dry_run) | 1535 dry_run, gerrit_obj) |
1475 if default_presubmit: | 1536 if default_presubmit: |
1476 if verbose: | 1537 if verbose: |
1477 output.write("Running default presubmit script.\n") | 1538 output.write("Running default presubmit script.\n") |
1478 fake_path = os.path.join(change.RepositoryRoot(), 'PRESUBMIT.py') | 1539 fake_path = os.path.join(change.RepositoryRoot(), 'PRESUBMIT.py') |
1479 results += executer.ExecPresubmitScript(default_presubmit, fake_path) | 1540 results += executer.ExecPresubmitScript(default_presubmit, fake_path) |
1480 for filename in presubmit_files: | 1541 for filename in presubmit_files: |
1481 filename = os.path.abspath(filename) | 1542 filename = os.path.abspath(filename) |
1482 if verbose: | 1543 if verbose: |
1483 output.write("Running %s\n" % filename) | 1544 output.write("Running %s\n" % filename) |
1484 # Accept CRLF presubmit script. | 1545 # Accept CRLF presubmit script. |
(...skipping 204 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
1689 "can be passed to this program.") | 1750 "can be passed to this program.") |
1690 if options.rietveld_email_file: | 1751 if options.rietveld_email_file: |
1691 with open(options.rietveld_email_file, "rb") as f: | 1752 with open(options.rietveld_email_file, "rb") as f: |
1692 options.rietveld_email = f.read().strip() | 1753 options.rietveld_email = f.read().strip() |
1693 | 1754 |
1694 change_class, files = load_files(options, args) | 1755 change_class, files = load_files(options, args) |
1695 if not change_class: | 1756 if not change_class: |
1696 parser.error('For unversioned directory, <files> is not optional.') | 1757 parser.error('For unversioned directory, <files> is not optional.') |
1697 logging.info('Found %d file(s).' % len(files)) | 1758 logging.info('Found %d file(s).' % len(files)) |
1698 | 1759 |
1699 rietveld_obj = None | 1760 rietveld_obj, gerrit_obj = None, None |
1761 | |
1700 if options.rietveld_url: | 1762 if options.rietveld_url: |
1701 # The empty password is permitted: '' is not None. | 1763 # The empty password is permitted: '' is not None. |
1702 if options.rietveld_private_key_file: | 1764 if options.rietveld_private_key_file: |
1703 rietveld_obj = rietveld.JwtOAuth2Rietveld( | 1765 rietveld_obj = rietveld.JwtOAuth2Rietveld( |
1704 options.rietveld_url, | 1766 options.rietveld_url, |
1705 options.rietveld_email, | 1767 options.rietveld_email, |
1706 options.rietveld_private_key_file) | 1768 options.rietveld_private_key_file) |
1707 else: | 1769 else: |
1708 rietveld_obj = rietveld.CachingRietveld( | 1770 rietveld_obj = rietveld.CachingRietveld( |
1709 options.rietveld_url, | 1771 options.rietveld_url, |
1710 auth_config, | 1772 auth_config, |
1711 options.rietveld_email) | 1773 options.rietveld_email) |
1712 if options.rietveld_fetch: | 1774 if options.rietveld_fetch: |
1713 assert options.issue | 1775 assert options.issue |
1714 props = rietveld_obj.get_issue_properties(options.issue, False) | 1776 props = rietveld_obj.get_issue_properties(options.issue, False) |
1715 options.author = props['owner_email'] | 1777 options.author = props['owner_email'] |
1716 options.description = props['description'] | 1778 options.description = props['description'] |
1717 logging.info('Got author: "%s"', options.author) | 1779 logging.info('Got author: "%s"', options.author) |
1718 logging.info('Got description: """\n%s\n"""', options.description) | 1780 logging.info('Got description: """\n%s\n"""', options.description) |
1719 | 1781 |
1720 if options.gerrit_url and options.gerrit_fetch: | 1782 if options.gerrit_url and options.gerrit_fetch: |
1783 assert options.issue and options.patchset | |
1721 rietveld_obj = None | 1784 rietveld_obj = None |
Michael Achenbach
2016/04/28 07:17:29
So, a hybrid is not supported? :)
E.g. a CL that'
tandrii(chromium)
2016/04/28 19:31:50
Well, just call it twice with two sets of args :)
| |
1722 assert options.issue and options.patchset | 1785 gerrit_obj = GerritAccessor(urlparse.urlparse(options.gerrit_url).netloc) |
1723 props = gerrit_util.GetChangeDetail( | 1786 options.author = gerrit_obj.GetChangeOwner(options.issue) |
1724 urlparse.urlparse(options.gerrit_url).netloc, str(options.issue), | 1787 options.description = gerrit_obj.GetChangeDescription(options.patchset) |
1725 ['ALL_REVISIONS']) | |
1726 options.author = props['owner']['email'] | |
1727 for rev, rev_info in props['revisions'].iteritems(): | |
1728 if str(rev_info['_number']) == str(options.patchset): | |
1729 options.description = gerrit_util.GetChangeDescriptionFromGitiles( | |
1730 rev_info['fetch']['http']['url'], rev) | |
1731 break | |
1732 else: | |
1733 print >> sys.stderr, ('Patchset %d was not found in Gerrit issue %d' % | |
1734 options.patchset, options.issue) | |
1735 return 2 | |
1736 logging.info('Got author: "%s"', options.author) | 1788 logging.info('Got author: "%s"', options.author) |
1737 logging.info('Got description: """\n%s\n"""', options.description) | 1789 logging.info('Got description: """\n%s\n"""', options.description) |
1738 | 1790 |
1739 try: | 1791 try: |
1740 with canned_check_filter(options.skip_canned): | 1792 with canned_check_filter(options.skip_canned): |
1741 results = DoPresubmitChecks( | 1793 results = DoPresubmitChecks( |
1742 change_class(options.name, | 1794 change_class(options.name, |
1743 options.description, | 1795 options.description, |
1744 options.root, | 1796 options.root, |
1745 files, | 1797 files, |
1746 options.issue, | 1798 options.issue, |
1747 options.patchset, | 1799 options.patchset, |
1748 options.author, | 1800 options.author, |
1749 upstream=options.upstream), | 1801 upstream=options.upstream), |
1750 options.commit, | 1802 options.commit, |
1751 options.verbose, | 1803 options.verbose, |
1752 sys.stdout, | 1804 sys.stdout, |
1753 sys.stdin, | 1805 sys.stdin, |
1754 options.default_presubmit, | 1806 options.default_presubmit, |
1755 options.may_prompt, | 1807 options.may_prompt, |
1756 rietveld_obj, | 1808 rietveld_obj, |
1757 options.dry_run) | 1809 options.dry_run, |
1810 gerrit_obj) | |
1758 return not results.should_continue() | 1811 return not results.should_continue() |
1759 except NonexistantCannedCheckFilter, e: | 1812 except NonexistantCannedCheckFilter, e: |
1760 print >> sys.stderr, ( | 1813 print >> sys.stderr, ( |
1761 'Attempted to skip nonexistent canned presubmit check: %s' % e.message) | 1814 'Attempted to skip nonexistent canned presubmit check: %s' % e.message) |
1762 return 2 | 1815 return 2 |
1763 except PresubmitFailure, e: | 1816 except PresubmitFailure, e: |
1764 print >> sys.stderr, e | 1817 print >> sys.stderr, e |
1765 print >> sys.stderr, 'Maybe your depot_tools is out of date?' | 1818 print >> sys.stderr, 'Maybe your depot_tools is out of date?' |
1766 print >> sys.stderr, 'If all fails, contact maruel@' | 1819 print >> sys.stderr, 'If all fails, contact maruel@' |
1767 return 2 | 1820 return 2 |
1768 | 1821 |
1769 | 1822 |
1770 if __name__ == '__main__': | 1823 if __name__ == '__main__': |
1771 fix_encoding.fix_encoding() | 1824 fix_encoding.fix_encoding() |
1772 try: | 1825 try: |
1773 sys.exit(main()) | 1826 sys.exit(main()) |
1774 except KeyboardInterrupt: | 1827 except KeyboardInterrupt: |
1775 sys.stderr.write('interrupted\n') | 1828 sys.stderr.write('interrupted\n') |
1776 sys.exit(1) | 1829 sys.exit(1) |
OLD | NEW |