2019-05-09 22:03:45 +00:00
|
|
|
package apply
|
2019-01-22 20:53:35 +00:00
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
|
|
|
"compress/gzip"
|
|
|
|
"encoding/base64"
|
|
|
|
"io/ioutil"
|
2020-04-22 22:34:19 +00:00
|
|
|
"strings"
|
|
|
|
|
2021-07-02 22:06:02 +00:00
|
|
|
jsonpatch "github.com/evanphx/json-patch"
|
2019-01-22 20:53:35 +00:00
|
|
|
"github.com/pkg/errors"
|
2021-07-02 22:06:02 +00:00
|
|
|
data2 "github.com/rancher/wrangler/pkg/data"
|
2020-03-26 21:07:15 +00:00
|
|
|
"github.com/rancher/wrangler/pkg/data/convert"
|
2021-07-02 22:06:02 +00:00
|
|
|
"github.com/rancher/wrangler/pkg/objectset"
|
2020-03-26 21:07:15 +00:00
|
|
|
patch2 "github.com/rancher/wrangler/pkg/patch"
|
2019-01-22 20:53:35 +00:00
|
|
|
"github.com/sirupsen/logrus"
|
|
|
|
"k8s.io/apimachinery/pkg/api/meta"
|
2019-12-12 01:27:03 +00:00
|
|
|
v1 "k8s.io/apimachinery/pkg/apis/meta/v1"
|
2019-01-22 20:53:35 +00:00
|
|
|
"k8s.io/apimachinery/pkg/apis/meta/v1/unstructured"
|
|
|
|
"k8s.io/apimachinery/pkg/runtime"
|
|
|
|
"k8s.io/apimachinery/pkg/runtime/schema"
|
|
|
|
"k8s.io/apimachinery/pkg/types"
|
|
|
|
"k8s.io/apimachinery/pkg/util/json"
|
|
|
|
"k8s.io/apimachinery/pkg/util/jsonmergepatch"
|
|
|
|
"k8s.io/apimachinery/pkg/util/strategicpatch"
|
2019-05-09 22:03:45 +00:00
|
|
|
"k8s.io/client-go/dynamic"
|
2019-01-22 20:53:35 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
const (
|
|
|
|
LabelApplied = "objectset.rio.cattle.io/applied"
|
|
|
|
)
|
|
|
|
|
2021-07-02 22:06:02 +00:00
|
|
|
var (
|
|
|
|
knownListKeys = map[string]bool{
|
|
|
|
"apiVersion": true,
|
|
|
|
"containerPort": true,
|
|
|
|
"devicePath": true,
|
|
|
|
"ip": true,
|
|
|
|
"kind": true,
|
|
|
|
"mountPath": true,
|
|
|
|
"name": true,
|
|
|
|
"port": true,
|
|
|
|
"topologyKey": true,
|
|
|
|
"type": true,
|
|
|
|
}
|
|
|
|
)
|
|
|
|
|
2019-12-12 01:27:03 +00:00
|
|
|
func prepareObjectForCreate(gvk schema.GroupVersionKind, obj runtime.Object) (runtime.Object, error) {
|
2021-07-02 22:06:02 +00:00
|
|
|
serialized, err := serializeApplied(obj)
|
2019-01-22 20:53:35 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
obj = obj.DeepCopyObject()
|
2019-12-12 01:27:03 +00:00
|
|
|
m, err := meta.Accessor(obj)
|
2019-01-22 20:53:35 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2019-12-12 01:27:03 +00:00
|
|
|
annotations := m.GetAnnotations()
|
2019-01-22 20:53:35 +00:00
|
|
|
if annotations == nil {
|
|
|
|
annotations = map[string]string{}
|
|
|
|
}
|
|
|
|
|
|
|
|
annotations[LabelApplied] = appliedToAnnotation(serialized)
|
2019-12-12 01:27:03 +00:00
|
|
|
m.SetAnnotations(annotations)
|
|
|
|
|
|
|
|
typed, err := meta.TypeAccessor(obj)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
apiVersion, kind := gvk.ToAPIVersionAndKind()
|
|
|
|
typed.SetAPIVersion(apiVersion)
|
|
|
|
typed.SetKind(kind)
|
2019-01-22 20:53:35 +00:00
|
|
|
|
|
|
|
return obj, nil
|
|
|
|
}
|
|
|
|
|
2019-12-12 01:27:03 +00:00
|
|
|
func originalAndModified(gvk schema.GroupVersionKind, oldMetadata v1.Object, newObject runtime.Object) ([]byte, []byte, error) {
|
2020-03-26 21:07:15 +00:00
|
|
|
original, err := getOriginalBytes(gvk, oldMetadata)
|
2019-01-22 20:53:35 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, nil, err
|
|
|
|
}
|
|
|
|
|
2019-12-12 01:27:03 +00:00
|
|
|
newObject, err = prepareObjectForCreate(gvk, newObject)
|
2019-01-22 20:53:35 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
modified, err := json.Marshal(newObject)
|
|
|
|
|
|
|
|
return original, modified, err
|
|
|
|
}
|
|
|
|
|
2019-05-09 22:03:45 +00:00
|
|
|
func emptyMaps(data map[string]interface{}, keys ...string) bool {
|
|
|
|
for _, key := range append(keys, "__invalid_key__") {
|
|
|
|
if len(data) == 0 {
|
|
|
|
// map is empty so all children are empty too
|
2019-01-22 20:53:35 +00:00
|
|
|
return true
|
2019-05-09 22:03:45 +00:00
|
|
|
} else if len(data) > 1 {
|
|
|
|
// map has more than one key so not empty
|
|
|
|
return false
|
2019-01-22 20:53:35 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
value, ok := data[key]
|
|
|
|
if !ok {
|
2019-05-09 22:03:45 +00:00
|
|
|
// map has one key but not what we are expecting so not considered empty
|
2019-01-22 20:53:35 +00:00
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
2020-03-26 21:07:15 +00:00
|
|
|
data = convert.ToMapInterface(value)
|
2019-01-22 20:53:35 +00:00
|
|
|
}
|
|
|
|
|
2019-05-09 22:03:45 +00:00
|
|
|
return true
|
2019-01-22 20:53:35 +00:00
|
|
|
}
|
|
|
|
|
2020-04-22 22:34:19 +00:00
|
|
|
func sanitizePatch(patch []byte, removeObjectSetAnnotation bool) ([]byte, error) {
|
2019-01-22 20:53:35 +00:00
|
|
|
mod := false
|
|
|
|
data := map[string]interface{}{}
|
|
|
|
err := json.Unmarshal(patch, &data)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
if _, ok := data["kind"]; ok {
|
|
|
|
mod = true
|
|
|
|
delete(data, "kind")
|
|
|
|
}
|
|
|
|
|
|
|
|
if _, ok := data["apiVersion"]; ok {
|
|
|
|
mod = true
|
|
|
|
delete(data, "apiVersion")
|
|
|
|
}
|
|
|
|
|
2021-07-02 22:06:02 +00:00
|
|
|
if _, ok := data["status"]; ok {
|
|
|
|
mod = true
|
|
|
|
delete(data, "status")
|
|
|
|
}
|
|
|
|
|
2019-01-22 20:53:35 +00:00
|
|
|
if deleted := removeCreationTimestamp(data); deleted {
|
|
|
|
mod = true
|
|
|
|
}
|
|
|
|
|
2020-04-22 22:34:19 +00:00
|
|
|
if removeObjectSetAnnotation {
|
|
|
|
metadata := convert.ToMapInterface(data2.GetValueN(data, "metadata"))
|
|
|
|
annotations := convert.ToMapInterface(data2.GetValueN(data, "metadata", "annotations"))
|
|
|
|
for k := range annotations {
|
|
|
|
if strings.HasPrefix(k, LabelPrefix) {
|
|
|
|
mod = true
|
|
|
|
delete(annotations, k)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if mod && len(annotations) == 0 {
|
|
|
|
delete(metadata, "annotations")
|
|
|
|
if len(metadata) == 0 {
|
|
|
|
delete(data, "metadata")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-05-09 22:03:45 +00:00
|
|
|
if emptyMaps(data, "metadata", "annotations") {
|
2019-01-22 20:53:35 +00:00
|
|
|
return []byte("{}"), nil
|
|
|
|
}
|
|
|
|
|
|
|
|
if !mod {
|
|
|
|
return patch, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
return json.Marshal(data)
|
|
|
|
}
|
|
|
|
|
2021-07-02 22:06:02 +00:00
|
|
|
func applyPatch(gvk schema.GroupVersionKind, reconciler Reconciler, patcher Patcher, debugID string, ignoreOriginal bool, oldObject, newObject runtime.Object, diffPatches [][]byte) (bool, error) {
|
2019-01-22 20:53:35 +00:00
|
|
|
oldMetadata, err := meta.Accessor(oldObject)
|
|
|
|
if err != nil {
|
|
|
|
return false, err
|
|
|
|
}
|
|
|
|
|
2019-12-12 01:27:03 +00:00
|
|
|
original, modified, err := originalAndModified(gvk, oldMetadata, newObject)
|
2019-01-22 20:53:35 +00:00
|
|
|
if err != nil {
|
|
|
|
return false, err
|
|
|
|
}
|
2021-07-02 22:06:02 +00:00
|
|
|
|
|
|
|
if ignoreOriginal {
|
|
|
|
original = nil
|
|
|
|
}
|
|
|
|
|
2019-01-22 20:53:35 +00:00
|
|
|
current, err := json.Marshal(oldObject)
|
|
|
|
if err != nil {
|
|
|
|
return false, err
|
|
|
|
}
|
|
|
|
|
2021-07-02 22:06:02 +00:00
|
|
|
patchType, patch, err := doPatch(gvk, original, modified, current, diffPatches)
|
2019-01-22 20:53:35 +00:00
|
|
|
if err != nil {
|
|
|
|
return false, errors.Wrap(err, "patch generation")
|
|
|
|
}
|
|
|
|
|
|
|
|
if string(patch) == "{}" {
|
|
|
|
return false, nil
|
|
|
|
}
|
|
|
|
|
2020-04-22 22:34:19 +00:00
|
|
|
patch, err = sanitizePatch(patch, false)
|
2019-01-22 20:53:35 +00:00
|
|
|
if err != nil {
|
|
|
|
return false, err
|
|
|
|
}
|
|
|
|
|
|
|
|
if string(patch) == "{}" {
|
|
|
|
return false, nil
|
|
|
|
}
|
|
|
|
|
2020-03-26 21:07:15 +00:00
|
|
|
logrus.Debugf("DesiredSet - Patch %s %s/%s for %s -- [%s, %s, %s, %s]", gvk, oldMetadata.GetNamespace(), oldMetadata.GetName(), debugID, patch, original, modified, current)
|
|
|
|
|
2019-12-12 01:27:03 +00:00
|
|
|
if reconciler != nil {
|
|
|
|
newObject, err := prepareObjectForCreate(gvk, newObject)
|
|
|
|
if err != nil {
|
|
|
|
return false, err
|
|
|
|
}
|
2020-03-26 21:07:15 +00:00
|
|
|
originalObject, err := getOriginalObject(gvk, oldMetadata)
|
|
|
|
if err != nil {
|
|
|
|
return false, err
|
|
|
|
}
|
2020-04-22 22:34:19 +00:00
|
|
|
if originalObject == nil {
|
|
|
|
originalObject = oldObject
|
|
|
|
}
|
2020-03-26 21:07:15 +00:00
|
|
|
handled, err := reconciler(originalObject, newObject)
|
2019-12-12 01:27:03 +00:00
|
|
|
if err != nil {
|
|
|
|
return false, err
|
|
|
|
}
|
|
|
|
if handled {
|
|
|
|
return true, nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-01-22 20:53:35 +00:00
|
|
|
logrus.Debugf("DesiredSet - Updated %s %s/%s for %s -- %s %s", gvk, oldMetadata.GetNamespace(), oldMetadata.GetName(), debugID, patchType, patch)
|
2019-05-09 22:03:45 +00:00
|
|
|
_, err = patcher(oldMetadata.GetNamespace(), oldMetadata.GetName(), patchType, patch)
|
2019-01-22 20:53:35 +00:00
|
|
|
|
|
|
|
return true, err
|
|
|
|
}
|
|
|
|
|
2020-04-22 22:34:19 +00:00
|
|
|
func (o *desiredSet) compareObjects(gvk schema.GroupVersionKind, reconciler Reconciler, patcher Patcher, client dynamic.NamespaceableResourceInterface, debugID string, oldObject, newObject runtime.Object, force bool) error {
|
2019-01-22 20:53:35 +00:00
|
|
|
oldMetadata, err := meta.Accessor(oldObject)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2020-04-22 22:34:19 +00:00
|
|
|
if o.createPlan {
|
|
|
|
o.plan.Objects = append(o.plan.Objects, oldObject)
|
|
|
|
}
|
|
|
|
|
2021-07-02 22:06:02 +00:00
|
|
|
diffPatches := o.diffPatches[patchKey{
|
|
|
|
GroupVersionKind: gvk,
|
|
|
|
ObjectKey: objectset.ObjectKey{
|
|
|
|
Namespace: oldMetadata.GetNamespace(),
|
|
|
|
Name: oldMetadata.GetName(),
|
|
|
|
},
|
|
|
|
}]
|
|
|
|
diffPatches = append(diffPatches, o.diffPatches[patchKey{
|
|
|
|
GroupVersionKind: gvk,
|
|
|
|
}]...)
|
|
|
|
|
|
|
|
if ran, err := applyPatch(gvk, reconciler, patcher, debugID, o.ignorePreviousApplied, oldObject, newObject, diffPatches); err != nil {
|
2019-01-22 20:53:35 +00:00
|
|
|
return err
|
|
|
|
} else if !ran {
|
|
|
|
logrus.Debugf("DesiredSet - No change(2) %s %s/%s for %s", gvk, oldMetadata.GetNamespace(), oldMetadata.GetName(), debugID)
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func removeCreationTimestamp(data map[string]interface{}) bool {
|
|
|
|
metadata, ok := data["metadata"]
|
|
|
|
if !ok {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
2020-03-26 21:07:15 +00:00
|
|
|
data = convert.ToMapInterface(metadata)
|
2019-01-22 20:53:35 +00:00
|
|
|
if _, ok := data["creationTimestamp"]; ok {
|
|
|
|
delete(data, "creationTimestamp")
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
2020-03-26 21:07:15 +00:00
|
|
|
func getOriginalObject(gvk schema.GroupVersionKind, obj v1.Object) (runtime.Object, error) {
|
2019-01-22 20:53:35 +00:00
|
|
|
original := appliedFromAnnotation(obj.GetAnnotations()[LabelApplied])
|
|
|
|
if len(original) == 0 {
|
2020-03-26 21:07:15 +00:00
|
|
|
return nil, nil
|
2019-01-22 20:53:35 +00:00
|
|
|
}
|
|
|
|
|
2019-07-12 17:13:20 +00:00
|
|
|
mapObj := map[string]interface{}{}
|
|
|
|
err := json.Unmarshal(original, &mapObj)
|
2019-01-22 20:53:35 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2019-07-12 17:13:20 +00:00
|
|
|
removeCreationTimestamp(mapObj)
|
2020-03-26 21:07:15 +00:00
|
|
|
return prepareObjectForCreate(gvk, &unstructured.Unstructured{
|
2019-07-12 17:13:20 +00:00
|
|
|
Object: mapObj,
|
2020-03-26 21:07:15 +00:00
|
|
|
})
|
|
|
|
}
|
2019-07-12 17:13:20 +00:00
|
|
|
|
2020-03-26 21:07:15 +00:00
|
|
|
func getOriginalBytes(gvk schema.GroupVersionKind, obj v1.Object) ([]byte, error) {
|
|
|
|
objCopy, err := getOriginalObject(gvk, obj)
|
2019-01-22 20:53:35 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2020-03-26 21:07:15 +00:00
|
|
|
if objCopy == nil {
|
|
|
|
return []byte("{}"), nil
|
|
|
|
}
|
2019-01-22 20:53:35 +00:00
|
|
|
return json.Marshal(objCopy)
|
|
|
|
}
|
|
|
|
|
|
|
|
func appliedFromAnnotation(str string) []byte {
|
|
|
|
if len(str) == 0 || str[0] == '{' {
|
|
|
|
return []byte(str)
|
|
|
|
}
|
|
|
|
|
|
|
|
b, err := base64.RawStdEncoding.DecodeString(str)
|
|
|
|
if err != nil {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
r, err := gzip.NewReader(bytes.NewBuffer(b))
|
|
|
|
if err != nil {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
b, err = ioutil.ReadAll(r)
|
|
|
|
if err != nil {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
return b
|
|
|
|
}
|
|
|
|
|
2021-07-02 22:06:02 +00:00
|
|
|
func pruneList(data []interface{}) []interface{} {
|
|
|
|
result := make([]interface{}, 0, len(data))
|
|
|
|
for _, v := range data {
|
|
|
|
switch typed := v.(type) {
|
|
|
|
case map[string]interface{}:
|
|
|
|
result = append(result, pruneValues(typed, true))
|
|
|
|
case []interface{}:
|
|
|
|
result = append(result, pruneList(typed))
|
|
|
|
default:
|
|
|
|
result = append(result, v)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return result
|
|
|
|
}
|
|
|
|
|
|
|
|
func pruneValues(data map[string]interface{}, isList bool) map[string]interface{} {
|
|
|
|
result := map[string]interface{}{}
|
|
|
|
for k, v := range data {
|
|
|
|
switch typed := v.(type) {
|
|
|
|
case map[string]interface{}:
|
|
|
|
result[k] = pruneValues(typed, false)
|
|
|
|
case []interface{}:
|
|
|
|
result[k] = pruneList(typed)
|
|
|
|
default:
|
|
|
|
if isList && knownListKeys[k] {
|
|
|
|
result[k] = v
|
|
|
|
} else {
|
|
|
|
switch x := v.(type) {
|
|
|
|
case string:
|
|
|
|
if len(x) > 64 {
|
|
|
|
result[k] = x[:64]
|
|
|
|
} else {
|
|
|
|
result[k] = v
|
|
|
|
}
|
|
|
|
case []byte:
|
|
|
|
result[k] = nil
|
|
|
|
default:
|
|
|
|
result[k] = v
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return result
|
|
|
|
}
|
|
|
|
|
|
|
|
func serializeApplied(obj runtime.Object) ([]byte, error) {
|
|
|
|
data, err := convert.EncodeToMap(obj)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
2019-01-22 20:53:35 +00:00
|
|
|
}
|
2021-07-02 22:06:02 +00:00
|
|
|
data = pruneValues(data, false)
|
|
|
|
return json.Marshal(data)
|
|
|
|
}
|
|
|
|
|
|
|
|
func appliedToAnnotation(b []byte) string {
|
2019-01-22 20:53:35 +00:00
|
|
|
buf := &bytes.Buffer{}
|
|
|
|
w := gzip.NewWriter(buf)
|
|
|
|
if _, err := w.Write(b); err != nil {
|
|
|
|
return string(b)
|
|
|
|
}
|
|
|
|
if err := w.Close(); err != nil {
|
|
|
|
return string(b)
|
|
|
|
}
|
|
|
|
return base64.RawStdEncoding.EncodeToString(buf.Bytes())
|
|
|
|
}
|
|
|
|
|
2021-07-02 22:06:02 +00:00
|
|
|
func stripIgnores(original, modified, current []byte, patches [][]byte) ([]byte, []byte, []byte, error) {
|
|
|
|
for _, patch := range patches {
|
|
|
|
patch, err := jsonpatch.DecodePatch(patch)
|
|
|
|
if err != nil {
|
|
|
|
return nil, nil, nil, err
|
|
|
|
}
|
|
|
|
if len(original) > 0 {
|
|
|
|
b, err := patch.Apply(original)
|
|
|
|
if err == nil {
|
|
|
|
original = b
|
|
|
|
}
|
|
|
|
}
|
|
|
|
b, err := patch.Apply(modified)
|
|
|
|
if err == nil {
|
|
|
|
modified = b
|
|
|
|
}
|
|
|
|
b, err = patch.Apply(current)
|
|
|
|
if err == nil {
|
|
|
|
current = b
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return original, modified, current, nil
|
|
|
|
}
|
|
|
|
|
2019-01-22 20:53:35 +00:00
|
|
|
// doPatch is adapted from "kubectl apply"
|
2021-07-02 22:06:02 +00:00
|
|
|
func doPatch(gvk schema.GroupVersionKind, original, modified, current []byte, diffPatch [][]byte) (types.PatchType, []byte, error) {
|
|
|
|
var (
|
|
|
|
patchType types.PatchType
|
|
|
|
patch []byte
|
|
|
|
)
|
|
|
|
|
|
|
|
original, modified, current, err := stripIgnores(original, modified, current, diffPatch)
|
|
|
|
if err != nil {
|
|
|
|
return patchType, nil, err
|
|
|
|
}
|
2019-01-22 20:53:35 +00:00
|
|
|
|
2020-03-26 21:07:15 +00:00
|
|
|
patchType, lookupPatchMeta, err := patch2.GetMergeStyle(gvk)
|
2019-01-22 20:53:35 +00:00
|
|
|
if err != nil {
|
|
|
|
return patchType, nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
if patchType == types.StrategicMergePatchType {
|
|
|
|
patch, err = strategicpatch.CreateThreeWayMergePatch(original, modified, current, lookupPatchMeta, true)
|
|
|
|
} else {
|
|
|
|
patch, err = jsonmergepatch.CreateThreeWayJSONMergePatch(original, modified, current)
|
|
|
|
}
|
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
logrus.Errorf("Failed to calcuated patch: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
return patchType, patch, err
|
|
|
|
}
|