mirror of
				https://github.com/docker/compose.git
				synced 2025-10-26 16:53:53 +01:00 
			
		
		
		
	- remove intermediate compose interface Signed-off-by: aiordache <anca.iordache@docker.com>
		
			
				
	
	
		
			183 lines
		
	
	
		
			5.3 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			183 lines
		
	
	
		
			5.3 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| // +build kube
 | |
| 
 | |
| /*
 | |
|    Copyright 2020 Docker Compose CLI 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 resources
 | |
| 
 | |
| import (
 | |
| 	"reflect"
 | |
| 	"testing"
 | |
| 
 | |
| 	"github.com/compose-spec/compose-go/types"
 | |
| 
 | |
| 	"github.com/stretchr/testify/assert"
 | |
| 	apiv1 "k8s.io/api/core/v1"
 | |
| )
 | |
| 
 | |
| /* FIXME
 | |
| func TestToPodWithPlacement(t *testing.T) {
 | |
| 	podTemplate := podTemplate(t, `
 | |
| version: "3"
 | |
| services:
 | |
|   redis:
 | |
|     image: redis:alpine
 | |
|     deploy:
 | |
|       placement:
 | |
|         constraints:
 | |
|           - node.platform.os == linux
 | |
|           - node.platform.arch == amd64
 | |
|           - node.hostname == node01
 | |
|           - node.labels.label1 == value1
 | |
|           - node.labels.label2.subpath != value2
 | |
| `)
 | |
| 
 | |
| 	expectedRequirements := []apiv1.NodeSelectorRequirement{
 | |
| 		{Key: "beta.kubernetes.io/os", Operator: apiv1.NodeSelectorOpIn, Values: []string{"linux"}},
 | |
| 		{Key: "beta.kubernetes.io/arch", Operator: apiv1.NodeSelectorOpIn, Values: []string{"amd64"}},
 | |
| 		{Key: "kubernetes.io/hostname", Operator: apiv1.NodeSelectorOpIn, Values: []string{"node01"}},
 | |
| 		{Key: "label1", Operator: apiv1.NodeSelectorOpIn, Values: []string{"value1"}},
 | |
| 		{Key: "label2.subpath", Operator: apiv1.NodeSelectorOpNotIn, Values: []string{"value2"}},
 | |
| 	}
 | |
| 
 | |
| 	requirements := podTemplate.Spec.Affinity.NodeAffinity.RequiredDuringSchedulingIgnoredDuringExecution.NodeSelectorTerms[0].MatchExpressions
 | |
| 
 | |
| 	sort.Slice(expectedRequirements, func(i, j int) bool { return expectedRequirements[i].Key < expectedRequirements[j].Key })
 | |
| 	sort.Slice(requirements, func(i, j int) bool { return requirements[i].Key < requirements[j].Key })
 | |
| 
 | |
| 	assert.EqualValues(t, expectedRequirements, requirements)
 | |
| }
 | |
| */
 | |
| 
 | |
| type keyValue struct {
 | |
| 	key   string
 | |
| 	value string
 | |
| }
 | |
| 
 | |
| func kv(key, value string) keyValue {
 | |
| 	return keyValue{key: key, value: value}
 | |
| }
 | |
| 
 | |
| func makeExpectedAffinity(kvs ...keyValue) *apiv1.Affinity {
 | |
| 
 | |
| 	var matchExpressions []apiv1.NodeSelectorRequirement
 | |
| 	for _, kv := range kvs {
 | |
| 		matchExpressions = append(
 | |
| 			matchExpressions,
 | |
| 			apiv1.NodeSelectorRequirement{
 | |
| 				Key:      kv.key,
 | |
| 				Operator: apiv1.NodeSelectorOpIn,
 | |
| 				Values:   []string{kv.value},
 | |
| 			},
 | |
| 		)
 | |
| 	}
 | |
| 	return &apiv1.Affinity{
 | |
| 		NodeAffinity: &apiv1.NodeAffinity{
 | |
| 			RequiredDuringSchedulingIgnoredDuringExecution: &apiv1.NodeSelector{
 | |
| 				NodeSelectorTerms: []apiv1.NodeSelectorTerm{
 | |
| 					{
 | |
| 						MatchExpressions: matchExpressions,
 | |
| 					},
 | |
| 				},
 | |
| 			},
 | |
| 		},
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func TestNodeAfinity(t *testing.T) {
 | |
| 	cases := []struct {
 | |
| 		name     string
 | |
| 		source   []string
 | |
| 		expected *apiv1.Affinity
 | |
| 	}{
 | |
| 		{
 | |
| 			name: "nil",
 | |
| 			expected: makeExpectedAffinity(
 | |
| 				kv(kubernetesOs, "linux"),
 | |
| 				kv(kubernetesArch, "amd64"),
 | |
| 			),
 | |
| 		},
 | |
| 		{
 | |
| 			name:   "hostname",
 | |
| 			source: []string{"node.hostname == test"},
 | |
| 			expected: makeExpectedAffinity(
 | |
| 				kv(kubernetesHostname, "test"),
 | |
| 				kv(kubernetesOs, "linux"),
 | |
| 				kv(kubernetesArch, "amd64"),
 | |
| 			),
 | |
| 		},
 | |
| 		{
 | |
| 			name:   "os",
 | |
| 			source: []string{"node.platform.os == windows"},
 | |
| 			expected: makeExpectedAffinity(
 | |
| 				kv(kubernetesOs, "windows"),
 | |
| 				kv(kubernetesArch, "amd64"),
 | |
| 			),
 | |
| 		},
 | |
| 		{
 | |
| 			name:   "arch",
 | |
| 			source: []string{"node.platform.arch == arm64"},
 | |
| 			expected: makeExpectedAffinity(
 | |
| 				kv(kubernetesArch, "arm64"),
 | |
| 				kv(kubernetesOs, "linux"),
 | |
| 			),
 | |
| 		},
 | |
| 		{
 | |
| 			name:   "custom-labels",
 | |
| 			source: []string{"node.platform.os == windows", "node.platform.arch == arm64"},
 | |
| 			expected: makeExpectedAffinity(
 | |
| 				kv(kubernetesArch, "arm64"),
 | |
| 				kv(kubernetesOs, "windows"),
 | |
| 			),
 | |
| 		},
 | |
| 	}
 | |
| 	for _, c := range cases {
 | |
| 		t.Run(c.name, func(t *testing.T) {
 | |
| 			result, err := toNodeAffinity(&types.DeployConfig{
 | |
| 				Placement: types.Placement{
 | |
| 					Constraints: c.source,
 | |
| 				},
 | |
| 			})
 | |
| 			assert.NoError(t, err)
 | |
| 			assert.True(t, nodeAffinityMatch(c.expected, result))
 | |
| 		})
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func nodeSelectorRequirementsToMap(source []apiv1.NodeSelectorRequirement, result map[string]apiv1.NodeSelectorRequirement) {
 | |
| 	for _, t := range source {
 | |
| 		result[t.Key] = t
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func nodeAffinityMatch(expected, actual *apiv1.Affinity) bool {
 | |
| 	expectedTerms := expected.NodeAffinity.RequiredDuringSchedulingIgnoredDuringExecution.NodeSelectorTerms
 | |
| 	actualTerms := actual.NodeAffinity.RequiredDuringSchedulingIgnoredDuringExecution.NodeSelectorTerms
 | |
| 	expectedExpressions := make(map[string]apiv1.NodeSelectorRequirement)
 | |
| 	expectedFields := make(map[string]apiv1.NodeSelectorRequirement)
 | |
| 	actualExpressions := make(map[string]apiv1.NodeSelectorRequirement)
 | |
| 	actualFields := make(map[string]apiv1.NodeSelectorRequirement)
 | |
| 	for _, v := range expectedTerms {
 | |
| 		nodeSelectorRequirementsToMap(v.MatchExpressions, expectedExpressions)
 | |
| 		nodeSelectorRequirementsToMap(v.MatchFields, expectedFields)
 | |
| 	}
 | |
| 	for _, v := range actualTerms {
 | |
| 		nodeSelectorRequirementsToMap(v.MatchExpressions, actualExpressions)
 | |
| 		nodeSelectorRequirementsToMap(v.MatchFields, actualFields)
 | |
| 	}
 | |
| 	return reflect.DeepEqual(expectedExpressions, actualExpressions) && reflect.DeepEqual(expectedFields, actualFields)
 | |
| }
 |