X-Git-Url: https://www.chiark.greenend.org.uk/ucgi/~mdw/git/stgit/blobdiff_plain/ddbbfd845a3813658370c26a7d7ea91ac6f57fae..f5f22afe39694a687b67077e1789600dc9ff47c9:/stgit/commands/pull.py diff --git a/stgit/commands/pull.py b/stgit/commands/pull.py index 2aba444..bec7fd7 100644 --- a/stgit/commands/pull.py +++ b/stgit/commands/pull.py @@ -20,74 +20,90 @@ from optparse import OptionParser, make_option from stgit.commands.common import * from stgit.utils import * +from stgit.out import * +from stgit.config import GitConfigException from stgit import stack, git help = 'pull the changes from the remote repository' -usage = """%prog [options] [] +usage = """%prog [options] [] -Pull the latest changes from the given URL or branch (defaulting to -'origin'). This command works by popping all the patches from the -stack, pulling the changes in the parent repository, setting the base -of the stack to the latest parent HEAD and pusing the patches back -(unless '--nopush' is specified). The 'push' operation can fail if -there are conflicts. They need to be resolved and the patch pushed -again. +Pull the latest changes from the given remote repository (defaulting +to branch..remote, or 'origin' if not set). This command works +by popping all the patches from the stack, pulling the changes in the +parent repository, setting the base of the stack to the latest parent +HEAD and pushing the patches back (unless '--nopush' is specified). +The 'push' operation can fail if there are conflicts. They need to be +resolved and the patch pushed again. -Note that this command doesn't perform any merge operation for the -base of the stack, it only performs merges with the patches being -pushed.""" +Check the 'git fetch' documentation for the format.""" +directory = DirectoryGotoToplevel() options = [make_option('-n', '--nopush', help = 'do not push the patches back after pulling', action = 'store_true'), - make_option('--head', metavar='OTHER_HEAD', - help = 'pull OTHER_HEAD instead of HEAD'), - make_option('--tag', - help = 'pull TAG')] - + make_option('-m', '--merged', + help = 'check for patches merged upstream', + action = 'store_true')] def func(parser, options, args): """Pull the changes from a remote repository """ - if len(args) == 0: - location = read_string(os.path.join(git.base_dir, 'branches', - 'origin')) - elif len(args) == 1: - location = args[0] - branch = os.path.join(git.base_dir, 'branches', location) - if os.path.isfile(branch): - location = read_string(branch) + policy = config.get('branch.%s.stgit.pull-policy' % crt_series.get_name()) or \ + config.get('stgit.pull-policy') + + if policy == 'rebase': + # parent is local + if len(args) == 1: + parser.error('specifying a repository is meaningless for policy="%s"' % policy) + if len(args) > 0: + parser.error('incorrect number of arguments') + else: - parser.error('incorrect number of arguments') + # parent is remote + if len(args) > 1: + parser.error('incorrect number of arguments') + + if len(args) >= 1: + repository = args[0] + else: + repository = crt_series.get_parent_remote() + + if crt_series.get_protected(): + raise CmdException, 'This branch is protected. Pulls are not permitted' check_local_changes() check_conflicts() - check_head_top_equal() + check_head_top_equal(crt_series) - # pop all patches - applied = crt_series.get_applied() - if len(applied) > 0: - print 'Popping all patches...', - sys.stdout.flush() - crt_series.pop_patch(applied[0]) - print 'done' + if policy not in ['pull', 'fetch-rebase', 'rebase']: + raise GitConfigException, 'Unsupported pull-policy "%s"' % policy - # pull the remote changes - print 'Pulling from "%s"...' % location - git.pull(location, options.head, options.tag) - print 'done' - - # push the patches back - if options.nopush: - applied = [] - for p in applied: - print 'Pushing patch "%s"...' % p, - sys.stdout.flush() - crt_series.push_patch(p) - if crt_series.empty_patch(p): - print 'done (empty patch)' - else: - print 'done' + applied = prepare_rebase(crt_series) - print_crt_patch() + # pull the remote changes + if policy == 'pull': + out.info('Pulling from "%s"' % repository) + git.pull(repository) + elif policy == 'fetch-rebase': + out.info('Fetching from "%s"' % repository) + git.fetch(repository) + try: + target = git.fetch_head() + except git.GitException: + out.error('Could not find the remote head to rebase onto - fix branch.%s.merge in .git/config' % crt_series.get_name()) + out.error('Pushing any patches back...') + post_rebase(crt_series, applied, False, False) + raise + + rebase(crt_series, target) + elif policy == 'rebase': + rebase(crt_series, crt_series.get_parent_branch()) + + post_rebase(crt_series, applied, options.nopush, options.merged) + + # maybe tidy up + if config.get('stgit.keepoptimized') == 'yes': + git.repack() + + print_crt_patch(crt_series)