sync.py 6.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222
  1. #
  2. # Copyright (C) 2008 The Android Open Source Project
  3. #
  4. # Licensed under the Apache License, Version 2.0 (the "License");
  5. # you may not use this file except in compliance with the License.
  6. # You may obtain a copy of the License at
  7. #
  8. # http://www.apache.org/licenses/LICENSE-2.0
  9. #
  10. # Unless required by applicable law or agreed to in writing, software
  11. # distributed under the License is distributed on an "AS IS" BASIS,
  12. # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  13. # See the License for the specific language governing permissions and
  14. # limitations under the License.
  15. from optparse import SUPPRESS_HELP
  16. import os
  17. import re
  18. import subprocess
  19. import sys
  20. import time
  21. from git_command import GIT
  22. from project import HEAD
  23. from command import Command, MirrorSafeCommand
  24. from error import RepoChangedException, GitError
  25. from project import R_HEADS
  26. from project import SyncBuffer
  27. from progress import Progress
  28. class Sync(Command, MirrorSafeCommand):
  29. common = True
  30. helpSummary = "Update working tree to the latest revision"
  31. helpUsage = """
  32. %prog [<project>...]
  33. """
  34. helpDescription = """
  35. The '%prog' command synchronizes local project directories
  36. with the remote repositories specified in the manifest. If a local
  37. project does not yet exist, it will clone a new local directory from
  38. the remote repository and set up tracking branches as specified in
  39. the manifest. If the local project already exists, '%prog'
  40. will update the remote branches and rebase any new local changes
  41. on top of the new remote changes.
  42. '%prog' will synchronize all projects listed at the command
  43. line. Projects can be specified either by name, or by a relative
  44. or absolute path to the project's local directory. If no projects
  45. are specified, '%prog' will synchronize all projects listed in
  46. the manifest.
  47. The -d/--detach option can be used to switch specified projects
  48. back to the manifest revision. This option is especially helpful
  49. if the project is currently on a topic branch, but the manifest
  50. revision is temporarily needed.
  51. """
  52. def _Options(self, p):
  53. p.add_option('-l','--local-only',
  54. dest='local_only', action='store_true',
  55. help="only update working tree, don't fetch")
  56. p.add_option('-n','--network-only',
  57. dest='network_only', action='store_true',
  58. help="fetch only, don't update working tree")
  59. p.add_option('-d','--detach',
  60. dest='detach_head', action='store_true',
  61. help='detach projects back to manifest revision')
  62. g = p.add_option_group('repo Version options')
  63. g.add_option('--no-repo-verify',
  64. dest='no_repo_verify', action='store_true',
  65. help='do not verify repo source code')
  66. g.add_option('--repo-upgraded',
  67. dest='repo_upgraded', action='store_true',
  68. help=SUPPRESS_HELP)
  69. def _Fetch(self, projects):
  70. fetched = set()
  71. pm = Progress('Fetching projects', len(projects))
  72. for project in projects:
  73. pm.update()
  74. if project.Sync_NetworkHalf():
  75. fetched.add(project.gitdir)
  76. else:
  77. print >>sys.stderr, 'error: Cannot fetch %s' % project.name
  78. sys.exit(1)
  79. pm.end()
  80. return fetched
  81. def Execute(self, opt, args):
  82. if opt.network_only and opt.detach_head:
  83. print >>sys.stderr, 'error: cannot combine -n and -d'
  84. sys.exit(1)
  85. if opt.network_only and opt.local_only:
  86. print >>sys.stderr, 'error: cannot combine -n and -l'
  87. sys.exit(1)
  88. rp = self.manifest.repoProject
  89. rp.PreSync()
  90. mp = self.manifest.manifestProject
  91. mp.PreSync()
  92. if opt.repo_upgraded:
  93. _PostRepoUpgrade(self.manifest)
  94. all = self.GetProjects(args, missing_ok=True)
  95. if not opt.local_only:
  96. to_fetch = []
  97. now = time.time()
  98. if (24 * 60 * 60) <= (now - rp.LastFetch):
  99. to_fetch.append(rp)
  100. to_fetch.append(mp)
  101. to_fetch.extend(all)
  102. fetched = self._Fetch(to_fetch)
  103. _PostRepoFetch(rp, opt.no_repo_verify)
  104. if opt.network_only:
  105. # bail out now; the rest touches the working tree
  106. return
  107. if mp.HasChanges:
  108. syncbuf = SyncBuffer(mp.config)
  109. mp.Sync_LocalHalf(syncbuf)
  110. if not syncbuf.Finish():
  111. sys.exit(1)
  112. self.manifest._Unload()
  113. all = self.GetProjects(args, missing_ok=True)
  114. missing = []
  115. for project in all:
  116. if project.gitdir not in fetched:
  117. missing.append(project)
  118. self._Fetch(missing)
  119. syncbuf = SyncBuffer(mp.config,
  120. detach_head = opt.detach_head)
  121. pm = Progress('Syncing work tree', len(all))
  122. for project in all:
  123. pm.update()
  124. if project.worktree:
  125. project.Sync_LocalHalf(syncbuf)
  126. pm.end()
  127. print >>sys.stderr
  128. if not syncbuf.Finish():
  129. sys.exit(1)
  130. def _PostRepoUpgrade(manifest):
  131. for project in manifest.projects.values():
  132. if project.Exists:
  133. project.PostRepoUpgrade()
  134. def _PostRepoFetch(rp, no_repo_verify=False, verbose=False):
  135. if rp.HasChanges:
  136. print >>sys.stderr, 'info: A new version of repo is available'
  137. print >>sys.stderr, ''
  138. if no_repo_verify or _VerifyTag(rp):
  139. syncbuf = SyncBuffer(rp.config)
  140. rp.Sync_LocalHalf(syncbuf)
  141. if not syncbuf.Finish():
  142. sys.exit(1)
  143. print >>sys.stderr, 'info: Restarting repo with latest version'
  144. raise RepoChangedException(['--repo-upgraded'])
  145. else:
  146. print >>sys.stderr, 'warning: Skipped upgrade to unverified version'
  147. else:
  148. if verbose:
  149. print >>sys.stderr, 'repo version %s is current' % rp.work_git.describe(HEAD)
  150. def _VerifyTag(project):
  151. gpg_dir = os.path.expanduser('~/.repoconfig/gnupg')
  152. if not os.path.exists(gpg_dir):
  153. print >>sys.stderr,\
  154. """warning: GnuPG was not available during last "repo init"
  155. warning: Cannot automatically authenticate repo."""
  156. return True
  157. remote = project.GetRemote(project.remote.name)
  158. ref = remote.ToLocal(project.revision)
  159. try:
  160. cur = project.bare_git.describe(ref)
  161. except GitError:
  162. cur = None
  163. if not cur \
  164. or re.compile(r'^.*-[0-9]{1,}-g[0-9a-f]{1,}$').match(cur):
  165. rev = project.revision
  166. if rev.startswith(R_HEADS):
  167. rev = rev[len(R_HEADS):]
  168. print >>sys.stderr
  169. print >>sys.stderr,\
  170. "warning: project '%s' branch '%s' is not signed" \
  171. % (project.name, rev)
  172. return False
  173. env = dict(os.environ)
  174. env['GIT_DIR'] = project.gitdir
  175. env['GNUPGHOME'] = gpg_dir
  176. cmd = [GIT, 'tag', '-v', cur]
  177. proc = subprocess.Popen(cmd,
  178. stdout = subprocess.PIPE,
  179. stderr = subprocess.PIPE,
  180. env = env)
  181. out = proc.stdout.read()
  182. proc.stdout.close()
  183. err = proc.stderr.read()
  184. proc.stderr.close()
  185. if proc.wait() != 0:
  186. print >>sys.stderr
  187. print >>sys.stderr, out
  188. print >>sys.stderr, err
  189. print >>sys.stderr
  190. return False
  191. return True