|  | # Copyright (C) 2010 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 sys | 
|  |  | 
|  | from color import Coloring | 
|  | from command import Command | 
|  | from git_command import GitCommand | 
|  |  | 
|  |  | 
|  | class RebaseColoring(Coloring): | 
|  | def __init__(self, config): | 
|  | Coloring.__init__(self, config, 'rebase') | 
|  | self.project = self.printer('project', attr='bold') | 
|  | self.fail = self.printer('fail', fg='red') | 
|  |  | 
|  |  | 
|  | class Rebase(Command): | 
|  | COMMON = True | 
|  | helpSummary = "Rebase local branches on upstream branch" | 
|  | helpUsage = """ | 
|  | %prog {[<project>...] | -i <project>...} | 
|  | """ | 
|  | helpDescription = """ | 
|  | '%prog' uses git rebase to move local changes in the current topic branch to | 
|  | the HEAD of the upstream history, useful when you have made commits in a topic | 
|  | branch but need to incorporate new upstream changes "underneath" them. | 
|  | """ | 
|  |  | 
|  | def _Options(self, p): | 
|  | g = p.get_option_group('--quiet') | 
|  | g.add_option('-i', '--interactive', | 
|  | dest="interactive", action="store_true", | 
|  | help="interactive rebase (single project only)") | 
|  |  | 
|  | p.add_option('--fail-fast', | 
|  | dest='fail_fast', action='store_true', | 
|  | help='stop rebasing after first error is hit') | 
|  | p.add_option('-f', '--force-rebase', | 
|  | dest='force_rebase', action='store_true', | 
|  | help='pass --force-rebase to git rebase') | 
|  | p.add_option('--no-ff', | 
|  | dest='ff', default=True, action='store_false', | 
|  | help='pass --no-ff to git rebase') | 
|  | p.add_option('--autosquash', | 
|  | dest='autosquash', action='store_true', | 
|  | help='pass --autosquash to git rebase') | 
|  | p.add_option('--whitespace', | 
|  | dest='whitespace', action='store', metavar='WS', | 
|  | help='pass --whitespace to git rebase') | 
|  | p.add_option('--auto-stash', | 
|  | dest='auto_stash', action='store_true', | 
|  | help='stash local modifications before starting') | 
|  | p.add_option('-m', '--onto-manifest', | 
|  | dest='onto_manifest', action='store_true', | 
|  | help='rebase onto the manifest version instead of upstream ' | 
|  | 'HEAD (this helps to make sure the local tree stays ' | 
|  | 'consistent if you previously synced to a manifest)') | 
|  |  | 
|  | def Execute(self, opt, args): | 
|  | all_projects = self.GetProjects(args, all_manifests=not opt.this_manifest_only) | 
|  | one_project = len(all_projects) == 1 | 
|  |  | 
|  | if opt.interactive and not one_project: | 
|  | print('error: interactive rebase not supported with multiple projects', | 
|  | file=sys.stderr) | 
|  | if len(args) == 1: | 
|  | print('note: project %s is mapped to more than one path' % (args[0],), | 
|  | file=sys.stderr) | 
|  | return 1 | 
|  |  | 
|  | # Setup the common git rebase args that we use for all projects. | 
|  | common_args = ['rebase'] | 
|  | if opt.whitespace: | 
|  | common_args.append('--whitespace=%s' % opt.whitespace) | 
|  | if opt.quiet: | 
|  | common_args.append('--quiet') | 
|  | if opt.force_rebase: | 
|  | common_args.append('--force-rebase') | 
|  | if not opt.ff: | 
|  | common_args.append('--no-ff') | 
|  | if opt.autosquash: | 
|  | common_args.append('--autosquash') | 
|  | if opt.interactive: | 
|  | common_args.append('-i') | 
|  |  | 
|  | config = self.manifest.manifestProject.config | 
|  | out = RebaseColoring(config) | 
|  | out.redirect(sys.stdout) | 
|  | _RelPath = lambda p: p.RelPath(local=opt.this_manifest_only) | 
|  |  | 
|  | ret = 0 | 
|  | for project in all_projects: | 
|  | if ret and opt.fail_fast: | 
|  | break | 
|  |  | 
|  | cb = project.CurrentBranch | 
|  | if not cb: | 
|  | if one_project: | 
|  | print("error: project %s has a detached HEAD" % _RelPath(project), | 
|  | file=sys.stderr) | 
|  | return 1 | 
|  | # ignore branches with detatched HEADs | 
|  | continue | 
|  |  | 
|  | upbranch = project.GetBranch(cb) | 
|  | if not upbranch.LocalMerge: | 
|  | if one_project: | 
|  | print("error: project %s does not track any remote branches" | 
|  | % _RelPath(project), file=sys.stderr) | 
|  | return 1 | 
|  | # ignore branches without remotes | 
|  | continue | 
|  |  | 
|  | args = common_args[:] | 
|  | if opt.onto_manifest: | 
|  | args.append('--onto') | 
|  | args.append(project.revisionExpr) | 
|  |  | 
|  | args.append(upbranch.LocalMerge) | 
|  |  | 
|  | out.project('project %s: rebasing %s -> %s', | 
|  | _RelPath(project), cb, upbranch.LocalMerge) | 
|  | out.nl() | 
|  | out.flush() | 
|  |  | 
|  | needs_stash = False | 
|  | if opt.auto_stash: | 
|  | stash_args = ["update-index", "--refresh", "-q"] | 
|  |  | 
|  | if GitCommand(project, stash_args).Wait() != 0: | 
|  | needs_stash = True | 
|  | # Dirty index, requires stash... | 
|  | stash_args = ["stash"] | 
|  |  | 
|  | if GitCommand(project, stash_args).Wait() != 0: | 
|  | ret += 1 | 
|  | continue | 
|  |  | 
|  | if GitCommand(project, args).Wait() != 0: | 
|  | ret += 1 | 
|  | continue | 
|  |  | 
|  | if needs_stash: | 
|  | stash_args.append('pop') | 
|  | stash_args.append('--quiet') | 
|  | if GitCommand(project, stash_args).Wait() != 0: | 
|  | ret += 1 | 
|  |  | 
|  | if ret: | 
|  | out.fail('%i projects had errors', ret) | 
|  | out.nl() | 
|  |  | 
|  | return ret |