# Copyright (C) 2008 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 functools import io from command import DEFAULT_LOCAL_JOBS from command import PagedCommand class Diff(PagedCommand): COMMON = True helpSummary = "Show changes between commit and working tree" helpUsage = """ %prog [...] The -u option causes '%prog' to generate diff output with file paths relative to the repository root, so the output can be applied to the Unix 'patch' command. """ PARALLEL_JOBS = DEFAULT_LOCAL_JOBS def _Options(self, p): p.add_option( "-u", "--absolute", dest="absolute", action="store_true", help="paths are relative to the repository root", ) @classmethod def _ExecuteOne(cls, absolute, local, project_idx): """Obtains the diff for a specific project. Args: absolute: Paths are relative to the root. local: a boolean, if True, the path is relative to the local (sub)manifest. If false, the path is relative to the outermost manifest. project_idx: Project index to get status of. Returns: The status of the project. """ buf = io.StringIO() project = cls.get_parallel_context()["projects"][project_idx] ret = project.PrintWorkTreeDiff(absolute, output_redir=buf, local=local) return (ret, buf.getvalue()) def Execute(self, opt, args): all_projects = self.GetProjects( args, all_manifests=not opt.this_manifest_only ) def _ProcessResults(_pool, _output, results): ret = 0 for state, output in results: if output: print(output, end="") if not state: ret = 1 return ret with self.ParallelContext(): self.get_parallel_context()["projects"] = all_projects return self.ExecuteInParallel( opt.jobs, functools.partial( self._ExecuteOne, opt.absolute, opt.this_manifest_only ), range(len(all_projects)), callback=_ProcessResults, ordered=True, chunksize=1, )