mirror of
https://github.com/project-zot/zot.git
synced 2025-01-06 22:40:28 -05:00
f618b1d4ef
* ci(deps): upgrade golangci-lint
Signed-off-by: Jan-Otto Kröpke <mail@jkroepke.de>
* build(deps): removed disabled linters
Signed-off-by: Jan-Otto Kröpke <mail@jkroepke.de>
* build(deps): go run github.com/daixiang0/gci@latest write .
Signed-off-by: Jan-Otto Kröpke <joe@cloudeteer.de>
* build(deps): go run golang.org/x/tools/cmd/goimports@latest -l -w .
Signed-off-by: Jan-Otto Kröpke <joe@cloudeteer.de>
* build(deps): go run github.com/bombsimon/wsl/v4/cmd...@latest -strict-append -test=true -fix ./...
Signed-off-by: Jan-Otto Kröpke <joe@cloudeteer.de>
* build(deps): go run github.com/catenacyber/perfsprint@latest -fix ./...
Signed-off-by: Jan-Otto Kröpke <joe@cloudeteer.de>
* build(deps): replace gomnd by mnd
Signed-off-by: Jan-Otto Kröpke <joe@cloudeteer.de>
* build(deps): make gqlgen
Signed-off-by: Jan-Otto Kröpke <joe@cloudeteer.de>
* build: Revert "build(deps): go run github.com/daixiang0/gci@latest write ."
This reverts commit 5bf8c42e1f
.
Signed-off-by: Jan-Otto Kröpke <joe@cloudeteer.de>
* build(deps): go run github.com/daixiang0/gci@latest write -s 'standard' -s default -s 'prefix(zotregistry.dev/zot)' .
Signed-off-by: Jan-Otto Kröpke <joe@cloudeteer.de>
* build(deps): make gqlgen
Signed-off-by: Jan-Otto Kröpke <joe@cloudeteer.de>
* fix: wsl issues
Signed-off-by: Jan-Otto Kröpke <joe@cloudeteer.de>
* fix: wsl issues
Signed-off-by: Jan-Otto Kröpke <joe@cloudeteer.de>
* fix: wsl issues
Signed-off-by: Jan-Otto Kröpke <joe@cloudeteer.de>
* fix: wsl issues
Signed-off-by: Jan-Otto Kröpke <joe@cloudeteer.de>
* fix: wsl issues
Signed-off-by: Jan-Otto Kröpke <joe@cloudeteer.de>
* fix: wsl issues
Signed-off-by: Jan-Otto Kröpke <joe@cloudeteer.de>
* fix: wsl issues
Signed-off-by: Jan-Otto Kröpke <joe@cloudeteer.de>
* fix: wsl issues
Signed-off-by: Jan-Otto Kröpke <joe@cloudeteer.de>
* fix: wsl issues
Signed-off-by: Jan-Otto Kröpke <joe@cloudeteer.de>
* fix: wsl issues
Signed-off-by: Jan-Otto Kröpke <joe@cloudeteer.de>
* fix: wsl issues
Signed-off-by: Jan-Otto Kröpke <joe@cloudeteer.de>
* fix: wsl issues
Signed-off-by: Jan-Otto Kröpke <joe@cloudeteer.de>
* fix: wsl issues
Signed-off-by: Jan-Otto Kröpke <joe@cloudeteer.de>
* fix: wsl issues
Signed-off-by: Jan-Otto Kröpke <joe@cloudeteer.de>
* fix: wsl issues
Signed-off-by: Jan-Otto Kröpke <joe@cloudeteer.de>
* fix: wsl issues
Signed-off-by: Jan-Otto Kröpke <joe@cloudeteer.de>
* fix: wsl issues
Signed-off-by: Jan-Otto Kröpke <joe@cloudeteer.de>
* fix: wsl issues
Signed-off-by: Jan-Otto Kröpke <joe@cloudeteer.de>
* fix: wsl issues
Signed-off-by: Jan-Otto Kröpke <joe@cloudeteer.de>
* fix: wsl issues
Signed-off-by: Jan-Otto Kröpke <joe@cloudeteer.de>
* fix: wsl issues
Signed-off-by: Jan-Otto Kröpke <joe@cloudeteer.de>
* fix: wsl issues
Signed-off-by: Jan-Otto Kröpke <joe@cloudeteer.de>
* fix: wsl issues
Signed-off-by: Jan-Otto Kröpke <joe@cloudeteer.de>
* fix: wsl issues
Signed-off-by: Jan-Otto Kröpke <joe@cloudeteer.de>
* fix: check-log issues
Signed-off-by: Jan-Otto Kröpke <joe@cloudeteer.de>
* fix: gci issues
Signed-off-by: Jan-Otto Kröpke <joe@cloudeteer.de>
* fix: tests
Signed-off-by: Jan-Otto Kröpke <joe@cloudeteer.de>
---------
Signed-off-by: Jan-Otto Kröpke <mail@jkroepke.de>
Signed-off-by: Jan-Otto Kröpke <joe@cloudeteer.de>
387 lines
9.7 KiB
Go
387 lines
9.7 KiB
Go
package cache
|
|
|
|
import (
|
|
"fmt"
|
|
"os"
|
|
"path"
|
|
"path/filepath"
|
|
"strings"
|
|
|
|
godigest "github.com/opencontainers/go-digest"
|
|
"go.etcd.io/bbolt"
|
|
|
|
zerr "zotregistry.dev/zot/errors"
|
|
zlog "zotregistry.dev/zot/pkg/log"
|
|
"zotregistry.dev/zot/pkg/storage/constants"
|
|
)
|
|
|
|
type BoltDBDriver struct {
|
|
rootDir string
|
|
db *bbolt.DB
|
|
log zlog.Logger
|
|
useRelPaths bool // whether or not to use relative paths, should be true for filesystem and false for s3
|
|
}
|
|
|
|
type BoltDBDriverParameters struct {
|
|
RootDir string
|
|
Name string
|
|
UseRelPaths bool
|
|
}
|
|
|
|
func NewBoltDBCache(parameters interface{}, log zlog.Logger) (*BoltDBDriver, error) {
|
|
properParameters, ok := parameters.(BoltDBDriverParameters)
|
|
if !ok {
|
|
log.Error().Err(zerr.ErrTypeAssertionFailed).Msgf("failed to cast type, expected type '%T' but got '%T'",
|
|
BoltDBDriverParameters{}, parameters)
|
|
|
|
return nil, zerr.ErrTypeAssertionFailed
|
|
}
|
|
|
|
err := os.MkdirAll(properParameters.RootDir, constants.DefaultDirPerms)
|
|
if err != nil {
|
|
log.Error().Err(err).Str("directory", properParameters.RootDir).Msg("failed to create directory for cache db")
|
|
|
|
return nil, err
|
|
}
|
|
|
|
dbPath := path.Join(properParameters.RootDir, properParameters.Name+constants.DBExtensionName)
|
|
dbOpts := &bbolt.Options{
|
|
Timeout: constants.DBCacheLockCheckTimeout,
|
|
FreelistType: bbolt.FreelistArrayType,
|
|
}
|
|
|
|
cacheDB, err := bbolt.Open(dbPath, 0o600, dbOpts) //nolint:mnd
|
|
if err != nil {
|
|
if strings.Contains(err.Error(), "timeout") {
|
|
err := fmt.Errorf("%w: %w, path '%s'", zerr.ErrTimeout, zerr.ErrDatabaseFileAlreadyInUse, dbPath)
|
|
|
|
log.Error().Err(err).Str("dbPath", dbPath).Msg("failed to create cache db")
|
|
|
|
return nil, err
|
|
}
|
|
|
|
log.Error().Err(err).Str("dbPath", dbPath).Msg("failed to create cache db")
|
|
|
|
return nil, err
|
|
}
|
|
|
|
if err := cacheDB.Update(func(tx *bbolt.Tx) error {
|
|
if _, err := tx.CreateBucketIfNotExists([]byte(constants.BlobsCache)); err != nil {
|
|
// this is a serious failure
|
|
log.Error().Err(err).Str("dbPath", dbPath).Msg("failed to create a root bucket")
|
|
|
|
return err
|
|
}
|
|
|
|
return nil
|
|
}); err != nil {
|
|
// something went wrong
|
|
log.Error().Err(err).Msg("failed to create a cache")
|
|
|
|
return nil, err
|
|
}
|
|
|
|
return &BoltDBDriver{
|
|
rootDir: properParameters.RootDir,
|
|
db: cacheDB,
|
|
useRelPaths: properParameters.UseRelPaths,
|
|
log: log,
|
|
}, nil
|
|
}
|
|
|
|
func (d *BoltDBDriver) UsesRelativePaths() bool {
|
|
return d.useRelPaths
|
|
}
|
|
|
|
func (d *BoltDBDriver) Name() string {
|
|
return "boltdb"
|
|
}
|
|
|
|
func (d *BoltDBDriver) PutBlob(digest godigest.Digest, path string) error {
|
|
if path == "" {
|
|
d.log.Error().Err(zerr.ErrEmptyValue).Str("digest", digest.String()).
|
|
Msg("failed to put blob due to empty path being provided")
|
|
|
|
return zerr.ErrEmptyValue
|
|
}
|
|
|
|
// use only relative (to rootDir) paths on blobs
|
|
var err error
|
|
if d.useRelPaths {
|
|
path, err = filepath.Rel(d.rootDir, path)
|
|
if err != nil {
|
|
d.log.Error().Err(err).Str("path", path).Msg("failed to get relative path")
|
|
}
|
|
}
|
|
|
|
if err := d.db.Update(func(tx *bbolt.Tx) error {
|
|
root := tx.Bucket([]byte(constants.BlobsCache))
|
|
if root == nil {
|
|
// this is a serious failure
|
|
err := zerr.ErrCacheRootBucket
|
|
d.log.Error().Err(err).Msg("failed to access root bucket")
|
|
|
|
return err
|
|
}
|
|
|
|
bucket, err := root.CreateBucketIfNotExists([]byte(digest.String()))
|
|
if err != nil {
|
|
// this is a serious failure
|
|
d.log.Error().Err(err).Str("bucket", digest.String()).Msg("failed to create a bucket")
|
|
|
|
return err
|
|
}
|
|
|
|
// create nested deduped bucket where we store all the deduped blobs + original blob
|
|
deduped, err := bucket.CreateBucketIfNotExists([]byte(constants.DuplicatesBucket))
|
|
if err != nil {
|
|
// this is a serious failure
|
|
d.log.Error().Err(err).Str("bucket", constants.DuplicatesBucket).Msg("failed to create a bucket")
|
|
|
|
return err
|
|
}
|
|
|
|
if err := deduped.Put([]byte(path), nil); err != nil {
|
|
d.log.Error().Err(err).Str("bucket", constants.DuplicatesBucket).Str("value", path).Msg("failed to put record")
|
|
|
|
return err
|
|
}
|
|
|
|
// create origin bucket and insert only the original blob
|
|
origin := bucket.Bucket([]byte(constants.OriginalBucket))
|
|
if origin == nil {
|
|
// if the bucket doesn't exist yet then 'path' is the original blob
|
|
origin, err := bucket.CreateBucket([]byte(constants.OriginalBucket))
|
|
if err != nil {
|
|
// this is a serious failure
|
|
d.log.Error().Err(err).Str("bucket", constants.OriginalBucket).Msg("failed to create a bucket")
|
|
|
|
return err
|
|
}
|
|
|
|
if err := origin.Put([]byte(path), nil); err != nil {
|
|
d.log.Error().Err(err).Str("bucket", constants.OriginalBucket).Str("value", path).Msg("failed to put record")
|
|
|
|
return err
|
|
}
|
|
}
|
|
|
|
return nil
|
|
}); err != nil {
|
|
return err
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func (d *BoltDBDriver) GetAllBlobs(digest godigest.Digest) ([]string, error) {
|
|
var blobPath strings.Builder
|
|
|
|
blobPaths := []string{}
|
|
|
|
if err := d.db.View(func(tx *bbolt.Tx) error {
|
|
root := tx.Bucket([]byte(constants.BlobsCache))
|
|
if root == nil {
|
|
// this is a serious failure
|
|
err := zerr.ErrCacheRootBucket
|
|
d.log.Error().Err(err).Msg("failed to access root bucket")
|
|
|
|
return err
|
|
}
|
|
|
|
bucket := root.Bucket([]byte(digest.String()))
|
|
if bucket != nil {
|
|
origin := bucket.Bucket([]byte(constants.OriginalBucket))
|
|
blobPath.Write(d.getOne(origin))
|
|
originBlob := blobPath.String()
|
|
|
|
blobPaths = append(blobPaths, originBlob)
|
|
|
|
deduped := bucket.Bucket([]byte(constants.DuplicatesBucket))
|
|
if deduped != nil {
|
|
cursor := deduped.Cursor()
|
|
|
|
for k, _ := cursor.First(); k != nil; k, _ = cursor.Next() {
|
|
var blobPath strings.Builder
|
|
|
|
blobPath.Write(k)
|
|
|
|
duplicateBlob := blobPath.String()
|
|
|
|
if duplicateBlob != originBlob {
|
|
blobPaths = append(blobPaths, duplicateBlob)
|
|
}
|
|
}
|
|
|
|
return nil
|
|
}
|
|
}
|
|
|
|
return zerr.ErrCacheMiss
|
|
}); err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
return blobPaths, nil
|
|
}
|
|
|
|
func (d *BoltDBDriver) GetBlob(digest godigest.Digest) (string, error) {
|
|
var blobPath strings.Builder
|
|
|
|
if err := d.db.View(func(tx *bbolt.Tx) error {
|
|
root := tx.Bucket([]byte(constants.BlobsCache))
|
|
if root == nil {
|
|
// this is a serious failure
|
|
err := zerr.ErrCacheRootBucket
|
|
d.log.Error().Err(err).Msg("failed to access root bucket")
|
|
|
|
return err
|
|
}
|
|
|
|
bucket := root.Bucket([]byte(digest.String()))
|
|
if bucket != nil {
|
|
origin := bucket.Bucket([]byte(constants.OriginalBucket))
|
|
blobPath.Write(d.getOne(origin))
|
|
|
|
return nil
|
|
}
|
|
|
|
return zerr.ErrCacheMiss
|
|
}); err != nil {
|
|
return "", err
|
|
}
|
|
|
|
return blobPath.String(), nil
|
|
}
|
|
|
|
func (d *BoltDBDriver) HasBlob(digest godigest.Digest, blob string) bool {
|
|
if err := d.db.View(func(tx *bbolt.Tx) error {
|
|
root := tx.Bucket([]byte(constants.BlobsCache))
|
|
if root == nil {
|
|
// this is a serious failure
|
|
err := zerr.ErrCacheRootBucket
|
|
d.log.Error().Err(err).Msg("failed to access root bucket")
|
|
|
|
return err
|
|
}
|
|
|
|
bucket := root.Bucket([]byte(digest.String()))
|
|
if bucket == nil {
|
|
return zerr.ErrCacheMiss
|
|
}
|
|
|
|
origin := bucket.Bucket([]byte(constants.OriginalBucket))
|
|
if origin == nil {
|
|
return zerr.ErrCacheMiss
|
|
}
|
|
|
|
deduped := bucket.Bucket([]byte(constants.DuplicatesBucket))
|
|
if deduped == nil {
|
|
return zerr.ErrCacheMiss
|
|
}
|
|
|
|
if origin.Get([]byte(blob)) == nil {
|
|
if deduped.Get([]byte(blob)) == nil {
|
|
return zerr.ErrCacheMiss
|
|
}
|
|
}
|
|
|
|
return nil
|
|
}); err != nil {
|
|
return false
|
|
}
|
|
|
|
return true
|
|
}
|
|
|
|
func (d *BoltDBDriver) getOne(bucket *bbolt.Bucket) []byte {
|
|
if bucket != nil {
|
|
cursor := bucket.Cursor()
|
|
k, _ := cursor.First()
|
|
|
|
return k
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func (d *BoltDBDriver) DeleteBlob(digest godigest.Digest, path string) error {
|
|
// use only relative (to rootDir) paths on blobs
|
|
var err error
|
|
if d.useRelPaths {
|
|
path, err = filepath.Rel(d.rootDir, path)
|
|
if err != nil {
|
|
d.log.Error().Err(err).Str("path", path).Msg("failed to get relative path")
|
|
}
|
|
}
|
|
|
|
if err := d.db.Update(func(tx *bbolt.Tx) error {
|
|
root := tx.Bucket([]byte(constants.BlobsCache))
|
|
if root == nil {
|
|
// this is a serious failure
|
|
err := zerr.ErrCacheRootBucket
|
|
d.log.Error().Err(err).Msg("failed to access root bucket")
|
|
|
|
return err
|
|
}
|
|
|
|
bucket := root.Bucket([]byte(digest.String()))
|
|
if bucket == nil {
|
|
return zerr.ErrCacheMiss
|
|
}
|
|
|
|
deduped := bucket.Bucket([]byte(constants.DuplicatesBucket))
|
|
if deduped == nil {
|
|
return zerr.ErrCacheMiss
|
|
}
|
|
|
|
if err := deduped.Delete([]byte(path)); err != nil {
|
|
d.log.Error().Err(err).Str("digest", digest.String()).Str("bucket", constants.DuplicatesBucket).
|
|
Str("path", path).Msg("failed to delete")
|
|
|
|
return err
|
|
}
|
|
|
|
origin := bucket.Bucket([]byte(constants.OriginalBucket))
|
|
if origin != nil {
|
|
originBlob := d.getOne(origin)
|
|
if originBlob != nil {
|
|
if err := origin.Delete([]byte(path)); err != nil {
|
|
d.log.Error().Err(err).Str("digest", digest.String()).Str("bucket", constants.OriginalBucket).
|
|
Str("path", path).Msg("failed to delete")
|
|
|
|
return err
|
|
}
|
|
|
|
// move next candidate to origin bucket, next GetKey will return this one and storage will move the content here
|
|
dedupedBlob := d.getOne(deduped)
|
|
if dedupedBlob != nil {
|
|
if err := origin.Put(dedupedBlob, nil); err != nil {
|
|
d.log.Error().Err(err).Str("digest", digest.String()).Str("bucket", constants.OriginalBucket).Str("path", path).
|
|
Msg("failed to put")
|
|
|
|
return err
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// if no key in origin bucket then digest bucket is empty, remove it
|
|
k := d.getOne(origin)
|
|
if k == nil {
|
|
d.log.Debug().Str("digest", digest.String()).Str("path", path).Msg("deleting empty bucket")
|
|
if err := root.DeleteBucket([]byte(digest)); err != nil {
|
|
d.log.Error().Err(err).Str("digest", digest.String()).Str("bucket", digest.String()).Str("path", path).
|
|
Msg("failed to delete")
|
|
|
|
return err
|
|
}
|
|
}
|
|
|
|
return nil
|
|
}); err != nil {
|
|
return err
|
|
}
|
|
|
|
return nil
|
|
}
|