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

Side by Side Diff: gclient.py

Issue 2808048: Reapply r51760 and r51761 that were reverted in r51767. (Closed)
Patch Set: Fix issues in File() highlighted by new unit tests Created 10 years, 5 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 | gclient_utils.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/python 1 #!/usr/bin/python
2 # Copyright (c) 2010 The Chromium Authors. All rights reserved. 2 # Copyright (c) 2010 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 7
8 Files 8 Files
9 .gclient : Current client configuration, written by 'config' command. 9 .gclient : Current client configuration, written by 'config' command.
10 Format is a Python script defining 'solutions', a list whose 10 Format is a Python script defining 'solutions', a list whose
(...skipping 31 matching lines...) Expand 10 before | Expand all | Expand 10 after
42 it will be removed from the list and the list will be extended 42 it will be removed from the list and the list will be extended
43 by the list of matching files. 43 by the list of matching files.
44 44
45 Example: 45 Example:
46 hooks = [ 46 hooks = [
47 { "pattern": "\\.(gif|jpe?g|pr0n|png)$", 47 { "pattern": "\\.(gif|jpe?g|pr0n|png)$",
48 "action": ["python", "image_indexer.py", "--all"]}, 48 "action": ["python", "image_indexer.py", "--all"]},
49 ] 49 ]
50 """ 50 """
51 51
52 __version__ = "0.4.1" 52 __version__ = "0.5"
53 53
54 import errno
55 import logging 54 import logging
56 import optparse 55 import optparse
57 import os 56 import os
58 import pprint 57 import pprint
59 import re 58 import re
60 import subprocess 59 import subprocess
61 import sys 60 import sys
62 import urlparse 61 import urlparse
63 import urllib 62 import urllib
64 63
(...skipping 25 matching lines...) Expand all
90 89
91 sub_target_name is an optional parameter if the module name in the other 90 sub_target_name is an optional parameter if the module name in the other
92 DEPS file is different. E.g., you might want to map src/net to net.""" 91 DEPS file is different. E.g., you might want to map src/net to net."""
93 self.module_name = module_name 92 self.module_name = module_name
94 self.sub_target_name = sub_target_name 93 self.sub_target_name = sub_target_name
95 94
96 def __str__(self): 95 def __str__(self):
97 return 'From(%s, %s)' % (repr(self.module_name), 96 return 'From(%s, %s)' % (repr(self.module_name),
98 repr(self.sub_target_name)) 97 repr(self.sub_target_name))
99 98
100 def GetUrl(self, target_name, sub_deps_base_url, root_dir, sub_deps):
101 """Resolve the URL for this From entry."""
102 sub_deps_target_name = target_name
103 if self.sub_target_name:
104 sub_deps_target_name = self.sub_target_name
105 url = sub_deps[sub_deps_target_name]
106 if url.startswith('/'):
107 # If it's a relative URL, we need to resolve the URL relative to the
108 # sub deps base URL.
109 if not isinstance(sub_deps_base_url, basestring):
110 sub_deps_base_url = sub_deps_base_url.GetPath()
111 scm = gclient_scm.CreateSCM(sub_deps_base_url, root_dir,
112 None)
113 url = scm.FullUrlForRelativeUrl(url)
114 return url
115
116 class FileImpl(object): 99 class FileImpl(object):
117 """Used to implement the File('') syntax which lets you sync a single file 100 """Used to implement the File('') syntax which lets you sync a single file
118 from a SVN repo.""" 101 from a SVN repo."""
119 102
120 def __init__(self, file_location): 103 def __init__(self, file_location):
121 self.file_location = file_location 104 self.file_location = file_location
122 105
123 def __str__(self): 106 def __str__(self):
124 return 'File("%s")' % self.file_location 107 return 'File("%s")' % self.file_location
125 108
(...skipping 27 matching lines...) Expand all
153 class Dependency(GClientKeywords): 136 class Dependency(GClientKeywords):
154 """Object that represents a dependency checkout.""" 137 """Object that represents a dependency checkout."""
155 DEPS_FILE = 'DEPS' 138 DEPS_FILE = 'DEPS'
156 139
157 def __init__(self, parent, name, url, safesync_url=None, custom_deps=None, 140 def __init__(self, parent, name, url, safesync_url=None, custom_deps=None,
158 custom_vars=None, deps_file=None): 141 custom_vars=None, deps_file=None):
159 GClientKeywords.__init__(self) 142 GClientKeywords.__init__(self)
160 self.parent = parent 143 self.parent = parent
161 self.name = name 144 self.name = name
162 self.url = url 145 self.url = url
146 self.parsed_url = None
163 # These 2 are only set in .gclient and not in DEPS files. 147 # These 2 are only set in .gclient and not in DEPS files.
164 self.safesync_url = safesync_url 148 self.safesync_url = safesync_url
165 self.custom_vars = custom_vars or {} 149 self.custom_vars = custom_vars or {}
166 self.custom_deps = custom_deps or {} 150 self.custom_deps = custom_deps or {}
167 self.deps_hooks = [] 151 self.deps_hooks = []
168 self.dependencies = [] 152 self.dependencies = []
169 self.deps_file = deps_file or self.DEPS_FILE 153 self.deps_file = deps_file or self.DEPS_FILE
154 # A cache of the files affected by the current operation, necessary for
155 # hooks.
156 self.file_list = []
170 self.deps_parsed = False 157 self.deps_parsed = False
171 self.direct_reference = False 158 self.direct_reference = False
172 159
173 # Sanity checks 160 # Sanity checks
174 if not self.name and self.parent: 161 if not self.name and self.parent:
175 raise gclient_utils.Error('Dependency without name') 162 raise gclient_utils.Error('Dependency without name')
163 if self.name in [d.name for d in self.tree(False)]:
164 raise gclient_utils.Error('Dependency %s specified more than once' %
165 self.name)
176 if not isinstance(self.url, 166 if not isinstance(self.url,
177 (basestring, self.FromImpl, self.FileImpl, None.__class__)): 167 (basestring, self.FromImpl, self.FileImpl, None.__class__)):
178 raise gclient_utils.Error('dependency url must be either a string, None, ' 168 raise gclient_utils.Error('dependency url must be either a string, None, '
179 'File() or From() instead of %s' % 169 'File() or From() instead of %s' %
180 self.url.__class__.__name__) 170 self.url.__class__.__name__)
181 if '/' in self.deps_file or '\\' in self.deps_file: 171 if '/' in self.deps_file or '\\' in self.deps_file:
182 raise gclient_utils.Error('deps_file name must not be a path, just a ' 172 raise gclient_utils.Error('deps_file name must not be a path, just a '
183 'filename. %s' % self.deps_file) 173 'filename. %s' % self.deps_file)
184 174
175 def LateOverride(self, url):
176 overriden_url = self.get_custom_deps(self.name, url)
177 if overriden_url != url:
178 self.parsed_url = overriden_url
179 logging.debug('%s, %s was overriden to %s' % (self.name, url,
180 self.parsed_url))
181 elif isinstance(url, self.FromImpl):
182 ref = [dep for dep in self.tree(True) if url.module_name == dep.name]
183 if not len(ref) == 1:
184 raise Exception('Failed to find one reference to %s. %s' % (
185 url.module_name, ref))
186 ref = ref[0]
187 sub_target = url.sub_target_name or url
188 # Make sure the referenced dependency DEPS file is loaded and file the
189 # inner referenced dependency.
190 ref.ParseDepsFile(False)
191 found_dep = None
192 for d in ref.dependencies:
193 if d.name == sub_target:
194 found_dep = d
195 break
196 if not found_dep:
197 raise Exception('Couldn\'t find %s in %s, referenced by %s' % (
198 sub_target, ref.name, self.name))
199 # Call LateOverride() again.
200 self.parsed_url = found_dep.LateOverride(found_dep.url)
201 logging.debug('%s, %s to %s' % (self.name, url, self.parsed_url))
202 elif isinstance(url, basestring):
203 parsed_url = urlparse.urlparse(url)
204 if not parsed_url[0]:
205 # A relative url. Fetch the real base.
206 path = parsed_url[2]
207 if not path.startswith('/'):
208 raise gclient_utils.Error(
209 'relative DEPS entry \'%s\' must begin with a slash' % url)
210 # Create a scm just to query the full url.
211 scm = gclient_scm.CreateSCM(self.parent.parsed_url, self.root_dir(),
212 None)
213 self.parsed_url = scm.FullUrlForRelativeUrl(url)
214 else:
215 self.parsed_url = url
216 logging.debug('%s, %s -> %s' % (self.name, url, self.parsed_url))
217 elif isinstance(url, self.FileImpl):
218 self.parsed_url = url
219 logging.debug('%s, %s -> %s (File)' % (self.name, url, self.parsed_url))
220 return self.parsed_url
221
185 def ParseDepsFile(self, direct_reference): 222 def ParseDepsFile(self, direct_reference):
186 """Parses the DEPS file for this dependency.""" 223 """Parses the DEPS file for this dependency."""
187 if direct_reference: 224 if direct_reference:
188 # Maybe it was referenced earlier by a From() keyword but it's now 225 # Maybe it was referenced earlier by a From() keyword but it's now
189 # directly referenced. 226 # directly referenced.
190 self.direct_reference = direct_reference 227 self.direct_reference = direct_reference
228 if self.deps_parsed:
229 return
191 self.deps_parsed = True 230 self.deps_parsed = True
192 filepath = os.path.join(self.root_dir(), self.name, self.deps_file) 231 filepath = os.path.join(self.root_dir(), self.name, self.deps_file)
193 if not os.path.isfile(filepath): 232 if not os.path.isfile(filepath):
194 return {} 233 return
195 deps_content = gclient_utils.FileRead(filepath) 234 deps_content = gclient_utils.FileRead(filepath)
196 235
197 # Eval the content. 236 # Eval the content.
198 # One thing is unintuitive, vars= {} must happen before Var() use. 237 # One thing is unintuitive, vars= {} must happen before Var() use.
199 local_scope = {} 238 local_scope = {}
200 var = self.VarImpl(self.custom_vars, local_scope) 239 var = self.VarImpl(self.custom_vars, local_scope)
201 global_scope = { 240 global_scope = {
202 'File': self.FileImpl, 241 'File': self.FileImpl,
203 'From': self.FromImpl, 242 'From': self.FromImpl,
204 'Var': var.Lookup, 243 'Var': var.Lookup,
(...skipping 30 matching lines...) Expand all
235 # the dictionary using paths relative to the directory containing 274 # the dictionary using paths relative to the directory containing
236 # the DEPS file. 275 # the DEPS file.
237 use_relative_paths = local_scope.get('use_relative_paths', False) 276 use_relative_paths = local_scope.get('use_relative_paths', False)
238 if use_relative_paths: 277 if use_relative_paths:
239 rel_deps = {} 278 rel_deps = {}
240 for d, url in deps.items(): 279 for d, url in deps.items():
241 # normpath is required to allow DEPS to use .. in their 280 # normpath is required to allow DEPS to use .. in their
242 # dependency local path. 281 # dependency local path.
243 rel_deps[os.path.normpath(os.path.join(self.name, d))] = url 282 rel_deps[os.path.normpath(os.path.join(self.name, d))] = url
244 deps = rel_deps 283 deps = rel_deps
245 # TODO(maruel): Add these dependencies into self.dependencies.
246 return deps
247 284
248 def _ParseAllDeps(self, solution_urls): 285 # Convert the deps into real Dependency.
249 """Parse the complete list of dependencies for the client. 286 for name, url in deps.iteritems():
287 if name in [s.name for s in self.dependencies]:
288 raise
289 self.dependencies.append(Dependency(self, name, url))
290 # Sort by name.
291 self.dependencies.sort(key=lambda x: x.name)
292 logging.info('Loaded: %s' % str(self))
250 293
251 Args: 294 def RunCommandRecursively(self, options, revision_overrides,
252 solution_urls: A dict mapping module names (as relative paths) to URLs 295 command, args, pm):
253 corresponding to the solutions specified by the client. This parameter 296 """Runs 'command' before parsing the DEPS in case it's a initial checkout
254 is passed as an optimization. 297 or a revert."""
298 assert self.file_list == []
299 # When running runhooks, there's no need to consult the SCM.
300 # All known hooks are expected to run unconditionally regardless of working
301 # copy state, so skip the SCM status check.
302 run_scm = command not in ('runhooks', None)
303 self.LateOverride(self.url)
304 if run_scm and self.parsed_url:
305 if isinstance(self.parsed_url, self.FileImpl):
306 # Special support for single-file checkout.
307 if not command in (None, 'cleanup', 'diff', 'pack', 'status'):
308 options.revision = self.parsed_url.GetRevision()
309 scm = gclient_scm.SVNWrapper(self.parsed_url.GetPath(),
310 self.root_dir(),
311 self.name)
312 scm.RunCommand('updatesingle', options,
313 args + [self.parsed_url.GetFilename()],
314 self.file_list)
315 else:
316 options.revision = revision_overrides.get(self.name)
317 scm = gclient_scm.CreateSCM(self.parsed_url, self.root_dir(), self.name)
318 scm.RunCommand(command, options, args, self.file_list)
319 self.file_list = [os.path.join(self.name, f.strip())
320 for f in self.file_list]
321 options.revision = None
322 if pm:
323 # The + 1 comes from the fact that .gclient is considered a step in
324 # itself, .i.e. this code is called one time for the .gclient. This is not
325 # conceptually correct but it simplifies code.
326 pm._total = len(self.tree(False)) + 1
327 pm.update()
328 if self.recursion_limit():
329 # Then we can parse the DEPS file.
330 self.ParseDepsFile(True)
331 if pm:
332 pm._total = len(self.tree(False)) + 1
333 pm.update(0)
334 # Parse the dependencies of this dependency.
335 for s in self.dependencies:
336 # TODO(maruel): All these can run concurrently! No need for threads,
337 # just buffer stdout&stderr on pipes and flush as they complete.
338 # Watch out for stdin.
339 s.RunCommandRecursively(options, revision_overrides, command, args, pm)
255 340
256 Returns: 341 def RunHooksRecursively(self, options):
257 A dict mapping module names (as relative paths) to URLs corresponding 342 """Evaluates all hooks, running actions as needed. RunCommandRecursively()
258 to the entire set of dependencies to checkout for the given client. 343 must have been called before to load the DEPS."""
344 # If "--force" was specified, run all hooks regardless of what files have
345 # changed.
346 if self.deps_hooks:
347 # TODO(maruel): If the user is using git or git-svn, then we don't know
348 # what files have changed so we always run all hooks. It'd be nice to fix
349 # that.
350 if (options.force or
351 gclient_scm.GetScmName(self.parsed_url) in ('git', None) or
352 os.path.isdir(os.path.join(self.root_dir(), self.name, '.git'))):
353 for hook_dict in self.deps_hooks:
354 self._RunHookAction(hook_dict, [])
355 else:
356 # TODO(phajdan.jr): We should know exactly when the paths are absolute.
357 # Convert all absolute paths to relative.
358 for i in range(len(self.file_list)):
359 # It depends on the command being executed (like runhooks vs sync).
360 if not os.path.isabs(self.file_list[i]):
361 continue
259 362
260 Raises: 363 prefix = os.path.commonprefix([self.root_dir().lower(),
261 Error: If a dependency conflicts with another dependency or of a solution. 364 self.file_list[i].lower()])
262 """ 365 self.file_list[i] = self.file_list[i][len(prefix):]
263 deps = {}
264 for solution in self.dependencies:
265 solution_deps = solution.ParseDepsFile(True)
266 366
267 # If a line is in custom_deps, but not in the solution, we want to append 367 # Strip any leading path separators.
268 # this line to the solution. 368 while (self.file_list[i].startswith('\\') or
269 for d in solution.custom_deps: 369 self.file_list[i].startswith('/')):
270 if d not in solution_deps: 370 self.file_list[i] = self.file_list[i][1:]
271 solution_deps[d] = solution.custom_deps[d]
272 371
273 for d in solution_deps: 372 # Run hooks on the basis of whether the files from the gclient operation
274 if d in solution.custom_deps: 373 # match each hook's pattern.
275 # Dependency is overriden. 374 for hook_dict in self.deps_hooks:
276 url = solution.custom_deps[d] 375 pattern = re.compile(hook_dict['pattern'])
277 if url is None: 376 matching_file_list = [f for f in self.file_list if pattern.search(f)]
278 continue 377 if matching_file_list:
279 else: 378 self._RunHookAction(hook_dict, matching_file_list)
280 url = solution_deps[d] 379 if self.recursion_limit():
281 # if we have a From reference dependent on another solution, then 380 for s in self.dependencies:
282 # just skip the From reference. When we pull deps for the solution, 381 s.RunHooksRecursively(options)
283 # we will take care of this dependency.
284 #
285 # If multiple solutions all have the same From reference, then we
286 # should only add one to our list of dependencies.
287 if isinstance(url, self.FromImpl):
288 if url.module_name in solution_urls:
289 # Already parsed.
290 continue
291 if d in deps and type(deps[d]) != str:
292 if url.module_name == deps[d].module_name:
293 continue
294 elif isinstance(url, str):
295 parsed_url = urlparse.urlparse(url)
296 scheme = parsed_url[0]
297 if not scheme:
298 # A relative url. Fetch the real base.
299 path = parsed_url[2]
300 if path[0] != "/":
301 raise gclient_utils.Error(
302 "relative DEPS entry \"%s\" must begin with a slash" % d)
303 # Create a scm just to query the full url.
304 scm = gclient_scm.CreateSCM(solution.url, self.root_dir(),
305 None)
306 url = scm.FullUrlForRelativeUrl(url)
307 if d in deps and deps[d] != url:
308 raise gclient_utils.Error(
309 "Solutions have conflicting versions of dependency \"%s\"" % d)
310 if d in solution_urls and solution_urls[d] != url:
311 raise gclient_utils.Error(
312 "Dependency \"%s\" conflicts with specified solution" % d)
313 # Grab the dependency.
314 deps[d] = url
315 return deps
316
317 def _RunHooks(self, command, file_list, is_using_git):
318 """Evaluates all hooks, running actions as needed.
319 """
320 # Hooks only run for these command types.
321 if not command in ('update', 'revert', 'runhooks'):
322 return
323
324 # Hooks only run when --nohooks is not specified
325 if self._options.nohooks:
326 return
327
328 # Get any hooks from the .gclient file.
329 hooks = self.deps_hooks[:]
330 # Add any hooks found in DEPS files.
331 for d in self.dependencies:
332 hooks.extend(d.deps_hooks)
333
334 # If "--force" was specified, run all hooks regardless of what files have
335 # changed. If the user is using git, then we don't know what files have
336 # changed so we always run all hooks.
337 if self._options.force or is_using_git:
338 for hook_dict in hooks:
339 self._RunHookAction(hook_dict, [])
340 return
341
342 # Run hooks on the basis of whether the files from the gclient operation
343 # match each hook's pattern.
344 for hook_dict in hooks:
345 pattern = re.compile(hook_dict['pattern'])
346 matching_file_list = [f for f in file_list if pattern.search(f)]
347 if matching_file_list:
348 self._RunHookAction(hook_dict, matching_file_list)
349 382
350 def _RunHookAction(self, hook_dict, matching_file_list): 383 def _RunHookAction(self, hook_dict, matching_file_list):
351 """Runs the action from a single hook.""" 384 """Runs the action from a single hook."""
352 logging.info(hook_dict) 385 logging.info(hook_dict)
353 logging.info(matching_file_list) 386 logging.info(matching_file_list)
354 command = hook_dict['action'][:] 387 command = hook_dict['action'][:]
355 if command[0] == 'python': 388 if command[0] == 'python':
356 # If the hook specified "python" as the first item, the action is a 389 # If the hook specified "python" as the first item, the action is a
357 # Python script. Run it by starting a new copy of the same 390 # Python script. Run it by starting a new copy of the same
358 # interpreter. 391 # interpreter.
(...skipping 23 matching lines...) Expand all
382 def get_custom_deps(self, name, url): 415 def get_custom_deps(self, name, url):
383 """Returns a custom deps if applicable.""" 416 """Returns a custom deps if applicable."""
384 if self.parent: 417 if self.parent:
385 url = self.parent.get_custom_deps(name, url) 418 url = self.parent.get_custom_deps(name, url)
386 # None is a valid return value to disable a dependency. 419 # None is a valid return value to disable a dependency.
387 return self.custom_deps.get(name, url) 420 return self.custom_deps.get(name, url)
388 421
389 def __str__(self): 422 def __str__(self):
390 out = [] 423 out = []
391 for i in ('name', 'url', 'safesync_url', 'custom_deps', 'custom_vars', 424 for i in ('name', 'url', 'safesync_url', 'custom_deps', 'custom_vars',
392 'deps_hooks'): 425 'deps_hooks', 'file_list'):
393 # 'deps_file' 426 # 'deps_file'
394 if self.__dict__[i]: 427 if self.__dict__[i]:
395 out.append('%s: %s' % (i, self.__dict__[i])) 428 out.append('%s: %s' % (i, self.__dict__[i]))
396 429
397 for d in self.dependencies: 430 for d in self.dependencies:
398 out.extend([' ' + x for x in str(d).splitlines()]) 431 out.extend([' ' + x for x in str(d).splitlines()])
399 out.append('') 432 out.append('')
400 return '\n'.join(out) 433 return '\n'.join(out)
401 434
402 def __repr__(self): 435 def __repr__(self):
(...skipping 95 matching lines...) Expand 10 before | Expand all | Expand 10 after
498 os.path.join(path, options.config_filename))) 531 os.path.join(path, options.config_filename)))
499 return client 532 return client
500 533
501 def SetDefaultConfig(self, solution_name, solution_url, safesync_url): 534 def SetDefaultConfig(self, solution_name, solution_url, safesync_url):
502 self.SetConfig(self.DEFAULT_CLIENT_FILE_TEXT % { 535 self.SetConfig(self.DEFAULT_CLIENT_FILE_TEXT % {
503 'solution_name': solution_name, 536 'solution_name': solution_name,
504 'solution_url': solution_url, 537 'solution_url': solution_url,
505 'safesync_url' : safesync_url, 538 'safesync_url' : safesync_url,
506 }) 539 })
507 540
508 def _SaveEntries(self, entries): 541 def _SaveEntries(self):
509 """Creates a .gclient_entries file to record the list of unique checkouts. 542 """Creates a .gclient_entries file to record the list of unique checkouts.
510 543
511 The .gclient_entries file lives in the same directory as .gclient. 544 The .gclient_entries file lives in the same directory as .gclient.
512 """ 545 """
513 # Sometimes pprint.pformat will use {', sometimes it'll use { ' ... It 546 # Sometimes pprint.pformat will use {', sometimes it'll use { ' ... It
514 # makes testing a bit too fun. 547 # makes testing a bit too fun.
515 result = pprint.pformat(entries, 2) 548 result = 'entries = {\n'
516 if result.startswith('{\''): 549 for entry in self.tree(False):
517 result = '{ \'' + result[2:] 550 # Skip over File() dependencies as we can't version them.
518 text = 'entries = \\\n' + result + '\n' 551 if not isinstance(entry.parsed_url, self.FileImpl):
552 result += ' %s: %s,\n' % (pprint.pformat(entry.name),
553 pprint.pformat(entry.parsed_url))
554 result += '}\n'
519 file_path = os.path.join(self.root_dir(), self._options.entries_filename) 555 file_path = os.path.join(self.root_dir(), self._options.entries_filename)
520 gclient_utils.FileWrite(file_path, text) 556 logging.info(result)
557 gclient_utils.FileWrite(file_path, result)
521 558
522 def _ReadEntries(self): 559 def _ReadEntries(self):
523 """Read the .gclient_entries file for the given client. 560 """Read the .gclient_entries file for the given client.
524 561
525 Returns: 562 Returns:
526 A sequence of solution names, which will be empty if there is the 563 A sequence of solution names, which will be empty if there is the
527 entries file hasn't been created yet. 564 entries file hasn't been created yet.
528 """ 565 """
529 scope = {} 566 scope = {}
530 filename = os.path.join(self.root_dir(), self._options.entries_filename) 567 filename = os.path.join(self.root_dir(), self._options.entries_filename)
(...skipping 40 matching lines...) Expand 10 before | Expand all | Expand 10 after
571 def RunOnDeps(self, command, args): 608 def RunOnDeps(self, command, args):
572 """Runs a command on each dependency in a client and its dependencies. 609 """Runs a command on each dependency in a client and its dependencies.
573 610
574 Args: 611 Args:
575 command: The command to use (e.g., 'status' or 'diff') 612 command: The command to use (e.g., 'status' or 'diff')
576 args: list of str - extra arguments to add to the command line. 613 args: list of str - extra arguments to add to the command line.
577 """ 614 """
578 if not self.dependencies: 615 if not self.dependencies:
579 raise gclient_utils.Error('No solution specified') 616 raise gclient_utils.Error('No solution specified')
580 revision_overrides = self._EnforceRevisions() 617 revision_overrides = self._EnforceRevisions()
581 618 pm = None
582 # When running runhooks --force, there's no need to consult the SCM.
583 # All known hooks are expected to run unconditionally regardless of working
584 # copy state, so skip the SCM status check.
585 run_scm = not (command == 'runhooks' and self._options.force)
586
587 entries = {}
588 file_list = []
589 # Run on the base solutions first.
590 for solution in self.dependencies:
591 name = solution.name
592 if name in entries:
593 raise gclient_utils.Error("solution %s specified more than once" % name)
594 url = solution.url
595 entries[name] = url
596 if run_scm and url:
597 self._options.revision = revision_overrides.get(name)
598 scm = gclient_scm.CreateSCM(url, self.root_dir(), name)
599 scm.RunCommand(command, self._options, args, file_list)
600 file_list = [os.path.join(name, f.strip()) for f in file_list]
601 self._options.revision = None
602
603 # Process the dependencies next (sort alphanumerically to ensure that
604 # containing directories get populated first and for readability)
605 deps = self._ParseAllDeps(entries)
606 deps_to_process = deps.keys()
607 deps_to_process.sort()
608
609 # First pass for direct dependencies.
610 if command == 'update' and not self._options.verbose: 619 if command == 'update' and not self._options.verbose:
611 pm = Progress('Syncing projects', len(deps_to_process)) 620 pm = Progress('Syncing projects', len(self.tree(False)) + 1)
612 for d in deps_to_process: 621 self.RunCommandRecursively(self._options, revision_overrides,
613 if command == 'update' and not self._options.verbose: 622 command, args, pm)
614 pm.update() 623 if pm:
615 if type(deps[d]) == str:
616 url = deps[d]
617 entries[d] = url
618 if run_scm:
619 self._options.revision = revision_overrides.get(d)
620 scm = gclient_scm.CreateSCM(url, self.root_dir(), d)
621 scm.RunCommand(command, self._options, args, file_list)
622 self._options.revision = None
623 elif isinstance(deps[d], self.FileImpl):
624 if command in (None, 'cleanup', 'diff', 'pack', 'status'):
625 continue
626 file_dep = deps[d]
627 self._options.revision = file_dep.GetRevision()
628 if run_scm:
629 scm = gclient_scm.SVNWrapper(file_dep.GetPath(), self.root_dir(), d)
630 scm.RunCommand('updatesingle', self._options,
631 args + [file_dep.GetFilename()], file_list)
632
633 if command == 'update' and not self._options.verbose:
634 pm.end() 624 pm.end()
635 625
636 # Second pass for inherited deps (via the From keyword) 626 # Once all the dependencies have been processed, it's now safe to run the
637 for d in deps_to_process: 627 # hooks.
638 if isinstance(deps[d], self.FromImpl): 628 if not self._options.nohooks:
639 # Getting the URL from the sub_deps file can involve having to resolve 629 self.RunHooksRecursively(self._options)
640 # a File() or having to resolve a relative URL. To resolve relative
641 # URLs, we need to pass in the orignal sub deps URL.
642 sub_deps_base_url = deps[deps[d].module_name]
643 sub_deps = Dependency(self, deps[d].module_name, sub_deps_base_url
644 ).ParseDepsFile(False)
645 url = deps[d].GetUrl(d, sub_deps_base_url, self.root_dir(), sub_deps)
646 entries[d] = url
647 if run_scm:
648 self._options.revision = revision_overrides.get(d)
649 scm = gclient_scm.CreateSCM(url, self.root_dir(), d)
650 scm.RunCommand(command, self._options, args, file_list)
651 self._options.revision = None
652
653 # Convert all absolute paths to relative.
654 for i in range(len(file_list)):
655 # TODO(phajdan.jr): We should know exactly when the paths are absolute.
656 # It depends on the command being executed (like runhooks vs sync).
657 if not os.path.isabs(file_list[i]):
658 continue
659
660 prefix = os.path.commonprefix([self.root_dir().lower(),
661 file_list[i].lower()])
662 file_list[i] = file_list[i][len(prefix):]
663
664 # Strip any leading path separators.
665 while file_list[i].startswith('\\') or file_list[i].startswith('/'):
666 file_list[i] = file_list[i][1:]
667
668 is_using_git = gclient_utils.IsUsingGit(self.root_dir(), entries.keys())
669 self._RunHooks(command, file_list, is_using_git)
670 630
671 if command == 'update': 631 if command == 'update':
672 # Notify the user if there is an orphaned entry in their working copy. 632 # Notify the user if there is an orphaned entry in their working copy.
673 # Only delete the directory if there are no changes in it, and 633 # Only delete the directory if there are no changes in it, and
674 # delete_unversioned_trees is set to true. 634 # delete_unversioned_trees is set to true.
675 prev_entries = self._ReadEntries() 635 entries = [i.name for i in self.tree(False)]
676 for entry in prev_entries: 636 for entry, prev_url in self._ReadEntries().iteritems():
677 # Fix path separator on Windows. 637 # Fix path separator on Windows.
678 entry_fixed = entry.replace('/', os.path.sep) 638 entry_fixed = entry.replace('/', os.path.sep)
679 e_dir = os.path.join(self.root_dir(), entry_fixed) 639 e_dir = os.path.join(self.root_dir(), entry_fixed)
680 # Use entry and not entry_fixed there. 640 # Use entry and not entry_fixed there.
681 if entry not in entries and os.path.exists(e_dir): 641 if entry not in entries and os.path.exists(e_dir):
682 modified_files = False 642 file_list = []
683 if isinstance(prev_entries, list): 643 scm = gclient_scm.CreateSCM(prev_url, self.root_dir(), entry_fixed)
684 # old .gclient_entries format was list, now dict 644 scm.status(self._options, [], file_list)
685 modified_files = gclient_scm.scm.SVN.CaptureStatus(e_dir) 645 modified_files = file_list != []
686 else:
687 file_list = []
688 scm = gclient_scm.CreateSCM(prev_entries[entry], self.root_dir(),
689 entry_fixed)
690 scm.status(self._options, [], file_list)
691 modified_files = file_list != []
692 if not self._options.delete_unversioned_trees or modified_files: 646 if not self._options.delete_unversioned_trees or modified_files:
693 # There are modified files in this entry. Keep warning until 647 # There are modified files in this entry. Keep warning until
694 # removed. 648 # removed.
695 print(('\nWARNING: \'%s\' is no longer part of this client. ' 649 print(('\nWARNING: \'%s\' is no longer part of this client. '
696 'It is recommended that you manually remove it.\n') % 650 'It is recommended that you manually remove it.\n') %
697 entry_fixed) 651 entry_fixed)
698 else: 652 else:
699 # Delete the entry 653 # Delete the entry
700 print('\n________ deleting \'%s\' in \'%s\'' % ( 654 print('\n________ deleting \'%s\' in \'%s\'' % (
701 entry_fixed, self.root_dir())) 655 entry_fixed, self.root_dir()))
702 gclient_utils.RemoveDirectory(e_dir) 656 gclient_utils.RemoveDirectory(e_dir)
703 # record the current list of entries for next time 657 # record the current list of entries for next time
704 self._SaveEntries(entries) 658 self._SaveEntries()
705 return 0 659 return 0
706 660
707 def PrintRevInfo(self): 661 def PrintRevInfo(self):
708 """Output revision info mapping for the client and its dependencies. 662 """Output revision info mapping for the client and its dependencies.
709 663
710 This allows the capture of an overall "revision" for the source tree that 664 This allows the capture of an overall "revision" for the source tree that
711 can be used to reproduce the same tree in the future. It is only useful for 665 can be used to reproduce the same tree in the future. It is only useful for
712 "unpinned dependencies", i.e. DEPS/deps references without a svn revision 666 "unpinned dependencies", i.e. DEPS/deps references without a svn revision
713 number or a git hash. A git branch name isn't "pinned" since the actual 667 number or a git hash. A git branch name isn't "pinned" since the actual
714 commit can change. 668 commit can change.
715 669
716 The --snapshot option allows creating a .gclient file to reproduce the tree. 670 The --snapshot option allows creating a .gclient file to reproduce the tree.
717 """ 671 """
718 if not self.dependencies: 672 if not self.dependencies:
719 raise gclient_utils.Error('No solution specified') 673 raise gclient_utils.Error('No solution specified')
674 # Load all the settings.
675 self.RunCommandRecursively(self._options, {}, None, [], None)
720 676
721 # Inner helper to generate base url and rev tuple
722 def GetURLAndRev(name, original_url): 677 def GetURLAndRev(name, original_url):
678 """Returns the revision-qualified SCM url."""
679 if isinstance(original_url, self.FileImpl):
680 return original_url.file_location
723 url, _ = gclient_utils.SplitUrlRevision(original_url) 681 url, _ = gclient_utils.SplitUrlRevision(original_url)
724 scm = gclient_scm.CreateSCM(original_url, self.root_dir(), name) 682 scm = gclient_scm.CreateSCM(original_url, self.root_dir(), name)
725 return (url, scm.revinfo(self._options, [], None)) 683 if not os.path.isdir(scm.checkout_path):
684 return None
685 return '%s@%s' % (url, scm.revinfo(self._options, [], None))
726 686
727 # text of the snapshot gclient file
728 new_gclient = ""
729 # Dictionary of { path : SCM url } to ensure no duplicate solutions
730 solution_names = {}
731 entries = {}
732 # Run on the base solutions first.
733 for solution in self.dependencies:
734 # Dictionary of { path : SCM url } to describe the gclient checkout
735 name = solution.name
736 if name in solution_names:
737 raise gclient_utils.Error("solution %s specified more than once" % name)
738 (url, rev) = GetURLAndRev(name, solution.url)
739 entries[name] = "%s@%s" % (url, rev)
740 solution_names[name] = "%s@%s" % (url, rev)
741
742 # Process the dependencies next (sort alphanumerically to ensure that
743 # containing directories get populated first and for readability)
744 deps = self._ParseAllDeps(entries)
745 deps_to_process = deps.keys()
746 deps_to_process.sort()
747
748 # First pass for direct dependencies.
749 for d in deps_to_process:
750 if type(deps[d]) == str:
751 (url, rev) = GetURLAndRev(d, deps[d])
752 entries[d] = "%s@%s" % (url, rev)
753
754 # Second pass for inherited deps (via the From keyword)
755 for d in deps_to_process:
756 if isinstance(deps[d], self.FromImpl):
757 deps_parent_url = entries[deps[d].module_name]
758 if deps_parent_url.find("@") < 0:
759 raise gclient_utils.Error("From %s missing revisioned url" %
760 deps[d].module_name)
761 sub_deps_base_url = deps[deps[d].module_name]
762 sub_deps = Dependency(self, deps[d].module_name, sub_deps_base_url
763 ).ParseDepsFile(False)
764 url = deps[d].GetUrl(d, sub_deps_base_url, self.root_dir(), sub_deps)
765 (url, rev) = GetURLAndRev(d, url)
766 entries[d] = "%s@%s" % (url, rev)
767
768 # Build the snapshot configuration string
769 if self._options.snapshot: 687 if self._options.snapshot:
770 url = entries.pop(name) 688 new_gclient = ''
771 custom_deps = ''.join([' \"%s\": \"%s\",\n' % (x, entries[x]) 689 # First level at .gclient
772 for x in sorted(entries.keys())]) 690 for d in self.dependencies:
773 691 entries = {}
774 new_gclient += self.DEFAULT_SNAPSHOT_SOLUTION_TEXT % { 692 def GrabDeps(sol):
775 'solution_name': name, 693 """Recursively grab dependencies."""
776 'solution_url': url, 694 for i in sol.dependencies:
777 'safesync_url' : '', 695 entries[i.name] = GetURLAndRev(i.name, i.parsed_url)
778 'solution_deps': custom_deps, 696 GrabDeps(i)
779 } 697 GrabDeps(d)
698 custom_deps = []
699 for k in sorted(entries.keys()):
700 if entries[k]:
701 # Quotes aren't escaped...
702 custom_deps.append(' \"%s\": \'%s\',\n' % (k, entries[k]))
703 else:
704 custom_deps.append(' \"%s\": None,\n' % k)
705 new_gclient += self.DEFAULT_SNAPSHOT_SOLUTION_TEXT % {
706 'solution_name': d.name,
707 'solution_url': d.url,
708 'safesync_url' : d.safesync_url or '',
709 'solution_deps': ''.join(custom_deps),
710 }
711 # Print the snapshot configuration file
712 print(self.DEFAULT_SNAPSHOT_FILE_TEXT % {'solution_list': new_gclient})
780 else: 713 else:
781 print(';\n'.join(['%s: %s' % (x, entries[x]) 714 entries = sorted(self.tree(False), key=lambda i: i.name)
782 for x in sorted(entries.keys())])) 715 for entry in entries:
783 716 revision = GetURLAndRev(entry.name, entry.parsed_url)
784 # Print the snapshot configuration file 717 line = '%s: %s' % (entry.name, revision)
785 if self._options.snapshot: 718 if not entry is entries[-1]:
786 config = self.DEFAULT_SNAPSHOT_FILE_TEXT % {'solution_list': new_gclient} 719 line += ';'
787 snapclient = GClient(self.root_dir(), self._options) 720 print line
788 snapclient.SetConfig(config)
789 print(snapclient.config_content)
790 721
791 def ParseDepsFile(self, direct_reference): 722 def ParseDepsFile(self, direct_reference):
792 """No DEPS to parse for a .gclient file.""" 723 """No DEPS to parse for a .gclient file."""
793 self.direct_reference = direct_reference 724 self.direct_reference = direct_reference
794 self.deps_parsed = True 725 self.deps_parsed = True
795 726
796 def root_dir(self): 727 def root_dir(self):
797 """Root directory of gclient checkout.""" 728 """Root directory of gclient checkout."""
798 return self._root_dir 729 return self._root_dir
799 730
(...skipping 300 matching lines...) Expand 10 before | Expand all | Expand 10 after
1100 'unpinned dependencies', i.e. DEPS/deps references without a svn revision 1031 'unpinned dependencies', i.e. DEPS/deps references without a svn revision
1101 number or a git hash. A git branch name isn't 'pinned' since the actual 1032 number or a git hash. A git branch name isn't 'pinned' since the actual
1102 commit can change. 1033 commit can change.
1103 """ 1034 """
1104 parser.add_option('--deps', dest='deps_os', metavar='OS_LIST', 1035 parser.add_option('--deps', dest='deps_os', metavar='OS_LIST',
1105 help='override deps for the specified (comma-separated) ' 1036 help='override deps for the specified (comma-separated) '
1106 'platform(s); \'all\' will process all deps_os ' 1037 'platform(s); \'all\' will process all deps_os '
1107 'references') 1038 'references')
1108 parser.add_option('-s', '--snapshot', action='store_true', 1039 parser.add_option('-s', '--snapshot', action='store_true',
1109 help='creates a snapshot .gclient file of the current ' 1040 help='creates a snapshot .gclient file of the current '
1110 'version of all repositories to reproduce the tree, ' 1041 'version of all repositories to reproduce the tree')
1111 'implies -a')
1112 (options, args) = parser.parse_args(args) 1042 (options, args) = parser.parse_args(args)
1113 client = GClient.LoadCurrentConfig(options) 1043 client = GClient.LoadCurrentConfig(options)
1114 if not client: 1044 if not client:
1115 raise gclient_utils.Error('client not configured; see \'gclient config\'') 1045 raise gclient_utils.Error('client not configured; see \'gclient config\'')
1116 client.PrintRevInfo() 1046 client.PrintRevInfo()
1117 return 0 1047 return 0
1118 1048
1119 1049
1120 def Command(name): 1050 def Command(name):
1121 return getattr(sys.modules[__name__], 'CMD' + name, None) 1051 return getattr(sys.modules[__name__], 'CMD' + name, None)
(...skipping 77 matching lines...) Expand 10 before | Expand all | Expand 10 after
1199 return CMDhelp(parser, argv) 1129 return CMDhelp(parser, argv)
1200 except gclient_utils.Error, e: 1130 except gclient_utils.Error, e:
1201 print >> sys.stderr, 'Error: %s' % str(e) 1131 print >> sys.stderr, 'Error: %s' % str(e)
1202 return 1 1132 return 1
1203 1133
1204 1134
1205 if '__main__' == __name__: 1135 if '__main__' == __name__:
1206 sys.exit(Main(sys.argv[1:])) 1136 sys.exit(Main(sys.argv[1:]))
1207 1137
1208 # vim: ts=2:sw=2:tw=80:et: 1138 # vim: ts=2:sw=2:tw=80:et:
OLDNEW
« no previous file with comments | « no previous file | gclient_utils.py » ('j') | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698