blob: c2376b658ffa8b69f89cdc62f3895be2250ab6e5 [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
Mike Frysinger979d5bd2020-02-09 02:28:34 -050015import optparse
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -070016import os
Conley Owensd21720d2012-04-16 11:02:21 -070017import platform
Conley Owens971de8e2012-04-16 10:36:08 -070018import re
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -070019import sys
Mike Frysingeracf63b22019-06-13 02:24:21 -040020import urllib.parse
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -070021
22from color import Coloring
Shawn O. Pearcec95583b2009-03-03 17:47:06 -080023from command import InteractiveCommand, MirrorSafeCommand
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -070024from error import ManifestParseError
Jonathan Nieder93719792015-03-17 11:29:58 -070025from project import SyncBuffer
Shawn O. Pearcef322b9a2011-09-19 14:50:58 -070026from git_config import GitConfig
Mike Frysinger82caef62020-02-11 18:51:08 -050027from git_command import git_require, MIN_GIT_VERSION_SOFT, MIN_GIT_VERSION_HARD
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):
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -070034 common = True
Mike Frysinger401c6f02021-02-18 15:20:15 -050035 helpSummary = "Initialize a repo client checkout in the current directory"
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -070036 helpUsage = """
Mike Frysinger401c6f02021-02-18 15:20:15 -050037%prog [options] [manifest url]
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -070038"""
39 helpDescription = """
40The '%prog' command is run once to install and initialize repo.
41The latest repo source code and manifest collection is downloaded
42from the server and is installed in the .repo/ directory in the
43current working directory.
44
Mike Frysinger401c6f02021-02-18 15:20:15 -050045When creating a new checkout, the manifest URL is the only required setting.
46It may be specified using the --manifest-url option, or as the first optional
47argument.
48
Shawn O. Pearce77bb4af2009-04-18 11:33:32 -070049The optional -b argument can be used to select the manifest branch
Mike Frysinger50a81de2020-09-06 15:51:21 -040050to checkout and use. If no branch is specified, the remote's default
Mike Frysinger23882b32021-02-23 15:43:07 -050051branch is used. This is equivalent to using -b HEAD.
Shawn O. Pearce77bb4af2009-04-18 11:33:32 -070052
53The optional -m argument can be used to specify an alternate manifest
54to be used. If no manifest is specified, the manifest default.xml
55will be used.
56
Shawn O. Pearce88443382010-10-08 10:02:09 +020057The --reference option can be used to point to a directory that
58has the content of a --mirror sync. This will make the working
59directory use as much data as possible from the local reference
60directory when fetching from the server. This will make the sync
61go a lot faster by reducing data traffic on the network.
62
Nikolai Merinov09f0abb2018-10-19 15:07:05 +050063The --dissociate option can be used to borrow the objects from
64the directory specified with the --reference option only to reduce
65network transfer, and stop borrowing from them after a first clone
66is made by making necessary local copies of borrowed objects.
67
Hu xiuyun9711a982015-12-11 11:16:41 +080068The --no-clone-bundle option disables any attempt to use
69$URL/clone.bundle to bootstrap a new Git repository from a
70resumeable bundle file on a content delivery network. This
71may be necessary if there are problems with the local Python
72HTTP client or proxy configuration, but the Git binary works.
Shawn O. Pearce88443382010-10-08 10:02:09 +020073
Mike Frysingerb8f7bb02018-10-10 01:05:11 -040074# Switching Manifest Branches
Shawn O. Pearce77bb4af2009-04-18 11:33:32 -070075
76To switch to another manifest branch, `repo init -b otherbranch`
77may be used in an existing client. However, as this only updates the
78manifest, a subsequent `repo sync` (or `repo sync -d`) is necessary
79to update the working directory files.
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -070080"""
81
Mike Frysinger66098f72020-02-05 00:01:59 -050082 def _Options(self, p, gitc_init=False):
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -070083 # Logging
84 g = p.add_option_group('Logging options')
Mike Frysingeredd3d452020-02-21 23:55:07 -050085 g.add_option('-v', '--verbose',
86 dest='output_mode', action='store_true',
87 help='show all output')
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -070088 g.add_option('-q', '--quiet',
Mike Frysingeredd3d452020-02-21 23:55:07 -050089 dest='output_mode', action='store_false',
90 help='only show errors')
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -070091
92 # Manifest
93 g = p.add_option_group('Manifest options')
94 g.add_option('-u', '--manifest-url',
Shawn O. Pearce34fb20f2011-11-30 13:41:02 -080095 dest='manifest_url',
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -070096 help='manifest repository location', metavar='URL')
Mike Frysinger23882b32021-02-23 15:43:07 -050097 g.add_option('-b', '--manifest-branch', metavar='REVISION',
98 help='manifest branch or revision (use HEAD for default)')
Mike Frysinger66098f72020-02-05 00:01:59 -050099 cbr_opts = ['--current-branch']
100 # The gitc-init subcommand allocates -c itself, but a lot of init users
101 # want -c, so try to satisfy both as best we can.
Dan Willemsen93293ca2020-02-06 17:00:00 -0800102 if not gitc_init:
Mike Frysinger66098f72020-02-05 00:01:59 -0500103 cbr_opts += ['-c']
104 g.add_option(*cbr_opts,
Naseer Ahmedf4dda9a2016-12-01 18:49:54 -0500105 dest='current_branch_only', action='store_true',
106 help='fetch only current manifest branch from server')
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -0700107 g.add_option('-m', '--manifest-name',
108 dest='manifest_name', default='default.xml',
109 help='initial manifest file', metavar='NAME.xml')
Shawn O. Pearcee284ad12008-11-04 07:37:10 -0800110 g.add_option('--mirror',
111 dest='mirror', action='store_true',
David Pursehouse3d07da82012-08-15 14:22:08 +0900112 help='create a replica of the remote repositories '
113 'rather than a client working directory')
Shawn O. Pearce88443382010-10-08 10:02:09 +0200114 g.add_option('--reference',
115 dest='reference',
116 help='location of mirror directory', metavar='DIR')
Nikolai Merinov09f0abb2018-10-19 15:07:05 +0500117 g.add_option('--dissociate',
118 dest='dissociate', action='store_true',
119 help='dissociate from reference mirrors after clone')
Doug Anderson30d45292011-05-04 15:01:04 -0700120 g.add_option('--depth', type='int', default=None,
121 dest='depth',
122 help='create a shallow clone with given depth; see git clone')
Xin Li745be2e2019-06-03 11:24:30 -0700123 g.add_option('--partial-clone', action='store_true',
124 dest='partial_clone',
125 help='perform partial clone (https://git-scm.com/'
126 'docs/gitrepository-layout#_code_partialclone_code)')
127 g.add_option('--clone-filter', action='store', default='blob:none',
128 dest='clone_filter',
129 help='filter for use with --partial-clone [default: %default]')
Mike Frysinger979d5bd2020-02-09 02:28:34 -0500130 # TODO(vapier): Expose option with real help text once this has been in the
131 # wild for a while w/out significant bug reports. Goal is by ~Sep 2020.
132 g.add_option('--worktree', action='store_true',
133 help=optparse.SUPPRESS_HELP)
Julien Campergue335f5ef2013-10-16 11:02:35 +0200134 g.add_option('--archive',
135 dest='archive', action='store_true',
136 help='checkout an archive instead of a git repository for '
137 'each project. See git archive.')
Martin Kellye4e94d22017-03-21 16:05:12 -0700138 g.add_option('--submodules',
139 dest='submodules', action='store_true',
140 help='sync any submodules associated with the manifest repo')
Raman Tenneti21dce3d2021-02-09 00:26:31 -0800141 g.add_option('--use-superproject', action='store_true',
142 help='use the manifest superproject to sync projects')
143 g.add_option('--no-use-superproject', action='store_false',
144 dest='use_superproject',
145 help='disable use of manifest superprojects')
Colin Cross5acde752012-03-28 20:15:45 -0700146 g.add_option('-g', '--groups',
David Holmer0a1c6a12012-11-14 19:19:00 -0500147 dest='groups', default='default',
148 help='restrict manifest projects to ones with specified '
149 'group(s) [default|all|G1,G2,G3|G4,-G5,-G6]',
Colin Cross5acde752012-03-28 20:15:45 -0700150 metavar='GROUP')
Conley Owensd21720d2012-04-16 11:02:21 -0700151 g.add_option('-p', '--platform',
152 dest='platform', default='auto',
Conley Owensbb1b5f52012-08-13 13:11:18 -0700153 help='restrict manifest projects to ones with a specified '
Conley Owensd21720d2012-04-16 11:02:21 -0700154 'platform group [auto|all|none|linux|darwin|...]',
155 metavar='PLATFORM')
Xin Lid79a4bc2020-05-20 16:03:45 -0700156 g.add_option('--clone-bundle', action='store_true',
157 help='force use of /clone.bundle on HTTP/HTTPS (default if not --partial-clone)')
Hu xiuyun9711a982015-12-11 11:16:41 +0800158 g.add_option('--no-clone-bundle',
Xin Lid79a4bc2020-05-20 16:03:45 -0700159 dest='clone_bundle', action='store_false',
160 help='disable use of /clone.bundle on HTTP/HTTPS (default if --partial-clone)')
Naseer Ahmedf4dda9a2016-12-01 18:49:54 -0500161 g.add_option('--no-tags',
Mike Frysingerc58ec4d2020-02-17 14:36:08 -0500162 dest='tags', default=True, action='store_false',
Naseer Ahmedf4dda9a2016-12-01 18:49:54 -0500163 help="don't fetch tags in the manifest")
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -0700164
165 # Tool
Shawn O. Pearcefd89b672009-04-18 11:28:57 -0700166 g = p.add_option_group('repo Version options')
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -0700167 g.add_option('--repo-url',
168 dest='repo_url',
169 help='repo repository location', metavar='URL')
Mike Frysinger58ac1672020-03-14 14:35:26 -0400170 g.add_option('--repo-rev', metavar='REV',
171 help='repo branch or revision')
172 g.add_option('--repo-branch', dest='repo_rev',
173 help=optparse.SUPPRESS_HELP)
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -0700174 g.add_option('--no-repo-verify',
Mike Frysingerc58ec4d2020-02-17 14:36:08 -0500175 dest='repo_verify', default=True, action='store_false',
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -0700176 help='do not verify repo source code')
177
Victor Boivie841be342011-04-05 11:31:10 +0200178 # Other
179 g = p.add_option_group('Other options')
180 g.add_option('--config-name',
181 dest='config_name', action="store_true", default=False,
182 help='Always prompt for name/e-mail')
183
David Pursehouse3f5ea0b2012-11-17 03:13:09 +0900184 def _RegisteredEnvironmentOptions(self):
185 return {'REPO_MANIFEST_URL': 'manifest_url',
186 'REPO_MIRROR_LOCATION': 'reference'}
187
Raman Tennetief99ec02021-03-04 10:29:40 -0800188 def _CloneSuperproject(self, opt):
189 """Clone the superproject based on the superproject's url and branch.
190
191 Args:
192 opt: Program options returned from optparse. See _Options().
193 """
Raman Tenneti21dce3d2021-02-09 00:26:31 -0800194 superproject = git_superproject.Superproject(self.manifest,
Raman Tennetief99ec02021-03-04 10:29:40 -0800195 self.repodir,
196 quiet=opt.quiet)
Raman Tenneti21dce3d2021-02-09 00:26:31 -0800197 if not superproject.Sync():
198 print('error: git update of superproject failed', file=sys.stderr)
199 sys.exit(1)
200
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -0700201 def _SyncManifest(self, opt):
202 m = self.manifest.manifestProject
Shawn O. Pearce5470df62009-03-09 18:51:58 -0700203 is_new = not m.Exists
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -0700204
Shawn O. Pearce5470df62009-03-09 18:51:58 -0700205 if is_new:
Shawn O. Pearce34fb20f2011-11-30 13:41:02 -0800206 if not opt.manifest_url:
Mike Frysinger401c6f02021-02-18 15:20:15 -0500207 print('fatal: manifest url is required.', file=sys.stderr)
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -0700208 sys.exit(1)
209
210 if not opt.quiet:
Mike Frysingerdcbfadf2020-02-22 00:04:39 -0500211 print('Downloading manifest from %s' %
212 (GitConfig.ForUser().UrlInsteadOf(opt.manifest_url),),
Sarah Owenscecd1d82012-11-01 22:59:27 -0700213 file=sys.stderr)
Victor Boivie2b30e3a2012-10-05 12:37:58 +0200214
215 # The manifest project object doesn't keep track of the path on the
216 # server where this git is located, so let's save that here.
217 mirrored_manifest_git = None
218 if opt.reference:
Anthony King7993f3c2015-06-03 17:21:56 +0100219 manifest_git_path = urllib.parse.urlparse(opt.manifest_url).path[1:]
Victor Boivie2b30e3a2012-10-05 12:37:58 +0200220 mirrored_manifest_git = os.path.join(opt.reference, manifest_git_path)
221 if not mirrored_manifest_git.endswith(".git"):
222 mirrored_manifest_git += ".git"
223 if not os.path.exists(mirrored_manifest_git):
Samuel Holland5f0e57d2018-01-22 11:00:24 -0600224 mirrored_manifest_git = os.path.join(opt.reference,
225 '.repo/manifests.git')
Victor Boivie2b30e3a2012-10-05 12:37:58 +0200226
227 m._InitGitDir(mirror_git=mirrored_manifest_git)
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -0700228
Nasser Grainawid92464e2019-05-21 10:41:35 -0600229 self._ConfigureDepth(opt)
230
Mike Frysinger50a81de2020-09-06 15:51:21 -0400231 # Set the remote URL before the remote branch as we might need it below.
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -0700232 if opt.manifest_url:
233 r = m.GetRemote(m.remote.name)
234 r.url = opt.manifest_url
235 r.ResetFetch()
236 r.Save()
237
Mike Frysinger50a81de2020-09-06 15:51:21 -0400238 if opt.manifest_branch:
Mike Frysinger23882b32021-02-23 15:43:07 -0500239 if opt.manifest_branch == 'HEAD':
240 opt.manifest_branch = m.ResolveRemoteHead()
241 if opt.manifest_branch is None:
242 print('fatal: unable to resolve HEAD', file=sys.stderr)
243 sys.exit(1)
Mike Frysinger50a81de2020-09-06 15:51:21 -0400244 m.revisionExpr = opt.manifest_branch
245 else:
246 if is_new:
247 default_branch = m.ResolveRemoteHead()
248 if default_branch is None:
249 # If the remote doesn't have HEAD configured, default to master.
250 default_branch = 'refs/heads/master'
251 m.revisionExpr = default_branch
252 else:
253 m.PreSync()
254
David Pursehouse1d947b32012-10-25 12:23:11 +0900255 groups = re.split(r'[,\s]+', opt.groups)
Pascal Muetschardc2a64dd2015-10-22 13:26:36 -0700256 all_platforms = ['linux', 'darwin', 'windows']
Conley Owensd21720d2012-04-16 11:02:21 -0700257 platformize = lambda x: 'platform-' + x
258 if opt.platform == 'auto':
259 if (not opt.mirror and
David Pursehouseabdf7502020-02-12 14:58:39 +0900260 not m.config.GetString('repo.mirror') == 'true'):
Conley Owensd21720d2012-04-16 11:02:21 -0700261 groups.append(platformize(platform.system().lower()))
262 elif opt.platform == 'all':
Colin Cross54657272012-04-23 13:39:48 -0700263 groups.extend(map(platformize, all_platforms))
Conley Owensd21720d2012-04-16 11:02:21 -0700264 elif opt.platform in all_platforms:
Pascal Muetschardc2a64dd2015-10-22 13:26:36 -0700265 groups.append(platformize(opt.platform))
Conley Owensd21720d2012-04-16 11:02:21 -0700266 elif opt.platform != 'none':
Sarah Owenscecd1d82012-11-01 22:59:27 -0700267 print('fatal: invalid platform flag', file=sys.stderr)
Conley Owensd21720d2012-04-16 11:02:21 -0700268 sys.exit(1)
269
Conley Owens971de8e2012-04-16 10:36:08 -0700270 groups = [x for x in groups if x]
271 groupstr = ','.join(groups)
David Holmer0a1c6a12012-11-14 19:19:00 -0500272 if opt.platform == 'auto' and groupstr == 'default,platform-' + platform.system().lower():
Conley Owens971de8e2012-04-16 10:36:08 -0700273 groupstr = None
274 m.config.SetString('manifest.groups', groupstr)
Colin Cross5acde752012-03-28 20:15:45 -0700275
Shawn O. Pearce88443382010-10-08 10:02:09 +0200276 if opt.reference:
277 m.config.SetString('repo.reference', opt.reference)
278
Nikolai Merinov09f0abb2018-10-19 15:07:05 +0500279 if opt.dissociate:
Mike Frysinger38867fb2021-02-09 23:14:41 -0500280 m.config.SetBoolean('repo.dissociate', opt.dissociate)
Nikolai Merinov09f0abb2018-10-19 15:07:05 +0500281
Mike Frysinger979d5bd2020-02-09 02:28:34 -0500282 if opt.worktree:
283 if opt.mirror:
284 print('fatal: --mirror and --worktree are incompatible',
285 file=sys.stderr)
286 sys.exit(1)
287 if opt.submodules:
288 print('fatal: --submodules and --worktree are incompatible',
289 file=sys.stderr)
290 sys.exit(1)
Mike Frysinger38867fb2021-02-09 23:14:41 -0500291 m.config.SetBoolean('repo.worktree', opt.worktree)
Mike Frysinger979d5bd2020-02-09 02:28:34 -0500292 if is_new:
293 m.use_git_worktrees = True
294 print('warning: --worktree is experimental!', file=sys.stderr)
295
Julien Campergue335f5ef2013-10-16 11:02:35 +0200296 if opt.archive:
297 if is_new:
Mike Frysinger38867fb2021-02-09 23:14:41 -0500298 m.config.SetBoolean('repo.archive', opt.archive)
Julien Campergue335f5ef2013-10-16 11:02:35 +0200299 else:
300 print('fatal: --archive is only supported when initializing a new '
301 'workspace.', file=sys.stderr)
302 print('Either delete the .repo folder in this workspace, or initialize '
303 'in another location.', file=sys.stderr)
304 sys.exit(1)
305
Shawn O. Pearcee284ad12008-11-04 07:37:10 -0800306 if opt.mirror:
Shawn O. Pearce5470df62009-03-09 18:51:58 -0700307 if is_new:
Mike Frysinger38867fb2021-02-09 23:14:41 -0500308 m.config.SetBoolean('repo.mirror', opt.mirror)
Shawn O. Pearce5470df62009-03-09 18:51:58 -0700309 else:
David Pursehouse25470982012-11-21 14:41:58 +0900310 print('fatal: --mirror is only supported when initializing a new '
311 'workspace.', file=sys.stderr)
312 print('Either delete the .repo folder in this workspace, or initialize '
313 'in another location.', file=sys.stderr)
Shawn O. Pearce5470df62009-03-09 18:51:58 -0700314 sys.exit(1)
Shawn O. Pearcee284ad12008-11-04 07:37:10 -0800315
Xin Li745be2e2019-06-03 11:24:30 -0700316 if opt.partial_clone:
317 if opt.mirror:
318 print('fatal: --mirror and --partial-clone are mutually exclusive',
319 file=sys.stderr)
320 sys.exit(1)
Mike Frysinger38867fb2021-02-09 23:14:41 -0500321 m.config.SetBoolean('repo.partialclone', opt.partial_clone)
Xin Li745be2e2019-06-03 11:24:30 -0700322 if opt.clone_filter:
323 m.config.SetString('repo.clonefilter', opt.clone_filter)
324 else:
325 opt.clone_filter = None
326
Xin Lid79a4bc2020-05-20 16:03:45 -0700327 if opt.clone_bundle is None:
328 opt.clone_bundle = False if opt.partial_clone else True
329 else:
Mike Frysinger38867fb2021-02-09 23:14:41 -0500330 m.config.SetBoolean('repo.clonebundle', opt.clone_bundle)
Xin Lid79a4bc2020-05-20 16:03:45 -0700331
Martin Kellye4e94d22017-03-21 16:05:12 -0700332 if opt.submodules:
Mike Frysinger38867fb2021-02-09 23:14:41 -0500333 m.config.SetBoolean('repo.submodules', opt.submodules)
Martin Kellye4e94d22017-03-21 16:05:12 -0700334
Raman Tenneti21dce3d2021-02-09 00:26:31 -0800335 if opt.use_superproject is not None:
336 m.config.SetBoolean('repo.superproject', opt.use_superproject)
337
Mike Frysingeredd3d452020-02-21 23:55:07 -0500338 if not m.Sync_NetworkHalf(is_new=is_new, quiet=opt.quiet, verbose=opt.verbose,
Mike Frysingerc58ec4d2020-02-17 14:36:08 -0500339 clone_bundle=opt.clone_bundle,
David Pursehouseabdf7502020-02-12 14:58:39 +0900340 current_branch_only=opt.current_branch_only,
Mike Frysingerc58ec4d2020-02-17 14:36:08 -0500341 tags=opt.tags, submodules=opt.submodules,
David Pursehouseabdf7502020-02-12 14:58:39 +0900342 clone_filter=opt.clone_filter):
Shawn O. Pearce1fc99f42009-03-17 08:06:18 -0700343 r = m.GetRemote(m.remote.name)
Sarah Owenscecd1d82012-11-01 22:59:27 -0700344 print('fatal: cannot obtain manifest %s' % r.url, file=sys.stderr)
Doug Anderson2630dd92011-04-07 13:36:30 -0700345
346 # Better delete the manifest git dir if we created it; otherwise next
347 # time (when user fixes problems) we won't go through the "is_new" logic.
348 if is_new:
Renaud Paquaya65adf72016-11-03 10:37:53 -0700349 platform_utils.rmtree(m.gitdir)
Shawn O. Pearce1fc99f42009-03-17 08:06:18 -0700350 sys.exit(1)
351
Florian Vallee5d016502012-06-07 17:19:26 +0200352 if opt.manifest_branch:
Martin Kelly224a31a2017-07-10 14:46:25 -0700353 m.MetaBranchSwitch(submodules=opt.submodules)
Florian Vallee5d016502012-06-07 17:19:26 +0200354
Shawn O. Pearce350cde42009-04-16 11:21:18 -0700355 syncbuf = SyncBuffer(m.config)
Martin Kellye4e94d22017-03-21 16:05:12 -0700356 m.Sync_LocalHalf(syncbuf, submodules=opt.submodules)
Shawn O. Pearce350cde42009-04-16 11:21:18 -0700357 syncbuf.Finish()
358
Shawn O. Pearcedf018832009-03-17 08:15:27 -0700359 if is_new or m.CurrentBranch is None:
Shawn O. Pearce0a389e92009-04-10 16:21:18 -0700360 if not m.StartBranch('default'):
Sarah Owenscecd1d82012-11-01 22:59:27 -0700361 print('fatal: cannot create default in manifest', file=sys.stderr)
Shawn O. Pearce0a389e92009-04-10 16:21:18 -0700362 sys.exit(1)
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -0700363
364 def _LinkManifest(self, name):
365 if not name:
Sarah Owenscecd1d82012-11-01 22:59:27 -0700366 print('fatal: manifest name (-m) is required.', file=sys.stderr)
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -0700367 sys.exit(1)
368
369 try:
370 self.manifest.Link(name)
Sarah Owensa5be53f2012-09-09 15:37:57 -0700371 except ManifestParseError as e:
Sarah Owenscecd1d82012-11-01 22:59:27 -0700372 print("fatal: manifest '%s' not available" % name, file=sys.stderr)
373 print('fatal: %s' % str(e), file=sys.stderr)
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -0700374 sys.exit(1)
375
Shawn O. Pearce37dbf2b2009-07-02 10:53:04 -0700376 def _Prompt(self, prompt, value):
Mike Frysingerab85fe72019-07-04 17:35:11 -0400377 print('%-10s [%s]: ' % (prompt, value), end='')
378 # TODO: When we require Python 3, use flush=True w/print above.
379 sys.stdout.flush()
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -0700380 a = sys.stdin.readline().strip()
Shawn O. Pearce37dbf2b2009-07-02 10:53:04 -0700381 if a == '':
382 return value
383 return a
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -0700384
Mike Frysinger0b888912020-02-21 22:48:40 -0500385 def _ShouldConfigureUser(self, opt):
Mike Frysinger8c1e9cb2020-09-06 14:53:18 -0400386 gc = self.client.globalConfig
Victor Boivie841be342011-04-05 11:31:10 +0200387 mp = self.manifest.manifestProject
388
389 # If we don't have local settings, get from global.
390 if not mp.config.Has('user.name') or not mp.config.Has('user.email'):
391 if not gc.Has('user.name') or not gc.Has('user.email'):
392 return True
393
394 mp.config.SetString('user.name', gc.GetString('user.name'))
395 mp.config.SetString('user.email', gc.GetString('user.email'))
396
Mike Frysinger0b888912020-02-21 22:48:40 -0500397 if not opt.quiet:
398 print()
399 print('Your identity is: %s <%s>' % (mp.config.GetString('user.name'),
400 mp.config.GetString('user.email')))
401 print("If you want to change this, please re-run 'repo init' with --config-name")
Victor Boivie841be342011-04-05 11:31:10 +0200402 return False
403
Mike Frysinger0b888912020-02-21 22:48:40 -0500404 def _ConfigureUser(self, opt):
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -0700405 mp = self.manifest.manifestProject
406
Shawn O. Pearce37dbf2b2009-07-02 10:53:04 -0700407 while True:
Mike Frysinger0b888912020-02-21 22:48:40 -0500408 if not opt.quiet:
409 print()
David Pursehouse54a4e602020-02-12 14:31:05 +0900410 name = self._Prompt('Your Name', mp.UserName)
Shawn O. Pearce37dbf2b2009-07-02 10:53:04 -0700411 email = self._Prompt('Your Email', mp.UserEmail)
412
Mike Frysinger0b888912020-02-21 22:48:40 -0500413 if not opt.quiet:
414 print()
Sarah Owenscecd1d82012-11-01 22:59:27 -0700415 print('Your identity is: %s <%s>' % (name, email))
Mike Frysingerab85fe72019-07-04 17:35:11 -0400416 print('is this correct [y/N]? ', end='')
417 # TODO: When we require Python 3, use flush=True w/print above.
418 sys.stdout.flush()
David Pursehousefc241242012-11-14 09:19:39 +0900419 a = sys.stdin.readline().strip().lower()
Nico Sallembien6d7508b2010-04-01 11:03:53 -0700420 if a in ('yes', 'y', 't', 'true'):
Shawn O. Pearce37dbf2b2009-07-02 10:53:04 -0700421 break
422
423 if name != mp.UserName:
424 mp.config.SetString('user.name', name)
425 if email != mp.UserEmail:
426 mp.config.SetString('user.email', email)
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -0700427
428 def _HasColorSet(self, gc):
429 for n in ['ui', 'diff', 'status']:
430 if gc.Has('color.%s' % n):
431 return True
432 return False
433
434 def _ConfigureColor(self):
Mike Frysinger8c1e9cb2020-09-06 14:53:18 -0400435 gc = self.client.globalConfig
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -0700436 if self._HasColorSet(gc):
437 return
438
439 class _Test(Coloring):
440 def __init__(self):
441 Coloring.__init__(self, gc, 'test color display')
442 self._on = True
443 out = _Test()
444
Sarah Owenscecd1d82012-11-01 22:59:27 -0700445 print()
446 print("Testing colorized output (for 'repo diff', 'repo status'):")
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -0700447
David Pursehouse8f62fb72012-11-14 12:09:38 +0900448 for c in ['black', 'red', 'green', 'yellow', 'blue', 'magenta', 'cyan']:
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -0700449 out.write(' ')
450 out.printer(fg=c)(' %-6s ', c)
451 out.write(' ')
452 out.printer(fg='white', bg='black')(' %s ' % 'white')
453 out.nl()
454
David Pursehouse8f62fb72012-11-14 12:09:38 +0900455 for c in ['bold', 'dim', 'ul', 'reverse']:
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -0700456 out.write(' ')
457 out.printer(fg='black', attr=c)(' %-6s ', c)
458 out.nl()
459
Mike Frysingerab85fe72019-07-04 17:35:11 -0400460 print('Enable color display in this user account (y/N)? ', end='')
461 # TODO: When we require Python 3, use flush=True w/print above.
462 sys.stdout.flush()
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -0700463 a = sys.stdin.readline().strip().lower()
464 if a in ('y', 'yes', 't', 'true', 'on'):
465 gc.SetString('color.ui', 'auto')
466
Doug Anderson30d45292011-05-04 15:01:04 -0700467 def _ConfigureDepth(self, opt):
468 """Configure the depth we'll sync down.
469
470 Args:
471 opt: Options from optparse. We care about opt.depth.
472 """
473 # Opt.depth will be non-None if user actually passed --depth to repo init.
474 if opt.depth is not None:
475 if opt.depth > 0:
476 # Positive values will set the depth.
477 depth = str(opt.depth)
478 else:
479 # Negative numbers will clear the depth; passing None to SetString
480 # will do that.
481 depth = None
482
483 # We store the depth in the main manifest project.
484 self.manifest.manifestProject.config.SetString('repo.depth', depth)
485
Mike Frysinger0b888912020-02-21 22:48:40 -0500486 def _DisplayResult(self, opt):
Yang Zhenhui75cc3532012-10-23 15:41:54 +0800487 if self.manifest.IsMirror:
488 init_type = 'mirror '
489 else:
490 init_type = ''
491
Mike Frysinger0b888912020-02-21 22:48:40 -0500492 if not opt.quiet:
493 print()
494 print('repo %shas been initialized in %s' %
495 (init_type, self.manifest.topdir))
Yang Zhenhui75cc3532012-10-23 15:41:54 +0800496
497 current_dir = os.getcwd()
498 if current_dir != self.manifest.topdir:
David Pursehouse35765962013-01-29 09:49:48 +0900499 print('If this is not the directory in which you want to initialize '
Sarah Owenscecd1d82012-11-01 22:59:27 -0700500 'repo, please run:')
501 print(' rm -r %s/.repo' % self.manifest.topdir)
502 print('and try again.')
Yang Zhenhui75cc3532012-10-23 15:41:54 +0800503
Mike Frysingerae6cb082019-08-27 01:10:59 -0400504 def ValidateOptions(self, opt, args):
Victor Boivie297e7c62012-10-05 14:50:05 +0200505 if opt.reference:
Samuel Hollandbaa00092018-01-22 10:57:29 -0600506 opt.reference = os.path.expanduser(opt.reference)
Victor Boivie297e7c62012-10-05 14:50:05 +0200507
Julien Campergue335f5ef2013-10-16 11:02:35 +0200508 # Check this here, else manifest will be tagged "not new" and init won't be
509 # possible anymore without removing the .repo/manifests directory.
510 if opt.archive and opt.mirror:
Mike Frysingerae6cb082019-08-27 01:10:59 -0400511 self.OptionParser.error('--mirror and --archive cannot be used together.')
512
Mike Frysinger0578ebf2020-08-27 01:50:12 -0400513 if args:
Mike Frysinger401c6f02021-02-18 15:20:15 -0500514 if opt.manifest_url:
515 self.OptionParser.error(
516 '--manifest-url option and URL argument both specified: only use '
517 'one to select the manifest URL.')
518
519 opt.manifest_url = args.pop(0)
520
521 if args:
522 self.OptionParser.error('too many arguments to init')
Mike Frysinger0578ebf2020-08-27 01:50:12 -0400523
Mike Frysingerae6cb082019-08-27 01:10:59 -0400524 def Execute(self, opt, args):
Mike Frysinger82caef62020-02-11 18:51:08 -0500525 git_require(MIN_GIT_VERSION_HARD, fail=True)
526 if not git_require(MIN_GIT_VERSION_SOFT):
527 print('repo: warning: git-%s+ will soon be required; please upgrade your '
528 'version of git to maintain support.'
529 % ('.'.join(str(x) for x in MIN_GIT_VERSION_SOFT),),
530 file=sys.stderr)
Julien Campergue335f5ef2013-10-16 11:02:35 +0200531
Mike Frysingeredd3d452020-02-21 23:55:07 -0500532 opt.quiet = opt.output_mode is False
533 opt.verbose = opt.output_mode is True
534
Mike Frysinger7936ce82020-02-29 02:53:41 -0500535 rp = self.manifest.repoProject
536
537 # Handle new --repo-url requests.
538 if opt.repo_url:
539 remote = rp.GetRemote('origin')
540 remote.url = opt.repo_url
541 remote.Save()
542
Mike Frysinger3599cc32020-02-29 02:53:41 -0500543 # Handle new --repo-rev requests.
544 if opt.repo_rev:
545 wrapper = Wrapper()
546 remote_ref, rev = wrapper.check_repo_rev(
547 rp.gitdir, opt.repo_rev, repo_verify=opt.repo_verify, quiet=opt.quiet)
548 branch = rp.GetBranch('default')
549 branch.merge = remote_ref
Mike Frysinger5e2f32f2020-12-05 22:57:19 -0500550 rp.work_git.reset('--hard', rev)
Mike Frysinger3599cc32020-02-29 02:53:41 -0500551 branch.Save()
552
Mike Frysinger979d5bd2020-02-09 02:28:34 -0500553 if opt.worktree:
554 # Older versions of git supported worktree, but had dangerous gc bugs.
555 git_require((2, 15, 0), fail=True, msg='git gc worktree corruption')
556
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -0700557 self._SyncManifest(opt)
558 self._LinkManifest(opt.manifest_name)
559
Raman Tenneti21dce3d2021-02-09 00:26:31 -0800560 if self.manifest.manifestProject.config.GetBoolean('repo.superproject'):
Raman Tennetief99ec02021-03-04 10:29:40 -0800561 self._CloneSuperproject(opt)
Raman Tenneti21dce3d2021-02-09 00:26:31 -0800562
Shawn O. Pearce8630f392009-03-19 10:17:12 -0700563 if os.isatty(0) and os.isatty(1) and not self.manifest.IsMirror:
Mike Frysinger0b888912020-02-21 22:48:40 -0500564 if opt.config_name or self._ShouldConfigureUser(opt):
565 self._ConfigureUser(opt)
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -0700566 self._ConfigureColor()
567
Mike Frysinger0b888912020-02-21 22:48:40 -0500568 self._DisplayResult(opt)