| # Copyright (C) 2009 The Android Open Source Project |
| # |
| # Licensed under the Apache License, Version 2.0 (the "License"); |
| # you may not use this file except in compliance with the License. |
| # You may obtain a copy of the License at |
| # |
| # http://www.apache.org/licenses/LICENSE-2.0 |
| # |
| # Unless required by applicable law or agreed to in writing, software |
| # distributed under the License is distributed on an "AS IS" BASIS, |
| # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| # See the License for the specific language governing permissions and |
| # limitations under the License. |
| |
| import json |
| import os |
| import sys |
| |
| from command import PagedCommand |
| from repo_logging import RepoLogger |
| |
| |
| logger = RepoLogger(__file__) |
| |
| |
| class Manifest(PagedCommand): |
| COMMON = False |
| helpSummary = "Manifest inspection utility" |
| helpUsage = """ |
| %prog [-o {-|NAME.xml}] [-m MANIFEST.xml] [-r] |
| """ |
| _helpDescription = """ |
| |
| With the -o option, exports the current manifest for inspection. |
| The manifest and (if present) local_manifests/ are combined |
| together to produce a single manifest file. This file can be stored |
| in a Git repository for use during future 'repo init' invocations. |
| |
| The -r option can be used to generate a manifest file with project |
| revisions set to the current commit hash. These are known as |
| "revision locked manifests", as they don't follow a particular branch. |
| In this case, the 'upstream' attribute is set to the ref we were on |
| when the manifest was generated. The 'dest-branch' attribute is set |
| to indicate the remote ref to push changes to via 'repo upload'. |
| """ |
| |
| @property |
| def helpDescription(self): |
| helptext = self._helpDescription + "\n" |
| r = os.path.dirname(__file__) |
| r = os.path.dirname(r) |
| with open(os.path.join(r, "docs", "manifest-format.md")) as fd: |
| for line in fd: |
| helptext += line |
| return helptext |
| |
| def _Options(self, p): |
| p.add_option( |
| "-r", |
| "--revision-as-HEAD", |
| dest="peg_rev", |
| action="store_true", |
| help="save revisions as current HEAD", |
| ) |
| p.add_option( |
| "-m", |
| "--manifest-name", |
| help="temporary manifest to use for this sync", |
| metavar="NAME.xml", |
| ) |
| p.add_option( |
| "--suppress-upstream-revision", |
| dest="peg_rev_upstream", |
| default=True, |
| action="store_false", |
| help="if in -r mode, do not write the upstream field " |
| "(only of use if the branch names for a sha1 manifest are " |
| "sensitive)", |
| ) |
| p.add_option( |
| "--suppress-dest-branch", |
| dest="peg_rev_dest_branch", |
| default=True, |
| action="store_false", |
| help="if in -r mode, do not write the dest-branch field " |
| "(only of use if the branch names for a sha1 manifest are " |
| "sensitive)", |
| ) |
| p.add_option( |
| "--json", |
| default=False, |
| action="store_true", |
| help="output manifest in JSON format (experimental)", |
| ) |
| p.add_option( |
| "--pretty", |
| default=False, |
| action="store_true", |
| help="format output for humans to read", |
| ) |
| p.add_option( |
| "--no-local-manifests", |
| default=False, |
| action="store_true", |
| dest="ignore_local_manifests", |
| help="ignore local manifests", |
| ) |
| p.add_option( |
| "-o", |
| "--output-file", |
| dest="output_file", |
| default="-", |
| help="file to save the manifest to. (Filename prefix for " |
| "multi-tree.)", |
| metavar="-|NAME.xml", |
| ) |
| |
| def _Output(self, opt): |
| # If alternate manifest is specified, override the manifest file that |
| # we're using. |
| if opt.manifest_name: |
| self.manifest.Override(opt.manifest_name, False) |
| |
| for manifest in self.ManifestList(opt): |
| output_file = opt.output_file |
| if output_file == "-": |
| fd = sys.stdout |
| else: |
| if manifest.path_prefix: |
| output_file = ( |
| f"{opt.output_file}:" |
| f'{manifest.path_prefix.replace("/", "%2f")}' |
| ) |
| fd = open(output_file, "w") |
| |
| manifest.SetUseLocalManifests(not opt.ignore_local_manifests) |
| |
| if opt.json: |
| logger.warning("warning: --json is experimental!") |
| doc = manifest.ToDict( |
| peg_rev=opt.peg_rev, |
| peg_rev_upstream=opt.peg_rev_upstream, |
| peg_rev_dest_branch=opt.peg_rev_dest_branch, |
| ) |
| |
| json_settings = { |
| # JSON style guide says Unicode characters are fully |
| # allowed. |
| "ensure_ascii": False, |
| # We use 2 space indent to match JSON style guide. |
| "indent": 2 if opt.pretty else None, |
| "separators": (",", ": ") if opt.pretty else (",", ":"), |
| "sort_keys": True, |
| } |
| fd.write(json.dumps(doc, **json_settings)) |
| else: |
| manifest.Save( |
| fd, |
| peg_rev=opt.peg_rev, |
| peg_rev_upstream=opt.peg_rev_upstream, |
| peg_rev_dest_branch=opt.peg_rev_dest_branch, |
| ) |
| if output_file != "-": |
| fd.close() |
| if manifest.path_prefix: |
| logger.warning( |
| "Saved %s submanifest to %s", |
| manifest.path_prefix, |
| output_file, |
| ) |
| else: |
| logger.warning("Saved manifest to %s", output_file) |
| |
| def ValidateOptions(self, opt, args): |
| if args: |
| self.Usage() |
| |
| def Execute(self, opt, args): |
| self._Output(opt) |