golangLAKEHOUSE/internal/storaged/registry_test.go
root 8cfcdb8e5f G0 D2: storaged S3 GET/PUT/LIST/DELETE · 3-lineage scrum · 4 fixes applied
Phase G0 Day 2 ships storaged: aws-sdk-go-v2 wrapper + chi routes
binding 127.0.0.1:3211 with 256 MiB MaxBytesReader, Content-Length
up-front 413, and a 4-slot non-blocking semaphore returning 503 +
Retry-After:5 when full. Acceptance smoke (6/6 probes) PASSES against
the dedicated MinIO bucket lakehouse-go-primary, isolated from the
Rust system's lakehouse bucket during coexistence.

Cross-lineage scrum on the shipped code:
  - Opus 4.7 (opencode): 1 BLOCK + 3 WARN + 3 INFO
  - Qwen3-coder (openrouter): 2 BLOCK + 1 WARN + 1 INFO (3 false positives)
  - Kimi K2-0905 (openrouter, after route-shopping past opencode's 4k
    cap and the direct adapter's empty-content reasoning bug):
    1 BLOCK + 2 WARN + 1 INFO

Fixed:
  C1 buildRegistry ctx cancel footgun → context.Background()
     (Opus + Kimi convergent; future credential refresh chains)
  C2 MaxBytesReader unwrap through manager.Uploader multipart
     goroutines → Content-Length up-front 413 + string-suffix fallback
     (Opus + Kimi convergent; latent 500-instead-of-413 in 5-256 MiB range)
  C3 Bucket.List unbounded accumulation → MaxListResults=10_000 cap
     (Opus + Kimi convergent; OOM guard)
  S1 PUT response Content-Type: application/json (Opus single-reviewer)

Strict validateKey policy (J approved): rejects empty, >1024B, NUL,
leading "/", ".." path components, CR/LF/tab control characters.
DELETE exposed at HTTP layer (J approved option A) for symmetry +
smoke ergonomics.

Build clean, vet clean, all unit tests pass, smoke 6/6 PASS after
every fix round. go.mod 1.23 → 1.24 (required by aws-sdk-go-v2).

Process finding worth recording: opencode caps non-streaming Kimi at
max_tokens=4096; the direct kimi.com adapter consumed 8192 tokens of
reasoning but surfaced empty content; openrouter/moonshotai/kimi-k2-0905
delivered structured output in ~33s. Future Kimi scrums should default
to that route.

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
2026-04-28 23:23:03 -05:00

50 lines
1.1 KiB
Go

package storaged
import (
"errors"
"testing"
)
func TestRegistry_RegisterAndResolve(t *testing.T) {
r := NewRegistry()
b := &Bucket{name: "primary"}
if err := r.Register(b); err != nil {
t.Fatalf("Register: %v", err)
}
got, err := r.Resolve("primary")
if err != nil {
t.Fatalf("Resolve: %v", err)
}
if got != b {
t.Errorf("Resolve returned different bucket")
}
}
func TestRegistry_DuplicateRegisterRejected(t *testing.T) {
r := NewRegistry()
if err := r.Register(&Bucket{name: "primary"}); err != nil {
t.Fatal(err)
}
if err := r.Register(&Bucket{name: "primary"}); err == nil {
t.Fatal("expected error on duplicate register")
}
}
func TestRegistry_UnknownReturnsErrBucketUnknown(t *testing.T) {
r := NewRegistry()
_, err := r.Resolve("nope")
if !errors.Is(err, ErrBucketUnknown) {
t.Fatalf("expected ErrBucketUnknown, got %v", err)
}
}
func TestRegistry_Names(t *testing.T) {
r := NewRegistry()
_ = r.Register(&Bucket{name: "primary"})
_ = r.Register(&Bucket{name: "archive"})
names := r.Names()
if len(names) != 2 {
t.Fatalf("expected 2 names, got %d", len(names))
}
}