blob: b19ab5fb95b883bab816f1c52558f5ab92ebcb79 [file] [log] [blame]
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -07001# Copyright (C) 2008 The Android Open Source Project
2#
3# Licensed under the Apache License, Version 2.0 (the "License");
4# you may not use this file except in compliance with the License.
5# You may obtain a copy of the License at
6#
7# http://www.apache.org/licenses/LICENSE-2.0
8#
9# Unless required by applicable law or agreed to in writing, software
10# distributed under the License is distributed on an "AS IS" BASIS,
11# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12# See the License for the specific language governing permissions and
13# limitations under the License.
14
Shawn O. Pearce438ee1c2008-11-03 09:59:36 -080015import errno
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -070016import filecmp
17import os
18import re
19import shutil
20import stat
21import sys
22import urllib2
23
24from color import Coloring
25from git_command import GitCommand
26from git_config import GitConfig, IsId
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -070027from error import GitError, ImportError, UploadError
Shawn O. Pearce559b8462009-03-02 12:56:08 -080028from error import ManifestInvalidRevisionError
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -070029from remote import Remote
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -070030
31HEAD = 'HEAD'
32R_HEADS = 'refs/heads/'
33R_TAGS = 'refs/tags/'
34R_PUB = 'refs/published/'
35R_M = 'refs/remotes/m/'
36
Shawn O. Pearce48244782009-04-16 08:25:57 -070037def _error(fmt, *args):
38 msg = fmt % args
39 print >>sys.stderr, 'error: %s' % msg
40
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -070041def not_rev(r):
42 return '^' + r
43
Shawn O. Pearceb54a3922009-01-05 16:18:58 -080044def sq(r):
45 return "'" + r.replace("'", "'\''") + "'"
Shawn O. Pearcec9ef7442008-11-03 10:32:09 -080046
47hook_list = None
48def repo_hooks():
49 global hook_list
50 if hook_list is None:
51 d = os.path.abspath(os.path.dirname(__file__))
52 d = os.path.join(d , 'hooks')
53 hook_list = map(lambda x: os.path.join(d, x), os.listdir(d))
54 return hook_list
55
56def relpath(dst, src):
57 src = os.path.dirname(src)
58 top = os.path.commonprefix([dst, src])
59 if top.endswith('/'):
60 top = top[:-1]
61 else:
62 top = os.path.dirname(top)
63
64 tmp = src
65 rel = ''
66 while top != tmp:
67 rel += '../'
68 tmp = os.path.dirname(tmp)
69 return rel + dst[len(top) + 1:]
70
71
Shawn O. Pearce632768b2008-10-23 11:58:52 -070072class DownloadedChange(object):
73 _commit_cache = None
74
75 def __init__(self, project, base, change_id, ps_id, commit):
76 self.project = project
77 self.base = base
78 self.change_id = change_id
79 self.ps_id = ps_id
80 self.commit = commit
81
82 @property
83 def commits(self):
84 if self._commit_cache is None:
85 self._commit_cache = self.project.bare_git.rev_list(
86 '--abbrev=8',
87 '--abbrev-commit',
88 '--pretty=oneline',
89 '--reverse',
90 '--date-order',
91 not_rev(self.base),
92 self.commit,
93 '--')
94 return self._commit_cache
95
96
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -070097class ReviewableBranch(object):
98 _commit_cache = None
99
100 def __init__(self, project, branch, base):
101 self.project = project
102 self.branch = branch
103 self.base = base
Shawn O. Pearcec99883f2008-11-11 17:12:43 -0800104 self.replace_changes = None
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -0700105
106 @property
107 def name(self):
108 return self.branch.name
109
110 @property
111 def commits(self):
112 if self._commit_cache is None:
113 self._commit_cache = self.project.bare_git.rev_list(
114 '--abbrev=8',
115 '--abbrev-commit',
116 '--pretty=oneline',
117 '--reverse',
118 '--date-order',
119 not_rev(self.base),
120 R_HEADS + self.name,
121 '--')
122 return self._commit_cache
123
124 @property
Shawn O. Pearcec99883f2008-11-11 17:12:43 -0800125 def unabbrev_commits(self):
126 r = dict()
127 for commit in self.project.bare_git.rev_list(
128 not_rev(self.base),
129 R_HEADS + self.name,
130 '--'):
131 r[commit[0:8]] = commit
132 return r
133
134 @property
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -0700135 def date(self):
136 return self.project.bare_git.log(
137 '--pretty=format:%cd',
138 '-n', '1',
139 R_HEADS + self.name,
140 '--')
141
Joe Onorato2896a792008-11-17 16:56:36 -0500142 def UploadForReview(self, people):
Shawn O. Pearcec99883f2008-11-11 17:12:43 -0800143 self.project.UploadForReview(self.name,
Joe Onorato2896a792008-11-17 16:56:36 -0500144 self.replace_changes,
145 people)
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -0700146
147 @property
148 def tip_url(self):
149 me = self.project.GetBranch(self.name)
150 commit = self.project.bare_git.rev_parse(R_HEADS + self.name)
151 return 'http://%s/r/%s' % (me.remote.review, commit[0:12])
152
Shawn O. Pearce0758d2f2008-10-22 13:13:40 -0700153 @property
154 def owner_email(self):
155 return self.project.UserEmail
156
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -0700157
158class StatusColoring(Coloring):
159 def __init__(self, config):
160 Coloring.__init__(self, config, 'status')
161 self.project = self.printer('header', attr = 'bold')
162 self.branch = self.printer('header', attr = 'bold')
163 self.nobranch = self.printer('nobranch', fg = 'red')
164
165 self.added = self.printer('added', fg = 'green')
166 self.changed = self.printer('changed', fg = 'red')
167 self.untracked = self.printer('untracked', fg = 'red')
168
169
170class DiffColoring(Coloring):
171 def __init__(self, config):
172 Coloring.__init__(self, config, 'diff')
173 self.project = self.printer('header', attr = 'bold')
174
175
176class _CopyFile:
Shawn O. Pearcec7a4eef2009-03-05 10:32:38 -0800177 def __init__(self, src, dest, abssrc, absdest):
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -0700178 self.src = src
179 self.dest = dest
Shawn O. Pearcec7a4eef2009-03-05 10:32:38 -0800180 self.abs_src = abssrc
181 self.abs_dest = absdest
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -0700182
183 def _Copy(self):
Shawn O. Pearcec7a4eef2009-03-05 10:32:38 -0800184 src = self.abs_src
185 dest = self.abs_dest
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -0700186 # copy file if it does not exist or is out of date
187 if not os.path.exists(dest) or not filecmp.cmp(src, dest):
188 try:
189 # remove existing file first, since it might be read-only
190 if os.path.exists(dest):
191 os.remove(dest)
192 shutil.copy(src, dest)
193 # make the file read-only
194 mode = os.stat(dest)[stat.ST_MODE]
195 mode = mode & ~(stat.S_IWUSR | stat.S_IWGRP | stat.S_IWOTH)
196 os.chmod(dest, mode)
197 except IOError:
Shawn O. Pearce48244782009-04-16 08:25:57 -0700198 _error('Cannot copy file %s to %s', src, dest)
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -0700199
200
201class Project(object):
202 def __init__(self,
203 manifest,
204 name,
205 remote,
206 gitdir,
207 worktree,
208 relpath,
209 revision):
210 self.manifest = manifest
211 self.name = name
212 self.remote = remote
213 self.gitdir = gitdir
214 self.worktree = worktree
215 self.relpath = relpath
216 self.revision = revision
217 self.snapshots = {}
218 self.extraRemotes = {}
219 self.copyfiles = []
220 self.config = GitConfig.ForRepository(
221 gitdir = self.gitdir,
222 defaults = self.manifest.globalConfig)
223
Shawn O. Pearcee284ad12008-11-04 07:37:10 -0800224 if self.worktree:
225 self.work_git = self._GitGetByExec(self, bare=False)
226 else:
227 self.work_git = None
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -0700228 self.bare_git = self._GitGetByExec(self, bare=True)
229
230 @property
231 def Exists(self):
232 return os.path.isdir(self.gitdir)
233
234 @property
235 def CurrentBranch(self):
236 """Obtain the name of the currently checked out branch.
237 The branch name omits the 'refs/heads/' prefix.
238 None is returned if the project is on a detached HEAD.
239 """
240 try:
241 b = self.work_git.GetHead()
242 except GitError:
243 return None
244 if b.startswith(R_HEADS):
245 return b[len(R_HEADS):]
246 return None
247
248 def IsDirty(self, consider_untracked=True):
249 """Is the working directory modified in some way?
250 """
251 self.work_git.update_index('-q',
252 '--unmerged',
253 '--ignore-missing',
254 '--refresh')
255 if self.work_git.DiffZ('diff-index','-M','--cached',HEAD):
256 return True
257 if self.work_git.DiffZ('diff-files'):
258 return True
259 if consider_untracked and self.work_git.LsOthers():
260 return True
261 return False
262
263 _userident_name = None
264 _userident_email = None
265
266 @property
267 def UserName(self):
268 """Obtain the user's personal name.
269 """
270 if self._userident_name is None:
271 self._LoadUserIdentity()
272 return self._userident_name
273
274 @property
275 def UserEmail(self):
276 """Obtain the user's email address. This is very likely
277 to be their Gerrit login.
278 """
279 if self._userident_email is None:
280 self._LoadUserIdentity()
281 return self._userident_email
282
283 def _LoadUserIdentity(self):
284 u = self.bare_git.var('GIT_COMMITTER_IDENT')
285 m = re.compile("^(.*) <([^>]*)> ").match(u)
286 if m:
287 self._userident_name = m.group(1)
288 self._userident_email = m.group(2)
289 else:
290 self._userident_name = ''
291 self._userident_email = ''
292
293 def GetRemote(self, name):
294 """Get the configuration for a single remote.
295 """
296 return self.config.GetRemote(name)
297
298 def GetBranch(self, name):
299 """Get the configuration for a single branch.
300 """
301 return self.config.GetBranch(name)
302
Shawn O. Pearce27b07322009-04-10 16:02:48 -0700303 def GetBranches(self):
304 """Get all existing local branches.
305 """
306 current = self.CurrentBranch
307 all = self.bare_git.ListRefs()
308 heads = {}
309 pubd = {}
310
311 for name, id in all.iteritems():
312 if name.startswith(R_HEADS):
313 name = name[len(R_HEADS):]
314 b = self.GetBranch(name)
315 b.current = name == current
316 b.published = None
317 b.revision = id
318 heads[name] = b
319
320 for name, id in all.iteritems():
321 if name.startswith(R_PUB):
322 name = name[len(R_PUB):]
323 b = heads.get(name)
324 if b:
325 b.published = id
326
327 return heads
328
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -0700329
330## Status Display ##
331
332 def PrintWorkTreeStatus(self):
333 """Prints the status of the repository to stdout.
334 """
335 if not os.path.isdir(self.worktree):
336 print ''
337 print 'project %s/' % self.relpath
338 print ' missing (run "repo sync")'
339 return
340
341 self.work_git.update_index('-q',
342 '--unmerged',
343 '--ignore-missing',
344 '--refresh')
345 di = self.work_git.DiffZ('diff-index', '-M', '--cached', HEAD)
346 df = self.work_git.DiffZ('diff-files')
347 do = self.work_git.LsOthers()
348 if not di and not df and not do:
Shawn O. Pearce161f4452009-04-10 17:41:44 -0700349 return 'CLEAN'
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -0700350
351 out = StatusColoring(self.config)
352 out.project('project %-40s', self.relpath + '/')
353
354 branch = self.CurrentBranch
355 if branch is None:
356 out.nobranch('(*** NO BRANCH ***)')
357 else:
358 out.branch('branch %s', branch)
359 out.nl()
360
361 paths = list()
362 paths.extend(di.keys())
363 paths.extend(df.keys())
364 paths.extend(do)
365
366 paths = list(set(paths))
367 paths.sort()
368
369 for p in paths:
370 try: i = di[p]
371 except KeyError: i = None
372
373 try: f = df[p]
374 except KeyError: f = None
375
376 if i: i_status = i.status.upper()
377 else: i_status = '-'
378
379 if f: f_status = f.status.lower()
380 else: f_status = '-'
381
382 if i and i.src_path:
Shawn O. Pearcefe086752009-03-03 13:49:48 -0800383 line = ' %s%s\t%s => %s (%s%%)' % (i_status, f_status,
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -0700384 i.src_path, p, i.level)
385 else:
386 line = ' %s%s\t%s' % (i_status, f_status, p)
387
388 if i and not f:
389 out.added('%s', line)
390 elif (i and f) or (not i and f):
391 out.changed('%s', line)
392 elif not i and not f:
393 out.untracked('%s', line)
394 else:
395 out.write('%s', line)
396 out.nl()
Shawn O. Pearce161f4452009-04-10 17:41:44 -0700397 return 'DIRTY'
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -0700398
399 def PrintWorkTreeDiff(self):
400 """Prints the status of the repository to stdout.
401 """
402 out = DiffColoring(self.config)
403 cmd = ['diff']
404 if out.is_on:
405 cmd.append('--color')
406 cmd.append(HEAD)
407 cmd.append('--')
408 p = GitCommand(self,
409 cmd,
410 capture_stdout = True,
411 capture_stderr = True)
412 has_diff = False
413 for line in p.process.stdout:
414 if not has_diff:
415 out.nl()
416 out.project('project %s/' % self.relpath)
417 out.nl()
418 has_diff = True
419 print line[:-1]
420 p.Wait()
421
422
423## Publish / Upload ##
424
425 def WasPublished(self, branch):
426 """Was the branch published (uploaded) for code review?
427 If so, returns the SHA-1 hash of the last published
428 state for the branch.
429 """
430 try:
431 return self.bare_git.rev_parse(R_PUB + branch)
432 except GitError:
433 return None
434
435 def CleanPublishedCache(self):
436 """Prunes any stale published refs.
437 """
438 heads = set()
439 canrm = {}
440 for name, id in self._allrefs.iteritems():
441 if name.startswith(R_HEADS):
442 heads.add(name)
443 elif name.startswith(R_PUB):
444 canrm[name] = id
445
446 for name, id in canrm.iteritems():
447 n = name[len(R_PUB):]
448 if R_HEADS + n not in heads:
449 self.bare_git.DeleteRef(name, id)
450
451 def GetUploadableBranches(self):
452 """List any branches which can be uploaded for review.
453 """
454 heads = {}
455 pubed = {}
456
457 for name, id in self._allrefs.iteritems():
458 if name.startswith(R_HEADS):
459 heads[name[len(R_HEADS):]] = id
460 elif name.startswith(R_PUB):
461 pubed[name[len(R_PUB):]] = id
462
463 ready = []
464 for branch, id in heads.iteritems():
465 if branch in pubed and pubed[branch] == id:
466 continue
467
Shawn O. Pearce35f25962008-11-11 17:03:13 -0800468 rb = self.GetUploadableBranch(branch)
469 if rb:
470 ready.append(rb)
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -0700471 return ready
472
Shawn O. Pearce35f25962008-11-11 17:03:13 -0800473 def GetUploadableBranch(self, branch_name):
474 """Get a single uploadable branch, or None.
475 """
476 branch = self.GetBranch(branch_name)
477 base = branch.LocalMerge
478 if branch.LocalMerge:
479 rb = ReviewableBranch(self, branch, base)
480 if rb.commits:
481 return rb
482 return None
483
Joe Onorato2896a792008-11-17 16:56:36 -0500484 def UploadForReview(self, branch=None, replace_changes=None, people=([],[])):
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -0700485 """Uploads the named branch for code review.
486 """
487 if branch is None:
488 branch = self.CurrentBranch
489 if branch is None:
490 raise GitError('not currently on a branch')
491
492 branch = self.GetBranch(branch)
493 if not branch.LocalMerge:
494 raise GitError('branch %s does not track a remote' % branch.name)
495 if not branch.remote.review:
496 raise GitError('remote %s has no review url' % branch.remote.name)
497
498 dest_branch = branch.merge
499 if not dest_branch.startswith(R_HEADS):
500 dest_branch = R_HEADS + dest_branch
501
Shawn O. Pearce339ba9f2008-11-06 09:52:51 -0800502 if not branch.remote.projectname:
503 branch.remote.projectname = self.name
504 branch.remote.Save()
505
Shawn O. Pearce370e3fa2009-01-26 10:55:39 -0800506 if branch.remote.ReviewProtocol == 'ssh':
Shawn O. Pearceb54a3922009-01-05 16:18:58 -0800507 if dest_branch.startswith(R_HEADS):
508 dest_branch = dest_branch[len(R_HEADS):]
509
510 rp = ['gerrit receive-pack']
511 for e in people[0]:
512 rp.append('--reviewer=%s' % sq(e))
513 for e in people[1]:
514 rp.append('--cc=%s' % sq(e))
515
516 cmd = ['push']
517 cmd.append('--receive-pack=%s' % " ".join(rp))
518 cmd.append(branch.remote.SshReviewUrl(self.UserEmail))
519 cmd.append('%s:refs/for/%s' % (R_HEADS + branch.name, dest_branch))
520 if replace_changes:
521 for change_id,commit_id in replace_changes.iteritems():
522 cmd.append('%s:refs/changes/%s/new' % (commit_id, change_id))
523 if GitCommand(self, cmd, bare = True).Wait() != 0:
524 raise UploadError('Upload failed')
525
526 else:
527 raise UploadError('Unsupported protocol %s' \
528 % branch.remote.review)
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -0700529
530 msg = "posted to %s for %s" % (branch.remote.review, dest_branch)
531 self.bare_git.UpdateRef(R_PUB + branch.name,
532 R_HEADS + branch.name,
533 message = msg)
534
535
536## Sync ##
537
538 def Sync_NetworkHalf(self):
539 """Perform only the network IO portion of the sync process.
540 Local working directory/branch state is not affected.
541 """
542 if not self.Exists:
543 print >>sys.stderr
544 print >>sys.stderr, 'Initializing project %s ...' % self.name
545 self._InitGitDir()
Shawn O. Pearcee284ad12008-11-04 07:37:10 -0800546
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -0700547 self._InitRemote()
548 for r in self.extraRemotes.values():
549 if not self._RemoteFetch(r.name):
550 return False
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -0700551 if not self._RemoteFetch():
552 return False
Shawn O. Pearcee284ad12008-11-04 07:37:10 -0800553
554 if self.worktree:
Shawn O. Pearcee284ad12008-11-04 07:37:10 -0800555 self._InitMRef()
556 else:
557 self._InitMirrorHead()
558 try:
559 os.remove(os.path.join(self.gitdir, 'FETCH_HEAD'))
560 except OSError:
561 pass
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -0700562 return True
Shawn O. Pearcec9ef7442008-11-03 10:32:09 -0800563
564 def PostRepoUpgrade(self):
565 self._InitHooks()
566
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -0700567 def _CopyFiles(self):
568 for file in self.copyfiles:
569 file._Copy()
570
Shawn O. Pearce350cde42009-04-16 11:21:18 -0700571 def Sync_LocalHalf(self, syncbuf):
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -0700572 """Perform only the local IO portion of the sync process.
573 Network access is not required.
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -0700574 """
575 self._InitWorkTree()
576 self.CleanPublishedCache()
577
578 rem = self.GetRemote(self.remote.name)
579 rev = rem.ToLocal(self.revision)
Shawn O. Pearce559b8462009-03-02 12:56:08 -0800580 try:
581 self.bare_git.rev_parse('--verify', '%s^0' % rev)
582 except GitError:
583 raise ManifestInvalidRevisionError(
584 'revision %s in %s not found' % (self.revision, self.name))
585
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -0700586 branch = self.CurrentBranch
587
Shawn O. Pearce350cde42009-04-16 11:21:18 -0700588 if branch is None or syncbuf.detach_head:
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -0700589 # Currently on a detached HEAD. The user is assumed to
590 # not have any local modifications worth worrying about.
591 #
Shawn O. Pearce350cde42009-04-16 11:21:18 -0700592 if os.path.exists(os.path.join(self.worktree, '.dotest')) \
593 or os.path.exists(os.path.join(self.worktree, '.git', 'rebase-apply')):
594 syncbuf.fail(self, _PriorSyncFailedError())
595 return
596
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -0700597 lost = self._revlist(not_rev(rev), HEAD)
598 if lost:
Shawn O. Pearce350cde42009-04-16 11:21:18 -0700599 syncbuf.info(self, "discarding %d commits", len(lost))
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -0700600 try:
601 self._Checkout(rev, quiet=True)
Shawn O. Pearce350cde42009-04-16 11:21:18 -0700602 except GitError, e:
603 syncbuf.fail(self, e)
604 return
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -0700605 self._CopyFiles()
Shawn O. Pearce350cde42009-04-16 11:21:18 -0700606 return
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -0700607
608 branch = self.GetBranch(branch)
609 merge = branch.LocalMerge
610
611 if not merge:
612 # The current branch has no tracking configuration.
613 # Jump off it to a deatched HEAD.
614 #
Shawn O. Pearce350cde42009-04-16 11:21:18 -0700615 syncbuf.info(self,
616 "leaving %s; does not track upstream",
617 branch.name)
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -0700618 try:
619 self._Checkout(rev, quiet=True)
Shawn O. Pearce350cde42009-04-16 11:21:18 -0700620 except GitError, e:
621 syncbuf.fail(self, e)
622 return
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -0700623 self._CopyFiles()
Shawn O. Pearce350cde42009-04-16 11:21:18 -0700624 return
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -0700625
626 upstream_gain = self._revlist(not_rev(HEAD), rev)
627 pub = self.WasPublished(branch.name)
628 if pub:
629 not_merged = self._revlist(not_rev(rev), pub)
630 if not_merged:
631 if upstream_gain:
632 # The user has published this branch and some of those
633 # commits are not yet merged upstream. We do not want
634 # to rewrite the published commits so we punt.
635 #
Shawn O. Pearce350cde42009-04-16 11:21:18 -0700636 syncbuf.info(self,
637 "branch %s is published but is now %d commits behind",
638 branch.name,
639 len(upstream_gain))
Shawn O. Pearce350cde42009-04-16 11:21:18 -0700640 return
Shawn O. Pearce23d77812008-10-30 11:06:57 -0700641 elif upstream_gain:
Shawn O. Pearcea54c5272008-10-30 11:03:00 -0700642 # We can fast-forward safely.
643 #
Shawn O. Pearce350cde42009-04-16 11:21:18 -0700644 def _doff():
Shawn O. Pearcea54c5272008-10-30 11:03:00 -0700645 self._FastForward(rev)
Shawn O. Pearce350cde42009-04-16 11:21:18 -0700646 self._CopyFiles()
647 syncbuf.later1(self, _doff)
648 return
Shawn O. Pearce23d77812008-10-30 11:06:57 -0700649 else:
650 # Trivially no changes in the upstream.
651 #
Shawn O. Pearce350cde42009-04-16 11:21:18 -0700652 return
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -0700653
654 if merge == rev:
655 try:
656 old_merge = self.bare_git.rev_parse('%s@{1}' % merge)
657 except GitError:
658 old_merge = merge
Shawn O. Pearce07346002008-10-21 07:09:27 -0700659 if old_merge == '0000000000000000000000000000000000000000' \
660 or old_merge == '':
661 old_merge = merge
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -0700662 else:
663 # The upstream switched on us. Time to cross our fingers
664 # and pray that the old upstream also wasn't in the habit
665 # of rebasing itself.
666 #
Shawn O. Pearce350cde42009-04-16 11:21:18 -0700667 syncbuf.info(self, "manifest switched %s...%s", merge, rev)
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -0700668 old_merge = merge
669
670 if rev == old_merge:
671 upstream_lost = []
672 else:
673 upstream_lost = self._revlist(not_rev(rev), old_merge)
674
675 if not upstream_lost and not upstream_gain:
676 # Trivially no changes caused by the upstream.
677 #
Shawn O. Pearce350cde42009-04-16 11:21:18 -0700678 return
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -0700679
680 if self.IsDirty(consider_untracked=False):
Shawn O. Pearce350cde42009-04-16 11:21:18 -0700681 syncbuf.fail(self, _DirtyError())
682 return
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -0700683
684 if upstream_lost:
685 # Upstream rebased. Not everything in HEAD
686 # may have been caused by the user.
687 #
Shawn O. Pearce350cde42009-04-16 11:21:18 -0700688 syncbuf.info(self,
689 "discarding %d commits removed from upstream",
690 len(upstream_lost))
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -0700691
692 branch.remote = rem
693 branch.merge = self.revision
694 branch.Save()
695
696 my_changes = self._revlist(not_rev(old_merge), HEAD)
697 if my_changes:
Shawn O. Pearce350cde42009-04-16 11:21:18 -0700698 def _dorebase():
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -0700699 self._Rebase(upstream = old_merge, onto = rev)
Shawn O. Pearce350cde42009-04-16 11:21:18 -0700700 self._CopyFiles()
701 syncbuf.later2(self, _dorebase)
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -0700702 elif upstream_lost:
703 try:
704 self._ResetHard(rev)
Shawn O. Pearce350cde42009-04-16 11:21:18 -0700705 self._CopyFiles()
706 except GitError, e:
707 syncbuf.fail(self, e)
708 return
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -0700709 else:
Shawn O. Pearce350cde42009-04-16 11:21:18 -0700710 def _doff():
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -0700711 self._FastForward(rev)
Shawn O. Pearce350cde42009-04-16 11:21:18 -0700712 self._CopyFiles()
713 syncbuf.later1(self, _doff)
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -0700714
Shawn O. Pearcec7a4eef2009-03-05 10:32:38 -0800715 def AddCopyFile(self, src, dest, absdest):
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -0700716 # dest should already be an absolute path, but src is project relative
717 # make src an absolute path
Shawn O. Pearcec7a4eef2009-03-05 10:32:38 -0800718 abssrc = os.path.join(self.worktree, src)
719 self.copyfiles.append(_CopyFile(src, dest, abssrc, absdest))
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -0700720
Shawn O. Pearce632768b2008-10-23 11:58:52 -0700721 def DownloadPatchSet(self, change_id, patch_id):
722 """Download a single patch set of a single change to FETCH_HEAD.
723 """
724 remote = self.GetRemote(self.remote.name)
725
726 cmd = ['fetch', remote.name]
727 cmd.append('refs/changes/%2.2d/%d/%d' \
728 % (change_id % 100, change_id, patch_id))
729 cmd.extend(map(lambda x: str(x), remote.fetch))
730 if GitCommand(self, cmd, bare=True).Wait() != 0:
731 return None
732 return DownloadedChange(self,
733 remote.ToLocal(self.revision),
734 change_id,
735 patch_id,
736 self.bare_git.rev_parse('FETCH_HEAD'))
737
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -0700738
739## Branch Management ##
740
741 def StartBranch(self, name):
742 """Create a new branch off the manifest's revision.
743 """
Shawn O. Pearce0a389e92009-04-10 16:21:18 -0700744 try:
745 self.bare_git.rev_parse(R_HEADS + name)
746 exists = True
747 except GitError:
748 exists = False;
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -0700749
Shawn O. Pearce0a389e92009-04-10 16:21:18 -0700750 if exists:
751 if name == self.CurrentBranch:
752 return True
753 else:
754 cmd = ['checkout', name, '--']
755 return GitCommand(self, cmd).Wait() == 0
756
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -0700757 else:
Shawn O. Pearce0a389e92009-04-10 16:21:18 -0700758 branch = self.GetBranch(name)
759 branch.remote = self.GetRemote(self.remote.name)
760 branch.merge = self.revision
761
762 rev = branch.LocalMerge
763 cmd = ['checkout', '-b', branch.name, rev]
764 if GitCommand(self, cmd).Wait() == 0:
765 branch.Save()
766 return True
767 else:
768 return False
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -0700769
Wink Saville02d79452009-04-10 13:01:24 -0700770 def CheckoutBranch(self, name):
771 """Checkout a local topic branch.
772 """
773
774 # Be sure the branch exists
775 try:
776 tip_rev = self.bare_git.rev_parse(R_HEADS + name)
777 except GitError:
778 return False;
779
780 # Do the checkout
781 cmd = ['checkout', name, '--']
Shawn O. Pearce2675c3f2009-04-10 16:20:25 -0700782 return GitCommand(self, cmd).Wait() == 0
Wink Saville02d79452009-04-10 13:01:24 -0700783
Shawn O. Pearce9fa44db2008-11-03 11:24:59 -0800784 def AbandonBranch(self, name):
785 """Destroy a local topic branch.
786 """
787 try:
788 tip_rev = self.bare_git.rev_parse(R_HEADS + name)
789 except GitError:
790 return
791
792 if self.CurrentBranch == name:
793 self._Checkout(
794 self.GetRemote(self.remote.name).ToLocal(self.revision),
795 quiet=True)
796
797 cmd = ['branch', '-D', name]
798 GitCommand(self, cmd, capture_stdout=True).Wait()
799
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -0700800 def PruneHeads(self):
801 """Prune any topic branches already merged into upstream.
802 """
803 cb = self.CurrentBranch
804 kill = []
Shawn O. Pearce3778f9d2009-03-02 12:30:50 -0800805 left = self._allrefs
806 for name in left.keys():
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -0700807 if name.startswith(R_HEADS):
808 name = name[len(R_HEADS):]
809 if cb is None or name != cb:
810 kill.append(name)
811
812 rev = self.GetRemote(self.remote.name).ToLocal(self.revision)
813 if cb is not None \
814 and not self._revlist(HEAD + '...' + rev) \
815 and not self.IsDirty(consider_untracked = False):
816 self.work_git.DetachHead(HEAD)
817 kill.append(cb)
818
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -0700819 if kill:
820 try:
821 old = self.bare_git.GetHead()
822 except GitError:
823 old = 'refs/heads/please_never_use_this_as_a_branch_name'
824
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -0700825 try:
826 self.bare_git.DetachHead(rev)
827
828 b = ['branch', '-d']
829 b.extend(kill)
830 b = GitCommand(self, b, bare=True,
831 capture_stdout=True,
832 capture_stderr=True)
833 b.Wait()
834 finally:
835 self.bare_git.SetHead(old)
Shawn O. Pearce3778f9d2009-03-02 12:30:50 -0800836 left = self._allrefs
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -0700837
Shawn O. Pearce3778f9d2009-03-02 12:30:50 -0800838 for branch in kill:
839 if (R_HEADS + branch) not in left:
840 self.CleanPublishedCache()
841 break
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -0700842
843 if cb and cb not in kill:
844 kill.append(cb)
Shawn O. Pearce7c6c64d2009-03-02 12:38:13 -0800845 kill.sort()
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -0700846
847 kept = []
848 for branch in kill:
Shawn O. Pearce3778f9d2009-03-02 12:30:50 -0800849 if (R_HEADS + branch) in left:
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -0700850 branch = self.GetBranch(branch)
851 base = branch.LocalMerge
852 if not base:
853 base = rev
854 kept.append(ReviewableBranch(self, branch, base))
855 return kept
856
857
858## Direct Git Commands ##
859
860 def _RemoteFetch(self, name=None):
861 if not name:
862 name = self.remote.name
Shawn O. Pearcee284ad12008-11-04 07:37:10 -0800863 cmd = ['fetch']
864 if not self.worktree:
865 cmd.append('--update-head-ok')
866 cmd.append(name)
867 return GitCommand(self, cmd, bare = True).Wait() == 0
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -0700868
869 def _Checkout(self, rev, quiet=False):
870 cmd = ['checkout']
871 if quiet:
872 cmd.append('-q')
873 cmd.append(rev)
874 cmd.append('--')
875 if GitCommand(self, cmd).Wait() != 0:
876 if self._allrefs:
877 raise GitError('%s checkout %s ' % (self.name, rev))
878
879 def _ResetHard(self, rev, quiet=True):
880 cmd = ['reset', '--hard']
881 if quiet:
882 cmd.append('-q')
883 cmd.append(rev)
884 if GitCommand(self, cmd).Wait() != 0:
885 raise GitError('%s reset --hard %s ' % (self.name, rev))
886
887 def _Rebase(self, upstream, onto = None):
Shawn O. Pearce19a83d82009-04-16 08:14:26 -0700888 cmd = ['rebase']
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -0700889 if onto is not None:
890 cmd.extend(['--onto', onto])
891 cmd.append(upstream)
Shawn O. Pearce19a83d82009-04-16 08:14:26 -0700892 if GitCommand(self, cmd).Wait() != 0:
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -0700893 raise GitError('%s rebase %s ' % (self.name, upstream))
894
895 def _FastForward(self, head):
896 cmd = ['merge', head]
897 if GitCommand(self, cmd).Wait() != 0:
898 raise GitError('%s merge %s ' % (self.name, head))
899
900 def _InitGitDir(self):
901 if not os.path.exists(self.gitdir):
902 os.makedirs(self.gitdir)
903 self.bare_git.init()
Shawn O. Pearce2816d4f2009-03-03 17:53:18 -0800904
905 if self.manifest.IsMirror:
906 self.config.SetString('core.bare', 'true')
907 else:
908 self.config.SetString('core.bare', None)
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -0700909
910 hooks = self._gitdir_path('hooks')
Shawn O. Pearcede646812008-10-29 14:38:12 -0700911 try:
912 to_rm = os.listdir(hooks)
913 except OSError:
914 to_rm = []
915 for old_hook in to_rm:
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -0700916 os.remove(os.path.join(hooks, old_hook))
Shawn O. Pearcec9ef7442008-11-03 10:32:09 -0800917 self._InitHooks()
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -0700918
919 m = self.manifest.manifestProject.config
920 for key in ['user.name', 'user.email']:
921 if m.Has(key, include_defaults = False):
922 self.config.SetString(key, m.GetString(key))
923
Shawn O. Pearcec9ef7442008-11-03 10:32:09 -0800924 def _InitHooks(self):
925 hooks = self._gitdir_path('hooks')
926 if not os.path.exists(hooks):
927 os.makedirs(hooks)
928 for stock_hook in repo_hooks():
929 dst = os.path.join(hooks, os.path.basename(stock_hook))
930 try:
931 os.symlink(relpath(stock_hook, dst), dst)
932 except OSError, e:
933 if e.errno == errno.EEXIST:
934 pass
935 elif e.errno == errno.EPERM:
936 raise GitError('filesystem must support symlinks')
937 else:
938 raise
939
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -0700940 def _InitRemote(self):
941 if self.remote.fetchUrl:
942 remote = self.GetRemote(self.remote.name)
943
944 url = self.remote.fetchUrl
945 while url.endswith('/'):
946 url = url[:-1]
947 url += '/%s.git' % self.name
948 remote.url = url
949 remote.review = self.remote.reviewUrl
Shawn O. Pearce339ba9f2008-11-06 09:52:51 -0800950 if remote.projectname is None:
951 remote.projectname = self.name
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -0700952
Shawn O. Pearcee284ad12008-11-04 07:37:10 -0800953 if self.worktree:
954 remote.ResetFetch(mirror=False)
955 else:
956 remote.ResetFetch(mirror=True)
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -0700957 remote.Save()
958
959 for r in self.extraRemotes.values():
960 remote = self.GetRemote(r.name)
961 remote.url = r.fetchUrl
962 remote.review = r.reviewUrl
Shawn O. Pearceae6e0942008-11-06 10:25:35 -0800963 if r.projectName:
964 remote.projectname = r.projectName
965 elif remote.projectname is None:
Shawn O. Pearce339ba9f2008-11-06 09:52:51 -0800966 remote.projectname = self.name
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -0700967 remote.ResetFetch()
968 remote.Save()
969
970 def _InitMRef(self):
971 if self.manifest.branch:
972 msg = 'manifest set to %s' % self.revision
973 ref = R_M + self.manifest.branch
974
975 if IsId(self.revision):
Marcelo E. Magallon21f73852008-12-31 04:44:37 +0000976 dst = self.revision + '^0'
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -0700977 self.bare_git.UpdateRef(ref, dst, message = msg, detach = True)
978 else:
979 remote = self.GetRemote(self.remote.name)
980 dst = remote.ToLocal(self.revision)
981 self.bare_git.symbolic_ref('-m', msg, ref, dst)
982
Shawn O. Pearcee284ad12008-11-04 07:37:10 -0800983 def _InitMirrorHead(self):
984 dst = self.GetRemote(self.remote.name).ToLocal(self.revision)
985 msg = 'manifest set to %s' % self.revision
986 self.bare_git.SetHead(dst, message=msg)
987
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -0700988 def _InitWorkTree(self):
989 dotgit = os.path.join(self.worktree, '.git')
990 if not os.path.exists(dotgit):
991 os.makedirs(dotgit)
992
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -0700993 for name in ['config',
994 'description',
995 'hooks',
996 'info',
997 'logs',
998 'objects',
999 'packed-refs',
1000 'refs',
1001 'rr-cache',
1002 'svn']:
Shawn O. Pearce438ee1c2008-11-03 09:59:36 -08001003 try:
Shawn O. Pearcec9ef7442008-11-03 10:32:09 -08001004 src = os.path.join(self.gitdir, name)
1005 dst = os.path.join(dotgit, name)
1006 os.symlink(relpath(src, dst), dst)
Shawn O. Pearce438ee1c2008-11-03 09:59:36 -08001007 except OSError, e:
1008 if e.errno == errno.EPERM:
1009 raise GitError('filesystem must support symlinks')
1010 else:
1011 raise
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -07001012
1013 rev = self.GetRemote(self.remote.name).ToLocal(self.revision)
1014 rev = self.bare_git.rev_parse('%s^0' % rev)
1015
1016 f = open(os.path.join(dotgit, HEAD), 'wb')
1017 f.write("%s\n" % rev)
1018 f.close()
1019
1020 cmd = ['read-tree', '--reset', '-u']
1021 cmd.append('-v')
1022 cmd.append('HEAD')
1023 if GitCommand(self, cmd).Wait() != 0:
1024 raise GitError("cannot initialize work tree")
1025
1026 def _gitdir_path(self, path):
1027 return os.path.join(self.gitdir, path)
1028
1029 def _revlist(self, *args):
1030 cmd = []
1031 cmd.extend(args)
1032 cmd.append('--')
1033 return self.work_git.rev_list(*args)
1034
1035 @property
1036 def _allrefs(self):
1037 return self.bare_git.ListRefs()
1038
1039 class _GitGetByExec(object):
1040 def __init__(self, project, bare):
1041 self._project = project
1042 self._bare = bare
1043
1044 def ListRefs(self, *args):
1045 cmdv = ['for-each-ref', '--format=%(objectname) %(refname)']
1046 cmdv.extend(args)
1047 p = GitCommand(self._project,
1048 cmdv,
1049 bare = self._bare,
1050 capture_stdout = True,
1051 capture_stderr = True)
1052 r = {}
1053 for line in p.process.stdout:
1054 id, name = line[:-1].split(' ', 2)
1055 r[name] = id
1056 if p.Wait() != 0:
1057 raise GitError('%s for-each-ref %s: %s' % (
1058 self._project.name,
1059 str(args),
1060 p.stderr))
1061 return r
1062
1063 def LsOthers(self):
1064 p = GitCommand(self._project,
1065 ['ls-files',
1066 '-z',
1067 '--others',
1068 '--exclude-standard'],
1069 bare = False,
1070 capture_stdout = True,
1071 capture_stderr = True)
1072 if p.Wait() == 0:
1073 out = p.stdout
1074 if out:
1075 return out[:-1].split("\0")
1076 return []
1077
1078 def DiffZ(self, name, *args):
1079 cmd = [name]
1080 cmd.append('-z')
1081 cmd.extend(args)
1082 p = GitCommand(self._project,
1083 cmd,
1084 bare = False,
1085 capture_stdout = True,
1086 capture_stderr = True)
1087 try:
1088 out = p.process.stdout.read()
1089 r = {}
1090 if out:
1091 out = iter(out[:-1].split('\0'))
1092 while out:
Shawn O. Pearce02dbb6d2008-10-21 13:59:08 -07001093 try:
1094 info = out.next()
1095 path = out.next()
1096 except StopIteration:
1097 break
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -07001098
1099 class _Info(object):
1100 def __init__(self, path, omode, nmode, oid, nid, state):
1101 self.path = path
1102 self.src_path = None
1103 self.old_mode = omode
1104 self.new_mode = nmode
1105 self.old_id = oid
1106 self.new_id = nid
1107
1108 if len(state) == 1:
1109 self.status = state
1110 self.level = None
1111 else:
1112 self.status = state[:1]
1113 self.level = state[1:]
1114 while self.level.startswith('0'):
1115 self.level = self.level[1:]
1116
1117 info = info[1:].split(' ')
1118 info =_Info(path, *info)
1119 if info.status in ('R', 'C'):
1120 info.src_path = info.path
1121 info.path = out.next()
1122 r[info.path] = info
1123 return r
1124 finally:
1125 p.Wait()
1126
1127 def GetHead(self):
1128 return self.symbolic_ref(HEAD)
1129
1130 def SetHead(self, ref, message=None):
1131 cmdv = []
1132 if message is not None:
1133 cmdv.extend(['-m', message])
1134 cmdv.append(HEAD)
1135 cmdv.append(ref)
1136 self.symbolic_ref(*cmdv)
1137
1138 def DetachHead(self, new, message=None):
1139 cmdv = ['--no-deref']
1140 if message is not None:
1141 cmdv.extend(['-m', message])
1142 cmdv.append(HEAD)
1143 cmdv.append(new)
1144 self.update_ref(*cmdv)
1145
1146 def UpdateRef(self, name, new, old=None,
1147 message=None,
1148 detach=False):
1149 cmdv = []
1150 if message is not None:
1151 cmdv.extend(['-m', message])
1152 if detach:
1153 cmdv.append('--no-deref')
1154 cmdv.append(name)
1155 cmdv.append(new)
1156 if old is not None:
1157 cmdv.append(old)
1158 self.update_ref(*cmdv)
1159
1160 def DeleteRef(self, name, old=None):
1161 if not old:
1162 old = self.rev_parse(name)
1163 self.update_ref('-d', name, old)
1164
1165 def rev_list(self, *args):
1166 cmdv = ['rev-list']
1167 cmdv.extend(args)
1168 p = GitCommand(self._project,
1169 cmdv,
1170 bare = self._bare,
1171 capture_stdout = True,
1172 capture_stderr = True)
1173 r = []
1174 for line in p.process.stdout:
1175 r.append(line[:-1])
1176 if p.Wait() != 0:
1177 raise GitError('%s rev-list %s: %s' % (
1178 self._project.name,
1179 str(args),
1180 p.stderr))
1181 return r
1182
1183 def __getattr__(self, name):
1184 name = name.replace('_', '-')
1185 def runner(*args):
1186 cmdv = [name]
1187 cmdv.extend(args)
1188 p = GitCommand(self._project,
1189 cmdv,
1190 bare = self._bare,
1191 capture_stdout = True,
1192 capture_stderr = True)
1193 if p.Wait() != 0:
1194 raise GitError('%s %s: %s' % (
1195 self._project.name,
1196 name,
1197 p.stderr))
1198 r = p.stdout
1199 if r.endswith('\n') and r.index('\n') == len(r) - 1:
1200 return r[:-1]
1201 return r
1202 return runner
1203
1204
Shawn O. Pearce350cde42009-04-16 11:21:18 -07001205class _PriorSyncFailedError(Exception):
1206 def __str__(self):
1207 return 'prior sync failed; rebase still in progress'
1208
1209class _DirtyError(Exception):
1210 def __str__(self):
1211 return 'contains uncommitted changes'
1212
1213class _InfoMessage(object):
1214 def __init__(self, project, text):
1215 self.project = project
1216 self.text = text
1217
1218 def Print(self, syncbuf):
1219 syncbuf.out.info('%s/: %s', self.project.relpath, self.text)
1220 syncbuf.out.nl()
1221
1222class _Failure(object):
1223 def __init__(self, project, why):
1224 self.project = project
1225 self.why = why
1226
1227 def Print(self, syncbuf):
1228 syncbuf.out.fail('error: %s/: %s',
1229 self.project.relpath,
1230 str(self.why))
1231 syncbuf.out.nl()
1232
1233class _Later(object):
1234 def __init__(self, project, action):
1235 self.project = project
1236 self.action = action
1237
1238 def Run(self, syncbuf):
1239 out = syncbuf.out
1240 out.project('project %s/', self.project.relpath)
1241 out.nl()
1242 try:
1243 self.action()
1244 out.nl()
1245 return True
1246 except GitError, e:
1247 out.nl()
1248 return False
1249
1250class _SyncColoring(Coloring):
1251 def __init__(self, config):
1252 Coloring.__init__(self, config, 'reposync')
1253 self.project = self.printer('header', attr = 'bold')
1254 self.info = self.printer('info')
1255 self.fail = self.printer('fail', fg='red')
1256
1257class SyncBuffer(object):
1258 def __init__(self, config, detach_head=False):
1259 self._messages = []
1260 self._failures = []
1261 self._later_queue1 = []
1262 self._later_queue2 = []
1263
1264 self.out = _SyncColoring(config)
1265 self.out.redirect(sys.stderr)
1266
1267 self.detach_head = detach_head
1268 self.clean = True
1269
1270 def info(self, project, fmt, *args):
1271 self._messages.append(_InfoMessage(project, fmt % args))
1272
1273 def fail(self, project, err=None):
1274 self._failures.append(_Failure(project, err))
1275 self.clean = False
1276
1277 def later1(self, project, what):
1278 self._later_queue1.append(_Later(project, what))
1279
1280 def later2(self, project, what):
1281 self._later_queue2.append(_Later(project, what))
1282
1283 def Finish(self):
1284 self._PrintMessages()
1285 self._RunLater()
1286 self._PrintMessages()
1287 return self.clean
1288
1289 def _RunLater(self):
1290 for q in ['_later_queue1', '_later_queue2']:
1291 if not self._RunQueue(q):
1292 return
1293
1294 def _RunQueue(self, queue):
1295 for m in getattr(self, queue):
1296 if not m.Run(self):
1297 self.clean = False
1298 return False
1299 setattr(self, queue, [])
1300 return True
1301
1302 def _PrintMessages(self):
1303 for m in self._messages:
1304 m.Print(self)
1305 for m in self._failures:
1306 m.Print(self)
1307
1308 self._messages = []
1309 self._failures = []
1310
1311
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -07001312class MetaProject(Project):
1313 """A special project housed under .repo.
1314 """
1315 def __init__(self, manifest, name, gitdir, worktree):
1316 repodir = manifest.repodir
1317 Project.__init__(self,
1318 manifest = manifest,
1319 name = name,
1320 gitdir = gitdir,
1321 worktree = worktree,
1322 remote = Remote('origin'),
1323 relpath = '.repo/%s' % name,
1324 revision = 'refs/heads/master')
1325
1326 def PreSync(self):
1327 if self.Exists:
1328 cb = self.CurrentBranch
1329 if cb:
1330 base = self.GetBranch(cb).merge
1331 if base:
1332 self.revision = base
1333
1334 @property
1335 def HasChanges(self):
1336 """Has the remote received new commits not yet checked out?
1337 """
1338 rev = self.GetRemote(self.remote.name).ToLocal(self.revision)
1339 if self._revlist(not_rev(HEAD), rev):
1340 return True
1341 return False