The Android Open Source Project | cf31fe9 | 2008-10-21 07:00:00 -0700 | [diff] [blame] | 1 | # 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 Frysinger | 8dbc07a | 2021-02-18 23:37:33 -0500 | [diff] [blame^] | 15 | import functools |
| 16 | import multiprocessing |
Simran Basi | b9a1b73 | 2015-08-20 12:19:28 -0700 | [diff] [blame] | 17 | import os |
The Android Open Source Project | cf31fe9 | 2008-10-21 07:00:00 -0700 | [diff] [blame] | 18 | import sys |
Simran Basi | b9a1b73 | 2015-08-20 12:19:28 -0700 | [diff] [blame] | 19 | |
Mike Frysinger | 8dbc07a | 2021-02-18 23:37:33 -0500 | [diff] [blame^] | 20 | from command import Command, DEFAULT_LOCAL_JOBS, WORKER_BATCH_SIZE |
Zac Livingston | 9ead97b | 2017-06-13 08:29:04 -0600 | [diff] [blame] | 21 | from git_config import IsImmutable |
The Android Open Source Project | cf31fe9 | 2008-10-21 07:00:00 -0700 | [diff] [blame] | 22 | from git_command import git |
Simran Basi | b9a1b73 | 2015-08-20 12:19:28 -0700 | [diff] [blame] | 23 | import gitc_utils |
Shawn O. Pearce | 0f0dfa3 | 2009-04-18 14:53:39 -0700 | [diff] [blame] | 24 | from progress import Progress |
Simran Basi | b9a1b73 | 2015-08-20 12:19:28 -0700 | [diff] [blame] | 25 | from project import SyncBuffer |
The Android Open Source Project | cf31fe9 | 2008-10-21 07:00:00 -0700 | [diff] [blame] | 26 | |
David Pursehouse | 819827a | 2020-02-12 15:20:19 +0900 | [diff] [blame] | 27 | |
The Android Open Source Project | cf31fe9 | 2008-10-21 07:00:00 -0700 | [diff] [blame] | 28 | class Start(Command): |
| 29 | common = True |
| 30 | helpSummary = "Start a new branch for development" |
| 31 | helpUsage = """ |
Ficus Kirkpatrick | 6f6cd77 | 2009-04-22 17:27:12 -0700 | [diff] [blame] | 32 | %prog <newbranchname> [--all | <project>...] |
Shawn O. Pearce | 06e556d | 2009-04-18 11:19:01 -0700 | [diff] [blame] | 33 | """ |
| 34 | helpDescription = """ |
| 35 | '%prog' begins a new branch of development, starting from the |
| 36 | revision specified in the manifest. |
The Android Open Source Project | cf31fe9 | 2008-10-21 07:00:00 -0700 | [diff] [blame] | 37 | """ |
Mike Frysinger | 8dbc07a | 2021-02-18 23:37:33 -0500 | [diff] [blame^] | 38 | PARALLEL_JOBS = DEFAULT_LOCAL_JOBS |
The Android Open Source Project | cf31fe9 | 2008-10-21 07:00:00 -0700 | [diff] [blame] | 39 | |
Ficus Kirkpatrick | 6f6cd77 | 2009-04-22 17:27:12 -0700 | [diff] [blame] | 40 | def _Options(self, p): |
Mike Frysinger | 8dbc07a | 2021-02-18 23:37:33 -0500 | [diff] [blame^] | 41 | super()._Options(p) |
Ficus Kirkpatrick | 6f6cd77 | 2009-04-22 17:27:12 -0700 | [diff] [blame] | 42 | p.add_option('--all', |
| 43 | dest='all', action='store_true', |
| 44 | help='begin branch in all projects') |
Theodore Dubois | 60fdc5c | 2019-07-30 12:14:25 -0700 | [diff] [blame] | 45 | p.add_option('-r', '--rev', '--revision', dest='revision', |
| 46 | help='point branch at this revision instead of upstream') |
| 47 | p.add_option('--head', dest='revision', action='store_const', const='HEAD', |
| 48 | help='abbreviation for --rev HEAD') |
Ficus Kirkpatrick | 6f6cd77 | 2009-04-22 17:27:12 -0700 | [diff] [blame] | 49 | |
Mike Frysinger | ae6cb08 | 2019-08-27 01:10:59 -0400 | [diff] [blame] | 50 | def ValidateOptions(self, opt, args): |
The Android Open Source Project | cf31fe9 | 2008-10-21 07:00:00 -0700 | [diff] [blame] | 51 | if not args: |
| 52 | self.Usage() |
| 53 | |
| 54 | nb = args[0] |
| 55 | if not git.check_ref_format('heads/%s' % nb): |
Mike Frysinger | ae6cb08 | 2019-08-27 01:10:59 -0400 | [diff] [blame] | 56 | self.OptionParser.error("'%s' is not a valid name" % nb) |
The Android Open Source Project | cf31fe9 | 2008-10-21 07:00:00 -0700 | [diff] [blame] | 57 | |
Mike Frysinger | 8dbc07a | 2021-02-18 23:37:33 -0500 | [diff] [blame^] | 58 | 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 Frysinger | ae6cb08 | 2019-08-27 01:10:59 -0400 | [diff] [blame] | 78 | def Execute(self, opt, args): |
| 79 | nb = args[0] |
Shawn O. Pearce | 0a389e9 | 2009-04-10 16:21:18 -0700 | [diff] [blame] | 80 | err = [] |
Ficus Kirkpatrick | 6f6cd77 | 2009-04-22 17:27:12 -0700 | [diff] [blame] | 81 | projects = [] |
| 82 | if not opt.all: |
| 83 | projects = args[1:] |
| 84 | if len(projects) < 1: |
David Pursehouse | 54a4e60 | 2020-02-12 14:31:05 +0900 | [diff] [blame] | 85 | projects = ['.'] # start it in the local project by default |
Ficus Kirkpatrick | 6f6cd77 | 2009-04-22 17:27:12 -0700 | [diff] [blame] | 86 | |
Dan Willemsen | 04197a5 | 2015-10-07 16:53:10 -0700 | [diff] [blame] | 87 | 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 Basi | b9a1b73 | 2015-08-20 12:19:28 -0700 | [diff] [blame] | 92 | if self.gitc_manifest: |
Dan Willemsen | 04197a5 | 2015-10-07 16:53:10 -0700 | [diff] [blame] | 93 | gitc_projects = self.GetProjects(projects, manifest=self.gitc_manifest, |
| 94 | missing_ok=True) |
| 95 | for project in gitc_projects: |
Simran Basi | b9a1b73 | 2015-08-20 12:19:28 -0700 | [diff] [blame] | 96 | if project.old_revision: |
| 97 | project.already_synced = True |
| 98 | else: |
| 99 | project.already_synced = False |
| 100 | project.old_revision = project.revisionExpr |
Simran Basi | b9a1b73 | 2015-08-20 12:19:28 -0700 | [diff] [blame] | 101 | project.revisionExpr = None |
| 102 | # Save the GITC manifest. |
| 103 | gitc_utils.save_manifest(self.gitc_manifest) |
Shawn O. Pearce | 0f0dfa3 | 2009-04-18 14:53:39 -0700 | [diff] [blame] | 104 | |
Dan Willemsen | 04197a5 | 2015-10-07 16:53:10 -0700 | [diff] [blame] | 105 | # Make sure we have a valid CWD |
| 106 | if not os.path.exists(os.getcwd()): |
| 107 | os.chdir(self.manifest.topdir) |
| 108 | |
Mike Frysinger | 8dbc07a | 2021-02-18 23:37:33 -0500 | [diff] [blame^] | 109 | pm = Progress('Syncing %s' % nb, len(all_projects)) |
| 110 | for project in all_projects: |
Dan Willemsen | 5ea32d1 | 2015-09-08 13:27:20 -0700 | [diff] [blame] | 111 | gitc_project = self.gitc_manifest.paths[project.relpath] |
| 112 | # Sync projects that have not been opened. |
Simran Basi | b9a1b73 | 2015-08-20 12:19:28 -0700 | [diff] [blame] | 113 | 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 Willemsen | 5ea32d1 | 2015-09-08 13:27:20 -0700 | [diff] [blame] | 122 | project.revisionId = gitc_project.old_revision |
Mike Frysinger | 8dbc07a | 2021-02-18 23:37:33 -0500 | [diff] [blame^] | 123 | pm.update() |
| 124 | pm.end() |
Simran Basi | b9a1b73 | 2015-08-20 12:19:28 -0700 | [diff] [blame] | 125 | |
Mike Frysinger | 8dbc07a | 2021-02-18 23:37:33 -0500 | [diff] [blame^] | 126 | def _ProcessResults(results): |
| 127 | for (result, project) in results: |
| 128 | if not result: |
| 129 | err.append(project) |
| 130 | pm.update() |
Dan Willemsen | 5ea32d1 | 2015-09-08 13:27:20 -0700 | [diff] [blame] | 131 | |
Mike Frysinger | 8dbc07a | 2021-02-18 23:37:33 -0500 | [diff] [blame^] | 132 | pm = Progress('Starting %s' % nb, len(all_projects)) |
| 133 | # 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. Pearce | 0f0dfa3 | 2009-04-18 14:53:39 -0700 | [diff] [blame] | 142 | pm.end() |
Shawn O. Pearce | 0a389e9 | 2009-04-10 16:21:18 -0700 | [diff] [blame] | 143 | |
| 144 | if err: |
Shawn O. Pearce | 0a389e9 | 2009-04-10 16:21:18 -0700 | [diff] [blame] | 145 | for p in err: |
Sarah Owens | cecd1d8 | 2012-11-01 22:59:27 -0700 | [diff] [blame] | 146 | print("error: %s/: cannot start %s" % (p.relpath, nb), |
| 147 | file=sys.stderr) |
Shawn O. Pearce | 0a389e9 | 2009-04-10 16:21:18 -0700 | [diff] [blame] | 148 | sys.exit(1) |