blob: aa2f915aeee2cf27351a21c88a523f9752efb93a [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 Frysinger8dbc07a2021-02-18 23:37:33 -050015import functools
16import multiprocessing
Simran Basib9a1b732015-08-20 12:19:28 -070017import os
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -070018import sys
Simran Basib9a1b732015-08-20 12:19:28 -070019
Mike Frysinger8dbc07a2021-02-18 23:37:33 -050020from command import Command, DEFAULT_LOCAL_JOBS, WORKER_BATCH_SIZE
Zac Livingston9ead97b2017-06-13 08:29:04 -060021from git_config import IsImmutable
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -070022from git_command import git
Simran Basib9a1b732015-08-20 12:19:28 -070023import gitc_utils
Shawn O. Pearce0f0dfa32009-04-18 14:53:39 -070024from progress import Progress
Simran Basib9a1b732015-08-20 12:19:28 -070025from project import SyncBuffer
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -070026
David Pursehouse819827a2020-02-12 15:20:19 +090027
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -070028class Start(Command):
29 common = True
30 helpSummary = "Start a new branch for development"
31 helpUsage = """
Ficus Kirkpatrick6f6cd772009-04-22 17:27:12 -070032%prog <newbranchname> [--all | <project>...]
Shawn O. Pearce06e556d2009-04-18 11:19:01 -070033"""
34 helpDescription = """
35'%prog' begins a new branch of development, starting from the
36revision specified in the manifest.
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -070037"""
Mike Frysinger8dbc07a2021-02-18 23:37:33 -050038 PARALLEL_JOBS = DEFAULT_LOCAL_JOBS
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -070039
Ficus Kirkpatrick6f6cd772009-04-22 17:27:12 -070040 def _Options(self, p):
41 p.add_option('--all',
42 dest='all', action='store_true',
43 help='begin branch in all projects')
Theodore Dubois60fdc5c2019-07-30 12:14:25 -070044 p.add_option('-r', '--rev', '--revision', dest='revision',
45 help='point branch at this revision instead of upstream')
Mike Frysingerd1e4fa72021-03-23 20:27:29 -040046 p.add_option('--head', '--HEAD',
47 dest='revision', action='store_const', const='HEAD',
Theodore Dubois60fdc5c2019-07-30 12:14:25 -070048 help='abbreviation for --rev HEAD')
Ficus Kirkpatrick6f6cd772009-04-22 17:27:12 -070049
Mike Frysingerae6cb082019-08-27 01:10:59 -040050 def ValidateOptions(self, opt, args):
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -070051 if not args:
52 self.Usage()
53
54 nb = args[0]
55 if not git.check_ref_format('heads/%s' % nb):
Mike Frysingerae6cb082019-08-27 01:10:59 -040056 self.OptionParser.error("'%s' is not a valid name" % nb)
The Android Open Source Projectcf31fe92008-10-21 07:00:00 -070057
Mike Frysinger8dbc07a2021-02-18 23:37:33 -050058 def _ExecuteOne(self, opt, nb, project):
59 """Start one project."""
60 # If the current revision is immutable, such as a SHA1, a tag or
61 # a change, then we can't push back to it. Substitute with
62 # dest_branch, if defined; or with manifest default revision instead.
63 branch_merge = ''
64 if IsImmutable(project.revisionExpr):
65 if project.dest_branch:
66 branch_merge = project.dest_branch
67 else:
68 branch_merge = self.manifest.default.revisionExpr
69
70 try:
71 ret = project.StartBranch(
72 nb, branch_merge=branch_merge, revision=opt.revision)
73 except Exception as e:
74 print('error: unable to checkout %s: %s' % (project.name, e), file=sys.stderr)
75 ret = False
76 return (ret, project)
77
Mike Frysingerae6cb082019-08-27 01:10:59 -040078 def Execute(self, opt, args):
79 nb = args[0]
Shawn O. Pearce0a389e92009-04-10 16:21:18 -070080 err = []
Ficus Kirkpatrick6f6cd772009-04-22 17:27:12 -070081 projects = []
82 if not opt.all:
83 projects = args[1:]
84 if len(projects) < 1:
David Pursehouse54a4e602020-02-12 14:31:05 +090085 projects = ['.'] # start it in the local project by default
Ficus Kirkpatrick6f6cd772009-04-22 17:27:12 -070086
Dan Willemsen04197a52015-10-07 16:53:10 -070087 all_projects = self.GetProjects(projects,
88 missing_ok=bool(self.gitc_manifest))
89
90 # This must happen after we find all_projects, since GetProjects may need
91 # the local directory, which will disappear once we save the GITC manifest.
Simran Basib9a1b732015-08-20 12:19:28 -070092 if self.gitc_manifest:
Dan Willemsen04197a52015-10-07 16:53:10 -070093 gitc_projects = self.GetProjects(projects, manifest=self.gitc_manifest,
94 missing_ok=True)
95 for project in gitc_projects:
Simran Basib9a1b732015-08-20 12:19:28 -070096 if project.old_revision:
97 project.already_synced = True
98 else:
99 project.already_synced = False
100 project.old_revision = project.revisionExpr
Simran Basib9a1b732015-08-20 12:19:28 -0700101 project.revisionExpr = None
102 # Save the GITC manifest.
103 gitc_utils.save_manifest(self.gitc_manifest)
Shawn O. Pearce0f0dfa32009-04-18 14:53:39 -0700104
Dan Willemsen04197a52015-10-07 16:53:10 -0700105 # Make sure we have a valid CWD
106 if not os.path.exists(os.getcwd()):
107 os.chdir(self.manifest.topdir)
108
Mike Frysinger151701e2021-04-13 15:07:21 -0400109 pm = Progress('Syncing %s' % nb, len(all_projects), quiet=opt.quiet)
Mike Frysinger8dbc07a2021-02-18 23:37:33 -0500110 for project in all_projects:
Dan Willemsen5ea32d12015-09-08 13:27:20 -0700111 gitc_project = self.gitc_manifest.paths[project.relpath]
112 # Sync projects that have not been opened.
Simran Basib9a1b732015-08-20 12:19:28 -0700113 if not gitc_project.already_synced:
114 proj_localdir = os.path.join(self.gitc_manifest.gitc_client_dir,
115 project.relpath)
116 project.worktree = proj_localdir
117 if not os.path.exists(proj_localdir):
118 os.makedirs(proj_localdir)
119 project.Sync_NetworkHalf()
120 sync_buf = SyncBuffer(self.manifest.manifestProject.config)
121 project.Sync_LocalHalf(sync_buf)
Dan Willemsen5ea32d12015-09-08 13:27:20 -0700122 project.revisionId = gitc_project.old_revision
Mike Frysinger8dbc07a2021-02-18 23:37:33 -0500123 pm.update()
124 pm.end()
Simran Basib9a1b732015-08-20 12:19:28 -0700125
Mike Frysinger8dbc07a2021-02-18 23:37:33 -0500126 def _ProcessResults(results):
127 for (result, project) in results:
128 if not result:
129 err.append(project)
130 pm.update()
Dan Willemsen5ea32d12015-09-08 13:27:20 -0700131
Mike Frysinger151701e2021-04-13 15:07:21 -0400132 pm = Progress('Starting %s' % nb, len(all_projects), quiet=opt.quiet)
Mike Frysinger8dbc07a2021-02-18 23:37:33 -0500133 # NB: Multiprocessing is heavy, so don't spin it up for one job.
134 if len(all_projects) == 1 or opt.jobs == 1:
135 _ProcessResults(self._ExecuteOne(opt, nb, x) for x in all_projects)
136 else:
137 with multiprocessing.Pool(opt.jobs) as pool:
138 results = pool.imap_unordered(
139 functools.partial(self._ExecuteOne, opt, nb), all_projects,
140 chunksize=WORKER_BATCH_SIZE)
141 _ProcessResults(results)
Shawn O. Pearce0f0dfa32009-04-18 14:53:39 -0700142 pm.end()
Shawn O. Pearce0a389e92009-04-10 16:21:18 -0700143
144 if err:
Shawn O. Pearce0a389e92009-04-10 16:21:18 -0700145 for p in err:
Sarah Owenscecd1d82012-11-01 22:59:27 -0700146 print("error: %s/: cannot start %s" % (p.relpath, nb),
147 file=sys.stderr)
Shawn O. Pearce0a389e92009-04-10 16:21:18 -0700148 sys.exit(1)