mirror of
				https://github.com/k3s-io/kubernetes.git
				synced 2025-10-22 06:59:03 +00:00 
			
		
		
		
	Add structured-merge-diff vendor
This is where most of the logic actually lives.
This commit is contained in:
		
							
								
								
									
										201
									
								
								vendor/sigs.k8s.io/structured-merge-diff/LICENSE
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										201
									
								
								vendor/sigs.k8s.io/structured-merge-diff/LICENSE
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,201 @@ | |||||||
|  |                                  Apache License | ||||||
|  |                            Version 2.0, January 2004 | ||||||
|  |                         http://www.apache.org/licenses/ | ||||||
|  |  | ||||||
|  |    TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION | ||||||
|  |  | ||||||
|  |    1. Definitions. | ||||||
|  |  | ||||||
|  |       "License" shall mean the terms and conditions for use, reproduction, | ||||||
|  |       and distribution as defined by Sections 1 through 9 of this document. | ||||||
|  |  | ||||||
|  |       "Licensor" shall mean the copyright owner or entity authorized by | ||||||
|  |       the copyright owner that is granting the License. | ||||||
|  |  | ||||||
|  |       "Legal Entity" shall mean the union of the acting entity and all | ||||||
|  |       other entities that control, are controlled by, or are under common | ||||||
|  |       control with that entity. For the purposes of this definition, | ||||||
|  |       "control" means (i) the power, direct or indirect, to cause the | ||||||
|  |       direction or management of such entity, whether by contract or | ||||||
|  |       otherwise, or (ii) ownership of fifty percent (50%) or more of the | ||||||
|  |       outstanding shares, or (iii) beneficial ownership of such entity. | ||||||
|  |  | ||||||
|  |       "You" (or "Your") shall mean an individual or Legal Entity | ||||||
|  |       exercising permissions granted by this License. | ||||||
|  |  | ||||||
|  |       "Source" form shall mean the preferred form for making modifications, | ||||||
|  |       including but not limited to software source code, documentation | ||||||
|  |       source, and configuration files. | ||||||
|  |  | ||||||
|  |       "Object" form shall mean any form resulting from mechanical | ||||||
|  |       transformation or translation of a Source form, including but | ||||||
|  |       not limited to compiled object code, generated documentation, | ||||||
|  |       and conversions to other media types. | ||||||
|  |  | ||||||
|  |       "Work" shall mean the work of authorship, whether in Source or | ||||||
|  |       Object form, made available under the License, as indicated by a | ||||||
|  |       copyright notice that is included in or attached to the work | ||||||
|  |       (an example is provided in the Appendix below). | ||||||
|  |  | ||||||
|  |       "Derivative Works" shall mean any work, whether in Source or Object | ||||||
|  |       form, that is based on (or derived from) the Work and for which the | ||||||
|  |       editorial revisions, annotations, elaborations, or other modifications | ||||||
|  |       represent, as a whole, an original work of authorship. For the purposes | ||||||
|  |       of this License, Derivative Works shall not include works that remain | ||||||
|  |       separable from, or merely link (or bind by name) to the interfaces of, | ||||||
|  |       the Work and Derivative Works thereof. | ||||||
|  |  | ||||||
|  |       "Contribution" shall mean any work of authorship, including | ||||||
|  |       the original version of the Work and any modifications or additions | ||||||
|  |       to that Work or Derivative Works thereof, that is intentionally | ||||||
|  |       submitted to Licensor for inclusion in the Work by the copyright owner | ||||||
|  |       or by an individual or Legal Entity authorized to submit on behalf of | ||||||
|  |       the copyright owner. For the purposes of this definition, "submitted" | ||||||
|  |       means any form of electronic, verbal, or written communication sent | ||||||
|  |       to the Licensor or its representatives, including but not limited to | ||||||
|  |       communication on electronic mailing lists, source code control systems, | ||||||
|  |       and issue tracking systems that are managed by, or on behalf of, the | ||||||
|  |       Licensor for the purpose of discussing and improving the Work, but | ||||||
|  |       excluding communication that is conspicuously marked or otherwise | ||||||
|  |       designated in writing by the copyright owner as "Not a Contribution." | ||||||
|  |  | ||||||
|  |       "Contributor" shall mean Licensor and any individual or Legal Entity | ||||||
|  |       on behalf of whom a Contribution has been received by Licensor and | ||||||
|  |       subsequently incorporated within the Work. | ||||||
|  |  | ||||||
|  |    2. Grant of Copyright License. Subject to the terms and conditions of | ||||||
|  |       this License, each Contributor hereby grants to You a perpetual, | ||||||
|  |       worldwide, non-exclusive, no-charge, royalty-free, irrevocable | ||||||
|  |       copyright license to reproduce, prepare Derivative Works of, | ||||||
|  |       publicly display, publicly perform, sublicense, and distribute the | ||||||
|  |       Work and such Derivative Works in Source or Object form. | ||||||
|  |  | ||||||
|  |    3. Grant of Patent License. Subject to the terms and conditions of | ||||||
|  |       this License, each Contributor hereby grants to You a perpetual, | ||||||
|  |       worldwide, non-exclusive, no-charge, royalty-free, irrevocable | ||||||
|  |       (except as stated in this section) patent license to make, have made, | ||||||
|  |       use, offer to sell, sell, import, and otherwise transfer the Work, | ||||||
|  |       where such license applies only to those patent claims licensable | ||||||
|  |       by such Contributor that are necessarily infringed by their | ||||||
|  |       Contribution(s) alone or by combination of their Contribution(s) | ||||||
|  |       with the Work to which such Contribution(s) was submitted. If You | ||||||
|  |       institute patent litigation against any entity (including a | ||||||
|  |       cross-claim or counterclaim in a lawsuit) alleging that the Work | ||||||
|  |       or a Contribution incorporated within the Work constitutes direct | ||||||
|  |       or contributory patent infringement, then any patent licenses | ||||||
|  |       granted to You under this License for that Work shall terminate | ||||||
|  |       as of the date such litigation is filed. | ||||||
|  |  | ||||||
|  |    4. Redistribution. You may reproduce and distribute copies of the | ||||||
|  |       Work or Derivative Works thereof in any medium, with or without | ||||||
|  |       modifications, and in Source or Object form, provided that You | ||||||
|  |       meet the following conditions: | ||||||
|  |  | ||||||
|  |       (a) You must give any other recipients of the Work or | ||||||
|  |           Derivative Works a copy of this License; and | ||||||
|  |  | ||||||
|  |       (b) You must cause any modified files to carry prominent notices | ||||||
|  |           stating that You changed the files; and | ||||||
|  |  | ||||||
|  |       (c) You must retain, in the Source form of any Derivative Works | ||||||
|  |           that You distribute, all copyright, patent, trademark, and | ||||||
|  |           attribution notices from the Source form of the Work, | ||||||
|  |           excluding those notices that do not pertain to any part of | ||||||
|  |           the Derivative Works; and | ||||||
|  |  | ||||||
|  |       (d) If the Work includes a "NOTICE" text file as part of its | ||||||
|  |           distribution, then any Derivative Works that You distribute must | ||||||
|  |           include a readable copy of the attribution notices contained | ||||||
|  |           within such NOTICE file, excluding those notices that do not | ||||||
|  |           pertain to any part of the Derivative Works, in at least one | ||||||
|  |           of the following places: within a NOTICE text file distributed | ||||||
|  |           as part of the Derivative Works; within the Source form or | ||||||
|  |           documentation, if provided along with the Derivative Works; or, | ||||||
|  |           within a display generated by the Derivative Works, if and | ||||||
|  |           wherever such third-party notices normally appear. The contents | ||||||
|  |           of the NOTICE file are for informational purposes only and | ||||||
|  |           do not modify the License. You may add Your own attribution | ||||||
|  |           notices within Derivative Works that You distribute, alongside | ||||||
|  |           or as an addendum to the NOTICE text from the Work, provided | ||||||
|  |           that such additional attribution notices cannot be construed | ||||||
|  |           as modifying the License. | ||||||
|  |  | ||||||
|  |       You may add Your own copyright statement to Your modifications and | ||||||
|  |       may provide additional or different license terms and conditions | ||||||
|  |       for use, reproduction, or distribution of Your modifications, or | ||||||
|  |       for any such Derivative Works as a whole, provided Your use, | ||||||
|  |       reproduction, and distribution of the Work otherwise complies with | ||||||
|  |       the conditions stated in this License. | ||||||
|  |  | ||||||
|  |    5. Submission of Contributions. Unless You explicitly state otherwise, | ||||||
|  |       any Contribution intentionally submitted for inclusion in the Work | ||||||
|  |       by You to the Licensor shall be under the terms and conditions of | ||||||
|  |       this License, without any additional terms or conditions. | ||||||
|  |       Notwithstanding the above, nothing herein shall supersede or modify | ||||||
|  |       the terms of any separate license agreement you may have executed | ||||||
|  |       with Licensor regarding such Contributions. | ||||||
|  |  | ||||||
|  |    6. Trademarks. This License does not grant permission to use the trade | ||||||
|  |       names, trademarks, service marks, or product names of the Licensor, | ||||||
|  |       except as required for reasonable and customary use in describing the | ||||||
|  |       origin of the Work and reproducing the content of the NOTICE file. | ||||||
|  |  | ||||||
|  |    7. Disclaimer of Warranty. Unless required by applicable law or | ||||||
|  |       agreed to in writing, Licensor provides the Work (and each | ||||||
|  |       Contributor provides its Contributions) on an "AS IS" BASIS, | ||||||
|  |       WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or | ||||||
|  |       implied, including, without limitation, any warranties or conditions | ||||||
|  |       of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A | ||||||
|  |       PARTICULAR PURPOSE. You are solely responsible for determining the | ||||||
|  |       appropriateness of using or redistributing the Work and assume any | ||||||
|  |       risks associated with Your exercise of permissions under this License. | ||||||
|  |  | ||||||
|  |    8. Limitation of Liability. In no event and under no legal theory, | ||||||
|  |       whether in tort (including negligence), contract, or otherwise, | ||||||
|  |       unless required by applicable law (such as deliberate and grossly | ||||||
|  |       negligent acts) or agreed to in writing, shall any Contributor be | ||||||
|  |       liable to You for damages, including any direct, indirect, special, | ||||||
|  |       incidental, or consequential damages of any character arising as a | ||||||
|  |       result of this License or out of the use or inability to use the | ||||||
|  |       Work (including but not limited to damages for loss of goodwill, | ||||||
|  |       work stoppage, computer failure or malfunction, or any and all | ||||||
|  |       other commercial damages or losses), even if such Contributor | ||||||
|  |       has been advised of the possibility of such damages. | ||||||
|  |  | ||||||
|  |    9. Accepting Warranty or Additional Liability. While redistributing | ||||||
|  |       the Work or Derivative Works thereof, You may choose to offer, | ||||||
|  |       and charge a fee for, acceptance of support, warranty, indemnity, | ||||||
|  |       or other liability obligations and/or rights consistent with this | ||||||
|  |       License. However, in accepting such obligations, You may act only | ||||||
|  |       on Your own behalf and on Your sole responsibility, not on behalf | ||||||
|  |       of any other Contributor, and only if You agree to indemnify, | ||||||
|  |       defend, and hold each Contributor harmless for any liability | ||||||
|  |       incurred by, or claims asserted against, such Contributor by reason | ||||||
|  |       of your accepting any such warranty or additional liability. | ||||||
|  |  | ||||||
|  |    END OF TERMS AND CONDITIONS | ||||||
|  |  | ||||||
|  |    APPENDIX: How to apply the Apache License to your work. | ||||||
|  |  | ||||||
|  |       To apply the Apache License to your work, attach the following | ||||||
|  |       boilerplate notice, with the fields enclosed by brackets "{}" | ||||||
|  |       replaced with your own identifying information. (Don't include | ||||||
|  |       the brackets!)  The text should be enclosed in the appropriate | ||||||
|  |       comment syntax for the file format. We also recommend that a | ||||||
|  |       file or class name and description of purpose be included on the | ||||||
|  |       same "printed page" as the copyright notice for easier | ||||||
|  |       identification within third-party archives. | ||||||
|  |  | ||||||
|  |    Copyright {yyyy} {name of copyright owner} | ||||||
|  |  | ||||||
|  |    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. | ||||||
							
								
								
									
										31
									
								
								vendor/sigs.k8s.io/structured-merge-diff/fieldpath/BUILD
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										31
									
								
								vendor/sigs.k8s.io/structured-merge-diff/fieldpath/BUILD
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,31 @@ | |||||||
|  | load("@io_bazel_rules_go//go:def.bzl", "go_library") | ||||||
|  |  | ||||||
|  | go_library( | ||||||
|  |     name = "go_default_library", | ||||||
|  |     srcs = [ | ||||||
|  |         "doc.go", | ||||||
|  |         "element.go", | ||||||
|  |         "fromvalue.go", | ||||||
|  |         "managers.go", | ||||||
|  |         "path.go", | ||||||
|  |         "set.go", | ||||||
|  |     ], | ||||||
|  |     importmap = "k8s.io/kubernetes/vendor/sigs.k8s.io/structured-merge-diff/fieldpath", | ||||||
|  |     importpath = "sigs.k8s.io/structured-merge-diff/fieldpath", | ||||||
|  |     visibility = ["//visibility:public"], | ||||||
|  |     deps = ["//vendor/sigs.k8s.io/structured-merge-diff/value:go_default_library"], | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | filegroup( | ||||||
|  |     name = "package-srcs", | ||||||
|  |     srcs = glob(["**"]), | ||||||
|  |     tags = ["automanaged"], | ||||||
|  |     visibility = ["//visibility:private"], | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | filegroup( | ||||||
|  |     name = "all-srcs", | ||||||
|  |     srcs = [":package-srcs"], | ||||||
|  |     tags = ["automanaged"], | ||||||
|  |     visibility = ["//visibility:public"], | ||||||
|  | ) | ||||||
							
								
								
									
										21
									
								
								vendor/sigs.k8s.io/structured-merge-diff/fieldpath/doc.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										21
									
								
								vendor/sigs.k8s.io/structured-merge-diff/fieldpath/doc.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,21 @@ | |||||||
|  | /* | ||||||
|  | Copyright 2018 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 fieldpath defines a way for referencing path elements (e.g., an | ||||||
|  | // index in an array, or a key in a map). It provides types for arranging these | ||||||
|  | // into paths for referencing nested fields, and for grouping those into sets, | ||||||
|  | // for referencing multiple nested fields. | ||||||
|  | package fieldpath | ||||||
							
								
								
									
										184
									
								
								vendor/sigs.k8s.io/structured-merge-diff/fieldpath/element.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										184
									
								
								vendor/sigs.k8s.io/structured-merge-diff/fieldpath/element.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,184 @@ | |||||||
|  | /* | ||||||
|  | Copyright 2018 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 fieldpath | ||||||
|  |  | ||||||
|  | import ( | ||||||
|  | 	"fmt" | ||||||
|  | 	"sort" | ||||||
|  | 	"strings" | ||||||
|  |  | ||||||
|  | 	"sigs.k8s.io/structured-merge-diff/value" | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | // PathElement describes how to select a child field given a containing object. | ||||||
|  | type PathElement struct { | ||||||
|  | 	// Exactly one of the following fields should be non-nil. | ||||||
|  |  | ||||||
|  | 	// FieldName selects a single field from a map (reminder: this is also | ||||||
|  | 	// how structs are represented). The containing object must be a map. | ||||||
|  | 	FieldName *string | ||||||
|  |  | ||||||
|  | 	// Key selects the list element which has fields matching those given. | ||||||
|  | 	// The containing object must be an associative list with map typed | ||||||
|  | 	// elements. | ||||||
|  | 	Key []value.Field | ||||||
|  |  | ||||||
|  | 	// Value selects the list element with the given value. The containing | ||||||
|  | 	// object must be an associative list with a primitive typed element | ||||||
|  | 	// (i.e., a set). | ||||||
|  | 	Value *value.Value | ||||||
|  |  | ||||||
|  | 	// Index selects a list element by its index number. The containing | ||||||
|  | 	// object must be an atomic list. | ||||||
|  | 	Index *int | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // String presents the path element as a human-readable string. | ||||||
|  | func (e PathElement) String() string { | ||||||
|  | 	switch { | ||||||
|  | 	case e.FieldName != nil: | ||||||
|  | 		return "." + *e.FieldName | ||||||
|  | 	case len(e.Key) > 0: | ||||||
|  | 		strs := make([]string, len(e.Key)) | ||||||
|  | 		for i, k := range e.Key { | ||||||
|  | 			strs[i] = fmt.Sprintf("%v=%v", k.Name, k.Value) | ||||||
|  | 		} | ||||||
|  | 		// The order must be canonical, since we use the string value | ||||||
|  | 		// in a set structure. | ||||||
|  | 		sort.Strings(strs) | ||||||
|  | 		return "[" + strings.Join(strs, ",") + "]" | ||||||
|  | 	case e.Value != nil: | ||||||
|  | 		return fmt.Sprintf("[=%v]", e.Value) | ||||||
|  | 	case e.Index != nil: | ||||||
|  | 		return fmt.Sprintf("[%v]", *e.Index) | ||||||
|  | 	default: | ||||||
|  | 		return "{{invalid path element}}" | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // KeyByFields is a helper function which constructs a key for an associative | ||||||
|  | // list type. `nameValues` must have an even number of entries, alternating | ||||||
|  | // names (type must be string) with values (type must be value.Value). If these | ||||||
|  | // conditions are not met, KeyByFields will panic--it's intended for static | ||||||
|  | // construction and shouldn't have user-produced values passed to it. | ||||||
|  | func KeyByFields(nameValues ...interface{}) []value.Field { | ||||||
|  | 	if len(nameValues)%2 != 0 { | ||||||
|  | 		panic("must have a value for every name") | ||||||
|  | 	} | ||||||
|  | 	out := []value.Field{} | ||||||
|  | 	for i := 0; i < len(nameValues)-1; i += 2 { | ||||||
|  | 		out = append(out, value.Field{ | ||||||
|  | 			Name:  nameValues[i].(string), | ||||||
|  | 			Value: nameValues[i+1].(value.Value), | ||||||
|  | 		}) | ||||||
|  | 	} | ||||||
|  | 	return out | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // PathElementSet is a set of path elements. | ||||||
|  | // TODO: serialize as a list. | ||||||
|  | type PathElementSet struct { | ||||||
|  | 	// The strange construction is because there's no way to test | ||||||
|  | 	// PathElements for equality (it can't be used as a key for a map). | ||||||
|  | 	members map[string]PathElement | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // Insert adds pe to the set. | ||||||
|  | func (s *PathElementSet) Insert(pe PathElement) { | ||||||
|  | 	serialized := pe.String() | ||||||
|  | 	if s.members == nil { | ||||||
|  | 		s.members = map[string]PathElement{ | ||||||
|  | 			serialized: pe, | ||||||
|  | 		} | ||||||
|  | 		return | ||||||
|  | 	} | ||||||
|  | 	if _, ok := s.members[serialized]; !ok { | ||||||
|  | 		s.members[serialized] = pe | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // Union returns a set containing elements that appear in either s or s2. | ||||||
|  | func (s *PathElementSet) Union(s2 *PathElementSet) *PathElementSet { | ||||||
|  | 	out := &PathElementSet{ | ||||||
|  | 		members: map[string]PathElement{}, | ||||||
|  | 	} | ||||||
|  | 	for k, v := range s.members { | ||||||
|  | 		out.members[k] = v | ||||||
|  | 	} | ||||||
|  | 	for k, v := range s2.members { | ||||||
|  | 		out.members[k] = v | ||||||
|  | 	} | ||||||
|  | 	return out | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // Intersection returns a set containing elements which appear in both s and s2. | ||||||
|  | func (s *PathElementSet) Intersection(s2 *PathElementSet) *PathElementSet { | ||||||
|  | 	out := &PathElementSet{ | ||||||
|  | 		members: map[string]PathElement{}, | ||||||
|  | 	} | ||||||
|  | 	for k, v := range s.members { | ||||||
|  | 		if _, ok := s2.members[k]; ok { | ||||||
|  | 			out.members[k] = v | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	return out | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // Difference returns a set containing elements which appear in s but not in s2. | ||||||
|  | func (s *PathElementSet) Difference(s2 *PathElementSet) *PathElementSet { | ||||||
|  | 	out := &PathElementSet{ | ||||||
|  | 		members: map[string]PathElement{}, | ||||||
|  | 	} | ||||||
|  | 	for k, v := range s.members { | ||||||
|  | 		if _, ok := s2.members[k]; !ok { | ||||||
|  | 			out.members[k] = v | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	return out | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // Size retuns the number of elements in the set. | ||||||
|  | func (s *PathElementSet) Size() int { return len(s.members) } | ||||||
|  |  | ||||||
|  | // Has returns true if pe is a member of the set. | ||||||
|  | func (s *PathElementSet) Has(pe PathElement) bool { | ||||||
|  | 	if s.members == nil { | ||||||
|  | 		return false | ||||||
|  | 	} | ||||||
|  | 	_, ok := s.members[pe.String()] | ||||||
|  | 	return ok | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // Equals returns true if s and s2 have exactly the same members. | ||||||
|  | func (s *PathElementSet) Equals(s2 *PathElementSet) bool { | ||||||
|  | 	if len(s.members) != len(s2.members) { | ||||||
|  | 		return false | ||||||
|  | 	} | ||||||
|  | 	for k := range s.members { | ||||||
|  | 		if _, ok := s2.members[k]; !ok { | ||||||
|  | 			return false | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	return true | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // Iterate calls f for each PathElement in the set. | ||||||
|  | func (s *PathElementSet) Iterate(f func(PathElement)) { | ||||||
|  | 	for _, pe := range s.members { | ||||||
|  | 		f(pe) | ||||||
|  | 	} | ||||||
|  | } | ||||||
							
								
								
									
										123
									
								
								vendor/sigs.k8s.io/structured-merge-diff/fieldpath/fromvalue.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										123
									
								
								vendor/sigs.k8s.io/structured-merge-diff/fieldpath/fromvalue.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,123 @@ | |||||||
|  | /* | ||||||
|  | Copyright 2018 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 fieldpath | ||||||
|  |  | ||||||
|  | import ( | ||||||
|  | 	"sigs.k8s.io/structured-merge-diff/value" | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | // SetFromValue creates a set containing every leaf field mentioned in v. | ||||||
|  | func SetFromValue(v value.Value) *Set { | ||||||
|  | 	s := NewSet() | ||||||
|  |  | ||||||
|  | 	w := objectWalker{ | ||||||
|  | 		path:  Path{}, | ||||||
|  | 		value: v, | ||||||
|  | 		do:    func(p Path) { s.Insert(p) }, | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	w.walk() | ||||||
|  | 	return s | ||||||
|  | } | ||||||
|  |  | ||||||
|  | type objectWalker struct { | ||||||
|  | 	path  Path | ||||||
|  | 	value value.Value | ||||||
|  |  | ||||||
|  | 	do func(Path) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (w *objectWalker) walk() { | ||||||
|  | 	switch { | ||||||
|  | 	case w.value.Null: | ||||||
|  | 	case w.value.FloatValue != nil: | ||||||
|  | 	case w.value.IntValue != nil: | ||||||
|  | 	case w.value.StringValue != nil: | ||||||
|  | 	case w.value.BooleanValue != nil: | ||||||
|  | 		// All leaf fields handled the same way (after the switch | ||||||
|  | 		// statement). | ||||||
|  |  | ||||||
|  | 	// Descend | ||||||
|  | 	case w.value.ListValue != nil: | ||||||
|  | 		// If the list were atomic, we'd break here, but we don't have | ||||||
|  | 		// a schema, so we can't tell. | ||||||
|  |  | ||||||
|  | 		for i, child := range w.value.ListValue.Items { | ||||||
|  | 			w2 := *w | ||||||
|  | 			w2.path = append(w.path, GuessBestListPathElement(i, child)) | ||||||
|  | 			w2.value = child | ||||||
|  | 			w2.walk() | ||||||
|  | 		} | ||||||
|  | 		return | ||||||
|  | 	case w.value.MapValue != nil: | ||||||
|  | 		// If the map/struct were atomic, we'd break here, but we don't | ||||||
|  | 		// have a schema, so we can't tell. | ||||||
|  |  | ||||||
|  | 		for i := range w.value.MapValue.Items { | ||||||
|  | 			child := w.value.MapValue.Items[i] | ||||||
|  | 			w2 := *w | ||||||
|  | 			w2.path = append(w.path, PathElement{FieldName: &child.Name}) | ||||||
|  | 			w2.value = child.Value | ||||||
|  | 			w2.walk() | ||||||
|  | 		} | ||||||
|  | 		return | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	// Leaf fields get added to the set. | ||||||
|  | 	if len(w.path) > 0 { | ||||||
|  | 		w.do(w.path) | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // AssociativeListCandidateFieldNames lists the field names which are | ||||||
|  | // considered keys if found in a list element. | ||||||
|  | var AssociativeListCandidateFieldNames = []string{ | ||||||
|  | 	"key", | ||||||
|  | 	"id", | ||||||
|  | 	"name", | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // GuessBestListPathElement guesses whether item is an associative list | ||||||
|  | // element, which should be referenced by key(s), or if it is not and therefore | ||||||
|  | // referencing by index is acceptable. Currently this is done by checking | ||||||
|  | // whether item has any of the fields listed in | ||||||
|  | // AssociativeListCandidateFieldNames which have scalar values. | ||||||
|  | func GuessBestListPathElement(index int, item value.Value) PathElement { | ||||||
|  | 	if item.MapValue == nil { | ||||||
|  | 		// Non map items could be parts of sets or regular "atomic" | ||||||
|  | 		// lists. We won't try to guess whether something should be a | ||||||
|  | 		// set or not. | ||||||
|  | 		return PathElement{Index: &index} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	var keys []value.Field | ||||||
|  | 	for _, name := range AssociativeListCandidateFieldNames { | ||||||
|  | 		f, ok := item.MapValue.Get(name) | ||||||
|  | 		if !ok { | ||||||
|  | 			continue | ||||||
|  | 		} | ||||||
|  | 		// only accept primitive/scalar types as keys. | ||||||
|  | 		if f.Value.Null || f.Value.MapValue != nil || f.Value.ListValue != nil { | ||||||
|  | 			continue | ||||||
|  | 		} | ||||||
|  | 		keys = append(keys, *f) | ||||||
|  | 	} | ||||||
|  | 	if len(keys) > 0 { | ||||||
|  | 		return PathElement{Key: keys} | ||||||
|  | 	} | ||||||
|  | 	return PathElement{Index: &index} | ||||||
|  | } | ||||||
							
								
								
									
										73
									
								
								vendor/sigs.k8s.io/structured-merge-diff/fieldpath/managers.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										73
									
								
								vendor/sigs.k8s.io/structured-merge-diff/fieldpath/managers.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,73 @@ | |||||||
|  | /* | ||||||
|  | Copyright 2018 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 fieldpath | ||||||
|  |  | ||||||
|  | // APIVersion describes the version of an object or of a fieldset. | ||||||
|  | type APIVersion string | ||||||
|  |  | ||||||
|  | // VersionedSet associates a version to a set. | ||||||
|  | type VersionedSet struct { | ||||||
|  | 	*Set | ||||||
|  | 	APIVersion APIVersion | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // ManagedFields is a map from manager to VersionedSet (what they own in | ||||||
|  | // what version). | ||||||
|  | type ManagedFields map[string]*VersionedSet | ||||||
|  |  | ||||||
|  | // Difference returns a symmetric difference between two Managers. If a | ||||||
|  | // given user's entry has version X in lhs and version Y in rhs, then | ||||||
|  | // the return value for that user will be from rhs. If the difference for | ||||||
|  | // a user is an empty set, that user will not be inserted in the map. | ||||||
|  | func (lhs ManagedFields) Difference(rhs ManagedFields) ManagedFields { | ||||||
|  | 	diff := ManagedFields{} | ||||||
|  |  | ||||||
|  | 	for manager, left := range lhs { | ||||||
|  | 		right, ok := rhs[manager] | ||||||
|  | 		if !ok { | ||||||
|  | 			if !left.Empty() { | ||||||
|  | 				diff[manager] = left | ||||||
|  | 			} | ||||||
|  | 			continue | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		// If we have sets in both but their version | ||||||
|  | 		// differs, we don't even diff and keep the | ||||||
|  | 		// entire thing. | ||||||
|  | 		if left.APIVersion != right.APIVersion { | ||||||
|  | 			diff[manager] = right | ||||||
|  | 			continue | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		newSet := left.Difference(right.Set).Union(right.Difference(left.Set)) | ||||||
|  | 		if !newSet.Empty() { | ||||||
|  | 			diff[manager] = &VersionedSet{ | ||||||
|  | 				Set:        newSet, | ||||||
|  | 				APIVersion: right.APIVersion, | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	for manager, set := range rhs { | ||||||
|  | 		if _, ok := lhs[manager]; ok { | ||||||
|  | 			// Already done | ||||||
|  | 			continue | ||||||
|  | 		} | ||||||
|  | 		if !set.Empty() { | ||||||
|  | 			diff[manager] = set | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return diff | ||||||
|  | } | ||||||
							
								
								
									
										75
									
								
								vendor/sigs.k8s.io/structured-merge-diff/fieldpath/path.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										75
									
								
								vendor/sigs.k8s.io/structured-merge-diff/fieldpath/path.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,75 @@ | |||||||
|  | /* | ||||||
|  | Copyright 2018 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 fieldpath | ||||||
|  |  | ||||||
|  | import ( | ||||||
|  | 	"fmt" | ||||||
|  | 	"strings" | ||||||
|  |  | ||||||
|  | 	"sigs.k8s.io/structured-merge-diff/value" | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | // Path describes how to select a potentially deeply-nested child field given a | ||||||
|  | // containing object. | ||||||
|  | type Path []PathElement | ||||||
|  |  | ||||||
|  | func (fp Path) String() string { | ||||||
|  | 	strs := make([]string, len(fp)) | ||||||
|  | 	for i := range fp { | ||||||
|  | 		strs[i] = fp[i].String() | ||||||
|  | 	} | ||||||
|  | 	return strings.Join(strs, "") | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // MakePath constructs a Path. The parts may be PathElements, ints, strings. | ||||||
|  | func MakePath(parts ...interface{}) (Path, error) { | ||||||
|  | 	var fp Path | ||||||
|  | 	for _, p := range parts { | ||||||
|  | 		switch t := p.(type) { | ||||||
|  | 		case PathElement: | ||||||
|  | 			fp = append(fp, t) | ||||||
|  | 		case int: | ||||||
|  | 			// TODO: Understand schema and object and convert this to the | ||||||
|  | 			// FieldSpecifier below if appropriate. | ||||||
|  | 			fp = append(fp, PathElement{Index: &t}) | ||||||
|  | 		case string: | ||||||
|  | 			fp = append(fp, PathElement{FieldName: &t}) | ||||||
|  | 		case []value.Field: | ||||||
|  | 			if len(t) == 0 { | ||||||
|  | 				return nil, fmt.Errorf("associative list key type path elements must have at least one key (got zero)") | ||||||
|  | 			} | ||||||
|  | 			fp = append(fp, PathElement{Key: t}) | ||||||
|  | 		case value.Value: | ||||||
|  | 			// TODO: understand schema and verify that this is a set type | ||||||
|  | 			// TODO: make a copy of t | ||||||
|  | 			fp = append(fp, PathElement{Value: &t}) | ||||||
|  | 		default: | ||||||
|  | 			return nil, fmt.Errorf("unable to make %#v into a path element", p) | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	return fp, nil | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // MakePathOrDie panics if parts can't be turned into a path. Good for things | ||||||
|  | // that are known at complie time. | ||||||
|  | func MakePathOrDie(parts ...interface{}) Path { | ||||||
|  | 	fp, err := MakePath(parts...) | ||||||
|  | 	if err != nil { | ||||||
|  | 		panic(err) | ||||||
|  | 	} | ||||||
|  | 	return fp | ||||||
|  | } | ||||||
							
								
								
									
										305
									
								
								vendor/sigs.k8s.io/structured-merge-diff/fieldpath/set.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										305
									
								
								vendor/sigs.k8s.io/structured-merge-diff/fieldpath/set.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,305 @@ | |||||||
|  | /* | ||||||
|  | Copyright 2018 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 fieldpath | ||||||
|  |  | ||||||
|  | import ( | ||||||
|  | 	"strings" | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | // Set identifies a set of fields. | ||||||
|  | type Set struct { | ||||||
|  | 	// Members lists fields that are part of the set. | ||||||
|  | 	// TODO: will be serialized as a list of path elements. | ||||||
|  | 	Members PathElementSet | ||||||
|  |  | ||||||
|  | 	// Children lists child fields which themselves have children that are | ||||||
|  | 	// members of the set. Appearance in this list does not imply membership. | ||||||
|  | 	// Note: this is a tree, not an arbitrary graph. | ||||||
|  | 	Children SetNodeMap | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // NewSet makes a set from a list of paths. | ||||||
|  | func NewSet(paths ...Path) *Set { | ||||||
|  | 	s := &Set{} | ||||||
|  | 	for _, p := range paths { | ||||||
|  | 		s.Insert(p) | ||||||
|  | 	} | ||||||
|  | 	return s | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // Insert adds the field identified by `p` to the set. Important: parent fields | ||||||
|  | // are NOT added to the set; if that is desired, they must be added separately. | ||||||
|  | func (s *Set) Insert(p Path) { | ||||||
|  | 	if len(p) == 0 { | ||||||
|  | 		// Zero-length path identifies the entire object; we don't | ||||||
|  | 		// track top-level ownership. | ||||||
|  | 		return | ||||||
|  | 	} | ||||||
|  | 	for { | ||||||
|  | 		if len(p) == 1 { | ||||||
|  | 			s.Members.Insert(p[0]) | ||||||
|  | 			return | ||||||
|  | 		} | ||||||
|  | 		s = s.Children.Descend(p[0]) | ||||||
|  | 		p = p[1:] | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // Union returns a Set containing elements which appear in either s or s2. | ||||||
|  | func (s *Set) Union(s2 *Set) *Set { | ||||||
|  | 	return &Set{ | ||||||
|  | 		Members:  *s.Members.Union(&s2.Members), | ||||||
|  | 		Children: *s.Children.Union(&s2.Children), | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // Intersection returns a Set containing leaf elements which appear in both s | ||||||
|  | // and s2. Intersection can be constructed from Union and Difference operations | ||||||
|  | // (example in the tests) but it's much faster to do it in one pass. | ||||||
|  | func (s *Set) Intersection(s2 *Set) *Set { | ||||||
|  | 	return &Set{ | ||||||
|  | 		Members:  *s.Members.Intersection(&s2.Members), | ||||||
|  | 		Children: *s.Children.Intersection(&s2.Children), | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // Difference returns a Set containing elements which: | ||||||
|  | // * appear in s | ||||||
|  | // * do not appear in s2 | ||||||
|  | // * and are not children of elements that appear in s2. | ||||||
|  | // | ||||||
|  | // In other words, for leaf fields, this acts like a regular set difference | ||||||
|  | // operation. When non leaf fields are compared with leaf fields ("parents" | ||||||
|  | // which contain "children"), the effect is: | ||||||
|  | // * parent - child = parent | ||||||
|  | // * child - parent = {empty set} | ||||||
|  | func (s *Set) Difference(s2 *Set) *Set { | ||||||
|  | 	return &Set{ | ||||||
|  | 		Members:  *s.Members.Difference(&s2.Members), | ||||||
|  | 		Children: *s.Children.Difference(s2), | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // Size returns the number of members of the set. | ||||||
|  | func (s *Set) Size() int { | ||||||
|  | 	return s.Members.Size() + s.Children.Size() | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // Empty returns true if there are no members of the set. It is a separate | ||||||
|  | // function from Size since it's common to check whether size > 0, and | ||||||
|  | // potentially much faster to return as soon as a single element is found. | ||||||
|  | func (s *Set) Empty() bool { | ||||||
|  | 	if s.Members.Size() > 0 { | ||||||
|  | 		return false | ||||||
|  | 	} | ||||||
|  | 	return s.Children.Empty() | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // Has returns true if the field referenced by `p` is a member of the set. | ||||||
|  | func (s *Set) Has(p Path) bool { | ||||||
|  | 	if len(p) == 0 { | ||||||
|  | 		// No one owns "the entire object" | ||||||
|  | 		return false | ||||||
|  | 	} | ||||||
|  | 	for { | ||||||
|  | 		if len(p) == 1 { | ||||||
|  | 			return s.Members.Has(p[0]) | ||||||
|  | 		} | ||||||
|  | 		var ok bool | ||||||
|  | 		s, ok = s.Children.Get(p[0]) | ||||||
|  | 		if !ok { | ||||||
|  | 			return false | ||||||
|  | 		} | ||||||
|  | 		p = p[1:] | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // Equals returns true if s and s2 have exactly the same members. | ||||||
|  | func (s *Set) Equals(s2 *Set) bool { | ||||||
|  | 	return s.Members.Equals(&s2.Members) && s.Children.Equals(&s2.Children) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // String returns the set one element per line. | ||||||
|  | func (s *Set) String() string { | ||||||
|  | 	elements := []string{} | ||||||
|  | 	s.Iterate(func(p Path) { | ||||||
|  | 		elements = append(elements, p.String()) | ||||||
|  | 	}) | ||||||
|  | 	return strings.Join(elements, "\n") | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // Iterate calls f once for each field that is a member of the set (preorder | ||||||
|  | // DFS). The path passed to f will be reused so make a copy if you wish to keep | ||||||
|  | // it. | ||||||
|  | func (s *Set) Iterate(f func(Path)) { | ||||||
|  | 	s.iteratePrefix(Path{}, f) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (s *Set) iteratePrefix(prefix Path, f func(Path)) { | ||||||
|  | 	s.Members.Iterate(func(pe PathElement) { f(append(prefix, pe)) }) | ||||||
|  | 	s.Children.iteratePrefix(prefix, f) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // setNode is a pair of PathElement / Set, for the purpose of expressing | ||||||
|  | // nested set membership. | ||||||
|  | type setNode struct { | ||||||
|  | 	pathElement PathElement | ||||||
|  | 	set         *Set | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // SetNodeMap is a map of PathElement to subset. | ||||||
|  | type SetNodeMap struct { | ||||||
|  | 	members map[string]setNode | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // Descend adds pe to the set if necessary, returning the associated subset. | ||||||
|  | func (s *SetNodeMap) Descend(pe PathElement) *Set { | ||||||
|  | 	serialized := pe.String() | ||||||
|  | 	if s.members == nil { | ||||||
|  | 		s.members = map[string]setNode{} | ||||||
|  | 	} | ||||||
|  | 	if n, ok := s.members[serialized]; ok { | ||||||
|  | 		return n.set | ||||||
|  | 	} | ||||||
|  | 	ss := &Set{} | ||||||
|  | 	s.members[serialized] = setNode{ | ||||||
|  | 		pathElement: pe, | ||||||
|  | 		set:         ss, | ||||||
|  | 	} | ||||||
|  | 	return ss | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // Size returns the sum of the number of members of all subsets. | ||||||
|  | func (s *SetNodeMap) Size() int { | ||||||
|  | 	count := 0 | ||||||
|  | 	for _, v := range s.members { | ||||||
|  | 		count += v.set.Size() | ||||||
|  | 	} | ||||||
|  | 	return count | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // Empty returns false if there's at least one member in some child set. | ||||||
|  | func (s *SetNodeMap) Empty() bool { | ||||||
|  | 	for _, n := range s.members { | ||||||
|  | 		if !n.set.Empty() { | ||||||
|  | 			return false | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	return true | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // Get returns (the associated set, true) or (nil, false) if there is none. | ||||||
|  | func (s *SetNodeMap) Get(pe PathElement) (*Set, bool) { | ||||||
|  | 	if s.members == nil { | ||||||
|  | 		return nil, false | ||||||
|  | 	} | ||||||
|  | 	serialized := pe.String() | ||||||
|  | 	if n, ok := s.members[serialized]; ok { | ||||||
|  | 		return n.set, true | ||||||
|  | 	} | ||||||
|  | 	return nil, false | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // Equals returns true if s and s2 have the same structure (same nested | ||||||
|  | // child sets). | ||||||
|  | func (s *SetNodeMap) Equals(s2 *SetNodeMap) bool { | ||||||
|  | 	if len(s.members) != len(s2.members) { | ||||||
|  | 		return false | ||||||
|  | 	} | ||||||
|  | 	for k, v := range s.members { | ||||||
|  | 		v2, ok := s2.members[k] | ||||||
|  | 		if !ok { | ||||||
|  | 			return false | ||||||
|  | 		} | ||||||
|  | 		if !v.set.Equals(v2.set) { | ||||||
|  | 			return false | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	return true | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // Union returns a SetNodeMap with members that appear in either s or s2. | ||||||
|  | func (s *SetNodeMap) Union(s2 *SetNodeMap) *SetNodeMap { | ||||||
|  | 	out := &SetNodeMap{} | ||||||
|  | 	for k, sn := range s.members { | ||||||
|  | 		pe := sn.pathElement | ||||||
|  | 		if sn2, ok := s2.members[k]; ok { | ||||||
|  | 			*out.Descend(pe) = *sn.set.Union(sn2.set) | ||||||
|  | 		} else { | ||||||
|  | 			*out.Descend(pe) = *sn.set | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	for k, sn2 := range s2.members { | ||||||
|  | 		pe := sn2.pathElement | ||||||
|  | 		if _, ok := s.members[k]; ok { | ||||||
|  | 			// already handled | ||||||
|  | 			continue | ||||||
|  | 		} | ||||||
|  | 		*out.Descend(pe) = *sn2.set | ||||||
|  | 	} | ||||||
|  | 	return out | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // Intersection returns a SetNodeMap with members that appear in both s and s2. | ||||||
|  | func (s *SetNodeMap) Intersection(s2 *SetNodeMap) *SetNodeMap { | ||||||
|  | 	out := &SetNodeMap{} | ||||||
|  | 	for k, sn := range s.members { | ||||||
|  | 		pe := sn.pathElement | ||||||
|  | 		if sn2, ok := s2.members[k]; ok { | ||||||
|  | 			i := *sn.set.Intersection(sn2.set) | ||||||
|  | 			if !i.Empty() { | ||||||
|  | 				*out.Descend(pe) = i | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	return out | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // Difference returns a SetNodeMap with members that appear in s but not in s2. | ||||||
|  | func (s *SetNodeMap) Difference(s2 *Set) *SetNodeMap { | ||||||
|  | 	out := &SetNodeMap{} | ||||||
|  | 	for k, sn := range s.members { | ||||||
|  | 		pe := sn.pathElement | ||||||
|  | 		if s2.Members.Has(pe) { | ||||||
|  | 			continue | ||||||
|  | 		} | ||||||
|  | 		if sn2, ok := s2.Children.members[k]; ok { | ||||||
|  | 			diff := *sn.set.Difference(sn2.set) | ||||||
|  | 			// We aren't permitted to add nodes with no elements. | ||||||
|  | 			if !diff.Empty() { | ||||||
|  | 				*out.Descend(pe) = diff | ||||||
|  | 			} | ||||||
|  | 		} else { | ||||||
|  | 			*out.Descend(pe) = *sn.set | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	return out | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // Iterate calls f for each PathElement in the set. | ||||||
|  | func (s *SetNodeMap) Iterate(f func(PathElement)) { | ||||||
|  | 	for _, n := range s.members { | ||||||
|  | 		f(n.pathElement) | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (s *SetNodeMap) iteratePrefix(prefix Path, f func(Path)) { | ||||||
|  | 	for _, n := range s.members { | ||||||
|  | 		pe := n.pathElement | ||||||
|  | 		n.set.iteratePrefix(append(prefix, pe), f) | ||||||
|  | 	} | ||||||
|  | } | ||||||
							
								
								
									
										30
									
								
								vendor/sigs.k8s.io/structured-merge-diff/merge/BUILD
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										30
									
								
								vendor/sigs.k8s.io/structured-merge-diff/merge/BUILD
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,30 @@ | |||||||
|  | load("@io_bazel_rules_go//go:def.bzl", "go_library") | ||||||
|  |  | ||||||
|  | go_library( | ||||||
|  |     name = "go_default_library", | ||||||
|  |     srcs = [ | ||||||
|  |         "conflict.go", | ||||||
|  |         "update.go", | ||||||
|  |     ], | ||||||
|  |     importmap = "k8s.io/kubernetes/vendor/sigs.k8s.io/structured-merge-diff/merge", | ||||||
|  |     importpath = "sigs.k8s.io/structured-merge-diff/merge", | ||||||
|  |     visibility = ["//visibility:public"], | ||||||
|  |     deps = [ | ||||||
|  |         "//vendor/sigs.k8s.io/structured-merge-diff/fieldpath:go_default_library", | ||||||
|  |         "//vendor/sigs.k8s.io/structured-merge-diff/typed:go_default_library", | ||||||
|  |     ], | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | filegroup( | ||||||
|  |     name = "package-srcs", | ||||||
|  |     srcs = glob(["**"]), | ||||||
|  |     tags = ["automanaged"], | ||||||
|  |     visibility = ["//visibility:private"], | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | filegroup( | ||||||
|  |     name = "all-srcs", | ||||||
|  |     srcs = [":package-srcs"], | ||||||
|  |     tags = ["automanaged"], | ||||||
|  |     visibility = ["//visibility:public"], | ||||||
|  | ) | ||||||
							
								
								
									
										91
									
								
								vendor/sigs.k8s.io/structured-merge-diff/merge/conflict.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										91
									
								
								vendor/sigs.k8s.io/structured-merge-diff/merge/conflict.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,91 @@ | |||||||
|  | /* | ||||||
|  | Copyright 2018 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 merge | ||||||
|  |  | ||||||
|  | import ( | ||||||
|  | 	"fmt" | ||||||
|  | 	"sort" | ||||||
|  | 	"strings" | ||||||
|  |  | ||||||
|  | 	"sigs.k8s.io/structured-merge-diff/fieldpath" | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | // Conflict is a conflict on a specific field with the current manager of | ||||||
|  | // that field. It does implement the error interface so that it can be | ||||||
|  | // used as an error. | ||||||
|  | type Conflict struct { | ||||||
|  | 	Manager string | ||||||
|  | 	Path    fieldpath.Path | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // Conflict is an error. | ||||||
|  | var _ error = Conflict{} | ||||||
|  |  | ||||||
|  | // Error formats the conflict as an error. | ||||||
|  | func (c Conflict) Error() string { | ||||||
|  | 	return fmt.Sprintf("conflict with %q: %v", c.Manager, c.Path) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // Conflicts accumulates multiple conflicts and aggregates them by managers. | ||||||
|  | type Conflicts []Conflict | ||||||
|  |  | ||||||
|  | var _ error = Conflicts{} | ||||||
|  |  | ||||||
|  | // Error prints the list of conflicts, grouped by sorted managers. | ||||||
|  | func (conflicts Conflicts) Error() string { | ||||||
|  | 	if len(conflicts) == 1 { | ||||||
|  | 		return conflicts[0].Error() | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	m := map[string][]fieldpath.Path{} | ||||||
|  | 	for _, conflict := range conflicts { | ||||||
|  | 		m[conflict.Manager] = append(m[conflict.Manager], conflict.Path) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	managers := []string{} | ||||||
|  | 	for manager := range m { | ||||||
|  | 		managers = append(managers, manager) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	// Print conflicts by sorted managers. | ||||||
|  | 	sort.Strings(managers) | ||||||
|  |  | ||||||
|  | 	messages := []string{} | ||||||
|  | 	for _, manager := range managers { | ||||||
|  | 		messages = append(messages, fmt.Sprintf("conflicts with %q:", manager)) | ||||||
|  | 		for _, path := range m[manager] { | ||||||
|  | 			messages = append(messages, fmt.Sprintf("- %v", path)) | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	return strings.Join(messages, "\n") | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // ConflictsFromManagers creates a list of conflicts given Managers sets. | ||||||
|  | func ConflictsFromManagers(sets fieldpath.ManagedFields) Conflicts { | ||||||
|  | 	conflicts := []Conflict{} | ||||||
|  |  | ||||||
|  | 	for manager, set := range sets { | ||||||
|  | 		set.Iterate(func(p fieldpath.Path) { | ||||||
|  | 			conflicts = append(conflicts, Conflict{ | ||||||
|  | 				Manager: manager, | ||||||
|  | 				Path:    p, | ||||||
|  | 			}) | ||||||
|  | 		}) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return conflicts | ||||||
|  | } | ||||||
							
								
								
									
										142
									
								
								vendor/sigs.k8s.io/structured-merge-diff/merge/update.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										142
									
								
								vendor/sigs.k8s.io/structured-merge-diff/merge/update.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,142 @@ | |||||||
|  | /* | ||||||
|  | Copyright 2018 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 merge | ||||||
|  |  | ||||||
|  | import ( | ||||||
|  | 	"fmt" | ||||||
|  |  | ||||||
|  | 	"sigs.k8s.io/structured-merge-diff/fieldpath" | ||||||
|  | 	"sigs.k8s.io/structured-merge-diff/typed" | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | // Converter is an interface to the conversion logic. The converter | ||||||
|  | // needs to be able to convert objects from one version to another. | ||||||
|  | type Converter interface { | ||||||
|  | 	Convert(object typed.TypedValue, version fieldpath.APIVersion) (typed.TypedValue, error) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // Updater is the object used to compute updated FieldSets and also | ||||||
|  | // merge the object on Apply. | ||||||
|  | type Updater struct { | ||||||
|  | 	Converter Converter | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (s *Updater) update(oldObject, newObject typed.TypedValue, version fieldpath.APIVersion, managers fieldpath.ManagedFields, workflow string, force bool) (fieldpath.ManagedFields, error) { | ||||||
|  | 	if managers == nil { | ||||||
|  | 		managers = fieldpath.ManagedFields{} | ||||||
|  | 	} | ||||||
|  | 	conflicts := fieldpath.ManagedFields{} | ||||||
|  | 	type Versioned struct { | ||||||
|  | 		oldObject typed.TypedValue | ||||||
|  | 		newObject typed.TypedValue | ||||||
|  | 	} | ||||||
|  | 	versions := map[fieldpath.APIVersion]Versioned{ | ||||||
|  | 		version: Versioned{ | ||||||
|  | 			oldObject: oldObject, | ||||||
|  | 			newObject: newObject, | ||||||
|  | 		}, | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	for manager, managerSet := range managers { | ||||||
|  | 		if manager == workflow { | ||||||
|  | 			continue | ||||||
|  | 		} | ||||||
|  | 		versioned, ok := versions[managerSet.APIVersion] | ||||||
|  | 		if !ok { | ||||||
|  | 			var err error | ||||||
|  | 			versioned.oldObject, err = s.Converter.Convert(oldObject, managerSet.APIVersion) | ||||||
|  | 			if err != nil { | ||||||
|  | 				return nil, fmt.Errorf("failed to convert old object: %v", err) | ||||||
|  | 			} | ||||||
|  | 			versioned.newObject, err = s.Converter.Convert(newObject, managerSet.APIVersion) | ||||||
|  | 			if err != nil { | ||||||
|  | 				return nil, fmt.Errorf("failed to convert new object: %v", err) | ||||||
|  | 			} | ||||||
|  | 			versions[managerSet.APIVersion] = versioned | ||||||
|  | 		} | ||||||
|  | 		compare, err := versioned.oldObject.Compare(versioned.newObject) | ||||||
|  | 		if err != nil { | ||||||
|  | 			return nil, fmt.Errorf("failed to compare objects: %v", err) | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		conflictSet := managerSet.Intersection(compare.Modified.Union(compare.Added)) | ||||||
|  | 		if !conflictSet.Empty() { | ||||||
|  | 			conflicts[manager] = &fieldpath.VersionedSet{ | ||||||
|  | 				Set:        conflictSet, | ||||||
|  | 				APIVersion: managerSet.APIVersion, | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if !force && len(conflicts) != 0 { | ||||||
|  | 		return nil, ConflictsFromManagers(conflicts) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	for manager, conflictSet := range conflicts { | ||||||
|  | 		managers[manager].Set = managers[manager].Set.Difference(conflictSet.Set) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return managers, nil | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // Update is the method you should call once you've merged your final | ||||||
|  | // object on CREATE/UPDATE/PATCH verbs. newObject must be the object | ||||||
|  | // that you intend to persist (after applying the patch if this is for a | ||||||
|  | // PATCH call), and liveObject must be the original object (empty if | ||||||
|  | // this is a CREATE call). | ||||||
|  | func (s *Updater) Update(liveObject, newObject typed.TypedValue, version fieldpath.APIVersion, managers fieldpath.ManagedFields, manager string) (fieldpath.ManagedFields, error) { | ||||||
|  | 	var err error | ||||||
|  | 	managers, err = s.update(liveObject, newObject, version, managers, manager, true) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return fieldpath.ManagedFields{}, err | ||||||
|  | 	} | ||||||
|  | 	compare, err := liveObject.Compare(newObject) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return fieldpath.ManagedFields{}, fmt.Errorf("failed to compare live and new objects: %v", err) | ||||||
|  | 	} | ||||||
|  | 	if _, ok := managers[manager]; !ok { | ||||||
|  | 		managers[manager] = &fieldpath.VersionedSet{ | ||||||
|  | 			Set: fieldpath.NewSet(), | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	managers[manager].Set = managers[manager].Set.Union(compare.Modified).Union(compare.Added).Difference(compare.Removed) | ||||||
|  | 	managers[manager].APIVersion = version | ||||||
|  | 	return managers, nil | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // Apply should be called when Apply is run, given the current object as | ||||||
|  | // well as the configuration that is applied. This will merge the object | ||||||
|  | // and return it. | ||||||
|  | func (s *Updater) Apply(liveObject, configObject typed.TypedValue, version fieldpath.APIVersion, managers fieldpath.ManagedFields, manager string, force bool) (typed.TypedValue, fieldpath.ManagedFields, error) { | ||||||
|  | 	newObject, err := liveObject.Merge(configObject) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return typed.TypedValue{}, fieldpath.ManagedFields{}, fmt.Errorf("failed to merge config: %v", err) | ||||||
|  | 	} | ||||||
|  | 	managers, err = s.update(liveObject, newObject, version, managers, manager, force) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return typed.TypedValue{}, fieldpath.ManagedFields{}, err | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	// TODO: Remove unconflicting removed fields | ||||||
|  |  | ||||||
|  | 	set, err := configObject.ToFieldSet() | ||||||
|  | 	if err != nil { | ||||||
|  | 		return typed.TypedValue{}, fieldpath.ManagedFields{}, fmt.Errorf("failed to get field set: %v", err) | ||||||
|  | 	} | ||||||
|  | 	managers[manager] = &fieldpath.VersionedSet{ | ||||||
|  | 		Set:        set, | ||||||
|  | 		APIVersion: version, | ||||||
|  | 	} | ||||||
|  | 	return newObject, managers, nil | ||||||
|  | } | ||||||
							
								
								
									
										27
									
								
								vendor/sigs.k8s.io/structured-merge-diff/schema/BUILD
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										27
									
								
								vendor/sigs.k8s.io/structured-merge-diff/schema/BUILD
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,27 @@ | |||||||
|  | load("@io_bazel_rules_go//go:def.bzl", "go_library") | ||||||
|  |  | ||||||
|  | go_library( | ||||||
|  |     name = "go_default_library", | ||||||
|  |     srcs = [ | ||||||
|  |         "doc.go", | ||||||
|  |         "elements.go", | ||||||
|  |         "schemaschema.go", | ||||||
|  |     ], | ||||||
|  |     importmap = "k8s.io/kubernetes/vendor/sigs.k8s.io/structured-merge-diff/schema", | ||||||
|  |     importpath = "sigs.k8s.io/structured-merge-diff/schema", | ||||||
|  |     visibility = ["//visibility:public"], | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | filegroup( | ||||||
|  |     name = "package-srcs", | ||||||
|  |     srcs = glob(["**"]), | ||||||
|  |     tags = ["automanaged"], | ||||||
|  |     visibility = ["//visibility:private"], | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | filegroup( | ||||||
|  |     name = "all-srcs", | ||||||
|  |     srcs = [":package-srcs"], | ||||||
|  |     tags = ["automanaged"], | ||||||
|  |     visibility = ["//visibility:public"], | ||||||
|  | ) | ||||||
							
								
								
									
										28
									
								
								vendor/sigs.k8s.io/structured-merge-diff/schema/doc.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										28
									
								
								vendor/sigs.k8s.io/structured-merge-diff/schema/doc.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,28 @@ | |||||||
|  | /* | ||||||
|  | Copyright 2018 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 schema defines a targeted schema language which allows one to | ||||||
|  | // represent all the schema information necessary to perform "structured" | ||||||
|  | // merges and diffs. | ||||||
|  | // | ||||||
|  | // Due to the targeted nature of the data model, the schema language can fit in | ||||||
|  | // just a few hundred lines of go code, making it much more understandable and | ||||||
|  | // concise than e.g. OpenAPI. | ||||||
|  | // | ||||||
|  | // This schema was derived by observing the API objects used by Kubernetes, and | ||||||
|  | // formalizing a model which allows certain operations ("apply") to be more | ||||||
|  | // well defined. It is currently missing one feature: one-of ("unions"). | ||||||
|  | package schema | ||||||
							
								
								
									
										219
									
								
								vendor/sigs.k8s.io/structured-merge-diff/schema/elements.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										219
									
								
								vendor/sigs.k8s.io/structured-merge-diff/schema/elements.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,219 @@ | |||||||
|  | /* | ||||||
|  | Copyright 2018 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 schema | ||||||
|  |  | ||||||
|  | // Schema is a list of named types. | ||||||
|  | type Schema struct { | ||||||
|  | 	Types []TypeDef `yaml:"types,omitempty"` | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // A TypeSpecifier references a particular type in a schema. | ||||||
|  | type TypeSpecifier struct { | ||||||
|  | 	Type   TypeRef `yaml:"type,omitempty"` | ||||||
|  | 	Schema Schema  `yaml:"schema,omitempty"` | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // TypeDef represents a named type in a schema. | ||||||
|  | type TypeDef struct { | ||||||
|  | 	// Top level types should be named. Every type must have a unique name. | ||||||
|  | 	Name string `yaml:"name,omitempty"` | ||||||
|  |  | ||||||
|  | 	Atom `yaml:"atom,omitempty,inline"` | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // TypeRef either refers to a named type or declares an inlined type. | ||||||
|  | type TypeRef struct { | ||||||
|  | 	// Either the name or one member of Atom should be set. | ||||||
|  | 	NamedType *string `yaml:"namedType,omitempty"` | ||||||
|  | 	Inlined   Atom    `yaml:",inline,omitempty"` | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // Atom represents the smallest possible pieces of the type system. | ||||||
|  | type Atom struct { | ||||||
|  | 	// Exactly one of the below must be set. | ||||||
|  | 	*Scalar  `yaml:"scalar,omitempty"` | ||||||
|  | 	*Struct  `yaml:"struct,omitempty"` | ||||||
|  | 	*List    `yaml:"list,omitempty"` | ||||||
|  | 	*Map     `yaml:"map,omitempty"` | ||||||
|  | 	*Untyped `yaml:"untyped,omitempty"` | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // Scalar (AKA "primitive") represents a type which has a single value which is | ||||||
|  | // either numeric, string, or boolean. | ||||||
|  | // | ||||||
|  | // TODO: split numeric into float/int? Something even more fine-grained? | ||||||
|  | type Scalar string | ||||||
|  |  | ||||||
|  | const ( | ||||||
|  | 	Numeric = Scalar("numeric") | ||||||
|  | 	String  = Scalar("string") | ||||||
|  | 	Boolean = Scalar("boolean") | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | // ElementRelationship is an enum of the different possible relationships | ||||||
|  | // between the elements of container types (maps, lists, structs, untyped). | ||||||
|  | type ElementRelationship string | ||||||
|  |  | ||||||
|  | const ( | ||||||
|  | 	// Associative only applies to lists (see the documentation there). | ||||||
|  | 	Associative = ElementRelationship("associative") | ||||||
|  | 	// Atomic makes container types (lists, maps, structs, untyped) behave | ||||||
|  | 	// as scalars / leaf fields (which is the default for untyped data). | ||||||
|  | 	Atomic = ElementRelationship("atomic") | ||||||
|  | 	// Separable means the items of the container type have no particular | ||||||
|  | 	// relationship (default behavior for maps and structs). | ||||||
|  | 	Separable = ElementRelationship("separable") | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | // Struct represents a type which is composed of a number of different fields. | ||||||
|  | // Each field has a name and a type. | ||||||
|  | // | ||||||
|  | // TODO: in the future, we will add one-of groups (sometimes called unions). | ||||||
|  | type Struct struct { | ||||||
|  | 	// Each struct field appears exactly once in this list. The order in | ||||||
|  | 	// this list defines the canonical field ordering. | ||||||
|  | 	Fields []StructField `yaml:"fields,omitempty"` | ||||||
|  |  | ||||||
|  | 	// TODO: Implement unions, either this way or by inlining. | ||||||
|  | 	// Unions are groupings of fields with special rules. They may refer to | ||||||
|  | 	// one or more fields in the above list. A given field from the above | ||||||
|  | 	// list may be referenced in exactly 0 or 1 places in the below list. | ||||||
|  | 	// Unions []Union `yaml:"unions,omitempty"` | ||||||
|  |  | ||||||
|  | 	// ElementRelationship states the relationship between the struct's items. | ||||||
|  | 	// * `separable` (or unset) implies that each element is 100% independent. | ||||||
|  | 	// * `atomic` implies that all elements depend on each other, and this | ||||||
|  | 	//   is effectively a scalar / leaf field; it doesn't make sense for | ||||||
|  | 	//   separate actors to set the elements. Example: an RGB color struct; | ||||||
|  | 	//   it would never make sense to "own" only one component of the | ||||||
|  | 	//   color. | ||||||
|  | 	// The default behavior for structs is `separable`; it's permitted to | ||||||
|  | 	// leave this unset to get the default behavior. | ||||||
|  | 	ElementRelationship ElementRelationship `yaml:"elementRelationship,omitempty"` | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // StructField pairs a field name with a field type. | ||||||
|  | type StructField struct { | ||||||
|  | 	// Name is the field name. | ||||||
|  | 	Name string `yaml:"name,omitempty"` | ||||||
|  | 	// Type is the field type. | ||||||
|  | 	Type TypeRef `yaml:"type,omitempty"` | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // List represents a type which contains a zero or more elements, all of the | ||||||
|  | // same subtype. Lists may be either associative: each element is more or less | ||||||
|  | // independent and could be managed by separate entities in the system; or | ||||||
|  | // atomic, where the elements are heavily dependent on each other: it is not | ||||||
|  | // sensible to change one element without considering the ramifications on all | ||||||
|  | // the other elements. | ||||||
|  | type List struct { | ||||||
|  | 	// ElementType is the type of the list's elements. | ||||||
|  | 	ElementType TypeRef `yaml:"elementType,omitempty"` | ||||||
|  |  | ||||||
|  | 	// ElementRelationship states the relationship between the list's elements | ||||||
|  | 	// and must have one of these values: | ||||||
|  | 	// * `atomic`: the list is treated as a single entity, like a scalar. | ||||||
|  | 	// * `associative`: | ||||||
|  | 	//   - If the list element is a scalar, the list is treated as a set. | ||||||
|  | 	//   - If the list element is a struct, the list is treated as a map. | ||||||
|  | 	//   - The list element must not be a map or a list itself. | ||||||
|  | 	// There is no default for this value for lists; all schemas must | ||||||
|  | 	// explicitly state the element relationship for all lists. | ||||||
|  | 	ElementRelationship ElementRelationship `yaml:"elementRelationship,omitempty"` | ||||||
|  |  | ||||||
|  | 	// Iff ElementRelationship is `associative`, and the element type is | ||||||
|  | 	// struct, then Keys must have non-zero length, and it lists the fields | ||||||
|  | 	// of the element's struct type which are to be used as the keys of the | ||||||
|  | 	// list. | ||||||
|  | 	// | ||||||
|  | 	// TODO: change this to "non-atomic struct" above and make the code reflect this. | ||||||
|  | 	// | ||||||
|  | 	// Each key must refer to a single field name (no nesting, not JSONPath). | ||||||
|  | 	Keys []string `yaml:"keys,omitempty"` | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // Map is a key-value pair. Its default semantics are the same as an | ||||||
|  | // associative list, but: | ||||||
|  | // * It is serialized differently: | ||||||
|  | //     map:  {"k": {"value": "v"}} | ||||||
|  | //     list: [{"key": "k", "value": "v"}] | ||||||
|  | // * Keys must be string typed. | ||||||
|  | // * Keys can't have multiple components. | ||||||
|  | // | ||||||
|  | // Although serialized the same, maps are different from structs in that each | ||||||
|  | // map item must have the same type. | ||||||
|  | // | ||||||
|  | // Optionally, maps may be atomic (for example, imagine representing an RGB | ||||||
|  | // color value--it doesn't make sense to have different actors own the R and G | ||||||
|  | // values). | ||||||
|  | type Map struct { | ||||||
|  | 	// ElementType is the type of the list's elements. | ||||||
|  | 	ElementType TypeRef `yaml:"elementType,omitempty"` | ||||||
|  |  | ||||||
|  | 	// ElementRelationship states the relationship between the map's items. | ||||||
|  | 	// * `separable` implies that each element is 100% independent. | ||||||
|  | 	// * `atomic` implies that all elements depend on each other, and this | ||||||
|  | 	//   is effectively a scalar / leaf field; it doesn't make sense for | ||||||
|  | 	//   separate actors to set the elements. | ||||||
|  | 	//   TODO: find a simple example. | ||||||
|  | 	// The default behavior for maps is `separable`; it's permitted to | ||||||
|  | 	// leave this unset to get the default behavior. | ||||||
|  | 	ElementRelationship ElementRelationship `yaml:"elementRelationship,omitempty"` | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // Untyped represents types that allow arbitrary content. (Think: plugin | ||||||
|  | // objects.) | ||||||
|  | type Untyped struct { | ||||||
|  | 	// ElementRelationship states the relationship between the items, if | ||||||
|  | 	// container-typed data happens to be present here. | ||||||
|  | 	// * `atomic` implies that all elements depend on each other, and this | ||||||
|  | 	//   is effectively a scalar / leaf field; it doesn't make sense for | ||||||
|  | 	//   separate actors to set the elements. | ||||||
|  | 	// TODO: support "guess" (guesses at associative list keys) | ||||||
|  | 	// TODO: support "lookup" (calls a lookup function to figure out the | ||||||
|  | 	//       schema based on the data) | ||||||
|  | 	// The default behavior for untyped data is `atomic`; it's permitted to | ||||||
|  | 	// leave this unset to get the default behavior. | ||||||
|  | 	ElementRelationship ElementRelationship `yaml:"elementRelationship,omitempty"` | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // FindNamedType is a convenience function that returns the referenced TypeDef, | ||||||
|  | // if it exists, or (nil, false) if it doesn't. | ||||||
|  | func (s Schema) FindNamedType(name string) (TypeDef, bool) { | ||||||
|  | 	for _, t := range s.Types { | ||||||
|  | 		if t.Name == name { | ||||||
|  | 			return t, true | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	return TypeDef{}, false | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // Resolve is a convenience function which returns the atom referenced, whether | ||||||
|  | // it is inline or named. Returns (Atom{}, false) if the type can't be resolved. | ||||||
|  | // | ||||||
|  | // This allows callers to not care about the difference between a (possibly | ||||||
|  | // inlined) reference and a definition. | ||||||
|  | func (s Schema) Resolve(tr TypeRef) (Atom, bool) { | ||||||
|  | 	if tr.NamedType != nil { | ||||||
|  | 		t, ok := s.FindNamedType(*tr.NamedType) | ||||||
|  | 		if !ok { | ||||||
|  | 			return Atom{}, false | ||||||
|  | 		} | ||||||
|  | 		return t.Atom, true | ||||||
|  | 	} | ||||||
|  | 	return tr.Inlined, true | ||||||
|  | } | ||||||
							
								
								
									
										128
									
								
								vendor/sigs.k8s.io/structured-merge-diff/schema/schemaschema.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										128
									
								
								vendor/sigs.k8s.io/structured-merge-diff/schema/schemaschema.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,128 @@ | |||||||
|  | /* | ||||||
|  | Copyright 2018 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 schema | ||||||
|  |  | ||||||
|  | // SchemaSchemaYAML is a schema against which you can validate other schemas. | ||||||
|  | // It will validate itself. It can be unmarshalled into a Schema type. | ||||||
|  | var SchemaSchemaYAML = `types: | ||||||
|  | - name: schema | ||||||
|  |   struct: | ||||||
|  |     fields: | ||||||
|  |       - name: types | ||||||
|  |         type: | ||||||
|  |           list: | ||||||
|  |             elementRelationship: associative | ||||||
|  |             elementType: | ||||||
|  |               namedType: typeDef | ||||||
|  |             keys: | ||||||
|  |             - name | ||||||
|  | - name: typeDef | ||||||
|  |   struct: | ||||||
|  |     fields: | ||||||
|  |     - name: name | ||||||
|  |       type: | ||||||
|  |         scalar: string | ||||||
|  |     - name: scalar | ||||||
|  |       type: | ||||||
|  |         scalar: string | ||||||
|  |     - name: struct | ||||||
|  |       type: | ||||||
|  |         namedType: struct | ||||||
|  |     - name: list | ||||||
|  |       type: | ||||||
|  |         namedType: list | ||||||
|  |     - name: map | ||||||
|  |       type: | ||||||
|  |         namedType: map | ||||||
|  |     - name: untyped | ||||||
|  |       type: | ||||||
|  |         namedType: untyped | ||||||
|  | - name: typeRef | ||||||
|  |   struct: | ||||||
|  |     fields: | ||||||
|  |     - name: namedType | ||||||
|  |       type: | ||||||
|  |         scalar: string | ||||||
|  |     - name: scalar | ||||||
|  |       type: | ||||||
|  |         scalar: string | ||||||
|  |     - name: struct | ||||||
|  |       type: | ||||||
|  |         namedType: struct | ||||||
|  |     - name: list | ||||||
|  |       type: | ||||||
|  |         namedType: list | ||||||
|  |     - name: map | ||||||
|  |       type: | ||||||
|  |         namedType: map | ||||||
|  |     - name: untyped | ||||||
|  |       type: | ||||||
|  |         namedType: untyped | ||||||
|  | - name: scalar | ||||||
|  |   scalar: string | ||||||
|  | - name: struct | ||||||
|  |   struct: | ||||||
|  |     fields: | ||||||
|  |     - name: fields | ||||||
|  |       type: | ||||||
|  |         list: | ||||||
|  |           elementType: | ||||||
|  |             namedType: structField | ||||||
|  |           elementRelationship: associative | ||||||
|  |           keys: [ "name" ] | ||||||
|  |     - name: elementRelationship | ||||||
|  |       type: | ||||||
|  |         scalar: string | ||||||
|  | - name: structField | ||||||
|  |   struct: | ||||||
|  |     fields: | ||||||
|  |     - name: name | ||||||
|  |       type: | ||||||
|  |         scalar: string | ||||||
|  |     - name: type | ||||||
|  |       type: | ||||||
|  |         namedType: typeRef | ||||||
|  | - name: list | ||||||
|  |   struct: | ||||||
|  |     fields: | ||||||
|  |     - name: elementType | ||||||
|  |       type: | ||||||
|  |         namedType: typeRef | ||||||
|  |     - name: elementRelationship | ||||||
|  |       type: | ||||||
|  |         scalar: string | ||||||
|  |     - name: keys | ||||||
|  |       type: | ||||||
|  |         list: | ||||||
|  |           elementType: | ||||||
|  |             scalar: string | ||||||
|  | - name: map | ||||||
|  |   struct: | ||||||
|  |     fields: | ||||||
|  |     - name: elementType | ||||||
|  |       type: | ||||||
|  |         namedType: typeRef | ||||||
|  |     - name: elementRelationship | ||||||
|  |       type: | ||||||
|  |         scalar: string | ||||||
|  | - name: untyped | ||||||
|  |   struct: | ||||||
|  |     fields: | ||||||
|  |     - name: elementRelationship | ||||||
|  |       type: | ||||||
|  |         scalar: string | ||||||
|  | ` | ||||||
							
								
								
									
										36
									
								
								vendor/sigs.k8s.io/structured-merge-diff/typed/BUILD
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										36
									
								
								vendor/sigs.k8s.io/structured-merge-diff/typed/BUILD
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,36 @@ | |||||||
|  | load("@io_bazel_rules_go//go:def.bzl", "go_library") | ||||||
|  |  | ||||||
|  | go_library( | ||||||
|  |     name = "go_default_library", | ||||||
|  |     srcs = [ | ||||||
|  |         "doc.go", | ||||||
|  |         "helpers.go", | ||||||
|  |         "merge.go", | ||||||
|  |         "parser.go", | ||||||
|  |         "typed.go", | ||||||
|  |         "validate.go", | ||||||
|  |     ], | ||||||
|  |     importmap = "k8s.io/kubernetes/vendor/sigs.k8s.io/structured-merge-diff/typed", | ||||||
|  |     importpath = "sigs.k8s.io/structured-merge-diff/typed", | ||||||
|  |     visibility = ["//visibility:public"], | ||||||
|  |     deps = [ | ||||||
|  |         "//vendor/gopkg.in/yaml.v2:go_default_library", | ||||||
|  |         "//vendor/sigs.k8s.io/structured-merge-diff/fieldpath:go_default_library", | ||||||
|  |         "//vendor/sigs.k8s.io/structured-merge-diff/schema:go_default_library", | ||||||
|  |         "//vendor/sigs.k8s.io/structured-merge-diff/value:go_default_library", | ||||||
|  |     ], | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | filegroup( | ||||||
|  |     name = "package-srcs", | ||||||
|  |     srcs = glob(["**"]), | ||||||
|  |     tags = ["automanaged"], | ||||||
|  |     visibility = ["//visibility:private"], | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | filegroup( | ||||||
|  |     name = "all-srcs", | ||||||
|  |     srcs = [":package-srcs"], | ||||||
|  |     tags = ["automanaged"], | ||||||
|  |     visibility = ["//visibility:public"], | ||||||
|  | ) | ||||||
							
								
								
									
										18
									
								
								vendor/sigs.k8s.io/structured-merge-diff/typed/doc.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										18
									
								
								vendor/sigs.k8s.io/structured-merge-diff/typed/doc.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,18 @@ | |||||||
|  | /* | ||||||
|  | Copyright 2018 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 typed contains logic for operating on values with given schemas. | ||||||
|  | package typed | ||||||
							
								
								
									
										246
									
								
								vendor/sigs.k8s.io/structured-merge-diff/typed/helpers.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										246
									
								
								vendor/sigs.k8s.io/structured-merge-diff/typed/helpers.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,246 @@ | |||||||
|  | /* | ||||||
|  | Copyright 2018 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 typed | ||||||
|  |  | ||||||
|  | import ( | ||||||
|  | 	"errors" | ||||||
|  | 	"fmt" | ||||||
|  | 	"strings" | ||||||
|  |  | ||||||
|  | 	"sigs.k8s.io/structured-merge-diff/fieldpath" | ||||||
|  | 	"sigs.k8s.io/structured-merge-diff/schema" | ||||||
|  | 	"sigs.k8s.io/structured-merge-diff/value" | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | // ValidationError reports an error about a particular field | ||||||
|  | type ValidationError struct { | ||||||
|  | 	Path         fieldpath.Path | ||||||
|  | 	ErrorMessage string | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // Error returns a human readable error message. | ||||||
|  | func (ve ValidationError) Error() string { | ||||||
|  | 	if len(ve.Path) == 0 { | ||||||
|  | 		return ve.ErrorMessage | ||||||
|  | 	} | ||||||
|  | 	return fmt.Sprintf("%s: %v", ve.Path, ve.ErrorMessage) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // ValidationErrors accumulates multiple validation error messages. | ||||||
|  | type ValidationErrors []ValidationError | ||||||
|  |  | ||||||
|  | // Error returns a human readable error message reporting each error in the | ||||||
|  | // list. | ||||||
|  | func (errs ValidationErrors) Error() string { | ||||||
|  | 	if len(errs) == 1 { | ||||||
|  | 		return errs[0].Error() | ||||||
|  | 	} | ||||||
|  | 	messages := []string{"errors:"} | ||||||
|  | 	for _, e := range errs { | ||||||
|  | 		messages = append(messages, "  "+e.Error()) | ||||||
|  | 	} | ||||||
|  | 	return strings.Join(messages, "\n") | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // errorFormatter makes it easy to keep a list of validation errors. They | ||||||
|  | // should all be packed into a single error object before leaving the package | ||||||
|  | // boundary, since it's weird to have functions not return a plain error type. | ||||||
|  | type errorFormatter struct { | ||||||
|  | 	path fieldpath.Path | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (ef *errorFormatter) descend(pe fieldpath.PathElement) { | ||||||
|  | 	ef.path = append(ef.path, pe) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (ef errorFormatter) errorf(format string, args ...interface{}) ValidationErrors { | ||||||
|  | 	return ValidationErrors{{ | ||||||
|  | 		Path:         append(fieldpath.Path{}, ef.path...), | ||||||
|  | 		ErrorMessage: fmt.Sprintf(format, args...), | ||||||
|  | 	}} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (ef errorFormatter) error(err error) ValidationErrors { | ||||||
|  | 	return ValidationErrors{{ | ||||||
|  | 		Path:         append(fieldpath.Path{}, ef.path...), | ||||||
|  | 		ErrorMessage: err.Error(), | ||||||
|  | 	}} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (ef errorFormatter) prefixError(prefix string, err error) ValidationErrors { | ||||||
|  | 	return ValidationErrors{{ | ||||||
|  | 		Path:         append(fieldpath.Path{}, ef.path...), | ||||||
|  | 		ErrorMessage: prefix + err.Error(), | ||||||
|  | 	}} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | type atomHandler interface { | ||||||
|  | 	doScalar(schema.Scalar) ValidationErrors | ||||||
|  | 	doStruct(schema.Struct) ValidationErrors | ||||||
|  | 	doList(schema.List) ValidationErrors | ||||||
|  | 	doMap(schema.Map) ValidationErrors | ||||||
|  | 	doUntyped(schema.Untyped) ValidationErrors | ||||||
|  |  | ||||||
|  | 	errorf(msg string, args ...interface{}) ValidationErrors | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func resolveSchema(s *schema.Schema, tr schema.TypeRef, ah atomHandler) ValidationErrors { | ||||||
|  | 	a, ok := s.Resolve(tr) | ||||||
|  | 	if !ok { | ||||||
|  | 		return ah.errorf("schema error: no type found matching: %v", *tr.NamedType) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	switch { | ||||||
|  | 	case a.Scalar != nil: | ||||||
|  | 		return ah.doScalar(*a.Scalar) | ||||||
|  | 	case a.Struct != nil: | ||||||
|  | 		return ah.doStruct(*a.Struct) | ||||||
|  | 	case a.List != nil: | ||||||
|  | 		return ah.doList(*a.List) | ||||||
|  | 	case a.Map != nil: | ||||||
|  | 		return ah.doMap(*a.Map) | ||||||
|  | 	case a.Untyped != nil: | ||||||
|  | 		return ah.doUntyped(*a.Untyped) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	name := "inlined" | ||||||
|  | 	if tr.NamedType != nil { | ||||||
|  | 		name = "named type: " + *tr.NamedType | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return ah.errorf("schema error: invalid atom: %v", name) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (ef errorFormatter) validateScalar(t schema.Scalar, v *value.Value, prefix string) (errs ValidationErrors) { | ||||||
|  | 	if v == nil { | ||||||
|  | 		return nil | ||||||
|  | 	} | ||||||
|  | 	switch t { | ||||||
|  | 	case schema.Numeric: | ||||||
|  | 		if v.FloatValue == nil && v.IntValue == nil { | ||||||
|  | 			// TODO: should the schema separate int and float? | ||||||
|  | 			return ef.errorf("%vexpected numeric (int or float), got %v", prefix, v) | ||||||
|  | 		} | ||||||
|  | 	case schema.String: | ||||||
|  | 		if v.StringValue == nil { | ||||||
|  | 			return ef.errorf("%vexpected string, got %v", prefix, v) | ||||||
|  | 		} | ||||||
|  | 	case schema.Boolean: | ||||||
|  | 		if v.BooleanValue == nil { | ||||||
|  | 			return ef.errorf("%vexpected boolean, got %v", prefix, v) | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	return nil | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // Returns the list, or an error. Reminder: nil is a valid list and might be returned. | ||||||
|  | func listValue(val value.Value) (*value.List, error) { | ||||||
|  | 	switch { | ||||||
|  | 	case val.Null: | ||||||
|  | 		// Null is a valid list. | ||||||
|  | 		return nil, nil | ||||||
|  | 	case val.ListValue != nil: | ||||||
|  | 		return val.ListValue, nil | ||||||
|  | 	default: | ||||||
|  | 		return nil, fmt.Errorf("expected list, got %v", val) | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // Returns the map, or an error. Reminder: nil is a valid map and might be returned. | ||||||
|  | func mapOrStructValue(val value.Value, typeName string) (*value.Map, error) { | ||||||
|  | 	switch { | ||||||
|  | 	case val.Null: | ||||||
|  | 		return nil, nil | ||||||
|  | 	case val.MapValue != nil: | ||||||
|  | 		return val.MapValue, nil | ||||||
|  | 	default: | ||||||
|  | 		return nil, fmt.Errorf("expected %v, got %v", typeName, val) | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (ef errorFormatter) rejectExtraStructFields(m *value.Map, allowedNames map[string]struct{}, prefix string) (errs ValidationErrors) { | ||||||
|  | 	if m == nil { | ||||||
|  | 		return nil | ||||||
|  | 	} | ||||||
|  | 	for _, f := range m.Items { | ||||||
|  | 		if _, allowed := allowedNames[f.Name]; !allowed { | ||||||
|  | 			errs = append(errs, ef.errorf("%vfield %v is not mentioned in the schema", prefix, f.Name)...) | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	return errs | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func keyedAssociativeListItemToPathElement(list schema.List, index int, child value.Value) (fieldpath.PathElement, error) { | ||||||
|  | 	pe := fieldpath.PathElement{} | ||||||
|  | 	if child.Null { | ||||||
|  | 		// For now, the keys are required which means that null entries | ||||||
|  | 		// are illegal. | ||||||
|  | 		return pe, errors.New("associative list with keys may not have a null element") | ||||||
|  | 	} | ||||||
|  | 	if child.MapValue == nil { | ||||||
|  | 		return pe, errors.New("associative list with keys may not have non-map elements") | ||||||
|  | 	} | ||||||
|  | 	for _, fieldName := range list.Keys { | ||||||
|  | 		var fieldValue value.Value | ||||||
|  | 		field, ok := child.MapValue.Get(fieldName) | ||||||
|  | 		if ok { | ||||||
|  | 			fieldValue = field.Value | ||||||
|  | 		} else { | ||||||
|  | 			// Treat keys as required. | ||||||
|  | 			return pe, fmt.Errorf("associative list with keys has an element that omits key field %q", fieldName) | ||||||
|  | 		} | ||||||
|  | 		pe.Key = append(pe.Key, value.Field{ | ||||||
|  | 			Name:  fieldName, | ||||||
|  | 			Value: fieldValue, | ||||||
|  | 		}) | ||||||
|  | 	} | ||||||
|  | 	return pe, nil | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func setItemToPathElement(list schema.List, index int, child value.Value) (fieldpath.PathElement, error) { | ||||||
|  | 	pe := fieldpath.PathElement{} | ||||||
|  | 	switch { | ||||||
|  | 	case child.MapValue != nil: | ||||||
|  | 		// TODO: atomic maps should be acceptable. | ||||||
|  | 		return pe, errors.New("associative list without keys has an element that's a map type") | ||||||
|  | 	case child.ListValue != nil: | ||||||
|  | 		// Should we support a set of lists? For the moment | ||||||
|  | 		// let's say we don't. | ||||||
|  | 		// TODO: atomic lists should be acceptable. | ||||||
|  | 		return pe, errors.New("not supported: associative list with lists as elements") | ||||||
|  | 	case child.Null: | ||||||
|  | 		return pe, errors.New("associative list without keys has an element that's an explicit null") | ||||||
|  | 	default: | ||||||
|  | 		// We are a set type. | ||||||
|  | 		pe.Value = &child | ||||||
|  | 		return pe, nil | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func listItemToPathElement(list schema.List, index int, child value.Value) (fieldpath.PathElement, error) { | ||||||
|  | 	if list.ElementRelationship == schema.Associative { | ||||||
|  | 		if len(list.Keys) > 0 { | ||||||
|  | 			return keyedAssociativeListItemToPathElement(list, index, child) | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		// If there's no keys, then we must be a set of primitives. | ||||||
|  | 		return setItemToPathElement(list, index, child) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	// Use the index as a key for atomic lists. | ||||||
|  | 	return fieldpath.PathElement{Index: &index}, nil | ||||||
|  | } | ||||||
							
								
								
									
										410
									
								
								vendor/sigs.k8s.io/structured-merge-diff/typed/merge.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										410
									
								
								vendor/sigs.k8s.io/structured-merge-diff/typed/merge.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,410 @@ | |||||||
|  | /* | ||||||
|  | Copyright 2018 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 typed | ||||||
|  |  | ||||||
|  | import ( | ||||||
|  | 	"sigs.k8s.io/structured-merge-diff/fieldpath" | ||||||
|  | 	"sigs.k8s.io/structured-merge-diff/schema" | ||||||
|  | 	"sigs.k8s.io/structured-merge-diff/value" | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | type mergingWalker struct { | ||||||
|  | 	errorFormatter | ||||||
|  | 	lhs     *value.Value | ||||||
|  | 	rhs     *value.Value | ||||||
|  | 	schema  *schema.Schema | ||||||
|  | 	typeRef schema.TypeRef | ||||||
|  |  | ||||||
|  | 	// How to merge. Called after schema validation for all leaf fields. | ||||||
|  | 	rule mergeRule | ||||||
|  |  | ||||||
|  | 	// If set, called after non-leaf items have been merged. (`out` is | ||||||
|  | 	// probably already set.) | ||||||
|  | 	postItemHook mergeRule | ||||||
|  |  | ||||||
|  | 	// output of the merge operation (nil if none) | ||||||
|  | 	out *value.Value | ||||||
|  |  | ||||||
|  | 	// internal housekeeping--don't set when constructing. | ||||||
|  | 	inLeaf bool // Set to true if we're in a "big leaf"--atomic map/list | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // merge rules examine w.lhs and w.rhs (up to one of which may be nil) and | ||||||
|  | // optionally set w.out. If lhs and rhs are both set, they will be of | ||||||
|  | // comparable type. | ||||||
|  | type mergeRule func(w *mergingWalker) | ||||||
|  |  | ||||||
|  | var ( | ||||||
|  | 	ruleKeepRHS = mergeRule(func(w *mergingWalker) { | ||||||
|  | 		if w.rhs != nil { | ||||||
|  | 			v := *w.rhs | ||||||
|  | 			w.out = &v | ||||||
|  | 		} else if w.lhs != nil { | ||||||
|  | 			v := *w.lhs | ||||||
|  | 			w.out = &v | ||||||
|  | 		} | ||||||
|  | 	}) | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | // merge sets w.out. | ||||||
|  | func (w *mergingWalker) merge() ValidationErrors { | ||||||
|  | 	if w.lhs == nil && w.rhs == nil { | ||||||
|  | 		// check this condidition here instead of everywhere below. | ||||||
|  | 		return w.errorf("at least one of lhs and rhs must be provided") | ||||||
|  | 	} | ||||||
|  | 	errs := resolveSchema(w.schema, w.typeRef, w) | ||||||
|  | 	if !w.inLeaf && w.postItemHook != nil { | ||||||
|  | 		w.postItemHook(w) | ||||||
|  | 	} | ||||||
|  | 	return errs | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // doLeaf should be called on leaves before descending into children, if there | ||||||
|  | // will be a descent. It modifies w.inLeaf. | ||||||
|  | func (w *mergingWalker) doLeaf() { | ||||||
|  | 	if w.inLeaf { | ||||||
|  | 		// We're in a "big leaf", an atomic map or list. Ignore | ||||||
|  | 		// subsequent leaves. | ||||||
|  | 		return | ||||||
|  | 	} | ||||||
|  | 	w.inLeaf = true | ||||||
|  |  | ||||||
|  | 	// We don't recurse into leaf fields for merging. | ||||||
|  | 	w.rule(w) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (w *mergingWalker) doScalar(t schema.Scalar) (errs ValidationErrors) { | ||||||
|  | 	errs = append(errs, w.validateScalar(t, w.lhs, "lhs: ")...) | ||||||
|  | 	errs = append(errs, w.validateScalar(t, w.rhs, "rhs: ")...) | ||||||
|  | 	if len(errs) > 0 { | ||||||
|  | 		return errs | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	// All scalars are leaf fields. | ||||||
|  | 	w.doLeaf() | ||||||
|  |  | ||||||
|  | 	return nil | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (w *mergingWalker) prepareDescent(pe fieldpath.PathElement, tr schema.TypeRef) *mergingWalker { | ||||||
|  | 	w2 := *w | ||||||
|  | 	w2.typeRef = tr | ||||||
|  | 	w2.errorFormatter.descend(pe) | ||||||
|  | 	w2.lhs = nil | ||||||
|  | 	w2.rhs = nil | ||||||
|  | 	w2.out = nil | ||||||
|  | 	return &w2 | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (w *mergingWalker) visitStructFields(t schema.Struct, lhs, rhs *value.Map) (errs ValidationErrors) { | ||||||
|  | 	out := &value.Map{} | ||||||
|  |  | ||||||
|  | 	valOrNil := func(m *value.Map, name string) *value.Value { | ||||||
|  | 		if m == nil { | ||||||
|  | 			return nil | ||||||
|  | 		} | ||||||
|  | 		val, ok := m.Get(name) | ||||||
|  | 		if ok { | ||||||
|  | 			return &val.Value | ||||||
|  | 		} | ||||||
|  | 		return nil | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	allowedNames := map[string]struct{}{} | ||||||
|  | 	for i := range t.Fields { | ||||||
|  | 		// I don't want to use the loop variable since a reference | ||||||
|  | 		// might outlive the loop iteration (in an error message). | ||||||
|  | 		f := t.Fields[i] | ||||||
|  | 		allowedNames[f.Name] = struct{}{} | ||||||
|  | 		w2 := w.prepareDescent(fieldpath.PathElement{FieldName: &f.Name}, f.Type) | ||||||
|  | 		w2.lhs = valOrNil(lhs, f.Name) | ||||||
|  | 		w2.rhs = valOrNil(rhs, f.Name) | ||||||
|  | 		if w2.lhs == nil && w2.rhs == nil { | ||||||
|  | 			// All fields are optional | ||||||
|  | 			continue | ||||||
|  | 		} | ||||||
|  | 		if newErrs := w2.merge(); len(newErrs) > 0 { | ||||||
|  | 			errs = append(errs, newErrs...) | ||||||
|  | 		} else if w2.out != nil { | ||||||
|  | 			out.Set(f.Name, *w2.out) | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	// All fields may be optional, but unknown fields are not allowed. | ||||||
|  | 	errs = append(errs, w.rejectExtraStructFields(lhs, allowedNames, "lhs: ")...) | ||||||
|  | 	errs = append(errs, w.rejectExtraStructFields(rhs, allowedNames, "rhs: ")...) | ||||||
|  | 	if len(errs) > 0 { | ||||||
|  | 		return errs | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if len(out.Items) > 0 { | ||||||
|  | 		w.out = &value.Value{MapValue: out} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return errs | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (w *mergingWalker) derefMapOrStruct(prefix, typeName string, v *value.Value, dest **value.Map) (errs ValidationErrors) { | ||||||
|  | 	// taking dest as input so that it can be called as a one-liner with | ||||||
|  | 	// append. | ||||||
|  | 	if v == nil { | ||||||
|  | 		return nil | ||||||
|  | 	} | ||||||
|  | 	m, err := mapOrStructValue(*v, typeName) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return w.prefixError(prefix, err) | ||||||
|  | 	} | ||||||
|  | 	*dest = m | ||||||
|  | 	return nil | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (w *mergingWalker) doStruct(t schema.Struct) (errs ValidationErrors) { | ||||||
|  | 	var lhs, rhs *value.Map | ||||||
|  | 	errs = append(errs, w.derefMapOrStruct("lhs: ", "struct", w.lhs, &lhs)...) | ||||||
|  | 	errs = append(errs, w.derefMapOrStruct("rhs: ", "struct", w.rhs, &rhs)...) | ||||||
|  | 	if len(errs) > 0 { | ||||||
|  | 		return errs | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	// If both lhs and rhs are empty/null, treat it as a | ||||||
|  | 	// leaf: this helps preserve the empty/null | ||||||
|  | 	// distinction. | ||||||
|  | 	emptyPromoteToLeaf := (lhs == nil || len(lhs.Items) == 0) && | ||||||
|  | 		(rhs == nil || len(rhs.Items) == 0) | ||||||
|  |  | ||||||
|  | 	if t.ElementRelationship == schema.Atomic || emptyPromoteToLeaf { | ||||||
|  | 		w.doLeaf() | ||||||
|  | 		return nil | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if lhs == nil && rhs == nil { | ||||||
|  | 		// nil is a valid map! | ||||||
|  | 		return nil | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	errs = w.visitStructFields(t, lhs, rhs) | ||||||
|  |  | ||||||
|  | 	// TODO: Check unions. | ||||||
|  |  | ||||||
|  | 	return errs | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (w *mergingWalker) visitListItems(t schema.List, lhs, rhs *value.List) (errs ValidationErrors) { | ||||||
|  | 	out := &value.List{} | ||||||
|  |  | ||||||
|  | 	// TODO: ordering is totally wrong. | ||||||
|  | 	// TODO: might as well make the map order work the same way. | ||||||
|  |  | ||||||
|  | 	// This is a cheap hack to at least make the output order stable. | ||||||
|  | 	rhsOrder := []fieldpath.PathElement{} | ||||||
|  |  | ||||||
|  | 	// First, collect all RHS children. | ||||||
|  | 	observedRHS := map[string]value.Value{} | ||||||
|  | 	if rhs != nil { | ||||||
|  | 		for i, child := range rhs.Items { | ||||||
|  | 			pe, err := listItemToPathElement(t, i, child) | ||||||
|  | 			if err != nil { | ||||||
|  | 				errs = append(errs, w.errorf("rhs: element %v: %v", i, err.Error())...) | ||||||
|  | 				// If we can't construct the path element, we can't | ||||||
|  | 				// even report errors deeper in the schema, so bail on | ||||||
|  | 				// this element. | ||||||
|  | 				continue | ||||||
|  | 			} | ||||||
|  | 			keyStr := pe.String() | ||||||
|  | 			if _, found := observedRHS[keyStr]; found { | ||||||
|  | 				errs = append(errs, w.errorf("rhs: duplicate entries for key %v", keyStr)...) | ||||||
|  | 			} | ||||||
|  | 			observedRHS[keyStr] = child | ||||||
|  | 			rhsOrder = append(rhsOrder, pe) | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	// Then merge with LHS children. | ||||||
|  | 	observedLHS := map[string]struct{}{} | ||||||
|  | 	if lhs != nil { | ||||||
|  | 		for i, child := range lhs.Items { | ||||||
|  | 			pe, err := listItemToPathElement(t, i, child) | ||||||
|  | 			if err != nil { | ||||||
|  | 				errs = append(errs, w.errorf("lhs: element %v: %v", i, err.Error())...) | ||||||
|  | 				// If we can't construct the path element, we can't | ||||||
|  | 				// even report errors deeper in the schema, so bail on | ||||||
|  | 				// this element. | ||||||
|  | 				continue | ||||||
|  | 			} | ||||||
|  | 			keyStr := pe.String() | ||||||
|  | 			if _, found := observedLHS[keyStr]; found { | ||||||
|  | 				errs = append(errs, w.errorf("lhs: duplicate entries for key %v", keyStr)...) | ||||||
|  | 				continue | ||||||
|  | 			} | ||||||
|  | 			observedLHS[keyStr] = struct{}{} | ||||||
|  | 			w2 := w.prepareDescent(pe, t.ElementType) | ||||||
|  | 			w2.lhs = &child | ||||||
|  | 			if rchild, ok := observedRHS[keyStr]; ok { | ||||||
|  | 				w2.rhs = &rchild | ||||||
|  | 			} | ||||||
|  | 			if newErrs := w2.merge(); len(newErrs) > 0 { | ||||||
|  | 				errs = append(errs, newErrs...) | ||||||
|  | 			} else if w2.out != nil { | ||||||
|  | 				out.Items = append(out.Items, *w2.out) | ||||||
|  | 			} | ||||||
|  | 			// Keep track of children that have been handled | ||||||
|  | 			delete(observedRHS, keyStr) | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	for _, rhsToCheck := range rhsOrder { | ||||||
|  | 		if unmergedChild, ok := observedRHS[rhsToCheck.String()]; ok { | ||||||
|  | 			w2 := w.prepareDescent(rhsToCheck, t.ElementType) | ||||||
|  | 			w2.rhs = &unmergedChild | ||||||
|  | 			if newErrs := w2.merge(); len(newErrs) > 0 { | ||||||
|  | 				errs = append(errs, newErrs...) | ||||||
|  | 			} else if w2.out != nil { | ||||||
|  | 				out.Items = append(out.Items, *w2.out) | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if len(out.Items) > 0 { | ||||||
|  | 		w.out = &value.Value{ListValue: out} | ||||||
|  | 	} | ||||||
|  | 	return errs | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (w *mergingWalker) derefList(prefix string, v *value.Value, dest **value.List) (errs ValidationErrors) { | ||||||
|  | 	// taking dest as input so that it can be called as a one-liner with | ||||||
|  | 	// append. | ||||||
|  | 	if v == nil { | ||||||
|  | 		return nil | ||||||
|  | 	} | ||||||
|  | 	l, err := listValue(*v) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return w.prefixError(prefix, err) | ||||||
|  | 	} | ||||||
|  | 	*dest = l | ||||||
|  | 	return nil | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (w *mergingWalker) doList(t schema.List) (errs ValidationErrors) { | ||||||
|  | 	var lhs, rhs *value.List | ||||||
|  | 	errs = append(errs, w.derefList("lhs: ", w.lhs, &lhs)...) | ||||||
|  | 	errs = append(errs, w.derefList("rhs: ", w.rhs, &rhs)...) | ||||||
|  | 	if len(errs) > 0 { | ||||||
|  | 		return errs | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	// If both lhs and rhs are empty/null, treat it as a | ||||||
|  | 	// leaf: this helps preserve the empty/null | ||||||
|  | 	// distinction. | ||||||
|  | 	emptyPromoteToLeaf := (lhs == nil || len(lhs.Items) == 0) && | ||||||
|  | 		(rhs == nil || len(rhs.Items) == 0) | ||||||
|  |  | ||||||
|  | 	if t.ElementRelationship == schema.Atomic || emptyPromoteToLeaf { | ||||||
|  | 		w.doLeaf() | ||||||
|  | 		return nil | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if lhs == nil && rhs == nil { | ||||||
|  | 		return nil | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	errs = w.visitListItems(t, lhs, rhs) | ||||||
|  |  | ||||||
|  | 	return errs | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (w *mergingWalker) visitMapItems(t schema.Map, lhs, rhs *value.Map) (errs ValidationErrors) { | ||||||
|  | 	out := &value.Map{} | ||||||
|  |  | ||||||
|  | 	if lhs != nil { | ||||||
|  | 		for _, litem := range lhs.Items { | ||||||
|  | 			name := litem.Name | ||||||
|  | 			w2 := w.prepareDescent(fieldpath.PathElement{FieldName: &name}, t.ElementType) | ||||||
|  | 			w2.lhs = &litem.Value | ||||||
|  | 			if rhs != nil { | ||||||
|  | 				if ritem, ok := rhs.Get(litem.Name); ok { | ||||||
|  | 					w2.rhs = &ritem.Value | ||||||
|  | 				} | ||||||
|  | 			} | ||||||
|  | 			if newErrs := w2.merge(); len(newErrs) > 0 { | ||||||
|  | 				errs = append(errs, newErrs...) | ||||||
|  | 			} else if w2.out != nil { | ||||||
|  | 				out.Set(name, *w2.out) | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if rhs != nil { | ||||||
|  | 		for _, ritem := range rhs.Items { | ||||||
|  | 			if lhs != nil { | ||||||
|  | 				if _, ok := lhs.Get(ritem.Name); ok { | ||||||
|  | 					continue | ||||||
|  | 				} | ||||||
|  | 			} | ||||||
|  |  | ||||||
|  | 			name := ritem.Name | ||||||
|  | 			w2 := w.prepareDescent(fieldpath.PathElement{FieldName: &name}, t.ElementType) | ||||||
|  | 			w2.rhs = &ritem.Value | ||||||
|  | 			if newErrs := w2.merge(); len(newErrs) > 0 { | ||||||
|  | 				errs = append(errs, newErrs...) | ||||||
|  | 			} else if w2.out != nil { | ||||||
|  | 				out.Set(name, *w2.out) | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if len(out.Items) > 0 { | ||||||
|  | 		w.out = &value.Value{MapValue: out} | ||||||
|  | 	} | ||||||
|  | 	return errs | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (w *mergingWalker) doMap(t schema.Map) (errs ValidationErrors) { | ||||||
|  | 	var lhs, rhs *value.Map | ||||||
|  | 	errs = append(errs, w.derefMapOrStruct("lhs: ", "map", w.lhs, &lhs)...) | ||||||
|  | 	errs = append(errs, w.derefMapOrStruct("rhs: ", "map", w.rhs, &rhs)...) | ||||||
|  | 	if len(errs) > 0 { | ||||||
|  | 		return errs | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	// If both lhs and rhs are empty/null, treat it as a | ||||||
|  | 	// leaf: this helps preserve the empty/null | ||||||
|  | 	// distinction. | ||||||
|  | 	emptyPromoteToLeaf := (lhs == nil || len(lhs.Items) == 0) && | ||||||
|  | 		(rhs == nil || len(rhs.Items) == 0) | ||||||
|  |  | ||||||
|  | 	if t.ElementRelationship == schema.Atomic || emptyPromoteToLeaf { | ||||||
|  | 		w.doLeaf() | ||||||
|  | 		return nil | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if lhs == nil && rhs == nil { | ||||||
|  | 		return nil | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	errs = w.visitMapItems(t, lhs, rhs) | ||||||
|  |  | ||||||
|  | 	return errs | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (w *mergingWalker) doUntyped(t schema.Untyped) (errs ValidationErrors) { | ||||||
|  | 	if t.ElementRelationship == "" || t.ElementRelationship == schema.Atomic { | ||||||
|  | 		// Untyped sections allow anything, and are considered leaf | ||||||
|  | 		// fields. | ||||||
|  | 		w.doLeaf() | ||||||
|  | 	} | ||||||
|  | 	return nil | ||||||
|  | } | ||||||
							
								
								
									
										114
									
								
								vendor/sigs.k8s.io/structured-merge-diff/typed/parser.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										114
									
								
								vendor/sigs.k8s.io/structured-merge-diff/typed/parser.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,114 @@ | |||||||
|  | /* | ||||||
|  | Copyright 2018 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 typed | ||||||
|  |  | ||||||
|  | import ( | ||||||
|  | 	"fmt" | ||||||
|  |  | ||||||
|  | 	yaml "gopkg.in/yaml.v2" | ||||||
|  | 	"sigs.k8s.io/structured-merge-diff/schema" | ||||||
|  | 	"sigs.k8s.io/structured-merge-diff/value" | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | // YAMLObject is an object encoded in YAML. | ||||||
|  | type YAMLObject string | ||||||
|  |  | ||||||
|  | // Parser implements YAMLParser and allows introspecting the schema. | ||||||
|  | type Parser struct { | ||||||
|  | 	Schema schema.Schema | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // create builds an unvalidated parser. | ||||||
|  | func create(schema YAMLObject) (*Parser, error) { | ||||||
|  | 	p := Parser{} | ||||||
|  | 	err := yaml.Unmarshal([]byte(schema), &p.Schema) | ||||||
|  | 	return &p, err | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func createOrDie(schema YAMLObject) *Parser { | ||||||
|  | 	p, err := create(schema) | ||||||
|  | 	if err != nil { | ||||||
|  | 		panic(fmt.Errorf("failed to create parser: %v", err)) | ||||||
|  | 	} | ||||||
|  | 	return p | ||||||
|  | } | ||||||
|  |  | ||||||
|  | var ssParser = createOrDie(YAMLObject(schema.SchemaSchemaYAML)) | ||||||
|  |  | ||||||
|  | // NewParser will build a YAMLParser from a schema. The schema is validated. | ||||||
|  | func NewParser(schema YAMLObject) (*Parser, error) { | ||||||
|  | 	_, err := ssParser.Type("schema").FromYAML(schema) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return nil, fmt.Errorf("unable to validate schema: %v", err) | ||||||
|  | 	} | ||||||
|  | 	return create(schema) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // TypeNames returns a list of types this parser understands. | ||||||
|  | func (p *Parser) TypeNames() (names []string) { | ||||||
|  | 	for _, td := range p.Schema.Types { | ||||||
|  | 		names = append(names, td.Name) | ||||||
|  | 	} | ||||||
|  | 	return names | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // Type returns a helper which can produce objects of the given type. Any | ||||||
|  | // errors are deferred until a further function is called. | ||||||
|  | func (p *Parser) Type(name string) *ParseableType { | ||||||
|  | 	return &ParseableType{ | ||||||
|  | 		parser:   p, | ||||||
|  | 		typename: name, | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // ParseableType allows for easy production of typed objects. | ||||||
|  | type ParseableType struct { | ||||||
|  | 	parser   *Parser | ||||||
|  | 	typename string | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // IsValid return true if p's schema and typename are valid. | ||||||
|  | func (p *ParseableType) IsValid() bool { | ||||||
|  | 	_, ok := p.parser.Schema.Resolve(schema.TypeRef{NamedType: &p.typename}) | ||||||
|  | 	return ok | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // New returns a new empty object with the current schema and the | ||||||
|  | // type "typename". | ||||||
|  | func (p *ParseableType) New() (TypedValue, error) { | ||||||
|  | 	return p.FromYAML(YAMLObject("{}")) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // FromYAML parses a yaml string into an object with the current schema | ||||||
|  | // and the type "typename" or an error if validation fails. | ||||||
|  | func (p *ParseableType) FromYAML(object YAMLObject) (TypedValue, error) { | ||||||
|  | 	v, err := value.FromYAML([]byte(object)) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return TypedValue{}, err | ||||||
|  | 	} | ||||||
|  | 	return AsTyped(v, &p.parser.Schema, p.typename) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // FromUnstructured converts a go interface to a TypedValue. It will return an | ||||||
|  | // error if the resulting object fails schema validation. | ||||||
|  | func (p *ParseableType) FromUnstructured(in interface{}) (TypedValue, error) { | ||||||
|  | 	v, err := value.FromUnstructured(in) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return TypedValue{}, err | ||||||
|  | 	} | ||||||
|  | 	return AsTyped(v, &p.parser.Schema, p.typename) | ||||||
|  | } | ||||||
							
								
								
									
										213
									
								
								vendor/sigs.k8s.io/structured-merge-diff/typed/typed.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										213
									
								
								vendor/sigs.k8s.io/structured-merge-diff/typed/typed.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,213 @@ | |||||||
|  | /* | ||||||
|  | Copyright 2018 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 typed | ||||||
|  |  | ||||||
|  | import ( | ||||||
|  | 	"fmt" | ||||||
|  | 	"reflect" | ||||||
|  |  | ||||||
|  | 	"sigs.k8s.io/structured-merge-diff/fieldpath" | ||||||
|  | 	"sigs.k8s.io/structured-merge-diff/schema" | ||||||
|  | 	"sigs.k8s.io/structured-merge-diff/value" | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | // TypedValue is a value of some specific type. | ||||||
|  | type TypedValue struct { | ||||||
|  | 	value   value.Value | ||||||
|  | 	typeRef schema.TypeRef | ||||||
|  | 	schema  *schema.Schema | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // AsTyped accepts a value and a type and returns a TypedValue. 'v' must have | ||||||
|  | // type 'typeName' in the schema. An error is returned if the v doesn't conform | ||||||
|  | // to the schema. | ||||||
|  | func AsTyped(v value.Value, s *schema.Schema, typeName string) (TypedValue, error) { | ||||||
|  | 	tv := TypedValue{ | ||||||
|  | 		value:   v, | ||||||
|  | 		typeRef: schema.TypeRef{NamedType: &typeName}, | ||||||
|  | 		schema:  s, | ||||||
|  | 	} | ||||||
|  | 	if err := tv.Validate(); err != nil { | ||||||
|  | 		return TypedValue{}, err | ||||||
|  | 	} | ||||||
|  | 	return tv, nil | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // AsValue removes the type from the TypedValue and only keeps the value. | ||||||
|  | func (tv TypedValue) AsValue() *value.Value { | ||||||
|  | 	return &tv.value | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // Validate returns an error with a list of every spec violation. | ||||||
|  | func (tv TypedValue) Validate() error { | ||||||
|  | 	if errs := tv.walker().validate(); len(errs) != 0 { | ||||||
|  | 		return errs | ||||||
|  | 	} | ||||||
|  | 	return nil | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // ToFieldSet creates a set containing every leaf field mentioned in tv, or | ||||||
|  | // validation errors, if any were encountered. | ||||||
|  | func (tv TypedValue) ToFieldSet() (*fieldpath.Set, error) { | ||||||
|  | 	s := fieldpath.NewSet() | ||||||
|  | 	w := tv.walker() | ||||||
|  | 	w.leafFieldCallback = func(p fieldpath.Path) { s.Insert(p) } | ||||||
|  | 	if errs := w.validate(); len(errs) != 0 { | ||||||
|  | 		return nil, errs | ||||||
|  | 	} | ||||||
|  | 	return s, nil | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // Merge returns the result of merging tv and pso ("partially specified | ||||||
|  | // object") together. Of note: | ||||||
|  | //  * No fields can be removed by this operation. | ||||||
|  | //  * If both tv and pso specify a given leaf field, the result will keep pso's | ||||||
|  | //    value. | ||||||
|  | //  * Container typed elements will have their items ordered: | ||||||
|  | //    * like tv, if pso doesn't change anything in the container | ||||||
|  | //    * like pso, if pso does change something in the container. | ||||||
|  | // tv and pso must both be of the same type (their Schema and TypeRef must | ||||||
|  | // match), or an error will be returned. Validation errors will be returned if | ||||||
|  | // the objects don't conform to the schema. | ||||||
|  | func (tv TypedValue) Merge(pso TypedValue) (TypedValue, error) { | ||||||
|  | 	return merge(tv, pso, ruleKeepRHS, nil) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // Comparison is the return value of a TypedValue.Compare() operation. | ||||||
|  | // | ||||||
|  | // No field will appear in more than one of the three fieldsets. If all of the | ||||||
|  | // fieldsets are empty, then the objects must have been equal. | ||||||
|  | type Comparison struct { | ||||||
|  | 	// Merged is the result of merging the two objects, as explained in the | ||||||
|  | 	// comments on TypedValue.Merge(). | ||||||
|  | 	Merged TypedValue | ||||||
|  |  | ||||||
|  | 	// Removed contains any fields removed by rhs (the right-hand-side | ||||||
|  | 	// object in the comparison). | ||||||
|  | 	Removed *fieldpath.Set | ||||||
|  | 	// Modified contains fields present in both objects but different. | ||||||
|  | 	Modified *fieldpath.Set | ||||||
|  | 	// Added contains any fields added by rhs. | ||||||
|  | 	Added *fieldpath.Set | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // IsSame returns true if the comparison returned no changes (the two | ||||||
|  | // compared objects are similar). | ||||||
|  | func (c *Comparison) IsSame() bool { | ||||||
|  | 	return c.Removed.Empty() && c.Modified.Empty() && c.Added.Empty() | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // String returns a human readable version of the comparison. | ||||||
|  | func (c *Comparison) String() string { | ||||||
|  | 	str := fmt.Sprintf("- Merged Object:\n%v\n", c.Merged.AsValue()) | ||||||
|  | 	if !c.Modified.Empty() { | ||||||
|  | 		str += fmt.Sprintf("- Modified Fields:\n%v\n", c.Modified) | ||||||
|  | 	} | ||||||
|  | 	if !c.Added.Empty() { | ||||||
|  | 		str += fmt.Sprintf("- Added Fields:\n%v\n", c.Added) | ||||||
|  | 	} | ||||||
|  | 	if !c.Removed.Empty() { | ||||||
|  | 		str += fmt.Sprintf("- Removed Fields:\n%v\n", c.Removed) | ||||||
|  | 	} | ||||||
|  | 	return str | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // Compare compares the two objects. See the comments on the `Comparison` | ||||||
|  | // struct for details on the return value. | ||||||
|  | // | ||||||
|  | // tv and rhs must both be of the same type (their Schema and TypeRef must | ||||||
|  | // match), or an error will be returned. Validation errors will be returned if | ||||||
|  | // the objects don't conform to the schema. | ||||||
|  | func (tv TypedValue) Compare(rhs TypedValue) (c *Comparison, err error) { | ||||||
|  | 	c = &Comparison{ | ||||||
|  | 		Removed:  fieldpath.NewSet(), | ||||||
|  | 		Modified: fieldpath.NewSet(), | ||||||
|  | 		Added:    fieldpath.NewSet(), | ||||||
|  | 	} | ||||||
|  | 	c.Merged, err = merge(tv, rhs, func(w *mergingWalker) { | ||||||
|  | 		if w.lhs == nil { | ||||||
|  | 			c.Added.Insert(w.path) | ||||||
|  | 		} else if w.rhs == nil { | ||||||
|  | 			c.Removed.Insert(w.path) | ||||||
|  | 		} else if !reflect.DeepEqual(w.rhs, w.lhs) { | ||||||
|  | 			// TODO: reflect.DeepEqual is not sufficient for this. | ||||||
|  | 			// Need to implement equality check on the value type. | ||||||
|  | 			c.Modified.Insert(w.path) | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		ruleKeepRHS(w) | ||||||
|  | 	}, func(w *mergingWalker) { | ||||||
|  | 		if w.lhs == nil { | ||||||
|  | 			c.Added.Insert(w.path) | ||||||
|  | 		} else if w.rhs == nil { | ||||||
|  | 			c.Removed.Insert(w.path) | ||||||
|  | 		} | ||||||
|  | 	}) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return nil, err | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	return c, nil | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func merge(lhs, rhs TypedValue, rule, postRule mergeRule) (TypedValue, error) { | ||||||
|  | 	if lhs.schema != rhs.schema { | ||||||
|  | 		return TypedValue{}, errorFormatter{}. | ||||||
|  | 			errorf("expected objects with types from the same schema") | ||||||
|  | 	} | ||||||
|  | 	if !reflect.DeepEqual(lhs.typeRef, rhs.typeRef) { | ||||||
|  | 		return TypedValue{}, errorFormatter{}. | ||||||
|  | 			errorf("expected objects of the same type, but got %v and %v", lhs.typeRef, rhs.typeRef) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	mw := mergingWalker{ | ||||||
|  | 		lhs:          &lhs.value, | ||||||
|  | 		rhs:          &rhs.value, | ||||||
|  | 		schema:       lhs.schema, | ||||||
|  | 		typeRef:      lhs.typeRef, | ||||||
|  | 		rule:         rule, | ||||||
|  | 		postItemHook: postRule, | ||||||
|  | 	} | ||||||
|  | 	errs := mw.merge() | ||||||
|  | 	if len(errs) > 0 { | ||||||
|  | 		return TypedValue{}, errs | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	out := TypedValue{ | ||||||
|  | 		schema:  lhs.schema, | ||||||
|  | 		typeRef: lhs.typeRef, | ||||||
|  | 	} | ||||||
|  | 	if mw.out == nil { | ||||||
|  | 		out.value = value.Value{Null: true} | ||||||
|  | 	} else { | ||||||
|  | 		out.value = *mw.out | ||||||
|  | 	} | ||||||
|  | 	return out, nil | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // AsTypeUnvalidated is just like WithType, but doesn't validate that the type | ||||||
|  | // conforms to the schema, for cases where that has already been checked or | ||||||
|  | // where you're going to call a method that validates as a side-effect (like | ||||||
|  | // ToFieldSet). | ||||||
|  | func AsTypedUnvalidated(v value.Value, s *schema.Schema, typeName string) TypedValue { | ||||||
|  | 	tv := TypedValue{ | ||||||
|  | 		value:   v, | ||||||
|  | 		typeRef: schema.TypeRef{NamedType: &typeName}, | ||||||
|  | 		schema:  s, | ||||||
|  | 	} | ||||||
|  | 	return tv | ||||||
|  | } | ||||||
							
								
								
									
										208
									
								
								vendor/sigs.k8s.io/structured-merge-diff/typed/validate.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										208
									
								
								vendor/sigs.k8s.io/structured-merge-diff/typed/validate.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,208 @@ | |||||||
|  | /* | ||||||
|  | Copyright 2018 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 typed | ||||||
|  |  | ||||||
|  | import ( | ||||||
|  | 	"sigs.k8s.io/structured-merge-diff/fieldpath" | ||||||
|  | 	"sigs.k8s.io/structured-merge-diff/schema" | ||||||
|  | 	"sigs.k8s.io/structured-merge-diff/value" | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | func (tv TypedValue) walker() *validatingObjectWalker { | ||||||
|  | 	return &validatingObjectWalker{ | ||||||
|  | 		value:   tv.value, | ||||||
|  | 		schema:  tv.schema, | ||||||
|  | 		typeRef: tv.typeRef, | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | type validatingObjectWalker struct { | ||||||
|  | 	errorFormatter | ||||||
|  | 	value   value.Value | ||||||
|  | 	schema  *schema.Schema | ||||||
|  | 	typeRef schema.TypeRef | ||||||
|  |  | ||||||
|  | 	// If set, this is called on "leaf fields": | ||||||
|  | 	//  * scalars: int/string/float/bool | ||||||
|  | 	//  * atomic maps and lists | ||||||
|  | 	//  * untyped fields | ||||||
|  | 	leafFieldCallback func(fieldpath.Path) | ||||||
|  |  | ||||||
|  | 	// internal housekeeping--don't set when constructing. | ||||||
|  | 	inLeaf bool // Set to true if we're in a "big leaf"--atomic map/list | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (v validatingObjectWalker) validate() ValidationErrors { | ||||||
|  | 	return resolveSchema(v.schema, v.typeRef, v) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // doLeaf should be called on leaves before descending into children, if there | ||||||
|  | // will be a descent. It modifies v.inLeaf. | ||||||
|  | func (v *validatingObjectWalker) doLeaf() { | ||||||
|  | 	if v.inLeaf { | ||||||
|  | 		// We're in a "big leaf", an atomic map or list. Ignore | ||||||
|  | 		// subsequent leaves. | ||||||
|  | 		return | ||||||
|  | 	} | ||||||
|  | 	v.inLeaf = true | ||||||
|  |  | ||||||
|  | 	if v.leafFieldCallback != nil { | ||||||
|  | 		// At the moment, this is only used to build fieldsets; we can | ||||||
|  | 		// add more than the path in here if needed. | ||||||
|  | 		v.leafFieldCallback(v.path) | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (v validatingObjectWalker) doScalar(t schema.Scalar) ValidationErrors { | ||||||
|  | 	if errs := v.validateScalar(t, &v.value, ""); len(errs) > 0 { | ||||||
|  | 		return errs | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	// All scalars are leaf fields. | ||||||
|  | 	v.doLeaf() | ||||||
|  |  | ||||||
|  | 	return nil | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (v validatingObjectWalker) visitStructFields(t schema.Struct, m *value.Map) (errs ValidationErrors) { | ||||||
|  | 	allowedNames := map[string]struct{}{} | ||||||
|  | 	for i := range t.Fields { | ||||||
|  | 		// I don't want to use the loop variable since a reference | ||||||
|  | 		// might outlive the loop iteration (in an error message). | ||||||
|  | 		f := t.Fields[i] | ||||||
|  | 		allowedNames[f.Name] = struct{}{} | ||||||
|  | 		child, ok := m.Get(f.Name) | ||||||
|  | 		if !ok { | ||||||
|  | 			// All fields are optional | ||||||
|  | 			continue | ||||||
|  | 		} | ||||||
|  | 		v2 := v | ||||||
|  | 		v2.errorFormatter.descend(fieldpath.PathElement{FieldName: &f.Name}) | ||||||
|  | 		v2.value = child.Value | ||||||
|  | 		v2.typeRef = f.Type | ||||||
|  | 		errs = append(errs, v2.validate()...) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	// All fields may be optional, but unknown fields are not allowed. | ||||||
|  | 	return append(errs, v.rejectExtraStructFields(m, allowedNames, "")...) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (v validatingObjectWalker) doStruct(t schema.Struct) (errs ValidationErrors) { | ||||||
|  | 	m, err := mapOrStructValue(v.value, "struct") | ||||||
|  | 	if err != nil { | ||||||
|  | 		return v.error(err) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if t.ElementRelationship == schema.Atomic { | ||||||
|  | 		v.doLeaf() | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if m == nil { | ||||||
|  | 		// nil is a valid map! | ||||||
|  | 		return nil | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	errs = v.visitStructFields(t, m) | ||||||
|  |  | ||||||
|  | 	// TODO: Check unions. | ||||||
|  |  | ||||||
|  | 	return errs | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (v validatingObjectWalker) visitListItems(t schema.List, list *value.List) (errs ValidationErrors) { | ||||||
|  | 	observedKeys := map[string]struct{}{} | ||||||
|  | 	for i, child := range list.Items { | ||||||
|  | 		pe, err := listItemToPathElement(t, i, child) | ||||||
|  | 		if err != nil { | ||||||
|  | 			errs = append(errs, v.errorf("element %v: %v", i, err.Error())...) | ||||||
|  | 			// If we can't construct the path element, we can't | ||||||
|  | 			// even report errors deeper in the schema, so bail on | ||||||
|  | 			// this element. | ||||||
|  | 			continue | ||||||
|  | 		} | ||||||
|  | 		keyStr := pe.String() | ||||||
|  | 		if _, found := observedKeys[keyStr]; found { | ||||||
|  | 			errs = append(errs, v.errorf("duplicate entries for key %v", keyStr)...) | ||||||
|  | 		} | ||||||
|  | 		observedKeys[keyStr] = struct{}{} | ||||||
|  | 		v2 := v | ||||||
|  | 		v2.errorFormatter.descend(pe) | ||||||
|  | 		v2.value = child | ||||||
|  | 		v2.typeRef = t.ElementType | ||||||
|  | 		errs = append(errs, v2.validate()...) | ||||||
|  | 	} | ||||||
|  | 	return errs | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (v validatingObjectWalker) doList(t schema.List) (errs ValidationErrors) { | ||||||
|  | 	list, err := listValue(v.value) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return v.error(err) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if t.ElementRelationship == schema.Atomic { | ||||||
|  | 		v.doLeaf() | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if list == nil { | ||||||
|  | 		return nil | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	errs = v.visitListItems(t, list) | ||||||
|  |  | ||||||
|  | 	return errs | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (v validatingObjectWalker) visitMapItems(t schema.Map, m *value.Map) (errs ValidationErrors) { | ||||||
|  | 	for _, item := range m.Items { | ||||||
|  | 		v2 := v | ||||||
|  | 		name := item.Name | ||||||
|  | 		v2.errorFormatter.descend(fieldpath.PathElement{FieldName: &name}) | ||||||
|  | 		v2.value = item.Value | ||||||
|  | 		v2.typeRef = t.ElementType | ||||||
|  | 		errs = append(errs, v2.validate()...) | ||||||
|  | 	} | ||||||
|  | 	return errs | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (v validatingObjectWalker) doMap(t schema.Map) (errs ValidationErrors) { | ||||||
|  | 	m, err := mapOrStructValue(v.value, "map") | ||||||
|  | 	if err != nil { | ||||||
|  | 		return v.error(err) | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if t.ElementRelationship == schema.Atomic { | ||||||
|  | 		v.doLeaf() | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if m == nil { | ||||||
|  | 		return nil | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	errs = v.visitMapItems(t, m) | ||||||
|  |  | ||||||
|  | 	return errs | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func (v validatingObjectWalker) doUntyped(t schema.Untyped) (errs ValidationErrors) { | ||||||
|  | 	if t.ElementRelationship == "" || t.ElementRelationship == schema.Atomic { | ||||||
|  | 		// Untyped sections allow anything, and are considered leaf | ||||||
|  | 		// fields. | ||||||
|  | 		v.doLeaf() | ||||||
|  | 	} | ||||||
|  | 	return nil | ||||||
|  | } | ||||||
							
								
								
									
										28
									
								
								vendor/sigs.k8s.io/structured-merge-diff/value/BUILD
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										28
									
								
								vendor/sigs.k8s.io/structured-merge-diff/value/BUILD
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,28 @@ | |||||||
|  | load("@io_bazel_rules_go//go:def.bzl", "go_library") | ||||||
|  |  | ||||||
|  | go_library( | ||||||
|  |     name = "go_default_library", | ||||||
|  |     srcs = [ | ||||||
|  |         "doc.go", | ||||||
|  |         "unstructured.go", | ||||||
|  |         "value.go", | ||||||
|  |     ], | ||||||
|  |     importmap = "k8s.io/kubernetes/vendor/sigs.k8s.io/structured-merge-diff/value", | ||||||
|  |     importpath = "sigs.k8s.io/structured-merge-diff/value", | ||||||
|  |     visibility = ["//visibility:public"], | ||||||
|  |     deps = ["//vendor/gopkg.in/yaml.v2:go_default_library"], | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | filegroup( | ||||||
|  |     name = "package-srcs", | ||||||
|  |     srcs = glob(["**"]), | ||||||
|  |     tags = ["automanaged"], | ||||||
|  |     visibility = ["//visibility:private"], | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | filegroup( | ||||||
|  |     name = "all-srcs", | ||||||
|  |     srcs = [":package-srcs"], | ||||||
|  |     tags = ["automanaged"], | ||||||
|  |     visibility = ["//visibility:public"], | ||||||
|  | ) | ||||||
							
								
								
									
										21
									
								
								vendor/sigs.k8s.io/structured-merge-diff/value/doc.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										21
									
								
								vendor/sigs.k8s.io/structured-merge-diff/value/doc.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,21 @@ | |||||||
|  | /* | ||||||
|  | Copyright 2018 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 value defines types for an in-memory representation of yaml or json | ||||||
|  | // objects, organized for convenient comparison with a schema (as defined by | ||||||
|  | // the sibling schema package). Functions for reading and writing the objects | ||||||
|  | // are also provided. | ||||||
|  | package value | ||||||
							
								
								
									
										234
									
								
								vendor/sigs.k8s.io/structured-merge-diff/value/unstructured.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										234
									
								
								vendor/sigs.k8s.io/structured-merge-diff/value/unstructured.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,234 @@ | |||||||
|  | /* | ||||||
|  | Copyright 2018 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 value | ||||||
|  |  | ||||||
|  | import ( | ||||||
|  | 	"encoding/json" | ||||||
|  | 	"fmt" | ||||||
|  |  | ||||||
|  | 	"gopkg.in/yaml.v2" | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | // FromYAML is a helper function for reading a YAML document; it attempts to | ||||||
|  | // preserve order of keys within maps/structs. This is as a convenience to | ||||||
|  | // humans keeping YAML documents, not because there is a behavior difference. | ||||||
|  | // | ||||||
|  | // Known bug: objects with top-level arrays don't parse correctly. | ||||||
|  | func FromYAML(input []byte) (Value, error) { | ||||||
|  | 	var decoded interface{} | ||||||
|  |  | ||||||
|  | 	if len(input) == 0 || (len(input) == 4 && string(input) == "null") { | ||||||
|  | 		// Special case since the yaml package doesn't accurately | ||||||
|  | 		// preserve this. | ||||||
|  | 		return Value{Null: true}, nil | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	// This attempts to enable order sensitivity; note the yaml package is | ||||||
|  | 	// broken for documents that have root-level arrays, hence the two-step | ||||||
|  | 	// approach. TODO: This is a horrific hack. Is it worth it? | ||||||
|  | 	var ms yaml.MapSlice | ||||||
|  | 	if err := yaml.Unmarshal(input, &ms); err == nil { | ||||||
|  | 		decoded = ms | ||||||
|  | 	} else if err := yaml.Unmarshal(input, &decoded); err != nil { | ||||||
|  | 		return Value{}, err | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	v, err := FromUnstructured(decoded) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return Value{}, fmt.Errorf("failed to interpret (%v):\n%s", err, input) | ||||||
|  | 	} | ||||||
|  | 	return v, nil | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // FromJSON is a helper function for reading a JSON document | ||||||
|  | func FromJSON(input []byte) (Value, error) { | ||||||
|  | 	var decoded interface{} | ||||||
|  |  | ||||||
|  | 	if err := json.Unmarshal(input, &decoded); err != nil { | ||||||
|  | 		return Value{}, err | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	v, err := FromUnstructured(decoded) | ||||||
|  | 	if err != nil { | ||||||
|  | 		return Value{}, fmt.Errorf("failed to interpret (%v):\n%s", err, input) | ||||||
|  | 	} | ||||||
|  | 	return v, nil | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // FromUnstructured will convert a go interface to a Value. | ||||||
|  | // It's most commonly expected to be used with map[string]interface{} as the | ||||||
|  | // input. `in` must not have any structures with cycles in them. | ||||||
|  | // yaml.MapSlice may be used for order-preservation. | ||||||
|  | func FromUnstructured(in interface{}) (Value, error) { | ||||||
|  | 	if in == nil { | ||||||
|  | 		return Value{Null: true}, nil | ||||||
|  | 	} | ||||||
|  | 	switch t := in.(type) { | ||||||
|  | 	case map[interface{}]interface{}: | ||||||
|  | 		m := Map{} | ||||||
|  | 		for rawKey, rawVal := range t { | ||||||
|  | 			k, ok := rawKey.(string) | ||||||
|  | 			if !ok { | ||||||
|  | 				return Value{}, fmt.Errorf("key %#v: not a string", k) | ||||||
|  | 			} | ||||||
|  | 			v, err := FromUnstructured(rawVal) | ||||||
|  | 			if err != nil { | ||||||
|  | 				return Value{}, fmt.Errorf("key %v: %v", k, err) | ||||||
|  | 			} | ||||||
|  | 			m.Set(k, v) | ||||||
|  | 		} | ||||||
|  | 		return Value{MapValue: &m}, nil | ||||||
|  | 	case map[string]interface{}: | ||||||
|  | 		m := Map{} | ||||||
|  | 		for k, rawVal := range t { | ||||||
|  | 			v, err := FromUnstructured(rawVal) | ||||||
|  | 			if err != nil { | ||||||
|  | 				return Value{}, fmt.Errorf("key %v: %v", k, err) | ||||||
|  | 			} | ||||||
|  | 			m.Set(k, v) | ||||||
|  | 		} | ||||||
|  | 		return Value{MapValue: &m}, nil | ||||||
|  | 	case yaml.MapSlice: | ||||||
|  | 		m := Map{} | ||||||
|  | 		for _, item := range t { | ||||||
|  | 			k, ok := item.Key.(string) | ||||||
|  | 			if !ok { | ||||||
|  | 				return Value{}, fmt.Errorf("key %#v is not a string", item.Key) | ||||||
|  | 			} | ||||||
|  | 			v, err := FromUnstructured(item.Value) | ||||||
|  | 			if err != nil { | ||||||
|  | 				return Value{}, fmt.Errorf("key %v: %v", k, err) | ||||||
|  | 			} | ||||||
|  | 			m.Set(k, v) | ||||||
|  | 		} | ||||||
|  | 		return Value{MapValue: &m}, nil | ||||||
|  | 	case []interface{}: | ||||||
|  | 		l := List{} | ||||||
|  | 		for i, rawVal := range t { | ||||||
|  | 			v, err := FromUnstructured(rawVal) | ||||||
|  | 			if err != nil { | ||||||
|  | 				return Value{}, fmt.Errorf("index %v: %v", i, err) | ||||||
|  | 			} | ||||||
|  | 			l.Items = append(l.Items, v) | ||||||
|  | 		} | ||||||
|  | 		return Value{ListValue: &l}, nil | ||||||
|  | 	case int: | ||||||
|  | 		n := Int(t) | ||||||
|  | 		return Value{IntValue: &n}, nil | ||||||
|  | 	case int8: | ||||||
|  | 		n := Int(t) | ||||||
|  | 		return Value{IntValue: &n}, nil | ||||||
|  | 	case int16: | ||||||
|  | 		n := Int(t) | ||||||
|  | 		return Value{IntValue: &n}, nil | ||||||
|  | 	case int32: | ||||||
|  | 		n := Int(t) | ||||||
|  | 		return Value{IntValue: &n}, nil | ||||||
|  | 	case int64: | ||||||
|  | 		n := Int(t) | ||||||
|  | 		return Value{IntValue: &n}, nil | ||||||
|  | 	case uint: | ||||||
|  | 		n := Int(t) | ||||||
|  | 		return Value{IntValue: &n}, nil | ||||||
|  | 	case uint8: | ||||||
|  | 		n := Int(t) | ||||||
|  | 		return Value{IntValue: &n}, nil | ||||||
|  | 	case uint16: | ||||||
|  | 		n := Int(t) | ||||||
|  | 		return Value{IntValue: &n}, nil | ||||||
|  | 	case uint32: | ||||||
|  | 		n := Int(t) | ||||||
|  | 		return Value{IntValue: &n}, nil | ||||||
|  | 	case float32: | ||||||
|  | 		f := Float(t) | ||||||
|  | 		return Value{FloatValue: &f}, nil | ||||||
|  | 	case float64: | ||||||
|  | 		f := Float(t) | ||||||
|  | 		return Value{FloatValue: &f}, nil | ||||||
|  | 	case string: | ||||||
|  | 		return StringValue(t), nil | ||||||
|  | 	case bool: | ||||||
|  | 		return BooleanValue(t), nil | ||||||
|  | 	default: | ||||||
|  | 		return Value{}, fmt.Errorf("type unimplemented: %t", in) | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // ToYAML is a helper function for producing a YAML document; it attempts to | ||||||
|  | // preserve order of keys within maps/structs. This is as a convenience to | ||||||
|  | // humans keeping YAML documents, not because there is a behavior difference. | ||||||
|  | func (v *Value) ToYAML() ([]byte, error) { | ||||||
|  | 	return yaml.Marshal(v.ToUnstructured(true)) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // ToJSON is a helper function for producing a JSon document. | ||||||
|  | func (v *Value) ToJSON() ([]byte, error) { | ||||||
|  | 	return json.Marshal(v.ToUnstructured(false)) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // ToUnstructured will convert the Value into a go-typed object. | ||||||
|  | // If preserveOrder is true, then maps will be converted to the yaml.MapSlice | ||||||
|  | // type. Otherwise, map[string]interface{} must be used-- this destroys | ||||||
|  | // ordering information and is not recommended if the result of this will be | ||||||
|  | // serialized. Other types: | ||||||
|  | // * list -> []interface{} | ||||||
|  | // * others -> corresponding go type, wrapped in an interface{} | ||||||
|  | // | ||||||
|  | // Of note, floats and ints will always come out as float64 and int64, | ||||||
|  | // respectively. | ||||||
|  | func (v *Value) ToUnstructured(preserveOrder bool) interface{} { | ||||||
|  | 	switch { | ||||||
|  | 	case v.FloatValue != nil: | ||||||
|  | 		f := float64(*v.FloatValue) | ||||||
|  | 		return f | ||||||
|  | 	case v.IntValue != nil: | ||||||
|  | 		i := int64(*v.IntValue) | ||||||
|  | 		return i | ||||||
|  | 	case v.StringValue != nil: | ||||||
|  | 		return string(*v.StringValue) | ||||||
|  | 	case v.BooleanValue != nil: | ||||||
|  | 		return bool(*v.BooleanValue) | ||||||
|  | 	case v.ListValue != nil: | ||||||
|  | 		out := []interface{}{} | ||||||
|  | 		for _, item := range v.ListValue.Items { | ||||||
|  | 			out = append(out, item.ToUnstructured(preserveOrder)) | ||||||
|  | 		} | ||||||
|  | 		return out | ||||||
|  | 	case v.MapValue != nil: | ||||||
|  | 		m := v.MapValue | ||||||
|  | 		if preserveOrder { | ||||||
|  | 			ms := make(yaml.MapSlice, len(m.Items)) | ||||||
|  | 			for i := range m.Items { | ||||||
|  | 				ms[i] = yaml.MapItem{ | ||||||
|  | 					Key:   m.Items[i].Name, | ||||||
|  | 					Value: m.Items[i].Value.ToUnstructured(preserveOrder), | ||||||
|  | 				} | ||||||
|  | 			} | ||||||
|  | 			return ms | ||||||
|  | 		} | ||||||
|  | 		// This case is unavoidably lossy. | ||||||
|  | 		out := map[string]interface{}{} | ||||||
|  | 		for i := range m.Items { | ||||||
|  | 			out[m.Items[i].Name] = m.Items[i].Value.ToUnstructured(preserveOrder) | ||||||
|  | 		} | ||||||
|  | 		return out | ||||||
|  | 	default: | ||||||
|  | 		fallthrough | ||||||
|  | 	case v.Null == true: | ||||||
|  | 		return nil | ||||||
|  | 	} | ||||||
|  | } | ||||||
							
								
								
									
										139
									
								
								vendor/sigs.k8s.io/structured-merge-diff/value/value.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										139
									
								
								vendor/sigs.k8s.io/structured-merge-diff/value/value.go
									
									
									
										generated
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,139 @@ | |||||||
|  | /* | ||||||
|  | Copyright 2018 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 value | ||||||
|  |  | ||||||
|  | import ( | ||||||
|  | 	"fmt" | ||||||
|  | 	"strings" | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | // A Value is an object; it corresponds to an 'atom' in the schema. | ||||||
|  | type Value struct { | ||||||
|  | 	// Exactly one of the below must be set. | ||||||
|  | 	FloatValue   *Float | ||||||
|  | 	IntValue     *Int | ||||||
|  | 	StringValue  *String | ||||||
|  | 	BooleanValue *Boolean | ||||||
|  | 	ListValue    *List | ||||||
|  | 	MapValue     *Map | ||||||
|  | 	Null         bool // represents an explicit `"foo" = null` | ||||||
|  | } | ||||||
|  |  | ||||||
|  | type Int int64 | ||||||
|  | type Float float64 | ||||||
|  | type String string | ||||||
|  | type Boolean bool | ||||||
|  |  | ||||||
|  | // Field is an individual key-value pair. | ||||||
|  | type Field struct { | ||||||
|  | 	Name  string | ||||||
|  | 	Value Value | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // List is a list of items. | ||||||
|  | type List struct { | ||||||
|  | 	Items []Value | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // Map is a map of key-value pairs. It represents both structs and maps. We use | ||||||
|  | // a list and a go-language map to preserve order. | ||||||
|  | // | ||||||
|  | // Set and Get helpers are provided. | ||||||
|  | type Map struct { | ||||||
|  | 	Items []Field | ||||||
|  |  | ||||||
|  | 	// may be nil; lazily constructed. | ||||||
|  | 	// TODO: Direct modifications to Items above will cause serious problems. | ||||||
|  | 	index map[string]*Field | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // Get returns the (Field, true) or (nil, false) if it is not present | ||||||
|  | func (m *Map) Get(key string) (*Field, bool) { | ||||||
|  | 	if m.index == nil { | ||||||
|  | 		m.index = map[string]*Field{} | ||||||
|  | 		for i := range m.Items { | ||||||
|  | 			f := &m.Items[i] | ||||||
|  | 			m.index[f.Name] = f | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	f, ok := m.index[key] | ||||||
|  | 	return f, ok | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // Set inserts or updates the given item. | ||||||
|  | func (m *Map) Set(key string, value Value) { | ||||||
|  | 	if f, ok := m.Get(key); ok { | ||||||
|  | 		f.Value = value | ||||||
|  | 		return | ||||||
|  | 	} | ||||||
|  | 	m.Items = append(m.Items, Field{Name: key, Value: value}) | ||||||
|  | 	m.index = nil // Since the append might have reallocated | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // StringValue returns s as a scalar string Value. | ||||||
|  | func StringValue(s string) Value { | ||||||
|  | 	s2 := String(s) | ||||||
|  | 	return Value{StringValue: &s2} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // IntValue returns i as a scalar numeric (integer) Value. | ||||||
|  | func IntValue(i int) Value { | ||||||
|  | 	i2 := Int(i) | ||||||
|  | 	return Value{IntValue: &i2} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // FloatValue returns f as a scalar numeric (float) Value. | ||||||
|  | func FloatValue(f float64) Value { | ||||||
|  | 	f2 := Float(f) | ||||||
|  | 	return Value{FloatValue: &f2} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // BooleanValue returns b as a scalar boolean Value. | ||||||
|  | func BooleanValue(b bool) Value { | ||||||
|  | 	b2 := Boolean(b) | ||||||
|  | 	return Value{BooleanValue: &b2} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // String returns a human-readable representation of the value. | ||||||
|  | func (v Value) String() string { | ||||||
|  | 	switch { | ||||||
|  | 	case v.FloatValue != nil: | ||||||
|  | 		return fmt.Sprintf("%v", *v.FloatValue) | ||||||
|  | 	case v.IntValue != nil: | ||||||
|  | 		return fmt.Sprintf("%v", *v.IntValue) | ||||||
|  | 	case v.StringValue != nil: | ||||||
|  | 		return fmt.Sprintf("%q", *v.StringValue) | ||||||
|  | 	case v.BooleanValue != nil: | ||||||
|  | 		return fmt.Sprintf("%v", *v.BooleanValue) | ||||||
|  | 	case v.ListValue != nil: | ||||||
|  | 		strs := []string{} | ||||||
|  | 		for _, item := range v.ListValue.Items { | ||||||
|  | 			strs = append(strs, item.String()) | ||||||
|  | 		} | ||||||
|  | 		return "[" + strings.Join(strs, ",") + "]" | ||||||
|  | 	case v.MapValue != nil: | ||||||
|  | 		strs := []string{} | ||||||
|  | 		for _, i := range v.MapValue.Items { | ||||||
|  | 			strs = append(strs, fmt.Sprintf("%v=%v", i.Name, i.Value)) | ||||||
|  | 		} | ||||||
|  | 		return "{" + strings.Join(strs, ";") + "}" | ||||||
|  | 	default: | ||||||
|  | 		fallthrough | ||||||
|  | 	case v.Null == true: | ||||||
|  | 		return "null" | ||||||
|  | 	} | ||||||
|  | } | ||||||
		Reference in New Issue
	
	Block a user