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

Side by Side Diff: gclient.py

Issue 25322002: Add support in gclient for pre-DEPS hooks (Closed) Base URL: https://chromium.googlesource.com/chromium/tools/depot_tools.git@master
Patch Set: Add --noprehooks flag Created 7 years, 2 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
« no previous file with comments | « no previous file | testing_support/fake_repos.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) 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 """Meta checkout manager supporting both Subversion and GIT.""" 6 """Meta checkout manager supporting both Subversion and GIT."""
7 # Files 7 # Files
8 # .gclient : Current client configuration, written by 'config' command. 8 # .gclient : Current client configuration, written by 'config' command.
9 # Format is a Python script defining 'solutions', a list whose 9 # Format is a Python script defining 'solutions', a list whose
10 # entries each are maps binding the strings "name" and "url" 10 # entries each are maps binding the strings "name" and "url"
(...skipping 37 matching lines...) Expand 10 before | Expand all | Expand 10 after
48 # 48 #
49 # Example: 49 # Example:
50 # hooks = [ 50 # hooks = [
51 # { "pattern": "\\.(gif|jpe?g|pr0n|png)$", 51 # { "pattern": "\\.(gif|jpe?g|pr0n|png)$",
52 # "action": ["python", "image_indexer.py", "--all"]}, 52 # "action": ["python", "image_indexer.py", "--all"]},
53 # { "pattern": ".", 53 # { "pattern": ".",
54 # "name": "gyp", 54 # "name": "gyp",
55 # "action": ["python", "src/build/gyp_chromium"]}, 55 # "action": ["python", "src/build/gyp_chromium"]},
56 # ] 56 # ]
57 # 57 #
58 # Pre-DEPS Hooks
59 # DEPS files may optionally contain a list named "pre_deps_hooks". These are
60 # the same as normal hooks, except that they run before the DEPS are
61 # processed. Pre-DEPS hooks always run unless the --noprehooks flag is used.
62 #
58 # Specifying a target OS 63 # Specifying a target OS
59 # An optional key named "target_os" may be added to a gclient file to specify 64 # An optional key named "target_os" may be added to a gclient file to specify
60 # one or more additional operating systems that should be considered when 65 # one or more additional operating systems that should be considered when
61 # processing the deps_os dict of a DEPS file. 66 # processing the deps_os dict of a DEPS file.
62 # 67 #
63 # Example: 68 # Example:
64 # target_os = [ "android" ] 69 # target_os = [ "android" ]
65 # 70 #
66 # If the "target_os_only" key is also present and true, then *only* the 71 # If the "target_os_only" key is also present and true, then *only* the
67 # operating systems listed in "target_os" will be used. 72 # operating systems listed in "target_os" will be used.
(...skipping 211 matching lines...) Expand 10 before | Expand all | Expand 10 after
279 def __init__(self, parent, name, url, safesync_url, managed, custom_deps, 284 def __init__(self, parent, name, url, safesync_url, managed, custom_deps,
280 custom_vars, custom_hooks, deps_file, should_process): 285 custom_vars, custom_hooks, deps_file, should_process):
281 gclient_utils.WorkItem.__init__(self, name) 286 gclient_utils.WorkItem.__init__(self, name)
282 DependencySettings.__init__( 287 DependencySettings.__init__(
283 self, parent, url, safesync_url, managed, custom_deps, custom_vars, 288 self, parent, url, safesync_url, managed, custom_deps, custom_vars,
284 custom_hooks, deps_file, should_process) 289 custom_hooks, deps_file, should_process)
285 290
286 # This is in both .gclient and DEPS files: 291 # This is in both .gclient and DEPS files:
287 self._deps_hooks = [] 292 self._deps_hooks = []
288 293
294 self._pre_deps_hooks = []
295
289 # Calculates properties: 296 # Calculates properties:
290 self._parsed_url = None 297 self._parsed_url = None
291 self._dependencies = [] 298 self._dependencies = []
292 # A cache of the files affected by the current operation, necessary for 299 # A cache of the files affected by the current operation, necessary for
293 # hooks. 300 # hooks.
294 self._file_list = [] 301 self._file_list = []
295 # If it is not set to True, the dependency wasn't processed for its child 302 # If it is not set to True, the dependency wasn't processed for its child
296 # dependency, i.e. its DEPS wasn't read. 303 # dependency, i.e. its DEPS wasn't read.
297 self._deps_parsed = False 304 self._deps_parsed = False
298 # This dependency has been processed, i.e. checked out 305 # This dependency has been processed, i.e. checked out
299 self._processed = False 306 self._processed = False
307 # This dependency had its pre-DEPS hooks run
308 self._pre_deps_hooks_ran = False
300 # This dependency had its hook run 309 # This dependency had its hook run
301 self._hooks_ran = False 310 self._hooks_ran = False
302 # This is the scm used to checkout self.url. It may be used by dependencies 311 # This is the scm used to checkout self.url. It may be used by dependencies
303 # to get the datetime of the revision we checked out. 312 # to get the datetime of the revision we checked out.
304 self._used_scm = None 313 self._used_scm = None
305 # The actual revision we ended up getting, or None if that information is 314 # The actual revision we ended up getting, or None if that information is
306 # unavailable 315 # unavailable
307 self._got_revision = None 316 self._got_revision = None
308 317
309 if not self.name and self.parent: 318 if not self.name and self.parent:
(...skipping 231 matching lines...) Expand 10 before | Expand all | Expand 10 after
541 hook_names_to_suppress = [c.get('name', '') for c in self.custom_hooks] 550 hook_names_to_suppress = [c.get('name', '') for c in self.custom_hooks]
542 for hook in local_scope.get('hooks', []): 551 for hook in local_scope.get('hooks', []):
543 if hook.get('name', '') not in hook_names_to_suppress: 552 if hook.get('name', '') not in hook_names_to_suppress:
544 hooks_to_run.append(hook) 553 hooks_to_run.append(hook)
545 554
546 # add the replacements and any additions 555 # add the replacements and any additions
547 for hook in self.custom_hooks: 556 for hook in self.custom_hooks:
548 if 'action' in hook: 557 if 'action' in hook:
549 hooks_to_run.append(hook) 558 hooks_to_run.append(hook)
550 559
560 self._pre_deps_hooks = [self.GetHookAction(hook, []) for hook in
561 local_scope.get('pre_deps_hooks', [])]
562
551 self.add_dependencies_and_close(deps_to_add, hooks_to_run) 563 self.add_dependencies_and_close(deps_to_add, hooks_to_run)
552 logging.info('ParseDepsFile(%s) done' % self.name) 564 logging.info('ParseDepsFile(%s) done' % self.name)
553 565
554 def add_dependencies_and_close(self, deps_to_add, hooks): 566 def add_dependencies_and_close(self, deps_to_add, hooks):
555 """Adds the dependencies, hooks and mark the parsing as done.""" 567 """Adds the dependencies, hooks and mark the parsing as done."""
556 for dep in deps_to_add: 568 for dep in deps_to_add:
557 if dep.verify_validity(): 569 if dep.verify_validity():
558 self.add_dependency(dep) 570 self.add_dependency(dep)
559 self._mark_as_parsed(hooks) 571 self._mark_as_parsed(hooks)
560 572
(...skipping 80 matching lines...) Expand 10 before | Expand all | Expand 10 after
641 continue 653 continue
642 prefix = os.path.commonprefix( 654 prefix = os.path.commonprefix(
643 [self.root.root_dir.lower(), file_list[i].lower()]) 655 [self.root.root_dir.lower(), file_list[i].lower()])
644 file_list[i] = file_list[i][len(prefix):] 656 file_list[i] = file_list[i][len(prefix):]
645 # Strip any leading path separators. 657 # Strip any leading path separators.
646 while file_list[i].startswith(('\\', '/')): 658 while file_list[i].startswith(('\\', '/')):
647 file_list[i] = file_list[i][1:] 659 file_list[i] = file_list[i][1:]
648 660
649 # Always parse the DEPS file. 661 # Always parse the DEPS file.
650 self.ParseDepsFile() 662 self.ParseDepsFile()
651
652 self._run_is_done(file_list or [], parsed_url) 663 self._run_is_done(file_list or [], parsed_url)
664 if not options.noprehooks:
665 self.RunPreDepsHooks()
653 666
654 if self.recursion_limit: 667 if self.recursion_limit:
655 # Parse the dependencies of this dependency. 668 # Parse the dependencies of this dependency.
656 for s in self.dependencies: 669 for s in self.dependencies:
657 work_queue.enqueue(s) 670 work_queue.enqueue(s)
658 671
659 if command == 'recurse': 672 if command == 'recurse':
660 if not isinstance(parsed_url, self.FileImpl): 673 if not isinstance(parsed_url, self.FileImpl):
661 # Skip file only checkout. 674 # Skip file only checkout.
662 scm = gclient_scm.GetScmName(parsed_url) 675 scm = gclient_scm.GetScmName(parsed_url)
(...skipping 121 matching lines...) Expand 10 before | Expand all | Expand 10 after
784 # failed. Users of this script may wish to treat hook action failures 797 # failed. Users of this script may wish to treat hook action failures
785 # differently from VC failures. 798 # differently from VC failures.
786 print >> sys.stderr, 'Error: %s' % str(e) 799 print >> sys.stderr, 'Error: %s' % str(e)
787 sys.exit(2) 800 sys.exit(2)
788 finally: 801 finally:
789 elapsed_time = time.time() - start_time 802 elapsed_time = time.time() - start_time
790 if elapsed_time > 10: 803 if elapsed_time > 10:
791 print "Hook '%s' took %.2f secs" % ( 804 print "Hook '%s' took %.2f secs" % (
792 gclient_utils.CommandToStr(hook), elapsed_time) 805 gclient_utils.CommandToStr(hook), elapsed_time)
793 806
807 def RunPreDepsHooks(self):
808 assert self.processed
809 assert self.deps_parsed
810 assert not self.pre_deps_hooks_ran
811 assert not self.hooks_ran
812 for s in self.dependencies:
813 assert not s.processed
814 self._pre_deps_hooks_ran = True
815 for hook in self.pre_deps_hooks:
816 try:
817 start_time = time.time()
818 gclient_utils.CheckCallAndFilterAndHeader(
819 hook, cwd=self.root.root_dir, always=True)
820 except (gclient_utils.Error, subprocess2.CalledProcessError), e:
821 # Use a discrete exit status code of 2 to indicate that a hook action
822 # failed. Users of this script may wish to treat hook action failures
823 # differently from VC failures.
824 print >> sys.stderr, 'Error: %s' % str(e)
825 sys.exit(2)
826 finally:
827 elapsed_time = time.time() - start_time
828 if elapsed_time > 10:
829 print "Hook '%s' took %.2f secs" % (
830 gclient_utils.CommandToStr(hook), elapsed_time)
831
832
794 def subtree(self, include_all): 833 def subtree(self, include_all):
795 """Breadth first recursion excluding root node.""" 834 """Breadth first recursion excluding root node."""
796 dependencies = self.dependencies 835 dependencies = self.dependencies
797 for d in dependencies: 836 for d in dependencies:
798 if d.should_process or include_all: 837 if d.should_process or include_all:
799 yield d 838 yield d
800 for d in dependencies: 839 for d in dependencies:
801 for i in d.subtree(include_all): 840 for i in d.subtree(include_all):
802 yield i 841 yield i
803 842
(...skipping 19 matching lines...) Expand all
823 def dependencies(self): 862 def dependencies(self):
824 return tuple(self._dependencies) 863 return tuple(self._dependencies)
825 864
826 @property 865 @property
827 @gclient_utils.lockedmethod 866 @gclient_utils.lockedmethod
828 def deps_hooks(self): 867 def deps_hooks(self):
829 return tuple(self._deps_hooks) 868 return tuple(self._deps_hooks)
830 869
831 @property 870 @property
832 @gclient_utils.lockedmethod 871 @gclient_utils.lockedmethod
872 def pre_deps_hooks(self):
873 return tuple(self._pre_deps_hooks)
874
875 @property
876 @gclient_utils.lockedmethod
833 def parsed_url(self): 877 def parsed_url(self):
834 return self._parsed_url 878 return self._parsed_url
835 879
836 @property 880 @property
837 @gclient_utils.lockedmethod 881 @gclient_utils.lockedmethod
838 def deps_parsed(self): 882 def deps_parsed(self):
839 """This is purely for debugging purposes. It's not used anywhere.""" 883 """This is purely for debugging purposes. It's not used anywhere."""
840 return self._deps_parsed 884 return self._deps_parsed
841 885
842 @property 886 @property
843 @gclient_utils.lockedmethod 887 @gclient_utils.lockedmethod
844 def processed(self): 888 def processed(self):
845 return self._processed 889 return self._processed
846 890
847 @property 891 @property
848 @gclient_utils.lockedmethod 892 @gclient_utils.lockedmethod
893 def pre_deps_hooks_ran(self):
894 return self._pre_deps_hooks_ran
895
896 @property
897 @gclient_utils.lockedmethod
849 def hooks_ran(self): 898 def hooks_ran(self):
850 return self._hooks_ran 899 return self._hooks_ran
851 900
852 @property 901 @property
853 @gclient_utils.lockedmethod 902 @gclient_utils.lockedmethod
854 def file_list(self): 903 def file_list(self):
855 return tuple(self._file_list) 904 return tuple(self._file_list)
856 905
857 @property 906 @property
858 def used_scm(self): 907 def used_scm(self):
(...skipping 689 matching lines...) Expand 10 before | Expand all | Expand 10 after
1548 } 1597 }
1549 } 1598 }
1550 } 1599 }
1551 """) 1600 """)
1552 def CMDsync(parser, args): 1601 def CMDsync(parser, args):
1553 """Checkout/update all modules.""" 1602 """Checkout/update all modules."""
1554 parser.add_option('-f', '--force', action='store_true', 1603 parser.add_option('-f', '--force', action='store_true',
1555 help='force update even for unchanged modules') 1604 help='force update even for unchanged modules')
1556 parser.add_option('-n', '--nohooks', action='store_true', 1605 parser.add_option('-n', '--nohooks', action='store_true',
1557 help='don\'t run hooks after the update is complete') 1606 help='don\'t run hooks after the update is complete')
1607 parser.add_option('-p', '--noprehooks', action='store_true',
1608 help='don\'t run pre-DEPS hooks')
1558 parser.add_option('-r', '--revision', action='append', 1609 parser.add_option('-r', '--revision', action='append',
1559 dest='revisions', metavar='REV', default=[], 1610 dest='revisions', metavar='REV', default=[],
1560 help='Enforces revision/hash for the solutions with the ' 1611 help='Enforces revision/hash for the solutions with the '
1561 'format src@rev. The src@ part is optional and can be ' 1612 'format src@rev. The src@ part is optional and can be '
1562 'skipped. -r can be used multiple times when .gclient ' 1613 'skipped. -r can be used multiple times when .gclient '
1563 'has multiple solutions configured and will work even ' 1614 'has multiple solutions configured and will work even '
1564 'if the src@ part is skipped. Note that specifying ' 1615 'if the src@ part is skipped. Note that specifying '
1565 '--revision means your safesync_url gets ignored.') 1616 '--revision means your safesync_url gets ignored.')
1566 parser.add_option('--with_branch_heads', action='store_true', 1617 parser.add_option('--with_branch_heads', action='store_true',
1567 help='Clone git "branch_heads" refspecs in addition to ' 1618 help='Clone git "branch_heads" refspecs in addition to '
(...skipping 85 matching lines...) Expand 10 before | Expand all | Expand 10 after
1653 """Reverts all modifications in every dependencies. 1704 """Reverts all modifications in every dependencies.
1654 1705
1655 That's the nuclear option to get back to a 'clean' state. It removes anything 1706 That's the nuclear option to get back to a 'clean' state. It removes anything
1656 that shows up in svn status.""" 1707 that shows up in svn status."""
1657 parser.add_option('--deps', dest='deps_os', metavar='OS_LIST', 1708 parser.add_option('--deps', dest='deps_os', metavar='OS_LIST',
1658 help='override deps for the specified (comma-separated) ' 1709 help='override deps for the specified (comma-separated) '
1659 'platform(s); \'all\' will process all deps_os ' 1710 'platform(s); \'all\' will process all deps_os '
1660 'references') 1711 'references')
1661 parser.add_option('-n', '--nohooks', action='store_true', 1712 parser.add_option('-n', '--nohooks', action='store_true',
1662 help='don\'t run hooks after the revert is complete') 1713 help='don\'t run hooks after the revert is complete')
1714 parser.add_option('-p', '--noprehooks', action='store_true',
1715 help='don\'t run pre-DEPS hooks')
1663 parser.add_option('--upstream', action='store_true', 1716 parser.add_option('--upstream', action='store_true',
1664 help='Make repo state match upstream branch.') 1717 help='Make repo state match upstream branch.')
1665 (options, args) = parser.parse_args(args) 1718 (options, args) = parser.parse_args(args)
1666 # --force is implied. 1719 # --force is implied.
1667 options.force = True 1720 options.force = True
1668 options.reset = False 1721 options.reset = False
1669 options.delete_unversioned_trees = False 1722 options.delete_unversioned_trees = False
1670 client = GClient.LoadCurrentConfig(options) 1723 client = GClient.LoadCurrentConfig(options)
1671 if not client: 1724 if not client:
1672 raise gclient_utils.Error('client not configured; see \'gclient config\'') 1725 raise gclient_utils.Error('client not configured; see \'gclient config\'')
(...skipping 113 matching lines...) Expand 10 before | Expand all | Expand 10 after
1786 self.error('--jobs must be 1 or higher') 1839 self.error('--jobs must be 1 or higher')
1787 1840
1788 # These hacks need to die. 1841 # These hacks need to die.
1789 if not hasattr(options, 'revisions'): 1842 if not hasattr(options, 'revisions'):
1790 # GClient.RunOnDeps expects it even if not applicable. 1843 # GClient.RunOnDeps expects it even if not applicable.
1791 options.revisions = [] 1844 options.revisions = []
1792 if not hasattr(options, 'head'): 1845 if not hasattr(options, 'head'):
1793 options.head = None 1846 options.head = None
1794 if not hasattr(options, 'nohooks'): 1847 if not hasattr(options, 'nohooks'):
1795 options.nohooks = True 1848 options.nohooks = True
1849 if not hasattr(options, 'noprehooks'):
1850 options.noprehooks = False
iannucci 2013/10/09 16:41:24 Hm... I don't think any of these are really necess
borenet 2013/10/10 18:24:29 If I understand correctly, these are defaults for
1796 if not hasattr(options, 'deps_os'): 1851 if not hasattr(options, 'deps_os'):
1797 options.deps_os = None 1852 options.deps_os = None
1798 if not hasattr(options, 'manually_grab_svn_rev'): 1853 if not hasattr(options, 'manually_grab_svn_rev'):
1799 options.manually_grab_svn_rev = None 1854 options.manually_grab_svn_rev = None
1800 if not hasattr(options, 'force'): 1855 if not hasattr(options, 'force'):
1801 options.force = None 1856 options.force = None
1802 return (options, args) 1857 return (options, args)
1803 1858
1804 1859
1805 def disable_buffering(): 1860 def disable_buffering():
(...skipping 27 matching lines...) Expand all
1833 raise 1888 raise
1834 except (gclient_utils.Error, subprocess2.CalledProcessError), e: 1889 except (gclient_utils.Error, subprocess2.CalledProcessError), e:
1835 print >> sys.stderr, 'Error: %s' % str(e) 1890 print >> sys.stderr, 'Error: %s' % str(e)
1836 return 1 1891 return 1
1837 1892
1838 1893
1839 if '__main__' == __name__: 1894 if '__main__' == __name__:
1840 sys.exit(Main(sys.argv[1:])) 1895 sys.exit(Main(sys.argv[1:]))
1841 1896
1842 # vim: ts=2:sw=2:tw=80:et: 1897 # vim: ts=2:sw=2:tw=80:et:
OLDNEW
« no previous file with comments | « no previous file | testing_support/fake_repos.py » ('j') | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698