Files
gotgt/pkg/scsi/backingstore/s3store/s3store_test.go
Lei Xue 76ab15b0df feat: add S3-compatible object storage backend
Add a new backend store that enables iSCSI targets backed by
S3-compatible object storage (AWS S3, MinIO, Ceph RGW, etc.).

The implementation uses a chunked storage strategy where the virtual
block device is divided into fixed-size chunks (default 4 MiB), each
stored as an independent S3 object. This enables efficient random
read/write access on top of object storage.

Key features:
- Chunked storage with configurable chunk size
- Sparse device support (unwritten chunks treated as zeros)
- Concurrent multi-chunk I/O via errgroup
- Per-chunk locking for safe read-modify-write
- AWS SDK v2 with default credential chain
- In-process gofakes3 test server (no Docker needed)
- 12 unit tests + 2 integration tests

Also updates CI workflow to run S3 backend tests and updates
README with S3 backend documentation.

Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
2026-03-16 16:22:57 +08:00

360 lines
8.5 KiB
Go

package s3store
import (
"bytes"
"context"
"fmt"
"io"
"sync"
"testing"
"github.com/aws/aws-sdk-go-v2/service/s3"
"github.com/aws/aws-sdk-go-v2/service/s3/types"
"github.com/gostor/gotgt/pkg/api"
"github.com/gostor/gotgt/pkg/config"
"github.com/gostor/gotgt/pkg/scsi"
)
// fakeS3Client is an in-memory S3 client for testing.
type fakeS3Client struct {
mu sync.Mutex
objects map[string][]byte // key -> data
}
func newFakeS3Client() *fakeS3Client {
return &fakeS3Client{objects: make(map[string][]byte)}
}
func (f *fakeS3Client) GetObject(_ context.Context, input *s3.GetObjectInput, _ ...func(*s3.Options)) (*s3.GetObjectOutput, error) {
f.mu.Lock()
defer f.mu.Unlock()
key := *input.Key
data, ok := f.objects[key]
if !ok {
return nil, &types.NoSuchKey{}
}
if input.Range != nil {
var start, end int64
_, err := fmt.Sscanf(*input.Range, "bytes=%d-%d", &start, &end)
if err == nil && start >= 0 && end < int64(len(data)) {
data = data[start : end+1]
}
}
return &s3.GetObjectOutput{
Body: io.NopCloser(bytes.NewReader(data)),
}, nil
}
func (f *fakeS3Client) PutObject(_ context.Context, input *s3.PutObjectInput, _ ...func(*s3.Options)) (*s3.PutObjectOutput, error) {
f.mu.Lock()
defer f.mu.Unlock()
key := *input.Key
data, err := io.ReadAll(input.Body)
if err != nil {
return nil, err
}
f.objects[key] = data
return &s3.PutObjectOutput{}, nil
}
func (f *fakeS3Client) DeleteObject(_ context.Context, input *s3.DeleteObjectInput, _ ...func(*s3.Options)) (*s3.DeleteObjectOutput, error) {
f.mu.Lock()
defer f.mu.Unlock()
delete(f.objects, *input.Key)
return &s3.DeleteObjectOutput{}, nil
}
func newTestStore(client *fakeS3Client, chunkSize int64, deviceSize uint64) *S3BackingStore {
return &S3BackingStore{
BaseBackingStore: scsi.BaseBackingStore{
Name: S3BackingStorage,
DataSize: deviceSize,
},
client: client,
bucket: "test-bucket",
prefix: "test/disk",
chunkSize: chunkSize,
}
}
func TestOpen_NewDevice(t *testing.T) {
client := newFakeS3Client()
bs := &S3BackingStore{
BaseBackingStore: scsi.BaseBackingStore{Name: S3BackingStorage},
client: client,
}
dev := &api.SCSILu{
BackendConfig: &config.BackendStorage{
DeviceSize: 1024 * 1024, // 1MiB
},
}
err := bs.Open(dev, "mybucket/myprefix")
if err != nil {
t.Fatalf("Open failed: %v", err)
}
if bs.bucket != "mybucket" {
t.Errorf("expected bucket=mybucket, got %s", bs.bucket)
}
if bs.prefix != "myprefix" {
t.Errorf("expected prefix=myprefix, got %s", bs.prefix)
}
if bs.DataSize != 1024*1024 {
t.Errorf("expected DataSize=1048576, got %d", bs.DataSize)
}
// Verify metadata was saved
if _, ok := client.objects["myprefix/_metadata"]; !ok {
t.Error("metadata object was not created")
}
}
func TestOpen_ExistingDevice(t *testing.T) {
client := newFakeS3Client()
// Pre-populate metadata
client.objects["myprefix/_metadata"] = []byte(`{"deviceSize":2097152,"chunkSize":1048576}`)
bs := &S3BackingStore{
BaseBackingStore: scsi.BaseBackingStore{Name: S3BackingStorage},
client: client,
}
dev := &api.SCSILu{
BackendConfig: &config.BackendStorage{},
}
err := bs.Open(dev, "mybucket/myprefix")
if err != nil {
t.Fatalf("Open failed: %v", err)
}
if bs.DataSize != 2097152 {
t.Errorf("expected DataSize=2097152, got %d", bs.DataSize)
}
if bs.chunkSize != 1048576 {
t.Errorf("expected chunkSize=1048576, got %d", bs.chunkSize)
}
}
func TestOpen_InvalidPath(t *testing.T) {
bs := &S3BackingStore{
BaseBackingStore: scsi.BaseBackingStore{Name: S3BackingStorage},
client: newFakeS3Client(),
}
dev := &api.SCSILu{BackendConfig: &config.BackendStorage{}}
err := bs.Open(dev, "nobucket")
if err == nil {
t.Error("expected error for invalid path")
}
}
func TestRead_SingleChunk(t *testing.T) {
client := newFakeS3Client()
bs := newTestStore(client, 1024, 4096)
// Write a chunk
chunkData := make([]byte, 1024)
for i := range chunkData {
chunkData[i] = 0xAB
}
client.objects["test/disk/chunk_0000000000"] = chunkData
data, err := bs.Read(100, 200)
if err != nil {
t.Fatalf("Read failed: %v", err)
}
if len(data) != 200 {
t.Fatalf("expected 200 bytes, got %d", len(data))
}
for i, b := range data {
if b != 0xAB {
t.Fatalf("byte %d: expected 0xAB, got 0x%02X", i, b)
}
}
}
func TestRead_CrossChunk(t *testing.T) {
client := newFakeS3Client()
bs := newTestStore(client, 100, 1000)
// Chunk 0: bytes 0-99, fill with 0xAA
chunk0 := make([]byte, 100)
for i := range chunk0 {
chunk0[i] = 0xAA
}
client.objects["test/disk/chunk_0000000000"] = chunk0
// Chunk 1: bytes 100-199, fill with 0xBB
chunk1 := make([]byte, 100)
for i := range chunk1 {
chunk1[i] = 0xBB
}
client.objects["test/disk/chunk_0000000001"] = chunk1
// Read across boundary: 50-149
data, err := bs.Read(50, 100)
if err != nil {
t.Fatalf("Read failed: %v", err)
}
if len(data) != 100 {
t.Fatalf("expected 100 bytes, got %d", len(data))
}
// First 50 bytes from chunk0 (0xAA)
for i := 0; i < 50; i++ {
if data[i] != 0xAA {
t.Fatalf("byte %d: expected 0xAA, got 0x%02X", i, data[i])
}
}
// Last 50 bytes from chunk1 (0xBB)
for i := 50; i < 100; i++ {
if data[i] != 0xBB {
t.Fatalf("byte %d: expected 0xBB, got 0x%02X", i, data[i])
}
}
}
func TestRead_SparseChunk(t *testing.T) {
client := newFakeS3Client()
bs := newTestStore(client, 100, 1000)
// Don't create any chunks - should return zeros
data, err := bs.Read(0, 100)
if err != nil {
t.Fatalf("Read failed: %v", err)
}
for i, b := range data {
if b != 0 {
t.Fatalf("byte %d: expected 0, got 0x%02X", i, b)
}
}
}
func TestWrite_SingleChunk(t *testing.T) {
client := newFakeS3Client()
bs := newTestStore(client, 1024, 4096)
wbuf := []byte{1, 2, 3, 4, 5}
err := bs.Write(wbuf, 10)
if err != nil {
t.Fatalf("Write failed: %v", err)
}
// The chunk should exist now (read-modify-write since it's partial)
chunk, ok := client.objects["test/disk/chunk_0000000000"]
if !ok {
t.Fatal("chunk was not created")
}
if len(chunk) != 1024 {
t.Fatalf("chunk size: expected 1024, got %d", len(chunk))
}
// Verify written data
for i, b := range wbuf {
if chunk[10+i] != b {
t.Fatalf("byte %d: expected %d, got %d", 10+i, b, chunk[10+i])
}
}
// Verify zeros around it
if chunk[9] != 0 || chunk[15] != 0 {
t.Fatal("surrounding bytes should be zero")
}
}
func TestWrite_FullChunk(t *testing.T) {
client := newFakeS3Client()
bs := newTestStore(client, 8, 64)
// Write exactly one full chunk
wbuf := []byte{1, 2, 3, 4, 5, 6, 7, 8}
err := bs.Write(wbuf, 0)
if err != nil {
t.Fatalf("Write failed: %v", err)
}
chunk := client.objects["test/disk/chunk_0000000000"]
if !bytes.Equal(chunk, wbuf) {
t.Fatalf("chunk data mismatch: %v vs %v", chunk, wbuf)
}
}
func TestWrite_CrossChunk(t *testing.T) {
client := newFakeS3Client()
bs := newTestStore(client, 8, 64)
// Write across chunk boundary (offset 6, len 4 -> chunks 0 and 1)
wbuf := []byte{0xA1, 0xA2, 0xA3, 0xA4}
err := bs.Write(wbuf, 6)
if err != nil {
t.Fatalf("Write failed: %v", err)
}
// Verify chunk 0: bytes 6-7 should be 0xA1, 0xA2
chunk0 := client.objects["test/disk/chunk_0000000000"]
if chunk0[6] != 0xA1 || chunk0[7] != 0xA2 {
t.Fatalf("chunk0 data mismatch at boundary: %v", chunk0[6:8])
}
// Verify chunk 1: bytes 0-1 should be 0xA3, 0xA4
chunk1 := client.objects["test/disk/chunk_0000000001"]
if chunk1[0] != 0xA3 || chunk1[1] != 0xA4 {
t.Fatalf("chunk1 data mismatch at boundary: %v", chunk1[0:2])
}
}
func TestWriteThenRead(t *testing.T) {
client := newFakeS3Client()
bs := newTestStore(client, 100, 1000)
// Write pattern across chunks
wbuf := make([]byte, 250)
for i := range wbuf {
wbuf[i] = byte(i % 256)
}
err := bs.Write(wbuf, 50)
if err != nil {
t.Fatalf("Write failed: %v", err)
}
// Read back
data, err := bs.Read(50, 250)
if err != nil {
t.Fatalf("Read failed: %v", err)
}
if !bytes.Equal(data, wbuf) {
t.Fatal("read data does not match written data")
}
}
func TestUnmap_FullChunk(t *testing.T) {
client := newFakeS3Client()
bs := newTestStore(client, 100, 1000)
// Create chunk
client.objects["test/disk/chunk_0000000001"] = make([]byte, 100)
err := bs.Unmap([]api.UnmapBlockDescriptor{
{Offset: 100, TL: 100}, // exactly chunk 1
})
if err != nil {
t.Fatalf("Unmap failed: %v", err)
}
if _, ok := client.objects["test/disk/chunk_0000000001"]; ok {
t.Error("chunk should have been deleted")
}
}
func TestSize(t *testing.T) {
bs := newTestStore(newFakeS3Client(), 1024, 8192)
dev := &api.SCSILu{}
if bs.Size(dev) != 8192 {
t.Errorf("expected 8192, got %d", bs.Size(dev))
}
}