OLD | NEW |
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 Loading... |
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 Loading... |
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 Loading... |
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 Loading... |
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 Loading... |
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 Loading... |
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 Loading... |
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 Loading... |
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 Loading... |
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: |
OLD | NEW |