123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163 |
- /*
- Copyright 2017 The Kubernetes Authors.
- Licensed under the Apache License, Version 2.0 (the "License");
- you may not use this file except in compliance with the License.
- You may obtain a copy of the License at
- http://www.apache.org/licenses/LICENSE-2.0
- Unless required by applicable law or agreed to in writing, software
- distributed under the License is distributed on an "AS IS" BASIS,
- WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- See the License for the specific language governing permissions and
- limitations under the License.
- */
- package strategy
- import (
- "fmt"
- "k8s.io/kubernetes/pkg/kubectl/apply"
- )
- func createMergeStrategy(options Options, strategic *delegatingStrategy) mergeStrategy {
- return mergeStrategy{
- strategic,
- options,
- }
- }
- // mergeStrategy merges the values in an Element into a single Result
- type mergeStrategy struct {
- strategic *delegatingStrategy
- options Options
- }
- // MergeList merges the lists in a ListElement into a single Result
- func (v mergeStrategy) MergeList(e apply.ListElement) (apply.Result, error) {
- // No merge logic if adding or deleting a field
- if result, done := v.doAddOrDelete(e); done {
- return result, nil
- }
- // Detect conflict in ListElement
- if err := v.doConflictDetect(e); err != nil {
- return apply.Result{}, err
- }
- // Merge each item in the list and append it to the list
- merged := []interface{}{}
- for _, value := range e.Values {
- // Recursively merge the list element before adding the value to the list
- m, err := value.Merge(v.strategic)
- if err != nil {
- return apply.Result{}, err
- }
- switch m.Operation {
- case apply.SET:
- // Keep the list item value
- merged = append(merged, m.MergedResult)
- case apply.DROP:
- // Drop the list item value
- default:
- panic(fmt.Errorf("Unexpected result operation type %+v", m))
- }
- }
- if len(merged) == 0 {
- // If the list is empty, return a nil entry
- return apply.Result{Operation: apply.SET, MergedResult: nil}, nil
- }
- // Return the merged list, and tell the caller to keep it
- return apply.Result{Operation: apply.SET, MergedResult: merged}, nil
- }
- // MergeMap merges the maps in a MapElement into a single Result
- func (v mergeStrategy) MergeMap(e apply.MapElement) (apply.Result, error) {
- // No merge logic if adding or deleting a field
- if result, done := v.doAddOrDelete(e); done {
- return result, nil
- }
- // Detect conflict in MapElement
- if err := v.doConflictDetect(e); err != nil {
- return apply.Result{}, err
- }
- return v.doMergeMap(e.GetValues())
- }
- // MergeMap merges the type instances in a TypeElement into a single Result
- func (v mergeStrategy) MergeType(e apply.TypeElement) (apply.Result, error) {
- // No merge logic if adding or deleting a field
- if result, done := v.doAddOrDelete(e); done {
- return result, nil
- }
- // Detect conflict in TypeElement
- if err := v.doConflictDetect(e); err != nil {
- return apply.Result{}, err
- }
- return v.doMergeMap(e.GetValues())
- }
- // do merges a recorded, local and remote map into a new object
- func (v mergeStrategy) doMergeMap(e map[string]apply.Element) (apply.Result, error) {
- // Merge each item in the list
- merged := map[string]interface{}{}
- for key, value := range e {
- // Recursively merge the map element before adding the value to the map
- result, err := value.Merge(v.strategic)
- if err != nil {
- return apply.Result{}, err
- }
- switch result.Operation {
- case apply.SET:
- // Keep the map item value
- merged[key] = result.MergedResult
- case apply.DROP:
- // Drop the map item value
- default:
- panic(fmt.Errorf("Unexpected result operation type %+v", result))
- }
- }
- // Return the merged map, and tell the caller to keep it
- if len(merged) == 0 {
- // Special case the empty map to set the field value to nil, but keep the field key
- // This is how the tests expect the structures to look when parsed from yaml
- return apply.Result{Operation: apply.SET, MergedResult: nil}, nil
- }
- return apply.Result{Operation: apply.SET, MergedResult: merged}, nil
- }
- func (v mergeStrategy) doAddOrDelete(e apply.Element) (apply.Result, bool) {
- if apply.IsAdd(e) {
- return apply.Result{Operation: apply.SET, MergedResult: e.GetLocal()}, true
- }
- // Delete the List
- if apply.IsDrop(e) {
- return apply.Result{Operation: apply.DROP}, true
- }
- return apply.Result{}, false
- }
- // MergePrimitive returns and error. Primitive elements can't be merged, only replaced.
- func (v mergeStrategy) MergePrimitive(diff apply.PrimitiveElement) (apply.Result, error) {
- return apply.Result{}, fmt.Errorf("Cannot merge primitive element %v", diff.Name)
- }
- // MergeEmpty returns an empty result
- func (v mergeStrategy) MergeEmpty(diff apply.EmptyElement) (apply.Result, error) {
- return apply.Result{Operation: apply.SET}, nil
- }
- // doConflictDetect returns error if element has conflict
- func (v mergeStrategy) doConflictDetect(e apply.Element) error {
- return v.strategic.doConflictDetect(e)
- }
- var _ apply.Strategy = &mergeStrategy{}
|