From 7530e0523e3c3093e4fe2120b93395bae5ef5571 Mon Sep 17 00:00:00 2001 From: Vladislav Sukhin Date: Wed, 11 Dec 2024 16:26:03 +0300 Subject: [PATCH 01/36] feat: pvc config Signed-off-by: Vladislav Sukhin --- api/v1/testkube.yaml | 33 +++++++++++++++++++ ...test_workflow_independent_step_parallel.go | 1 + .../model_test_workflow_pvc_config.go | 23 +++++++++++++ .../v1/testkube/model_test_workflow_spec.go | 1 + .../model_test_workflow_step_parallel.go | 1 + .../model_test_workflow_template_spec.go | 1 + 6 files changed, 60 insertions(+) create mode 100644 pkg/api/v1/testkube/model_test_workflow_pvc_config.go diff --git a/api/v1/testkube.yaml b/api/v1/testkube.yaml index 50079b8db75..d6852fd61a1 100644 --- a/api/v1/testkube.yaml +++ b/api/v1/testkube.yaml @@ -8797,6 +8797,10 @@ components: $ref: "#/components/schemas/TestWorkflowEvent" execution: $ref: "#/components/schemas/TestWorkflowTagSchema" + pvcs: + type: object + additionalProperties: + $ref: "#/components/schemas/TestWorkflowPVCConfig" TestWorkflowTemplateSpec: type: object @@ -8835,6 +8839,10 @@ components: $ref: "#/components/schemas/TestWorkflowEvent" execution: $ref: "#/components/schemas/TestWorkflowTagSchema" + pvcs: + type: object + additionalProperties: + $ref: "#/components/schemas/TestWorkflowPVCConfig" TestWorkflowStepControl: type: object @@ -10588,6 +10596,31 @@ components: must be defined type: boolean + TestWorkflowPVCConfig: + type: object + properties: + shared: + description: Specify whether the PVC should be shared between test workflow pods + type: boolean + accessModes: + description: 'Access mode for claim storage. More info: https://kubernetes.io/docs/concepts/storage/persistent-volumes/#access-modes' + type: array + items: + type: string + volumeMode: + description: 'Volume mode indicates the consumption of the volume as either a filesystem or block device. + More info: https://kubernetes.io/docs/concepts/storage/persistent-volumes/#volume-mode' + type: string + resources: + description: 'Resources required for PVC' + $ref: "#/components/schemas/TestWorkflowResources" + storageClassName: + description: 'Storage class name specifies the name of a StorageClass. More info: https://kubernetes.io/docs/concepts/storage/storage-classes/' + type: string + selector: + description: Only the volumes whose labels match the selector can be bound to the claim + $ref: "#/components/schemas/LabelSelector" + # # Errors # diff --git a/pkg/api/v1/testkube/model_test_workflow_independent_step_parallel.go b/pkg/api/v1/testkube/model_test_workflow_independent_step_parallel.go index ddb97dc54ce..a68339239ef 100644 --- a/pkg/api/v1/testkube/model_test_workflow_independent_step_parallel.go +++ b/pkg/api/v1/testkube/model_test_workflow_independent_step_parallel.go @@ -53,4 +53,5 @@ type TestWorkflowIndependentStepParallel struct { After []TestWorkflowIndependentStep `json:"after,omitempty"` Events []TestWorkflowEvent `json:"events,omitempty"` Execution *TestWorkflowTagSchema `json:"execution,omitempty"` + Pvcs map[string]TestWorkflowPvcConfig `json:"pvcs,omitempty"` } diff --git a/pkg/api/v1/testkube/model_test_workflow_pvc_config.go b/pkg/api/v1/testkube/model_test_workflow_pvc_config.go new file mode 100644 index 00000000000..d4ac765a78a --- /dev/null +++ b/pkg/api/v1/testkube/model_test_workflow_pvc_config.go @@ -0,0 +1,23 @@ +/* + * Testkube API + * + * Testkube provides a Kubernetes-native framework for test definition, execution and results + * + * API version: 1.0.0 + * Contact: testkube@kubeshop.io + * Generated by: Swagger Codegen (https://github.com/swagger-api/swagger-codegen.git) + */ +package testkube + +type TestWorkflowPvcConfig struct { + // Specify whether the PVC should be shared between test workflow pods + Shared bool `json:"shared,omitempty"` + // Access mode for claim storage. More info: https://kubernetes.io/docs/concepts/storage/persistent-volumes/#access-modes + AccessModes []string `json:"accessModes,omitempty"` + // Volume mode indicates the consumption of the volume as either a filesystem or block device. More info: https://kubernetes.io/docs/concepts/storage/persistent-volumes/#volume-mode + VolumeMode string `json:"volumeMode,omitempty"` + Resources *TestWorkflowResources `json:"resources,omitempty"` + // Storage class name specifies the name of a StorageClass. More info: https://kubernetes.io/docs/concepts/storage/storage-classes/ + StorageClassName string `json:"storageClassName,omitempty"` + Selector *LabelSelector `json:"selector,omitempty"` +} diff --git a/pkg/api/v1/testkube/model_test_workflow_spec.go b/pkg/api/v1/testkube/model_test_workflow_spec.go index 87a6c0e7ee8..f8a69366d4a 100644 --- a/pkg/api/v1/testkube/model_test_workflow_spec.go +++ b/pkg/api/v1/testkube/model_test_workflow_spec.go @@ -23,4 +23,5 @@ type TestWorkflowSpec struct { After []TestWorkflowStep `json:"after,omitempty"` Events []TestWorkflowEvent `json:"events,omitempty"` Execution *TestWorkflowTagSchema `json:"execution,omitempty"` + Pvcs map[string]TestWorkflowPvcConfig `json:"pvcs,omitempty"` } diff --git a/pkg/api/v1/testkube/model_test_workflow_step_parallel.go b/pkg/api/v1/testkube/model_test_workflow_step_parallel.go index 7c8a9ffcca7..d11b2e1d85a 100644 --- a/pkg/api/v1/testkube/model_test_workflow_step_parallel.go +++ b/pkg/api/v1/testkube/model_test_workflow_step_parallel.go @@ -55,4 +55,5 @@ type TestWorkflowStepParallel struct { After []TestWorkflowStep `json:"after,omitempty"` Events []TestWorkflowEvent `json:"events,omitempty"` Execution *TestWorkflowTagSchema `json:"execution,omitempty"` + Pvcs map[string]TestWorkflowPvcConfig `json:"pvcs,omitempty"` } diff --git a/pkg/api/v1/testkube/model_test_workflow_template_spec.go b/pkg/api/v1/testkube/model_test_workflow_template_spec.go index 17d16455a72..c1b9c34c401 100644 --- a/pkg/api/v1/testkube/model_test_workflow_template_spec.go +++ b/pkg/api/v1/testkube/model_test_workflow_template_spec.go @@ -22,4 +22,5 @@ type TestWorkflowTemplateSpec struct { After []TestWorkflowIndependentStep `json:"after,omitempty"` Events []TestWorkflowEvent `json:"events,omitempty"` Execution *TestWorkflowTagSchema `json:"execution,omitempty"` + Pvcs map[string]TestWorkflowPvcConfig `json:"pvcs,omitempty"` } From 6d67658ea0bf118a6c4a6d5023f43724860f93ef Mon Sep 17 00:00:00 2001 From: Vladislav Sukhin Date: Wed, 11 Dec 2024 17:17:37 +0300 Subject: [PATCH 02/36] fix: rename var Signed-off-by: Vladislav Sukhin --- api/v1/testkube.yaml | 10 +++++----- pkg/api/v1/testkube/model_test_workflow_pvc_config.go | 2 +- 2 files changed, 6 insertions(+), 6 deletions(-) diff --git a/api/v1/testkube.yaml b/api/v1/testkube.yaml index d6852fd61a1..0d98bd8a8df 100644 --- a/api/v1/testkube.yaml +++ b/api/v1/testkube.yaml @@ -8800,7 +8800,7 @@ components: pvcs: type: object additionalProperties: - $ref: "#/components/schemas/TestWorkflowPVCConfig" + $ref: "#/components/schemas/TestWorkflowPvcConfig" TestWorkflowTemplateSpec: type: object @@ -8842,7 +8842,7 @@ components: pvcs: type: object additionalProperties: - $ref: "#/components/schemas/TestWorkflowPVCConfig" + $ref: "#/components/schemas/TestWorkflowPvcConfig" TestWorkflowStepControl: type: object @@ -10596,11 +10596,11 @@ components: must be defined type: boolean - TestWorkflowPVCConfig: + TestWorkflowPvcConfig: type: object properties: shared: - description: Specify whether the PVC should be shared between test workflow pods + description: Specify whether the pvc should be shared between test workflow pods type: boolean accessModes: description: 'Access mode for claim storage. More info: https://kubernetes.io/docs/concepts/storage/persistent-volumes/#access-modes' @@ -10612,7 +10612,7 @@ components: More info: https://kubernetes.io/docs/concepts/storage/persistent-volumes/#volume-mode' type: string resources: - description: 'Resources required for PVC' + description: 'Resources required for pvc' $ref: "#/components/schemas/TestWorkflowResources" storageClassName: description: 'Storage class name specifies the name of a StorageClass. More info: https://kubernetes.io/docs/concepts/storage/storage-classes/' diff --git a/pkg/api/v1/testkube/model_test_workflow_pvc_config.go b/pkg/api/v1/testkube/model_test_workflow_pvc_config.go index d4ac765a78a..8e7de8dfbf9 100644 --- a/pkg/api/v1/testkube/model_test_workflow_pvc_config.go +++ b/pkg/api/v1/testkube/model_test_workflow_pvc_config.go @@ -10,7 +10,7 @@ package testkube type TestWorkflowPvcConfig struct { - // Specify whether the PVC should be shared between test workflow pods + // Specify whether the pvc should be shared between test workflow pods Shared bool `json:"shared,omitempty"` // Access mode for claim storage. More info: https://kubernetes.io/docs/concepts/storage/persistent-volumes/#access-modes AccessModes []string `json:"accessModes,omitempty"` From 76ea394881b26b007ed9a99874b839bf1dde5eab Mon Sep 17 00:00:00 2001 From: Vladislav Sukhin Date: Wed, 11 Dec 2024 18:08:55 +0300 Subject: [PATCH 03/36] fix: dep update Signed-off-by: Vladislav Sukhin --- go.mod | 2 +- go.sum | 4 ++-- 2 files changed, 3 insertions(+), 3 deletions(-) diff --git a/go.mod b/go.mod index f2e215d06b9..bdd0facf6c8 100644 --- a/go.mod +++ b/go.mod @@ -42,7 +42,7 @@ require ( github.com/keygen-sh/jsonapi-go v1.2.1 github.com/keygen-sh/keygen-go/v3 v3.2.0 github.com/kubepug/kubepug v1.7.1 - github.com/kubeshop/testkube-operator v1.17.55-0.20241118133003-70462ac10f4a + github.com/kubeshop/testkube-operator v1.17.55-0.20241211150637-26f07cccd389 github.com/minio/minio-go/v7 v7.0.66 github.com/montanaflynn/stats v0.7.1 github.com/moogar0880/problems v0.1.1 diff --git a/go.sum b/go.sum index 56f08fbd156..c4081dfec7b 100644 --- a/go.sum +++ b/go.sum @@ -336,8 +336,8 @@ github.com/kr/text v0.2.0 h1:5Nx0Ya0ZqY2ygV366QzturHI13Jq95ApcVaJBhpS+AY= github.com/kr/text v0.2.0/go.mod h1:eLer722TekiGuMkidMxC/pM04lWEeraHUUmBw8l2grE= github.com/kubepug/kubepug v1.7.1 h1:LKhfSxS8Y5mXs50v+3Lpyec+cogErDLcV7CMUuiaisw= github.com/kubepug/kubepug v1.7.1/go.mod h1:lv+HxD0oTFL7ZWjj0u6HKhMbbTIId3eG7aWIW0gyF8g= -github.com/kubeshop/testkube-operator v1.17.55-0.20241118133003-70462ac10f4a h1:xget2cwwqOL+K2Op9FPbMgfzj9lSVJAzZ9p48yxuFrE= -github.com/kubeshop/testkube-operator v1.17.55-0.20241118133003-70462ac10f4a/go.mod h1:P47tw1nKQFufdsZndyq2HG2MSa0zK/lU0XpRfZtEmIk= +github.com/kubeshop/testkube-operator v1.17.55-0.20241211150637-26f07cccd389 h1:dP8c/kJmIFzy6F07qG+Wmo3jq8dfEvjYcAItJHDyjJQ= +github.com/kubeshop/testkube-operator v1.17.55-0.20241211150637-26f07cccd389/go.mod h1:P47tw1nKQFufdsZndyq2HG2MSa0zK/lU0XpRfZtEmIk= github.com/leodido/go-urn v1.4.0 h1:WT9HwE9SGECu3lg4d/dIA+jxlljEa1/ffXKmRjqdmIQ= github.com/leodido/go-urn v1.4.0/go.mod h1:bvxc+MVxLKB4z00jd1z+Dvzr47oO32F/QSNjSBOlFxI= github.com/lithammer/fuzzysearch v1.1.8 h1:/HIuJnjHuXS8bKaiTMeeDlW2/AyIWk2brx1V8LFgLN4= From 6b99bc4cac74bef6c56dc9182fc7cfbfa9f4e6a0 Mon Sep 17 00:00:00 2001 From: Vladislav Sukhin Date: Wed, 11 Dec 2024 18:38:40 +0300 Subject: [PATCH 04/36] fix: map pvc to api Signed-off-by: Vladislav Sukhin --- pkg/mapper/testworkflows/kube_openapi.go | 15 +++++++++++++++ 1 file changed, 15 insertions(+) diff --git a/pkg/mapper/testworkflows/kube_openapi.go b/pkg/mapper/testworkflows/kube_openapi.go index 9076451da8f..ac221087d21 100644 --- a/pkg/mapper/testworkflows/kube_openapi.go +++ b/pkg/mapper/testworkflows/kube_openapi.go @@ -915,6 +915,7 @@ func MapStepParallelKubeToAPI(v testworkflowsv1.StepParallel) testkube.TestWorkf Run: common.MapPtr(v.Run, MapStepRunKubeToAPI), Execute: common.MapPtr(v.Execute, MapStepExecuteKubeToAPI), Artifacts: common.MapPtr(v.Artifacts, MapStepArtifactsKubeToAPI), + Pvcs: common.MapMap(v.Pvcs, MapPvcConfigKubeToAPI), } } @@ -948,6 +949,7 @@ func MapIndependentStepParallelKubeToAPI(v testworkflowsv1.IndependentStepParall Run: common.MapPtr(v.Run, MapStepRunKubeToAPI), Execute: common.MapPtr(v.Execute, MapStepExecuteKubeToAPI), Artifacts: common.MapPtr(v.Artifacts, MapStepArtifactsKubeToAPI), + Pvcs: common.MapMap(v.Pvcs, MapPvcConfigKubeToAPI), } } @@ -1134,6 +1136,7 @@ func MapSpecKubeToAPI(v testworkflowsv1.TestWorkflowSpec) testkube.TestWorkflowS After: common.MapSlice(v.After, MapStepKubeToAPI), Events: common.MapSlice(v.Events, MapEventKubeToAPI), Execution: common.MapPtr(v.Execution, MapTestWorkflowTagSchemaKubeToAPI), + Pvcs: common.MapMap(v.Pvcs, MapPvcConfigKubeToAPI), } } @@ -1150,6 +1153,7 @@ func MapTemplateSpecKubeToAPI(v testworkflowsv1.TestWorkflowTemplateSpec) testku After: common.MapSlice(v.After, MapIndependentStepKubeToAPI), Events: common.MapSlice(v.Events, MapEventKubeToAPI), Execution: common.MapPtr(v.Execution, MapTestWorkflowTagSchemaKubeToAPI), + Pvcs: common.MapMap(v.Pvcs, MapPvcConfigKubeToAPI), } } @@ -1206,3 +1210,14 @@ func MapTestWorkflowTagSchemaKubeToAPI(v testworkflowsv1.TestWorkflowTagSchema) Tags: v.Tags, } } + +func MapPvcConfigKubeToAPI(v testworkflowsv1.TestWorkflowPvcConfig) testkube.TestWorkflowPvcConfig { + return testkube.TestWorkflowPvcConfig{ + Shared: v.Shared, + AccessModes: v.AccessModes, + VolumeMode: v.VolumeMode, + Resources: common.MapPtr(v.Resources, MapResourcesKubeToAPI), + StorageClassName: v.StorageClassName, + Selector: common.MapPtr(v.Selector, MapSelectorToAPI), + } +} From a7c36c054180be8c6e89b9523f9c053bb12abb80 Mon Sep 17 00:00:00 2001 From: Vladislav Sukhin Date: Wed, 11 Dec 2024 18:57:04 +0300 Subject: [PATCH 05/36] fix: map to kube Signed-off-by: Vladislav Sukhin --- pkg/mapper/testworkflows/openapi_kube.go | 15 +++++++++++++++ 1 file changed, 15 insertions(+) diff --git a/pkg/mapper/testworkflows/openapi_kube.go b/pkg/mapper/testworkflows/openapi_kube.go index 86b9bf7edee..ca78b436e5f 100644 --- a/pkg/mapper/testworkflows/openapi_kube.go +++ b/pkg/mapper/testworkflows/openapi_kube.go @@ -951,6 +951,7 @@ func MapStepParallelAPIToKube(v testkube.TestWorkflowStepParallel) testworkflows Setup: common.MapSlice(v.Setup, MapStepAPIToKube), Steps: common.MapSlice(v.Steps, MapStepAPIToKube), After: common.MapSlice(v.After, MapStepAPIToKube), + Pvcs: common.MapMap(v.Pvcs, MapPvcConfigAPIToKube), }, StepControl: testworkflowsv1.StepControl{ Paused: v.Paused, @@ -995,6 +996,7 @@ func MapIndependentStepParallelAPIToKube(v testkube.TestWorkflowIndependentStepP Setup: common.MapSlice(v.Setup, MapIndependentStepAPIToKube), Steps: common.MapSlice(v.Steps, MapIndependentStepAPIToKube), After: common.MapSlice(v.After, MapIndependentStepAPIToKube), + Pvcs: common.MapMap(v.Pvcs, MapPvcConfigAPIToKube), }, StepControl: testworkflowsv1.StepControl{ Paused: v.Paused, @@ -1234,6 +1236,7 @@ func MapSpecAPIToKube(v testkube.TestWorkflowSpec) testworkflowsv1.TestWorkflowS Setup: common.MapSlice(v.Setup, MapStepAPIToKube), Steps: common.MapSlice(v.Steps, MapStepAPIToKube), After: common.MapSlice(v.After, MapStepAPIToKube), + Pvcs: common.MapMap(v.Pvcs, MapPvcConfigAPIToKube), } } @@ -1253,6 +1256,7 @@ func MapTemplateSpecAPIToKube(v testkube.TestWorkflowTemplateSpec) testworkflows Setup: common.MapSlice(v.Setup, MapIndependentStepAPIToKube), Steps: common.MapSlice(v.Steps, MapIndependentStepAPIToKube), After: common.MapSlice(v.After, MapIndependentStepAPIToKube), + Pvcs: common.MapMap(v.Pvcs, MapPvcConfigAPIToKube), } } @@ -1485,3 +1489,14 @@ func MapTestWorkflowTagSchemaAPIToKube(v testkube.TestWorkflowTagSchema) testwor Tags: v.Tags, } } + +func MapPvcConfigAPIToKube(v testkube.TestWorkflowPvcConfig) testworkflowsv1.TestWorkflowPvcConfig { + return testworkflowsv1.TestWorkflowPvcConfig{ + Shared: v.Shared, + AccessModes: v.AccessModes, + VolumeMode: v.VolumeMode, + Resources: common.MapPtr(v.Resources, MapResourcesAPIToKube), + StorageClassName: v.StorageClassName, + Selector: common.MapPtr(v.Selector, MapLabelSelectorAPIToKube), + } +} From 8e8d073b1f646b9c21248260e6625b26cf9631ba Mon Sep 17 00:00:00 2001 From: Vladislav Sukhin Date: Wed, 11 Dec 2024 21:38:41 +0300 Subject: [PATCH 06/36] fix: add pvc to control plane Signed-off-by: Vladislav Sukhin --- .../testworkflowprocessor/bundle.go | 11 +++++++ .../testworkflowprocessor/intermediate.go | 16 +++++++-- .../testworkflowprocessor/processor.go | 11 +++++++ .../testworkflowresolver/apply.go | 1 + .../testworkflowresolver/merge.go | 33 +++++++++++++++++++ 5 files changed, 70 insertions(+), 2 deletions(-) diff --git a/pkg/testworkflows/testworkflowprocessor/bundle.go b/pkg/testworkflows/testworkflowprocessor/bundle.go index 84bfe8a6947..fb49754391e 100644 --- a/pkg/testworkflows/testworkflowprocessor/bundle.go +++ b/pkg/testworkflows/testworkflowprocessor/bundle.go @@ -27,6 +27,7 @@ type BundleOptions struct { type Bundle struct { Secrets []corev1.Secret ConfigMaps []corev1.ConfigMap + Pvcs []corev1.PersistentVolumeClaim Job batchv1.Job Signature []stage.Signature FullSignature []stage.Signature @@ -50,6 +51,9 @@ func (b *Bundle) SetGroupId(groupId string) { for i := range b.Secrets { AnnotateGroupId(&b.Secrets[i], groupId) } + for i := range b.Pvcs { + AnnotateGroupId(&b.Pvcs[i], groupId) + } } func (b *Bundle) Deploy(ctx context.Context, clientSet kubernetes.Interface, namespace string) (err error) { @@ -68,6 +72,13 @@ func (b *Bundle) Deploy(ctx context.Context, clientSet kubernetes.Interface, nam return errors.Wrap(err, "failed to deploy config maps") } } + for _, item := range b.Pvcs { + _, err = clientSet.CoreV1().PersistentVolumeClaims(namespace).Create(ctx, &item, metav1.CreateOptions{}) + if err != nil { + return errors.Wrap(err, "failed to deploy pvcs") + } + } + _, err = clientSet.BatchV1().Jobs(namespace).Create(ctx, &b.Job, metav1.CreateOptions{}) return errors.Wrap(err, "failed to deploy job") } diff --git a/pkg/testworkflows/testworkflowprocessor/intermediate.go b/pkg/testworkflows/testworkflowprocessor/intermediate.go index b92b268338a..7b43adeb040 100644 --- a/pkg/testworkflows/testworkflowprocessor/intermediate.go +++ b/pkg/testworkflows/testworkflowprocessor/intermediate.go @@ -21,6 +21,7 @@ type Intermediate interface { ConfigMaps() []corev1.ConfigMap Secrets() []corev1.Secret Volumes() []corev1.Volume + Pvcs() []corev1.PersistentVolumeClaim AppendJobConfig(cfg *testworkflowsv1.JobConfig) Intermediate AppendPodConfig(cfg *testworkflowsv1.PodConfig) Intermediate @@ -28,6 +29,7 @@ type Intermediate interface { AddConfigMap(configMap corev1.ConfigMap) Intermediate AddSecret(secret corev1.Secret) Intermediate AddVolume(volume corev1.Volume) Intermediate + AddPvc(pvc corev1.PersistentVolumeClaim) Intermediate AddEmptyDirVolume(source *corev1.EmptyDirVolumeSource, mountPath string) corev1.VolumeMount @@ -47,8 +49,9 @@ type intermediate struct { Job testworkflowsv1.JobConfig `expr:"include"` // Actual Kubernetes resources to use - Secs []corev1.Secret `expr:"force"` - Cfgs []corev1.ConfigMap `expr:"force"` + Secs []corev1.Secret `expr:"force"` + Cfgs []corev1.ConfigMap `expr:"force"` + Ps []corev1.PersistentVolumeClaim `expr:"force"` // Storing files Files ConfigMapFiles `expr:"include"` @@ -87,6 +90,10 @@ func (s *intermediate) Volumes() []corev1.Volume { return append(s.Pod.Volumes, s.Files.Volumes()...) } +func (s *intermediate) Pvcs() []corev1.PersistentVolumeClaim { + return s.Ps +} + func (s *intermediate) AppendJobConfig(cfg *testworkflowsv1.JobConfig) Intermediate { s.Job = *testworkflowresolver.MergeJobConfig(&s.Job, cfg) return s @@ -102,6 +109,11 @@ func (s *intermediate) AddVolume(volume corev1.Volume) Intermediate { return s } +func (s *intermediate) AddPvc(pvc corev1.PersistentVolumeClaim) Intermediate { + s.Ps = append(s.Ps, pvc) + return s +} + func (s *intermediate) AddConfigMap(configMap corev1.ConfigMap) Intermediate { s.Cfgs = append(s.Cfgs, configMap) return s diff --git a/pkg/testworkflows/testworkflowprocessor/processor.go b/pkg/testworkflows/testworkflowprocessor/processor.go index 1a0d460a69c..af5d881f9bb 100644 --- a/pkg/testworkflows/testworkflowprocessor/processor.go +++ b/pkg/testworkflows/testworkflowprocessor/processor.go @@ -162,6 +162,16 @@ func (p *processor) Bundle(ctx context.Context, workflow *testworkflowsv1.TestWo } } + // Finalize Pvcs + pvcs := layer.Pvcs() + for i := range pvcs { + AnnotateControlledBy(&pvcs[i], options.Config.Resource.RootId, options.Config.Resource.Id) + err = expressions.FinalizeForce(&pvcs[i], machines...) + if err != nil { + return nil, errors.Wrap(err, "finalizing Pvc") + } + } + // Finalize Secrets secrets := append(layer.Secrets(), options.Secrets...) for i := range secrets { @@ -420,6 +430,7 @@ func (p *processor) Bundle(ctx context.Context, workflow *testworkflowsv1.TestWo bundle = &Bundle{ ConfigMaps: configMaps, Secrets: secrets, + Pvcs: pvcs, Job: jobSpec, Signature: sig, FullSignature: fullSig, diff --git a/pkg/testworkflows/testworkflowresolver/apply.go b/pkg/testworkflows/testworkflowresolver/apply.go index a3afdc9ea84..03c8b67da43 100644 --- a/pkg/testworkflows/testworkflowresolver/apply.go +++ b/pkg/testworkflows/testworkflowresolver/apply.go @@ -69,6 +69,7 @@ func injectTemplateToSpec(spec *testworkflowsv1.TestWorkflowSpec, template testw spec.Services = MergeMap(common.MapMap(template.Spec.Services, ConvertIndependentServiceToService), spec.Services) spec.Container = MergeContainerConfig(template.Spec.Container, spec.Container) spec.System = MergeSystem(template.Spec.System, spec.System) + spec.Pvcs = MergeMap(template.Spec.Pvcs, spec.Pvcs) // Include the steps from the template setup := common.MapSlice(template.Spec.Setup, ConvertIndependentStepToStep) diff --git a/pkg/testworkflows/testworkflowresolver/merge.go b/pkg/testworkflows/testworkflowresolver/merge.go index 1d503f1b84c..51cdd08d0dd 100644 --- a/pkg/testworkflows/testworkflowresolver/merge.go +++ b/pkg/testworkflows/testworkflowresolver/merge.go @@ -12,6 +12,7 @@ import ( "maps" corev1 "k8s.io/api/core/v1" + metav1 "k8s.io/apimachinery/pkg/apis/meta/v1" testworkflowsv1 "github.com/kubeshop/testkube-operator/api/testworkflows/v1" "github.com/kubeshop/testkube/internal/common" @@ -414,3 +415,35 @@ func MergeTags(dst, src map[string]string) map[string]string { return dst } + +func MergePvcConfig(dst, include *testworkflowsv1.TestWorkflowPvcConfig) *testworkflowsv1.TestWorkflowPvcConfig { + if dst == nil { + return include + } else if include == nil { + return dst + } + if include.Shared { + dst.Shared = include.Shared + } + dst.AccessModes = append(dst.AccessModes, include.AccessModes...) + if include.VolumeMode != "" { + dst.VolumeMode = include.VolumeMode + } + dst.Resources = MergeResources(dst.Resources, include.Resources) + if include.StorageClassName != "" { + dst.StorageClassName = include.StorageClassName + } + dst.Selector = MergeLabelSelector(dst.Selector, include.Selector) + return dst +} + +func MergeLabelSelector(dst, include *metav1.LabelSelector) *metav1.LabelSelector { + if dst == nil { + return include + } else if include == nil { + return dst + } + dst.MatchLabels = common.MergeMaps(dst.MatchLabels, include.MatchLabels) + dst.MatchExpressions = append(dst.MatchExpressions, include.MatchExpressions...) + return dst +} From 3bbba2548e6386ad541bc56cb8bfe03cedd255f4 Mon Sep 17 00:00:00 2001 From: Vladislav Sukhin Date: Thu, 12 Dec 2024 13:22:03 +0300 Subject: [PATCH 07/36] cleanup pvcs Signed-off-by: Vladislav Sukhin --- .../executionworker/controller/cleanup.go | 14 ++++++++++ .../mock_intermediate.go | 28 +++++++++++++++++++ 2 files changed, 42 insertions(+) diff --git a/pkg/testworkflows/executionworker/controller/cleanup.go b/pkg/testworkflows/executionworker/controller/cleanup.go index 1a6faa7d8b6..a70d7194ee9 100644 --- a/pkg/testworkflows/executionworker/controller/cleanup.go +++ b/pkg/testworkflows/executionworker/controller/cleanup.go @@ -57,6 +57,17 @@ func cleanupJobs(labelName string) func(ctx context.Context, clientSet kubernete } } +func cleanupPvcs(labelName string) func(ctx context.Context, clientSet kubernetes.Interface, namespace, id string) error { + return func(ctx context.Context, clientSet kubernetes.Interface, namespace, id string) error { + return clientSet.CoreV1().PersistentVolumeClaims(namespace).DeleteCollection(ctx, metav1.DeleteOptions{ + GracePeriodSeconds: common.Ptr(int64(0)), + PropagationPolicy: common.Ptr(metav1.DeletePropagationBackground), + }, metav1.ListOptions{ + LabelSelector: fmt.Sprintf("%s=%s", labelName, id), + }) + } +} + func Cleanup(ctx context.Context, clientSet kubernetes.Interface, namespace, id string) error { var errs []error var errsMu sync.Mutex @@ -70,6 +81,8 @@ func Cleanup(ctx context.Context, clientSet kubernetes.Interface, namespace, id cleanupConfigMaps(constants.ResourceIdLabelName), cleanupSecrets(constants.RootResourceIdLabelName), cleanupSecrets(constants.ResourceIdLabelName), + cleanupPvcs(constants.RootResourceIdLabelName), + cleanupPvcs(constants.ResourceIdLabelName), } wg.Add(len(ops)) for _, op := range ops { @@ -96,6 +109,7 @@ func CleanupGroup(ctx context.Context, clientSet kubernetes.Interface, namespace cleanupPods(constants.GroupIdLabelName), cleanupConfigMaps(constants.GroupIdLabelName), cleanupSecrets(constants.GroupIdLabelName), + cleanupPvcs(constants.GroupIdLabelName), } wg.Add(len(ops)) for _, op := range ops { diff --git a/pkg/testworkflows/testworkflowprocessor/mock_intermediate.go b/pkg/testworkflows/testworkflowprocessor/mock_intermediate.go index 119db5cbdbf..5d42526cf59 100644 --- a/pkg/testworkflows/testworkflowprocessor/mock_intermediate.go +++ b/pkg/testworkflows/testworkflowprocessor/mock_intermediate.go @@ -79,6 +79,20 @@ func (mr *MockIntermediateMockRecorder) AddEmptyDirVolume(arg0, arg1 interface{} return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "AddEmptyDirVolume", reflect.TypeOf((*MockIntermediate)(nil).AddEmptyDirVolume), arg0, arg1) } +// AddPvc mocks base method. +func (m *MockIntermediate) AddPvc(arg0 v10.PersistentVolumeClaim) Intermediate { + m.ctrl.T.Helper() + ret := m.ctrl.Call(m, "AddPvc", arg0) + ret0, _ := ret[0].(Intermediate) + return ret0 +} + +// AddPvc indicates an expected call of AddPvc. +func (mr *MockIntermediateMockRecorder) AddPvc(arg0 interface{}) *gomock.Call { + mr.mock.ctrl.T.Helper() + return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "AddPvc", reflect.TypeOf((*MockIntermediate)(nil).AddPvc), arg0) +} + // AddSecret mocks base method. func (m *MockIntermediate) AddSecret(arg0 v10.Secret) Intermediate { m.ctrl.T.Helper() @@ -220,6 +234,20 @@ func (mr *MockIntermediateMockRecorder) PodConfig() *gomock.Call { return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "PodConfig", reflect.TypeOf((*MockIntermediate)(nil).PodConfig)) } +// Pvcs mocks base method. +func (m *MockIntermediate) Pvcs() []v10.PersistentVolumeClaim { + m.ctrl.T.Helper() + ret := m.ctrl.Call(m, "Pvcs") + ret0, _ := ret[0].([]v10.PersistentVolumeClaim) + return ret0 +} + +// Pvcs indicates an expected call of Pvcs. +func (mr *MockIntermediateMockRecorder) Pvcs() *gomock.Call { + mr.mock.ctrl.T.Helper() + return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "Pvcs", reflect.TypeOf((*MockIntermediate)(nil).Pvcs)) +} + // Secrets mocks base method. func (m *MockIntermediate) Secrets() []v10.Secret { m.ctrl.T.Helper() From d9fcff8408e5eef7cc541d66543d1d2fb0b904da Mon Sep 17 00:00:00 2001 From: Vladislav Sukhin Date: Thu, 12 Dec 2024 14:49:44 +0300 Subject: [PATCH 08/36] fix: merge pvcs map Signed-off-by: Vladislav Sukhin --- .../testworkflowresolver/apply.go | 1 + .../testworkflowresolver/merge.go | 33 ------------------- 2 files changed, 1 insertion(+), 33 deletions(-) diff --git a/pkg/testworkflows/testworkflowresolver/apply.go b/pkg/testworkflows/testworkflowresolver/apply.go index 03c8b67da43..0808a09fda6 100644 --- a/pkg/testworkflows/testworkflowresolver/apply.go +++ b/pkg/testworkflows/testworkflowresolver/apply.go @@ -63,6 +63,7 @@ func injectTemplateToSpec(spec *testworkflowsv1.TestWorkflowSpec, template testw spec.Job = MergeJobConfig(template.Spec.Job, spec.Job) spec.Events = append(template.Spec.Events, spec.Events...) spec.Execution = MergeExecution(template.Spec.Execution, spec.Execution) + spec.Pvcs = MergeMap(template.Spec.Pvcs, spec.Pvcs) // Apply basic configuration spec.Content = MergeContent(template.Spec.Content, spec.Content) diff --git a/pkg/testworkflows/testworkflowresolver/merge.go b/pkg/testworkflows/testworkflowresolver/merge.go index 51cdd08d0dd..1d503f1b84c 100644 --- a/pkg/testworkflows/testworkflowresolver/merge.go +++ b/pkg/testworkflows/testworkflowresolver/merge.go @@ -12,7 +12,6 @@ import ( "maps" corev1 "k8s.io/api/core/v1" - metav1 "k8s.io/apimachinery/pkg/apis/meta/v1" testworkflowsv1 "github.com/kubeshop/testkube-operator/api/testworkflows/v1" "github.com/kubeshop/testkube/internal/common" @@ -415,35 +414,3 @@ func MergeTags(dst, src map[string]string) map[string]string { return dst } - -func MergePvcConfig(dst, include *testworkflowsv1.TestWorkflowPvcConfig) *testworkflowsv1.TestWorkflowPvcConfig { - if dst == nil { - return include - } else if include == nil { - return dst - } - if include.Shared { - dst.Shared = include.Shared - } - dst.AccessModes = append(dst.AccessModes, include.AccessModes...) - if include.VolumeMode != "" { - dst.VolumeMode = include.VolumeMode - } - dst.Resources = MergeResources(dst.Resources, include.Resources) - if include.StorageClassName != "" { - dst.StorageClassName = include.StorageClassName - } - dst.Selector = MergeLabelSelector(dst.Selector, include.Selector) - return dst -} - -func MergeLabelSelector(dst, include *metav1.LabelSelector) *metav1.LabelSelector { - if dst == nil { - return include - } else if include == nil { - return dst - } - dst.MatchLabels = common.MergeMaps(dst.MatchLabels, include.MatchLabels) - dst.MatchExpressions = append(dst.MatchExpressions, include.MatchExpressions...) - return dst -} From bd2cf57b1718bf18950bb48973d2d20b65416897 Mon Sep 17 00:00:00 2001 From: Vladislav Sukhin Date: Thu, 12 Dec 2024 15:13:16 +0300 Subject: [PATCH 09/36] fix: add pvcs to service spec Signed-off-by: Vladislav Sukhin --- api/v1/testkube.yaml | 7 ++++--- .../model_test_workflow_independent_service_spec.go | 3 ++- pkg/api/v1/testkube/model_test_workflow_pvc_config.go | 2 -- pkg/api/v1/testkube/model_test_workflow_service_spec.go | 3 ++- 4 files changed, 8 insertions(+), 7 deletions(-) diff --git a/api/v1/testkube.yaml b/api/v1/testkube.yaml index 0d98bd8a8df..3855d42115f 100644 --- a/api/v1/testkube.yaml +++ b/api/v1/testkube.yaml @@ -8678,6 +8678,10 @@ components: type: string readinessProbe: $ref: "#/components/schemas/Probe" + pvcs: + type: object + additionalProperties: + $ref: "#/components/schemas/TestWorkflowPvcConfig" TestWorkflowServiceSpec: type: object @@ -10599,9 +10603,6 @@ components: TestWorkflowPvcConfig: type: object properties: - shared: - description: Specify whether the pvc should be shared between test workflow pods - type: boolean accessModes: description: 'Access mode for claim storage. More info: https://kubernetes.io/docs/concepts/storage/persistent-volumes/#access-modes' type: array diff --git a/pkg/api/v1/testkube/model_test_workflow_independent_service_spec.go b/pkg/api/v1/testkube/model_test_workflow_independent_service_spec.go index 192eeadb608..57ea947f73d 100644 --- a/pkg/api/v1/testkube/model_test_workflow_independent_service_spec.go +++ b/pkg/api/v1/testkube/model_test_workflow_independent_service_spec.go @@ -41,5 +41,6 @@ type TestWorkflowIndependentServiceSpec struct { // matrix of parameters to spawn instances Matrix map[string]interface{} `json:"matrix,omitempty"` // parameters that should be distributed across sharded instances - Shards map[string]interface{} `json:"shards,omitempty"` + Shards map[string]interface{} `json:"shards,omitempty"` + Pvcs map[string]TestWorkflowPvcConfig `json:"pvcs,omitempty"` } diff --git a/pkg/api/v1/testkube/model_test_workflow_pvc_config.go b/pkg/api/v1/testkube/model_test_workflow_pvc_config.go index 8e7de8dfbf9..289c61df362 100644 --- a/pkg/api/v1/testkube/model_test_workflow_pvc_config.go +++ b/pkg/api/v1/testkube/model_test_workflow_pvc_config.go @@ -10,8 +10,6 @@ package testkube type TestWorkflowPvcConfig struct { - // Specify whether the pvc should be shared between test workflow pods - Shared bool `json:"shared,omitempty"` // Access mode for claim storage. More info: https://kubernetes.io/docs/concepts/storage/persistent-volumes/#access-modes AccessModes []string `json:"accessModes,omitempty"` // Volume mode indicates the consumption of the volume as either a filesystem or block device. More info: https://kubernetes.io/docs/concepts/storage/persistent-volumes/#volume-mode diff --git a/pkg/api/v1/testkube/model_test_workflow_service_spec.go b/pkg/api/v1/testkube/model_test_workflow_service_spec.go index ad915dd162e..4df57b081e1 100644 --- a/pkg/api/v1/testkube/model_test_workflow_service_spec.go +++ b/pkg/api/v1/testkube/model_test_workflow_service_spec.go @@ -42,5 +42,6 @@ type TestWorkflowServiceSpec struct { // matrix of parameters to spawn instances Matrix map[string]interface{} `json:"matrix,omitempty"` // parameters that should be distributed across sharded instances - Shards map[string]interface{} `json:"shards,omitempty"` + Shards map[string]interface{} `json:"shards,omitempty"` + Pvcs map[string]TestWorkflowPvcConfig `json:"pvcs,omitempty"` } From 0d02d16f04eb248905ad3029908cb61d7fb0133c Mon Sep 17 00:00:00 2001 From: Vladislav Sukhin Date: Thu, 12 Dec 2024 15:16:48 +0300 Subject: [PATCH 10/36] fix: remove duplication Signed-off-by: Vladislav Sukhin --- pkg/testworkflows/testworkflowresolver/apply.go | 1 - 1 file changed, 1 deletion(-) diff --git a/pkg/testworkflows/testworkflowresolver/apply.go b/pkg/testworkflows/testworkflowresolver/apply.go index 0808a09fda6..23b2845305e 100644 --- a/pkg/testworkflows/testworkflowresolver/apply.go +++ b/pkg/testworkflows/testworkflowresolver/apply.go @@ -70,7 +70,6 @@ func injectTemplateToSpec(spec *testworkflowsv1.TestWorkflowSpec, template testw spec.Services = MergeMap(common.MapMap(template.Spec.Services, ConvertIndependentServiceToService), spec.Services) spec.Container = MergeContainerConfig(template.Spec.Container, spec.Container) spec.System = MergeSystem(template.Spec.System, spec.System) - spec.Pvcs = MergeMap(template.Spec.Pvcs, spec.Pvcs) // Include the steps from the template setup := common.MapSlice(template.Spec.Setup, ConvertIndependentStepToStep) From b85e2743e76c47baa6b2a42629cbc900d58912bf Mon Sep 17 00:00:00 2001 From: Vladislav Sukhin Date: Thu, 12 Dec 2024 16:13:27 +0300 Subject: [PATCH 11/36] fix: dep update Signed-off-by: Vladislav Sukhin --- go.mod | 2 +- go.sum | 4 ++-- 2 files changed, 3 insertions(+), 3 deletions(-) diff --git a/go.mod b/go.mod index bdd0facf6c8..c97ba8eab84 100644 --- a/go.mod +++ b/go.mod @@ -42,7 +42,7 @@ require ( github.com/keygen-sh/jsonapi-go v1.2.1 github.com/keygen-sh/keygen-go/v3 v3.2.0 github.com/kubepug/kubepug v1.7.1 - github.com/kubeshop/testkube-operator v1.17.55-0.20241211150637-26f07cccd389 + github.com/kubeshop/testkube-operator v1.17.55-0.20241212130344-02ce60b6d6b1 github.com/minio/minio-go/v7 v7.0.66 github.com/montanaflynn/stats v0.7.1 github.com/moogar0880/problems v0.1.1 diff --git a/go.sum b/go.sum index c4081dfec7b..8161814abbc 100644 --- a/go.sum +++ b/go.sum @@ -336,8 +336,8 @@ github.com/kr/text v0.2.0 h1:5Nx0Ya0ZqY2ygV366QzturHI13Jq95ApcVaJBhpS+AY= github.com/kr/text v0.2.0/go.mod h1:eLer722TekiGuMkidMxC/pM04lWEeraHUUmBw8l2grE= github.com/kubepug/kubepug v1.7.1 h1:LKhfSxS8Y5mXs50v+3Lpyec+cogErDLcV7CMUuiaisw= github.com/kubepug/kubepug v1.7.1/go.mod h1:lv+HxD0oTFL7ZWjj0u6HKhMbbTIId3eG7aWIW0gyF8g= -github.com/kubeshop/testkube-operator v1.17.55-0.20241211150637-26f07cccd389 h1:dP8c/kJmIFzy6F07qG+Wmo3jq8dfEvjYcAItJHDyjJQ= -github.com/kubeshop/testkube-operator v1.17.55-0.20241211150637-26f07cccd389/go.mod h1:P47tw1nKQFufdsZndyq2HG2MSa0zK/lU0XpRfZtEmIk= +github.com/kubeshop/testkube-operator v1.17.55-0.20241212130344-02ce60b6d6b1 h1:IV87xKZ8ezTbyOp0Qu6hDrzR/9Zo9iEvzsdB9uG8mOg= +github.com/kubeshop/testkube-operator v1.17.55-0.20241212130344-02ce60b6d6b1/go.mod h1:P47tw1nKQFufdsZndyq2HG2MSa0zK/lU0XpRfZtEmIk= github.com/leodido/go-urn v1.4.0 h1:WT9HwE9SGECu3lg4d/dIA+jxlljEa1/ffXKmRjqdmIQ= github.com/leodido/go-urn v1.4.0/go.mod h1:bvxc+MVxLKB4z00jd1z+Dvzr47oO32F/QSNjSBOlFxI= github.com/lithammer/fuzzysearch v1.1.8 h1:/HIuJnjHuXS8bKaiTMeeDlW2/AyIWk2brx1V8LFgLN4= From fd2dd602230b0f52785ae849ef798886fd8bf9e3 Mon Sep 17 00:00:00 2001 From: Vladislav Sukhin Date: Thu, 12 Dec 2024 16:30:40 +0300 Subject: [PATCH 12/36] fix: setvice pvcs Signed-off-by: Vladislav Sukhin --- pkg/mapper/testworkflows/kube_openapi.go | 3 ++- pkg/mapper/testworkflows/openapi_kube.go | 3 ++- 2 files changed, 4 insertions(+), 2 deletions(-) diff --git a/pkg/mapper/testworkflows/kube_openapi.go b/pkg/mapper/testworkflows/kube_openapi.go index ac221087d21..79c69d3f8c6 100644 --- a/pkg/mapper/testworkflows/kube_openapi.go +++ b/pkg/mapper/testworkflows/kube_openapi.go @@ -1030,6 +1030,7 @@ func MapIndependentServiceSpecKubeToAPI(v testworkflowsv1.IndependentServiceSpec Logs: MapStringToBoxedString(v.Logs), RestartPolicy: string(v.RestartPolicy), ReadinessProbe: common.MapPtr(v.ReadinessProbe, MapProbeKubeToAPI), + Pvcs: common.MapMap(v.Pvcs, MapPvcConfigKubeToAPI), } } @@ -1059,6 +1060,7 @@ func MapServiceSpecKubeToAPI(v testworkflowsv1.ServiceSpec) testkube.TestWorkflo Logs: MapStringToBoxedString(v.Logs), RestartPolicy: string(v.RestartPolicy), ReadinessProbe: common.MapPtr(v.ReadinessProbe, MapProbeKubeToAPI), + Pvcs: common.MapMap(v.Pvcs, MapPvcConfigKubeToAPI), } } @@ -1213,7 +1215,6 @@ func MapTestWorkflowTagSchemaKubeToAPI(v testworkflowsv1.TestWorkflowTagSchema) func MapPvcConfigKubeToAPI(v testworkflowsv1.TestWorkflowPvcConfig) testkube.TestWorkflowPvcConfig { return testkube.TestWorkflowPvcConfig{ - Shared: v.Shared, AccessModes: v.AccessModes, VolumeMode: v.VolumeMode, Resources: common.MapPtr(v.Resources, MapResourcesKubeToAPI), diff --git a/pkg/mapper/testworkflows/openapi_kube.go b/pkg/mapper/testworkflows/openapi_kube.go index ca78b436e5f..0865309fbfc 100644 --- a/pkg/mapper/testworkflows/openapi_kube.go +++ b/pkg/mapper/testworkflows/openapi_kube.go @@ -1100,6 +1100,7 @@ func MapIndependentServiceSpecAPIToKube(v testkube.TestWorkflowIndependentServic Logs: MapBoxedStringToString(v.Logs), RestartPolicy: testworkflowsv1.ServiceRestartPolicy(v.RestartPolicy), ReadinessProbe: common.MapPtr(v.ReadinessProbe, MapProbeAPIToKube), + Pvcs: common.MapMap(v.Pvcs, MapPvcConfigAPIToKube), } } @@ -1136,6 +1137,7 @@ func MapServiceSpecAPIToKube(v testkube.TestWorkflowServiceSpec) testworkflowsv1 Logs: MapBoxedStringToString(v.Logs), RestartPolicy: testworkflowsv1.ServiceRestartPolicy(v.RestartPolicy), ReadinessProbe: common.MapPtr(v.ReadinessProbe, MapProbeAPIToKube), + Pvcs: common.MapMap(v.Pvcs, MapPvcConfigAPIToKube), }, } } @@ -1492,7 +1494,6 @@ func MapTestWorkflowTagSchemaAPIToKube(v testkube.TestWorkflowTagSchema) testwor func MapPvcConfigAPIToKube(v testkube.TestWorkflowPvcConfig) testworkflowsv1.TestWorkflowPvcConfig { return testworkflowsv1.TestWorkflowPvcConfig{ - Shared: v.Shared, AccessModes: v.AccessModes, VolumeMode: v.VolumeMode, Resources: common.MapPtr(v.Resources, MapResourcesAPIToKube), From 551f1b2b46d03fab3a8219b77142bac738cef280 Mon Sep 17 00:00:00 2001 From: Vladislav Sukhin Date: Thu, 12 Dec 2024 18:44:08 +0300 Subject: [PATCH 13/36] fi: dep update Signed-off-by: Vladislav Sukhin --- go.mod | 2 +- go.sum | 4 ++-- 2 files changed, 3 insertions(+), 3 deletions(-) diff --git a/go.mod b/go.mod index c97ba8eab84..a37d552c929 100644 --- a/go.mod +++ b/go.mod @@ -42,7 +42,7 @@ require ( github.com/keygen-sh/jsonapi-go v1.2.1 github.com/keygen-sh/keygen-go/v3 v3.2.0 github.com/kubepug/kubepug v1.7.1 - github.com/kubeshop/testkube-operator v1.17.55-0.20241212130344-02ce60b6d6b1 + github.com/kubeshop/testkube-operator v1.17.55-0.20241212153943-9d18068064f1 github.com/minio/minio-go/v7 v7.0.66 github.com/montanaflynn/stats v0.7.1 github.com/moogar0880/problems v0.1.1 diff --git a/go.sum b/go.sum index 8161814abbc..a468a77b2cf 100644 --- a/go.sum +++ b/go.sum @@ -336,8 +336,8 @@ github.com/kr/text v0.2.0 h1:5Nx0Ya0ZqY2ygV366QzturHI13Jq95ApcVaJBhpS+AY= github.com/kr/text v0.2.0/go.mod h1:eLer722TekiGuMkidMxC/pM04lWEeraHUUmBw8l2grE= github.com/kubepug/kubepug v1.7.1 h1:LKhfSxS8Y5mXs50v+3Lpyec+cogErDLcV7CMUuiaisw= github.com/kubepug/kubepug v1.7.1/go.mod h1:lv+HxD0oTFL7ZWjj0u6HKhMbbTIId3eG7aWIW0gyF8g= -github.com/kubeshop/testkube-operator v1.17.55-0.20241212130344-02ce60b6d6b1 h1:IV87xKZ8ezTbyOp0Qu6hDrzR/9Zo9iEvzsdB9uG8mOg= -github.com/kubeshop/testkube-operator v1.17.55-0.20241212130344-02ce60b6d6b1/go.mod h1:P47tw1nKQFufdsZndyq2HG2MSa0zK/lU0XpRfZtEmIk= +github.com/kubeshop/testkube-operator v1.17.55-0.20241212153943-9d18068064f1 h1:TmmGhFII/sIdEy6lMRHkRovG7Xr1pW4NkF9sPiPgtNc= +github.com/kubeshop/testkube-operator v1.17.55-0.20241212153943-9d18068064f1/go.mod h1:P47tw1nKQFufdsZndyq2HG2MSa0zK/lU0XpRfZtEmIk= github.com/leodido/go-urn v1.4.0 h1:WT9HwE9SGECu3lg4d/dIA+jxlljEa1/ffXKmRjqdmIQ= github.com/leodido/go-urn v1.4.0/go.mod h1:bvxc+MVxLKB4z00jd1z+Dvzr47oO32F/QSNjSBOlFxI= github.com/lithammer/fuzzysearch v1.1.8 h1:/HIuJnjHuXS8bKaiTMeeDlW2/AyIWk2brx1V8LFgLN4= From 6faffdb5244b56a1124514e6e0595a62d7af49eb Mon Sep 17 00:00:00 2001 From: Vladislav Sukhin Date: Thu, 12 Dec 2024 19:05:21 +0300 Subject: [PATCH 14/36] fix: append pvcs Signed-off-by: Vladislav Sukhin --- .../testworkflowprocessor/intermediate.go | 13 +++++---- .../mock_intermediate.go | 28 +++++++++---------- .../testworkflowprocessor/processor.go | 4 ++- .../testworkflowresolver/apply.go | 1 + .../testworkflowresolver/merge.go | 13 +++++++++ 5 files changed, 39 insertions(+), 20 deletions(-) diff --git a/pkg/testworkflows/testworkflowprocessor/intermediate.go b/pkg/testworkflows/testworkflowprocessor/intermediate.go index 7b43adeb040..cd908f26839 100644 --- a/pkg/testworkflows/testworkflowprocessor/intermediate.go +++ b/pkg/testworkflows/testworkflowprocessor/intermediate.go @@ -4,6 +4,7 @@ import ( "fmt" corev1 "k8s.io/api/core/v1" + metav1 "k8s.io/apimachinery/pkg/apis/meta/v1" testworkflowsv1 "github.com/kubeshop/testkube-operator/api/testworkflows/v1" "github.com/kubeshop/testkube/pkg/testworkflows/testworkflowprocessor/stage" @@ -25,11 +26,11 @@ type Intermediate interface { AppendJobConfig(cfg *testworkflowsv1.JobConfig) Intermediate AppendPodConfig(cfg *testworkflowsv1.PodConfig) Intermediate + AppendPvc(cfg map[string]corev1.PersistentVolumeClaimSpec) Intermediate AddConfigMap(configMap corev1.ConfigMap) Intermediate AddSecret(secret corev1.Secret) Intermediate AddVolume(volume corev1.Volume) Intermediate - AddPvc(pvc corev1.PersistentVolumeClaim) Intermediate AddEmptyDirVolume(source *corev1.EmptyDirVolumeSource, mountPath string) corev1.VolumeMount @@ -104,13 +105,15 @@ func (s *intermediate) AppendPodConfig(cfg *testworkflowsv1.PodConfig) Intermedi return s } -func (s *intermediate) AddVolume(volume corev1.Volume) Intermediate { - s.Pod.Volumes = append(s.Pod.Volumes, volume) +func (s *intermediate) AppendPvc(cfg map[string]corev1.PersistentVolumeClaimSpec) Intermediate { + for name, spec := range cfg { + s.Ps = append(s.Ps, corev1.PersistentVolumeClaim{ObjectMeta: metav1.ObjectMeta{Name: name}, Spec: spec}) + } return s } -func (s *intermediate) AddPvc(pvc corev1.PersistentVolumeClaim) Intermediate { - s.Ps = append(s.Ps, pvc) +func (s *intermediate) AddVolume(volume corev1.Volume) Intermediate { + s.Pod.Volumes = append(s.Pod.Volumes, volume) return s } diff --git a/pkg/testworkflows/testworkflowprocessor/mock_intermediate.go b/pkg/testworkflows/testworkflowprocessor/mock_intermediate.go index 5d42526cf59..a5627ed0d50 100644 --- a/pkg/testworkflows/testworkflowprocessor/mock_intermediate.go +++ b/pkg/testworkflows/testworkflowprocessor/mock_intermediate.go @@ -79,20 +79,6 @@ func (mr *MockIntermediateMockRecorder) AddEmptyDirVolume(arg0, arg1 interface{} return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "AddEmptyDirVolume", reflect.TypeOf((*MockIntermediate)(nil).AddEmptyDirVolume), arg0, arg1) } -// AddPvc mocks base method. -func (m *MockIntermediate) AddPvc(arg0 v10.PersistentVolumeClaim) Intermediate { - m.ctrl.T.Helper() - ret := m.ctrl.Call(m, "AddPvc", arg0) - ret0, _ := ret[0].(Intermediate) - return ret0 -} - -// AddPvc indicates an expected call of AddPvc. -func (mr *MockIntermediateMockRecorder) AddPvc(arg0 interface{}) *gomock.Call { - mr.mock.ctrl.T.Helper() - return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "AddPvc", reflect.TypeOf((*MockIntermediate)(nil).AddPvc), arg0) -} - // AddSecret mocks base method. func (m *MockIntermediate) AddSecret(arg0 v10.Secret) Intermediate { m.ctrl.T.Helper() @@ -164,6 +150,20 @@ func (mr *MockIntermediateMockRecorder) AppendPodConfig(arg0 interface{}) *gomoc return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "AppendPodConfig", reflect.TypeOf((*MockIntermediate)(nil).AppendPodConfig), arg0) } +// AppendPvc mocks base method. +func (m *MockIntermediate) AppendPvc(arg0 map[string]v10.PersistentVolumeClaimSpec) Intermediate { + m.ctrl.T.Helper() + ret := m.ctrl.Call(m, "AppendPvc", arg0) + ret0, _ := ret[0].(Intermediate) + return ret0 +} + +// AppendPvc indicates an expected call of AppendPvc. +func (mr *MockIntermediateMockRecorder) AppendPvc(arg0 interface{}) *gomock.Call { + mr.mock.ctrl.T.Helper() + return mr.mock.ctrl.RecordCallWithMethodType(mr.mock, "AppendPvc", reflect.TypeOf((*MockIntermediate)(nil).AppendPvc), arg0) +} + // ConfigMaps mocks base method. func (m *MockIntermediate) ConfigMaps() []v10.ConfigMap { m.ctrl.T.Helper() diff --git a/pkg/testworkflows/testworkflowprocessor/processor.go b/pkg/testworkflows/testworkflowprocessor/processor.go index af5d881f9bb..bc5674addfd 100644 --- a/pkg/testworkflows/testworkflowprocessor/processor.go +++ b/pkg/testworkflows/testworkflowprocessor/processor.go @@ -23,6 +23,7 @@ import ( "github.com/kubeshop/testkube/pkg/testworkflows/testworkflowprocessor/action/actiontypes/lite" "github.com/kubeshop/testkube/pkg/testworkflows/testworkflowprocessor/constants" "github.com/kubeshop/testkube/pkg/testworkflows/testworkflowprocessor/stage" + "github.com/kubeshop/testkube/pkg/testworkflows/testworkflowresolver" ) //go:generate mockgen -destination=./mock_processor.go -package=testworkflowprocessor "github.com/kubeshop/testkube/pkg/testworkflows/testworkflowprocessor" Processor @@ -106,7 +107,8 @@ func (p *processor) Bundle(ctx context.Context, workflow *testworkflowsv1.TestWo // Initialize intermediate layer layer := NewIntermediate(). AppendPodConfig(workflow.Spec.Pod). - AppendJobConfig(workflow.Spec.Job) + AppendJobConfig(workflow.Spec.Job). + AppendPvc(common.MapMap(workflow.Spec.Pvcs, testworkflowresolver.ConvertTestWorkflowPvcConfigToPersistentVolumeClaimSpec)) layer.ContainerDefaults(). ApplyCR(constants.DefaultContainerConfig.DeepCopy()). AppendVolumeMounts(layer.AddEmptyDirVolume(nil, constants.DefaultInternalPath)). diff --git a/pkg/testworkflows/testworkflowresolver/apply.go b/pkg/testworkflows/testworkflowresolver/apply.go index 23b2845305e..07ee151165e 100644 --- a/pkg/testworkflows/testworkflowresolver/apply.go +++ b/pkg/testworkflows/testworkflowresolver/apply.go @@ -126,6 +126,7 @@ func InjectServiceTemplate(svc *testworkflowsv1.ServiceSpec, template testworkfl svc.Pod = MergePodConfig(template.Spec.Pod, svc.Pod) svc.Content = MergeContent(template.Spec.Content, svc.Content) svc.ContainerConfig = *MergeContainerConfig(template.Spec.Container, &svc.ContainerConfig) + svc.Pvcs = MergeMap(template.Spec.Pvcs, svc.Pvcs) return nil } diff --git a/pkg/testworkflows/testworkflowresolver/merge.go b/pkg/testworkflows/testworkflowresolver/merge.go index 1d503f1b84c..9d932cee8d6 100644 --- a/pkg/testworkflows/testworkflowresolver/merge.go +++ b/pkg/testworkflows/testworkflowresolver/merge.go @@ -373,6 +373,7 @@ func ConvertIndependentStepParallelToStepParallel(step testworkflowsv1.Independe Setup: common.MapSlice(step.TestWorkflowTemplateSpec.Setup, ConvertIndependentStepToStep), Steps: common.MapSlice(step.TestWorkflowTemplateSpec.Steps, ConvertIndependentStepToStep), After: common.MapSlice(step.TestWorkflowTemplateSpec.After, ConvertIndependentStepToStep), + Pvcs: step.TestWorkflowTemplateSpec.Pvcs, }, StepControl: step.StepControl, StepOperations: step.StepOperations, @@ -414,3 +415,15 @@ func MergeTags(dst, src map[string]string) map[string]string { return dst } + +func ConvertTestWorkflowPvcConfigToPersistentVolumeClaimSpec(pvc testworkflowsv1.TestWorkflowPvcConfig) corev1.PersistentVolumeClaimSpec { + return corev1.PersistentVolumeClaimSpec{ + AccessModes: common.MapSlice(pvc.AccessModes, func(s string) corev1.PersistentVolumeAccessMode { + return corev1.PersistentVolumeAccessMode(s) + }), + Selector: pvc.Selector, + // Resources: pvc.Resources.Limits, + StorageClassName: &pvc.StorageClassName, + VolumeMode: (*corev1.PersistentVolumeMode)(&pvc.VolumeMode), + } +} From da9e7f13417821955c1f1502bb4ddfbdccb3789b Mon Sep 17 00:00:00 2001 From: Vladislav Sukhin Date: Thu, 12 Dec 2024 19:35:07 +0300 Subject: [PATCH 15/36] fix: copy service pvcs Signed-off-by: Vladislav Sukhin --- cmd/tcl/testworkflow-toolkit/commands/services.go | 1 + 1 file changed, 1 insertion(+) diff --git a/cmd/tcl/testworkflow-toolkit/commands/services.go b/cmd/tcl/testworkflow-toolkit/commands/services.go index b32f3231fb9..ce692cc849b 100644 --- a/cmd/tcl/testworkflow-toolkit/commands/services.go +++ b/cmd/tcl/testworkflow-toolkit/commands/services.go @@ -156,6 +156,7 @@ func NewServicesCmd() *cobra.Command { Steps: []testworkflowsv1.Step{ {StepOperations: testworkflowsv1.StepOperations{Run: common.Ptr(svcSpec.StepRun)}}, }, + Pvcs: svcSpec.Pvcs, } spec.Steps[0].Run.ContainerConfig = testworkflowsv1.ContainerConfig{} From 541721126fa860b2177bc39dec0685868f3fe145 Mon Sep 17 00:00:00 2001 From: Vladislav Sukhin Date: Thu, 12 Dec 2024 19:43:24 +0300 Subject: [PATCH 16/36] fix: pvc name for execution Signed-off-by: Vladislav Sukhin --- pkg/testworkflows/testworkflowprocessor/intermediate.go | 8 +++++++- 1 file changed, 7 insertions(+), 1 deletion(-) diff --git a/pkg/testworkflows/testworkflowprocessor/intermediate.go b/pkg/testworkflows/testworkflowprocessor/intermediate.go index cd908f26839..9be9b86fe09 100644 --- a/pkg/testworkflows/testworkflowprocessor/intermediate.go +++ b/pkg/testworkflows/testworkflowprocessor/intermediate.go @@ -106,8 +106,14 @@ func (s *intermediate) AppendPodConfig(cfg *testworkflowsv1.PodConfig) Intermedi } func (s *intermediate) AppendPvc(cfg map[string]corev1.PersistentVolumeClaimSpec) Intermediate { + ref := s.NextRef() for name, spec := range cfg { - s.Ps = append(s.Ps, corev1.PersistentVolumeClaim{ObjectMeta: metav1.ObjectMeta{Name: name}, Spec: spec}) + s.Ps = append(s.Ps, corev1.PersistentVolumeClaim{ + ObjectMeta: metav1.ObjectMeta{ + Name: fmt.Sprintf("%s-%s", name, ref), + }, + Spec: spec, + }) } return s } From 597a69a4c4a1662935fcf6820454626d31d9e184 Mon Sep 17 00:00:00 2001 From: Vladislav Sukhin Date: Thu, 12 Dec 2024 21:44:02 +0300 Subject: [PATCH 17/36] fix: register pvc names Signed-off-by: Vladislav Sukhin --- .../testworkflowconfig/expressions.go | 22 ++++++++++++++++++- .../testworkflowprocessor/processor.go | 3 ++- 2 files changed, 23 insertions(+), 2 deletions(-) diff --git a/pkg/testworkflows/testworkflowconfig/expressions.go b/pkg/testworkflows/testworkflowconfig/expressions.go index 3cd5cd330b8..b951bddb63a 100644 --- a/pkg/testworkflows/testworkflowconfig/expressions.go +++ b/pkg/testworkflows/testworkflowconfig/expressions.go @@ -1,6 +1,12 @@ package testworkflowconfig -import "github.com/kubeshop/testkube/pkg/expressions" +import ( + "strings" + + corev1 "k8s.io/api/core/v1" + + "github.com/kubeshop/testkube/pkg/expressions" +) func CreateExecutionMachine(cfg *ExecutionConfig) expressions.Machine { return expressions.NewMachine(). @@ -77,3 +83,17 @@ func CreateWorkerMachine(cfg *WorkerConfig) expressions.Machine { }) return expressions.CombinedMachines(machine) } + +func CreatePvcMachine(pvcs []corev1.PersistentVolumeClaim) expressions.Machine { + pvcMap := make(map[string]string) + for _, pvc := range pvcs { + name := pvc.Name + if index := strings.LastIndex(name, "-"); index != -1 { + name = name[:index] + } + + pvcMap[name+".name"] = pvc.Name + } + + return expressions.NewMachine().RegisterStringMap("pvcs", pvcMap) +} diff --git a/pkg/testworkflows/testworkflowprocessor/processor.go b/pkg/testworkflows/testworkflowprocessor/processor.go index bc5674addfd..43e8b054a4d 100644 --- a/pkg/testworkflows/testworkflowprocessor/processor.go +++ b/pkg/testworkflows/testworkflowprocessor/processor.go @@ -119,7 +119,8 @@ func (p *processor) Bundle(ctx context.Context, workflow *testworkflowsv1.TestWo machines = append(machines, createSecretMachine(mapEnv), testworkflowconfig.CreateWorkerMachine(&options.Config.Worker), - testworkflowconfig.CreateResourceMachine(&options.Config.Resource)) + testworkflowconfig.CreateResourceMachine(&options.Config.Resource), + testworkflowconfig.CreatePvcMachine(layer.Pvcs())) // Fetch resource root and resource ID if options.Config.Resource.Id == "" { From bd3aad2819967cadc472e829615c793be948c371 Mon Sep 17 00:00:00 2001 From: Vladislav Sukhin Date: Fri, 13 Dec 2024 16:08:28 +0300 Subject: [PATCH 18/36] fix: convert pvc Signed-off-by: Vladislav Sukhin --- api/v1/testkube.yaml | 3 ++ go.mod | 2 +- go.sum | 2 + .../model_test_workflow_pvc_config.go | 6 ++- pkg/mapper/testworkflows/kube_openapi.go | 1 + pkg/mapper/testworkflows/openapi_kube.go | 1 + .../testworkflowresolver/merge.go | 51 ++++++++++++++++++- 7 files changed, 61 insertions(+), 5 deletions(-) diff --git a/api/v1/testkube.yaml b/api/v1/testkube.yaml index 3855d42115f..60336c86c10 100644 --- a/api/v1/testkube.yaml +++ b/api/v1/testkube.yaml @@ -10618,6 +10618,9 @@ components: storageClassName: description: 'Storage class name specifies the name of a StorageClass. More info: https://kubernetes.io/docs/concepts/storage/storage-classes/' type: string + volumeName: + description: 'Volume name is used to identify the volume' + type: string selector: description: Only the volumes whose labels match the selector can be bound to the claim $ref: "#/components/schemas/LabelSelector" diff --git a/go.mod b/go.mod index a37d552c929..f3e1bd55f07 100644 --- a/go.mod +++ b/go.mod @@ -42,7 +42,7 @@ require ( github.com/keygen-sh/jsonapi-go v1.2.1 github.com/keygen-sh/keygen-go/v3 v3.2.0 github.com/kubepug/kubepug v1.7.1 - github.com/kubeshop/testkube-operator v1.17.55-0.20241212153943-9d18068064f1 + github.com/kubeshop/testkube-operator v1.17.55-0.20241213114930-df3327436207 github.com/minio/minio-go/v7 v7.0.66 github.com/montanaflynn/stats v0.7.1 github.com/moogar0880/problems v0.1.1 diff --git a/go.sum b/go.sum index a468a77b2cf..d55ba5f3ef2 100644 --- a/go.sum +++ b/go.sum @@ -338,6 +338,8 @@ github.com/kubepug/kubepug v1.7.1 h1:LKhfSxS8Y5mXs50v+3Lpyec+cogErDLcV7CMUuiaisw github.com/kubepug/kubepug v1.7.1/go.mod h1:lv+HxD0oTFL7ZWjj0u6HKhMbbTIId3eG7aWIW0gyF8g= github.com/kubeshop/testkube-operator v1.17.55-0.20241212153943-9d18068064f1 h1:TmmGhFII/sIdEy6lMRHkRovG7Xr1pW4NkF9sPiPgtNc= github.com/kubeshop/testkube-operator v1.17.55-0.20241212153943-9d18068064f1/go.mod h1:P47tw1nKQFufdsZndyq2HG2MSa0zK/lU0XpRfZtEmIk= +github.com/kubeshop/testkube-operator v1.17.55-0.20241213114930-df3327436207 h1:IQvoge5CpG0Z1uEjb/rcA4SN+ZiAmPrO0XSTLLlPDeg= +github.com/kubeshop/testkube-operator v1.17.55-0.20241213114930-df3327436207/go.mod h1:P47tw1nKQFufdsZndyq2HG2MSa0zK/lU0XpRfZtEmIk= github.com/leodido/go-urn v1.4.0 h1:WT9HwE9SGECu3lg4d/dIA+jxlljEa1/ffXKmRjqdmIQ= github.com/leodido/go-urn v1.4.0/go.mod h1:bvxc+MVxLKB4z00jd1z+Dvzr47oO32F/QSNjSBOlFxI= github.com/lithammer/fuzzysearch v1.1.8 h1:/HIuJnjHuXS8bKaiTMeeDlW2/AyIWk2brx1V8LFgLN4= diff --git a/pkg/api/v1/testkube/model_test_workflow_pvc_config.go b/pkg/api/v1/testkube/model_test_workflow_pvc_config.go index 289c61df362..77cacf8f69e 100644 --- a/pkg/api/v1/testkube/model_test_workflow_pvc_config.go +++ b/pkg/api/v1/testkube/model_test_workflow_pvc_config.go @@ -16,6 +16,8 @@ type TestWorkflowPvcConfig struct { VolumeMode string `json:"volumeMode,omitempty"` Resources *TestWorkflowResources `json:"resources,omitempty"` // Storage class name specifies the name of a StorageClass. More info: https://kubernetes.io/docs/concepts/storage/storage-classes/ - StorageClassName string `json:"storageClassName,omitempty"` - Selector *LabelSelector `json:"selector,omitempty"` + StorageClassName string `json:"storageClassName,omitempty"` + // Volume name is used to identify the volume + VolumeName string `json:"volumeName,omitempty"` + Selector *LabelSelector `json:"selector,omitempty"` } diff --git a/pkg/mapper/testworkflows/kube_openapi.go b/pkg/mapper/testworkflows/kube_openapi.go index 79c69d3f8c6..5509cc6736d 100644 --- a/pkg/mapper/testworkflows/kube_openapi.go +++ b/pkg/mapper/testworkflows/kube_openapi.go @@ -1219,6 +1219,7 @@ func MapPvcConfigKubeToAPI(v testworkflowsv1.TestWorkflowPvcConfig) testkube.Tes VolumeMode: v.VolumeMode, Resources: common.MapPtr(v.Resources, MapResourcesKubeToAPI), StorageClassName: v.StorageClassName, + VolumeName: v.VolumeName, Selector: common.MapPtr(v.Selector, MapSelectorToAPI), } } diff --git a/pkg/mapper/testworkflows/openapi_kube.go b/pkg/mapper/testworkflows/openapi_kube.go index 0865309fbfc..caf0f3860ec 100644 --- a/pkg/mapper/testworkflows/openapi_kube.go +++ b/pkg/mapper/testworkflows/openapi_kube.go @@ -1498,6 +1498,7 @@ func MapPvcConfigAPIToKube(v testkube.TestWorkflowPvcConfig) testworkflowsv1.Tes VolumeMode: v.VolumeMode, Resources: common.MapPtr(v.Resources, MapResourcesAPIToKube), StorageClassName: v.StorageClassName, + VolumeName: v.VolumeName, Selector: common.MapPtr(v.Selector, MapLabelSelectorAPIToKube), } } diff --git a/pkg/testworkflows/testworkflowresolver/merge.go b/pkg/testworkflows/testworkflowresolver/merge.go index 9d932cee8d6..9da13ade7a2 100644 --- a/pkg/testworkflows/testworkflowresolver/merge.go +++ b/pkg/testworkflows/testworkflowresolver/merge.go @@ -12,6 +12,8 @@ import ( "maps" corev1 "k8s.io/api/core/v1" + "k8s.io/apimachinery/pkg/api/resource" + "k8s.io/apimachinery/pkg/util/intstr" testworkflowsv1 "github.com/kubeshop/testkube-operator/api/testworkflows/v1" "github.com/kubeshop/testkube/internal/common" @@ -421,9 +423,54 @@ func ConvertTestWorkflowPvcConfigToPersistentVolumeClaimSpec(pvc testworkflowsv1 AccessModes: common.MapSlice(pvc.AccessModes, func(s string) corev1.PersistentVolumeAccessMode { return corev1.PersistentVolumeAccessMode(s) }), - Selector: pvc.Selector, - // Resources: pvc.Resources.Limits, + Selector: pvc.Selector, + Resources: ConvertResourcesToVolumeResourceRequirements(pvc.Resources), + VolumeName: pvc.VolumeName, StorageClassName: &pvc.StorageClassName, VolumeMode: (*corev1.PersistentVolumeMode)(&pvc.VolumeMode), } } + +func ConvertResourcesToVolumeResourceRequirements(r *testworkflowsv1.Resources) corev1.VolumeResourceRequirements { + var limits, requests corev1.ResourceList + if r != nil { + if len(r.Limits) != 0 { + limits = make(corev1.ResourceList) + } + + if len(r.Requests) != 0 { + requests = make(corev1.ResourceList) + } + + for key, value := range r.Limits { + var quantity resource.Quantity + if value.Type == intstr.Int { + quantity = *resource.NewQuantity(int64(value.IntVal), resource.BinarySI) + } + + if value.Type == intstr.String { + quantity, _ = resource.ParseQuantity(value.String()) + } + + limits[key] = quantity + } + + for key, value := range r.Requests { + var quantity resource.Quantity + if value.Type == intstr.Int { + quantity = *resource.NewQuantity(int64(value.IntVal), resource.BinarySI) + } + + if value.Type == intstr.String { + quantity, _ = resource.ParseQuantity(value.String()) + } + + requests[key] = quantity + } + } + + return corev1.VolumeResourceRequirements{ + Limits: limits, + Requests: requests, + } +} From 95c8992e9905b54935098673814ba1d88556a086 Mon Sep 17 00:00:00 2001 From: Vladislav Sukhin Date: Fri, 13 Dec 2024 17:14:07 +0300 Subject: [PATCH 19/36] fix: move fields to boxed strings Signed-off-by: Vladislav Sukhin --- api/v1/testkube.yaml | 4 ++-- pkg/api/v1/testkube/model_test_workflow_pvc_config.go | 10 ++++------ 2 files changed, 6 insertions(+), 8 deletions(-) diff --git a/api/v1/testkube.yaml b/api/v1/testkube.yaml index 60336c86c10..065638042c4 100644 --- a/api/v1/testkube.yaml +++ b/api/v1/testkube.yaml @@ -10611,13 +10611,13 @@ components: volumeMode: description: 'Volume mode indicates the consumption of the volume as either a filesystem or block device. More info: https://kubernetes.io/docs/concepts/storage/persistent-volumes/#volume-mode' - type: string + $ref: "#/components/schemas/BoxedString" resources: description: 'Resources required for pvc' $ref: "#/components/schemas/TestWorkflowResources" storageClassName: description: 'Storage class name specifies the name of a StorageClass. More info: https://kubernetes.io/docs/concepts/storage/storage-classes/' - type: string + $ref: "#/components/schemas/BoxedString" volumeName: description: 'Volume name is used to identify the volume' type: string diff --git a/pkg/api/v1/testkube/model_test_workflow_pvc_config.go b/pkg/api/v1/testkube/model_test_workflow_pvc_config.go index 77cacf8f69e..02b3f00cdb5 100644 --- a/pkg/api/v1/testkube/model_test_workflow_pvc_config.go +++ b/pkg/api/v1/testkube/model_test_workflow_pvc_config.go @@ -11,12 +11,10 @@ package testkube type TestWorkflowPvcConfig struct { // Access mode for claim storage. More info: https://kubernetes.io/docs/concepts/storage/persistent-volumes/#access-modes - AccessModes []string `json:"accessModes,omitempty"` - // Volume mode indicates the consumption of the volume as either a filesystem or block device. More info: https://kubernetes.io/docs/concepts/storage/persistent-volumes/#volume-mode - VolumeMode string `json:"volumeMode,omitempty"` - Resources *TestWorkflowResources `json:"resources,omitempty"` - // Storage class name specifies the name of a StorageClass. More info: https://kubernetes.io/docs/concepts/storage/storage-classes/ - StorageClassName string `json:"storageClassName,omitempty"` + AccessModes []string `json:"accessModes,omitempty"` + VolumeMode *BoxedString `json:"volumeMode,omitempty"` + Resources *TestWorkflowResources `json:"resources,omitempty"` + StorageClassName *BoxedString `json:"storageClassName,omitempty"` // Volume name is used to identify the volume VolumeName string `json:"volumeName,omitempty"` Selector *LabelSelector `json:"selector,omitempty"` From 54b352d5ceaaf8c723455bb60bc5386fde1a082b Mon Sep 17 00:00:00 2001 From: Vladislav Sukhin Date: Fri, 13 Dec 2024 17:59:35 +0300 Subject: [PATCH 20/36] fix: dep update Signed-off-by: Vladislav Sukhin --- go.mod | 2 +- go.sum | 6 ++---- pkg/mapper/testworkflows/kube_openapi.go | 4 ++-- pkg/mapper/testworkflows/openapi_kube.go | 4 ++-- pkg/testworkflows/testworkflowresolver/merge.go | 4 ++-- 5 files changed, 9 insertions(+), 11 deletions(-) diff --git a/go.mod b/go.mod index f3e1bd55f07..d83e255cfb4 100644 --- a/go.mod +++ b/go.mod @@ -42,7 +42,7 @@ require ( github.com/keygen-sh/jsonapi-go v1.2.1 github.com/keygen-sh/keygen-go/v3 v3.2.0 github.com/kubepug/kubepug v1.7.1 - github.com/kubeshop/testkube-operator v1.17.55-0.20241213114930-df3327436207 + github.com/kubeshop/testkube-operator v1.17.55-0.20241213144851-e0d37fb5899a github.com/minio/minio-go/v7 v7.0.66 github.com/montanaflynn/stats v0.7.1 github.com/moogar0880/problems v0.1.1 diff --git a/go.sum b/go.sum index d55ba5f3ef2..35bbe8851a3 100644 --- a/go.sum +++ b/go.sum @@ -336,10 +336,8 @@ github.com/kr/text v0.2.0 h1:5Nx0Ya0ZqY2ygV366QzturHI13Jq95ApcVaJBhpS+AY= github.com/kr/text v0.2.0/go.mod h1:eLer722TekiGuMkidMxC/pM04lWEeraHUUmBw8l2grE= github.com/kubepug/kubepug v1.7.1 h1:LKhfSxS8Y5mXs50v+3Lpyec+cogErDLcV7CMUuiaisw= github.com/kubepug/kubepug v1.7.1/go.mod h1:lv+HxD0oTFL7ZWjj0u6HKhMbbTIId3eG7aWIW0gyF8g= -github.com/kubeshop/testkube-operator v1.17.55-0.20241212153943-9d18068064f1 h1:TmmGhFII/sIdEy6lMRHkRovG7Xr1pW4NkF9sPiPgtNc= -github.com/kubeshop/testkube-operator v1.17.55-0.20241212153943-9d18068064f1/go.mod h1:P47tw1nKQFufdsZndyq2HG2MSa0zK/lU0XpRfZtEmIk= -github.com/kubeshop/testkube-operator v1.17.55-0.20241213114930-df3327436207 h1:IQvoge5CpG0Z1uEjb/rcA4SN+ZiAmPrO0XSTLLlPDeg= -github.com/kubeshop/testkube-operator v1.17.55-0.20241213114930-df3327436207/go.mod h1:P47tw1nKQFufdsZndyq2HG2MSa0zK/lU0XpRfZtEmIk= +github.com/kubeshop/testkube-operator v1.17.55-0.20241213144851-e0d37fb5899a h1:nQDxKFu67lz9BH8isLnP7ttp7cgBWYnRDMV9VxMaYH0= +github.com/kubeshop/testkube-operator v1.17.55-0.20241213144851-e0d37fb5899a/go.mod h1:P47tw1nKQFufdsZndyq2HG2MSa0zK/lU0XpRfZtEmIk= github.com/leodido/go-urn v1.4.0 h1:WT9HwE9SGECu3lg4d/dIA+jxlljEa1/ffXKmRjqdmIQ= github.com/leodido/go-urn v1.4.0/go.mod h1:bvxc+MVxLKB4z00jd1z+Dvzr47oO32F/QSNjSBOlFxI= github.com/lithammer/fuzzysearch v1.1.8 h1:/HIuJnjHuXS8bKaiTMeeDlW2/AyIWk2brx1V8LFgLN4= diff --git a/pkg/mapper/testworkflows/kube_openapi.go b/pkg/mapper/testworkflows/kube_openapi.go index 5509cc6736d..333a2c4d121 100644 --- a/pkg/mapper/testworkflows/kube_openapi.go +++ b/pkg/mapper/testworkflows/kube_openapi.go @@ -1216,9 +1216,9 @@ func MapTestWorkflowTagSchemaKubeToAPI(v testworkflowsv1.TestWorkflowTagSchema) func MapPvcConfigKubeToAPI(v testworkflowsv1.TestWorkflowPvcConfig) testkube.TestWorkflowPvcConfig { return testkube.TestWorkflowPvcConfig{ AccessModes: v.AccessModes, - VolumeMode: v.VolumeMode, + VolumeMode: MapStringToBoxedString(v.VolumeMode), Resources: common.MapPtr(v.Resources, MapResourcesKubeToAPI), - StorageClassName: v.StorageClassName, + StorageClassName: MapStringToBoxedString(v.StorageClassName), VolumeName: v.VolumeName, Selector: common.MapPtr(v.Selector, MapSelectorToAPI), } diff --git a/pkg/mapper/testworkflows/openapi_kube.go b/pkg/mapper/testworkflows/openapi_kube.go index caf0f3860ec..fa5a76abc01 100644 --- a/pkg/mapper/testworkflows/openapi_kube.go +++ b/pkg/mapper/testworkflows/openapi_kube.go @@ -1495,9 +1495,9 @@ func MapTestWorkflowTagSchemaAPIToKube(v testkube.TestWorkflowTagSchema) testwor func MapPvcConfigAPIToKube(v testkube.TestWorkflowPvcConfig) testworkflowsv1.TestWorkflowPvcConfig { return testworkflowsv1.TestWorkflowPvcConfig{ AccessModes: v.AccessModes, - VolumeMode: v.VolumeMode, + VolumeMode: MapBoxedStringToString(v.VolumeMode), Resources: common.MapPtr(v.Resources, MapResourcesAPIToKube), - StorageClassName: v.StorageClassName, + StorageClassName: MapBoxedStringToString(v.StorageClassName), VolumeName: v.VolumeName, Selector: common.MapPtr(v.Selector, MapLabelSelectorAPIToKube), } diff --git a/pkg/testworkflows/testworkflowresolver/merge.go b/pkg/testworkflows/testworkflowresolver/merge.go index 9da13ade7a2..3ebf623ab09 100644 --- a/pkg/testworkflows/testworkflowresolver/merge.go +++ b/pkg/testworkflows/testworkflowresolver/merge.go @@ -426,8 +426,8 @@ func ConvertTestWorkflowPvcConfigToPersistentVolumeClaimSpec(pvc testworkflowsv1 Selector: pvc.Selector, Resources: ConvertResourcesToVolumeResourceRequirements(pvc.Resources), VolumeName: pvc.VolumeName, - StorageClassName: &pvc.StorageClassName, - VolumeMode: (*corev1.PersistentVolumeMode)(&pvc.VolumeMode), + StorageClassName: pvc.StorageClassName, + VolumeMode: (*corev1.PersistentVolumeMode)(pvc.VolumeMode), } } From 3520a1a760c3e44a8ed957f85490ffc7e36aa82f Mon Sep 17 00:00:00 2001 From: Vladislav Sukhin Date: Mon, 16 Dec 2024 16:29:45 +0300 Subject: [PATCH 21/36] fix: pass pvc names Signed-off-by: Vladislav Sukhin --- cmd/tcl/testworkflow-toolkit/commands/parallel.go | 1 + cmd/tcl/testworkflow-toolkit/commands/services.go | 1 + pkg/testworkflows/testworkflowconfig/config.go | 1 + pkg/testworkflows/testworkflowconfig/expressions.go | 10 ++++------ pkg/testworkflows/testworkflowprocessor/processor.go | 6 +++++- 5 files changed, 12 insertions(+), 7 deletions(-) diff --git a/cmd/tcl/testworkflow-toolkit/commands/parallel.go b/cmd/tcl/testworkflow-toolkit/commands/parallel.go index 75d6f233df9..3c3a192f2e4 100644 --- a/cmd/tcl/testworkflow-toolkit/commands/parallel.go +++ b/cmd/tcl/testworkflow-toolkit/commands/parallel.go @@ -205,6 +205,7 @@ func NewParallelCmd() *cobra.Command { testworkflowconfig.CreateResourceMachine(&cfg.Resource), testworkflowconfig.CreateWorkerMachine(&cfg.Worker), baseMachine, + testworkflowconfig.CreatePvcMachine(cfg.Execution.Pvcs), params.MachineAt(index), ) diff --git a/cmd/tcl/testworkflow-toolkit/commands/services.go b/cmd/tcl/testworkflow-toolkit/commands/services.go index ce692cc849b..c357c49363a 100644 --- a/cmd/tcl/testworkflow-toolkit/commands/services.go +++ b/cmd/tcl/testworkflow-toolkit/commands/services.go @@ -261,6 +261,7 @@ func NewServicesCmd() *cobra.Command { testworkflowconfig.CreateResourceMachine(&cfg.Resource), testworkflowconfig.CreateWorkerMachine(&cfg.Worker), baseMachine, + testworkflowconfig.CreatePvcMachine(cfg.Execution.Pvcs), params.MachineAt(index), ) diff --git a/pkg/testworkflows/testworkflowconfig/config.go b/pkg/testworkflows/testworkflowconfig/config.go index 8f41f3c8cfe..670b7538aed 100644 --- a/pkg/testworkflows/testworkflowconfig/config.go +++ b/pkg/testworkflows/testworkflowconfig/config.go @@ -22,6 +22,7 @@ type ExecutionConfig struct { OrganizationId string `json:"o,omitempty"` EnvironmentId string `json:"e,omitempty"` ParentIds string `json:"p,omitempty"` + PvcNames []string `json:"P,omitempty"` } type WorkflowConfig struct { diff --git a/pkg/testworkflows/testworkflowconfig/expressions.go b/pkg/testworkflows/testworkflowconfig/expressions.go index b951bddb63a..b611b32a0d3 100644 --- a/pkg/testworkflows/testworkflowconfig/expressions.go +++ b/pkg/testworkflows/testworkflowconfig/expressions.go @@ -3,8 +3,6 @@ package testworkflowconfig import ( "strings" - corev1 "k8s.io/api/core/v1" - "github.com/kubeshop/testkube/pkg/expressions" ) @@ -84,15 +82,15 @@ func CreateWorkerMachine(cfg *WorkerConfig) expressions.Machine { return expressions.CombinedMachines(machine) } -func CreatePvcMachine(pvcs []corev1.PersistentVolumeClaim) expressions.Machine { +func CreatePvcMachine(pvcNames []string) expressions.Machine { pvcMap := make(map[string]string) - for _, pvc := range pvcs { - name := pvc.Name + for _, pvcName := range pvcNames { + name := pvcName if index := strings.LastIndex(name, "-"); index != -1 { name = name[:index] } - pvcMap[name+".name"] = pvc.Name + pvcMap[name+".name"] = pvcName } return expressions.NewMachine().RegisterStringMap("pvcs", pvcMap) diff --git a/pkg/testworkflows/testworkflowprocessor/processor.go b/pkg/testworkflows/testworkflowprocessor/processor.go index 43e8b054a4d..9b65608d5f1 100644 --- a/pkg/testworkflows/testworkflowprocessor/processor.go +++ b/pkg/testworkflows/testworkflowprocessor/processor.go @@ -120,7 +120,9 @@ func (p *processor) Bundle(ctx context.Context, workflow *testworkflowsv1.TestWo createSecretMachine(mapEnv), testworkflowconfig.CreateWorkerMachine(&options.Config.Worker), testworkflowconfig.CreateResourceMachine(&options.Config.Resource), - testworkflowconfig.CreatePvcMachine(layer.Pvcs())) + testworkflowconfig.CreatePvcMachine( + common.MapSlice(layer.Pvcs(), func(p corev1.PersistentVolumeClaim) string { return p.Name })), + ) // Fetch resource root and resource ID if options.Config.Resource.Id == "" { @@ -174,6 +176,8 @@ func (p *processor) Bundle(ctx context.Context, workflow *testworkflowsv1.TestWo return nil, errors.Wrap(err, "finalizing Pvc") } } + pvcNames := common.MapSlice(pvcs, func(p corev1.PersistentVolumeClaim) string { return p.Name }) + options.Config.Execution.PvcNames = pvcNames // Finalize Secrets secrets := append(layer.Secrets(), options.Secrets...) From 6effa79bbd8e430eb7dc4cfaa918a98f6ffe9a21 Mon Sep 17 00:00:00 2001 From: Vladislav Sukhin Date: Mon, 16 Dec 2024 16:35:36 +0300 Subject: [PATCH 22/36] fix: renname cobfig field Signed-off-by: Vladislav Sukhin --- cmd/tcl/testworkflow-toolkit/commands/parallel.go | 2 +- cmd/tcl/testworkflow-toolkit/commands/services.go | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/cmd/tcl/testworkflow-toolkit/commands/parallel.go b/cmd/tcl/testworkflow-toolkit/commands/parallel.go index 3c3a192f2e4..d06f0bd0eda 100644 --- a/cmd/tcl/testworkflow-toolkit/commands/parallel.go +++ b/cmd/tcl/testworkflow-toolkit/commands/parallel.go @@ -205,7 +205,7 @@ func NewParallelCmd() *cobra.Command { testworkflowconfig.CreateResourceMachine(&cfg.Resource), testworkflowconfig.CreateWorkerMachine(&cfg.Worker), baseMachine, - testworkflowconfig.CreatePvcMachine(cfg.Execution.Pvcs), + testworkflowconfig.CreatePvcMachine(cfg.Execution.PvcNames), params.MachineAt(index), ) diff --git a/cmd/tcl/testworkflow-toolkit/commands/services.go b/cmd/tcl/testworkflow-toolkit/commands/services.go index c357c49363a..96784dd001d 100644 --- a/cmd/tcl/testworkflow-toolkit/commands/services.go +++ b/cmd/tcl/testworkflow-toolkit/commands/services.go @@ -261,7 +261,7 @@ func NewServicesCmd() *cobra.Command { testworkflowconfig.CreateResourceMachine(&cfg.Resource), testworkflowconfig.CreateWorkerMachine(&cfg.Worker), baseMachine, - testworkflowconfig.CreatePvcMachine(cfg.Execution.Pvcs), + testworkflowconfig.CreatePvcMachine(cfg.Execution.PvcNames), params.MachineAt(index), ) From f89592ec3739b88838c9932cde115381037e9d5f Mon Sep 17 00:00:00 2001 From: Vladislav Sukhin Date: Mon, 16 Dec 2024 17:09:57 +0300 Subject: [PATCH 23/36] fix: remove var Signed-off-by: Vladislav Sukhin --- pkg/testworkflows/testworkflowprocessor/processor.go | 3 +-- 1 file changed, 1 insertion(+), 2 deletions(-) diff --git a/pkg/testworkflows/testworkflowprocessor/processor.go b/pkg/testworkflows/testworkflowprocessor/processor.go index 9b65608d5f1..870eed6df6b 100644 --- a/pkg/testworkflows/testworkflowprocessor/processor.go +++ b/pkg/testworkflows/testworkflowprocessor/processor.go @@ -176,8 +176,7 @@ func (p *processor) Bundle(ctx context.Context, workflow *testworkflowsv1.TestWo return nil, errors.Wrap(err, "finalizing Pvc") } } - pvcNames := common.MapSlice(pvcs, func(p corev1.PersistentVolumeClaim) string { return p.Name }) - options.Config.Execution.PvcNames = pvcNames + options.Config.Execution.PvcNames = common.MapSlice(pvcs, func(p corev1.PersistentVolumeClaim) string { return p.Name }) // Finalize Secrets secrets := append(layer.Secrets(), options.Secrets...) From c47ff99b72c8a8208a1e7d6c5fceb6ec2fbf8461 Mon Sep 17 00:00:00 2001 From: Vladislav Sukhin Date: Mon, 16 Dec 2024 22:31:22 +0300 Subject: [PATCH 24/36] fix: add new pvc config fields Signed-off-by: Vladislav Sukhin --- api/v1/testkube.yaml | 35 +++++++++++ go.mod | 2 +- go.sum | 4 +- .../model_test_workflow_pvc_config.go | 7 ++- .../model_typed_local_object_reference.go | 19 ++++++ .../testkube/model_typed_object_reference.go | 20 +++++++ pkg/mapper/testworkflows/kube_openapi.go | 8 +-- pkg/mapper/testworkflows/openapi_kube.go | 10 ++-- .../testworkflowprocessor/processor.go | 3 +- .../testworkflowresolver/merge.go | 59 ------------------- 10 files changed, 92 insertions(+), 75 deletions(-) create mode 100644 pkg/api/v1/testkube/model_typed_local_object_reference.go create mode 100644 pkg/api/v1/testkube/model_typed_object_reference.go diff --git a/api/v1/testkube.yaml b/api/v1/testkube.yaml index 065638042c4..76f45fb906e 100644 --- a/api/v1/testkube.yaml +++ b/api/v1/testkube.yaml @@ -10624,6 +10624,41 @@ components: selector: description: Only the volumes whose labels match the selector can be bound to the claim $ref: "#/components/schemas/LabelSelector" + dataSource: + description: 'Data source field can be used to specify either: + * An existing VolumeSnapshot object (snapshot.storage.k8s.io/VolumeSnapshot) + * An existing PVC (PersistentVolumeClaim)' + $ref: "#/components/schemas/TypedLocalObjectReference" + dataSourceRef: + description: 'Data source reference specifies the object from which to populate the volume with data, if a non-empty volume is desired' + $ref: "#/components/schemas/TypedObjectReference" + volumeAttributesClassName: + description: 'Volume attributes class name may be used to set the VolumeAttributesClass used by this claim. More info: https://kubernetes.io/docs/concepts/storage/persistent-volumes#volumeattributesclass' + $ref: "#/components/schemas/BoxedString" + + TypedLocalObjectReference: + description: TypedLocalObjectReference contains enough information to let you locate the typed referenced object inside the same namespace + type: object + properties: + apiGroup: + description: api group is the group for the resource being referenced + $ref: "#/components/schemas/BoxedString" + kind: + description: kind is the type of resource being referenced + type: string + name: + description: name is the name of resource being referenced + type: string + + TypedObjectReference: + description: TypedObjectReference contains enough information to let you locate the typed referenced object inside the specified namespace + type: object + allOf: + - $ref: "#/components/schemas/TypedLocalObjectReference" + properties: + namespace: + description: Namespace is the namespace of resource being referenced + $ref: "#/components/schemas/BoxedString" # # Errors diff --git a/go.mod b/go.mod index d83e255cfb4..5b564cbcdc5 100644 --- a/go.mod +++ b/go.mod @@ -42,7 +42,7 @@ require ( github.com/keygen-sh/jsonapi-go v1.2.1 github.com/keygen-sh/keygen-go/v3 v3.2.0 github.com/kubepug/kubepug v1.7.1 - github.com/kubeshop/testkube-operator v1.17.55-0.20241213144851-e0d37fb5899a + github.com/kubeshop/testkube-operator v1.17.55-0.20241216161809-fd2aba982921 github.com/minio/minio-go/v7 v7.0.66 github.com/montanaflynn/stats v0.7.1 github.com/moogar0880/problems v0.1.1 diff --git a/go.sum b/go.sum index 35bbe8851a3..7de5a55a896 100644 --- a/go.sum +++ b/go.sum @@ -336,8 +336,8 @@ github.com/kr/text v0.2.0 h1:5Nx0Ya0ZqY2ygV366QzturHI13Jq95ApcVaJBhpS+AY= github.com/kr/text v0.2.0/go.mod h1:eLer722TekiGuMkidMxC/pM04lWEeraHUUmBw8l2grE= github.com/kubepug/kubepug v1.7.1 h1:LKhfSxS8Y5mXs50v+3Lpyec+cogErDLcV7CMUuiaisw= github.com/kubepug/kubepug v1.7.1/go.mod h1:lv+HxD0oTFL7ZWjj0u6HKhMbbTIId3eG7aWIW0gyF8g= -github.com/kubeshop/testkube-operator v1.17.55-0.20241213144851-e0d37fb5899a h1:nQDxKFu67lz9BH8isLnP7ttp7cgBWYnRDMV9VxMaYH0= -github.com/kubeshop/testkube-operator v1.17.55-0.20241213144851-e0d37fb5899a/go.mod h1:P47tw1nKQFufdsZndyq2HG2MSa0zK/lU0XpRfZtEmIk= +github.com/kubeshop/testkube-operator v1.17.55-0.20241216161809-fd2aba982921 h1:vvTobFjWYu0N9wyPA89sliFQz2KYf0jhVAsTobUqZfo= +github.com/kubeshop/testkube-operator v1.17.55-0.20241216161809-fd2aba982921/go.mod h1:P47tw1nKQFufdsZndyq2HG2MSa0zK/lU0XpRfZtEmIk= github.com/leodido/go-urn v1.4.0 h1:WT9HwE9SGECu3lg4d/dIA+jxlljEa1/ffXKmRjqdmIQ= github.com/leodido/go-urn v1.4.0/go.mod h1:bvxc+MVxLKB4z00jd1z+Dvzr47oO32F/QSNjSBOlFxI= github.com/lithammer/fuzzysearch v1.1.8 h1:/HIuJnjHuXS8bKaiTMeeDlW2/AyIWk2brx1V8LFgLN4= diff --git a/pkg/api/v1/testkube/model_test_workflow_pvc_config.go b/pkg/api/v1/testkube/model_test_workflow_pvc_config.go index 02b3f00cdb5..a7a03cd1266 100644 --- a/pkg/api/v1/testkube/model_test_workflow_pvc_config.go +++ b/pkg/api/v1/testkube/model_test_workflow_pvc_config.go @@ -16,6 +16,9 @@ type TestWorkflowPvcConfig struct { Resources *TestWorkflowResources `json:"resources,omitempty"` StorageClassName *BoxedString `json:"storageClassName,omitempty"` // Volume name is used to identify the volume - VolumeName string `json:"volumeName,omitempty"` - Selector *LabelSelector `json:"selector,omitempty"` + VolumeName string `json:"volumeName,omitempty"` + Selector *LabelSelector `json:"selector,omitempty"` + DataSource *TypedLocalObjectReference `json:"dataSource,omitempty"` + DataSourceRef *TypedObjectReference `json:"dataSourceRef,omitempty"` + VolumeAttributesClassName *BoxedString `json:"volumeAttributesClassName,omitempty"` } diff --git a/pkg/api/v1/testkube/model_typed_local_object_reference.go b/pkg/api/v1/testkube/model_typed_local_object_reference.go new file mode 100644 index 00000000000..4701fd64b49 --- /dev/null +++ b/pkg/api/v1/testkube/model_typed_local_object_reference.go @@ -0,0 +1,19 @@ +/* + * Testkube API + * + * Testkube provides a Kubernetes-native framework for test definition, execution and results + * + * API version: 1.0.0 + * Contact: testkube@kubeshop.io + * Generated by: Swagger Codegen (https://github.com/swagger-api/swagger-codegen.git) + */ +package testkube + +// TypedLocalObjectReference contains enough information to let you locate the typed referenced object inside the same namespace +type TypedLocalObjectReference struct { + ApiGroup *BoxedString `json:"apiGroup,omitempty"` + // kind is the type of resource being referenced + Kind string `json:"kind,omitempty"` + // name is the name of resource being referenced + Name string `json:"name,omitempty"` +} diff --git a/pkg/api/v1/testkube/model_typed_object_reference.go b/pkg/api/v1/testkube/model_typed_object_reference.go new file mode 100644 index 00000000000..18617f8d0b5 --- /dev/null +++ b/pkg/api/v1/testkube/model_typed_object_reference.go @@ -0,0 +1,20 @@ +/* + * Testkube API + * + * Testkube provides a Kubernetes-native framework for test definition, execution and results + * + * API version: 1.0.0 + * Contact: testkube@kubeshop.io + * Generated by: Swagger Codegen (https://github.com/swagger-api/swagger-codegen.git) + */ +package testkube + +// TypedObjectReference contains enough information to let you locate the typed referenced object inside the specified namespace +type TypedObjectReference struct { + ApiGroup *BoxedString `json:"apiGroup,omitempty"` + // kind is the type of resource being referenced + Kind string `json:"kind,omitempty"` + // name is the name of resource being referenced + Name string `json:"name,omitempty"` + Namespace *BoxedString `json:"namespace,omitempty"` +} diff --git a/pkg/mapper/testworkflows/kube_openapi.go b/pkg/mapper/testworkflows/kube_openapi.go index 333a2c4d121..81dc81d1e52 100644 --- a/pkg/mapper/testworkflows/kube_openapi.go +++ b/pkg/mapper/testworkflows/kube_openapi.go @@ -1213,11 +1213,11 @@ func MapTestWorkflowTagSchemaKubeToAPI(v testworkflowsv1.TestWorkflowTagSchema) } } -func MapPvcConfigKubeToAPI(v testworkflowsv1.TestWorkflowPvcConfig) testkube.TestWorkflowPvcConfig { +func MapPvcConfigKubeToAPI(v corev1.PersistentVolumeClaimSpec) testkube.TestWorkflowPvcConfig { return testkube.TestWorkflowPvcConfig{ - AccessModes: v.AccessModes, - VolumeMode: MapStringToBoxedString(v.VolumeMode), - Resources: common.MapPtr(v.Resources, MapResourcesKubeToAPI), + // AccessModes: v.AccessModes, + // VolumeMode: MapStringToBoxedString(v.VolumeMode), + // Resources: common.MapPtr(v.Resources, MapResourcesKubeToAPI), StorageClassName: MapStringToBoxedString(v.StorageClassName), VolumeName: v.VolumeName, Selector: common.MapPtr(v.Selector, MapSelectorToAPI), diff --git a/pkg/mapper/testworkflows/openapi_kube.go b/pkg/mapper/testworkflows/openapi_kube.go index fa5a76abc01..9841a1b5d5f 100644 --- a/pkg/mapper/testworkflows/openapi_kube.go +++ b/pkg/mapper/testworkflows/openapi_kube.go @@ -1492,11 +1492,11 @@ func MapTestWorkflowTagSchemaAPIToKube(v testkube.TestWorkflowTagSchema) testwor } } -func MapPvcConfigAPIToKube(v testkube.TestWorkflowPvcConfig) testworkflowsv1.TestWorkflowPvcConfig { - return testworkflowsv1.TestWorkflowPvcConfig{ - AccessModes: v.AccessModes, - VolumeMode: MapBoxedStringToString(v.VolumeMode), - Resources: common.MapPtr(v.Resources, MapResourcesAPIToKube), +func MapPvcConfigAPIToKube(v testkube.TestWorkflowPvcConfig) corev1.PersistentVolumeClaimSpec { + return corev1.PersistentVolumeClaimSpec{ + // AccessModes: v.AccessModes, + // VolumeMode: MapBoxedStringToString(v.VolumeMode), + // Resources: common.MapPtr(v.Resources, MapResourcesAPIToKube), StorageClassName: MapBoxedStringToString(v.StorageClassName), VolumeName: v.VolumeName, Selector: common.MapPtr(v.Selector, MapLabelSelectorAPIToKube), diff --git a/pkg/testworkflows/testworkflowprocessor/processor.go b/pkg/testworkflows/testworkflowprocessor/processor.go index 870eed6df6b..3f3b6e948df 100644 --- a/pkg/testworkflows/testworkflowprocessor/processor.go +++ b/pkg/testworkflows/testworkflowprocessor/processor.go @@ -23,7 +23,6 @@ import ( "github.com/kubeshop/testkube/pkg/testworkflows/testworkflowprocessor/action/actiontypes/lite" "github.com/kubeshop/testkube/pkg/testworkflows/testworkflowprocessor/constants" "github.com/kubeshop/testkube/pkg/testworkflows/testworkflowprocessor/stage" - "github.com/kubeshop/testkube/pkg/testworkflows/testworkflowresolver" ) //go:generate mockgen -destination=./mock_processor.go -package=testworkflowprocessor "github.com/kubeshop/testkube/pkg/testworkflows/testworkflowprocessor" Processor @@ -108,7 +107,7 @@ func (p *processor) Bundle(ctx context.Context, workflow *testworkflowsv1.TestWo layer := NewIntermediate(). AppendPodConfig(workflow.Spec.Pod). AppendJobConfig(workflow.Spec.Job). - AppendPvc(common.MapMap(workflow.Spec.Pvcs, testworkflowresolver.ConvertTestWorkflowPvcConfigToPersistentVolumeClaimSpec)) + AppendPvc(workflow.Spec.Pvcs) layer.ContainerDefaults(). ApplyCR(constants.DefaultContainerConfig.DeepCopy()). AppendVolumeMounts(layer.AddEmptyDirVolume(nil, constants.DefaultInternalPath)). diff --git a/pkg/testworkflows/testworkflowresolver/merge.go b/pkg/testworkflows/testworkflowresolver/merge.go index 3ebf623ab09..cdf602ca43b 100644 --- a/pkg/testworkflows/testworkflowresolver/merge.go +++ b/pkg/testworkflows/testworkflowresolver/merge.go @@ -12,8 +12,6 @@ import ( "maps" corev1 "k8s.io/api/core/v1" - "k8s.io/apimachinery/pkg/api/resource" - "k8s.io/apimachinery/pkg/util/intstr" testworkflowsv1 "github.com/kubeshop/testkube-operator/api/testworkflows/v1" "github.com/kubeshop/testkube/internal/common" @@ -417,60 +415,3 @@ func MergeTags(dst, src map[string]string) map[string]string { return dst } - -func ConvertTestWorkflowPvcConfigToPersistentVolumeClaimSpec(pvc testworkflowsv1.TestWorkflowPvcConfig) corev1.PersistentVolumeClaimSpec { - return corev1.PersistentVolumeClaimSpec{ - AccessModes: common.MapSlice(pvc.AccessModes, func(s string) corev1.PersistentVolumeAccessMode { - return corev1.PersistentVolumeAccessMode(s) - }), - Selector: pvc.Selector, - Resources: ConvertResourcesToVolumeResourceRequirements(pvc.Resources), - VolumeName: pvc.VolumeName, - StorageClassName: pvc.StorageClassName, - VolumeMode: (*corev1.PersistentVolumeMode)(pvc.VolumeMode), - } -} - -func ConvertResourcesToVolumeResourceRequirements(r *testworkflowsv1.Resources) corev1.VolumeResourceRequirements { - var limits, requests corev1.ResourceList - if r != nil { - if len(r.Limits) != 0 { - limits = make(corev1.ResourceList) - } - - if len(r.Requests) != 0 { - requests = make(corev1.ResourceList) - } - - for key, value := range r.Limits { - var quantity resource.Quantity - if value.Type == intstr.Int { - quantity = *resource.NewQuantity(int64(value.IntVal), resource.BinarySI) - } - - if value.Type == intstr.String { - quantity, _ = resource.ParseQuantity(value.String()) - } - - limits[key] = quantity - } - - for key, value := range r.Requests { - var quantity resource.Quantity - if value.Type == intstr.Int { - quantity = *resource.NewQuantity(int64(value.IntVal), resource.BinarySI) - } - - if value.Type == intstr.String { - quantity, _ = resource.ParseQuantity(value.String()) - } - - requests[key] = quantity - } - } - - return corev1.VolumeResourceRequirements{ - Limits: limits, - Requests: requests, - } -} From f2dd57bf92971a5019dd3492c23f6827b46e427b Mon Sep 17 00:00:00 2001 From: Vladislav Sukhin Date: Tue, 17 Dec 2024 14:27:09 +0300 Subject: [PATCH 25/36] fix: pvc mapping Signed-off-by: Vladislav Sukhin --- pkg/mapper/testworkflows/kube_openapi.go | 61 +++++++++++++++++++--- pkg/mapper/testworkflows/openapi_kube.go | 64 +++++++++++++++++++++--- 2 files changed, 113 insertions(+), 12 deletions(-) diff --git a/pkg/mapper/testworkflows/kube_openapi.go b/pkg/mapper/testworkflows/kube_openapi.go index 81dc81d1e52..099686ae9f7 100644 --- a/pkg/mapper/testworkflows/kube_openapi.go +++ b/pkg/mapper/testworkflows/kube_openapi.go @@ -1213,13 +1213,62 @@ func MapTestWorkflowTagSchemaKubeToAPI(v testworkflowsv1.TestWorkflowTagSchema) } } +func MapTypeLocalObjectReferenceKubeToAPI(v corev1.TypedLocalObjectReference) testkube.TypedLocalObjectReference { + return testkube.TypedLocalObjectReference{ + ApiGroup: MapStringToBoxedString(v.APIGroup), + Kind: v.Kind, + Name: v.Name, + } +} + +func MapTypeObjectReferenceKubeToAPI(v corev1.TypedObjectReference) testkube.TypedObjectReference { + return testkube.TypedObjectReference{ + ApiGroup: MapStringToBoxedString(v.APIGroup), + Kind: v.Kind, + Name: v.Name, + Namespace: MapStringToBoxedString(v.Namespace), + } +} + +func MapVolumeResourceRequirementsKubeToAPI(v corev1.VolumeResourceRequirements) *testkube.TestWorkflowResources { + return &testkube.TestWorkflowResources{ + Limits: MapResourcesListKubeCoreToAPI(v.Limits), + Requests: MapResourcesListKubeCoreToAPI(v.Requests), + } +} + +func MapResourcesListKubeCoreToAPI(v corev1.ResourceList) *testkube.TestWorkflowResourcesList { + if len(v) == 0 { + return nil + } + + res := &testkube.TestWorkflowResourcesList{} + if q, ok := v[corev1.ResourceCPU]; !ok { + res.Cpu = q.String() + } + if q, ok := v[corev1.ResourceMemory]; !ok { + res.Memory = q.String() + } + if q, ok := v[corev1.ResourceStorage]; !ok { + res.Storage = q.String() + } + if q, ok := v[corev1.ResourceEphemeralStorage]; !ok { + res.EphemeralStorage = q.String() + } + return res +} + func MapPvcConfigKubeToAPI(v corev1.PersistentVolumeClaimSpec) testkube.TestWorkflowPvcConfig { return testkube.TestWorkflowPvcConfig{ - // AccessModes: v.AccessModes, - // VolumeMode: MapStringToBoxedString(v.VolumeMode), - // Resources: common.MapPtr(v.Resources, MapResourcesKubeToAPI), - StorageClassName: MapStringToBoxedString(v.StorageClassName), - VolumeName: v.VolumeName, - Selector: common.MapPtr(v.Selector, MapSelectorToAPI), + AccessModes: common.MapSlice(v.AccessModes, + func(v corev1.PersistentVolumeAccessMode) string { return (string)(v) }), + VolumeMode: MapStringToBoxedString((*string)(v.VolumeMode)), + Resources: MapVolumeResourceRequirementsKubeToAPI(v.Resources), + StorageClassName: MapStringToBoxedString(v.StorageClassName), + VolumeName: v.VolumeName, + Selector: common.MapPtr(v.Selector, MapSelectorToAPI), + DataSource: common.MapPtr(v.DataSource, MapTypeLocalObjectReferenceKubeToAPI), + DataSourceRef: common.MapPtr(v.DataSourceRef, MapTypeObjectReferenceKubeToAPI), + VolumeAttributesClassName: MapStringToBoxedString(v.VolumeAttributesClassName), } } diff --git a/pkg/mapper/testworkflows/openapi_kube.go b/pkg/mapper/testworkflows/openapi_kube.go index 9841a1b5d5f..388548da53b 100644 --- a/pkg/mapper/testworkflows/openapi_kube.go +++ b/pkg/mapper/testworkflows/openapi_kube.go @@ -1492,13 +1492,65 @@ func MapTestWorkflowTagSchemaAPIToKube(v testkube.TestWorkflowTagSchema) testwor } } +func MapTypeLocalObjectReferenceAPIToKube(v testkube.TypedLocalObjectReference) corev1.TypedLocalObjectReference { + return corev1.TypedLocalObjectReference{ + APIGroup: MapBoxedStringToString(v.ApiGroup), + Kind: v.Kind, + Name: v.Name, + } +} + +func MapTypeObjectReferenceAPIToKube(v testkube.TypedObjectReference) corev1.TypedObjectReference { + return corev1.TypedObjectReference{ + APIGroup: MapBoxedStringToString(v.ApiGroup), + Kind: v.Kind, + Name: v.Name, + Namespace: MapBoxedStringToString(v.Namespace), + } +} + +func MapVolumeResourceRequirementsAPIToKube(v *testkube.TestWorkflowResources) corev1.VolumeResourceRequirements { + if v == nil { + return corev1.VolumeResourceRequirements{} + } + + return corev1.VolumeResourceRequirements{ + Limits: MapResourcesListAPIToKubeCore(v.Limits), + Requests: MapResourcesListAPIToKubeCore(v.Requests), + } +} + +func MapResourcesListAPIToKubeCore(v *testkube.TestWorkflowResourcesList) corev1.ResourceList { + if v == nil { + return nil + } + res := make(map[corev1.ResourceName]resource.Quantity) + if v.Cpu != "" { + res[corev1.ResourceCPU], _ = resource.ParseQuantity(v.Cpu) + } + if v.Memory != "" { + res[corev1.ResourceMemory], _ = resource.ParseQuantity(v.Memory) + } + if v.Storage != "" { + res[corev1.ResourceStorage], _ = resource.ParseQuantity(v.Storage) + } + if v.EphemeralStorage != "" { + res[corev1.ResourceEphemeralStorage], _ = resource.ParseQuantity(v.EphemeralStorage) + } + return res +} + func MapPvcConfigAPIToKube(v testkube.TestWorkflowPvcConfig) corev1.PersistentVolumeClaimSpec { return corev1.PersistentVolumeClaimSpec{ - // AccessModes: v.AccessModes, - // VolumeMode: MapBoxedStringToString(v.VolumeMode), - // Resources: common.MapPtr(v.Resources, MapResourcesAPIToKube), - StorageClassName: MapBoxedStringToString(v.StorageClassName), - VolumeName: v.VolumeName, - Selector: common.MapPtr(v.Selector, MapLabelSelectorAPIToKube), + AccessModes: common.MapSlice(v.AccessModes, + func(v string) corev1.PersistentVolumeAccessMode { return (corev1.PersistentVolumeAccessMode)(v) }), + VolumeMode: (*corev1.PersistentVolumeMode)(MapBoxedStringToString(v.VolumeMode)), + Resources: MapVolumeResourceRequirementsAPIToKube(v.Resources), + StorageClassName: MapBoxedStringToString(v.StorageClassName), + VolumeName: v.VolumeName, + Selector: common.MapPtr(v.Selector, MapLabelSelectorAPIToKube), + DataSource: common.MapPtr(v.DataSource, MapTypeLocalObjectReferenceAPIToKube), + DataSourceRef: common.MapPtr(v.DataSourceRef, MapTypeObjectReferenceAPIToKube), + VolumeAttributesClassName: MapBoxedStringToString(v.VolumeAttributesClassName), } } From a6f70f419107527bd09956c5d364d5b4768b75a4 Mon Sep 17 00:00:00 2001 From: Vladislav Sukhin Date: Tue, 17 Dec 2024 21:29:24 +0300 Subject: [PATCH 26/36] fix: json tag Signed-off-by: Vladislav Sukhin --- pkg/testworkflows/testworkflowconfig/config.go | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/pkg/testworkflows/testworkflowconfig/config.go b/pkg/testworkflows/testworkflowconfig/config.go index 670b7538aed..202707579be 100644 --- a/pkg/testworkflows/testworkflowconfig/config.go +++ b/pkg/testworkflows/testworkflowconfig/config.go @@ -22,7 +22,7 @@ type ExecutionConfig struct { OrganizationId string `json:"o,omitempty"` EnvironmentId string `json:"e,omitempty"` ParentIds string `json:"p,omitempty"` - PvcNames []string `json:"P,omitempty"` + PvcNames []string `json:"c,omitempty"` } type WorkflowConfig struct { From bc49ed20f0be92b588b488760355f9be1ad92a10 Mon Sep 17 00:00:00 2001 From: Vladislav Sukhin Date: Wed, 18 Dec 2024 13:00:54 +0300 Subject: [PATCH 27/36] fix: use proper pvc name Signed-off-by: Vladislav Sukhin --- .../testworkflowconfig/config.go | 2 +- .../testworkflowconfig/expressions.go | 11 ++--------- .../testworkflowprocessor/processor.go | 19 ++++++++++++++++--- 3 files changed, 19 insertions(+), 13 deletions(-) diff --git a/pkg/testworkflows/testworkflowconfig/config.go b/pkg/testworkflows/testworkflowconfig/config.go index 202707579be..c3efe42bb9a 100644 --- a/pkg/testworkflows/testworkflowconfig/config.go +++ b/pkg/testworkflows/testworkflowconfig/config.go @@ -22,7 +22,7 @@ type ExecutionConfig struct { OrganizationId string `json:"o,omitempty"` EnvironmentId string `json:"e,omitempty"` ParentIds string `json:"p,omitempty"` - PvcNames []string `json:"c,omitempty"` + PvcNames map[string]string `json:"c,omitempty"` } type WorkflowConfig struct { diff --git a/pkg/testworkflows/testworkflowconfig/expressions.go b/pkg/testworkflows/testworkflowconfig/expressions.go index b611b32a0d3..b5d85df25c0 100644 --- a/pkg/testworkflows/testworkflowconfig/expressions.go +++ b/pkg/testworkflows/testworkflowconfig/expressions.go @@ -1,8 +1,6 @@ package testworkflowconfig import ( - "strings" - "github.com/kubeshop/testkube/pkg/expressions" ) @@ -82,14 +80,9 @@ func CreateWorkerMachine(cfg *WorkerConfig) expressions.Machine { return expressions.CombinedMachines(machine) } -func CreatePvcMachine(pvcNames []string) expressions.Machine { +func CreatePvcMachine(pvcNames map[string]string) expressions.Machine { pvcMap := make(map[string]string) - for _, pvcName := range pvcNames { - name := pvcName - if index := strings.LastIndex(name, "-"); index != -1 { - name = name[:index] - } - + for name, pvcName := range pvcNames { pvcMap[name+".name"] = pvcName } diff --git a/pkg/testworkflows/testworkflowprocessor/processor.go b/pkg/testworkflows/testworkflowprocessor/processor.go index 3f3b6e948df..b3526b5d3ed 100644 --- a/pkg/testworkflows/testworkflowprocessor/processor.go +++ b/pkg/testworkflows/testworkflowprocessor/processor.go @@ -6,6 +6,7 @@ import ( "fmt" "maps" "path/filepath" + "strings" "time" "github.com/pkg/errors" @@ -115,12 +116,18 @@ func (p *processor) Bundle(ctx context.Context, workflow *testworkflowsv1.TestWo AppendVolumeMounts(layer.AddEmptyDirVolume(nil, constants.DefaultDataPath)) mapEnv := make(map[string]corev1.EnvVarSource) + mapPvc := make(map[string]string) + for _, pvc := range layer.Pvcs() { + if index := strings.LastIndex(pvc.Name, "-"); index != -1 { + mapPvc[pvc.Name[:index]] = pvc.Name[index+1:] + } + } + machines = append(machines, createSecretMachine(mapEnv), testworkflowconfig.CreateWorkerMachine(&options.Config.Worker), testworkflowconfig.CreateResourceMachine(&options.Config.Resource), - testworkflowconfig.CreatePvcMachine( - common.MapSlice(layer.Pvcs(), func(p corev1.PersistentVolumeClaim) string { return p.Name })), + testworkflowconfig.CreatePvcMachine(mapPvc), ) // Fetch resource root and resource ID @@ -168,14 +175,20 @@ func (p *processor) Bundle(ctx context.Context, workflow *testworkflowsv1.TestWo // Finalize Pvcs pvcs := layer.Pvcs() + mapPvc = make(map[string]string) for i := range pvcs { AnnotateControlledBy(&pvcs[i], options.Config.Resource.RootId, options.Config.Resource.Id) err = expressions.FinalizeForce(&pvcs[i], machines...) if err != nil { return nil, errors.Wrap(err, "finalizing Pvc") } + + if index := strings.LastIndex(pvcs[i].Name, "-"); index != -1 { + mapPvc[pvcs[i].Name[:index]] = pvcs[i].Name[index+1:] + pvcs[i].Name = pvcs[i].Name[index+1:] + } } - options.Config.Execution.PvcNames = common.MapSlice(pvcs, func(p corev1.PersistentVolumeClaim) string { return p.Name }) + options.Config.Execution.PvcNames = mapPvc // Finalize Secrets secrets := append(layer.Secrets(), options.Secrets...) From ea3e45c983db55f258391f31a13be9a087a8933c Mon Sep 17 00:00:00 2001 From: Vladislav Sukhin Date: Wed, 18 Dec 2024 14:03:21 +0300 Subject: [PATCH 28/36] fix: merge maps Signed-off-by: Vladislav Sukhin --- pkg/testworkflows/testworkflowprocessor/processor.go | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/pkg/testworkflows/testworkflowprocessor/processor.go b/pkg/testworkflows/testworkflowprocessor/processor.go index b3526b5d3ed..0d14c9b3083 100644 --- a/pkg/testworkflows/testworkflowprocessor/processor.go +++ b/pkg/testworkflows/testworkflowprocessor/processor.go @@ -188,7 +188,7 @@ func (p *processor) Bundle(ctx context.Context, workflow *testworkflowsv1.TestWo pvcs[i].Name = pvcs[i].Name[index+1:] } } - options.Config.Execution.PvcNames = mapPvc + options.Config.Execution.PvcNames = common.MergeMaps(options.Config.Execution.PvcNames, mapPvc) // Finalize Secrets secrets := append(layer.Secrets(), options.Secrets...) From 691a5e94b29f9f6301648f7979c3ce4d292a9c3a Mon Sep 17 00:00:00 2001 From: Vladislav Sukhin Date: Wed, 18 Dec 2024 14:25:38 +0300 Subject: [PATCH 29/36] fix: api spec format Signed-off-by: Vladislav Sukhin --- api/v1/testkube.yaml | 5 +++-- 1 file changed, 3 insertions(+), 2 deletions(-) diff --git a/api/v1/testkube.yaml b/api/v1/testkube.yaml index 76f45fb906e..f060a0e820a 100644 --- a/api/v1/testkube.yaml +++ b/api/v1/testkube.yaml @@ -10626,14 +10626,15 @@ components: $ref: "#/components/schemas/LabelSelector" dataSource: description: 'Data source field can be used to specify either: - * An existing VolumeSnapshot object (snapshot.storage.k8s.io/VolumeSnapshot) + * An existing VolumeSnapshot object (snapshot.storage.k8s.io/VolumeSnapshot) * An existing PVC (PersistentVolumeClaim)' $ref: "#/components/schemas/TypedLocalObjectReference" dataSourceRef: description: 'Data source reference specifies the object from which to populate the volume with data, if a non-empty volume is desired' $ref: "#/components/schemas/TypedObjectReference" volumeAttributesClassName: - description: 'Volume attributes class name may be used to set the VolumeAttributesClass used by this claim. More info: https://kubernetes.io/docs/concepts/storage/persistent-volumes#volumeattributesclass' + description: 'Volume attributes class name may be used to set the VolumeAttributesClass used by this claim. + More info: https://kubernetes.io/docs/concepts/storage/persistent-volumes#volumeattributesclass' $ref: "#/components/schemas/BoxedString" TypedLocalObjectReference: From 19fadbf4e213cf615ab335178a82ce2b602ef652 Mon Sep 17 00:00:00 2001 From: Vladislav Sukhin Date: Wed, 18 Dec 2024 15:41:07 +0300 Subject: [PATCH 30/36] fix: typo Signed-off-by: Vladislav Sukhin --- pkg/mapper/testworkflows/kube_openapi.go | 8 ++++---- 1 file changed, 4 insertions(+), 4 deletions(-) diff --git a/pkg/mapper/testworkflows/kube_openapi.go b/pkg/mapper/testworkflows/kube_openapi.go index 099686ae9f7..0b92ea60b97 100644 --- a/pkg/mapper/testworkflows/kube_openapi.go +++ b/pkg/mapper/testworkflows/kube_openapi.go @@ -1243,16 +1243,16 @@ func MapResourcesListKubeCoreToAPI(v corev1.ResourceList) *testkube.TestWorkflow } res := &testkube.TestWorkflowResourcesList{} - if q, ok := v[corev1.ResourceCPU]; !ok { + if q, ok := v[corev1.ResourceCPU]; ok { res.Cpu = q.String() } - if q, ok := v[corev1.ResourceMemory]; !ok { + if q, ok := v[corev1.ResourceMemory]; ok { res.Memory = q.String() } - if q, ok := v[corev1.ResourceStorage]; !ok { + if q, ok := v[corev1.ResourceStorage]; ok { res.Storage = q.String() } - if q, ok := v[corev1.ResourceEphemeralStorage]; !ok { + if q, ok := v[corev1.ResourceEphemeralStorage]; ok { res.EphemeralStorage = q.String() } return res From f242e4575e70de61580c3198d9bd11e76e2a8707 Mon Sep 17 00:00:00 2001 From: Vladislav Sukhin Date: Wed, 18 Dec 2024 16:36:36 +0300 Subject: [PATCH 31/36] fix: format methods Signed-off-by: Vladislav Sukhin --- pkg/mapper/testworkflows/kube_openapi.go | 4 ++++ pkg/mapper/testworkflows/openapi_kube.go | 5 +++++ 2 files changed, 9 insertions(+) diff --git a/pkg/mapper/testworkflows/kube_openapi.go b/pkg/mapper/testworkflows/kube_openapi.go index 0b92ea60b97..cb1ac92b34f 100644 --- a/pkg/mapper/testworkflows/kube_openapi.go +++ b/pkg/mapper/testworkflows/kube_openapi.go @@ -1246,15 +1246,19 @@ func MapResourcesListKubeCoreToAPI(v corev1.ResourceList) *testkube.TestWorkflow if q, ok := v[corev1.ResourceCPU]; ok { res.Cpu = q.String() } + if q, ok := v[corev1.ResourceMemory]; ok { res.Memory = q.String() } + if q, ok := v[corev1.ResourceStorage]; ok { res.Storage = q.String() } + if q, ok := v[corev1.ResourceEphemeralStorage]; ok { res.EphemeralStorage = q.String() } + return res } diff --git a/pkg/mapper/testworkflows/openapi_kube.go b/pkg/mapper/testworkflows/openapi_kube.go index 388548da53b..9523414b2de 100644 --- a/pkg/mapper/testworkflows/openapi_kube.go +++ b/pkg/mapper/testworkflows/openapi_kube.go @@ -1524,19 +1524,24 @@ func MapResourcesListAPIToKubeCore(v *testkube.TestWorkflowResourcesList) corev1 if v == nil { return nil } + res := make(map[corev1.ResourceName]resource.Quantity) if v.Cpu != "" { res[corev1.ResourceCPU], _ = resource.ParseQuantity(v.Cpu) } + if v.Memory != "" { res[corev1.ResourceMemory], _ = resource.ParseQuantity(v.Memory) } + if v.Storage != "" { res[corev1.ResourceStorage], _ = resource.ParseQuantity(v.Storage) } + if v.EphemeralStorage != "" { res[corev1.ResourceEphemeralStorage], _ = resource.ParseQuantity(v.EphemeralStorage) } + return res } From 978e7aab92abb737688c258c9edc6fca7edc7fcc Mon Sep 17 00:00:00 2001 From: Vladislav Sukhin Date: Wed, 18 Dec 2024 17:59:17 +0300 Subject: [PATCH 32/36] fix: use map instead of slice Signed-off-by: Vladislav Sukhin --- .../testworkflowprocessor/intermediate.go | 20 +++++++------ .../testworkflowprocessor/processor.go | 30 ++++++++----------- 2 files changed, 24 insertions(+), 26 deletions(-) diff --git a/pkg/testworkflows/testworkflowprocessor/intermediate.go b/pkg/testworkflows/testworkflowprocessor/intermediate.go index 9be9b86fe09..4bec3dd0f34 100644 --- a/pkg/testworkflows/testworkflowprocessor/intermediate.go +++ b/pkg/testworkflows/testworkflowprocessor/intermediate.go @@ -22,7 +22,7 @@ type Intermediate interface { ConfigMaps() []corev1.ConfigMap Secrets() []corev1.Secret Volumes() []corev1.Volume - Pvcs() []corev1.PersistentVolumeClaim + Pvcs() map[string]corev1.PersistentVolumeClaim AppendJobConfig(cfg *testworkflowsv1.JobConfig) Intermediate AppendPodConfig(cfg *testworkflowsv1.PodConfig) Intermediate @@ -50,9 +50,9 @@ type intermediate struct { Job testworkflowsv1.JobConfig `expr:"include"` // Actual Kubernetes resources to use - Secs []corev1.Secret `expr:"force"` - Cfgs []corev1.ConfigMap `expr:"force"` - Ps []corev1.PersistentVolumeClaim `expr:"force"` + Secs []corev1.Secret `expr:"force"` + Cfgs []corev1.ConfigMap `expr:"force"` + Ps map[string]corev1.PersistentVolumeClaim `expr:"force"` // Storing files Files ConfigMapFiles `expr:"include"` @@ -64,7 +64,9 @@ func NewIntermediate() Intermediate { RefCounter: ref, Root: stage.NewGroupStage("", true), Container: stage.NewContainer(), - Files: NewConfigMapFiles(fmt.Sprintf("{{resource.id}}-%s", ref.NextRef()), nil)} + Files: NewConfigMapFiles(fmt.Sprintf("{{resource.id}}-%s", ref.NextRef()), nil), + Ps: make(map[string]corev1.PersistentVolumeClaim), + } } func (s *intermediate) ContainerDefaults() stage.Container { @@ -91,7 +93,7 @@ func (s *intermediate) Volumes() []corev1.Volume { return append(s.Pod.Volumes, s.Files.Volumes()...) } -func (s *intermediate) Pvcs() []corev1.PersistentVolumeClaim { +func (s *intermediate) Pvcs() map[string]corev1.PersistentVolumeClaim { return s.Ps } @@ -108,12 +110,12 @@ func (s *intermediate) AppendPodConfig(cfg *testworkflowsv1.PodConfig) Intermedi func (s *intermediate) AppendPvc(cfg map[string]corev1.PersistentVolumeClaimSpec) Intermediate { ref := s.NextRef() for name, spec := range cfg { - s.Ps = append(s.Ps, corev1.PersistentVolumeClaim{ + s.Ps[name] = corev1.PersistentVolumeClaim{ ObjectMeta: metav1.ObjectMeta{ - Name: fmt.Sprintf("%s-%s", name, ref), + Name: ref, }, Spec: spec, - }) + } } return s } diff --git a/pkg/testworkflows/testworkflowprocessor/processor.go b/pkg/testworkflows/testworkflowprocessor/processor.go index 0d14c9b3083..6b4639df870 100644 --- a/pkg/testworkflows/testworkflowprocessor/processor.go +++ b/pkg/testworkflows/testworkflowprocessor/processor.go @@ -6,7 +6,6 @@ import ( "fmt" "maps" "path/filepath" - "strings" "time" "github.com/pkg/errors" @@ -116,18 +115,11 @@ func (p *processor) Bundle(ctx context.Context, workflow *testworkflowsv1.TestWo AppendVolumeMounts(layer.AddEmptyDirVolume(nil, constants.DefaultDataPath)) mapEnv := make(map[string]corev1.EnvVarSource) - mapPvc := make(map[string]string) - for _, pvc := range layer.Pvcs() { - if index := strings.LastIndex(pvc.Name, "-"); index != -1 { - mapPvc[pvc.Name[:index]] = pvc.Name[index+1:] - } - } - machines = append(machines, createSecretMachine(mapEnv), testworkflowconfig.CreateWorkerMachine(&options.Config.Worker), testworkflowconfig.CreateResourceMachine(&options.Config.Resource), - testworkflowconfig.CreatePvcMachine(mapPvc), + testworkflowconfig.CreatePvcMachine(common.MapMap(layer.Pvcs(), func(v corev1.PersistentVolumeClaim) string { return v.Name })), ) // Fetch resource root and resource ID @@ -174,19 +166,23 @@ func (p *processor) Bundle(ctx context.Context, workflow *testworkflowsv1.TestWo } // Finalize Pvcs - pvcs := layer.Pvcs() - mapPvc = make(map[string]string) - for i := range pvcs { - AnnotateControlledBy(&pvcs[i], options.Config.Resource.RootId, options.Config.Resource.Id) - err = expressions.FinalizeForce(&pvcs[i], machines...) + pvcs := make([]corev1.PersistentVolumeClaim, 0) + mapPvc := make(map[string]string) + for name, spec := range layer.Pvcs() { + AnnotateControlledBy(&spec, options.Config.Resource.RootId, options.Config.Resource.Id) + err = expressions.FinalizeForce(&spec, machines...) if err != nil { return nil, errors.Wrap(err, "finalizing Pvc") } - if index := strings.LastIndex(pvcs[i].Name, "-"); index != -1 { - mapPvc[pvcs[i].Name[:index]] = pvcs[i].Name[index+1:] - pvcs[i].Name = pvcs[i].Name[index+1:] + data := struct{ value string }{value: name} + err = expressions.FinalizeForce(&data, machines...) + if err != nil { + return nil, errors.Wrap(err, "finalizing name") } + + mapPvc[data.value] = spec.Name + pvcs = append(pvcs, spec) } options.Config.Execution.PvcNames = common.MergeMaps(options.Config.Execution.PvcNames, mapPvc) From 1ae2d0a6df356e66c64737d59b821651d97ee545 Mon Sep 17 00:00:00 2001 From: Vladislav Sukhin Date: Wed, 18 Dec 2024 18:04:24 +0300 Subject: [PATCH 33/36] fix: rename method Signed-off-by: Vladislav Sukhin --- pkg/testworkflows/testworkflowprocessor/intermediate.go | 7 +++---- pkg/testworkflows/testworkflowprocessor/processor.go | 2 +- 2 files changed, 4 insertions(+), 5 deletions(-) diff --git a/pkg/testworkflows/testworkflowprocessor/intermediate.go b/pkg/testworkflows/testworkflowprocessor/intermediate.go index 4bec3dd0f34..8b22b3a11f5 100644 --- a/pkg/testworkflows/testworkflowprocessor/intermediate.go +++ b/pkg/testworkflows/testworkflowprocessor/intermediate.go @@ -26,7 +26,7 @@ type Intermediate interface { AppendJobConfig(cfg *testworkflowsv1.JobConfig) Intermediate AppendPodConfig(cfg *testworkflowsv1.PodConfig) Intermediate - AppendPvc(cfg map[string]corev1.PersistentVolumeClaimSpec) Intermediate + AppendPvcs(cfg map[string]corev1.PersistentVolumeClaimSpec) Intermediate AddConfigMap(configMap corev1.ConfigMap) Intermediate AddSecret(secret corev1.Secret) Intermediate @@ -107,12 +107,11 @@ func (s *intermediate) AppendPodConfig(cfg *testworkflowsv1.PodConfig) Intermedi return s } -func (s *intermediate) AppendPvc(cfg map[string]corev1.PersistentVolumeClaimSpec) Intermediate { - ref := s.NextRef() +func (s *intermediate) AppendPvcs(cfg map[string]corev1.PersistentVolumeClaimSpec) Intermediate { for name, spec := range cfg { s.Ps[name] = corev1.PersistentVolumeClaim{ ObjectMeta: metav1.ObjectMeta{ - Name: ref, + Name: s.NextRef(), }, Spec: spec, } diff --git a/pkg/testworkflows/testworkflowprocessor/processor.go b/pkg/testworkflows/testworkflowprocessor/processor.go index 6b4639df870..049738434eb 100644 --- a/pkg/testworkflows/testworkflowprocessor/processor.go +++ b/pkg/testworkflows/testworkflowprocessor/processor.go @@ -107,7 +107,7 @@ func (p *processor) Bundle(ctx context.Context, workflow *testworkflowsv1.TestWo layer := NewIntermediate(). AppendPodConfig(workflow.Spec.Pod). AppendJobConfig(workflow.Spec.Job). - AppendPvc(workflow.Spec.Pvcs) + AppendPvcs(workflow.Spec.Pvcs) layer.ContainerDefaults(). ApplyCR(constants.DefaultContainerConfig.DeepCopy()). AppendVolumeMounts(layer.AddEmptyDirVolume(nil, constants.DefaultInternalPath)). From 9de43137fa747b5dcf6d1118c689c74b75cdd4ce Mon Sep 17 00:00:00 2001 From: Vladislav Sukhin Date: Wed, 18 Dec 2024 18:50:30 +0300 Subject: [PATCH 34/36] fix: unique ref Signed-off-by: Vladislav Sukhin --- pkg/testworkflows/testworkflowprocessor/intermediate.go | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/pkg/testworkflows/testworkflowprocessor/intermediate.go b/pkg/testworkflows/testworkflowprocessor/intermediate.go index 8b22b3a11f5..c4483fcbb7b 100644 --- a/pkg/testworkflows/testworkflowprocessor/intermediate.go +++ b/pkg/testworkflows/testworkflowprocessor/intermediate.go @@ -111,7 +111,7 @@ func (s *intermediate) AppendPvcs(cfg map[string]corev1.PersistentVolumeClaimSpe for name, spec := range cfg { s.Ps[name] = corev1.PersistentVolumeClaim{ ObjectMeta: metav1.ObjectMeta{ - Name: s.NextRef(), + Name: fmt.Sprintf("{{resource.id}}-%s", s.NextRef()), }, Spec: spec, } From 2ecd4239195db2d22e08ef83de8de477952084bb Mon Sep 17 00:00:00 2001 From: Vladislav Sukhin Date: Wed, 18 Dec 2024 21:53:27 +0300 Subject: [PATCH 35/36] fix: use root id Signed-off-by: Vladislav Sukhin --- pkg/testworkflows/testworkflowprocessor/intermediate.go | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/pkg/testworkflows/testworkflowprocessor/intermediate.go b/pkg/testworkflows/testworkflowprocessor/intermediate.go index c4483fcbb7b..2432bfa98ad 100644 --- a/pkg/testworkflows/testworkflowprocessor/intermediate.go +++ b/pkg/testworkflows/testworkflowprocessor/intermediate.go @@ -111,7 +111,7 @@ func (s *intermediate) AppendPvcs(cfg map[string]corev1.PersistentVolumeClaimSpe for name, spec := range cfg { s.Ps[name] = corev1.PersistentVolumeClaim{ ObjectMeta: metav1.ObjectMeta{ - Name: fmt.Sprintf("{{resource.id}}-%s", s.NextRef()), + Name: fmt.Sprintf("{{resource.root}}-%s", s.NextRef()), }, Spec: spec, } From 6c46caab57c4fdeacbfdb9915da11c6754e75728 Mon Sep 17 00:00:00 2001 From: Vladislav Sukhin Date: Thu, 19 Dec 2024 10:25:02 +0300 Subject: [PATCH 36/36] fix: dep update Signed-off-by: Vladislav Sukhin --- go.mod | 2 +- go.sum | 4 ++-- 2 files changed, 3 insertions(+), 3 deletions(-) diff --git a/go.mod b/go.mod index 5b564cbcdc5..c4e5b664e73 100644 --- a/go.mod +++ b/go.mod @@ -42,7 +42,7 @@ require ( github.com/keygen-sh/jsonapi-go v1.2.1 github.com/keygen-sh/keygen-go/v3 v3.2.0 github.com/kubepug/kubepug v1.7.1 - github.com/kubeshop/testkube-operator v1.17.55-0.20241216161809-fd2aba982921 + github.com/kubeshop/testkube-operator v1.17.55-0.20241219071935-0330347ee160 github.com/minio/minio-go/v7 v7.0.66 github.com/montanaflynn/stats v0.7.1 github.com/moogar0880/problems v0.1.1 diff --git a/go.sum b/go.sum index 7de5a55a896..c05bb6f2fb0 100644 --- a/go.sum +++ b/go.sum @@ -336,8 +336,8 @@ github.com/kr/text v0.2.0 h1:5Nx0Ya0ZqY2ygV366QzturHI13Jq95ApcVaJBhpS+AY= github.com/kr/text v0.2.0/go.mod h1:eLer722TekiGuMkidMxC/pM04lWEeraHUUmBw8l2grE= github.com/kubepug/kubepug v1.7.1 h1:LKhfSxS8Y5mXs50v+3Lpyec+cogErDLcV7CMUuiaisw= github.com/kubepug/kubepug v1.7.1/go.mod h1:lv+HxD0oTFL7ZWjj0u6HKhMbbTIId3eG7aWIW0gyF8g= -github.com/kubeshop/testkube-operator v1.17.55-0.20241216161809-fd2aba982921 h1:vvTobFjWYu0N9wyPA89sliFQz2KYf0jhVAsTobUqZfo= -github.com/kubeshop/testkube-operator v1.17.55-0.20241216161809-fd2aba982921/go.mod h1:P47tw1nKQFufdsZndyq2HG2MSa0zK/lU0XpRfZtEmIk= +github.com/kubeshop/testkube-operator v1.17.55-0.20241219071935-0330347ee160 h1:Bocn/Cgx/XLEsqvU3IdK5U/cFDJDFD3tES/1KS2wLkY= +github.com/kubeshop/testkube-operator v1.17.55-0.20241219071935-0330347ee160/go.mod h1:P47tw1nKQFufdsZndyq2HG2MSa0zK/lU0XpRfZtEmIk= github.com/leodido/go-urn v1.4.0 h1:WT9HwE9SGECu3lg4d/dIA+jxlljEa1/ffXKmRjqdmIQ= github.com/leodido/go-urn v1.4.0/go.mod h1:bvxc+MVxLKB4z00jd1z+Dvzr47oO32F/QSNjSBOlFxI= github.com/lithammer/fuzzysearch v1.1.8 h1:/HIuJnjHuXS8bKaiTMeeDlW2/AyIWk2brx1V8LFgLN4=