1
0
Fork 0
mirror of https://github.com/kyverno/kyverno.git synced 2025-03-31 03:45:17 +00:00

feat: add raw api call support (#3820)

Signed-off-by: Charles-Edouard Brétéché <charled.breteche@gmail.com>

Signed-off-by: Charles-Edouard Brétéché <charled.breteche@gmail.com>
Co-authored-by: shuting <shuting@nirmata.com>
This commit is contained in:
Charles-Edouard Brétéché 2022-08-23 18:52:54 +02:00 committed by GitHub
parent 6c93a925ab
commit 5cc97993dc
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
4 changed files with 43 additions and 2 deletions

View file

@ -2,6 +2,7 @@ package dclient
import (
"context"
"errors"
"fmt"
"time"
@ -32,6 +33,7 @@ type Interface interface {
Discovery() IDiscovery
// SetDiscovery sets the discovery client implementation
SetDiscovery(discoveryClient IDiscovery)
RawAbsPath(path string) ([]byte, error)
// GetResource returns the resource in unstructured/json format
GetResource(apiVersion string, kind string, namespace string, name string, subresources ...string) (*unstructured.Unstructured, error)
// PatchResource patches the resource
@ -55,6 +57,7 @@ type client struct {
discoveryClient IDiscovery
clientConfig *rest.Config
kclient kubernetes.Interface
restClient rest.Interface
}
// NewClient creates new instance of client
@ -67,6 +70,7 @@ func NewClient(config *rest.Config, kclient *kubernetes.Clientset, resync time.D
client: dclient,
clientConfig: config,
kclient: kclient,
restClient: kclient.RESTClient(),
}
// Set discovery client
discoveryClient := &serverPreferredResources{
@ -129,6 +133,13 @@ func (c *client) GetResource(apiVersion string, kind string, namespace string, n
return c.getResourceInterface(apiVersion, kind, namespace).Get(context.TODO(), name, metav1.GetOptions{}, subresources...)
}
func (c *client) RawAbsPath(path string) ([]byte, error) {
if c.restClient == nil {
return nil, errors.New("rest client not supported")
}
return c.restClient.Get().AbsPath(path).DoRaw(context.TODO())
}
// PatchResource patches the resource
func (c *client) PatchResource(apiVersion string, kind string, namespace string, name string, patch []byte) (*unstructured.Unstructured, error) {
return c.getResourceInterface(apiVersion, kind, namespace).Patch(context.TODO(), name, types.JSONPatchType, patch, metav1.PatchOptions{})

View file

@ -12,6 +12,7 @@ type APIPath struct {
ResourceType string
Name string
Namespace string
Raw string
}
// NewAPIPath validates and parses an API path.
@ -20,6 +21,12 @@ func NewAPIPath(path string) (*APIPath, error) {
trimmedPath := strings.Trim(path, "/ ")
paths := strings.Split(trimmedPath, "/")
if paths[0] == "apis" && len(paths) > 7 {
return &APIPath{
Raw: path,
}, nil
}
if len(paths) < 3 || len(paths) > 7 {
return nil, fmt.Errorf("invalid path length %s", path)
}

View file

@ -4,6 +4,22 @@ import (
"testing"
)
func Test_Raw(t *testing.T) {
f := func(path string) {
p, err := NewAPIPath(path)
if err != nil {
t.Error(err)
return
}
if p.Raw != path {
t.Errorf("expected raw path %s got %s", path, p.Raw)
}
}
f("/apis/cluster.karmada.io/v1alpha1/clusters/member1/proxy/api/v1/namespaces/{{ request.namespace }}/pods")
}
func Test_Paths(t *testing.T) {
f := func(path, expected string) {
p, err := NewAPIPath(path)

View file

@ -313,6 +313,11 @@ func fetchAPIData(log logr.Logger, entry kyvernov1.ContextEntry, ctx *PolicyCont
if err != nil {
return nil, fmt.Errorf("failed to add resource with urlPath: %s: %v", p, err)
}
} else if p.Raw != "" {
jsonData, err = getResource(ctx, p)
if err != nil {
return nil, fmt.Errorf("failed to get resource with raw url\n: %s: %v", p, err)
}
} else {
jsonData, err = loadResourceList(ctx, p)
if err != nil {
@ -340,15 +345,17 @@ func loadResource(ctx *PolicyContext, p *APIPath) ([]byte, error) {
if ctx.Client == nil {
return nil, fmt.Errorf("API client is not available")
}
r, err := ctx.Client.GetResource(p.Version, p.ResourceType, p.Namespace, p.Name)
if err != nil {
return nil, err
}
return r.MarshalJSON()
}
func getResource(ctx *PolicyContext, p *APIPath) ([]byte, error) {
return ctx.Client.RawAbsPath(p.Raw)
}
func loadConfigMap(logger logr.Logger, entry kyvernov1.ContextEntry, ctx *PolicyContext) error {
data, err := fetchConfigMap(logger, entry, ctx)
if err != nil {