2021-06-08 23:11:18 +03:00
|
|
|
package sync
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"encoding/json"
|
2022-03-07 10:45:10 +02:00
|
|
|
"errors"
|
2021-06-08 23:11:18 +03:00
|
|
|
"fmt"
|
2021-10-28 12:10:01 +03:00
|
|
|
"io"
|
2021-06-08 23:11:18 +03:00
|
|
|
"os"
|
|
|
|
"regexp"
|
2021-12-02 19:45:26 +02:00
|
|
|
goSync "sync"
|
2021-06-08 23:11:18 +03:00
|
|
|
"time"
|
|
|
|
|
|
|
|
"github.com/Masterminds/semver"
|
|
|
|
"github.com/containers/common/pkg/retry"
|
|
|
|
"github.com/containers/image/v5/copy"
|
|
|
|
"github.com/containers/image/v5/docker"
|
|
|
|
"github.com/containers/image/v5/docker/reference"
|
|
|
|
"github.com/containers/image/v5/signature"
|
|
|
|
"github.com/containers/image/v5/types"
|
|
|
|
ispec "github.com/opencontainers/image-spec/specs-go/v1"
|
|
|
|
"gopkg.in/resty.v1"
|
2022-03-07 10:45:10 +02:00
|
|
|
zerr "zotregistry.io/zot/errors"
|
2022-02-24 12:31:36 -08:00
|
|
|
"zotregistry.io/zot/pkg/api/constants"
|
2021-12-04 03:50:58 +00:00
|
|
|
"zotregistry.io/zot/pkg/log"
|
|
|
|
"zotregistry.io/zot/pkg/storage"
|
2022-02-18 13:36:50 +02:00
|
|
|
"zotregistry.io/zot/pkg/test"
|
2021-06-08 23:11:18 +03:00
|
|
|
)
|
|
|
|
|
|
|
|
const (
|
2022-05-25 13:23:06 +03:00
|
|
|
SyncBlobUploadDir = ".sync"
|
|
|
|
httpMaxRedirectsCount = 15
|
2021-06-08 23:11:18 +03:00
|
|
|
)
|
|
|
|
|
|
|
|
// /v2/_catalog struct.
|
|
|
|
type catalog struct {
|
|
|
|
Repositories []string `json:"repositories"`
|
|
|
|
}
|
|
|
|
|
|
|
|
// key is registry address.
|
|
|
|
type CredentialsFile map[string]Credentials
|
|
|
|
|
|
|
|
type Credentials struct {
|
|
|
|
Username string
|
|
|
|
Password string
|
|
|
|
}
|
|
|
|
|
|
|
|
type Config struct {
|
2021-12-28 15:29:30 +02:00
|
|
|
Enable *bool
|
2021-06-08 23:11:18 +03:00
|
|
|
CredentialsFile string
|
|
|
|
Registries []RegistryConfig
|
|
|
|
}
|
|
|
|
|
|
|
|
type RegistryConfig struct {
|
2021-12-29 17:14:56 +02:00
|
|
|
URLs []string
|
2021-06-08 23:11:18 +03:00
|
|
|
PollInterval time.Duration
|
|
|
|
Content []Content
|
|
|
|
TLSVerify *bool
|
|
|
|
OnDemand bool
|
|
|
|
CertDir string
|
2022-01-10 18:06:12 +02:00
|
|
|
MaxRetries *int
|
|
|
|
RetryDelay *time.Duration
|
2022-03-07 10:45:10 +02:00
|
|
|
OnlySigned *bool
|
2021-06-08 23:11:18 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
type Content struct {
|
2022-03-10 17:39:11 +02:00
|
|
|
Prefix string
|
|
|
|
Tags *Tags
|
|
|
|
Destination string `mapstructure:",omitempty"`
|
|
|
|
StripPrefix bool
|
2021-06-08 23:11:18 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
type Tags struct {
|
|
|
|
Regex *string
|
|
|
|
Semver *bool
|
|
|
|
}
|
|
|
|
|
|
|
|
// getUpstreamCatalog gets all repos from a registry.
|
2021-12-29 17:14:56 +02:00
|
|
|
func getUpstreamCatalog(client *resty.Client, upstreamURL string, log log.Logger) (catalog, error) {
|
2022-03-21 17:37:23 +00:00
|
|
|
var catalog catalog
|
2021-06-08 23:11:18 +03:00
|
|
|
|
2022-02-24 12:31:36 -08:00
|
|
|
registryCatalogURL := fmt.Sprintf("%s%s%s", upstreamURL, constants.RoutePrefix, constants.ExtCatalogPrefix)
|
2021-06-08 23:11:18 +03:00
|
|
|
|
|
|
|
resp, err := client.R().SetHeader("Content-Type", "application/json").Get(registryCatalogURL)
|
|
|
|
if err != nil {
|
|
|
|
log.Err(err).Msgf("couldn't query %s", registryCatalogURL)
|
2021-12-13 19:23:31 +00:00
|
|
|
|
2022-03-21 17:37:23 +00:00
|
|
|
return catalog, err
|
2021-06-08 23:11:18 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
if resp.IsError() {
|
|
|
|
log.Error().Msgf("couldn't query %s, status code: %d, body: %s", registryCatalogURL,
|
|
|
|
resp.StatusCode(), resp.Body())
|
2021-12-13 19:23:31 +00:00
|
|
|
|
2022-03-21 17:37:23 +00:00
|
|
|
return catalog, zerr.ErrSyncMissingCatalog
|
2021-06-08 23:11:18 +03:00
|
|
|
}
|
|
|
|
|
2022-03-21 17:37:23 +00:00
|
|
|
err = json.Unmarshal(resp.Body(), &catalog)
|
2021-06-08 23:11:18 +03:00
|
|
|
if err != nil {
|
|
|
|
log.Err(err).Str("body", string(resp.Body())).Msg("couldn't unmarshal registry's catalog")
|
2021-12-13 19:23:31 +00:00
|
|
|
|
2022-03-21 17:37:23 +00:00
|
|
|
return catalog, err
|
2021-06-08 23:11:18 +03:00
|
|
|
}
|
|
|
|
|
2022-03-21 17:37:23 +00:00
|
|
|
return catalog, nil
|
2021-06-08 23:11:18 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
// getImageTags lists all tags in a repository.
|
|
|
|
// It returns a string slice of tags and any error encountered.
|
|
|
|
func getImageTags(ctx context.Context, sysCtx *types.SystemContext, repoRef reference.Named) ([]string, error) {
|
|
|
|
dockerRef, err := docker.NewReference(reference.TagNameOnly(repoRef))
|
2022-02-18 13:36:50 +02:00
|
|
|
// hard to reach test case, injected error, see pkg/test/dev.go
|
|
|
|
if err = test.Error(err); err != nil {
|
2021-06-08 23:11:18 +03:00
|
|
|
return nil, err // Should never happen for a reference with tag and no digest
|
|
|
|
}
|
|
|
|
|
|
|
|
tags, err := docker.GetRepositoryTags(ctx, sysCtx, dockerRef)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return tags, nil
|
|
|
|
}
|
|
|
|
|
2021-10-28 12:10:01 +03:00
|
|
|
// filterImagesByTagRegex filters images by tag regex given in the config.
|
2021-06-08 23:11:18 +03:00
|
|
|
func filterImagesByTagRegex(upstreamReferences *[]types.ImageReference, content Content, log log.Logger) error {
|
|
|
|
refs := *upstreamReferences
|
|
|
|
|
|
|
|
if content.Tags == nil {
|
|
|
|
// no need to filter anything
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
if content.Tags.Regex != nil {
|
|
|
|
log.Info().Msgf("start filtering using the regular expression: %s", *content.Tags.Regex)
|
|
|
|
|
|
|
|
tagReg, err := regexp.Compile(*content.Tags.Regex)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2021-12-13 19:23:31 +00:00
|
|
|
numTags := 0
|
2021-06-08 23:11:18 +03:00
|
|
|
|
|
|
|
for _, ref := range refs {
|
|
|
|
tagged := getTagFromRef(ref, log)
|
|
|
|
if tagged != nil {
|
|
|
|
if tagReg.MatchString(tagged.Tag()) {
|
2021-12-13 19:23:31 +00:00
|
|
|
refs[numTags] = ref
|
|
|
|
numTags++
|
2021-06-08 23:11:18 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-12-13 19:23:31 +00:00
|
|
|
refs = refs[:numTags]
|
2021-06-08 23:11:18 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
*upstreamReferences = refs
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// filterImagesBySemver filters images by checking if their tags are semver compliant.
|
|
|
|
func filterImagesBySemver(upstreamReferences *[]types.ImageReference, content Content, log log.Logger) {
|
|
|
|
refs := *upstreamReferences
|
|
|
|
|
|
|
|
if content.Tags == nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
if content.Tags.Semver != nil && *content.Tags.Semver {
|
|
|
|
log.Info().Msg("start filtering using semver compliant rule")
|
|
|
|
|
2021-12-13 19:23:31 +00:00
|
|
|
numTags := 0
|
2021-06-08 23:11:18 +03:00
|
|
|
|
|
|
|
for _, ref := range refs {
|
|
|
|
tagged := getTagFromRef(ref, log)
|
|
|
|
if tagged != nil {
|
|
|
|
_, ok := semver.NewVersion(tagged.Tag())
|
|
|
|
if ok == nil {
|
2021-12-13 19:23:31 +00:00
|
|
|
refs[numTags] = ref
|
|
|
|
numTags++
|
2021-06-08 23:11:18 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-12-13 19:23:31 +00:00
|
|
|
refs = refs[:numTags]
|
2021-06-08 23:11:18 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
*upstreamReferences = refs
|
|
|
|
}
|
|
|
|
|
|
|
|
// imagesToCopyFromRepos lists all images given a registry name and its repos.
|
2022-02-10 16:17:49 +02:00
|
|
|
func imagesToCopyFromUpstream(ctx context.Context, registryName string, repos []string,
|
2022-03-21 17:37:23 +00:00
|
|
|
upstreamCtx *types.SystemContext, content Content, log log.Logger,
|
2022-04-05 18:18:31 +03:00
|
|
|
) (map[string][]types.ImageReference, error) {
|
|
|
|
upstreamReferences := make(map[string][]types.ImageReference)
|
2021-06-08 23:11:18 +03:00
|
|
|
|
|
|
|
for _, repoName := range repos {
|
2022-04-05 18:18:31 +03:00
|
|
|
repoUpstreamReferences := make([]types.ImageReference, 0)
|
|
|
|
|
2021-06-08 23:11:18 +03:00
|
|
|
repoRef, err := parseRepositoryReference(fmt.Sprintf("%s/%s", registryName, repoName))
|
|
|
|
if err != nil {
|
2022-04-21 11:09:08 +03:00
|
|
|
log.Error().Str("errorType", TypeOf(err)).
|
|
|
|
Err(err).Msgf("couldn't parse repository reference: %s", repoRef)
|
2021-12-13 19:23:31 +00:00
|
|
|
|
2021-06-08 23:11:18 +03:00
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2022-02-10 16:17:49 +02:00
|
|
|
tags, err := getImageTags(ctx, upstreamCtx, repoRef)
|
2021-06-08 23:11:18 +03:00
|
|
|
if err != nil {
|
2022-04-21 11:09:08 +03:00
|
|
|
log.Error().Str("errorType", TypeOf(err)).
|
|
|
|
Err(err).Msgf("couldn't fetch tags for %s", repoRef)
|
2021-12-13 19:23:31 +00:00
|
|
|
|
2021-06-08 23:11:18 +03:00
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, tag := range tags {
|
2021-12-07 20:26:26 +02:00
|
|
|
// don't copy cosign signature, containers/image doesn't support it
|
|
|
|
// we will copy it manually later
|
|
|
|
if isCosignTag(tag) {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
2021-06-08 23:11:18 +03:00
|
|
|
taggedRef, err := reference.WithTag(repoRef, tag)
|
|
|
|
if err != nil {
|
|
|
|
log.Err(err).Msgf("error creating a reference for repository %s and tag %q", repoRef.Name(), tag)
|
2021-12-13 19:23:31 +00:00
|
|
|
|
2021-06-08 23:11:18 +03:00
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
ref, err := docker.NewReference(taggedRef)
|
|
|
|
if err != nil {
|
|
|
|
log.Err(err).Msgf("cannot obtain a valid image reference for transport %q and reference %s",
|
|
|
|
docker.Transport.Name(), taggedRef.String())
|
2021-12-13 19:23:31 +00:00
|
|
|
|
2021-06-08 23:11:18 +03:00
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2022-04-05 18:18:31 +03:00
|
|
|
repoUpstreamReferences = append(repoUpstreamReferences, ref)
|
2021-06-08 23:11:18 +03:00
|
|
|
}
|
|
|
|
|
2022-04-05 18:18:31 +03:00
|
|
|
upstreamReferences[repoName] = repoUpstreamReferences
|
2021-06-08 23:11:18 +03:00
|
|
|
|
2022-04-05 18:18:31 +03:00
|
|
|
log.Debug().Msgf("repo: %s - upstream refs to be copied: %v", repoName, upstreamReferences)
|
|
|
|
|
|
|
|
err = filterImagesByTagRegex(&repoUpstreamReferences, content, log)
|
|
|
|
if err != nil {
|
|
|
|
return map[string][]types.ImageReference{}, err
|
|
|
|
}
|
2021-06-08 23:11:18 +03:00
|
|
|
|
2022-04-05 18:18:31 +03:00
|
|
|
log.Debug().Msgf("repo: %s - remaining upstream refs to be copied: %v", repoName, repoUpstreamReferences)
|
2022-01-10 18:06:12 +02:00
|
|
|
|
2022-04-05 18:18:31 +03:00
|
|
|
filterImagesBySemver(&repoUpstreamReferences, content, log)
|
2021-06-08 23:11:18 +03:00
|
|
|
|
2022-04-05 18:18:31 +03:00
|
|
|
log.Debug().Msgf("repo: %s - remaining upstream refs to be copied: %v", repoName, repoUpstreamReferences)
|
|
|
|
|
|
|
|
upstreamReferences[repoName] = repoUpstreamReferences
|
|
|
|
}
|
2021-06-08 23:11:18 +03:00
|
|
|
|
|
|
|
return upstreamReferences, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func getCopyOptions(upstreamCtx, localCtx *types.SystemContext) copy.Options {
|
|
|
|
options := copy.Options{
|
2021-12-07 20:26:26 +02:00
|
|
|
DestinationCtx: localCtx,
|
|
|
|
SourceCtx: upstreamCtx,
|
|
|
|
ReportWriter: io.Discard,
|
|
|
|
ForceManifestMIMEType: ispec.MediaTypeImageManifest, // force only oci manifest MIME type
|
2022-05-25 13:24:02 +03:00
|
|
|
PreserveDigests: true,
|
2021-06-08 23:11:18 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
return options
|
|
|
|
}
|
|
|
|
|
|
|
|
func getUpstreamContext(regCfg *RegistryConfig, credentials Credentials) *types.SystemContext {
|
|
|
|
upstreamCtx := &types.SystemContext{}
|
|
|
|
upstreamCtx.DockerCertPath = regCfg.CertDir
|
|
|
|
upstreamCtx.DockerDaemonCertPath = regCfg.CertDir
|
|
|
|
|
|
|
|
if regCfg.TLSVerify != nil && *regCfg.TLSVerify {
|
|
|
|
upstreamCtx.DockerDaemonInsecureSkipTLSVerify = false
|
|
|
|
upstreamCtx.DockerInsecureSkipTLSVerify = types.NewOptionalBool(false)
|
|
|
|
} else {
|
|
|
|
upstreamCtx.DockerDaemonInsecureSkipTLSVerify = true
|
|
|
|
upstreamCtx.DockerInsecureSkipTLSVerify = types.NewOptionalBool(true)
|
|
|
|
}
|
|
|
|
|
|
|
|
if credentials != (Credentials{}) {
|
|
|
|
upstreamCtx.DockerAuthConfig = &types.DockerAuthConfig{
|
|
|
|
Username: credentials.Username,
|
|
|
|
Password: credentials.Password,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return upstreamCtx
|
|
|
|
}
|
|
|
|
|
2022-04-05 18:18:31 +03:00
|
|
|
// nolint:gocyclo // offloading some of the functionalities from here would make the code harder to follow
|
2022-06-24 16:08:47 +03:00
|
|
|
func syncRegistry(ctx context.Context, regCfg RegistryConfig,
|
|
|
|
upstreamURL string,
|
2022-02-10 16:17:49 +02:00
|
|
|
storeController storage.StoreController, localCtx *types.SystemContext,
|
2022-03-07 10:45:10 +02:00
|
|
|
policyCtx *signature.PolicyContext, credentials Credentials,
|
2022-03-21 17:37:23 +00:00
|
|
|
retryOptions *retry.RetryOptions, log log.Logger,
|
|
|
|
) error {
|
2021-12-29 17:14:56 +02:00
|
|
|
log.Info().Msgf("syncing registry: %s", upstreamURL)
|
2021-06-08 23:11:18 +03:00
|
|
|
|
|
|
|
var err error
|
|
|
|
|
|
|
|
log.Debug().Msg("getting upstream context")
|
|
|
|
|
|
|
|
upstreamCtx := getUpstreamContext(®Cfg, credentials)
|
|
|
|
options := getCopyOptions(upstreamCtx, localCtx)
|
|
|
|
|
2022-03-07 10:45:10 +02:00
|
|
|
httpClient, registryURL, err := getHTTPClient(®Cfg, upstreamURL, credentials, log)
|
2021-12-07 20:26:26 +02:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2022-03-07 10:45:10 +02:00
|
|
|
var catalog catalog
|
|
|
|
|
2022-02-10 16:17:49 +02:00
|
|
|
if err = retry.RetryIfNecessary(ctx, func() error {
|
2021-12-29 17:14:56 +02:00
|
|
|
catalog, err = getUpstreamCatalog(httpClient, upstreamURL, log)
|
2021-12-13 19:23:31 +00:00
|
|
|
|
2021-06-08 23:11:18 +03:00
|
|
|
return err
|
|
|
|
}, retryOptions); err != nil {
|
2022-04-21 11:09:08 +03:00
|
|
|
log.Error().Str("errorType", TypeOf(err)).
|
|
|
|
Err(err).Msg("error while getting upstream catalog, retrying...")
|
2021-12-13 19:23:31 +00:00
|
|
|
|
2021-06-08 23:11:18 +03:00
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2021-11-25 14:04:39 +02:00
|
|
|
log.Info().Msgf("filtering %d repos based on sync prefixes", len(catalog.Repositories))
|
2021-06-08 23:11:18 +03:00
|
|
|
|
2021-11-25 14:04:39 +02:00
|
|
|
repos := filterRepos(catalog.Repositories, regCfg.Content, log)
|
2021-06-08 23:11:18 +03:00
|
|
|
|
|
|
|
log.Info().Msgf("got repos: %v", repos)
|
|
|
|
|
2022-04-05 18:18:31 +03:00
|
|
|
upstreamAddr := StripRegistryTransport(upstreamURL)
|
|
|
|
|
|
|
|
reposWithContentID := make(map[string][]struct {
|
2022-03-10 17:39:11 +02:00
|
|
|
ref types.ImageReference
|
|
|
|
content Content
|
2022-04-05 18:18:31 +03:00
|
|
|
})
|
2021-12-29 17:14:56 +02:00
|
|
|
|
2021-06-08 23:11:18 +03:00
|
|
|
for contentID, repos := range repos {
|
|
|
|
r := repos
|
2022-03-10 17:39:11 +02:00
|
|
|
contentID := contentID
|
2021-06-08 23:11:18 +03:00
|
|
|
|
2022-02-10 16:17:49 +02:00
|
|
|
if err = retry.RetryIfNecessary(ctx, func() error {
|
2022-04-05 18:18:31 +03:00
|
|
|
for _, repo := range r {
|
|
|
|
refs, err := imagesToCopyFromUpstream(ctx, upstreamAddr, r, upstreamCtx, regCfg.Content[contentID], log)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, ref := range refs[repo] {
|
|
|
|
reposWithContentID[repo] = append(reposWithContentID[repo], struct {
|
|
|
|
ref types.ImageReference
|
|
|
|
content Content
|
|
|
|
}{
|
|
|
|
ref: ref,
|
|
|
|
content: regCfg.Content[contentID],
|
|
|
|
})
|
|
|
|
}
|
2022-03-10 17:39:11 +02:00
|
|
|
}
|
2021-12-13 19:23:31 +00:00
|
|
|
|
2022-04-05 18:18:31 +03:00
|
|
|
return nil
|
2021-06-08 23:11:18 +03:00
|
|
|
}, retryOptions); err != nil {
|
2022-04-21 11:09:08 +03:00
|
|
|
log.Error().Str("errorType", TypeOf(err)).
|
|
|
|
Err(err).Msg("error while getting images references from upstream, retrying...")
|
2021-12-13 19:23:31 +00:00
|
|
|
|
2021-06-08 23:11:18 +03:00
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-04-05 18:18:31 +03:00
|
|
|
for remoteRepo, imageList := range reposWithContentID {
|
2022-03-07 10:45:10 +02:00
|
|
|
select {
|
|
|
|
case <-ctx.Done():
|
|
|
|
return ctx.Err()
|
|
|
|
default:
|
|
|
|
break
|
|
|
|
}
|
2021-06-08 23:11:18 +03:00
|
|
|
|
2022-04-05 18:18:31 +03:00
|
|
|
remoteRepoCopy := remoteRepo
|
|
|
|
imageStore := storeController.GetImageStore(remoteRepoCopy)
|
2021-06-08 23:11:18 +03:00
|
|
|
|
2022-04-05 18:18:31 +03:00
|
|
|
localCachePath, err := getLocalCachePath(imageStore, remoteRepoCopy)
|
2022-03-07 10:45:10 +02:00
|
|
|
if err != nil {
|
2022-04-21 11:09:08 +03:00
|
|
|
log.Error().Str("errorType", TypeOf(err)).
|
|
|
|
Err(err).Msgf("couldn't get localCachePath for %s", remoteRepoCopy)
|
2022-03-07 10:45:10 +02:00
|
|
|
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2022-04-05 18:18:31 +03:00
|
|
|
if localCachePath != "" {
|
|
|
|
defer os.RemoveAll(localCachePath)
|
2022-03-07 10:45:10 +02:00
|
|
|
}
|
|
|
|
|
2022-04-05 18:18:31 +03:00
|
|
|
for _, image := range imageList {
|
|
|
|
localRepo := remoteRepoCopy
|
|
|
|
upstreamImageRef := image.ref
|
2022-03-07 10:45:10 +02:00
|
|
|
|
2022-04-05 18:18:31 +03:00
|
|
|
upstreamImageDigest, err := docker.GetDigest(ctx, upstreamCtx, upstreamImageRef)
|
|
|
|
if err != nil {
|
|
|
|
log.Error().Err(err).Msgf("couldn't get upstream image %s manifest", upstreamImageRef.DockerReference())
|
2022-03-07 10:45:10 +02:00
|
|
|
|
2022-04-05 18:18:31 +03:00
|
|
|
return err
|
2022-03-07 10:45:10 +02:00
|
|
|
}
|
|
|
|
|
2022-04-05 18:18:31 +03:00
|
|
|
tag := getTagFromRef(upstreamImageRef, log).Tag()
|
|
|
|
// get upstream signatures
|
|
|
|
cosignManifest, err := getCosignManifest(httpClient, *registryURL, remoteRepoCopy,
|
|
|
|
upstreamImageDigest.String(), log)
|
|
|
|
if err != nil && !errors.Is(err, zerr.ErrSyncSignatureNotFound) {
|
|
|
|
log.Error().Err(err).Msgf("couldn't get upstream image %s cosign manifest", upstreamImageRef.DockerReference())
|
2022-03-07 10:45:10 +02:00
|
|
|
|
2022-04-05 18:18:31 +03:00
|
|
|
return err
|
|
|
|
}
|
2021-12-17 18:34:22 +02:00
|
|
|
|
2022-04-05 18:18:31 +03:00
|
|
|
refs, err := getNotaryRefs(httpClient, *registryURL, remoteRepoCopy, upstreamImageDigest.String(), log)
|
|
|
|
if err != nil && !errors.Is(err, zerr.ErrSyncSignatureNotFound) {
|
|
|
|
log.Error().Err(err).Msgf("couldn't get upstream image %s notary references", upstreamImageRef.DockerReference())
|
2022-03-07 10:45:10 +02:00
|
|
|
|
2022-04-05 18:18:31 +03:00
|
|
|
return err
|
2022-03-07 10:45:10 +02:00
|
|
|
}
|
|
|
|
|
2022-04-05 18:18:31 +03:00
|
|
|
// check if upstream image is signed
|
|
|
|
if cosignManifest == nil && len(refs.References) == 0 {
|
|
|
|
// upstream image not signed
|
|
|
|
if regCfg.OnlySigned != nil && *regCfg.OnlySigned {
|
|
|
|
// skip unsigned images
|
|
|
|
log.Info().Msgf("skipping image without signature %s", upstreamImageRef.DockerReference())
|
2022-03-07 10:45:10 +02:00
|
|
|
|
2022-04-05 18:18:31 +03:00
|
|
|
continue
|
2022-03-07 10:45:10 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-04-05 18:18:31 +03:00
|
|
|
skipImage, err := canSkipImage(localRepo, tag, upstreamImageDigest.String(), imageStore, log)
|
2022-03-07 10:45:10 +02:00
|
|
|
if err != nil {
|
2022-04-05 18:18:31 +03:00
|
|
|
log.Error().Err(err).Msgf("couldn't check if the upstream image %s can be skipped",
|
2022-03-07 10:45:10 +02:00
|
|
|
upstreamImageRef.DockerReference())
|
2022-04-05 18:18:31 +03:00
|
|
|
|
|
|
|
return err
|
2022-03-07 10:45:10 +02:00
|
|
|
}
|
|
|
|
|
2022-04-05 18:18:31 +03:00
|
|
|
// sync only differences
|
|
|
|
if skipImage {
|
|
|
|
log.Info().Msgf("already synced image %s, checking its signatures", upstreamImageRef.DockerReference())
|
2022-03-07 10:45:10 +02:00
|
|
|
|
2022-04-05 18:18:31 +03:00
|
|
|
skipNotarySig, err := canSkipNotarySignature(localRepo, tag, upstreamImageDigest.String(),
|
|
|
|
refs, imageStore, log)
|
|
|
|
if err != nil {
|
|
|
|
log.Error().Err(err).Msgf("couldn't check if the upstream image %s notary signature can be skipped",
|
|
|
|
upstreamImageRef.DockerReference())
|
2022-03-07 10:45:10 +02:00
|
|
|
}
|
|
|
|
|
2022-04-05 18:18:31 +03:00
|
|
|
if !skipNotarySig {
|
|
|
|
if err = retry.RetryIfNecessary(ctx, func() error {
|
|
|
|
err = syncNotarySignature(httpClient, imageStore, *registryURL, localRepo, remoteRepoCopy,
|
|
|
|
upstreamImageDigest.String(), refs, log)
|
2021-12-17 18:34:22 +02:00
|
|
|
|
2022-04-05 18:18:31 +03:00
|
|
|
return err
|
|
|
|
}, retryOptions); err != nil {
|
|
|
|
log.Error().Err(err).Msgf("couldn't copy notary signature for %s", upstreamImageRef.DockerReference())
|
|
|
|
}
|
|
|
|
}
|
2021-12-13 19:23:31 +00:00
|
|
|
|
2022-04-05 18:18:31 +03:00
|
|
|
skipCosignSig, err := canSkipCosignSignature(localRepo, tag, upstreamImageDigest.String(),
|
|
|
|
cosignManifest, imageStore, log)
|
|
|
|
if err != nil {
|
|
|
|
log.Error().Err(err).Msgf("couldn't check if the upstream image %s cosign signature can be skipped",
|
|
|
|
upstreamImageRef.DockerReference())
|
|
|
|
}
|
2021-06-08 23:11:18 +03:00
|
|
|
|
2022-04-05 18:18:31 +03:00
|
|
|
if !skipCosignSig {
|
|
|
|
if err = retry.RetryIfNecessary(ctx, func() error {
|
|
|
|
err = syncCosignSignature(httpClient, imageStore, *registryURL, localRepo, remoteRepoCopy,
|
|
|
|
upstreamImageDigest.String(), cosignManifest, log)
|
2022-02-18 13:36:50 +02:00
|
|
|
|
2022-04-05 18:18:31 +03:00
|
|
|
return err
|
|
|
|
}, retryOptions); err != nil {
|
|
|
|
log.Error().Err(err).Msgf("couldn't copy cosign signature for %s", upstreamImageRef.DockerReference())
|
|
|
|
}
|
|
|
|
}
|
2021-06-08 23:11:18 +03:00
|
|
|
|
2022-04-05 18:18:31 +03:00
|
|
|
continue
|
|
|
|
}
|
2021-12-13 19:23:31 +00:00
|
|
|
|
2022-04-05 18:18:31 +03:00
|
|
|
localImageRef, err := getLocalImageRef(localCachePath, localRepo, tag)
|
|
|
|
if err != nil {
|
2022-04-21 11:09:08 +03:00
|
|
|
log.Error().Str("errorType", TypeOf(err)).
|
|
|
|
Err(err).Msgf("couldn't obtain a valid image reference for reference %s/%s:%s",
|
2022-04-05 18:18:31 +03:00
|
|
|
localCachePath, localRepo, tag)
|
2021-12-13 19:23:31 +00:00
|
|
|
|
2022-04-05 18:18:31 +03:00
|
|
|
return err
|
|
|
|
}
|
2021-10-28 12:10:01 +03:00
|
|
|
|
2022-04-05 18:18:31 +03:00
|
|
|
log.Info().Msgf("copying image %s to %s", upstreamImageRef.DockerReference(), localCachePath)
|
2021-12-13 19:23:31 +00:00
|
|
|
|
2022-04-05 18:18:31 +03:00
|
|
|
if err = retry.RetryIfNecessary(ctx, func() error {
|
|
|
|
_, err = copy.Image(ctx, policyCtx, localImageRef, upstreamImageRef, &options)
|
2021-12-07 20:26:26 +02:00
|
|
|
|
2022-04-05 18:18:31 +03:00
|
|
|
return err
|
|
|
|
}, retryOptions); err != nil {
|
2022-04-21 11:09:08 +03:00
|
|
|
log.Error().Str("errorType", TypeOf(err)).
|
|
|
|
Err(err).Msgf("error while copying image %s to %s",
|
2022-04-05 18:18:31 +03:00
|
|
|
upstreamImageRef.DockerReference(), localCachePath)
|
2022-03-07 10:45:10 +02:00
|
|
|
|
2022-04-05 18:18:31 +03:00
|
|
|
return err
|
|
|
|
}
|
|
|
|
// push from cache to repo
|
|
|
|
err = pushSyncedLocalImage(localRepo, tag, localCachePath, imageStore, log)
|
|
|
|
if err != nil {
|
2022-04-21 11:09:08 +03:00
|
|
|
log.Error().Str("errorType", TypeOf(err)).
|
|
|
|
Err(err).Msgf("error while pushing synced cached image %s",
|
2022-04-05 18:18:31 +03:00
|
|
|
fmt.Sprintf("%s/%s:%s", localCachePath, localRepo, tag))
|
2021-12-07 20:26:26 +02:00
|
|
|
|
2022-04-05 18:18:31 +03:00
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
refs, err = getNotaryRefs(httpClient, *registryURL, remoteRepoCopy, upstreamImageDigest.String(), log)
|
|
|
|
if err = retry.RetryIfNecessary(ctx, func() error {
|
|
|
|
err = syncNotarySignature(httpClient, imageStore, *registryURL, localRepo,
|
|
|
|
remoteRepoCopy, upstreamImageDigest.String(), refs, log)
|
|
|
|
|
|
|
|
return err
|
|
|
|
}, retryOptions); err != nil {
|
2022-04-21 11:09:08 +03:00
|
|
|
log.Error().Str("errorType", TypeOf(err)).
|
|
|
|
Err(err).Msgf("couldn't copy notary signature for %s", upstreamImageRef.DockerReference())
|
2022-04-05 18:18:31 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
cosignManifest, err = getCosignManifest(httpClient, *registryURL, remoteRepoCopy,
|
|
|
|
upstreamImageDigest.String(), log)
|
|
|
|
if err = retry.RetryIfNecessary(ctx, func() error {
|
|
|
|
err = syncCosignSignature(httpClient, imageStore, *registryURL, localRepo,
|
|
|
|
remoteRepoCopy, upstreamImageDigest.String(), cosignManifest, log)
|
|
|
|
|
|
|
|
return err
|
|
|
|
}, retryOptions); err != nil {
|
2022-04-21 11:09:08 +03:00
|
|
|
log.Error().Str("errorType", TypeOf(err)).
|
|
|
|
Err(err).Msgf("couldn't copy cosign signature for %s", upstreamImageRef.DockerReference())
|
2022-04-05 18:18:31 +03:00
|
|
|
}
|
2021-12-07 20:26:26 +02:00
|
|
|
}
|
2021-06-08 23:11:18 +03:00
|
|
|
}
|
|
|
|
|
2021-12-29 17:14:56 +02:00
|
|
|
log.Info().Msgf("finished syncing %s", upstreamAddr)
|
2021-06-08 23:11:18 +03:00
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2021-10-28 12:10:01 +03:00
|
|
|
func getLocalContexts(log log.Logger) (*types.SystemContext, *signature.PolicyContext, error) {
|
2021-06-08 23:11:18 +03:00
|
|
|
log.Debug().Msg("getting local context")
|
|
|
|
|
|
|
|
var policy *signature.Policy
|
|
|
|
|
|
|
|
var err error
|
|
|
|
|
|
|
|
localCtx := &types.SystemContext{}
|
2021-12-07 20:26:26 +02:00
|
|
|
// preserve compression
|
|
|
|
localCtx.OCIAcceptUncompressedLayers = true
|
2021-06-08 23:11:18 +03:00
|
|
|
|
2021-10-28 12:10:01 +03:00
|
|
|
// accept any image with or without signature
|
|
|
|
policy = &signature.Policy{Default: []signature.PolicyRequirement{signature.NewPRInsecureAcceptAnything()}}
|
2021-06-08 23:11:18 +03:00
|
|
|
|
|
|
|
policyContext, err := signature.NewPolicyContext(policy)
|
2022-02-18 13:36:50 +02:00
|
|
|
if err := test.Error(err); err != nil {
|
2022-04-21 11:09:08 +03:00
|
|
|
log.Error().Str("errorType", TypeOf(err)).
|
|
|
|
Err(err).Msg("couldn't create policy context")
|
2021-12-13 19:23:31 +00:00
|
|
|
|
2021-06-08 23:11:18 +03:00
|
|
|
return &types.SystemContext{}, &signature.PolicyContext{}, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return localCtx, policyContext, nil
|
|
|
|
}
|
|
|
|
|
2022-06-24 16:08:47 +03:00
|
|
|
func Run(ctx context.Context, cfg Config,
|
|
|
|
storeController storage.StoreController,
|
2022-03-21 17:37:23 +00:00
|
|
|
wtgrp *goSync.WaitGroup, logger log.Logger,
|
|
|
|
) error {
|
2021-06-08 23:11:18 +03:00
|
|
|
var credentialsFile CredentialsFile
|
|
|
|
|
2021-10-28 12:10:01 +03:00
|
|
|
var err error
|
|
|
|
|
2021-06-08 23:11:18 +03:00
|
|
|
if cfg.CredentialsFile != "" {
|
|
|
|
credentialsFile, err = getFileCredentials(cfg.CredentialsFile)
|
|
|
|
if err != nil {
|
2022-04-21 11:09:08 +03:00
|
|
|
logger.Error().Str("errortype", TypeOf(err)).
|
|
|
|
Err(err).Msgf("couldn't get registry credentials from %s", cfg.CredentialsFile)
|
2021-12-13 19:23:31 +00:00
|
|
|
|
2021-06-08 23:11:18 +03:00
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-10-28 12:10:01 +03:00
|
|
|
localCtx, policyCtx, err := getLocalContexts(logger)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
// for each upstream registry, start a go routine.
|
2021-06-08 23:11:18 +03:00
|
|
|
for _, regCfg := range cfg.Registries {
|
2021-10-25 15:05:03 +03:00
|
|
|
// if content not provided, don't run periodically sync
|
2021-11-25 14:04:39 +02:00
|
|
|
if len(regCfg.Content) == 0 {
|
2021-12-29 17:14:56 +02:00
|
|
|
logger.Info().Msgf("sync config content not configured for %v, will not run periodically sync", regCfg.URLs)
|
2021-12-13 19:23:31 +00:00
|
|
|
|
2021-10-25 15:05:03 +03:00
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
// if pollInterval is not provided, don't run periodically sync
|
|
|
|
if regCfg.PollInterval == 0 {
|
2021-12-29 17:14:56 +02:00
|
|
|
logger.Warn().Msgf("sync config PollInterval not configured for %v, will not run periodically sync", regCfg.URLs)
|
2021-12-13 19:23:31 +00:00
|
|
|
|
2021-11-25 14:04:39 +02:00
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
2021-10-22 12:56:55 +03:00
|
|
|
ticker := time.NewTicker(regCfg.PollInterval)
|
2021-06-08 23:11:18 +03:00
|
|
|
|
2021-10-28 12:10:01 +03:00
|
|
|
// fork a new zerolog child to avoid data race
|
2022-03-07 10:45:10 +02:00
|
|
|
tlogger := log.Logger{Logger: logger.Logger}
|
|
|
|
|
|
|
|
retryOptions := &retry.RetryOptions{}
|
|
|
|
|
|
|
|
if regCfg.MaxRetries != nil {
|
|
|
|
retryOptions.MaxRetry = *regCfg.MaxRetries
|
|
|
|
if regCfg.RetryDelay != nil {
|
|
|
|
retryOptions.Delay = *regCfg.RetryDelay
|
|
|
|
}
|
|
|
|
}
|
2021-06-08 23:11:18 +03:00
|
|
|
|
2021-10-25 15:05:03 +03:00
|
|
|
// schedule each registry sync
|
2022-02-10 16:17:49 +02:00
|
|
|
go func(ctx context.Context, regCfg RegistryConfig, logger log.Logger) {
|
|
|
|
for {
|
2021-12-17 18:37:02 +02:00
|
|
|
// increment reference since will be busy, so shutdown has to wait
|
2021-12-13 19:23:31 +00:00
|
|
|
wtgrp.Add(1)
|
2021-12-17 18:37:02 +02:00
|
|
|
|
2021-12-29 17:14:56 +02:00
|
|
|
for _, upstreamURL := range regCfg.URLs {
|
|
|
|
upstreamAddr := StripRegistryTransport(upstreamURL)
|
|
|
|
// first try syncing main registry
|
2022-02-10 16:17:49 +02:00
|
|
|
if err := syncRegistry(ctx, regCfg, upstreamURL, storeController, localCtx, policyCtx,
|
2022-03-07 10:45:10 +02:00
|
|
|
credentialsFile[upstreamAddr], retryOptions, logger); err != nil {
|
2022-04-21 11:09:08 +03:00
|
|
|
logger.Error().Str("errortype", TypeOf(err)).
|
|
|
|
Err(err).Str("registry", upstreamURL).
|
2022-02-10 16:17:49 +02:00
|
|
|
Msg("sync exited with error, falling back to auxiliary registries if any")
|
2021-12-29 17:14:56 +02:00
|
|
|
} else {
|
|
|
|
// if success fall back to main registry
|
|
|
|
break
|
|
|
|
}
|
2021-06-08 23:11:18 +03:00
|
|
|
}
|
2021-12-02 19:45:26 +02:00
|
|
|
// mark as done after a single sync run
|
2021-12-13 19:23:31 +00:00
|
|
|
wtgrp.Done()
|
2022-02-10 16:17:49 +02:00
|
|
|
|
|
|
|
select {
|
|
|
|
case <-ctx.Done():
|
|
|
|
ticker.Stop()
|
|
|
|
|
|
|
|
return
|
|
|
|
case <-ticker.C:
|
|
|
|
// run on intervals
|
|
|
|
continue
|
|
|
|
}
|
2021-06-08 23:11:18 +03:00
|
|
|
}
|
2022-02-10 16:17:49 +02:00
|
|
|
}(ctx, regCfg, tlogger)
|
2021-06-08 23:11:18 +03:00
|
|
|
}
|
|
|
|
|
2021-10-28 12:10:01 +03:00
|
|
|
logger.Info().Msg("finished setting up sync")
|
2021-06-08 23:11:18 +03:00
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|