mirror of
https://github.com/project-zot/zot.git
synced 2026-06-18 05:28:07 +08:00
79439bbf63
Add support for configurable identity attributes in mTLS authentication, allowing identity extraction from CommonName, Subject DN, Email SAN, URI SAN, or DNSName SAN with fallback chain support. Includes regex pattern matching for URI SANs (e.g., SPIFFE workload IDs). - Add MTLSConfig with identity attributes, URISANPattern, and index fields - Implement extractMTLSIdentity with fallback chain logic - Move the mtls tests in the api package to pkg/api/mtls_test.go Signed-off-by: Andrei Aaron <andreifdaaron@gmail.com>
1099 lines
27 KiB
Go
1099 lines
27 KiB
Go
package config
|
|
|
|
import (
|
|
"encoding/json"
|
|
"maps"
|
|
"os"
|
|
"sync"
|
|
"time"
|
|
|
|
distspec "github.com/opencontainers/distribution-spec/specs-go"
|
|
"github.com/tiendc/go-deepcopy"
|
|
|
|
"zotregistry.dev/zot/v2/pkg/compat"
|
|
extconf "zotregistry.dev/zot/v2/pkg/extensions/config"
|
|
storageConstants "zotregistry.dev/zot/v2/pkg/storage/constants"
|
|
)
|
|
|
|
var (
|
|
Commit string //nolint: gochecknoglobals
|
|
ReleaseTag string //nolint: gochecknoglobals
|
|
BinaryType string //nolint: gochecknoglobals
|
|
GoVersion string //nolint: gochecknoglobals
|
|
|
|
openIDSupportedProviders = [...]string{"google", "gitlab", "oidc"} //nolint: gochecknoglobals
|
|
oauth2SupportedProviders = [...]string{"github"} //nolint: gochecknoglobals
|
|
|
|
)
|
|
|
|
type StorageConfig struct {
|
|
RootDirectory string
|
|
Dedupe bool
|
|
RemoteCache bool
|
|
GC bool
|
|
Commit bool
|
|
GCDelay time.Duration // applied for blobs
|
|
GCInterval time.Duration
|
|
Retention ImageRetention
|
|
StorageDriver map[string]any `mapstructure:",omitempty"`
|
|
CacheDriver map[string]any `mapstructure:",omitempty"`
|
|
|
|
// GCMaxSchedulerDelay is the maximum random delay for GC task scheduling
|
|
// This field is not configurable by the end user
|
|
GCMaxSchedulerDelay time.Duration `yaml:"-"`
|
|
}
|
|
|
|
type ImageRetention struct {
|
|
DryRun bool
|
|
Delay time.Duration // applied for referrers and untagged
|
|
Policies []RetentionPolicy
|
|
}
|
|
|
|
type RetentionPolicy struct {
|
|
Repositories []string
|
|
DeleteReferrers bool
|
|
DeleteUntagged *bool
|
|
KeepTags []KeepTagsPolicy
|
|
}
|
|
|
|
type KeepTagsPolicy struct {
|
|
Patterns []string
|
|
PulledWithin *time.Duration
|
|
PushedWithin *time.Duration
|
|
MostRecentlyPushedCount int
|
|
MostRecentlyPulledCount int
|
|
}
|
|
|
|
type TLSConfig struct {
|
|
Cert string
|
|
Key string
|
|
CACert string
|
|
}
|
|
|
|
type MTLSConfig struct {
|
|
// IdentityAttibutes is an ordered list of identity attributes to try
|
|
// Options: "CommonName", "Subject", "Email", "URI", "DNSName" (case-insensitive)
|
|
// Default: ["CommonName"] (backward compatible)
|
|
IdentityAttibutes []string `json:"identityAttributes,omitempty" mapstructure:"identityAttributes,omitempty"`
|
|
|
|
// URISANPattern is a regex pattern to extract identity from URI SAN
|
|
// Only used when IdentityAttibutes contains "URI"
|
|
// Example: "spiffe://example.org/workload/(.*)" extracts the workload ID
|
|
// If empty, uses the full URI SAN value
|
|
URISANPattern string `json:"uriSanPattern,omitempty" mapstructure:"uriSanPattern,omitempty"`
|
|
|
|
// URISANIndex specifies which URI SAN to use if multiple exist (0-based)
|
|
// Maps to cert.URIs[index] - the URIs field is a slice, so index is needed
|
|
// Default: 0 (first URI)
|
|
URISANIndex int `json:"uriSanIndex,omitempty" mapstructure:"uriSanIndex,omitempty"`
|
|
|
|
// DNSANIndex specifies which DNS SAN to use if multiple exist (0-based)
|
|
// Maps to cert.DNSNames[index] - the DNSNames field is a slice, so index is needed
|
|
// Default: 0 (first DNS name)
|
|
DNSANIndex int `json:"dnsSanIndex,omitempty" mapstructure:"dnsSanIndex,omitempty"`
|
|
|
|
// EmailSANIndex specifies which Email SAN to use if multiple exist (0-based)
|
|
// Maps to cert.EmailAddresses[index] - the EmailAddresses field is a slice, so index is needed
|
|
// Default: 0 (first email)
|
|
EmailSANIndex int `json:"emailSanIndex,omitempty" mapstructure:"emailSanIndex,omitempty"`
|
|
}
|
|
|
|
type AuthHTPasswd struct {
|
|
Path string
|
|
}
|
|
|
|
type AuthConfig struct {
|
|
FailDelay int
|
|
HTPasswd AuthHTPasswd
|
|
LDAP *LDAPConfig
|
|
Bearer *BearerConfig
|
|
OpenID *OpenIDConfig
|
|
APIKey bool
|
|
SessionKeysFile string
|
|
SessionHashKey []byte `json:"-"`
|
|
SessionEncryptKey []byte `json:"-"`
|
|
SessionDriver map[string]any `mapstructure:",omitempty"`
|
|
SecureSession *bool `json:"secureSession,omitempty" mapstructure:"secureSession,omitempty"`
|
|
MTLS *MTLSConfig `json:"mtls,omitempty" mapstructure:"mtls,omitempty"`
|
|
}
|
|
|
|
// IsLdapAuthEnabled checks if LDAP authentication is enabled in this auth config.
|
|
func (a *AuthConfig) IsLdapAuthEnabled() bool {
|
|
return a != nil && a.LDAP != nil
|
|
}
|
|
|
|
// IsHtpasswdAuthEnabled checks if HTPasswd authentication is enabled in this auth config.
|
|
func (a *AuthConfig) IsHtpasswdAuthEnabled() bool {
|
|
return a != nil && a.HTPasswd.Path != ""
|
|
}
|
|
|
|
// IsBearerAuthEnabled checks if Bearer authentication is enabled in this auth config.
|
|
func (a *AuthConfig) IsBearerAuthEnabled() bool {
|
|
return a != nil && a.Bearer != nil && a.Bearer.Cert != "" && a.Bearer.Realm != "" && a.Bearer.Service != ""
|
|
}
|
|
|
|
// IsOpenIDAuthEnabled checks if OpenID authentication is enabled in this auth config.
|
|
func (a *AuthConfig) IsOpenIDAuthEnabled() bool {
|
|
if a == nil || a.OpenID == nil {
|
|
return false
|
|
}
|
|
|
|
for provider := range a.OpenID.Providers {
|
|
if IsOpenIDSupported(provider) || IsOauth2Supported(provider) {
|
|
return true
|
|
}
|
|
}
|
|
|
|
return false
|
|
}
|
|
|
|
// IsAPIKeyEnabled checks if API Key authentication is enabled in this auth config.
|
|
func (a *AuthConfig) IsAPIKeyEnabled() bool {
|
|
return a != nil && a.APIKey
|
|
}
|
|
|
|
// IsBasicAuthnEnabled checks if any basic authentication method is enabled in this auth config.
|
|
func (a *AuthConfig) IsBasicAuthnEnabled() bool {
|
|
if a == nil {
|
|
return false
|
|
}
|
|
|
|
return a.IsHtpasswdAuthEnabled() || a.IsLdapAuthEnabled() || a.IsOpenIDAuthEnabled() || a.IsAPIKeyEnabled()
|
|
}
|
|
|
|
// GetFailDelay returns the configured fail delay for authentication attempts.
|
|
func (a *AuthConfig) GetFailDelay() int {
|
|
if a == nil {
|
|
return 0
|
|
}
|
|
|
|
return a.FailDelay
|
|
}
|
|
|
|
// GetMTLSConfig returns the mTLS configuration if it exists.
|
|
func (a *AuthConfig) GetMTLSConfig() *MTLSConfig {
|
|
if a == nil {
|
|
return nil
|
|
}
|
|
|
|
return a.MTLS
|
|
}
|
|
|
|
type BearerConfig struct {
|
|
Realm string
|
|
Service string
|
|
Cert string
|
|
}
|
|
|
|
type SessionKeys struct {
|
|
HashKey string
|
|
EncryptKey string `mapstructure:",omitempty"`
|
|
}
|
|
|
|
type OpenIDConfig struct {
|
|
Providers map[string]OpenIDProviderConfig
|
|
}
|
|
|
|
type OpenIDCredentials struct {
|
|
ClientID string
|
|
ClientSecret string
|
|
}
|
|
|
|
type OpenIDProviderConfig struct {
|
|
CredentialsFile string
|
|
Name string
|
|
ClientID string
|
|
ClientSecret string
|
|
KeyPath string
|
|
Issuer string
|
|
AuthURL string
|
|
TokenURL string
|
|
Scopes []string
|
|
ClaimMapping *ClaimMapping `mapstructure:",omitempty"`
|
|
}
|
|
|
|
// ClaimMapping specifies how OpenID claims are mapped to application fields.
|
|
// It allows customization of which claim is used as the username when authenticating users.
|
|
type ClaimMapping struct {
|
|
// Username specifies which OpenID claim to use as the username for the authenticated user.
|
|
// Acceptable values include "preferred_username", "email", "sub", "name", or any custom claim name.
|
|
// If not configured, the default is "email".
|
|
Username string `mapstructure:"username,omitempty"`
|
|
}
|
|
|
|
type MethodRatelimitConfig struct {
|
|
Method string
|
|
Rate int
|
|
}
|
|
|
|
type RatelimitConfig struct {
|
|
Rate *int // requests per second
|
|
Methods []MethodRatelimitConfig `mapstructure:",omitempty"`
|
|
}
|
|
|
|
//nolint:maligned
|
|
type HTTPConfig struct {
|
|
Address string
|
|
ExternalURL string `mapstructure:",omitempty"`
|
|
Port string
|
|
AllowOrigin string // comma separated
|
|
TLS *TLSConfig
|
|
Auth *AuthConfig
|
|
AccessControl *AccessControlConfig `mapstructure:"accessControl,omitempty"`
|
|
Realm string
|
|
Ratelimit *RatelimitConfig `mapstructure:",omitempty"`
|
|
Compat []compat.MediaCompatibility `mapstructure:",omitempty"`
|
|
}
|
|
|
|
type SchedulerConfig struct {
|
|
NumWorkers int
|
|
}
|
|
|
|
// ClusterConfig contains the scale-out configuration which is identical for all zot replicas.
|
|
type ClusterConfig struct {
|
|
// contains the "host:port" of all the zot instances participating
|
|
// in the cluster.
|
|
Members []string `json:"members" mapstructure:"members"`
|
|
|
|
// contains the hash key that is required for siphash.
|
|
// must be a 128-bit (16-byte) key
|
|
// https://github.com/dchest/siphash?tab=readme-ov-file#func-newkey-byte-hashhash64
|
|
HashKey string `json:"hashKey" mapstructure:"hashKey"`
|
|
|
|
// contains client TLS config.
|
|
TLS *TLSConfig `json:"tls" mapstructure:"tls"`
|
|
|
|
// private field for storing Proxy details such as internal socket list.
|
|
Proxy *ClusterRequestProxyConfig `json:"-" mapstructure:"-"`
|
|
}
|
|
|
|
// IsClustered returns true if the cluster configuration represents a multi-node cluster.
|
|
func (c *ClusterConfig) IsClustered() bool {
|
|
return c != nil && len(c.Members) > 1
|
|
}
|
|
|
|
type ClusterRequestProxyConfig struct {
|
|
// holds the cluster socket (IP:port) derived from the host's
|
|
// interface configuration and the listening port of the HTTP server.
|
|
LocalMemberClusterSocket string
|
|
// index of the local member cluster socket in the members array.
|
|
LocalMemberClusterSocketIndex uint64
|
|
}
|
|
|
|
type LDAPCredentials struct {
|
|
BindDN string
|
|
BindPassword string
|
|
}
|
|
|
|
type LDAPConfig struct {
|
|
CredentialsFile string
|
|
Port int
|
|
Insecure bool
|
|
StartTLS bool // if !Insecure, then StartTLS or LDAPs
|
|
SkipVerify bool
|
|
SubtreeSearch bool
|
|
Address string
|
|
bindDN string `json:"-"`
|
|
bindPassword string `json:"-"`
|
|
UserGroupAttribute string
|
|
BaseDN string
|
|
UserAttribute string
|
|
UserFilter string
|
|
CACert string
|
|
}
|
|
|
|
func (ldapConf *LDAPConfig) BindDN() string {
|
|
if ldapConf == nil {
|
|
return ""
|
|
}
|
|
|
|
return ldapConf.bindDN
|
|
}
|
|
|
|
func (ldapConf *LDAPConfig) SetBindDN(bindDN string) *LDAPConfig {
|
|
if ldapConf == nil {
|
|
return nil
|
|
}
|
|
ldapConf.bindDN = bindDN
|
|
|
|
return ldapConf
|
|
}
|
|
|
|
func (ldapConf *LDAPConfig) BindPassword() string {
|
|
if ldapConf == nil {
|
|
return ""
|
|
}
|
|
|
|
return ldapConf.bindPassword
|
|
}
|
|
|
|
func (ldapConf *LDAPConfig) SetBindPassword(bindPassword string) *LDAPConfig {
|
|
if ldapConf == nil {
|
|
return nil
|
|
}
|
|
ldapConf.bindPassword = bindPassword
|
|
|
|
return ldapConf
|
|
}
|
|
|
|
type LogConfig struct {
|
|
Level string
|
|
Output string
|
|
Audit string
|
|
}
|
|
|
|
type GlobalStorageConfig struct {
|
|
StorageConfig `mapstructure:",squash"`
|
|
|
|
SubPaths map[string]StorageConfig
|
|
}
|
|
|
|
type AccessControlConfig struct {
|
|
Repositories Repositories `json:"repositories" mapstructure:"repositories"`
|
|
AdminPolicy Policy
|
|
Groups Groups
|
|
Metrics Metrics
|
|
}
|
|
|
|
// IsAuthzEnabled checks if authorization is enabled (access control is configured).
|
|
func (config *AccessControlConfig) IsAuthzEnabled() bool {
|
|
return config != nil
|
|
}
|
|
|
|
func (config *AccessControlConfig) AnonymousPolicyExists() bool {
|
|
if config == nil {
|
|
return false
|
|
}
|
|
|
|
for _, repository := range config.Repositories {
|
|
if len(repository.AnonymousPolicy) > 0 {
|
|
return true
|
|
}
|
|
}
|
|
|
|
return false
|
|
}
|
|
|
|
// ContainsOnlyAnonymousPolicy checks if the access control configuration contains only anonymous policies.
|
|
func (config *AccessControlConfig) ContainsOnlyAnonymousPolicy() bool {
|
|
if config == nil {
|
|
return true
|
|
}
|
|
|
|
// Check if admin policy has any actions or users
|
|
if len(config.AdminPolicy.Actions)+len(config.AdminPolicy.Users) > 0 {
|
|
return false
|
|
}
|
|
|
|
anonymousPolicyPresent := false
|
|
|
|
for _, repository := range config.Repositories {
|
|
// Check if repository has default policy
|
|
if len(repository.DefaultPolicy) > 0 {
|
|
return false
|
|
}
|
|
|
|
// Check if repository has anonymous policy
|
|
if len(repository.AnonymousPolicy) > 0 {
|
|
anonymousPolicyPresent = true
|
|
}
|
|
|
|
// Check if repository has any non-empty policies
|
|
for _, policy := range repository.Policies {
|
|
if len(policy.Actions)+len(policy.Users) > 0 {
|
|
return false
|
|
}
|
|
}
|
|
}
|
|
|
|
return anonymousPolicyPresent
|
|
}
|
|
|
|
// GetRepositories safely gets a copy of the repositories configuration.
|
|
func (config *AccessControlConfig) GetRepositories() Repositories {
|
|
if config == nil {
|
|
return nil
|
|
}
|
|
|
|
// Return a copy to avoid race conditions
|
|
reposCopy := make(Repositories)
|
|
maps.Copy(reposCopy, config.Repositories)
|
|
|
|
return reposCopy
|
|
}
|
|
|
|
// GetAdminPolicy safely gets a copy of the admin policy.
|
|
func (config *AccessControlConfig) GetAdminPolicy() Policy {
|
|
if config == nil {
|
|
return Policy{}
|
|
}
|
|
|
|
return config.AdminPolicy
|
|
}
|
|
|
|
// GetMetrics safely gets a copy of the metrics configuration.
|
|
func (config *AccessControlConfig) GetMetrics() Metrics {
|
|
if config == nil {
|
|
return Metrics{}
|
|
}
|
|
|
|
return config.Metrics
|
|
}
|
|
|
|
// GetGroups safely gets a copy of the groups configuration.
|
|
func (config *AccessControlConfig) GetGroups() Groups {
|
|
if config == nil {
|
|
return nil
|
|
}
|
|
|
|
// Return a copy to avoid race conditions
|
|
groupsCopy := make(Groups)
|
|
maps.Copy(groupsCopy, config.Groups)
|
|
|
|
return groupsCopy
|
|
}
|
|
|
|
type (
|
|
Repositories map[string]PolicyGroup
|
|
Groups map[string]Group
|
|
)
|
|
|
|
type Group struct {
|
|
Users []string
|
|
}
|
|
|
|
type PolicyGroup struct {
|
|
Policies []Policy
|
|
DefaultPolicy []string
|
|
AnonymousPolicy []string
|
|
}
|
|
|
|
type Policy struct {
|
|
Users []string
|
|
Actions []string
|
|
Groups []string
|
|
}
|
|
|
|
type Metrics struct {
|
|
Users []string
|
|
}
|
|
|
|
type Config struct {
|
|
DistSpecVersion string `json:"distSpecVersion" mapstructure:"distSpecVersion"`
|
|
GoVersion string
|
|
Commit string
|
|
ReleaseTag string
|
|
BinaryType string
|
|
Storage GlobalStorageConfig
|
|
HTTP HTTPConfig
|
|
Log *LogConfig
|
|
Extensions *extconf.ExtensionConfig
|
|
Scheduler *SchedulerConfig `json:"scheduler" mapstructure:",omitempty"`
|
|
Cluster *ClusterConfig `json:"cluster" mapstructure:",omitempty"`
|
|
|
|
// Mutex to protect concurrent access to config fields
|
|
mu sync.RWMutex
|
|
}
|
|
|
|
func New() *Config {
|
|
return &Config{
|
|
DistSpecVersion: distspec.Version,
|
|
GoVersion: GoVersion,
|
|
Commit: Commit,
|
|
ReleaseTag: ReleaseTag,
|
|
BinaryType: BinaryType,
|
|
Storage: GlobalStorageConfig{
|
|
StorageConfig: StorageConfig{
|
|
Dedupe: true,
|
|
GC: true,
|
|
GCDelay: storageConstants.DefaultGCDelay,
|
|
GCInterval: storageConstants.DefaultGCInterval,
|
|
Retention: ImageRetention{},
|
|
},
|
|
},
|
|
HTTP: HTTPConfig{Address: "127.0.0.1", Port: "8080", Auth: &AuthConfig{FailDelay: 0}},
|
|
Log: &LogConfig{Level: "debug"},
|
|
}
|
|
}
|
|
|
|
func (expConfig StorageConfig) ParamsEqual(actConfig StorageConfig) bool {
|
|
return expConfig.GC == actConfig.GC && expConfig.Dedupe == actConfig.Dedupe &&
|
|
expConfig.GCDelay == actConfig.GCDelay && expConfig.GCInterval == actConfig.GCInterval
|
|
}
|
|
|
|
// isRetentionEnabledInternal checks if retention is enabled without acquiring a lock (internal use only).
|
|
func (c *Config) isRetentionEnabledInternal() bool {
|
|
if c == nil {
|
|
return false
|
|
}
|
|
|
|
var needsMetaDB bool
|
|
|
|
for _, retentionPolicy := range c.Storage.Retention.Policies {
|
|
for _, tagRetentionPolicy := range retentionPolicy.KeepTags {
|
|
if c.isTagsRetentionEnabled(tagRetentionPolicy) {
|
|
needsMetaDB = true
|
|
}
|
|
}
|
|
}
|
|
|
|
for _, subpath := range c.Storage.SubPaths {
|
|
for _, retentionPolicy := range subpath.Retention.Policies {
|
|
for _, tagRetentionPolicy := range retentionPolicy.KeepTags {
|
|
if c.isTagsRetentionEnabled(tagRetentionPolicy) {
|
|
needsMetaDB = true
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
return needsMetaDB
|
|
}
|
|
|
|
// isTagsRetentionEnabled checks if tags retention is enabled for a specific policy (internal use only).
|
|
func (c *Config) isTagsRetentionEnabled(tagRetentionPolicy KeepTagsPolicy) bool {
|
|
if tagRetentionPolicy.MostRecentlyPulledCount != 0 ||
|
|
tagRetentionPolicy.MostRecentlyPushedCount != 0 ||
|
|
tagRetentionPolicy.PulledWithin != nil ||
|
|
tagRetentionPolicy.PushedWithin != nil {
|
|
return true
|
|
}
|
|
|
|
return false
|
|
}
|
|
|
|
// Sanitize makes a sanitized copy of the config removing any secrets.
|
|
func (c *Config) Sanitize() *Config {
|
|
if c == nil {
|
|
return nil
|
|
}
|
|
|
|
c.mu.RLock()
|
|
defer c.mu.RUnlock()
|
|
|
|
sanitizedConfig := &Config{}
|
|
|
|
if err := DeepCopy(c, sanitizedConfig); err != nil {
|
|
panic(err)
|
|
}
|
|
|
|
// Sanitize HTTP config
|
|
if c.HTTP.Auth != nil {
|
|
// Sanitize LDAP bind password
|
|
if c.HTTP.Auth.LDAP != nil && c.HTTP.Auth.LDAP.bindPassword != "" {
|
|
sanitizedConfig.HTTP.Auth.LDAP = &LDAPConfig{}
|
|
|
|
if err := DeepCopy(c.HTTP.Auth.LDAP, sanitizedConfig.HTTP.Auth.LDAP); err != nil {
|
|
panic(err)
|
|
}
|
|
|
|
sanitizedConfig.HTTP.Auth.LDAP.bindPassword = "******"
|
|
}
|
|
|
|
// Sanitize OpenID client secrets
|
|
if c.HTTP.Auth.OpenID != nil {
|
|
sanitizedConfig.HTTP.Auth.OpenID = &OpenIDConfig{
|
|
Providers: make(map[string]OpenIDProviderConfig),
|
|
}
|
|
|
|
for provider, config := range c.HTTP.Auth.OpenID.Providers {
|
|
sanitizedConfig.HTTP.Auth.OpenID.Providers[provider] = OpenIDProviderConfig{
|
|
Name: config.Name,
|
|
ClientID: config.ClientID,
|
|
ClientSecret: "******",
|
|
KeyPath: config.KeyPath,
|
|
Issuer: config.Issuer,
|
|
AuthURL: config.AuthURL,
|
|
TokenURL: config.TokenURL,
|
|
Scopes: config.Scopes,
|
|
ClaimMapping: config.ClaimMapping,
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
if c.Extensions.IsEventRecorderEnabled() {
|
|
for i, sink := range c.Extensions.Events.Sinks {
|
|
if sink.Credentials == nil {
|
|
continue
|
|
}
|
|
|
|
if err := DeepCopy(&c.Extensions.Events.Sinks[i], &sanitizedConfig.Extensions.Events.Sinks[i]); err != nil {
|
|
panic(err)
|
|
}
|
|
|
|
sanitizedConfig.Extensions.Events.Sinks[i].Credentials.Password = "******"
|
|
}
|
|
}
|
|
|
|
return sanitizedConfig
|
|
}
|
|
|
|
// UpdateReloadableConfig updates only the fields that can be reloaded at runtime.
|
|
func (c *Config) UpdateReloadableConfig(newConfig *Config) {
|
|
if c == nil {
|
|
return
|
|
}
|
|
|
|
c.mu.Lock()
|
|
defer c.mu.Unlock()
|
|
|
|
// Update storage configuration
|
|
c.Storage.GC = newConfig.Storage.GC
|
|
c.Storage.Dedupe = newConfig.Storage.Dedupe
|
|
c.Storage.GCDelay = newConfig.Storage.GCDelay
|
|
c.Storage.GCInterval = newConfig.Storage.GCInterval
|
|
|
|
// Only update retention if we have a metaDB already in place
|
|
if c.isRetentionEnabledInternal() {
|
|
c.Storage.Retention = newConfig.Storage.Retention
|
|
}
|
|
|
|
// Update storage subpaths configuration
|
|
for subPath, storageConfig := range newConfig.Storage.SubPaths {
|
|
subPathConfig, ok := c.Storage.SubPaths[subPath]
|
|
if !ok {
|
|
continue
|
|
}
|
|
|
|
subPathConfig.GC = storageConfig.GC
|
|
subPathConfig.Dedupe = storageConfig.Dedupe
|
|
subPathConfig.GCDelay = storageConfig.GCDelay
|
|
subPathConfig.GCInterval = storageConfig.GCInterval
|
|
|
|
// Only update retention if we have a metaDB already in place
|
|
if c.isRetentionEnabledInternal() {
|
|
subPathConfig.Retention = storageConfig.Retention
|
|
}
|
|
|
|
c.Storage.SubPaths[subPath] = subPathConfig
|
|
}
|
|
|
|
// Update authentication configuration
|
|
if c.HTTP.Auth != nil && newConfig.HTTP.Auth != nil {
|
|
c.HTTP.Auth.HTPasswd = newConfig.HTTP.Auth.HTPasswd
|
|
c.HTTP.Auth.LDAP = newConfig.HTTP.Auth.LDAP
|
|
c.HTTP.Auth.APIKey = newConfig.HTTP.Auth.APIKey
|
|
c.HTTP.Auth.OpenID = newConfig.HTTP.Auth.OpenID
|
|
c.HTTP.Auth.SecureSession = newConfig.HTTP.Auth.SecureSession
|
|
c.HTTP.Auth.MTLS = newConfig.HTTP.Auth.MTLS
|
|
}
|
|
|
|
// Initialize and update AccessControlConfig
|
|
if newConfig.HTTP.AccessControl != nil && c.HTTP.AccessControl == nil {
|
|
c.HTTP.AccessControl = &AccessControlConfig{}
|
|
}
|
|
|
|
if newConfig.HTTP.AccessControl == nil {
|
|
c.HTTP.AccessControl = nil
|
|
} else {
|
|
// Update AccessControlConfig fields directly
|
|
c.HTTP.AccessControl.Repositories = newConfig.HTTP.AccessControl.Repositories
|
|
c.HTTP.AccessControl.AdminPolicy = newConfig.HTTP.AccessControl.AdminPolicy
|
|
c.HTTP.AccessControl.Metrics = newConfig.HTTP.AccessControl.Metrics
|
|
c.HTTP.AccessControl.Groups = newConfig.HTTP.AccessControl.Groups
|
|
}
|
|
|
|
// Initialize and update ExtensionConfig
|
|
if newConfig.Extensions != nil && c.Extensions == nil {
|
|
c.Extensions = &extconf.ExtensionConfig{}
|
|
}
|
|
|
|
if newConfig.Extensions == nil {
|
|
c.Extensions = nil
|
|
} else if c.Extensions != nil {
|
|
// Update sync extension
|
|
c.Extensions.Sync = newConfig.Extensions.Sync
|
|
|
|
// Update search extension
|
|
if newConfig.Extensions.Search != nil && newConfig.Extensions.Search.CVE != nil {
|
|
// Only update if search is enabled
|
|
if c.Extensions.IsSearchEnabled() {
|
|
if c.Extensions.Search != nil {
|
|
c.Extensions.Search.CVE = newConfig.Extensions.Search.CVE
|
|
}
|
|
}
|
|
} else {
|
|
// Remove search CVE config if not present in new config
|
|
if c.Extensions.Search != nil {
|
|
c.Extensions.Search.CVE = nil
|
|
}
|
|
}
|
|
|
|
// Update scrub extension
|
|
c.Extensions.Scrub = newConfig.Extensions.Scrub
|
|
}
|
|
}
|
|
|
|
// CopyAuthConfig returns a copy of the auth config if it exists.
|
|
func (c *Config) CopyAuthConfig() *AuthConfig {
|
|
if c == nil {
|
|
return nil
|
|
}
|
|
|
|
c.mu.RLock()
|
|
defer c.mu.RUnlock()
|
|
|
|
if c.HTTP.Auth == nil {
|
|
return nil
|
|
}
|
|
|
|
// Return a deep copy using tiendc/go-deepcopy to avoid race conditions
|
|
authCopy := &AuthConfig{}
|
|
_ = deepcopy.Copy(authCopy, c.HTTP.Auth)
|
|
|
|
return authCopy
|
|
}
|
|
|
|
// CopyAccessControlConfig returns a copy of the access control config if it exists.
|
|
func (c *Config) CopyAccessControlConfig() *AccessControlConfig {
|
|
if c == nil {
|
|
return nil
|
|
}
|
|
|
|
c.mu.RLock()
|
|
defer c.mu.RUnlock()
|
|
|
|
if c.HTTP.AccessControl == nil {
|
|
return nil
|
|
}
|
|
|
|
// Return a deep copy using tiendc/go-deepcopy to avoid race conditions
|
|
accessControlCopy := &AccessControlConfig{}
|
|
_ = deepcopy.Copy(accessControlCopy, c.HTTP.AccessControl)
|
|
|
|
return accessControlCopy
|
|
}
|
|
|
|
// CopyStorageConfig returns a copy of the storage config.
|
|
func (c *Config) CopyStorageConfig() GlobalStorageConfig {
|
|
if c == nil {
|
|
return GlobalStorageConfig{}
|
|
}
|
|
|
|
c.mu.RLock()
|
|
defer c.mu.RUnlock()
|
|
|
|
// Return a deep copy using tiendc/go-deepcopy to avoid race conditions
|
|
storageCopy := GlobalStorageConfig{}
|
|
_ = deepcopy.Copy(&storageCopy, &c.Storage)
|
|
|
|
return storageCopy
|
|
}
|
|
|
|
// CopyExtensionsConfig returns a copy of the extensions config if it exists.
|
|
func (c *Config) CopyExtensionsConfig() *extconf.ExtensionConfig {
|
|
if c == nil {
|
|
return nil
|
|
}
|
|
|
|
c.mu.RLock()
|
|
defer c.mu.RUnlock()
|
|
|
|
if c.Extensions == nil {
|
|
return nil
|
|
}
|
|
|
|
// Return a deep copy using tiendc/go-deepcopy to avoid race conditions
|
|
extensionsCopy := &extconf.ExtensionConfig{}
|
|
_ = deepcopy.Copy(extensionsCopy, c.Extensions)
|
|
|
|
return extensionsCopy
|
|
}
|
|
|
|
// CopyLogConfig returns a copy of the log config if it exists.
|
|
func (c *Config) CopyLogConfig() *LogConfig {
|
|
if c == nil {
|
|
return nil
|
|
}
|
|
|
|
c.mu.RLock()
|
|
defer c.mu.RUnlock()
|
|
|
|
if c.Log == nil {
|
|
return nil
|
|
}
|
|
|
|
// Return a copy to avoid race conditions
|
|
logCopy := *c.Log
|
|
|
|
return &logCopy
|
|
}
|
|
|
|
// CopyClusterConfig returns a copy of the cluster config if it exists.
|
|
func (c *Config) CopyClusterConfig() *ClusterConfig {
|
|
if c == nil {
|
|
return nil
|
|
}
|
|
|
|
c.mu.RLock()
|
|
defer c.mu.RUnlock()
|
|
|
|
if c.Cluster == nil {
|
|
return nil
|
|
}
|
|
|
|
// Return a deep copy using tiendc/go-deepcopy to avoid race conditions
|
|
clusterCopy := &ClusterConfig{}
|
|
_ = deepcopy.Copy(clusterCopy, c.Cluster)
|
|
|
|
return clusterCopy
|
|
}
|
|
|
|
// CopySchedulerConfig returns a copy of the scheduler config if it exists.
|
|
func (c *Config) CopySchedulerConfig() *SchedulerConfig {
|
|
if c == nil {
|
|
return nil
|
|
}
|
|
|
|
c.mu.RLock()
|
|
defer c.mu.RUnlock()
|
|
|
|
if c.Scheduler == nil {
|
|
return nil
|
|
}
|
|
|
|
// Return a copy to avoid race conditions
|
|
schedulerCopy := *c.Scheduler
|
|
|
|
return &schedulerCopy
|
|
}
|
|
|
|
// CopyTLSConfig returns a copy of the TLS config.
|
|
func (c *Config) CopyTLSConfig() *TLSConfig {
|
|
if c == nil {
|
|
return nil
|
|
}
|
|
|
|
c.mu.RLock()
|
|
defer c.mu.RUnlock()
|
|
|
|
if c.HTTP.TLS == nil {
|
|
return nil
|
|
}
|
|
|
|
// Return a copy to avoid race conditions
|
|
tlsCopy := *c.HTTP.TLS
|
|
|
|
return &tlsCopy
|
|
}
|
|
|
|
// CopyRatelimit returns a copy of the rate limit config.
|
|
func (c *Config) CopyRatelimit() *RatelimitConfig {
|
|
if c == nil {
|
|
return nil
|
|
}
|
|
|
|
c.mu.RLock()
|
|
defer c.mu.RUnlock()
|
|
|
|
if c.HTTP.Ratelimit == nil {
|
|
return nil
|
|
}
|
|
|
|
// Return a deep copy using tiendc/go-deepcopy to avoid race conditions
|
|
ratelimitCopy := &RatelimitConfig{}
|
|
_ = deepcopy.Copy(ratelimitCopy, c.HTTP.Ratelimit)
|
|
|
|
return ratelimitCopy
|
|
}
|
|
|
|
// GetVersionInfo returns version information (read-only, safe to access directly).
|
|
func (c *Config) GetVersionInfo() (string, string, string, string) {
|
|
if c == nil {
|
|
return "", "", "", ""
|
|
}
|
|
|
|
c.mu.RLock()
|
|
defer c.mu.RUnlock()
|
|
|
|
return c.Commit, c.BinaryType, c.GoVersion, c.DistSpecVersion
|
|
}
|
|
|
|
// GetRealm returns the HTTP realm value.
|
|
func (c *Config) GetRealm() string {
|
|
if c == nil {
|
|
return ""
|
|
}
|
|
|
|
c.mu.RLock()
|
|
defer c.mu.RUnlock()
|
|
|
|
return c.HTTP.Realm
|
|
}
|
|
|
|
// GetCompat returns a copy of the compatibility config.
|
|
func (c *Config) GetCompat() []compat.MediaCompatibility {
|
|
if c == nil {
|
|
return nil
|
|
}
|
|
|
|
c.mu.RLock()
|
|
defer c.mu.RUnlock()
|
|
|
|
if c.HTTP.Compat == nil {
|
|
return nil
|
|
}
|
|
|
|
// Return a copy to avoid race conditions
|
|
compatCopy := make([]compat.MediaCompatibility, len(c.HTTP.Compat))
|
|
copy(compatCopy, c.HTTP.Compat)
|
|
|
|
return compatCopy
|
|
}
|
|
|
|
// GetHTTPAddress returns the HTTP address.
|
|
func (c *Config) GetHTTPAddress() string {
|
|
if c == nil {
|
|
return ""
|
|
}
|
|
|
|
c.mu.RLock()
|
|
defer c.mu.RUnlock()
|
|
|
|
return c.HTTP.Address
|
|
}
|
|
|
|
// GetHTTPPort returns the HTTP port.
|
|
func (c *Config) GetHTTPPort() string {
|
|
if c == nil {
|
|
return ""
|
|
}
|
|
|
|
c.mu.RLock()
|
|
defer c.mu.RUnlock()
|
|
|
|
return c.HTTP.Port
|
|
}
|
|
|
|
// GetAllowOrigin returns the CORS allow origin configuration.
|
|
func (c *Config) GetAllowOrigin() string {
|
|
if c == nil {
|
|
return ""
|
|
}
|
|
|
|
c.mu.RLock()
|
|
defer c.mu.RUnlock()
|
|
|
|
return c.HTTP.AllowOrigin
|
|
}
|
|
|
|
// IsMTLSAuthEnabled checks if mTLS authentication is enabled.
|
|
func (c *Config) IsMTLSAuthEnabled() bool {
|
|
if c == nil {
|
|
return false
|
|
}
|
|
|
|
c.mu.RLock()
|
|
defer c.mu.RUnlock()
|
|
|
|
// mTLS is enabled if TLS is configured with client CA certificates
|
|
if c.HTTP.TLS != nil &&
|
|
c.HTTP.TLS.Key != "" &&
|
|
c.HTTP.TLS.Cert != "" &&
|
|
c.HTTP.TLS.CACert != "" {
|
|
return true
|
|
}
|
|
|
|
return false
|
|
}
|
|
|
|
// IsRetentionEnabled checks if tags retention is enabled.
|
|
func (c *Config) IsRetentionEnabled() bool {
|
|
if c == nil {
|
|
return false
|
|
}
|
|
|
|
c.mu.RLock()
|
|
defer c.mu.RUnlock()
|
|
|
|
return c.isRetentionEnabledInternal()
|
|
}
|
|
|
|
// IsCompatEnabled checks if compatibility mode is enabled.
|
|
func (c *Config) IsCompatEnabled() bool {
|
|
if c == nil {
|
|
return false
|
|
}
|
|
|
|
c.mu.RLock()
|
|
defer c.mu.RUnlock()
|
|
|
|
return len(c.HTTP.Compat) > 0
|
|
}
|
|
|
|
// UseSecureSession returns whether cookies should have the Secure flag set.
|
|
// If TLS is configured, always returns true. Otherwise, returns the value
|
|
// of SecureSession if set, or false by default.
|
|
func (c *Config) UseSecureSession() bool {
|
|
if c == nil {
|
|
return false
|
|
}
|
|
|
|
c.mu.RLock()
|
|
defer c.mu.RUnlock()
|
|
|
|
// If TLS is configured, cookies should be secure
|
|
if c.HTTP.TLS != nil {
|
|
return true
|
|
}
|
|
|
|
// If TLS is not configured, check if SecureSession is explicitly set in auth config
|
|
if c.HTTP.Auth != nil && c.HTTP.Auth.SecureSession != nil {
|
|
return *c.HTTP.Auth.SecureSession
|
|
}
|
|
|
|
// Default to false if TLS is not configured and no explicit setting
|
|
return false
|
|
}
|
|
|
|
// IsOpenIDSupported checks if the provider supports OpenID.
|
|
func IsOpenIDSupported(provider string) bool {
|
|
for _, supportedProvider := range openIDSupportedProviders {
|
|
if supportedProvider == provider {
|
|
return true
|
|
}
|
|
}
|
|
|
|
return false
|
|
}
|
|
|
|
// IsOauth2Supported checks if the provider supports OAuth2.
|
|
func IsOauth2Supported(provider string) bool {
|
|
for _, supportedProvider := range oauth2SupportedProviders {
|
|
if supportedProvider == provider {
|
|
return true
|
|
}
|
|
}
|
|
|
|
return false
|
|
}
|
|
|
|
// SameFile compare two files.
|
|
// This method will first do the stat of two file and compare using os.SameFile method.
|
|
func SameFile(str1, str2 string) (bool, error) {
|
|
sFile, err := os.Stat(str1)
|
|
if err != nil {
|
|
return false, err
|
|
}
|
|
|
|
tFile, err := os.Stat(str2)
|
|
if err != nil {
|
|
return false, err
|
|
}
|
|
|
|
return os.SameFile(sFile, tFile), nil
|
|
}
|
|
|
|
// DeepCopy performs a deep copy of src into dst using JSON marshaling/unmarshaling.
|
|
func DeepCopy(src, dst any) error {
|
|
bytes, err := json.Marshal(src)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
err = json.Unmarshal(bytes, dst)
|
|
|
|
return err
|
|
}
|