From eac43ebb7c57c73239123289eda7e02a1400fdda Mon Sep 17 00:00:00 2001 From: Cody Littley Date: Tue, 5 Nov 2024 13:21:51 -0600 Subject: [PATCH 1/2] Use keys that do not collide when data is stored in a single bucket. Signed-off-by: Cody Littley --- common/aws/cli.go | 5 -- common/aws/s3/client.go | 6 +- common/aws/s3/fragment.go | 27 +++--- common/aws/s3/fragment_test.go | 90 +++++-------------- common/aws/s3/scoped_keys.go | 55 ++++++++++++ disperser/apiserver/disperse_blob_v2.go | 2 +- disperser/apiserver/server_v2_test.go | 2 +- .../common/v2/blobstore/s3_blob_store.go | 12 +-- .../common/v2/blobstore/s3_blob_store_test.go | 10 ++- relay/chunkstore/chunk_reader.go | 8 +- relay/chunkstore/chunk_writer.go | 8 +- 11 files changed, 112 insertions(+), 113 deletions(-) create mode 100644 common/aws/s3/scoped_keys.go diff --git a/common/aws/cli.go b/common/aws/cli.go index e88618d454..79ba02b1fe 100644 --- a/common/aws/cli.go +++ b/common/aws/cli.go @@ -28,9 +28,6 @@ type ClientConfig struct { // EndpointURL of the S3 endpoint to use. If this is not set then the default AWS S3 endpoint will be used. EndpointURL string - // FragmentPrefixChars is the number of characters of the key to use as the prefix for fragmented files. - // A value of "3" for the key "ABCDEFG" will result in the prefix "ABC". Default is 3. - FragmentPrefixChars int // FragmentParallelismFactor helps determine the size of the pool of workers to help upload/download files. // A non-zero value for this parameter adds a number of workers equal to the number of cores times this value. // Default is 8. In general, the number of workers here can be a lot larger than the number of cores because the @@ -120,7 +117,6 @@ func ReadClientConfig(ctx *cli.Context, flagPrefix string) ClientConfig { AccessKey: ctx.GlobalString(common.PrefixFlag(flagPrefix, AccessKeyIdFlagName)), SecretAccessKey: ctx.GlobalString(common.PrefixFlag(flagPrefix, SecretAccessKeyFlagName)), EndpointURL: ctx.GlobalString(common.PrefixFlag(flagPrefix, EndpointURLFlagName)), - FragmentPrefixChars: ctx.GlobalInt(common.PrefixFlag(flagPrefix, FragmentPrefixCharsFlagName)), FragmentParallelismFactor: ctx.GlobalInt(common.PrefixFlag(flagPrefix, FragmentParallelismFactorFlagName)), FragmentParallelismConstant: ctx.GlobalInt(common.PrefixFlag(flagPrefix, FragmentParallelismConstantFlagName)), FragmentReadTimeout: ctx.GlobalDuration(common.PrefixFlag(flagPrefix, FragmentReadTimeoutFlagName)), @@ -132,7 +128,6 @@ func ReadClientConfig(ctx *cli.Context, flagPrefix string) ClientConfig { func DefaultClientConfig() *ClientConfig { return &ClientConfig{ Region: "us-east-2", - FragmentPrefixChars: 3, FragmentParallelismFactor: 8, FragmentParallelismConstant: 0, FragmentReadTimeout: 30 * time.Second, diff --git a/common/aws/s3/client.go b/common/aws/s3/client.go index ddc3ce4e14..1cc4692551 100644 --- a/common/aws/s3/client.go +++ b/common/aws/s3/client.go @@ -37,7 +37,7 @@ type client struct { var _ Client = (*client)(nil) -func NewClient(ctx context.Context, cfg commonaws.ClientConfig, logger logging.Logger) (*client, error) { +func NewClient(ctx context.Context, cfg commonaws.ClientConfig, logger logging.Logger) (Client, error) { var err error once.Do(func() { customResolver := aws.EndpointResolverWithOptionsFunc( @@ -196,7 +196,7 @@ func (s *client) FragmentedUploadObject( data []byte, fragmentSize int) error { - fragments, err := breakIntoFragments(key, data, s.cfg.FragmentPrefixChars, fragmentSize) + fragments, err := breakIntoFragments(key, data, fragmentSize) if err != nil { return err } @@ -251,7 +251,7 @@ func (s *client) FragmentedDownloadObject( return nil, errors.New("fragmentSize must be greater than 0") } - fragmentKeys, err := getFragmentKeys(key, s.cfg.FragmentPrefixChars, getFragmentCount(fileSize, fragmentSize)) + fragmentKeys, err := getFragmentKeys(key, getFragmentCount(fileSize, fragmentSize)) if err != nil { return nil, err } diff --git a/common/aws/s3/fragment.go b/common/aws/s3/fragment.go index 21da697d96..f89e8a9092 100644 --- a/common/aws/s3/fragment.go +++ b/common/aws/s3/fragment.go @@ -19,19 +19,12 @@ func getFragmentCount(fileSize int, fragmentSize int) int { // getFragmentKey returns the key for the fragment at the given index. // -// Fragment keys take the form of "prefix/body-index[f]". The prefix is the first prefixLength characters -// of the file key. The body is the file key. The index is the index of the fragment. The character "f" is appended -// to the key of the last fragment in the series. +// Fragment keys take the form of "body-index[f]". The index is the index of the fragment. The character "f" is +// appended to the key of the last fragment in the series. // -// Example: fileKey="abc123", prefixLength=2, fragmentCount=3 -// The keys will be "ab/abc123-0", "ab/abc123-1", "ab/abc123-2f" -func getFragmentKey(fileKey string, prefixLength int, fragmentCount int, index int) (string, error) { - var prefix string - if prefixLength > len(fileKey) { - prefix = fileKey - } else { - prefix = fileKey[:prefixLength] - } +// Example: fileKey="abc123", fragmentCount=3 +// The keys will be "abc123-0", "abc123-1", "abc123-2f" +func getFragmentKey(fileKey string, fragmentCount int, index int) (string, error) { postfix := "" if fragmentCount-1 == index { @@ -42,7 +35,7 @@ func getFragmentKey(fileKey string, prefixLength int, fragmentCount int, index i return "", fmt.Errorf("index %d is too high for fragment count %d", index, fragmentCount) } - return fmt.Sprintf("%s/%s-%d%s", prefix, fileKey, index, postfix), nil + return fmt.Sprintf("%s-%d%s", fileKey, index, postfix), nil } // Fragment is a subset of a file. @@ -53,7 +46,7 @@ type Fragment struct { } // breakIntoFragments breaks a file into fragments of the given size. -func breakIntoFragments(fileKey string, data []byte, prefixLength int, fragmentSize int) ([]*Fragment, error) { +func breakIntoFragments(fileKey string, data []byte, fragmentSize int) ([]*Fragment, error) { fragmentCount := getFragmentCount(len(data), fragmentSize) fragments := make([]*Fragment, fragmentCount) for i := 0; i < fragmentCount; i++ { @@ -63,7 +56,7 @@ func breakIntoFragments(fileKey string, data []byte, prefixLength int, fragmentS end = len(data) } - fragmentKey, err := getFragmentKey(fileKey, prefixLength, fragmentCount, i) + fragmentKey, err := getFragmentKey(fileKey, fragmentCount, i) if err != nil { return nil, err } @@ -77,10 +70,10 @@ func breakIntoFragments(fileKey string, data []byte, prefixLength int, fragmentS } // getFragmentKeys returns the keys for all fragments of a file. -func getFragmentKeys(fileKey string, prefixLength int, fragmentCount int) ([]string, error) { +func getFragmentKeys(fileKey string, fragmentCount int) ([]string, error) { keys := make([]string, fragmentCount) for i := 0; i < fragmentCount; i++ { - fragmentKey, err := getFragmentKey(fileKey, prefixLength, fragmentCount, i) + fragmentKey, err := getFragmentKey(fileKey, fragmentCount, i) if err != nil { return nil, err } diff --git a/common/aws/s3/fragment_test.go b/common/aws/s3/fragment_test.go index 04271ce8e8..598c0290b9 100644 --- a/common/aws/s3/fragment_test.go +++ b/common/aws/s3/fragment_test.go @@ -52,31 +52,6 @@ func TestGetFragmentCount(t *testing.T) { assert.Equal(t, expectedFragmentCount, fragmentCount) } -// Fragment keys take the form of "prefix/body-index[f]". Verify the prefix part of the key. -func TestPrefix(t *testing.T) { - tu.InitializeRandom() - - keyLength := rand.Intn(10) + 10 - key := tu.RandomString(keyLength) - - for i := 0; i < keyLength*2; i++ { - fragmentCount := rand.Intn(10) + 10 - fragmentIndex := rand.Intn(fragmentCount) - fragmentKey, err := getFragmentKey(key, i, fragmentCount, fragmentIndex) - assert.NoError(t, err) - - parts := strings.Split(fragmentKey, "/") - assert.Equal(t, 2, len(parts)) - prefix := parts[0] - - if i >= keyLength { - assert.Equal(t, key, prefix) - } else { - assert.Equal(t, key[:i], prefix) - } - } -} - // Fragment keys take the form of "prefix/body-index[f]". Verify the body part of the key. func TestKeyBody(t *testing.T) { tu.InitializeRandom() @@ -86,12 +61,10 @@ func TestKeyBody(t *testing.T) { key := tu.RandomString(keyLength) fragmentCount := rand.Intn(10) + 10 fragmentIndex := rand.Intn(fragmentCount) - fragmentKey, err := getFragmentKey(key, rand.Intn(10), fragmentCount, fragmentIndex) + fragmentKey, err := getFragmentKey(key, fragmentCount, fragmentIndex) assert.NoError(t, err) - parts := strings.Split(fragmentKey, "/") - assert.Equal(t, 2, len(parts)) - parts = strings.Split(parts[1], "-") + parts := strings.Split(fragmentKey, "-") assert.Equal(t, 2, len(parts)) body := parts[0] @@ -106,12 +79,10 @@ func TestKeyIndex(t *testing.T) { for i := 0; i < 10; i++ { fragmentCount := rand.Intn(10) + 10 index := rand.Intn(fragmentCount) - fragmentKey, err := getFragmentKey(tu.RandomString(10), rand.Intn(10), fragmentCount, index) + fragmentKey, err := getFragmentKey(tu.RandomString(10), fragmentCount, index) assert.NoError(t, err) - parts := strings.Split(fragmentKey, "/") - assert.Equal(t, 2, len(parts)) - parts = strings.Split(parts[1], "-") + parts := strings.Split(fragmentKey, "-") assert.Equal(t, 2, len(parts)) indexStr := parts[1] assert.True(t, strings.HasPrefix(indexStr, fmt.Sprintf("%d", index))) @@ -126,7 +97,7 @@ func TestKeyPostfix(t *testing.T) { segmentCount := rand.Intn(10) + 10 for i := 0; i < segmentCount; i++ { - fragmentKey, err := getFragmentKey(tu.RandomString(10), rand.Intn(10), segmentCount, i) + fragmentKey, err := getFragmentKey(tu.RandomString(10), segmentCount, i) assert.NoError(t, err) if i == segmentCount-1 { @@ -139,41 +110,35 @@ func TestKeyPostfix(t *testing.T) { // TestExampleInGodoc tests the example provided in the documentation for getFragmentKey(). // -// Example: fileKey="abc123", prefixLength=2, fragmentCount=3 -// The keys will be "ab/abc123-0", "ab/abc123-1", "ab/abc123-2f" +// Example: fileKey="abc123", fragmentCount=3 +// The keys will be "abc123-0", "abc123-1", "abc123-2f" func TestExampleInGodoc(t *testing.T) { fileKey := "abc123" - prefixLength := 2 fragmentCount := 3 - fragmentKeys, err := getFragmentKeys(fileKey, prefixLength, fragmentCount) + fragmentKeys, err := getFragmentKeys(fileKey, fragmentCount) assert.NoError(t, err) assert.Equal(t, 3, len(fragmentKeys)) - assert.Equal(t, "ab/abc123-0", fragmentKeys[0]) - assert.Equal(t, "ab/abc123-1", fragmentKeys[1]) - assert.Equal(t, "ab/abc123-2f", fragmentKeys[2]) + assert.Equal(t, "abc123-0", fragmentKeys[0]) + assert.Equal(t, "abc123-1", fragmentKeys[1]) + assert.Equal(t, "abc123-2f", fragmentKeys[2]) } func TestGetFragmentKeys(t *testing.T) { tu.InitializeRandom() fileKey := tu.RandomString(10) - prefixLength := rand.Intn(3) + 1 fragmentCount := rand.Intn(10) + 10 - fragmentKeys, err := getFragmentKeys(fileKey, prefixLength, fragmentCount) + fragmentKeys, err := getFragmentKeys(fileKey, fragmentCount) assert.NoError(t, err) assert.Equal(t, fragmentCount, len(fragmentKeys)) for i := 0; i < fragmentCount; i++ { - expectedKey, err := getFragmentKey(fileKey, prefixLength, fragmentCount, i) + expectedKey, err := getFragmentKey(fileKey, fragmentCount, i) assert.NoError(t, err) assert.Equal(t, expectedKey, fragmentKeys[i]) - parts := strings.Split(fragmentKeys[i], "/") - assert.Equal(t, 2, len(parts)) - parsedPrefix := parts[0] - assert.Equal(t, fileKey[:prefixLength], parsedPrefix) - parts = strings.Split(parts[1], "-") + parts := strings.Split(fragmentKeys[i], "-") assert.Equal(t, 2, len(parts)) parsedKey := parts[0] assert.Equal(t, fileKey, parsedKey) @@ -192,17 +157,16 @@ func TestGetFragments(t *testing.T) { fileKey := tu.RandomString(10) data := tu.RandomBytes(1000) - prefixLength := rand.Intn(3) + 1 fragmentSize := rand.Intn(100) + 100 - fragments, err := breakIntoFragments(fileKey, data, prefixLength, fragmentSize) + fragments, err := breakIntoFragments(fileKey, data, fragmentSize) assert.NoError(t, err) assert.Equal(t, getFragmentCount(len(data), fragmentSize), len(fragments)) totalSize := 0 for i, fragment := range fragments { - fragmentKey, err := getFragmentKey(fileKey, prefixLength, len(fragments), i) + fragmentKey, err := getFragmentKey(fileKey, len(fragments), i) assert.NoError(t, err) assert.Equal(t, fragmentKey, fragment.FragmentKey) @@ -224,14 +188,13 @@ func TestGetFragmentsSmallFile(t *testing.T) { fileKey := tu.RandomString(10) data := tu.RandomBytes(10) - prefixLength := rand.Intn(3) + 1 fragmentSize := rand.Intn(100) + 100 - fragments, err := breakIntoFragments(fileKey, data, prefixLength, fragmentSize) + fragments, err := breakIntoFragments(fileKey, data, fragmentSize) assert.NoError(t, err) assert.Equal(t, 1, len(fragments)) - fragmentKey, err := getFragmentKey(fileKey, prefixLength, 1, 0) + fragmentKey, err := getFragmentKey(fileKey, 1, 0) assert.NoError(t, err) assert.Equal(t, fragmentKey, fragments[0].FragmentKey) assert.Equal(t, data, fragments[0].Data) @@ -244,13 +207,12 @@ func TestGetFragmentsExactlyOnePerfectlySizedFile(t *testing.T) { fileKey := tu.RandomString(10) fragmentSize := rand.Intn(100) + 100 data := tu.RandomBytes(fragmentSize) - prefixLength := rand.Intn(3) + 1 - fragments, err := breakIntoFragments(fileKey, data, prefixLength, fragmentSize) + fragments, err := breakIntoFragments(fileKey, data, fragmentSize) assert.NoError(t, err) assert.Equal(t, 1, len(fragments)) - fragmentKey, err := getFragmentKey(fileKey, prefixLength, 1, 0) + fragmentKey, err := getFragmentKey(fileKey, 1, 0) assert.NoError(t, err) assert.Equal(t, fragmentKey, fragments[0].FragmentKey) assert.Equal(t, data, fragments[0].Data) @@ -262,10 +224,9 @@ func TestRecombineFragments(t *testing.T) { fileKey := tu.RandomString(10) data := tu.RandomBytes(1000) - prefixLength := rand.Intn(3) + 1 fragmentSize := rand.Intn(100) + 100 - fragments, err := breakIntoFragments(fileKey, data, prefixLength, fragmentSize) + fragments, err := breakIntoFragments(fileKey, data, fragmentSize) assert.NoError(t, err) recombinedData, err := recombineFragments(fragments) assert.NoError(t, err) @@ -287,10 +248,9 @@ func TestRecombineFragmentsSmallFile(t *testing.T) { fileKey := tu.RandomString(10) data := tu.RandomBytes(10) - prefixLength := rand.Intn(3) + 1 fragmentSize := rand.Intn(100) + 100 - fragments, err := breakIntoFragments(fileKey, data, prefixLength, fragmentSize) + fragments, err := breakIntoFragments(fileKey, data, fragmentSize) assert.NoError(t, err) assert.Equal(t, 1, len(fragments)) recombinedData, err := recombineFragments(fragments) @@ -303,10 +263,9 @@ func TestMissingFragment(t *testing.T) { fileKey := tu.RandomString(10) data := tu.RandomBytes(1000) - prefixLength := rand.Intn(3) + 1 fragmentSize := rand.Intn(100) + 100 - fragments, err := breakIntoFragments(fileKey, data, prefixLength, fragmentSize) + fragments, err := breakIntoFragments(fileKey, data, fragmentSize) assert.NoError(t, err) fragmentIndexToSkip := rand.Intn(len(fragments)) @@ -321,10 +280,9 @@ func TestMissingFinalFragment(t *testing.T) { fileKey := tu.RandomString(10) data := tu.RandomBytes(1000) - prefixLength := rand.Intn(3) + 1 fragmentSize := rand.Intn(100) + 100 - fragments, err := breakIntoFragments(fileKey, data, prefixLength, fragmentSize) + fragments, err := breakIntoFragments(fileKey, data, fragmentSize) assert.NoError(t, err) fragments = fragments[:len(fragments)-1] diff --git a/common/aws/s3/scoped_keys.go b/common/aws/s3/scoped_keys.go new file mode 100644 index 0000000000..3ef9a7bdc2 --- /dev/null +++ b/common/aws/s3/scoped_keys.go @@ -0,0 +1,55 @@ +package s3 + +import ( + "fmt" + v2 "github.com/Layr-Labs/eigenda/core/v2" +) + +const ( + // prefixLength is the number of characters to use from the base key to form the prefix. + // Assuming keys take the form of a random hash in hex, 3 will yield 16^3 = 4096 possible prefixes. + // This is currently hard coded because it is not expected to change, and it would require migration + // to change it that we have not yet implemented. + prefixLength = 3 + + // blobNamespace is the postfix for a blob key. + blobNamespace = "blob" + + // chunkNamespace is the postfix for a chunk key. + chunkNamespace = "chunk" + + // proofNamespace is the postfix for a proof key. + proofNamespace = "proof" +) + +// ScopedKey returns a key that is scoped to a "namespace". Keys take the form of "prefix/namespace/baseKey". +// Although there is no runtime enforcement, neither the base key nor the namespace should contain any +// non-alphanumeric characters. +func ScopedKey(namespace string, baseKey string, prefixLength int) string { + var prefix string + if prefixLength > len(baseKey) { + prefix = baseKey + } else { + prefix = baseKey[:prefixLength] + } + + return fmt.Sprintf("%s/%s/%s", prefix, namespace, baseKey) +} + +// ScopedBlobKey returns a key scoped to the blob namespace. Used to name files containing blobs in S3. +// A key scoped for blobs will never collide with a key scoped for chunks or proofs. +func ScopedBlobKey(blobKey v2.BlobKey) string { + return ScopedKey(blobNamespace, blobKey.Hex(), prefixLength) +} + +// ScopedChunkKey returns a key scoped to the chunk namespace. Used to name files containing chunks in S3. +// A key scoped for chunks will never collide with a key scoped for blobs or proofs. +func ScopedChunkKey(blobKey v2.BlobKey) string { + return ScopedKey(chunkNamespace, blobKey.Hex(), prefixLength) +} + +// ScopedProofKey returns a key scoped to the proof namespace. Used to name files containing proofs in S3. +// A key scoped for proofs will never collide with a key scoped for blobs or chunks. +func ScopedProofKey(blobKey v2.BlobKey) string { + return ScopedKey(proofNamespace, blobKey.Hex(), prefixLength) +} diff --git a/disperser/apiserver/disperse_blob_v2.go b/disperser/apiserver/disperse_blob_v2.go index 831b4b26c3..5b04b81c06 100644 --- a/disperser/apiserver/disperse_blob_v2.go +++ b/disperser/apiserver/disperse_blob_v2.go @@ -50,7 +50,7 @@ func (s *DispersalServerV2) StoreBlob(ctx context.Context, data []byte, blobHead return v2.BlobKey{}, err } - if err := s.blobStore.StoreBlob(ctx, blobKey.Hex(), data); err != nil { + if err := s.blobStore.StoreBlob(ctx, blobKey, data); err != nil { return v2.BlobKey{}, err } diff --git a/disperser/apiserver/server_v2_test.go b/disperser/apiserver/server_v2_test.go index 3bd0715844..0a008237a3 100644 --- a/disperser/apiserver/server_v2_test.go +++ b/disperser/apiserver/server_v2_test.go @@ -84,7 +84,7 @@ func TestV2DisperseBlob(t *testing.T) { assert.Equal(t, blobKey[:], reply.BlobKey) // Check if the blob is stored - storedData, err := c.BlobStore.GetBlob(ctx, blobKey.Hex()) + storedData, err := c.BlobStore.GetBlob(ctx, blobKey) assert.NoError(t, err) assert.Equal(t, data, storedData) diff --git a/disperser/common/v2/blobstore/s3_blob_store.go b/disperser/common/v2/blobstore/s3_blob_store.go index c7575c4a99..2d3f595442 100644 --- a/disperser/common/v2/blobstore/s3_blob_store.go +++ b/disperser/common/v2/blobstore/s3_blob_store.go @@ -2,6 +2,7 @@ package blobstore import ( "context" + v2 "github.com/Layr-Labs/eigenda/core/v2" "github.com/Layr-Labs/eigenda/common/aws/s3" "github.com/Layr-Labs/eigenda/disperser/common" @@ -24,8 +25,9 @@ func NewBlobStore(s3BucketName string, s3Client s3.Client, logger logging.Logger } // StoreBlob adds a blob to the blob store -func (b *BlobStore) StoreBlob(ctx context.Context, blobKey string, data []byte) error { - err := b.s3Client.UploadObject(ctx, b.bucketName, blobKey, data) +func (b *BlobStore) StoreBlob(ctx context.Context, key v2.BlobKey, data []byte) error { + + err := b.s3Client.UploadObject(ctx, b.bucketName, s3.ScopedBlobKey(key), data) if err != nil { b.logger.Errorf("failed to upload blob in bucket %s: %v", b.bucketName, err) return err @@ -34,10 +36,10 @@ func (b *BlobStore) StoreBlob(ctx context.Context, blobKey string, data []byte) } // GetBlob retrieves a blob from the blob store -func (b *BlobStore) GetBlob(ctx context.Context, blobKey string) ([]byte, error) { - data, err := b.s3Client.DownloadObject(ctx, b.bucketName, blobKey) +func (b *BlobStore) GetBlob(ctx context.Context, key v2.BlobKey) ([]byte, error) { + data, err := b.s3Client.DownloadObject(ctx, b.bucketName, s3.ScopedBlobKey(key)) if errors.Is(err, s3.ErrObjectNotFound) { - b.logger.Warnf("blob not found in bucket %s: %s", b.bucketName, blobKey) + b.logger.Warnf("blob not found in bucket %s: %s", b.bucketName, key) return nil, common.ErrBlobNotFound } diff --git a/disperser/common/v2/blobstore/s3_blob_store_test.go b/disperser/common/v2/blobstore/s3_blob_store_test.go index 2ca3b9c44d..756c9b5de0 100644 --- a/disperser/common/v2/blobstore/s3_blob_store_test.go +++ b/disperser/common/v2/blobstore/s3_blob_store_test.go @@ -2,21 +2,25 @@ package blobstore_test import ( "context" + tu "github.com/Layr-Labs/eigenda/common/testutils" + v2 "github.com/Layr-Labs/eigenda/core/v2" "testing" "github.com/stretchr/testify/assert" ) func TestStoreGetBlob(t *testing.T) { - err := blobStore.StoreBlob(context.Background(), "testBlobKey", []byte("testBlobData")) + testBlobKey := v2.BlobKey(tu.RandomBytes(32)) + err := blobStore.StoreBlob(context.Background(), testBlobKey, []byte("testBlobData")) assert.NoError(t, err) - data, err := blobStore.GetBlob(context.Background(), "testBlobKey") + data, err := blobStore.GetBlob(context.Background(), testBlobKey) assert.NoError(t, err) assert.Equal(t, []byte("testBlobData"), data) } func TestGetBlobNotFound(t *testing.T) { - data, err := blobStore.GetBlob(context.Background(), "nonExistentBlobKey") + testBlobKey := v2.BlobKey(tu.RandomBytes(32)) + data, err := blobStore.GetBlob(context.Background(), testBlobKey) assert.Error(t, err) assert.Nil(t, data) } diff --git a/relay/chunkstore/chunk_reader.go b/relay/chunkstore/chunk_reader.go index 843a0276d8..30977a6777 100644 --- a/relay/chunkstore/chunk_reader.go +++ b/relay/chunkstore/chunk_reader.go @@ -58,9 +58,7 @@ func (r *chunkReader) GetChunkProofs( ctx context.Context, blobKey v2.BlobKey) ([]*encoding.Proof, error) { - s3Key := blobKey.Hex() - - bytes, err := r.client.DownloadObject(ctx, r.bucket, s3Key) + bytes, err := r.client.DownloadObject(ctx, r.bucket, s3.ScopedProofKey(blobKey)) if err != nil { r.logger.Error("Failed to download chunks from S3: %v", err) return nil, fmt.Errorf("failed to download chunks from S3: %w", err) @@ -92,12 +90,10 @@ func (r *chunkReader) GetChunkCoefficients( blobKey v2.BlobKey, fragmentInfo *encoding.FragmentInfo) ([]*rs.Frame, error) { - s3Key := blobKey.Hex() - bytes, err := r.client.FragmentedDownloadObject( ctx, r.bucket, - s3Key, + s3.ScopedChunkKey(blobKey), int(fragmentInfo.TotalChunkSizeBytes), int(fragmentInfo.FragmentSizeBytes)) diff --git a/relay/chunkstore/chunk_writer.go b/relay/chunkstore/chunk_writer.go index ea54f28b19..6072d9630c 100644 --- a/relay/chunkstore/chunk_writer.go +++ b/relay/chunkstore/chunk_writer.go @@ -47,15 +47,13 @@ func NewChunkWriter( } func (c *chunkWriter) PutChunkProofs(ctx context.Context, blobKey v2.BlobKey, proofs []*encoding.Proof) error { - s3Key := blobKey.Hex() - bytes := make([]byte, 0, bn254.SizeOfG1AffineCompressed*len(proofs)) for _, proof := range proofs { proofBytes := proof.Bytes() bytes = append(bytes, proofBytes[:]...) } - err := c.s3Client.UploadObject(ctx, c.bucketName, s3Key, bytes) + err := c.s3Client.UploadObject(ctx, c.bucketName, s3.ScopedProofKey(blobKey), bytes) if err != nil { c.logger.Error("Failed to upload chunks to S3: %v", err) @@ -70,15 +68,13 @@ func (c *chunkWriter) PutChunkCoefficients( blobKey v2.BlobKey, frames []*rs.Frame) (*encoding.FragmentInfo, error) { - s3Key := blobKey.Hex() - bytes, err := rs.GnarkEncodeFrames(frames) if err != nil { c.logger.Error("Failed to encode frames: %v", err) return nil, fmt.Errorf("failed to encode frames: %w", err) } - err = c.s3Client.FragmentedUploadObject(ctx, c.bucketName, s3Key, bytes, c.fragmentSize) + err = c.s3Client.FragmentedUploadObject(ctx, c.bucketName, s3.ScopedChunkKey(blobKey), bytes, c.fragmentSize) if err != nil { c.logger.Error("Failed to upload chunks to S3: %v", err) return nil, fmt.Errorf("failed to upload chunks to S3: %w", err) From 2610af3c5881b3741adc815442ae4f4b216cd9d2 Mon Sep 17 00:00:00 2001 From: Cody Littley Date: Tue, 5 Nov 2024 13:33:30 -0600 Subject: [PATCH 2/2] Clean up godocs. Signed-off-by: Cody Littley --- common/aws/s3/scoped_keys.go | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/common/aws/s3/scoped_keys.go b/common/aws/s3/scoped_keys.go index 3ef9a7bdc2..3b558c5bc1 100644 --- a/common/aws/s3/scoped_keys.go +++ b/common/aws/s3/scoped_keys.go @@ -12,13 +12,13 @@ const ( // to change it that we have not yet implemented. prefixLength = 3 - // blobNamespace is the postfix for a blob key. + // blobNamespace is the namespace for a blob key. blobNamespace = "blob" - // chunkNamespace is the postfix for a chunk key. + // chunkNamespace is the namespace for a chunk key. chunkNamespace = "chunk" - // proofNamespace is the postfix for a proof key. + // proofNamespace is the namespace for a proof key. proofNamespace = "proof" )