@@ -3628,19 +3628,63 @@ def cmd_diagram(parser, options):
3628
3628
)
3629
3629
3630
3630
3631
+ def reparent_recursively (git , start_commit , parents , end_commit ):
3632
+ """Change the parents of start_commit and its descendants.
3633
+
3634
+ Change start_commit to have the specified parents, and reparent
3635
+ all commits on the ancestry path between start_commit and
3636
+ end_commit accordingly. Return the replacement end_commit.
3637
+ start_commit, parents, and end_commit must all be resolved OIDs.
3638
+
3639
+ """
3640
+
3641
+ # A map {old_oid : new_oid} keeping track of which replacements
3642
+ # have to be made:
3643
+ replacements = {}
3644
+
3645
+ # Reparent start_commit:
3646
+ replacements [start_commit ] = git .reparent (start_commit , parents )
3647
+
3648
+ for (commit , parents ) in git .rev_list_with_parents (
3649
+ '--ancestry-path' , '--topo-order' , '--reverse' ,
3650
+ '%s..%s' % (start_commit , end_commit )
3651
+ ):
3652
+ parents = [replacements .get (p , p ) for p in parents ]
3653
+ replacements [commit ] = git .reparent (commit , parents )
3654
+
3655
+ try :
3656
+ return replacements [end_commit ]
3657
+ except KeyError :
3658
+ raise ValueError (
3659
+ "%s is not an ancestor of %s" % (start_commit , end_commit ),
3660
+ )
3661
+
3662
+
3631
3663
def cmd_reparent (parser , options ):
3632
3664
git = GitRepository ()
3633
3665
try :
3634
- commit_sha1 = git .get_commit_sha1 ('HEAD' )
3666
+ commit = git .get_commit_sha1 (options .commit )
3667
+ except ValueError :
3668
+ sys .exit ('%s is not a valid commit' , options .commit )
3669
+
3670
+ try :
3671
+ head = git .get_commit_sha1 ('HEAD' )
3635
3672
except ValueError :
3636
3673
sys .exit ('HEAD is not a valid commit' )
3637
3674
3638
3675
try :
3639
- parent_sha1s = [git .get_commit_sha1 (p ) for p in options .parents ]
3676
+ parents = [git .get_commit_sha1 (p ) for p in options .parents ]
3677
+ except ValueError as e :
3678
+ sys .exit (e .message )
3679
+
3680
+ sys .stderr .write ('Reparenting %s..HEAD\n ' % (options .commit ,))
3681
+
3682
+ try :
3683
+ new_head = reparent_recursively (git , commit , parents , head )
3640
3684
except ValueError as e :
3641
3685
sys .exit (e .message )
3642
3686
3643
- sys .stdout .write ('%s\n ' % (git . reparent ( commit_sha1 , parent_sha1s ) ,))
3687
+ sys .stdout .write ('%s\n ' % (new_head ,))
3644
3688
3645
3689
3646
3690
def main (args ):
@@ -3927,10 +3971,24 @@ def main(args):
3927
3971
3928
3972
subparser = subparsers .add_parser (
3929
3973
'reparent' ,
3930
- help = 'change the parents of the HEAD commit' ,
3974
+ help = (
3975
+ 'change the parents of the specified commit and propagate the '
3976
+ 'change to HEAD'
3977
+ ),
3931
3978
)
3932
3979
subparser .add_argument (
3933
- 'parents' , nargs = '*' , help = '[PARENT...]' ,
3980
+ '--commit' , metavar = 'COMMIT' , default = 'HEAD' ,
3981
+ help = (
3982
+ 'target commit to reparent. Create a new commit identical to '
3983
+ 'this one, but having the specified parents. Then create '
3984
+ 'new versions of all descendants of this commit all the way to '
3985
+ 'HEAD, incorporating the modified commit. Output the SHA-1 of '
3986
+ 'the replacement HEAD commit.'
3987
+ ),
3988
+ )
3989
+ subparser .add_argument (
3990
+ 'parents' , nargs = '*' , metavar = 'PARENT' ,
3991
+ help = 'a list of commits' ,
3934
3992
)
3935
3993
3936
3994
options = parser .parse_args (args )
0 commit comments