aboutsummaryrefslogtreecommitdiff
path: root/object/storer/chain
diff options
context:
space:
mode:
authorGravatar Runxi Yu2026-03-25 14:34:50 +0000
committerGravatar Runxi Yu2026-03-25 14:34:50 +0000
commite4a7aa0742f5070299d37e8421c99d67f0af3f90 (patch)
tree36d89781476a92e61280c5ff232a2773e4092c0e /object/storer/chain
parent*: delta -> packfile/delta (diff)
signatureNo signature
*: object/store -> object/storer v0.1.107
Diffstat (limited to 'object/storer/chain')
-rw-r--r--object/storer/chain/bytes.go46
-rw-r--r--object/storer/chain/chain.go14
-rw-r--r--object/storer/chain/close.go8
-rw-r--r--object/storer/chain/header.go28
-rw-r--r--object/storer/chain/new.go13
-rw-r--r--object/storer/chain/reader.go47
-rw-r--r--object/storer/chain/refresh.go17
-rw-r--r--object/storer/chain/size.go27
8 files changed, 200 insertions, 0 deletions
diff --git a/object/storer/chain/bytes.go b/object/storer/chain/bytes.go
new file mode 100644
index 00000000..c3ec1eb8
--- /dev/null
+++ b/object/storer/chain/bytes.go
@@ -0,0 +1,46 @@
+package chain
+
+import (
+ "errors"
+ "fmt"
+
+ objectid "codeberg.org/lindenii/furgit/object/id"
+ "codeberg.org/lindenii/furgit/object/storer"
+ objecttype "codeberg.org/lindenii/furgit/object/type"
+)
+
+// ReadBytesFull reads a full serialized object from the first backend that has it.
+func (chain *Chain) ReadBytesFull(id objectid.ObjectID) ([]byte, error) {
+ for i, backend := range chain.backends {
+ full, err := backend.ReadBytesFull(id)
+ if err == nil {
+ return full, nil
+ }
+
+ if errors.Is(err, objectstorer.ErrObjectNotFound) {
+ continue
+ }
+
+ return nil, fmt.Errorf("objectstorer: backend %d read bytes full: %w", i, err)
+ }
+
+ return nil, objectstorer.ErrObjectNotFound
+}
+
+// ReadBytesContent reads an object's type and content bytes from the first backend that has it.
+func (chain *Chain) ReadBytesContent(id objectid.ObjectID) (objecttype.Type, []byte, error) {
+ for i, backend := range chain.backends {
+ ty, content, err := backend.ReadBytesContent(id)
+ if err == nil {
+ return ty, content, nil
+ }
+
+ if errors.Is(err, objectstorer.ErrObjectNotFound) {
+ continue
+ }
+
+ return objecttype.TypeInvalid, nil, fmt.Errorf("objectstorer: backend %d read bytes content: %w", i, err)
+ }
+
+ return objecttype.TypeInvalid, nil, objectstorer.ErrObjectNotFound
+}
diff --git a/object/storer/chain/chain.go b/object/storer/chain/chain.go
new file mode 100644
index 00000000..8502b590
--- /dev/null
+++ b/object/storer/chain/chain.go
@@ -0,0 +1,14 @@
+// Package chain provides a wrapper object storage backend to query a chain of
+// backends.
+package chain
+
+import (
+ "codeberg.org/lindenii/furgit/object/storer"
+)
+
+// Chain queries multiple object databases in order.
+//
+// Chain borrows its backend stores.
+type Chain struct {
+ backends []objectstorer.Store
+}
diff --git a/object/storer/chain/close.go b/object/storer/chain/close.go
new file mode 100644
index 00000000..6bd74565
--- /dev/null
+++ b/object/storer/chain/close.go
@@ -0,0 +1,8 @@
+package chain
+
+// Close releases wrapper-local resources.
+//
+// Chain borrows its backends, so Close does not close them.
+//
+// Repeated calls to Close are undefined behavior.
+func (chain *Chain) Close() error { return nil }
diff --git a/object/storer/chain/header.go b/object/storer/chain/header.go
new file mode 100644
index 00000000..e7791e9e
--- /dev/null
+++ b/object/storer/chain/header.go
@@ -0,0 +1,28 @@
+package chain
+
+import (
+ "errors"
+ "fmt"
+
+ objectid "codeberg.org/lindenii/furgit/object/id"
+ "codeberg.org/lindenii/furgit/object/storer"
+ objecttype "codeberg.org/lindenii/furgit/object/type"
+)
+
+// ReadHeader reads object header data from the first backend that has it.
+func (chain *Chain) ReadHeader(id objectid.ObjectID) (objecttype.Type, int64, error) {
+ for i, backend := range chain.backends {
+ ty, size, err := backend.ReadHeader(id)
+ if err == nil {
+ return ty, size, nil
+ }
+
+ if errors.Is(err, objectstorer.ErrObjectNotFound) {
+ continue
+ }
+
+ return objecttype.TypeInvalid, 0, fmt.Errorf("objectstorer: backend %d read header: %w", i, err)
+ }
+
+ return objecttype.TypeInvalid, 0, objectstorer.ErrObjectNotFound
+}
diff --git a/object/storer/chain/new.go b/object/storer/chain/new.go
new file mode 100644
index 00000000..f7a4f141
--- /dev/null
+++ b/object/storer/chain/new.go
@@ -0,0 +1,13 @@
+package chain
+
+import "codeberg.org/lindenii/furgit/object/storer"
+
+// New creates an ordered object database chain.
+//
+// The provided backends must be non-nil and distinct.
+// Chain borrows the provided backends and does not close them in Close.
+func New(backends ...objectstorer.Store) *Chain {
+ return &Chain{
+ backends: append([]objectstorer.Store(nil), backends...),
+ }
+}
diff --git a/object/storer/chain/reader.go b/object/storer/chain/reader.go
new file mode 100644
index 00000000..3ac8cce7
--- /dev/null
+++ b/object/storer/chain/reader.go
@@ -0,0 +1,47 @@
+package chain
+
+import (
+ "errors"
+ "fmt"
+ "io"
+
+ objectid "codeberg.org/lindenii/furgit/object/id"
+ "codeberg.org/lindenii/furgit/object/storer"
+ objecttype "codeberg.org/lindenii/furgit/object/type"
+)
+
+// ReadReaderFull reads a full serialized object stream from the first backend that has it.
+func (chain *Chain) ReadReaderFull(id objectid.ObjectID) (io.ReadCloser, error) {
+ for i, backend := range chain.backends {
+ reader, err := backend.ReadReaderFull(id)
+ if err == nil {
+ return reader, nil
+ }
+
+ if errors.Is(err, objectstorer.ErrObjectNotFound) {
+ continue
+ }
+
+ return nil, fmt.Errorf("objectstorer: backend %d read reader full: %w", i, err)
+ }
+
+ return nil, objectstorer.ErrObjectNotFound
+}
+
+// ReadReaderContent reads an object's type, declared content length, and content stream from the first backend that has it.
+func (chain *Chain) ReadReaderContent(id objectid.ObjectID) (objecttype.Type, int64, io.ReadCloser, error) {
+ for i, backend := range chain.backends {
+ ty, size, reader, err := backend.ReadReaderContent(id)
+ if err == nil {
+ return ty, size, reader, nil
+ }
+
+ if errors.Is(err, objectstorer.ErrObjectNotFound) {
+ continue
+ }
+
+ return objecttype.TypeInvalid, 0, nil, fmt.Errorf("objectstorer: backend %d read reader content: %w", i, err)
+ }
+
+ return objecttype.TypeInvalid, 0, nil, objectstorer.ErrObjectNotFound
+}
diff --git a/object/storer/chain/refresh.go b/object/storer/chain/refresh.go
new file mode 100644
index 00000000..c47352dc
--- /dev/null
+++ b/object/storer/chain/refresh.go
@@ -0,0 +1,17 @@
+package chain
+
+import "errors"
+
+// Refresh forwards refresh calls to all backends.
+func (chain *Chain) Refresh() error {
+ var errs []error
+
+ for _, backend := range chain.backends {
+ err := backend.Refresh()
+ if err != nil {
+ errs = append(errs, err)
+ }
+ }
+
+ return errors.Join(errs...)
+}
diff --git a/object/storer/chain/size.go b/object/storer/chain/size.go
new file mode 100644
index 00000000..6ad7d12c
--- /dev/null
+++ b/object/storer/chain/size.go
@@ -0,0 +1,27 @@
+package chain
+
+import (
+ "errors"
+ "fmt"
+
+ objectid "codeberg.org/lindenii/furgit/object/id"
+ "codeberg.org/lindenii/furgit/object/storer"
+)
+
+// ReadSize reads object content length from the first backend that has it.
+func (chain *Chain) ReadSize(id objectid.ObjectID) (int64, error) {
+ for i, backend := range chain.backends {
+ size, err := backend.ReadSize(id)
+ if err == nil {
+ return size, nil
+ }
+
+ if errors.Is(err, objectstorer.ErrObjectNotFound) {
+ continue
+ }
+
+ return 0, fmt.Errorf("objectstorer: backend %d read size: %w", i, err)
+ }
+
+ return 0, objectstorer.ErrObjectNotFound
+}