mirror of
https://github.com/kyverno/kyverno.git
synced 2025-03-09 09:26:54 +00:00
feat(gctx): move ready check to runtime Signed-off-by: Khaled Emara <khaled.emara@nirmata.com> Co-authored-by: shuting <shuting@nirmata.com>
198 lines
4.8 KiB
Go
198 lines
4.8 KiB
Go
package externalapi
|
|
|
|
import (
|
|
"context"
|
|
"encoding/json"
|
|
"fmt"
|
|
"sync"
|
|
"time"
|
|
|
|
"github.com/go-logr/logr"
|
|
kyvernov1 "github.com/kyverno/kyverno/api/kyverno/v1"
|
|
kyvernov2alpha1 "github.com/kyverno/kyverno/api/kyverno/v2alpha1"
|
|
"github.com/kyverno/kyverno/pkg/client/clientset/versioned"
|
|
kyvernov2alpha1listers "github.com/kyverno/kyverno/pkg/client/listers/kyverno/v2alpha1"
|
|
"github.com/kyverno/kyverno/pkg/engine/apicall"
|
|
"github.com/kyverno/kyverno/pkg/engine/jmespath"
|
|
"github.com/kyverno/kyverno/pkg/event"
|
|
entryevent "github.com/kyverno/kyverno/pkg/globalcontext/event"
|
|
"github.com/kyverno/kyverno/pkg/globalcontext/store"
|
|
controllerutils "github.com/kyverno/kyverno/pkg/utils/controller"
|
|
corev1 "k8s.io/api/core/v1"
|
|
"k8s.io/apimachinery/pkg/util/wait"
|
|
"k8s.io/client-go/util/retry"
|
|
)
|
|
|
|
type entry struct {
|
|
sync.Mutex
|
|
dataMap map[string]any
|
|
err error
|
|
stop func()
|
|
projections []store.Projection
|
|
}
|
|
|
|
func New(
|
|
ctx context.Context,
|
|
gce *kyvernov2alpha1.GlobalContextEntry,
|
|
eventGen event.Interface,
|
|
kyvernoClient versioned.Interface,
|
|
gceLister kyvernov2alpha1listers.GlobalContextEntryLister,
|
|
logger logr.Logger,
|
|
client apicall.ClientInterface,
|
|
call kyvernov1.APICall,
|
|
period time.Duration,
|
|
maxResponseLength int64,
|
|
shouldUpdateStatus bool,
|
|
jp jmespath.Interface,
|
|
) (store.Entry, error) {
|
|
var group wait.Group
|
|
ctx, cancel := context.WithCancel(ctx)
|
|
stop := func() {
|
|
// Send stop signal to informer's goroutine
|
|
cancel()
|
|
// Wait for the group to terminate
|
|
group.Wait()
|
|
}
|
|
|
|
projections := make([]store.Projection, 0)
|
|
for _, p := range gce.Spec.Projections {
|
|
jpQuery, err := jp.Query(p.JMESPath)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("failed to parse jmespath query: %s", err)
|
|
}
|
|
projections = append(projections, store.Projection{
|
|
Name: p.Name,
|
|
JP: jpQuery,
|
|
})
|
|
}
|
|
|
|
e := &entry{
|
|
dataMap: make(map[string]any),
|
|
stop: stop,
|
|
projections: projections,
|
|
}
|
|
|
|
group.StartWithContext(ctx, func(ctx context.Context) {
|
|
config := apicall.NewAPICallConfiguration(maxResponseLength)
|
|
caller := apicall.NewExecutor(logger, "globalcontext", client, config)
|
|
|
|
wait.UntilWithContext(ctx, func(ctx context.Context) {
|
|
if data, err := doCall(ctx, caller, call, gce.Spec.APICall.RetryLimit); err != nil {
|
|
e.setData(nil, err)
|
|
|
|
logger.Error(err, "failed to get data from api caller")
|
|
|
|
eventGen.Add(entryevent.NewErrorEvent(corev1.ObjectReference{
|
|
APIVersion: gce.APIVersion,
|
|
Kind: gce.Kind,
|
|
Name: gce.Name,
|
|
Namespace: gce.Namespace,
|
|
UID: gce.UID,
|
|
}, err))
|
|
} else {
|
|
e.setData(data, nil)
|
|
|
|
logger.V(4).Info("api call success", "data", data)
|
|
|
|
if shouldUpdateStatus {
|
|
if updateErr := updateStatus(ctx, gce, kyvernoClient); updateErr != nil {
|
|
logger.Error(updateErr, "failed to update status")
|
|
}
|
|
}
|
|
}
|
|
}, period)
|
|
})
|
|
|
|
return e, nil
|
|
}
|
|
|
|
func (e *entry) Get(projection string) (any, error) {
|
|
e.Lock()
|
|
defer e.Unlock()
|
|
|
|
if e.err != nil {
|
|
return nil, e.err
|
|
}
|
|
|
|
data := e.dataMap[projection]
|
|
if data == nil {
|
|
return nil, fmt.Errorf("no data available")
|
|
}
|
|
|
|
return data, nil
|
|
}
|
|
|
|
func (e *entry) Stop() {
|
|
e.Lock()
|
|
defer e.Unlock()
|
|
e.stop()
|
|
}
|
|
|
|
func (e *entry) setData(data any, err error) {
|
|
e.Lock()
|
|
defer e.Unlock()
|
|
|
|
if err != nil {
|
|
e.err = err
|
|
} else {
|
|
if len(e.projections) > 0 {
|
|
var jsonData any
|
|
if bytes, ok := data.([]byte); ok {
|
|
err = json.Unmarshal(bytes, &jsonData)
|
|
if err != nil {
|
|
e.err = err
|
|
return
|
|
}
|
|
} else {
|
|
e.err = fmt.Errorf("data is not a byte array")
|
|
return
|
|
}
|
|
|
|
for _, projection := range e.projections {
|
|
result, err := projection.JP.Search(jsonData)
|
|
if err != nil {
|
|
e.err = err
|
|
return
|
|
}
|
|
e.dataMap[projection.Name] = result
|
|
}
|
|
e.err = nil
|
|
} else {
|
|
e.dataMap[""] = data
|
|
}
|
|
}
|
|
}
|
|
|
|
func doCall(ctx context.Context, caller apicall.Executor, call kyvernov1.APICall, retryLimit int) (any, error) {
|
|
var result any
|
|
backoff := wait.Backoff{
|
|
Duration: retry.DefaultBackoff.Duration,
|
|
Factor: retry.DefaultBackoff.Factor,
|
|
Jitter: retry.DefaultBackoff.Jitter,
|
|
Steps: retryLimit,
|
|
}
|
|
|
|
retryError := retry.OnError(backoff, func(err error) bool {
|
|
return err != nil
|
|
}, func() error {
|
|
var exeErr error
|
|
result, exeErr = caller.Execute(ctx, &call)
|
|
return exeErr
|
|
})
|
|
|
|
return result, retryError
|
|
}
|
|
|
|
func updateStatus(ctx context.Context, gce *kyvernov2alpha1.GlobalContextEntry, kyvernoClient versioned.Interface) error {
|
|
retryErr := retry.RetryOnConflict(retry.DefaultRetry, func() error {
|
|
return controllerutils.UpdateStatus(ctx, gce, kyvernoClient.KyvernoV2alpha1().GlobalContextEntries(), func(latest *kyvernov2alpha1.GlobalContextEntry) error {
|
|
if latest == nil {
|
|
return fmt.Errorf("failed to update status: %s", gce.GetName())
|
|
}
|
|
latest.Status.UpdateRefreshTime()
|
|
return nil
|
|
}, nil)
|
|
})
|
|
|
|
return retryErr
|
|
}
|