2022-11-22 20:29:57 +02:00
|
|
|
package cache
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
2023-01-09 22:37:44 +02:00
|
|
|
"strings"
|
2022-11-22 20:29:57 +02:00
|
|
|
|
|
|
|
"github.com/aws/aws-sdk-go-v2/aws"
|
|
|
|
"github.com/aws/aws-sdk-go-v2/config"
|
|
|
|
"github.com/aws/aws-sdk-go-v2/feature/dynamodb/attributevalue"
|
|
|
|
"github.com/aws/aws-sdk-go-v2/service/dynamodb"
|
|
|
|
"github.com/aws/aws-sdk-go-v2/service/dynamodb/types"
|
|
|
|
godigest "github.com/opencontainers/go-digest"
|
|
|
|
|
2024-02-01 06:34:07 +02:00
|
|
|
zerr "zotregistry.dev/zot/errors"
|
|
|
|
zlog "zotregistry.dev/zot/pkg/log"
|
2022-11-22 20:29:57 +02:00
|
|
|
)
|
|
|
|
|
|
|
|
type DynamoDBDriver struct {
|
|
|
|
client *dynamodb.Client
|
|
|
|
log zlog.Logger
|
|
|
|
tableName string
|
|
|
|
}
|
|
|
|
|
|
|
|
type DynamoDBDriverParameters struct {
|
|
|
|
Endpoint, Region, TableName string
|
|
|
|
}
|
|
|
|
|
|
|
|
type Blob struct {
|
2023-10-10 20:29:07 +03:00
|
|
|
Digest string `dynamodbav:"Digest,string"`
|
|
|
|
DuplicateBlobPath []string `dynamodbav:"DuplicateBlobPath,stringset"`
|
|
|
|
OriginalBlobPath string `dynamodbav:"OriginalBlobPath,string"`
|
2022-11-22 20:29:57 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
func (d *DynamoDBDriver) NewTable(tableName string) error {
|
|
|
|
//nolint:gomnd
|
|
|
|
_, err := d.client.CreateTable(context.TODO(), &dynamodb.CreateTableInput{
|
|
|
|
TableName: &tableName,
|
|
|
|
AttributeDefinitions: []types.AttributeDefinition{
|
|
|
|
{
|
|
|
|
AttributeName: aws.String("Digest"),
|
|
|
|
AttributeType: types.ScalarAttributeTypeS,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
KeySchema: []types.KeySchemaElement{
|
|
|
|
{
|
|
|
|
AttributeName: aws.String("Digest"),
|
|
|
|
KeyType: types.KeyTypeHash,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
ProvisionedThroughput: &types.ProvisionedThroughput{
|
|
|
|
ReadCapacityUnits: aws.Int64(10),
|
|
|
|
WriteCapacityUnits: aws.Int64(5),
|
|
|
|
},
|
|
|
|
})
|
2023-01-09 22:37:44 +02:00
|
|
|
if err != nil && !strings.Contains(err.Error(), "Table already exists") {
|
2022-11-22 20:29:57 +02:00
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
d.tableName = tableName
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2023-11-24 10:38:36 +02:00
|
|
|
func NewDynamoDBCache(parameters interface{}, log zlog.Logger) (*DynamoDBDriver, error) {
|
2022-11-22 20:29:57 +02:00
|
|
|
properParameters, ok := parameters.(DynamoDBDriverParameters)
|
|
|
|
if !ok {
|
2023-12-08 00:05:02 -08:00
|
|
|
log.Error().Err(zerr.ErrTypeAssertionFailed).Msgf("failed to cast type, expected type '%T' but got '%T'",
|
2023-11-24 10:38:36 +02:00
|
|
|
BoltDBDriverParameters{}, parameters)
|
|
|
|
|
|
|
|
return nil, zerr.ErrTypeAssertionFailed
|
2022-11-22 20:29:57 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// custom endpoint resolver to point to localhost
|
2024-06-12 22:51:32 -07:00
|
|
|
customResolver := aws.EndpointResolverWithOptionsFunc( //nolint: staticcheck
|
2022-11-22 20:29:57 +02:00
|
|
|
func(service, region string, options ...interface{}) (aws.Endpoint, error) {
|
2024-06-12 22:51:32 -07:00
|
|
|
return aws.Endpoint{ //nolint: staticcheck
|
2022-11-22 20:29:57 +02:00
|
|
|
PartitionID: "aws",
|
|
|
|
URL: properParameters.Endpoint,
|
|
|
|
SigningRegion: region,
|
|
|
|
}, nil
|
|
|
|
})
|
|
|
|
|
|
|
|
// Using the SDK's default configuration, loading additional config
|
|
|
|
// and credentials values from the environment variables, shared
|
|
|
|
// credentials, and shared configuration files
|
2023-05-26 21:08:19 +03:00
|
|
|
cfg, err := config.LoadDefaultConfig(context.Background(), config.WithRegion(properParameters.Region),
|
2024-06-12 22:51:32 -07:00
|
|
|
config.WithEndpointResolverWithOptions(customResolver)) //nolint: staticcheck
|
2022-11-22 20:29:57 +02:00
|
|
|
if err != nil {
|
2023-12-08 00:05:02 -08:00
|
|
|
log.Error().Err(err).Msg("failed to load AWS SDK config for dynamodb")
|
2022-11-22 20:29:57 +02:00
|
|
|
|
2023-11-24 10:38:36 +02:00
|
|
|
return nil, err
|
2022-11-22 20:29:57 +02:00
|
|
|
}
|
|
|
|
|
2023-01-09 22:37:44 +02:00
|
|
|
driver := &DynamoDBDriver{client: dynamodb.NewFromConfig(cfg), tableName: properParameters.TableName, log: log}
|
|
|
|
|
|
|
|
err = driver.NewTable(driver.tableName)
|
|
|
|
if err != nil {
|
2023-12-08 00:05:02 -08:00
|
|
|
log.Error().Err(err).Str("tableName", driver.tableName).Msg("failed to create table for cache")
|
2023-11-24 10:38:36 +02:00
|
|
|
|
|
|
|
return nil, err
|
2023-01-09 22:37:44 +02:00
|
|
|
}
|
|
|
|
|
2022-11-22 20:29:57 +02:00
|
|
|
// Using the Config value, create the DynamoDB client
|
2023-11-24 10:38:36 +02:00
|
|
|
return driver, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (d *DynamoDBDriver) SetTableName(table string) {
|
|
|
|
d.tableName = table
|
2022-11-22 20:29:57 +02:00
|
|
|
}
|
|
|
|
|
2023-09-01 20:54:39 +03:00
|
|
|
func (d *DynamoDBDriver) UsesRelativePaths() bool {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
2022-11-22 20:29:57 +02:00
|
|
|
func (d *DynamoDBDriver) Name() string {
|
|
|
|
return "dynamodb"
|
|
|
|
}
|
|
|
|
|
2023-10-10 20:29:07 +03:00
|
|
|
// Returns the original blob.
|
2022-11-22 20:29:57 +02:00
|
|
|
func (d *DynamoDBDriver) GetBlob(digest godigest.Digest) (string, error) {
|
|
|
|
resp, err := d.client.GetItem(context.TODO(), &dynamodb.GetItemInput{
|
|
|
|
TableName: aws.String(d.tableName),
|
|
|
|
Key: map[string]types.AttributeValue{
|
|
|
|
"Digest": &types.AttributeValueMemberS{Value: digest.String()},
|
|
|
|
},
|
|
|
|
})
|
|
|
|
if err != nil {
|
2023-04-28 05:44:22 +03:00
|
|
|
d.log.Error().Err(err).Str("tableName", d.tableName).Msg("failed to get blob")
|
2022-11-22 20:29:57 +02:00
|
|
|
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
|
|
|
|
out := Blob{}
|
|
|
|
|
|
|
|
if resp.Item == nil {
|
|
|
|
return "", zerr.ErrCacheMiss
|
|
|
|
}
|
|
|
|
|
|
|
|
_ = attributevalue.UnmarshalMap(resp.Item, &out)
|
|
|
|
|
2023-10-10 20:29:07 +03:00
|
|
|
return out.OriginalBlobPath, nil
|
2022-11-22 20:29:57 +02:00
|
|
|
}
|
|
|
|
|
2024-07-08 11:35:44 -07:00
|
|
|
func (d *DynamoDBDriver) GetAllBlobs(digest godigest.Digest) ([]string, error) {
|
|
|
|
blobPaths := []string{}
|
|
|
|
|
|
|
|
resp, err := d.client.GetItem(context.TODO(), &dynamodb.GetItemInput{
|
|
|
|
TableName: aws.String(d.tableName),
|
|
|
|
Key: map[string]types.AttributeValue{
|
|
|
|
"Digest": &types.AttributeValueMemberS{Value: digest.String()},
|
|
|
|
},
|
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
d.log.Error().Err(err).Str("tableName", d.tableName).Msg("failed to get blob")
|
|
|
|
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
out := Blob{}
|
|
|
|
|
|
|
|
if resp.Item == nil {
|
|
|
|
d.log.Debug().Err(zerr.ErrCacheMiss).Str("digest", string(digest)).Msg("failed to find blob in cache")
|
|
|
|
|
|
|
|
return nil, zerr.ErrCacheMiss
|
|
|
|
}
|
|
|
|
|
|
|
|
_ = attributevalue.UnmarshalMap(resp.Item, &out)
|
|
|
|
|
|
|
|
blobPaths = append(blobPaths, out.OriginalBlobPath)
|
|
|
|
|
|
|
|
for _, item := range out.DuplicateBlobPath {
|
|
|
|
if item != out.OriginalBlobPath {
|
|
|
|
blobPaths = append(blobPaths, item)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return blobPaths, nil
|
|
|
|
}
|
|
|
|
|
2022-11-22 20:29:57 +02:00
|
|
|
func (d *DynamoDBDriver) PutBlob(digest godigest.Digest, path string) error {
|
|
|
|
if path == "" {
|
2023-12-08 00:05:02 -08:00
|
|
|
d.log.Error().Err(zerr.ErrEmptyValue).Str("digest", digest.String()).
|
|
|
|
Msg("failed to put blob because the path provided is empty")
|
2022-11-22 20:29:57 +02:00
|
|
|
|
|
|
|
return zerr.ErrEmptyValue
|
|
|
|
}
|
|
|
|
|
2023-10-10 20:29:07 +03:00
|
|
|
if originBlob, _ := d.GetBlob(digest); originBlob == "" {
|
|
|
|
// first entry, so add original blob
|
|
|
|
if err := d.putOriginBlob(digest, path); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
expression := "ADD DuplicateBlobPath :i"
|
2022-11-22 20:29:57 +02:00
|
|
|
attrPath := types.AttributeValueMemberSS{Value: []string{path}}
|
|
|
|
|
2023-10-10 20:29:07 +03:00
|
|
|
if err := d.updateItem(digest, expression, map[string]types.AttributeValue{":i": &attrPath}); err != nil {
|
2023-12-08 00:05:02 -08:00
|
|
|
d.log.Error().Err(err).Str("digest", digest.String()).Str("path", path).Msg("failed to put blob")
|
2022-11-22 20:29:57 +02:00
|
|
|
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (d *DynamoDBDriver) HasBlob(digest godigest.Digest, path string) bool {
|
|
|
|
resp, err := d.client.GetItem(context.TODO(), &dynamodb.GetItemInput{
|
|
|
|
TableName: aws.String(d.tableName),
|
|
|
|
Key: map[string]types.AttributeValue{
|
|
|
|
"Digest": &types.AttributeValueMemberS{Value: digest.String()},
|
|
|
|
},
|
|
|
|
})
|
|
|
|
if err != nil {
|
2023-04-28 05:44:22 +03:00
|
|
|
d.log.Error().Err(err).Str("tableName", d.tableName).Msg("failed to get blob")
|
2022-11-22 20:29:57 +02:00
|
|
|
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
out := Blob{}
|
|
|
|
|
|
|
|
if resp.Item == nil {
|
2023-12-08 00:05:02 -08:00
|
|
|
d.log.Debug().Err(zerr.ErrCacheMiss).Str("digest", string(digest)).Msg("failed to find blob in cache")
|
2022-11-22 20:29:57 +02:00
|
|
|
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
_ = attributevalue.UnmarshalMap(resp.Item, &out)
|
|
|
|
|
2023-10-10 20:29:07 +03:00
|
|
|
if out.OriginalBlobPath == path {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, item := range out.DuplicateBlobPath {
|
2022-11-22 20:29:57 +02:00
|
|
|
if item == path {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-12-08 00:05:02 -08:00
|
|
|
d.log.Debug().Err(zerr.ErrCacheMiss).Str("digest", string(digest)).Msg("failed to find blob in cache")
|
2022-11-22 20:29:57 +02:00
|
|
|
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
func (d *DynamoDBDriver) DeleteBlob(digest godigest.Digest, path string) error {
|
|
|
|
marshaledKey, _ := attributevalue.MarshalMap(map[string]interface{}{"Digest": digest.String()})
|
|
|
|
|
2023-10-10 20:29:07 +03:00
|
|
|
expression := "DELETE DuplicateBlobPath :i"
|
2022-11-22 20:29:57 +02:00
|
|
|
attrPath := types.AttributeValueMemberSS{Value: []string{path}}
|
|
|
|
|
2023-10-10 20:29:07 +03:00
|
|
|
if err := d.updateItem(digest, expression, map[string]types.AttributeValue{":i": &attrPath}); err != nil {
|
2023-12-08 00:05:02 -08:00
|
|
|
d.log.Error().Err(err).Str("digest", digest.String()).Str("path", path).Msg("failed to delete")
|
2022-11-22 20:29:57 +02:00
|
|
|
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2023-10-10 20:29:07 +03:00
|
|
|
originBlob, _ := d.GetBlob(digest)
|
|
|
|
// if original blob is the one deleted
|
|
|
|
if originBlob == path {
|
|
|
|
// move duplicate blob to original, storage will move content here
|
2023-11-24 10:38:36 +02:00
|
|
|
originBlob, _ = d.GetDuplicateBlob(digest)
|
2023-10-10 20:29:07 +03:00
|
|
|
if originBlob != "" {
|
|
|
|
if err := d.putOriginBlob(digest, originBlob); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2022-11-22 20:29:57 +02:00
|
|
|
|
2023-10-10 20:29:07 +03:00
|
|
|
if originBlob == "" {
|
2022-11-22 20:29:57 +02:00
|
|
|
d.log.Debug().Str("digest", digest.String()).Str("path", path).Msg("deleting empty bucket")
|
|
|
|
|
|
|
|
_, _ = d.client.DeleteItem(context.TODO(), &dynamodb.DeleteItemInput{
|
|
|
|
Key: marshaledKey,
|
|
|
|
TableName: &d.tableName,
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
2023-10-10 20:29:07 +03:00
|
|
|
|
2023-11-24 10:38:36 +02:00
|
|
|
func (d *DynamoDBDriver) GetDuplicateBlob(digest godigest.Digest) (string, error) {
|
2023-10-10 20:29:07 +03:00
|
|
|
resp, err := d.client.GetItem(context.TODO(), &dynamodb.GetItemInput{
|
|
|
|
TableName: aws.String(d.tableName),
|
|
|
|
Key: map[string]types.AttributeValue{
|
|
|
|
"Digest": &types.AttributeValueMemberS{Value: digest.String()},
|
|
|
|
},
|
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
d.log.Error().Err(err).Str("tableName", d.tableName).Msg("failed to get blob")
|
|
|
|
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
|
|
|
|
out := Blob{}
|
|
|
|
|
|
|
|
if resp.Item == nil {
|
|
|
|
return "", zerr.ErrCacheMiss
|
|
|
|
}
|
|
|
|
|
|
|
|
_ = attributevalue.UnmarshalMap(resp.Item, &out)
|
|
|
|
|
|
|
|
if len(out.DuplicateBlobPath) == 0 {
|
|
|
|
return "", nil
|
|
|
|
}
|
|
|
|
|
|
|
|
return out.DuplicateBlobPath[0], nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (d *DynamoDBDriver) putOriginBlob(digest godigest.Digest, path string) error {
|
|
|
|
expression := "SET OriginalBlobPath = :s"
|
|
|
|
attrPath := types.AttributeValueMemberS{Value: path}
|
|
|
|
|
|
|
|
if err := d.updateItem(digest, expression, map[string]types.AttributeValue{":s": &attrPath}); err != nil {
|
2023-12-08 00:05:02 -08:00
|
|
|
d.log.Error().Err(err).Str("digest", digest.String()).Str("path", path).Msg("failed to put original blob")
|
2023-10-10 20:29:07 +03:00
|
|
|
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (d *DynamoDBDriver) updateItem(digest godigest.Digest, expression string,
|
|
|
|
expressionAttVals map[string]types.AttributeValue,
|
|
|
|
) error {
|
|
|
|
marshaledKey, _ := attributevalue.MarshalMap(map[string]interface{}{"Digest": digest.String()})
|
|
|
|
|
|
|
|
_, err := d.client.UpdateItem(context.TODO(), &dynamodb.UpdateItemInput{
|
|
|
|
Key: marshaledKey,
|
|
|
|
TableName: &d.tableName,
|
|
|
|
UpdateExpression: &expression,
|
|
|
|
ExpressionAttributeValues: expressionAttVals,
|
|
|
|
})
|
|
|
|
|
|
|
|
return err
|
|
|
|
}
|