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

Side by Side Diff: presubmit_support.py

Issue 6897005: presubmit_support: cache the result of ChangedContents. (Closed) Base URL: http://src.chromium.org/svn/trunk/tools/depot_tools/
Patch Set: Fix spelling thing. Created 9 years, 8 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/presubmit_unittest.py » ('j') | no next file with comments »
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) 2011 The Chromium Authors. All rights reserved. 2 # Copyright (c) 2011 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.6.1' 9 __version__ = '1.6.1'
10 10
11 # TODO(joi) Add caching where appropriate/needed. The API is designed to allow 11 # TODO(joi) Add caching where appropriate/needed. The API is designed to allow
12 # caching (between all different invocations of presubmit scripts for a given 12 # caching (between all different invocations of presubmit scripts for a given
13 # change). We should add it as our presubmit scripts start feeling slow. 13 # change). We should add it as our presubmit scripts start feeling slow.
14 14
15 import copy
M-A Ruel 2011/04/23 01:05:03 not needed.
ncarter (slow) 2011/04/26 17:15:19 Done.
15 import cPickle # Exposed through the API. 16 import cPickle # Exposed through the API.
16 import cStringIO # Exposed through the API. 17 import cStringIO # Exposed through the API.
17 import fnmatch 18 import fnmatch
18 import glob 19 import glob
19 import logging 20 import logging
20 import marshal # Exposed through the API. 21 import marshal # Exposed through the API.
21 import optparse 22 import optparse
22 import os # Somewhat exposed through the API. 23 import os # Somewhat exposed through the API.
23 import pickle # Exposed through the API. 24 import pickle # Exposed through the API.
24 import random 25 import random
(...skipping 364 matching lines...) Expand 10 before | Expand all | Expand 10 after
389 390
390 This is useful for doing line-by-line regex checks, like checking for 391 This is useful for doing line-by-line regex checks, like checking for
391 trailing whitespace. 392 trailing whitespace.
392 393
393 Yields: 394 Yields:
394 a 3 tuple: 395 a 3 tuple:
395 the AffectedFile instance of the current file; 396 the AffectedFile instance of the current file;
396 integer line number (1-based); and 397 integer line number (1-based); and
397 the contents of the line as a string. 398 the contents of the line as a string.
398 399
399 Note: The cariage return (LF or CR) is stripped off. 400 Note: The carriage return (LF or CR) is stripped off.
400 """ 401 """
401 files = self.AffectedSourceFiles(source_file_filter) 402 files = self.AffectedSourceFiles(source_file_filter)
402 return _RightHandSideLinesImpl(files) 403 return _RightHandSideLinesImpl(files)
403 404
404 def ReadFile(self, file_item, mode='r'): 405 def ReadFile(self, file_item, mode='r'):
405 """Reads an arbitrary file. 406 """Reads an arbitrary file.
406 407
407 Deny reading anything outside the repository. 408 Deny reading anything outside the repository.
408 """ 409 """
409 if isinstance(file_item, AffectedFile): 410 if isinstance(file_item, AffectedFile):
410 file_item = file_item.AbsoluteLocalPath() 411 file_item = file_item.AbsoluteLocalPath()
411 if not file_item.startswith(self.change.RepositoryRoot()): 412 if not file_item.startswith(self.change.RepositoryRoot()):
412 raise IOError('Access outside the repository root is denied.') 413 raise IOError('Access outside the repository root is denied.')
413 return gclient_utils.FileRead(file_item, mode) 414 return gclient_utils.FileRead(file_item, mode)
414 415
415 416
416 class AffectedFile(object): 417 class AffectedFile(object):
417 """Representation of a file in a change.""" 418 """Representation of a file in a change."""
418 # Method could be a function 419 # Method could be a function
419 # pylint: disable=R0201 420 # pylint: disable=R0201
420 def __init__(self, path, action, repository_root=''): 421 def __init__(self, path, action, repository_root=''):
421 self._path = path 422 self._path = path
422 self._action = action 423 self._action = action
423 self._local_root = repository_root 424 self._local_root = repository_root
424 self._is_directory = None 425 self._is_directory = None
425 self._properties = {} 426 self._properties = {}
427 self._cached_changed_contents = None
428 self._cached_new_contents = None
426 logging.debug('%s(%s)' % (self.__class__.__name__, self._path)) 429 logging.debug('%s(%s)' % (self.__class__.__name__, self._path))
427 430
428 def ServerPath(self): 431 def ServerPath(self):
429 """Returns a path string that identifies the file in the SCM system. 432 """Returns a path string that identifies the file in the SCM system.
430 433
431 Returns the empty string if the file does not exist in SCM. 434 Returns the empty string if the file does not exist in SCM.
432 """ 435 """
433 return "" 436 return ""
434 437
435 def LocalPath(self): 438 def LocalPath(self):
(...skipping 32 matching lines...) Expand 10 before | Expand all | Expand 10 after
468 Deleted files are not text file.""" 471 Deleted files are not text file."""
469 raise NotImplementedError() # Implement when needed 472 raise NotImplementedError() # Implement when needed
470 473
471 def NewContents(self): 474 def NewContents(self):
472 """Returns an iterator over the lines in the new version of file. 475 """Returns an iterator over the lines in the new version of file.
473 476
474 The new version is the file in the user's workspace, i.e. the "right hand 477 The new version is the file in the user's workspace, i.e. the "right hand
475 side". 478 side".
476 479
477 Contents will be empty if the file is a directory or does not exist. 480 Contents will be empty if the file is a directory or does not exist.
478 Note: The cariage returns (LF or CR) are stripped off. 481 Note: The carriage returns (LF or CR) are stripped off.
479 """ 482 """
480 if self.IsDirectory(): 483 if self._cached_new_contents is None:
481 return [] 484 if self.IsDirectory():
482 else: 485 self._cached_new_contents = []
483 return gclient_utils.FileRead(self.AbsoluteLocalPath(), 486 else:
484 'rU').splitlines() 487 self._cached_new_contents = gclient_utils.FileRead(
M-A Ruel 2011/04/23 01:05:03 It should trap IOError and set to []. The file cou
ncarter (slow) 2011/04/26 17:15:19 Done. Also, added unittest.
488 self.AbsoluteLocalPath(), 'rU').splitlines()
489 return copy.deepcopy(self._cached_new_contents)
M-A Ruel 2011/04/23 01:05:03 return self._cached_new_contents[:]
ncarter (slow) 2011/04/26 17:15:19 Of course. Done.
485 490
486 def OldContents(self): 491 def OldContents(self):
487 """Returns an iterator over the lines in the old version of file. 492 """Returns an iterator over the lines in the old version of file.
488 493
489 The old version is the file in depot, i.e. the "left hand side". 494 The old version is the file in depot, i.e. the "left hand side".
490 """ 495 """
491 raise NotImplementedError() # Implement when needed 496 raise NotImplementedError() # Implement when needed
492 497
493 def OldFileTempPath(self): 498 def OldFileTempPath(self):
494 """Returns the path on local disk where the old contents resides. 499 """Returns the path on local disk where the old contents resides.
495 500
496 The old version is the file in depot, i.e. the "left hand side". 501 The old version is the file in depot, i.e. the "left hand side".
497 This is a read-only cached copy of the old contents. *DO NOT* try to 502 This is a read-only cached copy of the old contents. *DO NOT* try to
498 modify this file. 503 modify this file.
499 """ 504 """
500 raise NotImplementedError() # Implement if/when needed. 505 raise NotImplementedError() # Implement if/when needed.
501 506
502 def ChangedContents(self): 507 def ChangedContents(self):
503 """Returns a list of tuples (line number, line text) of all new lines. 508 """Returns a list of tuples (line number, line text) of all new lines.
504 509
505 This relies on the scm diff output describing each changed code section 510 This relies on the scm diff output describing each changed code section
506 with a line of the form 511 with a line of the form
507 512
508 ^@@ <old line num>,<old size> <new line num>,<new size> @@$ 513 ^@@ <old line num>,<old size> <new line num>,<new size> @@$
509 """ 514 """
510 new_lines = [] 515 if self._cached_changed_contents is not None:
516 return copy.deepcopy(self._cached_changed_contents)
517 self._cached_changed_contents = []
511 line_num = 0 518 line_num = 0
512 519
513 if self.IsDirectory(): 520 if self.IsDirectory():
514 return [] 521 return []
515 522
516 for line in self.GenerateScmDiff().splitlines(): 523 for line in self.GenerateScmDiff().splitlines():
517 m = re.match(r'^@@ [0-9\,\+\-]+ \+([0-9]+)\,[0-9]+ @@', line) 524 m = re.match(r'^@@ [0-9\,\+\-]+ \+([0-9]+)\,[0-9]+ @@', line)
518 if m: 525 if m:
519 line_num = int(m.groups(1)[0]) 526 line_num = int(m.groups(1)[0])
520 continue 527 continue
521 if line.startswith('+') and not line.startswith('++'): 528 if line.startswith('+') and not line.startswith('++'):
522 new_lines.append((line_num, line[1:])) 529 self._cached_changed_contents.append((line_num, line[1:]))
523 if not line.startswith('-'): 530 if not line.startswith('-'):
524 line_num += 1 531 line_num += 1
525 return new_lines 532 return copy.deepcopy(self._cached_changed_contents)
526 533
527 def __str__(self): 534 def __str__(self):
528 return self.LocalPath() 535 return self.LocalPath()
529 536
530 def GenerateScmDiff(self): 537 def GenerateScmDiff(self):
531 raise NotImplementedError() # Implemented in derived classes. 538 raise NotImplementedError() # Implemented in derived classes.
532 539
533 540
534 class SvnAffectedFile(AffectedFile): 541 class SvnAffectedFile(AffectedFile):
535 """Representation of a file in a change out of a Subversion checkout.""" 542 """Representation of a file in a change out of a Subversion checkout."""
(...skipping 677 matching lines...) Expand 10 before | Expand all | Expand 10 after
1213 except PresubmitFailure, e: 1220 except PresubmitFailure, e:
1214 print >> sys.stderr, e 1221 print >> sys.stderr, e
1215 print >> sys.stderr, 'Maybe your depot_tools is out of date?' 1222 print >> sys.stderr, 'Maybe your depot_tools is out of date?'
1216 print >> sys.stderr, 'If all fails, contact maruel@' 1223 print >> sys.stderr, 'If all fails, contact maruel@'
1217 return 2 1224 return 2
1218 1225
1219 1226
1220 if __name__ == '__main__': 1227 if __name__ == '__main__':
1221 fix_encoding.fix_encoding() 1228 fix_encoding.fix_encoding()
1222 sys.exit(Main(None)) 1229 sys.exit(Main(None))
OLDNEW
« no previous file with comments | « no previous file | tests/presubmit_unittest.py » ('j') | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698