blob: b6c891aca0ce121035ffdc3f59befc46e3d9ae2d [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
15import os
Conley Owensd21720d2012-04-16 11:02:21 -070016import platform
Conley Owens971de8e2012-04-16 10:36:08 -070017import re
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -070018import sys
Mike Frysingeracf63b22019-06-13 02:24:21 -040019import urllib.parse
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -070020
21from color import Coloring
Shawn O. Pearcec95583b2009-03-03 17:47:06 -080022from command import InteractiveCommand, MirrorSafeCommand
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -070023from error import ManifestParseError
Jonathan Nieder93719792015-03-17 11:29:58 -070024from project import SyncBuffer
Shawn O. Pearcef322b9a2011-09-19 14:50:58 -070025from git_config import GitConfig
Mike Frysinger82caef62020-02-11 18:51:08 -050026from git_command import git_require, MIN_GIT_VERSION_SOFT, MIN_GIT_VERSION_HARD
Jack Neusc474c9c2021-07-26 23:08:54 +000027import fetch
Raman Tenneti21dce3d2021-02-09 00:26:31 -080028import git_superproject
Renaud Paquaya65adf72016-11-03 10:37:53 -070029import platform_utils
Mike Frysinger3599cc32020-02-29 02:53:41 -050030from wrapper import Wrapper
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -070031
David Pursehouse819827a2020-02-12 15:20:19 +090032
Shawn O. Pearcec95583b2009-03-03 17:47:06 -080033class Init(InteractiveCommand, MirrorSafeCommand):
Mike Frysinger4f210542021-06-14 16:05:19 -040034 COMMON = True
LaMont Jonescc879a92021-11-18 22:40:18 +000035 MULTI_MANIFEST_SUPPORT = False
Mike Frysinger401c6f02021-02-18 15:20:15 -050036 helpSummary = "Initialize a repo client checkout in the current directory"
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -070037 helpUsage = """
Mike Frysinger401c6f02021-02-18 15:20:15 -050038%prog [options] [manifest url]
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -070039"""
40 helpDescription = """
41The '%prog' command is run once to install and initialize repo.
42The latest repo source code and manifest collection is downloaded
43from the server and is installed in the .repo/ directory in the
44current working directory.
45
Mike Frysinger401c6f02021-02-18 15:20:15 -050046When creating a new checkout, the manifest URL is the only required setting.
47It may be specified using the --manifest-url option, or as the first optional
48argument.
49
Shawn O. Pearce77bb4af2009-04-18 11:33:32 -070050The optional -b argument can be used to select the manifest branch
Mike Frysinger50a81de2020-09-06 15:51:21 -040051to checkout and use. If no branch is specified, the remote's default
Mike Frysinger23882b32021-02-23 15:43:07 -050052branch is used. This is equivalent to using -b HEAD.
Shawn O. Pearce77bb4af2009-04-18 11:33:32 -070053
54The optional -m argument can be used to specify an alternate manifest
55to be used. If no manifest is specified, the manifest default.xml
56will be used.
57
Jack Neusc474c9c2021-07-26 23:08:54 +000058If the --standalone-manifest argument is set, the manifest will be downloaded
59directly from the specified --manifest-url as a static file (rather than
60setting up a manifest git checkout). With --standalone-manifest, the manifest
61will be fully static and will not be re-downloaded during subsesquent
62`repo init` and `repo sync` calls.
63
Shawn O. Pearce88443382010-10-08 10:02:09 +020064The --reference option can be used to point to a directory that
65has the content of a --mirror sync. This will make the working
66directory use as much data as possible from the local reference
67directory when fetching from the server. This will make the sync
68go a lot faster by reducing data traffic on the network.
69
Nikolai Merinov09f0abb2018-10-19 15:07:05 +050070The --dissociate option can be used to borrow the objects from
71the directory specified with the --reference option only to reduce
72network transfer, and stop borrowing from them after a first clone
73is made by making necessary local copies of borrowed objects.
74
Hu xiuyun9711a982015-12-11 11:16:41 +080075The --no-clone-bundle option disables any attempt to use
76$URL/clone.bundle to bootstrap a new Git repository from a
77resumeable bundle file on a content delivery network. This
78may be necessary if there are problems with the local Python
79HTTP client or proxy configuration, but the Git binary works.
Shawn O. Pearce88443382010-10-08 10:02:09 +020080
Mike Frysingerb8f7bb02018-10-10 01:05:11 -040081# Switching Manifest Branches
Shawn O. Pearce77bb4af2009-04-18 11:33:32 -070082
83To switch to another manifest branch, `repo init -b otherbranch`
84may be used in an existing client. However, as this only updates the
85manifest, a subsequent `repo sync` (or `repo sync -d`) is necessary
86to update the working directory files.
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -070087"""
88
Mike Frysinger9180a072021-04-13 14:57:40 -040089 def _CommonOptions(self, p):
90 """Disable due to re-use of Wrapper()."""
91
Mike Frysinger66098f72020-02-05 00:01:59 -050092 def _Options(self, p, gitc_init=False):
Mike Frysinger9a734a32021-04-08 19:14:15 -040093 Wrapper().InitParser(p, gitc_init=gitc_init)
LaMont Jonescc879a92021-11-18 22:40:18 +000094 m = p.add_option_group('Multi-manifest')
95 m.add_option('--outer-manifest', action='store_true',
96 help='operate starting at the outermost manifest')
97 m.add_option('--no-outer-manifest', dest='outer_manifest',
98 action='store_false', default=None,
99 help='do not operate on outer manifests')
100 m.add_option('--this-manifest-only', action='store_true', default=None,
101 help='only operate on this (sub)manifest')
102 m.add_option('--no-this-manifest-only', '--all-manifests',
103 dest='this_manifest_only', action='store_false',
104 help='operate on this manifest and its submanifests')
Victor Boivie841be342011-04-05 11:31:10 +0200105
David Pursehouse3f5ea0b2012-11-17 03:13:09 +0900106 def _RegisteredEnvironmentOptions(self):
107 return {'REPO_MANIFEST_URL': 'manifest_url',
108 'REPO_MIRROR_LOCATION': 'reference'}
109
Raman Tennetief99ec02021-03-04 10:29:40 -0800110 def _CloneSuperproject(self, opt):
111 """Clone the superproject based on the superproject's url and branch.
112
113 Args:
114 opt: Program options returned from optparse. See _Options().
115 """
Raman Tenneti21dce3d2021-02-09 00:26:31 -0800116 superproject = git_superproject.Superproject(self.manifest,
Raman Tennetief99ec02021-03-04 10:29:40 -0800117 self.repodir,
Raman Tenneti784e16f2021-06-11 17:29:45 -0700118 self.git_event_log,
Raman Tennetief99ec02021-03-04 10:29:40 -0800119 quiet=opt.quiet)
Raman Tenneti784e16f2021-06-11 17:29:45 -0700120 sync_result = superproject.Sync()
121 if not sync_result.success:
Raman Tenneti8db30d62021-07-06 21:30:06 -0700122 print('warning: git update of superproject failed, repo sync will not '
123 'use superproject to fetch source; while this error is not fatal, '
124 'and you can continue to run repo sync, please run repo init with '
125 'the --no-use-superproject option to stop seeing this warning',
126 file=sys.stderr)
127 if sync_result.fatal and opt.use_superproject is not None:
Raman Tenneti784e16f2021-06-11 17:29:45 -0700128 sys.exit(1)
Raman Tenneti21dce3d2021-02-09 00:26:31 -0800129
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -0700130 def _SyncManifest(self, opt):
LaMont Jones9b03f152022-03-29 23:01:18 +0000131 """Call manifestProject.Sync with arguments from opt.
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -0700132
LaMont Jones9b03f152022-03-29 23:01:18 +0000133 Args:
134 opt: options from optparse.
135 """
136 if not self.manifest.manifestProject.Sync(
137 manifest_url=opt.manifest_url,
138 manifest_branch=opt.manifest_branch,
139 standalone_manifest=opt.standalone_manifest,
140 groups=opt.groups,
141 platform=opt.platform,
142 mirror=opt.mirror,
143 dissociate=opt.dissociate,
144 reference=opt.reference,
145 worktree=opt.worktree,
146 submodules=opt.submodules,
147 archive=opt.archive,
148 partial_clone=opt.partial_clone,
149 clone_filter=opt.clone_filter,
150 partial_clone_exclude=opt.partial_clone_exclude,
151 clone_bundle=opt.clone_bundle,
152 git_lfs=opt.git_lfs,
153 use_superproject=opt.use_superproject,
154 verbose=opt.verbose,
155 current_branch_only=opt.current_branch_only,
156 tags=opt.tags,
157 depth=opt.depth):
Conley Owensd21720d2012-04-16 11:02:21 -0700158 sys.exit(1)
159
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -0700160 def _LinkManifest(self, name):
161 if not name:
Sarah Owenscecd1d82012-11-01 22:59:27 -0700162 print('fatal: manifest name (-m) is required.', file=sys.stderr)
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -0700163 sys.exit(1)
164
165 try:
166 self.manifest.Link(name)
Sarah Owensa5be53f2012-09-09 15:37:57 -0700167 except ManifestParseError as e:
Sarah Owenscecd1d82012-11-01 22:59:27 -0700168 print("fatal: manifest '%s' not available" % name, file=sys.stderr)
169 print('fatal: %s' % str(e), file=sys.stderr)
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -0700170 sys.exit(1)
171
Shawn O. Pearce37dbf2b2009-07-02 10:53:04 -0700172 def _Prompt(self, prompt, value):
Mike Frysingerab85fe72019-07-04 17:35:11 -0400173 print('%-10s [%s]: ' % (prompt, value), end='')
174 # TODO: When we require Python 3, use flush=True w/print above.
175 sys.stdout.flush()
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -0700176 a = sys.stdin.readline().strip()
Shawn O. Pearce37dbf2b2009-07-02 10:53:04 -0700177 if a == '':
178 return value
179 return a
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -0700180
Mike Frysinger0b888912020-02-21 22:48:40 -0500181 def _ShouldConfigureUser(self, opt):
Mike Frysinger8c1e9cb2020-09-06 14:53:18 -0400182 gc = self.client.globalConfig
Victor Boivie841be342011-04-05 11:31:10 +0200183 mp = self.manifest.manifestProject
184
185 # If we don't have local settings, get from global.
186 if not mp.config.Has('user.name') or not mp.config.Has('user.email'):
187 if not gc.Has('user.name') or not gc.Has('user.email'):
188 return True
189
190 mp.config.SetString('user.name', gc.GetString('user.name'))
191 mp.config.SetString('user.email', gc.GetString('user.email'))
192
Mike Frysinger0b888912020-02-21 22:48:40 -0500193 if not opt.quiet:
194 print()
195 print('Your identity is: %s <%s>' % (mp.config.GetString('user.name'),
196 mp.config.GetString('user.email')))
197 print("If you want to change this, please re-run 'repo init' with --config-name")
Victor Boivie841be342011-04-05 11:31:10 +0200198 return False
199
Mike Frysinger0b888912020-02-21 22:48:40 -0500200 def _ConfigureUser(self, opt):
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -0700201 mp = self.manifest.manifestProject
202
Shawn O. Pearce37dbf2b2009-07-02 10:53:04 -0700203 while True:
Mike Frysinger0b888912020-02-21 22:48:40 -0500204 if not opt.quiet:
205 print()
David Pursehouse54a4e602020-02-12 14:31:05 +0900206 name = self._Prompt('Your Name', mp.UserName)
Shawn O. Pearce37dbf2b2009-07-02 10:53:04 -0700207 email = self._Prompt('Your Email', mp.UserEmail)
208
Mike Frysinger0b888912020-02-21 22:48:40 -0500209 if not opt.quiet:
210 print()
Sarah Owenscecd1d82012-11-01 22:59:27 -0700211 print('Your identity is: %s <%s>' % (name, email))
Mike Frysingerab85fe72019-07-04 17:35:11 -0400212 print('is this correct [y/N]? ', end='')
213 # TODO: When we require Python 3, use flush=True w/print above.
214 sys.stdout.flush()
David Pursehousefc241242012-11-14 09:19:39 +0900215 a = sys.stdin.readline().strip().lower()
Nico Sallembien6d7508b2010-04-01 11:03:53 -0700216 if a in ('yes', 'y', 't', 'true'):
Shawn O. Pearce37dbf2b2009-07-02 10:53:04 -0700217 break
218
219 if name != mp.UserName:
220 mp.config.SetString('user.name', name)
221 if email != mp.UserEmail:
222 mp.config.SetString('user.email', email)
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -0700223
224 def _HasColorSet(self, gc):
225 for n in ['ui', 'diff', 'status']:
226 if gc.Has('color.%s' % n):
227 return True
228 return False
229
230 def _ConfigureColor(self):
Mike Frysinger8c1e9cb2020-09-06 14:53:18 -0400231 gc = self.client.globalConfig
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -0700232 if self._HasColorSet(gc):
233 return
234
235 class _Test(Coloring):
236 def __init__(self):
237 Coloring.__init__(self, gc, 'test color display')
238 self._on = True
239 out = _Test()
240
Sarah Owenscecd1d82012-11-01 22:59:27 -0700241 print()
242 print("Testing colorized output (for 'repo diff', 'repo status'):")
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -0700243
David Pursehouse8f62fb72012-11-14 12:09:38 +0900244 for c in ['black', 'red', 'green', 'yellow', 'blue', 'magenta', 'cyan']:
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -0700245 out.write(' ')
246 out.printer(fg=c)(' %-6s ', c)
247 out.write(' ')
248 out.printer(fg='white', bg='black')(' %s ' % 'white')
249 out.nl()
250
David Pursehouse8f62fb72012-11-14 12:09:38 +0900251 for c in ['bold', 'dim', 'ul', 'reverse']:
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -0700252 out.write(' ')
253 out.printer(fg='black', attr=c)(' %-6s ', c)
254 out.nl()
255
Mike Frysingerab85fe72019-07-04 17:35:11 -0400256 print('Enable color display in this user account (y/N)? ', end='')
257 # TODO: When we require Python 3, use flush=True w/print above.
258 sys.stdout.flush()
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -0700259 a = sys.stdin.readline().strip().lower()
260 if a in ('y', 'yes', 't', 'true', 'on'):
261 gc.SetString('color.ui', 'auto')
262
Mike Frysinger0b888912020-02-21 22:48:40 -0500263 def _DisplayResult(self, opt):
Yang Zhenhui75cc3532012-10-23 15:41:54 +0800264 if self.manifest.IsMirror:
265 init_type = 'mirror '
266 else:
267 init_type = ''
268
Mike Frysinger0b888912020-02-21 22:48:40 -0500269 if not opt.quiet:
270 print()
271 print('repo %shas been initialized in %s' %
272 (init_type, self.manifest.topdir))
Yang Zhenhui75cc3532012-10-23 15:41:54 +0800273
274 current_dir = os.getcwd()
275 if current_dir != self.manifest.topdir:
David Pursehouse35765962013-01-29 09:49:48 +0900276 print('If this is not the directory in which you want to initialize '
Sarah Owenscecd1d82012-11-01 22:59:27 -0700277 'repo, please run:')
278 print(' rm -r %s/.repo' % self.manifest.topdir)
279 print('and try again.')
Yang Zhenhui75cc3532012-10-23 15:41:54 +0800280
Mike Frysingerae6cb082019-08-27 01:10:59 -0400281 def ValidateOptions(self, opt, args):
Victor Boivie297e7c62012-10-05 14:50:05 +0200282 if opt.reference:
Samuel Hollandbaa00092018-01-22 10:57:29 -0600283 opt.reference = os.path.expanduser(opt.reference)
Victor Boivie297e7c62012-10-05 14:50:05 +0200284
Julien Campergue335f5ef2013-10-16 11:02:35 +0200285 # Check this here, else manifest will be tagged "not new" and init won't be
286 # possible anymore without removing the .repo/manifests directory.
Raman Tenneti6bd89aa2021-11-16 11:48:09 -0800287 if opt.mirror:
288 if opt.archive:
289 self.OptionParser.error('--mirror and --archive cannot be used '
290 'together.')
291 if opt.use_superproject is not None:
292 self.OptionParser.error('--mirror and --use-superproject cannot be '
293 'used together.')
Mike Frysingerae6cb082019-08-27 01:10:59 -0400294
Raman Tenneti4a478ed2021-11-17 18:38:24 -0800295 if opt.standalone_manifest and (opt.manifest_branch or
296 opt.manifest_name != 'default.xml'):
Jack Neusc474c9c2021-07-26 23:08:54 +0000297 self.OptionParser.error('--manifest-branch and --manifest-name cannot'
298 ' be used with --standalone-manifest.')
299
Mike Frysinger0578ebf2020-08-27 01:50:12 -0400300 if args:
Mike Frysinger401c6f02021-02-18 15:20:15 -0500301 if opt.manifest_url:
302 self.OptionParser.error(
303 '--manifest-url option and URL argument both specified: only use '
304 'one to select the manifest URL.')
305
306 opt.manifest_url = args.pop(0)
307
308 if args:
309 self.OptionParser.error('too many arguments to init')
Mike Frysinger0578ebf2020-08-27 01:50:12 -0400310
Mike Frysingerae6cb082019-08-27 01:10:59 -0400311 def Execute(self, opt, args):
Mike Frysinger82caef62020-02-11 18:51:08 -0500312 git_require(MIN_GIT_VERSION_HARD, fail=True)
313 if not git_require(MIN_GIT_VERSION_SOFT):
314 print('repo: warning: git-%s+ will soon be required; please upgrade your '
315 'version of git to maintain support.'
316 % ('.'.join(str(x) for x in MIN_GIT_VERSION_SOFT),),
317 file=sys.stderr)
Julien Campergue335f5ef2013-10-16 11:02:35 +0200318
Mike Frysinger7936ce82020-02-29 02:53:41 -0500319 rp = self.manifest.repoProject
320
321 # Handle new --repo-url requests.
322 if opt.repo_url:
323 remote = rp.GetRemote('origin')
324 remote.url = opt.repo_url
325 remote.Save()
326
Mike Frysinger3599cc32020-02-29 02:53:41 -0500327 # Handle new --repo-rev requests.
328 if opt.repo_rev:
329 wrapper = Wrapper()
Mike Frysinger4aa85842022-01-25 02:10:28 -0500330 try:
331 remote_ref, rev = wrapper.check_repo_rev(
332 rp.gitdir, opt.repo_rev, repo_verify=opt.repo_verify, quiet=opt.quiet)
333 except wrapper.CloneFailure:
334 print('fatal: double check your --repo-rev setting.', file=sys.stderr)
335 sys.exit(1)
Mike Frysinger3599cc32020-02-29 02:53:41 -0500336 branch = rp.GetBranch('default')
337 branch.merge = remote_ref
Mike Frysinger5e2f32f2020-12-05 22:57:19 -0500338 rp.work_git.reset('--hard', rev)
Mike Frysinger3599cc32020-02-29 02:53:41 -0500339 branch.Save()
340
Mike Frysinger979d5bd2020-02-09 02:28:34 -0500341 if opt.worktree:
342 # Older versions of git supported worktree, but had dangerous gc bugs.
343 git_require((2, 15, 0), fail=True, msg='git gc worktree corruption')
344
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -0700345 self._SyncManifest(opt)
346 self._LinkManifest(opt.manifest_name)
347
Raman Tenneti21dce3d2021-02-09 00:26:31 -0800348 if self.manifest.manifestProject.config.GetBoolean('repo.superproject'):
Raman Tennetief99ec02021-03-04 10:29:40 -0800349 self._CloneSuperproject(opt)
Raman Tenneti21dce3d2021-02-09 00:26:31 -0800350
Shawn O. Pearce8630f392009-03-19 10:17:12 -0700351 if os.isatty(0) and os.isatty(1) and not self.manifest.IsMirror:
Mike Frysinger0b888912020-02-21 22:48:40 -0500352 if opt.config_name or self._ShouldConfigureUser(opt):
353 self._ConfigureUser(opt)
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -0700354 self._ConfigureColor()
355
Mike Frysinger0b888912020-02-21 22:48:40 -0500356 self._DisplayResult(opt)