package main

import (
	"fmt"
	"os"
	"path"
	"reflect"
	"regexp"
	"strings"
	"text/template"

	"github.com/kyverno/kyverno/pkg/client/clientset/versioned"
	apiserver "k8s.io/apiextensions-apiserver/pkg/client/clientset/clientset"
	"k8s.io/apimachinery/pkg/util/sets"
	"k8s.io/client-go/dynamic"
	"k8s.io/client-go/kubernetes"
	"k8s.io/client-go/metadata"
	aggregator "k8s.io/kube-aggregator/pkg/client/clientset_generated/clientset"
)

const (
	resourceTpl = `
package resource

import (
	"fmt"
	"time"
	"github.com/go-logr/logr"
	"github.com/kyverno/kyverno/pkg/metrics"
	"github.com/kyverno/kyverno/pkg/tracing"
	"go.opentelemetry.io/otel/attribute"
	"go.opentelemetry.io/otel/codes"
	"go.opentelemetry.io/otel/trace"
	"go.uber.org/multierr"
	{{- range $package := Packages .Target.Type }}
	{{ Pkg $package }} {{ Quote $package }}
	{{- end }}
)

func WithLogging(inner {{ GoType .Target.Type }}, logger logr.Logger) {{ GoType .Target.Type }} {
	return &withLogging{inner, logger}
}

func WithMetrics(inner {{ GoType .Target.Type }}, recorder metrics.Recorder) {{ GoType .Target.Type }} {
	return &withMetrics{inner, recorder}
}

func WithTracing(inner {{ GoType .Target.Type }}, client, kind string) {{ GoType .Target.Type }} {
	return &withTracing{inner, client, kind}
}

type withLogging struct {
	inner  {{ GoType .Target.Type }}
	logger logr.Logger
}

{{- range $operation := .Target.Operations }}
func (c *withLogging) {{ $operation.Method.Name }}(
	{{- range $i, $arg := Args $operation.Method -}}
	{{- if $arg.IsVariadic -}}
	arg{{ $i }} ...{{ GoType $arg.Type.Elem }},
	{{- else -}}
	arg{{ $i }} {{ GoType $arg.Type }},
	{{- end -}}
	{{- end -}}
) (
	{{- range $return := Returns $operation.Method -}}
	{{ GoType $return }},
	{{- end -}}
) {
	start := time.Now()
	logger := c.logger.WithValues("operation", {{ Quote $operation.Method.Name }})
	{{ range $i, $ret := Returns $operation.Method }}ret{{ $i }}{{ if not $ret.IsLast -}},{{- end }} {{ end }} := c.inner.{{ $operation.Method.Name }}(
		{{- range $i, $arg := Args $operation.Method -}}
		{{- if $arg.IsVariadic -}}
		arg{{ $i }}...,
		{{- else -}}
		arg{{ $i }},
		{{- end -}}
		{{- end -}}
	)
	{{- if $operation.HasError }}
	if err := multierr.Combine(
		{{- range $i, $ret := Returns $operation.Method -}}
		{{- if $ret.IsError -}}
		ret{{ $i }},
		{{- end -}}
		{{- end -}}
	); err != nil {
		logger.Error(err, "{{ $operation.Method.Name }} failed", "duration", time.Since(start))
	} else {
		logger.Info("{{ $operation.Method.Name }} done", "duration", time.Since(start))
	}
	{{- else }}
	logger.Info("{{ $operation.Method.Name }} done", "duration", time.Since(start))
	{{- end }}
	return	{{ range $i, $ret := Returns $operation.Method -}}
	ret{{ $i }}{{ if not $ret.IsLast -}},{{- end }}
	{{- end }}
}
{{- end }}

type withMetrics struct {
	inner    {{ GoType .Target.Type }}
	recorder metrics.Recorder
}

{{- range $operation := .Target.Operations }}
func (c *withMetrics) {{ $operation.Method.Name }}(
	{{- range $i, $arg := Args $operation.Method -}}
	{{- if $arg.IsVariadic -}}
	arg{{ $i }} ...{{ GoType $arg.Type.Elem }},
	{{- else -}}
	arg{{ $i }} {{ GoType $arg.Type }},
	{{- end -}}
	{{- end -}}
) (
	{{- range $return := Returns $operation.Method -}}
	{{ GoType $return }},
	{{- end -}}
) {
	{{- if $operation.HasContext }}
	defer c.recorder.RecordWithContext(arg0, {{ Quote (SnakeCase $operation.Method.Name) }})
	{{- else }}
	defer c.recorder.Record({{ Quote (SnakeCase $operation.Method.Name) }})
	{{- end }}
	return c.inner.{{ $operation.Method.Name }}(
		{{- range $i, $arg := Args $operation.Method -}}
		{{- if $arg.IsVariadic -}}
		arg{{ $i }}...,
		{{- else -}}
		arg{{ $i }},
		{{- end -}}
		{{- end -}}
	)
}
{{- end }}

type withTracing struct {
	inner    {{ GoType .Target.Type }}
	client   string
	kind     string
}

{{- range $operation := .Target.Operations }}
func (c *withTracing) {{ $operation.Method.Name }}(
	{{- range $i, $arg := Args $operation.Method -}}
	{{- if $arg.IsVariadic -}}
	arg{{ $i }} ...{{ GoType $arg.Type.Elem }},
	{{- else -}}
	arg{{ $i }} {{ GoType $arg.Type }},
	{{- end -}}
	{{- end -}}
) (
	{{- range $return := Returns $operation.Method -}}
	{{ GoType $return }},
	{{- end -}}
) {
	{{- if not $operation.HasContext }}
	return c.inner.{{ $operation.Method.Name }}(
		{{- range $i, $arg := Args $operation.Method -}}
		{{- if $arg.IsVariadic -}}
		arg{{ $i }}...,
		{{- else -}}
		arg{{ $i }},
		{{- end -}}
		{{- end -}}
	)
	{{- else }}
	var span trace.Span
	if tracing.IsInSpan(arg0) {
		arg0, span = tracing.StartChildSpan(
			arg0,
			"",
			fmt.Sprintf("KUBE %s/%s/%s", c.client, c.kind, {{ Quote $operation.Method.Name }}),
			trace.WithAttributes(
				tracing.KubeClientGroupKey.String(c.client),
				tracing.KubeClientKindKey.String(c.kind),
				tracing.KubeClientOperationKey.String({{ Quote $operation.Method.Name }}),
			),
		)
		defer span.End()
	}
	{{ range $i, $ret := Returns $operation.Method }}ret{{ $i }}{{ if not $ret.IsLast -}},{{- end }} {{ end }} := c.inner.{{ $operation.Method.Name }}(
		{{- range $i, $arg := Args $operation.Method -}}
		{{- if $arg.IsVariadic -}}
		arg{{ $i }}...,
		{{- else -}}
		arg{{ $i }},
		{{- end -}}
		{{- end -}}
	)
	if span != nil {
		{{- if $operation.HasError }}
		{{- range $i, $ret := Returns $operation.Method }}
		{{- if $ret.IsError }}
		tracing.SetSpanStatus(span, ret{{ $i }})
		{{- end }}
		{{- end }}
		{{- end }}
	}
	return	{{ range $i, $ret := Returns $operation.Method -}}
	ret{{ $i }}{{ if not $ret.IsLast -}},{{- end }}
	{{- end }}
	{{- end }}
}
{{- end }}
`
	clientTpl = `
package client

import (
	"github.com/go-logr/logr"
	"github.com/kyverno/kyverno/pkg/metrics"
	"k8s.io/client-go/rest"
	{{- range $package := Packages .Target.Type }}
	{{ Pkg $package }} {{ Quote $package }}
	{{- end }}
	{{- range $method, $resource := .Target.Resources }}
	{{ ToLower $method.Name }} "github.com/kyverno/kyverno/{{ $.Folder }}/{{ ToLower $method.Name }}"
	{{- end }}
)

func WithMetrics(inner {{ GoType .Target.Type }}, metrics metrics.MetricsConfigManager, clientType metrics.ClientType) {{ GoType .Target.Type }} {
	return &withMetrics{inner, metrics, clientType}
}

func WithTracing(inner {{ GoType .Target.Type }}, client string) {{ GoType .Target.Type }} {
	return &withTracing{inner, client}
}

func WithLogging(inner {{ GoType .Target.Type }}, logger logr.Logger) {{ GoType .Target.Type }} {
	return &withLogging{inner, logger}
}

type withMetrics struct {
	inner      {{ GoType .Target }}
	metrics    metrics.MetricsConfigManager
	clientType metrics.ClientType
}
func (c *withMetrics) RESTClient() rest.Interface {
	return c.inner.RESTClient()
}
{{- range $method, $resource := .Target.Resources }}
func (c *withMetrics) {{ $method.Name }}({{- if $method.IsNamespaced -}}namespace string{{- end -}}) {{ GoType $resource.Type }} {
	{{- if $method.IsNamespaced }}
	recorder := metrics.NamespacedClientQueryRecorder(c.metrics, namespace, {{ Quote $resource.Kind }}, c.clientType)
	{{- else }}
	recorder := metrics.ClusteredClientQueryRecorder(c.metrics, {{ Quote $resource.Kind }}, c.clientType)
	{{- end }}
	return 	{{ ToLower $method.Name }}.WithMetrics(c.inner.{{ $method.Name }}(
		{{- if $method.IsNamespaced -}}namespace{{- end -}}
	), recorder)
}
{{- end }}

type withTracing struct {
	inner  {{ GoType .Target }}
	client string
}
func (c *withTracing) RESTClient() rest.Interface {
	return c.inner.RESTClient()
}
{{- range $method, $resource := .Target.Resources }}
func (c *withTracing) {{ $method.Name }}({{- if $method.IsNamespaced -}}namespace string{{- end -}}) {{ GoType $resource.Type }} {
	return 	{{ ToLower $method.Name }}.WithTracing(c.inner.{{ $method.Name }}(
		{{- if $method.IsNamespaced -}}namespace{{- end -}}), c.client, {{ Quote $resource.Kind -}}
	)
}
{{- end }}

type withLogging struct {
	inner  {{ GoType .Target }}
	logger logr.Logger
}
func (c *withLogging) RESTClient() rest.Interface {
	return c.inner.RESTClient()
}
{{- range $method, $resource := .Target.Resources }}
func (c *withLogging) {{ $method.Name }}({{- if $method.IsNamespaced -}}namespace string{{- end -}}) {{ GoType $resource.Type }} {
	return 	{{ ToLower $method.Name }}.WithLogging(c.inner.{{ $method.Name }}(
		{{- if $method.IsNamespaced -}}namespace{{- end -}}), c.logger.WithValues("resource", {{ Quote $method.Name }})
		{{- if $method.IsNamespaced -}}.WithValues("namespace", namespace){{- end -}}
	)
}
{{- end }}
`
	clientsetTpl = `
package clientset

import (
	"github.com/go-logr/logr"
	"github.com/kyverno/kyverno/pkg/metrics"
	{{- range $package := Packages .Target.Type }}
	{{ Pkg $package }} {{ Quote $package }}
	{{- end }}
	{{- range $resourceMethod, $resource := .Target.Resources }}
	{{ ToLower $resourceMethod.Name }} "github.com/kyverno/kyverno/{{ $.Folder }}/{{ ToLower $resourceMethod.Name }}"
	{{- end }}
	{{- range $clientMethod, $client := .Target.Clients }}
	{{ ToLower $clientMethod.Name }} "github.com/kyverno/kyverno/{{ $.Folder }}/{{ ToLower $clientMethod.Name }}"
	{{- end }}
)

type clientset struct {
	{{- range $resourceMethod, $resource := .Target.Resources }}
	{{ ToLower $resourceMethod.Name }} {{ GoType $resource.Type }}
	{{- end }}
	{{- range $clientMethod, $client := .Target.Clients }}
	{{ ToLower $clientMethod.Name }} {{ GoType $client.Type }}
	{{- end }}
}

{{- range $resourceMethod, $resource := .Target.Resources }}
func (c *clientset) {{ $resourceMethod.Name }}() {{ GoType $resource.Type }}{
	return c.{{ ToLower $resourceMethod.Name }}
}
{{- end }}

{{- range $clientMethod, $client := .Target.Clients }}
func (c *clientset) {{ $clientMethod.Name }}() {{ GoType $client.Type }}{
	return c.{{ ToLower $clientMethod.Name }}
}
{{- end }}

func WrapWithMetrics(inner {{ GoType .Target }}, m metrics.MetricsConfigManager, clientType metrics.ClientType) {{ GoType .Target }} {
	return &clientset{
		{{- range $resourceMethod, $resource := .Target.Resources }}
		{{ ToLower $resourceMethod.Name }}: {{ ToLower $resourceMethod.Name }}.WithMetrics(inner.{{ $resourceMethod.Name }}(), metrics.ClusteredClientQueryRecorder(m, {{ Quote $resource.Kind }}, clientType)),
		{{- end }}
		{{- range $clientMethod, $client := .Target.Clients }}
		{{ ToLower $clientMethod.Name }}: {{ ToLower $clientMethod.Name }}.WithMetrics(inner.{{ $clientMethod.Name }}(), m, clientType),
		{{- end }}
	}
}

func WrapWithTracing(inner {{ GoType .Target }}) {{ GoType .Target }} {
	return &clientset{
		{{- range $resourceMethod, $resource := .Target.Resources }}
		{{ ToLower $resourceMethod.Name }}: {{ ToLower $resourceMethod.Name }}.WithTracing(inner.{{ $resourceMethod.Name }}(), {{ Quote $resourceMethod.Name }}, ""),
		{{- end }}
		{{- range $clientMethod, $client := .Target.Clients }}
		{{ ToLower $clientMethod.Name }}: {{ ToLower $clientMethod.Name }}.WithTracing(inner.{{ $clientMethod.Name }}(), {{ Quote $clientMethod.Name }}),
		{{- end }}
	}
}

func WrapWithLogging(inner {{ GoType .Target }}, logger logr.Logger) {{ GoType .Target }} {
	return &clientset{
		{{- range $resourceMethod, $resource := .Target.Resources }}
		{{ ToLower $resourceMethod.Name }}: {{ ToLower $resourceMethod.Name }}.WithLogging(inner.{{ $resourceMethod.Name }}(), logger.WithValues("group", {{ Quote $resourceMethod.Name }})),
		{{- end }}
		{{- range $clientMethod, $client := .Target.Clients }}
		{{ ToLower $clientMethod.Name }}: {{ ToLower $clientMethod.Name }}.WithLogging(inner.{{ $clientMethod.Name }}(), logger.WithValues("group", {{ Quote $clientMethod.Name }})),
		{{- end }}
	}
}
`
	interfaceTpl = `
package clientset

import (
	"github.com/go-logr/logr"
	"github.com/kyverno/kyverno/pkg/metrics"
	{{- range $package := Packages .Target.Type }}
	{{ Pkg $package }} {{ Quote $package }}
	{{- end }}
	{{- range $clientMethod, $client := .Target.Clients }}
	{{ ToLower $clientMethod.Name }} "github.com/kyverno/kyverno/{{ $.Folder }}/{{ ToLower $clientMethod.Name }}"
	{{- end }}
)

type UpstreamInterface = {{ GoType .Target.Type }}

type Interface interface {
	UpstreamInterface
	WithMetrics(metrics.MetricsConfigManager, metrics.ClientType) Interface
	WithTracing() Interface
	WithLogging(logr.Logger) Interface
}

func From(inner {{ GoType .Target }}, opts ...NewOption) Interface {
	i := from(inner)
	for _, opt := range opts {
		i = opt(i)
	}
	return i
}

type NewOption func (Interface) Interface

func WithMetrics(m metrics.MetricsConfigManager, t metrics.ClientType) NewOption {
	return func(i Interface) Interface {
		return i.WithMetrics(m, t)
	}
}

func WithTracing() NewOption {
	return func(i Interface) Interface {
		return i.WithTracing()
	}
}

func WithLogging(logger logr.Logger) NewOption {
	return func(i Interface) Interface {
		return i.WithLogging(logger)
	}
}

func NewForConfig(c *rest.Config, opts ...NewOption) (Interface, error) {
	inner, err := {{ Pkg .Target.Type.PkgPath }}.NewForConfig(c)
	if err != nil {
		return nil, err
	}
	return From(inner, opts...), nil
}

func NewForConfigAndClient(c *rest.Config, httpClient *http.Client, opts ...NewOption) (Interface, error) {
	inner, err := {{ Pkg .Target.Type.PkgPath }}.NewForConfigAndClient(c, httpClient)
	if err != nil {
		return nil, err
	}
	return From(inner, opts...), nil
}

func NewForConfigOrDie(c *rest.Config, opts ...NewOption) Interface {
	return From({{ Pkg .Target.Type.PkgPath }}.NewForConfigOrDie(c), opts...)
}

type wrapper struct {
	{{ GoType .Target.Type }}
}

func from(inner {{ GoType .Target }}, opts ...NewOption) Interface {
	return &wrapper{inner}
}

func (i *wrapper) WithMetrics(m metrics.MetricsConfigManager, t metrics.ClientType) Interface {
	return from(WrapWithMetrics(i, m, t))
}

func (i *wrapper) WithTracing() Interface {
	return from(WrapWithTracing(i))
}

func (i *wrapper) WithLogging(logger logr.Logger) Interface {
	return from(WrapWithLogging(i, logger))
}
`
)

var (
	matchFirstCap = regexp.MustCompile("(.)([A-Z][a-z]+)")
	matchAllCap   = regexp.MustCompile("([a-z0-9])([A-Z])")
)

type arg struct {
	reflect.Type
	IsVariadic bool
}

func (a arg) IsError() bool {
	return goType(a.Type) == "error"
}

type ret struct {
	reflect.Type
	IsLast bool
}

func (r ret) IsError() bool {
	return goType(r.Type) == "error"
}

type operation struct {
	reflect.Method
}

func (o operation) HasContext() bool {
	return o.Method.Type.NumIn() > 0 && goType(o.Method.Type.In(0)) == "context.Context"
}

func (o operation) HasError() bool {
	for _, out := range getOuts(o.Method) {
		if goType(out) == "error" {
			return true
		}
	}
	return false
}

type resource struct {
	reflect.Type
	Operations []operation
}

func (r resource) Kind() string {
	return strings.ReplaceAll(r.Type.Name(), "Interface", "")
}

type resourceKey reflect.Method

func (r resourceKey) IsNamespaced() bool {
	return r.Type.NumIn() == 1
}

type client struct {
	reflect.Type
	Resources map[resourceKey]resource
}

type clientset struct {
	reflect.Type
	Clients   map[reflect.Method]client
	Resources map[resourceKey]resource
}

func getIns(in reflect.Method) []reflect.Type {
	var out []reflect.Type
	for i := 0; i < in.Type.NumIn(); i++ {
		out = append(out, in.Type.In(i))
	}
	return out
}

func getOuts(in reflect.Method) []ret {
	var out []ret
	for i := 0; i < in.Type.NumOut(); i++ {
		out = append(out, ret{
			Type:   in.Type.Out(i),
			IsLast: i == in.Type.NumOut()-1,
		})
	}
	return out
}

func getMethods(in reflect.Type) []reflect.Method {
	var out []reflect.Method
	for i := 0; i < in.NumMethod(); i++ {
		out = append(out, in.Method(i))
	}
	return out
}

func packageAlias(in string) string {
	alias := in
	alias = strings.ReplaceAll(alias, ".", "_")
	alias = strings.ReplaceAll(alias, "-", "_")
	alias = strings.ReplaceAll(alias, "/", "_")
	return alias
}

func goType(in reflect.Type) string {
	switch in.Kind() {
	case reflect.Pointer:
		return "*" + goType(in.Elem())
	case reflect.Array:
		return "[]" + goType(in.Elem())
	case reflect.Slice:
		return "[]" + goType(in.Elem())
	case reflect.Map:
		return "map[" + goType(in.Key()) + "]" + goType(in.Elem())
	}
	pack := packageAlias(in.PkgPath())
	if pack == "" {
		return in.Name()
	}
	return pack + "." + in.Name()
}

func toSnakeCase(str string) string {
	snake := matchFirstCap.ReplaceAllString(str, "${1}_${2}")
	snake = matchAllCap.ReplaceAllString(snake, "${1}_${2}")
	return strings.ToLower(snake)
}

func parseResource(in reflect.Type) resource {
	r := resource{
		Type: in,
	}
	for _, operationMethod := range getMethods(in) {
		o := operation{
			Method: operationMethod,
		}
		r.Operations = append(r.Operations, o)
	}
	return r
}

func parseClient(in reflect.Type) client {
	c := client{
		Type:      in,
		Resources: map[resourceKey]resource{},
	}
	for _, resourceMethod := range getMethods(in) {
		// resource methods return only the resosurce interface type
		if resourceMethod.Type.NumOut() == 1 && resourceMethod.Name != "RESTClient" {
			resourceType := resourceMethod.Type.Out(0)
			r := resource{
				Type: resourceType,
			}
			for _, operationMethod := range getMethods(resourceType) {
				o := operation{
					Method: operationMethod,
				}
				r.Operations = append(r.Operations, o)
			}
			c.Resources[resourceKey(resourceMethod)] = r
		}
	}
	return c
}

func parseClientset(in reflect.Type) clientset {
	cs := clientset{
		Type:      in,
		Clients:   map[reflect.Method]client{},
		Resources: map[resourceKey]resource{},
	}
	for _, clientMethod := range getMethods(in) {
		// client methods return only the client interface type
		if clientMethod.Type.NumOut() == 1 && clientMethod.Name != "Discovery" {
			cs.Clients[clientMethod] = parseClient(clientMethod.Type.Out(0))
		} else if clientMethod.Name == "Discovery" {
			cs.Resources[resourceKey(clientMethod)] = parseResource(clientMethod.Type.Out(0))
		}
	}
	return cs
}

func parseImports(in reflect.Type) []string {
	imports := sets.New(in.PkgPath())
	for _, m := range getMethods(in) {
		for _, i := range getIns(m) {
			if i.Kind() == reflect.Pointer {
				i = i.Elem()
			}
			if i.PkgPath() != "" {
				imports.Insert(i.PkgPath())
			}
		}
		for _, i := range getOuts(m) {
			pkg := i.PkgPath()
			if i.Kind() == reflect.Pointer {
				pkg = i.Elem().PkgPath()
			}
			if pkg != "" {
				imports.Insert(pkg)
			}
		}
	}
	return sets.List(imports)
}

func executeTemplate(tpl string, data interface{}, folder string, file string) {
	tmpl := template.New("xxx")
	tmpl.Funcs(
		template.FuncMap{
			"ToLower": func(in string) string {
				return strings.ToLower(in)
			},
			"Quote": func(in string) string {
				return `"` + in + `"`
			},
			"SnakeCase": func(in string) string {
				return toSnakeCase(in)
			},
			"Args": func(in reflect.Method) []arg {
				var out []arg
				for i, a := range getIns(in) {
					out = append(out, arg{
						Type:       a,
						IsVariadic: in.Type.IsVariadic() && i == in.Type.NumIn()-1,
					})
				}
				return out
			},
			"Returns": func(in reflect.Method) []ret {
				return getOuts(in)
			},
			"Pkg": func(in string) string {
				return packageAlias(in)
			},
			"GoType": func(in reflect.Type) string {
				return goType(in)
			},
			"Packages": func(in reflect.Type) []string {
				return parseImports(in)
			},
		},
	)
	if tmpl, err := tmpl.Parse(tpl); err != nil {
		panic(err)
	} else {
		if err := os.MkdirAll(folder, 0o755); err != nil {
			panic(fmt.Sprintf("Failed to create directories for %s", folder))
		}
		f, err := os.Create(path.Join(folder, file))
		if err != nil {
			panic(fmt.Sprintf("Failed to create file %s", path.Join(folder, file)))
		}
		if err := tmpl.Execute(f, map[string]interface{}{
			"Folder": folder,
			"Target": data,
		}); err != nil {
			panic(err)
		}
	}
}

func generateResource(r resource, folder string) {
	executeTemplate(resourceTpl, r, folder, "resource.generated.go")
}

func generateClient(c client, folder string) {
	executeTemplate(clientTpl, c, folder, "client.generated.go")
	for m, r := range c.Resources {
		generateResource(r, path.Join(folder, strings.ToLower(m.Name)))
	}
}

func generateClientset(cs clientset, folder string) {
	executeTemplate(clientsetTpl, cs, folder, "clientset.generated.go")
	for m, c := range cs.Clients {
		generateClient(c, path.Join(folder, strings.ToLower(m.Name)))
	}
	for m, r := range cs.Resources {
		generateResource(r, path.Join(folder, strings.ToLower(m.Name)))
	}
}

func generateInterface(cs clientset, folder string) {
	executeTemplate(interfaceTpl, cs, folder, "interface.generated.go")
}

func main() {
	kube := parseClientset(reflect.TypeOf((*kubernetes.Interface)(nil)).Elem())
	generateClientset(kube, "pkg/clients/kube")
	generateInterface(kube, "pkg/clients/kube")
	kyverno := parseClientset(reflect.TypeOf((*versioned.Interface)(nil)).Elem())
	generateClientset(kyverno, "pkg/clients/kyverno")
	generateInterface(kyverno, "pkg/clients/kyverno")
	dynamicInterface := parseClientset(reflect.TypeOf((*dynamic.Interface)(nil)).Elem())
	dynamicResource := parseResource(reflect.TypeOf((*dynamic.ResourceInterface)(nil)).Elem())
	generateResource(dynamicResource, "pkg/clients/dynamic/resource")
	generateInterface(dynamicInterface, "pkg/clients/dynamic")
	metadataInterface := parseClientset(reflect.TypeOf((*metadata.Interface)(nil)).Elem())
	metadataResource := parseResource(reflect.TypeOf((*metadata.ResourceInterface)(nil)).Elem())
	generateInterface(metadataInterface, "pkg/clients/metadata")
	generateResource(metadataResource, "pkg/clients/metadata/resource")
	apiserverInterface := parseClientset(reflect.TypeOf((*apiserver.Interface)(nil)).Elem())
	generateClientset(apiserverInterface, "pkg/clients/apiserver")
	generateInterface(apiserverInterface, "pkg/clients/apiserver")
	aggregatorInterface := parseClientset(reflect.TypeOf((*aggregator.Interface)(nil)).Elem())
	generateClientset(aggregatorInterface, "pkg/clients/aggregator")
	generateInterface(aggregatorInterface, "pkg/clients/aggregator")
}