The Android Open Source Project | cf31fe9 | 2008-10-21 07:00:00 -0700 | [diff] [blame] | 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 | |
| 16 | import os |
Doug Anderson | 2630dd9 | 2011-04-07 13:36:30 -0700 | [diff] [blame] | 17 | import shutil |
The Android Open Source Project | cf31fe9 | 2008-10-21 07:00:00 -0700 | [diff] [blame] | 18 | import sys |
| 19 | |
| 20 | from color import Coloring |
Shawn O. Pearce | c95583b | 2009-03-03 17:47:06 -0800 | [diff] [blame] | 21 | from command import InteractiveCommand, MirrorSafeCommand |
The Android Open Source Project | cf31fe9 | 2008-10-21 07:00:00 -0700 | [diff] [blame] | 22 | from error import ManifestParseError |
Shawn O. Pearce | 350cde4 | 2009-04-16 11:21:18 -0700 | [diff] [blame] | 23 | from project import SyncBuffer |
Shawn O. Pearce | f322b9a | 2011-09-19 14:50:58 -0700 | [diff] [blame] | 24 | from git_config import GitConfig |
Shawn O. Pearce | 2ec00b9 | 2009-06-12 09:32:50 -0700 | [diff] [blame] | 25 | from git_command import git_require, MIN_GIT_VERSION |
Victor Boivie | ee1c2f5 | 2011-04-19 10:50:12 +0200 | [diff] [blame] | 26 | from git_config import GitConfig |
The Android Open Source Project | cf31fe9 | 2008-10-21 07:00:00 -0700 | [diff] [blame] | 27 | |
Shawn O. Pearce | c95583b | 2009-03-03 17:47:06 -0800 | [diff] [blame] | 28 | class Init(InteractiveCommand, MirrorSafeCommand): |
The Android Open Source Project | cf31fe9 | 2008-10-21 07:00:00 -0700 | [diff] [blame] | 29 | common = True |
| 30 | helpSummary = "Initialize repo in the current directory" |
| 31 | helpUsage = """ |
| 32 | %prog [options] |
| 33 | """ |
| 34 | helpDescription = """ |
| 35 | The '%prog' command is run once to install and initialize repo. |
| 36 | The latest repo source code and manifest collection is downloaded |
| 37 | from the server and is installed in the .repo/ directory in the |
| 38 | current working directory. |
| 39 | |
Victor Boivie | ee1c2f5 | 2011-04-19 10:50:12 +0200 | [diff] [blame] | 40 | The optional -u argument can be used to specify a URL to the |
| 41 | manifest project. It is also possible to have a git configuration |
| 42 | section as below to use 'identifier' as argument to -u: |
| 43 | |
| 44 | [repo-manifest "identifier"] |
| 45 | url = ... |
| 46 | server = ... |
| 47 | reference = ... |
| 48 | |
| 49 | Only the url is required - the others are optional. |
| 50 | |
| 51 | If no -u argument is specified, the 'repo-manifest' section named |
| 52 | 'default' will be used if present. |
| 53 | |
Shawn O. Pearce | 77bb4af | 2009-04-18 11:33:32 -0700 | [diff] [blame] | 54 | The optional -b argument can be used to select the manifest branch |
| 55 | to checkout and use. If no branch is specified, master is assumed. |
| 56 | |
| 57 | The optional -m argument can be used to specify an alternate manifest |
| 58 | to be used. If no manifest is specified, the manifest default.xml |
| 59 | will be used. |
| 60 | |
Shawn O. Pearce | 8844338 | 2010-10-08 10:02:09 +0200 | [diff] [blame] | 61 | The --reference option can be used to point to a directory that |
| 62 | has the content of a --mirror sync. This will make the working |
| 63 | directory use as much data as possible from the local reference |
| 64 | directory when fetching from the server. This will make the sync |
| 65 | go a lot faster by reducing data traffic on the network. |
| 66 | |
| 67 | |
Shawn O. Pearce | 77bb4af | 2009-04-18 11:33:32 -0700 | [diff] [blame] | 68 | Switching Manifest Branches |
| 69 | --------------------------- |
| 70 | |
| 71 | To switch to another manifest branch, `repo init -b otherbranch` |
| 72 | may be used in an existing client. However, as this only updates the |
| 73 | manifest, a subsequent `repo sync` (or `repo sync -d`) is necessary |
| 74 | to update the working directory files. |
The Android Open Source Project | cf31fe9 | 2008-10-21 07:00:00 -0700 | [diff] [blame] | 75 | """ |
| 76 | |
| 77 | def _Options(self, p): |
| 78 | # Logging |
| 79 | g = p.add_option_group('Logging options') |
| 80 | g.add_option('-q', '--quiet', |
| 81 | dest="quiet", action="store_true", default=False, |
| 82 | help="be quiet") |
| 83 | |
| 84 | # Manifest |
| 85 | g = p.add_option_group('Manifest options') |
| 86 | g.add_option('-u', '--manifest-url', |
Victor Boivie | ee1c2f5 | 2011-04-19 10:50:12 +0200 | [diff] [blame] | 87 | dest='manifest_url', default='default', |
The Android Open Source Project | cf31fe9 | 2008-10-21 07:00:00 -0700 | [diff] [blame] | 88 | help='manifest repository location', metavar='URL') |
| 89 | g.add_option('-b', '--manifest-branch', |
| 90 | dest='manifest_branch', |
| 91 | help='manifest branch or revision', metavar='REVISION') |
| 92 | g.add_option('-m', '--manifest-name', |
| 93 | dest='manifest_name', default='default.xml', |
| 94 | help='initial manifest file', metavar='NAME.xml') |
Shawn O. Pearce | e284ad1 | 2008-11-04 07:37:10 -0800 | [diff] [blame] | 95 | g.add_option('--mirror', |
| 96 | dest='mirror', action='store_true', |
| 97 | help='mirror the forrest') |
Shawn O. Pearce | 8844338 | 2010-10-08 10:02:09 +0200 | [diff] [blame] | 98 | g.add_option('--reference', |
| 99 | dest='reference', |
| 100 | help='location of mirror directory', metavar='DIR') |
Doug Anderson | 30d4529 | 2011-05-04 15:01:04 -0700 | [diff] [blame] | 101 | g.add_option('--depth', type='int', default=None, |
| 102 | dest='depth', |
| 103 | help='create a shallow clone with given depth; see git clone') |
The Android Open Source Project | cf31fe9 | 2008-10-21 07:00:00 -0700 | [diff] [blame] | 104 | |
| 105 | # Tool |
Shawn O. Pearce | fd89b67 | 2009-04-18 11:28:57 -0700 | [diff] [blame] | 106 | g = p.add_option_group('repo Version options') |
The Android Open Source Project | cf31fe9 | 2008-10-21 07:00:00 -0700 | [diff] [blame] | 107 | g.add_option('--repo-url', |
| 108 | dest='repo_url', |
| 109 | help='repo repository location', metavar='URL') |
| 110 | g.add_option('--repo-branch', |
| 111 | dest='repo_branch', |
| 112 | help='repo branch or revision', metavar='REVISION') |
| 113 | g.add_option('--no-repo-verify', |
| 114 | dest='no_repo_verify', action='store_true', |
| 115 | help='do not verify repo source code') |
| 116 | |
Victor Boivie | 841be34 | 2011-04-05 11:31:10 +0200 | [diff] [blame] | 117 | # Other |
| 118 | g = p.add_option_group('Other options') |
| 119 | g.add_option('--config-name', |
| 120 | dest='config_name', action="store_true", default=False, |
| 121 | help='Always prompt for name/e-mail') |
| 122 | |
The Android Open Source Project | cf31fe9 | 2008-10-21 07:00:00 -0700 | [diff] [blame] | 123 | def _SyncManifest(self, opt): |
| 124 | m = self.manifest.manifestProject |
Shawn O. Pearce | 5470df6 | 2009-03-09 18:51:58 -0700 | [diff] [blame] | 125 | is_new = not m.Exists |
Victor Boivie | ee1c2f5 | 2011-04-19 10:50:12 +0200 | [diff] [blame] | 126 | manifest_server = None |
| 127 | |
| 128 | # The manifest url may point out a manifest section in the config |
| 129 | key = 'repo-manifest.%s.' % opt.manifest_url |
| 130 | if GitConfig.ForUser().GetString(key + 'url'): |
| 131 | opt.manifest_url = GitConfig.ForUser().GetString(key + 'url') |
| 132 | |
| 133 | # Also copy other options to the manifest config if not specified already. |
| 134 | if not opt.reference: |
| 135 | opt.reference = GitConfig.ForUser().GetString(key + 'reference') |
| 136 | manifest_server = GitConfig.ForUser().GetString(key + 'server') |
The Android Open Source Project | cf31fe9 | 2008-10-21 07:00:00 -0700 | [diff] [blame] | 137 | |
Shawn O. Pearce | 5470df6 | 2009-03-09 18:51:58 -0700 | [diff] [blame] | 138 | if is_new: |
Victor Boivie | ee1c2f5 | 2011-04-19 10:50:12 +0200 | [diff] [blame] | 139 | if not opt.manifest_url or opt.manifest_url == 'default': |
| 140 | print >>sys.stderr, """\ |
| 141 | fatal: missing manifest url (-u) and no default found. |
| 142 | |
| 143 | tip: The global git configuration key 'repo-manifest.default.url' can |
| 144 | be used to specify a default url.""" |
The Android Open Source Project | cf31fe9 | 2008-10-21 07:00:00 -0700 | [diff] [blame] | 145 | sys.exit(1) |
| 146 | |
| 147 | if not opt.quiet: |
Shawn O. Pearce | f322b9a | 2011-09-19 14:50:58 -0700 | [diff] [blame] | 148 | print >>sys.stderr, 'Get %s' \ |
| 149 | % GitConfig.ForUser().UrlInsteadOf(opt.manifest_url) |
The Android Open Source Project | cf31fe9 | 2008-10-21 07:00:00 -0700 | [diff] [blame] | 150 | m._InitGitDir() |
| 151 | |
| 152 | if opt.manifest_branch: |
Shawn O. Pearce | 3c8dea1 | 2009-05-29 18:38:17 -0700 | [diff] [blame] | 153 | m.revisionExpr = opt.manifest_branch |
The Android Open Source Project | cf31fe9 | 2008-10-21 07:00:00 -0700 | [diff] [blame] | 154 | else: |
Shawn O. Pearce | 3c8dea1 | 2009-05-29 18:38:17 -0700 | [diff] [blame] | 155 | m.revisionExpr = 'refs/heads/master' |
The Android Open Source Project | cf31fe9 | 2008-10-21 07:00:00 -0700 | [diff] [blame] | 156 | else: |
| 157 | if opt.manifest_branch: |
Shawn O. Pearce | 3c8dea1 | 2009-05-29 18:38:17 -0700 | [diff] [blame] | 158 | m.revisionExpr = opt.manifest_branch |
The Android Open Source Project | cf31fe9 | 2008-10-21 07:00:00 -0700 | [diff] [blame] | 159 | else: |
| 160 | m.PreSync() |
| 161 | |
| 162 | if opt.manifest_url: |
| 163 | r = m.GetRemote(m.remote.name) |
| 164 | r.url = opt.manifest_url |
| 165 | r.ResetFetch() |
| 166 | r.Save() |
| 167 | |
Victor Boivie | ee1c2f5 | 2011-04-19 10:50:12 +0200 | [diff] [blame] | 168 | if manifest_server: |
| 169 | m.config.SetString('repo.manifest-server', manifest_server) |
| 170 | |
Shawn O. Pearce | 8844338 | 2010-10-08 10:02:09 +0200 | [diff] [blame] | 171 | if opt.reference: |
| 172 | m.config.SetString('repo.reference', opt.reference) |
| 173 | |
Shawn O. Pearce | e284ad1 | 2008-11-04 07:37:10 -0800 | [diff] [blame] | 174 | if opt.mirror: |
Shawn O. Pearce | 5470df6 | 2009-03-09 18:51:58 -0700 | [diff] [blame] | 175 | if is_new: |
| 176 | m.config.SetString('repo.mirror', 'true') |
| 177 | else: |
| 178 | print >>sys.stderr, 'fatal: --mirror not supported on existing client' |
| 179 | sys.exit(1) |
Shawn O. Pearce | e284ad1 | 2008-11-04 07:37:10 -0800 | [diff] [blame] | 180 | |
Shawn O. Pearce | f322b9a | 2011-09-19 14:50:58 -0700 | [diff] [blame] | 181 | if not m.Sync_NetworkHalf(is_new=is_new): |
Shawn O. Pearce | 1fc99f4 | 2009-03-17 08:06:18 -0700 | [diff] [blame] | 182 | r = m.GetRemote(m.remote.name) |
| 183 | print >>sys.stderr, 'fatal: cannot obtain manifest %s' % r.url |
Doug Anderson | 2630dd9 | 2011-04-07 13:36:30 -0700 | [diff] [blame] | 184 | |
| 185 | # Better delete the manifest git dir if we created it; otherwise next |
| 186 | # time (when user fixes problems) we won't go through the "is_new" logic. |
| 187 | if is_new: |
| 188 | shutil.rmtree(m.gitdir) |
Shawn O. Pearce | 1fc99f4 | 2009-03-17 08:06:18 -0700 | [diff] [blame] | 189 | sys.exit(1) |
| 190 | |
Shawn O. Pearce | 350cde4 | 2009-04-16 11:21:18 -0700 | [diff] [blame] | 191 | syncbuf = SyncBuffer(m.config) |
| 192 | m.Sync_LocalHalf(syncbuf) |
| 193 | syncbuf.Finish() |
| 194 | |
Shawn O. Pearce | df01883 | 2009-03-17 08:15:27 -0700 | [diff] [blame] | 195 | if is_new or m.CurrentBranch is None: |
Shawn O. Pearce | 0a389e9 | 2009-04-10 16:21:18 -0700 | [diff] [blame] | 196 | if not m.StartBranch('default'): |
| 197 | print >>sys.stderr, 'fatal: cannot create default in manifest' |
| 198 | sys.exit(1) |
The Android Open Source Project | cf31fe9 | 2008-10-21 07:00:00 -0700 | [diff] [blame] | 199 | |
| 200 | def _LinkManifest(self, name): |
| 201 | if not name: |
| 202 | print >>sys.stderr, 'fatal: manifest name (-m) is required.' |
| 203 | sys.exit(1) |
| 204 | |
| 205 | try: |
| 206 | self.manifest.Link(name) |
| 207 | except ManifestParseError, e: |
| 208 | print >>sys.stderr, "fatal: manifest '%s' not available" % name |
| 209 | print >>sys.stderr, 'fatal: %s' % str(e) |
| 210 | sys.exit(1) |
| 211 | |
Shawn O. Pearce | 37dbf2b | 2009-07-02 10:53:04 -0700 | [diff] [blame] | 212 | def _Prompt(self, prompt, value): |
The Android Open Source Project | cf31fe9 | 2008-10-21 07:00:00 -0700 | [diff] [blame] | 213 | mp = self.manifest.manifestProject |
| 214 | |
| 215 | sys.stdout.write('%-10s [%s]: ' % (prompt, value)) |
| 216 | a = sys.stdin.readline().strip() |
Shawn O. Pearce | 37dbf2b | 2009-07-02 10:53:04 -0700 | [diff] [blame] | 217 | if a == '': |
| 218 | return value |
| 219 | return a |
The Android Open Source Project | cf31fe9 | 2008-10-21 07:00:00 -0700 | [diff] [blame] | 220 | |
Victor Boivie | 841be34 | 2011-04-05 11:31:10 +0200 | [diff] [blame] | 221 | def _ShouldConfigureUser(self): |
| 222 | gc = self.manifest.globalConfig |
| 223 | mp = self.manifest.manifestProject |
| 224 | |
| 225 | # If we don't have local settings, get from global. |
| 226 | if not mp.config.Has('user.name') or not mp.config.Has('user.email'): |
| 227 | if not gc.Has('user.name') or not gc.Has('user.email'): |
| 228 | return True |
| 229 | |
| 230 | mp.config.SetString('user.name', gc.GetString('user.name')) |
| 231 | mp.config.SetString('user.email', gc.GetString('user.email')) |
| 232 | |
| 233 | print '' |
| 234 | print 'Your identity is: %s <%s>' % (mp.config.GetString('user.name'), |
| 235 | mp.config.GetString('user.email')) |
| 236 | print 'If you want to change this, please re-run \'repo init\' with --config-name' |
| 237 | return False |
| 238 | |
The Android Open Source Project | cf31fe9 | 2008-10-21 07:00:00 -0700 | [diff] [blame] | 239 | def _ConfigureUser(self): |
| 240 | mp = self.manifest.manifestProject |
| 241 | |
Shawn O. Pearce | 37dbf2b | 2009-07-02 10:53:04 -0700 | [diff] [blame] | 242 | while True: |
| 243 | print '' |
| 244 | name = self._Prompt('Your Name', mp.UserName) |
| 245 | email = self._Prompt('Your Email', mp.UserEmail) |
| 246 | |
| 247 | print '' |
| 248 | print 'Your identity is: %s <%s>' % (name, email) |
Mike Frysinger | e931127 | 2011-08-11 15:46:43 -0400 | [diff] [blame] | 249 | sys.stdout.write('is this correct [y/N]? ') |
Nico Sallembien | 6d7508b | 2010-04-01 11:03:53 -0700 | [diff] [blame] | 250 | a = sys.stdin.readline().strip() |
| 251 | if a in ('yes', 'y', 't', 'true'): |
Shawn O. Pearce | 37dbf2b | 2009-07-02 10:53:04 -0700 | [diff] [blame] | 252 | break |
| 253 | |
| 254 | if name != mp.UserName: |
| 255 | mp.config.SetString('user.name', name) |
| 256 | if email != mp.UserEmail: |
| 257 | mp.config.SetString('user.email', email) |
The Android Open Source Project | cf31fe9 | 2008-10-21 07:00:00 -0700 | [diff] [blame] | 258 | |
| 259 | def _HasColorSet(self, gc): |
| 260 | for n in ['ui', 'diff', 'status']: |
| 261 | if gc.Has('color.%s' % n): |
| 262 | return True |
| 263 | return False |
| 264 | |
| 265 | def _ConfigureColor(self): |
| 266 | gc = self.manifest.globalConfig |
| 267 | if self._HasColorSet(gc): |
| 268 | return |
| 269 | |
| 270 | class _Test(Coloring): |
| 271 | def __init__(self): |
| 272 | Coloring.__init__(self, gc, 'test color display') |
| 273 | self._on = True |
| 274 | out = _Test() |
| 275 | |
| 276 | print '' |
| 277 | print "Testing colorized output (for 'repo diff', 'repo status'):" |
| 278 | |
| 279 | for c in ['black','red','green','yellow','blue','magenta','cyan']: |
| 280 | out.write(' ') |
| 281 | out.printer(fg=c)(' %-6s ', c) |
| 282 | out.write(' ') |
| 283 | out.printer(fg='white', bg='black')(' %s ' % 'white') |
| 284 | out.nl() |
| 285 | |
| 286 | for c in ['bold','dim','ul','reverse']: |
| 287 | out.write(' ') |
| 288 | out.printer(fg='black', attr=c)(' %-6s ', c) |
| 289 | out.nl() |
| 290 | |
Mike Frysinger | e931127 | 2011-08-11 15:46:43 -0400 | [diff] [blame] | 291 | sys.stdout.write('Enable color display in this user account (y/N)? ') |
The Android Open Source Project | cf31fe9 | 2008-10-21 07:00:00 -0700 | [diff] [blame] | 292 | a = sys.stdin.readline().strip().lower() |
| 293 | if a in ('y', 'yes', 't', 'true', 'on'): |
| 294 | gc.SetString('color.ui', 'auto') |
| 295 | |
Doug Anderson | 30d4529 | 2011-05-04 15:01:04 -0700 | [diff] [blame] | 296 | def _ConfigureDepth(self, opt): |
| 297 | """Configure the depth we'll sync down. |
| 298 | |
| 299 | Args: |
| 300 | opt: Options from optparse. We care about opt.depth. |
| 301 | """ |
| 302 | # Opt.depth will be non-None if user actually passed --depth to repo init. |
| 303 | if opt.depth is not None: |
| 304 | if opt.depth > 0: |
| 305 | # Positive values will set the depth. |
| 306 | depth = str(opt.depth) |
| 307 | else: |
| 308 | # Negative numbers will clear the depth; passing None to SetString |
| 309 | # will do that. |
| 310 | depth = None |
| 311 | |
| 312 | # We store the depth in the main manifest project. |
| 313 | self.manifest.manifestProject.config.SetString('repo.depth', depth) |
| 314 | |
The Android Open Source Project | cf31fe9 | 2008-10-21 07:00:00 -0700 | [diff] [blame] | 315 | def Execute(self, opt, args): |
Shawn O. Pearce | 2ec00b9 | 2009-06-12 09:32:50 -0700 | [diff] [blame] | 316 | git_require(MIN_GIT_VERSION, fail=True) |
The Android Open Source Project | cf31fe9 | 2008-10-21 07:00:00 -0700 | [diff] [blame] | 317 | self._SyncManifest(opt) |
| 318 | self._LinkManifest(opt.manifest_name) |
| 319 | |
Shawn O. Pearce | 8630f39 | 2009-03-19 10:17:12 -0700 | [diff] [blame] | 320 | if os.isatty(0) and os.isatty(1) and not self.manifest.IsMirror: |
Victor Boivie | 841be34 | 2011-04-05 11:31:10 +0200 | [diff] [blame] | 321 | if opt.config_name or self._ShouldConfigureUser(): |
| 322 | self._ConfigureUser() |
The Android Open Source Project | cf31fe9 | 2008-10-21 07:00:00 -0700 | [diff] [blame] | 323 | self._ConfigureColor() |
| 324 | |
Doug Anderson | 30d4529 | 2011-05-04 15:01:04 -0700 | [diff] [blame] | 325 | self._ConfigureDepth(opt) |
| 326 | |
Shawn O. Pearce | 8630f39 | 2009-03-19 10:17:12 -0700 | [diff] [blame] | 327 | if self.manifest.IsMirror: |
Shawn O. Pearce | e284ad1 | 2008-11-04 07:37:10 -0800 | [diff] [blame] | 328 | type = 'mirror ' |
| 329 | else: |
| 330 | type = '' |
| 331 | |
The Android Open Source Project | cf31fe9 | 2008-10-21 07:00:00 -0700 | [diff] [blame] | 332 | print '' |
Shawn O. Pearce | e284ad1 | 2008-11-04 07:37:10 -0800 | [diff] [blame] | 333 | print 'repo %sinitialized in %s' % (type, self.manifest.topdir) |