2020-03-26 21:07:15 +00:00
|
|
|
package patch
|
|
|
|
|
|
|
|
import (
|
|
|
|
"sync"
|
|
|
|
|
|
|
|
"github.com/rancher/wrangler/pkg/gvk"
|
|
|
|
"k8s.io/apimachinery/pkg/runtime"
|
|
|
|
"k8s.io/apimachinery/pkg/runtime/schema"
|
|
|
|
"k8s.io/apimachinery/pkg/types"
|
|
|
|
"k8s.io/apimachinery/pkg/util/strategicpatch"
|
|
|
|
"k8s.io/client-go/kubernetes/scheme"
|
|
|
|
)
|
|
|
|
|
|
|
|
var (
|
|
|
|
patchCache = map[schema.GroupVersionKind]patchCacheEntry{}
|
|
|
|
patchCacheLock = sync.Mutex{}
|
|
|
|
)
|
|
|
|
|
|
|
|
type patchCacheEntry struct {
|
|
|
|
patchType types.PatchType
|
|
|
|
lookup strategicpatch.LookupPatchMeta
|
|
|
|
}
|
|
|
|
|
|
|
|
func isJSONPatch(patch []byte) bool {
|
|
|
|
// a JSON patch is a list
|
|
|
|
return len(patch) > 0 && patch[0] == '['
|
|
|
|
}
|
|
|
|
|
|
|
|
func GetPatchStyle(original, patch []byte) (types.PatchType, strategicpatch.LookupPatchMeta, error) {
|
|
|
|
if isJSONPatch(patch) {
|
|
|
|
return types.JSONPatchType, nil, nil
|
|
|
|
}
|
|
|
|
gvk, ok, err := gvk.Detect(original)
|
|
|
|
if err != nil {
|
|
|
|
return "", nil, err
|
|
|
|
}
|
|
|
|
if !ok {
|
|
|
|
return types.MergePatchType, nil, nil
|
|
|
|
}
|
|
|
|
return GetMergeStyle(gvk)
|
|
|
|
}
|
|
|
|
|
|
|
|
func GetMergeStyle(gvk schema.GroupVersionKind) (types.PatchType, strategicpatch.LookupPatchMeta, error) {
|
|
|
|
var (
|
|
|
|
patchType types.PatchType
|
|
|
|
lookupPatchMeta strategicpatch.LookupPatchMeta
|
|
|
|
)
|
|
|
|
|
|
|
|
patchCacheLock.Lock()
|
|
|
|
entry, ok := patchCache[gvk]
|
|
|
|
patchCacheLock.Unlock()
|
|
|
|
|
|
|
|
if ok {
|
|
|
|
return entry.patchType, entry.lookup, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
versionedObject, err := scheme.Scheme.New(gvk)
|
|
|
|
|
2021-07-02 22:06:02 +00:00
|
|
|
if runtime.IsNotRegisteredError(err) || gvk.Kind == "CustomResourceDefinition" {
|
2020-03-26 21:07:15 +00:00
|
|
|
patchType = types.MergePatchType
|
|
|
|
} else if err != nil {
|
|
|
|
return patchType, nil, err
|
|
|
|
} else {
|
|
|
|
patchType = types.StrategicMergePatchType
|
|
|
|
lookupPatchMeta, err = strategicpatch.NewPatchMetaFromStruct(versionedObject)
|
|
|
|
if err != nil {
|
|
|
|
return patchType, nil, err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
patchCacheLock.Lock()
|
|
|
|
patchCache[gvk] = patchCacheEntry{
|
|
|
|
patchType: patchType,
|
|
|
|
lookup: lookupPatchMeta,
|
|
|
|
}
|
|
|
|
patchCacheLock.Unlock()
|
|
|
|
|
|
|
|
return patchType, lookupPatchMeta, nil
|
|
|
|
}
|