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

Side by Side Diff: presubmit_support.py

Issue 1927773002: Reland Implement owners check in presubmit for Gerrit. (Closed) Base URL: https://chromium.googlesource.com/chromium/tools/depot_tools.git@P300
Patch Set: thanks pylint Created 4 years, 7 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
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 """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
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
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
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
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
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)
OLDNEW

Powered by Google App Engine
This is Rietveld 408576698