Skip to content

Commit e7d1b2b

Browse files
pks-tpks-gitlab
andauthored
rebase: Add wrapper for git_rebase_inmemory_index() (#900)
* rebase: Fix missing initialization of the repo pointer While the `Rebase` structure has a pointer to the repository the rebase is creatde in, this pointer isn't ever initialized. Fix this. * rebase: Add wrapper for `git_rebase_inmemory_index()` Add a new wrapper for `git_rebase_inmemory_index()`, which can be used to retrieve the index for an in-memory rebase. Co-authored-by: Patrick Steinhardt <[email protected]>
1 parent c598ea5 commit e7d1b2b

File tree

2 files changed

+99
-4
lines changed

2 files changed

+99
-4
lines changed

rebase.go

+25-4
Original file line numberDiff line numberDiff line change
@@ -318,7 +318,7 @@ func (r *Repository) InitRebase(branch *AnnotatedCommit, upstream *AnnotatedComm
318318
return nil, MakeGitError(ret)
319319
}
320320

321-
return newRebaseFromC(ptr, cOpts), nil
321+
return newRebaseFromC(ptr, r, cOpts), nil
322322
}
323323

324324
// OpenRebase opens an existing rebase that was previously started by either an invocation of InitRebase or by another client.
@@ -340,7 +340,7 @@ func (r *Repository) OpenRebase(opts *RebaseOptions) (*Rebase, error) {
340340
return nil, MakeGitError(ret)
341341
}
342342

343-
return newRebaseFromC(ptr, cOpts), nil
343+
return newRebaseFromC(ptr, r, cOpts), nil
344344
}
345345

346346
// OperationAt gets the rebase operation specified by the given index.
@@ -392,6 +392,27 @@ func (rebase *Rebase) Next() (*RebaseOperation, error) {
392392
return newRebaseOperationFromC(ptr), nil
393393
}
394394

395+
// InmemoryIndex gets the index produced by the last operation, which is the
396+
// result of `Next()` and which will be committed by the next invocation of
397+
// `Commit()`. This is useful for resolving conflicts in an in-memory rebase
398+
// before committing them.
399+
//
400+
// This is only applicable for in-memory rebases; for rebases within a working
401+
// directory, the changes were applied to the repository's index.
402+
func (rebase *Rebase) InmemoryIndex() (*Index, error) {
403+
runtime.LockOSThread()
404+
defer runtime.UnlockOSThread()
405+
406+
var ptr *C.git_index
407+
err := C.git_rebase_inmemory_index(&ptr, rebase.ptr)
408+
runtime.KeepAlive(rebase)
409+
if err < 0 {
410+
return nil, MakeGitError(err)
411+
}
412+
413+
return newIndexFromC(ptr, rebase.r), nil
414+
}
415+
395416
// Commit commits the current patch.
396417
// You must have resolved any conflicts that were introduced during the patch application from the Next() invocation.
397418
func (rebase *Rebase) Commit(ID *Oid, author, committer *Signature, message string) error {
@@ -457,8 +478,8 @@ func (r *Rebase) Free() {
457478
freeRebaseOptions(r.options)
458479
}
459480

460-
func newRebaseFromC(ptr *C.git_rebase, opts *C.git_rebase_options) *Rebase {
461-
rebase := &Rebase{ptr: ptr, options: opts}
481+
func newRebaseFromC(ptr *C.git_rebase, repo *Repository, opts *C.git_rebase_options) *Rebase {
482+
rebase := &Rebase{ptr: ptr, r: repo, options: opts}
462483
runtime.SetFinalizer(rebase, (*Rebase).Free)
463484
return rebase
464485
}

rebase_test.go

+74
Original file line numberDiff line numberDiff line change
@@ -14,6 +14,80 @@ import (
1414

1515
// Tests
1616

17+
func TestRebaseInMemoryWithConflict(t *testing.T) {
18+
repo := createTestRepo(t)
19+
defer cleanupTestRepo(t, repo)
20+
seedTestRepo(t, repo)
21+
22+
// Create two branches with common history, where both modify "common-file"
23+
// in a conflicting way.
24+
_, err := commitSomething(repo, "common-file", "a\nb\nc\n", commitOptions{})
25+
checkFatal(t, err)
26+
checkFatal(t, createBranch(repo, "branch-a"))
27+
checkFatal(t, createBranch(repo, "branch-b"))
28+
29+
checkFatal(t, repo.SetHead("refs/heads/branch-a"))
30+
_, err = commitSomething(repo, "common-file", "1\nb\nc\n", commitOptions{})
31+
checkFatal(t, err)
32+
33+
checkFatal(t, repo.SetHead("refs/heads/branch-b"))
34+
_, err = commitSomething(repo, "common-file", "x\nb\nc\n", commitOptions{})
35+
checkFatal(t, err)
36+
37+
branchA, err := repo.LookupBranch("branch-a", BranchLocal)
38+
checkFatal(t, err)
39+
onto, err := repo.AnnotatedCommitFromRef(branchA.Reference)
40+
checkFatal(t, err)
41+
42+
// We then rebase "branch-b" onto "branch-a" in-memory, which should result
43+
// in a conflict.
44+
rebase, err := repo.InitRebase(nil, nil, onto, &RebaseOptions{InMemory: 1})
45+
checkFatal(t, err)
46+
47+
_, err = rebase.Next()
48+
checkFatal(t, err)
49+
50+
index, err := rebase.InmemoryIndex()
51+
checkFatal(t, err)
52+
53+
// We simply resolve the conflict and commit the rebase.
54+
if !index.HasConflicts() {
55+
t.Fatal("expected index to have conflicts")
56+
}
57+
58+
conflict, err := index.Conflict("common-file")
59+
checkFatal(t, err)
60+
61+
resolvedBlobID, err := repo.CreateBlobFromBuffer([]byte("resolved contents"))
62+
checkFatal(t, err)
63+
64+
resolvedEntry := *conflict.Our
65+
resolvedEntry.Id = resolvedBlobID
66+
checkFatal(t, index.Add(&resolvedEntry))
67+
checkFatal(t, index.RemoveConflict("common-file"))
68+
69+
var commitID Oid
70+
checkFatal(t, rebase.Commit(&commitID, signature(), signature(), "rebased message"))
71+
checkFatal(t, rebase.Finish())
72+
73+
// And then assert that we can look up the new merge commit, and that the
74+
// "common-file" has the expected contents.
75+
commit, err := repo.LookupCommit(&commitID)
76+
checkFatal(t, err)
77+
if commit.Message() != "rebased message" {
78+
t.Fatalf("unexpected commit message %q", commit.Message())
79+
}
80+
81+
tree, err := commit.Tree()
82+
checkFatal(t, err)
83+
84+
blob, err := repo.LookupBlob(tree.EntryByName("common-file").Id)
85+
checkFatal(t, err)
86+
if string(blob.Contents()) != "resolved contents" {
87+
t.Fatalf("unexpected resolved blob contents %q", string(blob.Contents()))
88+
}
89+
}
90+
1791
func TestRebaseAbort(t *testing.T) {
1892
// TEST DATA
1993

0 commit comments

Comments
 (0)