Some checks failed
Docker image builds / build (push) Waiting to run
Asset Build Verification / asset_verification (push) Has been cancelled
Docs deploy / build (push) Has been cancelled
Go Mod Tidy Check / go_mod_tidy_check (push) Has been cancelled
Go / go_tests (push) Has been cancelled
Package builds (unstable) / package_builds (push) Has been cancelled
Smoke tests / smoke-test (default-config-macro) (push) Has been cancelled
Smoke tests / smoke-test (docker-registry) (push) Has been cancelled
Smoke tests / smoke-test (double_slash) (push) Has been cancelled
Smoke tests / smoke-test (forced-language) (push) Has been cancelled
Smoke tests / smoke-test (git-clone) (push) Has been cancelled
Smoke tests / smoke-test (git-push) (push) Has been cancelled
Smoke tests / smoke-test (healthcheck) (push) Has been cancelled
Smoke tests / smoke-test (i18n) (push) Has been cancelled
Smoke tests / smoke-test (log-file) (push) Has been cancelled
Smoke tests / smoke-test (nginx) (push) Has been cancelled
Smoke tests / smoke-test (palemoon/amd64) (push) Has been cancelled
Smoke tests / smoke-test (robots_txt) (push) Has been cancelled
Check Spelling / Check Spelling (push) Has been cancelled
SSH CI / ssh (aarch64-16k) (push) Has been cancelled
SSH CI / ssh (aarch64-4k) (push) Has been cancelled
SSH CI / ssh (ppc64le) (push) Has been cancelled
SSH CI / ssh (riscv64) (push) Has been cancelled
zizmor / zizmor latest via PyPI (push) Has been cancelled
78 lines
2.1 KiB
Go
78 lines
2.1 KiB
Go
package s3api
|
|
|
|
import (
|
|
"bytes"
|
|
"context"
|
|
"fmt"
|
|
"io"
|
|
"strconv"
|
|
"strings"
|
|
"time"
|
|
|
|
"git.sad.ovh/sophie/nuke/lib/store"
|
|
"github.com/aws/aws-sdk-go-v2/service/s3"
|
|
)
|
|
|
|
type Store struct {
|
|
s3 S3API
|
|
bucket string
|
|
}
|
|
|
|
func (s *Store) Delete(ctx context.Context, key string) error {
|
|
normKey := strings.ReplaceAll(key, ":", "/")
|
|
// Emulate not found by probing first.
|
|
if _, err := s.s3.HeadObject(ctx, &s3.HeadObjectInput{Bucket: &s.bucket, Key: &normKey}); err != nil {
|
|
return fmt.Errorf("%w: %w", store.ErrNotFound, err)
|
|
}
|
|
if _, err := s.s3.DeleteObject(ctx, &s3.DeleteObjectInput{Bucket: &s.bucket, Key: &normKey}); err != nil {
|
|
return fmt.Errorf("can't delete from s3: %w", err)
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (s *Store) Get(ctx context.Context, key string) ([]byte, error) {
|
|
normKey := strings.ReplaceAll(key, ":", "/")
|
|
out, err := s.s3.GetObject(ctx, &s3.GetObjectInput{
|
|
Bucket: &s.bucket,
|
|
Key: &normKey,
|
|
})
|
|
if err != nil {
|
|
return nil, fmt.Errorf("%w: %w", store.ErrNotFound, err)
|
|
}
|
|
defer out.Body.Close()
|
|
if msStr, ok := out.Metadata["x-nuke-expiry-ms"]; ok && msStr != "" {
|
|
if ms, err := strconv.ParseInt(msStr, 10, 64); err == nil {
|
|
if time.Now().UnixMilli() >= ms {
|
|
_, _ = s.s3.DeleteObject(ctx, &s3.DeleteObjectInput{Bucket: &s.bucket, Key: &normKey})
|
|
return nil, store.ErrNotFound
|
|
}
|
|
}
|
|
}
|
|
b, err := io.ReadAll(out.Body)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("can't read s3 object: %w", err)
|
|
}
|
|
return b, nil
|
|
}
|
|
|
|
func (s *Store) Set(ctx context.Context, key string, value []byte, expiry time.Duration) error {
|
|
normKey := strings.ReplaceAll(key, ":", "/")
|
|
// S3 has no native TTL; we store object with metadata X-Nuke-Expiry as epoch seconds.
|
|
var meta map[string]string
|
|
if expiry > 0 {
|
|
exp := time.Now().Add(expiry).UnixMilli()
|
|
meta = map[string]string{"x-nuke-expiry-ms": fmt.Sprintf("%d", exp)}
|
|
}
|
|
_, err := s.s3.PutObject(ctx, &s3.PutObjectInput{
|
|
Bucket: &s.bucket,
|
|
Key: &normKey,
|
|
Body: bytes.NewReader(value),
|
|
Metadata: meta,
|
|
})
|
|
if err != nil {
|
|
return fmt.Errorf("can't put s3 object: %w", err)
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (Store) IsPersistent() bool { return true }
|