pkg/containercredentials/config.go (89 lines of code) (raw):
/*
Copyright 2023 Amazon.com, Inc. or its affiliates. All Rights Reserved.
Licensed under the Apache License, Version 2.0 (the "License").
You may not use this file except in compliance with the License.
A copy of the License is located at
http://www.apache.org/licenses/LICENSE-2.0
or in the "license" file accompanying this file. This file is distributed
on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either
express or implied. See the License for the specific language governing
permissions and limitations under the License.
*/
package containercredentials
import (
"context"
"encoding/json"
"fmt"
"github.com/aws/amazon-eks-pod-identity-webhook/pkg/filesystem"
"k8s.io/klog/v2"
"sync"
)
type Config interface {
Get(namespace string, serviceAccount string) *PatchConfig
}
type FileConfig struct {
audience string
mountPath string
volumeName string
tokenPath string
fullUri string
watcher *filesystem.FileWatcher
identityConfigObject *IdentityConfigObject
cache map[Identity]bool
mu sync.RWMutex // guards cache
}
type PatchConfig struct {
Audience string
MountPath string
VolumeName string
TokenPath string
FullUri string
}
func NewFileConfig(audience, mountPath, volumeName, tokenPath, fullUri string) *FileConfig {
return &FileConfig{
audience: audience,
mountPath: mountPath,
volumeName: volumeName,
tokenPath: tokenPath,
fullUri: fullUri,
identityConfigObject: nil,
cache: make(map[Identity]bool),
}
}
// StartWatcher creates and starts a fsnotify watcher on the target config file.
// The watcher runs continuously until the context is cancelled. When the file is updated,
// Load will be invoked, and thus will refresh the cache.
func (f *FileConfig) StartWatcher(ctx context.Context, filePath string) error {
f.watcher = filesystem.NewFileWatcher("container-credential-config", filePath, f.Load)
return f.watcher.Watch(ctx)
}
func (f *FileConfig) Load(content []byte) error {
f.mu.Lock()
defer f.mu.Unlock()
if content == nil || len(content) == 0 {
klog.Info("Container credentials config file is empty, clearing cache")
f.identityConfigObject = nil
f.cache = nil
return nil
}
var configObject IdentityConfigObject
if err := json.Unmarshal(content, &configObject); err != nil {
return fmt.Errorf("error Unmarshalling container credentials config file: %v", err)
}
newCache := make(map[Identity]bool)
for _, item := range configObject.Identities {
klog.V(5).Infof("Adding SA %s/%s to container credentials config cache", item.Namespace, item.ServiceAccount)
newCache[item] = true
}
f.identityConfigObject = &configObject
f.cache = newCache
klog.Info("Successfully loaded container credentials config file")
return nil
}
func (f *FileConfig) Get(namespace string, serviceAccount string) *PatchConfig {
key := Identity{
Namespace: namespace,
ServiceAccount: serviceAccount,
}
if f.getCacheItem(key) {
return &PatchConfig{
Audience: f.audience,
MountPath: f.mountPath,
VolumeName: f.volumeName,
TokenPath: f.tokenPath,
FullUri: f.fullUri,
}
}
return nil
}
func (f *FileConfig) getCacheItem(identity Identity) bool {
f.mu.RLock()
defer f.mu.RUnlock()
return f.cache[identity]
}