summaryrefslogtreecommitdiff
path: root/mempack.go
diff options
context:
space:
mode:
authorlhchavez <[email protected]>2020-02-23 15:08:45 +0000
committerlhchavez <[email protected]>2020-02-23 15:08:45 +0000
commit3c88bd9f1aebc53ea9d77937829f8c155aa834c3 (patch)
tree8556d00fb42f2fcab5ba748e49e21e1141b9d604 /mempack.go
parentc75e0221d70bc471917aa3de34ca1ead1a747987 (diff)
parent21d618136f415486d95965e75af80c0e6688a0d5 (diff)
Merge remote-tracking branch 'upstream/master' into cherrypick-commit
Diffstat (limited to 'mempack.go')
-rw-r--r--mempack.go91
1 files changed, 91 insertions, 0 deletions
diff --git a/mempack.go b/mempack.go
new file mode 100644
index 0000000..bdea224
--- /dev/null
+++ b/mempack.go
@@ -0,0 +1,91 @@
+package git
+
+/*
+#include <git2.h>
+#include <git2/sys/mempack.h>
+
+extern int git_mempack_new(git_odb_backend **out);
+extern int git_mempack_dump(git_buf *pack, git_repository *repo, git_odb_backend *backend);
+extern int git_mempack_reset(git_odb_backend *backend);
+extern void _go_git_odb_backend_free(git_odb_backend *backend);
+*/
+import "C"
+
+import (
+ "runtime"
+ "unsafe"
+)
+
+// Mempack is a custom ODB backend that permits packing object in-memory.
+type Mempack struct {
+ ptr *C.git_odb_backend
+}
+
+// NewMempack creates a new mempack instance and registers it to the ODB.
+func NewMempack(odb *Odb) (mempack *Mempack, err error) {
+ mempack = new(Mempack)
+
+ runtime.LockOSThread()
+ defer runtime.UnlockOSThread()
+
+ ret := C.git_mempack_new(&mempack.ptr)
+ if ret < 0 {
+ return nil, MakeGitError(ret)
+ }
+
+ ret = C.git_odb_add_backend(odb.ptr, mempack.ptr, C.int(999))
+ runtime.KeepAlive(odb)
+ if ret < 0 {
+ // Since git_odb_add_alternate() takes ownership of the ODB backend, the
+ // only case in which we free the mempack's memory is if it fails to be
+ // added to the ODB.
+ C._go_git_odb_backend_free(mempack.ptr)
+ return nil, MakeGitError(ret)
+ }
+
+ return mempack, nil
+}
+
+// Dump dumps all the queued in-memory writes to a packfile.
+//
+// It is the caller's responsibility to ensure that the generated packfile is
+// available to the repository (e.g. by writing it to disk, or doing something
+// crazy like distributing it across several copies of the repository over a
+// network).
+//
+// Once the generated packfile is available to the repository, call
+// Mempack.Reset to cleanup the memory store.
+//
+// Calling Mempack.Reset before the packfile has been written to disk will
+// result in an inconsistent repository (the objects in the memory store won't
+// be accessible).
+func (mempack *Mempack) Dump(repository *Repository) ([]byte, error) {
+ buf := C.git_buf{}
+
+ runtime.LockOSThread()
+ defer runtime.UnlockOSThread()
+
+ ret := C.git_mempack_dump(&buf, repository.ptr, mempack.ptr)
+ runtime.KeepAlive(repository)
+ if ret < 0 {
+ return nil, MakeGitError(ret)
+ }
+ defer C.git_buf_dispose(&buf)
+
+ return C.GoBytes(unsafe.Pointer(buf.ptr), C.int(buf.size)), nil
+}
+
+// Reset resets the memory packer by clearing all the queued objects.
+//
+// This assumes that Mempack.Dump has been called before to store all the
+// queued objects into a single packfile.
+func (mempack *Mempack) Reset() error {
+ runtime.LockOSThread()
+ defer runtime.UnlockOSThread()
+
+ ret := C.git_mempack_reset(mempack.ptr)
+ if ret < 0 {
+ return MakeGitError(ret)
+ }
+ return nil
+}