diff --git a/theodolite-quarkus/src/main/kotlin/theodolite/k8s/K8sCustomResourceWrapper.kt b/theodolite-quarkus/src/main/kotlin/theodolite/k8s/K8sCustomResourceWrapper.kt new file mode 100644 index 0000000000000000000000000000000000000000..8b3ac08265d5caa75f602b18e279e36efd24e187 --- /dev/null +++ b/theodolite-quarkus/src/main/kotlin/theodolite/k8s/K8sCustomResourceWrapper.kt @@ -0,0 +1,54 @@ +package theodolite.k8s + +import io.fabric8.kubernetes.client.CustomResource +import io.fabric8.kubernetes.client.NamespacedKubernetesClient +import io.fabric8.kubernetes.client.dsl.base.CustomResourceDefinitionContext + +/** +* Fabric8 handles custom resources as plain HashMaps. These need to be handled differently than normal +* Kubernetes resources. The K8sCustomResourceWrapper class provides a wrapper to deploy and delete +* custom resources in a uniform way. +* +* @property map custom resource as plain hashmap +* @constructor Create empty K8s custom resource wrapper +*/ +class K8sCustomResourceWrapper(private val map : Map<String,String>) : CustomResource() { + + + /** + * Deploy a custom resource + * + * @param client a namespaced Kubernetes client which are used to deploy the CR object. + */ + fun deploy(client : NamespacedKubernetesClient){ + val kind = this.map["kind"] + // Search the CustomResourceDefinition to which the CR Object belongs. + // This should be exactly one if the CRD is registered for Kubernetes, zero otherwise. + val crds = client.apiextensions().v1beta1().customResourceDefinitions().list() + crds.items + .filter { crd -> crd.toString().contains("kind=$kind") } + .map { crd -> CustomResourceDefinitionContext.fromCrd(crd) } + .forEach { context -> + client.customResource(context).createOrReplace(client.configuration.namespace,this.map as Map<String, Any>) + } + } + + /** + * Delete a custom resource + * + * @param client a namespaced Kubernetes client which are used to delete the CR object. + */ + fun delete(client : NamespacedKubernetesClient){ + val kind = this.map["kind"] + val metadata = this.map["metadata"] as HashMap<String,String> + val name = metadata["name"] + + val crds = client.apiextensions().v1beta1().customResourceDefinitions().list() + crds.items + .filter { crd -> crd.toString().contains("kind=$kind") } + .map { crd -> CustomResourceDefinitionContext.fromCrd(crd) } + .forEach { context -> + client.customResource(context).delete(client.configuration.namespace,name) } + } + +} \ No newline at end of file diff --git a/theodolite-quarkus/src/main/kotlin/theodolite/k8s/K8sManager.kt b/theodolite-quarkus/src/main/kotlin/theodolite/k8s/K8sManager.kt index 6a4c5827afdb5e0b40120ca72cc165b2310a71e2..38a07545513d7713f61e484702cb48143ab3aed0 100644 --- a/theodolite-quarkus/src/main/kotlin/theodolite/k8s/K8sManager.kt +++ b/theodolite-quarkus/src/main/kotlin/theodolite/k8s/K8sManager.kt @@ -7,6 +7,7 @@ import io.fabric8.kubernetes.api.model.apps.Deployment import io.fabric8.kubernetes.api.model.apps.StatefulSet import io.fabric8.kubernetes.client.NamespacedKubernetesClient + class K8sManager(private val client: NamespacedKubernetesClient) { fun deploy(resource: KubernetesResource) { when (resource) { @@ -18,6 +19,7 @@ class K8sManager(private val client: NamespacedKubernetesClient) { this.client.configMaps().createOrReplace(resource) is StatefulSet -> this.client.apps().statefulSets().createOrReplace(resource) + is K8sCustomResourceWrapper -> resource.deploy(client) else -> throw IllegalArgumentException("Unknown Kubernetes resource.") } } @@ -32,6 +34,7 @@ class K8sManager(private val client: NamespacedKubernetesClient) { this.client.configMaps().delete(resource) is StatefulSet -> this.client.apps().statefulSets().delete(resource) + is K8sCustomResourceWrapper -> resource.delete(client) else -> throw IllegalArgumentException("Unknown Kubernetes resource.") } } diff --git a/theodolite-quarkus/src/main/kotlin/theodolite/k8s/K8sResourceLoader.kt b/theodolite-quarkus/src/main/kotlin/theodolite/k8s/K8sResourceLoader.kt index dbc31de96df4d1a4a48347c5f33853c0c98e1f95..658af46611ec7e1e784d286391ecf1e25a25366e 100644 --- a/theodolite-quarkus/src/main/kotlin/theodolite/k8s/K8sResourceLoader.kt +++ b/theodolite-quarkus/src/main/kotlin/theodolite/k8s/K8sResourceLoader.kt @@ -3,10 +3,10 @@ package theodolite.k8s import io.fabric8.kubernetes.api.model.ConfigMap import io.fabric8.kubernetes.api.model.KubernetesResource import io.fabric8.kubernetes.api.model.Service -import io.fabric8.kubernetes.api.model.apiextensions.v1beta1.CustomResourceDefinition import io.fabric8.kubernetes.api.model.apps.Deployment import io.fabric8.kubernetes.client.NamespacedKubernetesClient import mu.KotlinLogging +import theodolite.util.YamlParser private val logger = KotlinLogging.logger {} @@ -22,12 +22,12 @@ class K8sResourceLoader(private val client: NamespacedKubernetesClient) { } /** - * Parses a Service from a service yaml + * Parses a CustomResource from a yaml * @param path of the yaml file - * @return service from fabric8 + * @return customResource from fabric8 */ - private fun loadServiceMonitor(path: String): CustomResourceDefinition { - return loadGenericResource(path) { x: String -> client.customResourceDefinitions().load(x).get() } + fun loadCustomResource(path: String): K8sCustomResourceWrapper { + return loadGenericResource(path) { x: String -> K8sCustomResourceWrapper(YamlParser().parse(path, HashMap<String, String>()::class.java)!!) } } /** @@ -73,9 +73,17 @@ class K8sResourceLoader(private val client: NamespacedKubernetesClient) { return when (kind) { "Deployment" -> loadDeployment(path) "Service" -> loadService(path) - "ServiceMonitor" -> loadServiceMonitor(path) + "ServiceMonitor" -> loadCustomResource(path) "ConfigMap" -> loadConfigmap(path) - else -> throw IllegalArgumentException("Unknown resource with type $kind located in $path") + else -> { + logger.warn { "Try to load $kind from $path as Custom ressource" } + try{ + loadCustomResource(path) + } catch (e:Exception){ + logger.error { "Error during loading of unspecified CustomResource: $e" } + throw e + } + } } } } diff --git a/theodolite-quarkus/src/main/resources/yaml/BenchmarkExecution.yaml b/theodolite-quarkus/src/main/resources/yaml/BenchmarkExecution.yaml index 94c671008d0910be38ececd8558c689173b991ad..ef74e14f94b65f077fdec7e6f0b61772a7f3079f 100644 --- a/theodolite-quarkus/src/main/resources/yaml/BenchmarkExecution.yaml +++ b/theodolite-quarkus/src/main/resources/yaml/BenchmarkExecution.yaml @@ -1,5 +1,5 @@ name: "Theodolite Test Context" -benchmark: "benchmarkType" +benchmark: "uc1-kstreams" load: loadType: "NumSensors" loadValues: @@ -11,7 +11,7 @@ resources: slos: - sloType: "lag trend" threshold: 1000 - prometheusUrl: "http://localhost:32656" + prometheusUrl: "http://prometheus-operated:9090" externalSloUrl: "http://localhost:80/evaluate-slope" offset: 0 warmup: 0 diff --git a/theodolite-quarkus/src/main/resources/yaml/BenchmarkType.yaml b/theodolite-quarkus/src/main/resources/yaml/BenchmarkType.yaml index 8f6ee0e05efd4cbb8f6a5cb08d6fc048f1a8ee8a..60eb3bb9c31e3eab3e70f916b450372d56db4968 100644 --- a/theodolite-quarkus/src/main/resources/yaml/BenchmarkType.yaml +++ b/theodolite-quarkus/src/main/resources/yaml/BenchmarkType.yaml @@ -1,8 +1,9 @@ -name: "theodolite ist cool" +name: "uc1-kstreams" appResource: - "uc1-kstreams-deployment.yaml" - "aggregation-service.yaml" - "jmx-configmap.yaml" + - "uc1-service-monitor.yaml" loadGenResource: - "uc1-load-generator-deployment.yaml" - "uc1-load-generator-service.yaml" @@ -19,7 +20,7 @@ loadTypes: container: "workload-generator" variableName: "NUM_SENSORS" kafkaConfig: - bootstrapServer: "localhost:31290" + bootstrapServer: "theodolite-cp-kafka:9092" topics: - name: "input" numPartitions: 40 diff --git a/theodolite-quarkus/src/main/resources/yaml/service-monitor.yaml b/theodolite-quarkus/src/main/resources/yaml/uc1-service-monitor.yaml similarity index 100% rename from theodolite-quarkus/src/main/resources/yaml/service-monitor.yaml rename to theodolite-quarkus/src/main/resources/yaml/uc1-service-monitor.yaml