2023-05-31 20:26:23 +03:00
|
|
|
//go:build sync
|
|
|
|
// +build sync
|
|
|
|
|
2021-06-08 23:11:18 +03:00
|
|
|
package sync
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
2023-05-31 20:26:23 +03:00
|
|
|
"errors"
|
2022-01-10 18:06:12 +02:00
|
|
|
"sync"
|
|
|
|
"time"
|
2021-06-08 23:11:18 +03:00
|
|
|
|
|
|
|
"github.com/containers/common/pkg/retry"
|
2022-10-20 19:39:20 +03:00
|
|
|
|
2024-02-01 06:34:07 +02:00
|
|
|
zerr "zotregistry.dev/zot/errors"
|
|
|
|
"zotregistry.dev/zot/pkg/common"
|
|
|
|
"zotregistry.dev/zot/pkg/log"
|
2022-11-15 08:21:49 +02:00
|
|
|
)
|
|
|
|
|
2023-05-31 20:26:23 +03:00
|
|
|
type request struct {
|
|
|
|
repo string
|
|
|
|
reference string
|
|
|
|
// used for background retries, at most one background retry per service
|
|
|
|
serviceID int
|
|
|
|
isBackground bool
|
2022-03-07 10:45:10 +02:00
|
|
|
}
|
|
|
|
|
2023-05-31 20:26:23 +03:00
|
|
|
/*
|
|
|
|
a request can be an image/signature/sbom
|
|
|
|
|
|
|
|
keep track of all parallel requests, if two requests of same image/signature/sbom comes at the same time,
|
|
|
|
process just the first one, also keep track of all background retrying routines.
|
|
|
|
*/
|
|
|
|
type BaseOnDemand struct {
|
|
|
|
services []Service
|
|
|
|
// map[request]chan err
|
|
|
|
requestStore *sync.Map
|
|
|
|
log log.Logger
|
2022-01-10 18:06:12 +02:00
|
|
|
}
|
|
|
|
|
2023-05-31 20:26:23 +03:00
|
|
|
func NewOnDemand(log log.Logger) *BaseOnDemand {
|
|
|
|
return &BaseOnDemand{log: log, requestStore: &sync.Map{}}
|
2022-01-10 18:06:12 +02:00
|
|
|
}
|
|
|
|
|
2023-05-31 20:26:23 +03:00
|
|
|
func (onDemand *BaseOnDemand) Add(service Service) {
|
|
|
|
onDemand.services = append(onDemand.services, service)
|
2022-01-10 18:06:12 +02:00
|
|
|
}
|
|
|
|
|
2023-09-05 19:48:56 +03:00
|
|
|
func (onDemand *BaseOnDemand) SyncImage(ctx context.Context, repo, reference string) error {
|
2023-05-31 20:26:23 +03:00
|
|
|
req := request{
|
|
|
|
repo: repo,
|
|
|
|
reference: reference,
|
|
|
|
}
|
2022-01-10 18:06:12 +02:00
|
|
|
|
2023-05-31 20:26:23 +03:00
|
|
|
val, found := onDemand.requestStore.Load(req)
|
2022-01-10 18:06:12 +02:00
|
|
|
if found {
|
2023-05-31 20:26:23 +03:00
|
|
|
onDemand.log.Info().Str("repo", repo).Str("reference", reference).
|
|
|
|
Msg("image already demanded, waiting on channel")
|
|
|
|
|
|
|
|
syncResult, _ := val.(chan error)
|
2022-01-10 18:06:12 +02:00
|
|
|
|
2023-05-31 20:26:23 +03:00
|
|
|
err, ok := <-syncResult
|
2022-01-10 18:06:12 +02:00
|
|
|
// if channel closed exit
|
|
|
|
if !ok {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2023-05-31 20:26:23 +03:00
|
|
|
syncResult := make(chan error)
|
|
|
|
onDemand.requestStore.Store(req, syncResult)
|
|
|
|
|
|
|
|
defer onDemand.requestStore.Delete(req)
|
|
|
|
defer close(syncResult)
|
2022-01-10 18:06:12 +02:00
|
|
|
|
2023-09-05 19:48:56 +03:00
|
|
|
go onDemand.syncImage(ctx, repo, reference, syncResult)
|
2021-06-08 23:11:18 +03:00
|
|
|
|
2023-05-31 20:26:23 +03:00
|
|
|
err, ok := <-syncResult
|
2022-01-10 18:06:12 +02:00
|
|
|
if !ok {
|
2021-12-07 20:26:26 +02:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2022-01-10 18:06:12 +02:00
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2023-09-05 19:48:56 +03:00
|
|
|
func (onDemand *BaseOnDemand) SyncReference(ctx context.Context, repo string,
|
|
|
|
subjectDigestStr string, referenceType string,
|
|
|
|
) error {
|
2023-05-31 20:26:23 +03:00
|
|
|
var err error
|
2022-01-10 18:06:12 +02:00
|
|
|
|
2023-05-31 20:26:23 +03:00
|
|
|
for _, service := range onDemand.services {
|
|
|
|
err = service.SetNextAvailableURL()
|
2021-06-08 23:11:18 +03:00
|
|
|
if err != nil {
|
2023-05-31 20:26:23 +03:00
|
|
|
return err
|
2021-06-08 23:11:18 +03:00
|
|
|
}
|
2021-12-13 19:23:31 +00:00
|
|
|
|
2023-09-05 19:48:56 +03:00
|
|
|
err = service.SyncReference(ctx, repo, subjectDigestStr, referenceType)
|
2023-05-31 20:26:23 +03:00
|
|
|
if err != nil {
|
2021-06-08 23:11:18 +03:00
|
|
|
continue
|
2023-05-31 20:26:23 +03:00
|
|
|
} else {
|
|
|
|
return nil
|
2021-06-08 23:11:18 +03:00
|
|
|
}
|
2023-05-31 20:26:23 +03:00
|
|
|
}
|
2021-06-08 23:11:18 +03:00
|
|
|
|
2023-05-31 20:26:23 +03:00
|
|
|
return err
|
|
|
|
}
|
2021-10-25 15:05:03 +03:00
|
|
|
|
2023-09-05 19:48:56 +03:00
|
|
|
func (onDemand *BaseOnDemand) syncImage(ctx context.Context, repo, reference string, syncResult chan error) {
|
2023-05-31 20:26:23 +03:00
|
|
|
var err error
|
|
|
|
for serviceID, service := range onDemand.services {
|
|
|
|
err = service.SetNextAvailableURL()
|
2024-02-14 19:18:10 +02:00
|
|
|
|
|
|
|
isPingErr := errors.Is(err, zerr.ErrSyncPingRegistry)
|
|
|
|
if err != nil && !isPingErr {
|
2023-05-31 20:26:23 +03:00
|
|
|
syncResult <- err
|
2022-01-10 18:06:12 +02:00
|
|
|
|
2023-05-31 20:26:23 +03:00
|
|
|
return
|
2022-01-10 18:06:12 +02:00
|
|
|
}
|
|
|
|
|
2024-02-14 19:18:10 +02:00
|
|
|
// no need to try to sync inline if there is a ping error, we want to retry in background
|
|
|
|
if !isPingErr {
|
|
|
|
err = service.SyncImage(ctx, repo, reference)
|
|
|
|
}
|
|
|
|
|
|
|
|
if err != nil || isPingErr {
|
2023-05-31 20:26:23 +03:00
|
|
|
if errors.Is(err, zerr.ErrManifestNotFound) ||
|
|
|
|
errors.Is(err, zerr.ErrSyncImageFilteredOut) ||
|
|
|
|
errors.Is(err, zerr.ErrSyncImageNotSigned) {
|
|
|
|
continue
|
2023-03-07 19:58:42 +02:00
|
|
|
}
|
|
|
|
|
2023-05-31 20:26:23 +03:00
|
|
|
req := request{
|
|
|
|
repo: repo,
|
|
|
|
reference: reference,
|
|
|
|
serviceID: serviceID,
|
|
|
|
isBackground: true,
|
2022-01-10 18:06:12 +02:00
|
|
|
}
|
2023-03-07 19:58:42 +02:00
|
|
|
|
2023-05-31 20:26:23 +03:00
|
|
|
// if there is already a background routine, skip
|
|
|
|
if _, requested := onDemand.requestStore.LoadOrStore(req, struct{}{}); requested {
|
2022-03-07 10:45:10 +02:00
|
|
|
continue
|
|
|
|
}
|
2021-10-28 12:10:01 +03:00
|
|
|
|
2023-05-31 20:26:23 +03:00
|
|
|
retryOptions := service.GetRetryOptions()
|
|
|
|
|
|
|
|
if retryOptions.MaxRetry > 0 {
|
|
|
|
// retry in background
|
|
|
|
go func(service Service) {
|
2022-01-10 18:06:12 +02:00
|
|
|
// remove image after syncing
|
|
|
|
defer func() {
|
2023-05-31 20:26:23 +03:00
|
|
|
onDemand.requestStore.Delete(req)
|
|
|
|
onDemand.log.Info().Str("repo", repo).Str("reference", reference).
|
|
|
|
Msg("sync routine for image exited")
|
2022-01-10 18:06:12 +02:00
|
|
|
}()
|
|
|
|
|
2023-05-31 20:26:23 +03:00
|
|
|
onDemand.log.Info().Str("repo", repo).Str(reference, "reference").Str("err", err.Error()).
|
2023-12-08 00:05:02 -08:00
|
|
|
Str("component", "sync").Msg("starting routine to copy image, because of error")
|
2023-05-31 20:26:23 +03:00
|
|
|
|
2022-01-10 18:06:12 +02:00
|
|
|
time.Sleep(retryOptions.Delay)
|
|
|
|
|
2023-09-05 19:48:56 +03:00
|
|
|
// retrying in background, can't use the same context which should be cancelled by now.
|
2023-05-31 20:26:23 +03:00
|
|
|
if err = retry.RetryIfNecessary(context.Background(), func() error {
|
2023-09-05 19:48:56 +03:00
|
|
|
err := service.SyncImage(context.Background(), repo, reference)
|
2022-01-10 18:06:12 +02:00
|
|
|
|
|
|
|
return err
|
|
|
|
}, retryOptions); err != nil {
|
2023-05-31 20:26:23 +03:00
|
|
|
onDemand.log.Error().Str("errorType", common.TypeOf(err)).Str("repo", repo).Str("reference", reference).
|
2023-12-08 00:05:02 -08:00
|
|
|
Err(err).Str("component", "sync").Msg("failed to copy image")
|
2022-01-10 18:06:12 +02:00
|
|
|
}
|
2023-05-31 20:26:23 +03:00
|
|
|
}(service)
|
2021-12-07 20:26:26 +02:00
|
|
|
}
|
2023-05-31 20:26:23 +03:00
|
|
|
} else {
|
|
|
|
break
|
2021-06-08 23:11:18 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-05-31 20:26:23 +03:00
|
|
|
syncResult <- err
|
2022-11-15 08:21:49 +02:00
|
|
|
}
|