From 13272134528bbd342712f37a79b6ee57c10eefea Mon Sep 17 00:00:00 2001 From: Andre Weber Date: Fri, 15 Mar 2024 09:19:06 +0100 Subject: [PATCH 1/8] test: Introduce DockerDatabrokerContainer --- gradle/libs.versions.toml | 3 + kuksa-sdk/build.gradle.kts | 3 + .../DataBrokerConnectorAuthenticationTest.kt | 41 ++++- .../databroker/DataBrokerConfig.kt | 13 +- .../databroker/DataBrokerConnectionTest.kt | 14 ++ .../databroker/DataBrokerConnectorProvider.kt | 18 ++- .../DataBrokerConnectorSecureTest.kt | 22 ++- .../databroker/DataBrokerConnectorTest.kt | 14 ++ .../databroker/DataBrokerTransporterTest.kt | 14 ++ .../docker/DockerDatabrokerContainer.kt | 142 ++++++++++++++++++ .../DockerInsecureDatabrokerContainer.kt | 43 ++++++ .../docker/DockerSecureDatabrokerContainer.kt | 62 ++++++++ .../subscription/DataBrokerSubscriberTest.kt | 14 ++ .../actuate-provide-all.token | 0 .../test/resources/authentication/jwt.key.pub | 14 ++ .../{ => authentication}/provide-all.token | 0 .../{ => authentication}/read-all.token | 0 kuksa-sdk/src/test/resources/{ => tls}/CA.pem | 0 kuksa-sdk/src/test/resources/tls/Server.key | 27 ++++ kuksa-sdk/src/test/resources/tls/Server.pem | 28 ++++ .../databroker/DebugDataBrokerConfig.kt | 23 +++ .../databroker/ReleaseDataBrokerConfig.kt | 22 +++ 22 files changed, 490 insertions(+), 27 deletions(-) create mode 100644 kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/databroker/docker/DockerDatabrokerContainer.kt create mode 100644 kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/databroker/docker/DockerInsecureDatabrokerContainer.kt create mode 100644 kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/databroker/docker/DockerSecureDatabrokerContainer.kt rename kuksa-sdk/src/test/resources/{ => authentication}/actuate-provide-all.token (100%) create mode 100644 kuksa-sdk/src/test/resources/authentication/jwt.key.pub rename kuksa-sdk/src/test/resources/{ => authentication}/provide-all.token (100%) rename kuksa-sdk/src/test/resources/{ => authentication}/read-all.token (100%) rename kuksa-sdk/src/test/resources/{ => tls}/CA.pem (100%) create mode 100644 kuksa-sdk/src/test/resources/tls/Server.key create mode 100644 kuksa-sdk/src/test/resources/tls/Server.pem create mode 100644 kuksa-sdk/src/testDebug/kotlin/org/eclipse/kuksa/connectivity/databroker/DebugDataBrokerConfig.kt create mode 100644 kuksa-sdk/src/testRelease/kotlin/org/eclipse/kuksa/connectivity/databroker/ReleaseDataBrokerConfig.kt diff --git a/gradle/libs.versions.toml b/gradle/libs.versions.toml index eb16ec18..dc73e24b 100644 --- a/gradle/libs.versions.toml +++ b/gradle/libs.versions.toml @@ -5,6 +5,7 @@ detekt = "1.23.5" datastore = "1.0.0" constraintlayoutCompose = "1.0.1" datastorePreferences = "1.0.0" +dockerJavaCore = "3.3.6" dokka = "1.9.10" gson = "2.10.1" kotlin = "1.9.22" @@ -34,6 +35,8 @@ androidx-datastore = { module = "androidx.datastore:datastore", version.ref = "d androidx-datastore-preferences = { module = "androidx.datastore:datastore-preferences", version.ref = "datastorePreferences" } androidx-lifecycle-runtime-compose = { module = "androidx.lifecycle:lifecycle-runtime-compose", version.ref = "androidxLifecycle" } androidx-runtime-livedata = { module = "androidx.compose.runtime:runtime-livedata", version.ref = "runtimeLivedata" } +docker-java-core = { module = "com.github.docker-java:docker-java-core", version.ref = "dockerJavaCore" } +docker-java-transport-httpclient5 = { module = "com.github.docker-java:docker-java-transport-httpclient5", version.ref = "dockerJavaCore" } grpc-okhttp = { group = "io.grpc", name = "grpc-okhttp", version.ref = "grpc" } grpc-protobuf = { group = "io.grpc", name = "grpc-protobuf-lite", version.ref = "grpc" } grpc-stub = { group = "io.grpc", name = "grpc-stub", version.ref = "grpc" } diff --git a/kuksa-sdk/build.gradle.kts b/kuksa-sdk/build.gradle.kts index be14acbb..0289305a 100644 --- a/kuksa-sdk/build.gradle.kts +++ b/kuksa-sdk/build.gradle.kts @@ -94,6 +94,9 @@ dependencies { testImplementation(libs.kotlinx.coroutines.test) testImplementation(libs.kotest) testImplementation(libs.mockk) + + testImplementation(libs.docker.java.core) + testImplementation(libs.docker.java.transport.httpclient5) } publish { diff --git a/kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/authentication/DataBrokerConnectorAuthenticationTest.kt b/kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/authentication/DataBrokerConnectorAuthenticationTest.kt index cfa5e847..b8c1558a 100644 --- a/kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/authentication/DataBrokerConnectorAuthenticationTest.kt +++ b/kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/authentication/DataBrokerConnectorAuthenticationTest.kt @@ -1,5 +1,5 @@ /* - * Copyright (c) 2023 Contributors to the Eclipse Foundation + * Copyright (c) 2023 - 2024 Contributors to the Eclipse Foundation * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -14,6 +14,7 @@ * limitations under the License. * * SPDX-License-Identifier: Apache-2.0 + * */ package org.eclipse.kuksa.connectivity.authentication @@ -21,6 +22,8 @@ package org.eclipse.kuksa.connectivity.authentication import io.kotest.core.spec.style.BehaviorSpec import io.kotest.matchers.shouldBe import org.eclipse.kuksa.connectivity.databroker.DataBrokerConnectorProvider +import org.eclipse.kuksa.connectivity.databroker.docker.DockerDatabrokerContainer +import org.eclipse.kuksa.connectivity.databroker.docker.DockerSecureDatabrokerContainer import org.eclipse.kuksa.connectivity.databroker.request.FetchRequest import org.eclipse.kuksa.connectivity.databroker.request.UpdateRequest import org.eclipse.kuksa.proto.v1.Types @@ -29,7 +32,6 @@ import org.eclipse.kuksa.test.kotest.Authentication import org.eclipse.kuksa.test.kotest.CustomDatabroker import org.eclipse.kuksa.test.kotest.Insecure import org.eclipse.kuksa.test.kotest.Integration -import java.io.InputStream import kotlin.random.Random import kotlin.random.nextInt @@ -40,15 +42,32 @@ import kotlin.random.nextInt class DataBrokerConnectorAuthenticationTest : BehaviorSpec({ tags(Integration, Authentication, Insecure, CustomDatabroker) + var databrokerContainer: DockerDatabrokerContainer? = null + beforeSpec { + databrokerContainer = DockerSecureDatabrokerContainer() + .apply { + start() + } + } + + afterSpec { + databrokerContainer?.stop() + } + val random = Random(System.nanoTime()) + val tlsCertificate = TestResourceFile("tls/CA.pem") given("A DataBrokerConnectorProvider") { val dataBrokerConnectorProvider = DataBrokerConnectorProvider() val speedVssPath = "Vehicle.Speed" and("an insecure DataBrokerConnector with a READ_WRITE_ALL JWT") { - val jwtFileStream = JwtType.READ_WRITE_ALL.asInputStream() - val dataBrokerConnector = dataBrokerConnectorProvider.createInsecure(jwtFileStream = jwtFileStream) + val jwtFile = JwtType.READ_WRITE_ALL + + val dataBrokerConnector = dataBrokerConnectorProvider.createSecure( + rootCertFileStream = tlsCertificate.inputStream(), + jwtFileStream = jwtFile.asInputStream(), + ) and("a successfully established connection") { val connection = dataBrokerConnector.connect() @@ -77,8 +96,11 @@ class DataBrokerConnectorAuthenticationTest : BehaviorSpec({ } and("an insecure DataBrokerConnector with a READ_ALL JWT") { - val jwtFileStream = JwtType.READ_ALL.asInputStream() - val dataBrokerConnector = dataBrokerConnectorProvider.createInsecure(jwtFileStream = jwtFileStream) + val jwtFile = JwtType.READ_ALL + val dataBrokerConnector = dataBrokerConnectorProvider.createSecure( + rootCertFileStream = tlsCertificate.inputStream(), + jwtFileStream = jwtFile.asInputStream(), + ) and("a successfully established connection") { val connection = dataBrokerConnector.connect() @@ -106,8 +128,11 @@ class DataBrokerConnectorAuthenticationTest : BehaviorSpec({ } and("an insecure DataBrokerConnector with a READ_WRITE_ALL_VALUES_ONLY JWT") { - val jwtFileStream = JwtType.READ_WRITE_ALL_VALUES_ONLY.asInputStream() - val dataBrokerConnector = dataBrokerConnectorProvider.createInsecure(jwtFileStream = jwtFileStream) + val jwtFile = JwtType.READ_WRITE_ALL_VALUES_ONLY + val dataBrokerConnector = dataBrokerConnectorProvider.createSecure( + rootCertFileStream = tlsCertificate.inputStream(), + jwtFileStream = jwtFile.asInputStream(), + ) and("a successfully established connection") { val connection = dataBrokerConnector.connect() diff --git a/kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/databroker/DataBrokerConfig.kt b/kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/databroker/DataBrokerConfig.kt index 4c754649..eafe5055 100644 --- a/kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/databroker/DataBrokerConfig.kt +++ b/kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/databroker/DataBrokerConfig.kt @@ -1,5 +1,5 @@ /* - * Copyright (c) 2023 Contributors to the Eclipse Foundation + * Copyright (c) 2023 - 2024 Contributors to the Eclipse Foundation * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. @@ -14,17 +14,14 @@ * limitations under the License. * * SPDX-License-Identifier: Apache-2.0 + * */ package org.eclipse.kuksa.connectivity.databroker import java.util.concurrent.TimeUnit -object DataBrokerConfig { - const val HOST = "127.0.0.1" - const val PORT = 55556 +const val DATABROKER_HOST = "127.0.0.1" - // low timeout should be okay, since we are testing against a local service - const val TIMEOUT_SECONDS = 3L - val TIMEOUT_UNIT = TimeUnit.SECONDS -} +const val DATABROKER_TIMEOUT_SECONDS = 5L +val DATABROKER_TIMEOUT_UNIT = TimeUnit.SECONDS diff --git a/kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/databroker/DataBrokerConnectionTest.kt b/kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/databroker/DataBrokerConnectionTest.kt index 3a992aa6..e85d78f3 100644 --- a/kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/databroker/DataBrokerConnectionTest.kt +++ b/kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/databroker/DataBrokerConnectionTest.kt @@ -30,6 +30,8 @@ import io.mockk.mockk import io.mockk.slot import io.mockk.verify import kotlinx.coroutines.runBlocking +import org.eclipse.kuksa.connectivity.databroker.docker.DockerDatabrokerContainer +import org.eclipse.kuksa.connectivity.databroker.docker.DockerInsecureDatabrokerContainer import org.eclipse.kuksa.connectivity.databroker.listener.DisconnectListener import org.eclipse.kuksa.connectivity.databroker.listener.VssPathListener import org.eclipse.kuksa.connectivity.databroker.request.FetchRequest @@ -50,6 +52,18 @@ import kotlin.time.Duration.Companion.seconds class DataBrokerConnectionTest : BehaviorSpec({ tags(Integration, DefaultDatabroker) + var databrokerContainer: DockerDatabrokerContainer? = null + beforeSpec { + databrokerContainer = DockerInsecureDatabrokerContainer() + .apply { + start() + } + } + + afterSpec { + databrokerContainer?.stop() + } + given("A successfully established connection to the DataBroker") { val dataBrokerConnection = connectToDataBrokerBlocking() diff --git a/kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/databroker/DataBrokerConnectorProvider.kt b/kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/databroker/DataBrokerConnectorProvider.kt index 53df36bb..e7c39ad4 100644 --- a/kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/databroker/DataBrokerConnectorProvider.kt +++ b/kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/databroker/DataBrokerConnectorProvider.kt @@ -24,15 +24,17 @@ import io.grpc.ManagedChannel import io.grpc.ManagedChannelBuilder import io.grpc.TlsChannelCredentials import org.eclipse.kuksa.connectivity.authentication.JsonWebToken +import org.eclipse.kuksa.connectivity.authentication.JwtType import org.eclipse.kuksa.model.TimeoutConfig +import org.eclipse.kuksa.test.TestResourceFile import java.io.IOException import java.io.InputStream class DataBrokerConnectorProvider { lateinit var managedChannel: ManagedChannel fun createInsecure( - host: String = DataBrokerConfig.HOST, - port: Int = DataBrokerConfig.PORT, + host: String = DATABROKER_HOST, + port: Int = DATABROKER_PORT, jwtFileStream: InputStream? = null, ): DataBrokerConnector { managedChannel = ManagedChannelBuilder.forAddress(host, port).usePlaintext().build() @@ -46,16 +48,16 @@ class DataBrokerConnectorProvider { managedChannel, jsonWebToken, ).apply { - timeoutConfig = TimeoutConfig(DataBrokerConfig.TIMEOUT_SECONDS, DataBrokerConfig.TIMEOUT_UNIT) + timeoutConfig = TimeoutConfig(DATABROKER_TIMEOUT_SECONDS, DATABROKER_TIMEOUT_UNIT) } } fun createSecure( - host: String = DataBrokerConfig.HOST, - port: Int = DataBrokerConfig.PORT, - rootCertFileStream: InputStream, + host: String = DATABROKER_HOST, + port: Int = DATABROKER_PORT, overrideAuthority: String = "", - jwtFileStream: InputStream? = null, + rootCertFileStream: InputStream = TestResourceFile("tls/CA.pem").inputStream(), + jwtFileStream: InputStream? = JwtType.READ_WRITE_ALL.asInputStream(), ): DataBrokerConnector { val tlsCredentials: ChannelCredentials try { @@ -86,7 +88,7 @@ class DataBrokerConnectorProvider { managedChannel, jsonWebToken, ).apply { - timeoutConfig = TimeoutConfig(DataBrokerConfig.TIMEOUT_SECONDS, DataBrokerConfig.TIMEOUT_UNIT) + timeoutConfig = TimeoutConfig(DATABROKER_TIMEOUT_SECONDS, DATABROKER_TIMEOUT_UNIT) } } } diff --git a/kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/databroker/DataBrokerConnectorSecureTest.kt b/kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/databroker/DataBrokerConnectorSecureTest.kt index 0ff03f1f..3995c046 100644 --- a/kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/databroker/DataBrokerConnectorSecureTest.kt +++ b/kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/databroker/DataBrokerConnectorSecureTest.kt @@ -19,6 +19,8 @@ package org.eclipse.kuksa.connectivity.databroker import io.kotest.core.spec.style.BehaviorSpec +import org.eclipse.kuksa.connectivity.databroker.docker.DockerDatabrokerContainer +import org.eclipse.kuksa.connectivity.databroker.docker.DockerSecureDatabrokerContainer import org.eclipse.kuksa.test.TestResourceFile import org.eclipse.kuksa.test.kotest.CustomDatabroker import org.eclipse.kuksa.test.kotest.Integration @@ -29,13 +31,27 @@ import org.junit.jupiter.api.Assertions class DataBrokerConnectorSecureTest : BehaviorSpec({ tags(Integration, Secure, CustomDatabroker) + var databrokerContainer: DockerDatabrokerContainer? = null + beforeSpec { + databrokerContainer = DockerSecureDatabrokerContainer() + .apply { + start() + } + } + + afterSpec { + databrokerContainer?.stop() + } + given("A DataBrokerConnectorProvider") { val dataBrokerConnectorProvider = DataBrokerConnectorProvider() and("a secure DataBrokerConnector with valid Host, Port and TLS certificate") { - val certificate = TestResourceFile("CA.pem") - val dataBrokerConnector = - dataBrokerConnectorProvider.createSecure(rootCertFileStream = certificate.inputStream()) + val tlsCertificate = TestResourceFile("tls/CA.pem") + + val dataBrokerConnector = dataBrokerConnectorProvider.createSecure( + rootCertFileStream = tlsCertificate.inputStream(), + ) `when`("Trying to establish a secure connection") { val connection = dataBrokerConnector.connect() diff --git a/kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/databroker/DataBrokerConnectorTest.kt b/kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/databroker/DataBrokerConnectorTest.kt index 63e30827..b9eb5d73 100644 --- a/kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/databroker/DataBrokerConnectorTest.kt +++ b/kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/databroker/DataBrokerConnectorTest.kt @@ -22,12 +22,26 @@ import io.kotest.assertions.throwables.shouldThrow import io.kotest.core.spec.style.BehaviorSpec import io.kotest.matchers.shouldNotBe import org.eclipse.kuksa.test.kotest.DefaultDatabroker +import org.eclipse.kuksa.connectivity.databroker.docker.DockerDatabrokerContainer +import org.eclipse.kuksa.connectivity.databroker.docker.DockerInsecureDatabrokerContainer import org.eclipse.kuksa.test.kotest.Insecure import org.eclipse.kuksa.test.kotest.Integration class DataBrokerConnectorTest : BehaviorSpec({ tags(Integration, Insecure, DefaultDatabroker) + var databrokerContainer: DockerDatabrokerContainer? = null + beforeSpec { + databrokerContainer = DockerInsecureDatabrokerContainer() + .apply { + start() + } + } + + afterSpec { + databrokerContainer?.stop() + } + given("A DataBrokerConnectorProvider") { val dataBrokerConnectorProvider = DataBrokerConnectorProvider() diff --git a/kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/databroker/DataBrokerTransporterTest.kt b/kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/databroker/DataBrokerTransporterTest.kt index 21b6cdeb..6a053967 100644 --- a/kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/databroker/DataBrokerTransporterTest.kt +++ b/kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/databroker/DataBrokerTransporterTest.kt @@ -25,6 +25,8 @@ import io.kotest.matchers.shouldNotBe import io.kotest.matchers.types.instanceOf import io.mockk.mockk import io.mockk.verify +import org.eclipse.kuksa.connectivity.databroker.docker.DockerDatabrokerContainer +import org.eclipse.kuksa.connectivity.databroker.docker.DockerInsecureDatabrokerContainer import org.eclipse.kuksa.connectivity.databroker.listener.VssPathListener import org.eclipse.kuksa.extensions.updateRandomFloatValue import org.eclipse.kuksa.proto.v1.KuksaValV1 @@ -37,6 +39,18 @@ import kotlin.random.Random class DataBrokerTransporterTest : BehaviorSpec({ tags(Integration, Insecure, DefaultDatabroker) + var databrokerContainer: DockerDatabrokerContainer? = null + beforeSpec { + databrokerContainer = DockerInsecureDatabrokerContainer() + .apply { + start() + } + } + + afterSpec { + databrokerContainer?.stop() + } + given("An active Connection to the DataBroker") { val dataBrokerConnectorProvider = DataBrokerConnectorProvider() val connector = dataBrokerConnectorProvider.createInsecure() diff --git a/kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/databroker/docker/DockerDatabrokerContainer.kt b/kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/databroker/docker/DockerDatabrokerContainer.kt new file mode 100644 index 00000000..6f3a911a --- /dev/null +++ b/kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/databroker/docker/DockerDatabrokerContainer.kt @@ -0,0 +1,142 @@ +/* + * Copyright (c) 2023 - 2024 Contributors to the Eclipse Foundation + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + * + * SPDX-License-Identifier: Apache-2.0 + * + */ + +package org.eclipse.kuksa.connectivity.databroker.docker + +import com.github.dockerjava.api.DockerClient +import com.github.dockerjava.api.command.CreateContainerResponse +import com.github.dockerjava.api.command.PullImageResultCallback +import com.github.dockerjava.api.command.WaitContainerResultCallback +import com.github.dockerjava.api.exception.NotModifiedException +import com.github.dockerjava.api.model.ExposedPort +import com.github.dockerjava.api.model.HostConfig +import com.github.dockerjava.api.model.InternetProtocol +import com.github.dockerjava.api.model.PortBinding +import com.github.dockerjava.api.model.Ports +import com.github.dockerjava.core.DefaultDockerClientConfig +import com.github.dockerjava.core.DockerClientImpl +import com.github.dockerjava.httpclient5.ApacheDockerHttpClient +import org.eclipse.kuksa.connectivity.databroker.DATABROKER_TIMEOUT_SECONDS +import java.time.Duration +import java.util.concurrent.TimeUnit + +private const val KEY_PROPERTY_DATABROKER_TAG = "databroker.tag" +private const val KEY_ENV_DATABROKER_TAG = "DATABROKER_TAG" +private const val DEFAULT_DATABROKER_TAG = "master" + +/** + * Starts and stops the Databroker Docker Container. Per default the image with the master tag is pulled and started. + * The version of the image can be influenced by either a System Property with value "databroker.tag" or an + * Environment Variable with value "DATABROKER_TAG". + *``` + * Samples + * + * System Property: + * ./gradlew clean build -Ddatabroker.tag="0.4.1" + * + * + * Environment Variable: + * export DATABROKER_TAG="0.4.1" + * ./gradlew clean build + * + * or + * + * DATABROKER_TAG="0.4.1" ./gradlew clean build + * ``` + */ +abstract class DockerDatabrokerContainer( + protected val containerName: String, + protected val port: Int, +) { + protected open val hostConfig: HostConfig = HostConfig.newHostConfig() + .withNetworkMode("host") + .withAutoRemove(true) + .withPortBindings( + PortBinding( + Ports.Binding("0.0.0.0", ""), + ExposedPort(port, InternetProtocol.TCP), + ), + ) + + protected val repository = "ghcr.io/eclipse/kuksa.val/databroker" + + private val databrokerTag = System.getProperty(KEY_PROPERTY_DATABROKER_TAG) + ?: System.getenv(KEY_ENV_DATABROKER_TAG) + ?: DEFAULT_DATABROKER_TAG + + protected val dockerClient: DockerClient + + init { + val config = DefaultDockerClientConfig.createDefaultConfigBuilder().build() + + val timeout = Duration.ofSeconds(DATABROKER_TIMEOUT_SECONDS) + val dockerHttpClient = ApacheDockerHttpClient.Builder() + .dockerHost(config.dockerHost) + .sslConfig(config.sslConfig) + .connectionTimeout(timeout) + .responseTimeout(timeout) + .build() + + dockerClient = DockerClientImpl.getInstance(config, dockerHttpClient) + } + + fun start() { + stop() + + pullDatabrokerImage(databrokerTag) + val databrokerContainer = createDatabrokerContainer(databrokerTag) + startContainer(databrokerContainer.id) + } + + fun stop() { + val databrokerNames = listOf(containerName) + val dockerContainers = dockerClient.listContainersCmd() + .withNameFilter(databrokerNames) + .exec() + + dockerContainers.forEach { container -> + try { + dockerClient.stopContainerCmd(container.id).exec() + } catch (_: NotModifiedException) { + // thrown when a container is already stopped + } + } + } + + private fun pullDatabrokerImage(tag: String) { + dockerClient.pullImageCmd(repository) + .withTag(tag) + .exec(PullImageResultCallback()) + .awaitCompletion(30, TimeUnit.SECONDS) + } + + protected abstract fun createDatabrokerContainer(tag: String): CreateContainerResponse + + private fun startContainer(containerId: String) { + try { + dockerClient.startContainerCmd(containerId).exec() + + dockerClient.waitContainerCmd(containerId) + .exec(WaitContainerResultCallback()) + .awaitCompletion(5, TimeUnit.SECONDS) + } catch (_: NotModifiedException) { + // thrown when a container is already started + } + } +} diff --git a/kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/databroker/docker/DockerInsecureDatabrokerContainer.kt b/kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/databroker/docker/DockerInsecureDatabrokerContainer.kt new file mode 100644 index 00000000..a9f9a5e1 --- /dev/null +++ b/kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/databroker/docker/DockerInsecureDatabrokerContainer.kt @@ -0,0 +1,43 @@ +/* + * Copyright (c) 2023 - 2024 Contributors to the Eclipse Foundation + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + * + * SPDX-License-Identifier: Apache-2.0 + * + */ + +package org.eclipse.kuksa.connectivity.databroker.docker + +import com.github.dockerjava.api.command.CreateContainerResponse +import org.eclipse.kuksa.connectivity.databroker.DATABROKER_CONTAINER_NAME +import org.eclipse.kuksa.connectivity.databroker.DATABROKER_PORT + +// no tls, no authentication +class DockerInsecureDatabrokerContainer( + containerName: String = DATABROKER_CONTAINER_NAME, + port: Int = DATABROKER_PORT, +) : DockerDatabrokerContainer(containerName, port) { + + @Suppress("ArgumentListWrapping", "ktlint:standard:argument-list-wrapping") // better key-value pair readability + override fun createDatabrokerContainer(tag: String): CreateContainerResponse { + return dockerClient.createContainerCmd("$repository:$tag") + .withName(containerName) + .withHostConfig(hostConfig) + .withCmd( + "--port", "$port", + "--insecure", + ) + .exec() + } +} diff --git a/kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/databroker/docker/DockerSecureDatabrokerContainer.kt b/kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/databroker/docker/DockerSecureDatabrokerContainer.kt new file mode 100644 index 00000000..634c749c --- /dev/null +++ b/kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/databroker/docker/DockerSecureDatabrokerContainer.kt @@ -0,0 +1,62 @@ +/* + * Copyright (c) 2023 - 2024 Contributors to the Eclipse Foundation + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + * + * SPDX-License-Identifier: Apache-2.0 + * + */ + +package org.eclipse.kuksa.connectivity.databroker.docker + +import com.github.dockerjava.api.command.CreateContainerResponse +import com.github.dockerjava.api.model.AccessMode +import com.github.dockerjava.api.model.Bind +import com.github.dockerjava.api.model.HostConfig +import com.github.dockerjava.api.model.Volume +import org.eclipse.kuksa.connectivity.databroker.DATABROKER_CONTAINER_NAME +import org.eclipse.kuksa.connectivity.databroker.DATABROKER_PORT +import org.eclipse.kuksa.test.TestResourceFile + +// tls enabled, authentication enabled +class DockerSecureDatabrokerContainer( + containerName: String = DATABROKER_CONTAINER_NAME, + port: Int = DATABROKER_PORT, +) : DockerDatabrokerContainer(containerName, port) { + + private val authenticationFolder = TestResourceFile("authentication").toString() + private val authenticationMount = "/resources/authentication" + + private val tlsFolder = TestResourceFile("tls").toString() + private val tlsMount = "/resources/tls" + + override val hostConfig: HostConfig = super.hostConfig + .withBinds( + Bind(tlsFolder, Volume(tlsMount), AccessMode.ro), + Bind(authenticationFolder, Volume(authenticationMount), AccessMode.ro), + ) + + @Suppress("ArgumentListWrapping", "ktlint:standard:argument-list-wrapping") // better key-value pair readability + override fun createDatabrokerContainer(tag: String): CreateContainerResponse { + return dockerClient.createContainerCmd("$repository:$tag") + .withName(containerName) + .withHostConfig(hostConfig) + .withCmd( + "--port", "$port", + "--tls-cert", "$tlsMount/Server.pem", + "--tls-private-key", "$tlsMount/Server.key", + "--jwt-public-key", "$authenticationMount/jwt.key.pub", + ) + .exec() + } +} diff --git a/kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/databroker/subscription/DataBrokerSubscriberTest.kt b/kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/databroker/subscription/DataBrokerSubscriberTest.kt index 3c174221..3fb4f93b 100644 --- a/kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/databroker/subscription/DataBrokerSubscriberTest.kt +++ b/kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/databroker/subscription/DataBrokerSubscriberTest.kt @@ -29,6 +29,8 @@ import io.mockk.verify import kotlinx.coroutines.delay import org.eclipse.kuksa.connectivity.databroker.DataBrokerConnectorProvider import org.eclipse.kuksa.connectivity.databroker.DataBrokerTransporter +import org.eclipse.kuksa.connectivity.databroker.docker.DockerDatabrokerContainer +import org.eclipse.kuksa.connectivity.databroker.docker.DockerInsecureDatabrokerContainer import org.eclipse.kuksa.connectivity.databroker.listener.VssPathListener import org.eclipse.kuksa.extensions.toggleBoolean import org.eclipse.kuksa.extensions.updateRandomFloatValue @@ -48,6 +50,18 @@ import kotlin.time.Duration.Companion.seconds class DataBrokerSubscriberTest : BehaviorSpec({ tags(Integration, Insecure, DefaultDatabroker) + var databrokerContainer: DockerDatabrokerContainer? = null + beforeSpec { + databrokerContainer = DockerInsecureDatabrokerContainer() + .apply { + start() + } + } + + afterSpec { + databrokerContainer?.stop() + } + given("An active Connection to the DataBroker") { val dataBrokerConnectorProvider = DataBrokerConnectorProvider() val connector = dataBrokerConnectorProvider.createInsecure() diff --git a/kuksa-sdk/src/test/resources/actuate-provide-all.token b/kuksa-sdk/src/test/resources/authentication/actuate-provide-all.token similarity index 100% rename from kuksa-sdk/src/test/resources/actuate-provide-all.token rename to kuksa-sdk/src/test/resources/authentication/actuate-provide-all.token diff --git a/kuksa-sdk/src/test/resources/authentication/jwt.key.pub b/kuksa-sdk/src/test/resources/authentication/jwt.key.pub new file mode 100644 index 00000000..d9f78534 --- /dev/null +++ b/kuksa-sdk/src/test/resources/authentication/jwt.key.pub @@ -0,0 +1,14 @@ +-----BEGIN PUBLIC KEY----- +MIICIjANBgkqhkiG9w0BAQEFAAOCAg8AMIICCgKCAgEA6ScE9EKXEWVyYhzfhfvg ++LC8NseiuEjfrdFx3HKkb31bRw/SeS0Rye0KDP7uzffwreKf6wWYGxVUPYmyKC7j +Pji5MpDBGM9r3pIZSvPUFdpTE5TiRHFBxWbqPSYt954BTLq4rMu/W+oq5Pdfnugb +voYpLf0dclBl1g9KyszkDnItz3TYbWhGMbsUSfyeSPzH0IADzLoifxbc5mgiR73N +CA/4yNSpfLoqWgQ2vdTM1182sMSmxfqSgMzIMUX/tiaXGdkoKITF1sULlLyWfTo9 +79XRZ0hmUwvfzr3OjMZNoClpYSVbKY+vtxHyux9KOOtv9lPMsgYIaPXvisrsneDZ +fCS0afOfjgR96uHIe2UPSGAXru3yGziqEfpRZoxsgXaOe905ordLD5bSX14xkN7N +Cz7rxDLlxPQyxp4Vhog7p/QeUyydBpZjq2bAE5GAJtiu+XGvG8RypzJFKFQwMNsw +g1BoZVD0mb0MtU8KQmHcZIfY0FVer/CR0mUjfl1rHbtoJB+RY03lQvYNAD04ibAG +NI1RhlTziu35Xo6NDEgs9hVs9k3WrtF+ZUxhivWmP2VXhWruRakVkC1NzKGh54e5 +/KlluFbBNpWgvWZqzWo9Jr7/fzHtR0Q0IZwkxh+Vd/bUZya1uLKqP+sTcc+aTHbn +AEiqOjPq0D6X45wCzIwjILUCAwEAAQ== +-----END PUBLIC KEY----- diff --git a/kuksa-sdk/src/test/resources/provide-all.token b/kuksa-sdk/src/test/resources/authentication/provide-all.token similarity index 100% rename from kuksa-sdk/src/test/resources/provide-all.token rename to kuksa-sdk/src/test/resources/authentication/provide-all.token diff --git a/kuksa-sdk/src/test/resources/read-all.token b/kuksa-sdk/src/test/resources/authentication/read-all.token similarity index 100% rename from kuksa-sdk/src/test/resources/read-all.token rename to kuksa-sdk/src/test/resources/authentication/read-all.token diff --git a/kuksa-sdk/src/test/resources/CA.pem b/kuksa-sdk/src/test/resources/tls/CA.pem similarity index 100% rename from kuksa-sdk/src/test/resources/CA.pem rename to kuksa-sdk/src/test/resources/tls/CA.pem diff --git a/kuksa-sdk/src/test/resources/tls/Server.key b/kuksa-sdk/src/test/resources/tls/Server.key new file mode 100644 index 00000000..2e0e5f9a --- /dev/null +++ b/kuksa-sdk/src/test/resources/tls/Server.key @@ -0,0 +1,27 @@ +-----BEGIN RSA PRIVATE KEY----- +MIIEpQIBAAKCAQEA0FXTgl4B1nXO0AhHGwPWv/nD73sAnyhSUqUHKUq9WFg/u2ML +YX9OFjAgYvXTQGkEdGYJc6e/8TtNae8hQYsX6JVT669yg5vatIf3/GVhvxhngNwp +arLGYXyRV7QCgoBrdzpbTuUhhw51kWzgj1oUw+/XUTrepAvzZa6dh1ieKQSfmJUi +6OdcFCagIC1gVAXBv8fPQKjoznmtXs4rmryKfstgtU4ZXHx+wDQaafoUiveZbfVn +toi4UW926+IUI6xD4y3wVVoJyiit//T96eJ7QQsfeSvQGufTSTr3jS+w3oUpG/// +UZCLdFWfitZk5z3ERGY/ysmGXuRrDKOI1lJsaQIDAQABAoIBADbye/+FVcy/c5Vw +qXhZkdk+Qcw2z9oqY1QCzJm6aagUVnLQ056aWfkGJnSTS21QhWlBxDppwvkX3/oR +pN7Jlbu1LtYZhwVpJMy6k9BQ2O/yutHEv3OxxRrPJPJRutu6jR18Gbr8OdSKHr4F +TlDgvFtVNZ/p+/CeakqmPEwHNmewo6IfpkBNKHDmi1NpYoEFoyfb0XDVH8m87k5A +fso1hlDbzPQlJ840ebaf0DQBmeQNE3r5NeOMaMOPZzHUY2Ps/+AtCRUhIiAaw7cI +qkW3Vqkg6Lm1zjWv4ZUVsxz6mPKEzYX94CFPKIlWFDOLXYx/Ne1c2O2IIyyaUFrh +sD3dESUCgYEA6+N90wM0JahyOytwlHwmFtycqY8cNd1HGpR50C6Tly4ogsBD6j6h +UNnUdbDDqiI4y+ZAle+z4hqboW2UWc3yHPZG7Lxn94PgE7HLCCOsz8Uwpm2Xord0 +Uk4yQANzn1/PmbOubxIsTVhz37gotSEj/NFgU0r61bpeDBBHCd2uU18CgYEA4hj0 +S2F6+xbAbpVgdDBCxubMKbKT9jaxVgjr+M22Vrt+EM1zGkIQUhhpHCS5n6h9hmT3 +JJ2UZDuYD/J0N5kBcCVkHWFQZVfhURmQ6xcYgJmz97XkXNlTrAkKZwSXnud3wVY6 +ooz6yz71eA6E+Jln9SVnjO+kBnVvWylqD4OLXTcCgYEA3E+PX7o5RQarEbpDnlrJ +VEbdhrujlGdDln6fuqEVtXgl2+BoTeoKzjF0bisWw9rMgxtcrOzAa+d//WgTy0A+ +5W/a1BYvYAvqB1rhjouLRk4cXwQyQIXo/UoNQp42qd5ZTWt0+kXV3LNFHeipFGqM +av6+YWzBE0bJuaimQH+r5i8CgYEA4QOixQ6LvS4Eb0m8h4WsP5VcZLcz6BrAXHZe +mevo5uKL8R72yZAr+/gNS5QFJK8j1cfP6qHlF+fzSxOps9dThg/AVjkOMP3H4NWH +01/V486UPBfK4NjtG86XirUYTG3iUgiGLFYQFoxe8Y/JqDvMKT6DktDANilTnK9X +vX7WCPsCgYEAjMy6OfWhSeGMFYB8QTUkIIf57mdSGYbpVxEh8UnpZJPq62AZj+FX +Rzyu5CLLkg3PTVOyzsXnBOxBwkjje72Wv7tKaq9uKYkNPyeYIiM5VZTEKzPOAQTJ +eO3xy/LbOM1Plfy/koGN5FiHLL+EBdKn42F+ytj0FEYls0V0sZ+kbcQ= +-----END RSA PRIVATE KEY----- diff --git a/kuksa-sdk/src/test/resources/tls/Server.pem b/kuksa-sdk/src/test/resources/tls/Server.pem new file mode 100644 index 00000000..4796388c --- /dev/null +++ b/kuksa-sdk/src/test/resources/tls/Server.pem @@ -0,0 +1,28 @@ +-----BEGIN CERTIFICATE----- +MIIEvDCCA6SgAwIBAgIUficQzSD+1RjwvQQBvs3jGdvrkSgwDQYJKoZIhvcNAQEL +BQAwgZQxCzAJBgNVBAYTAkNBMRAwDgYDVQQIDAdPbnRhcmlvMQ8wDQYDVQQHDAZP +dHRhd2ExJTAjBgNVBAoMHEVjbGlwc2Uub3JnIEZvdW5kYXRpb24sIEluYy4xFTAT +BgNVBAMMDGxvY2FsaG9zdC1jYTEkMCIGCSqGSIb3DQEJARYVa3Vrc2EtZGV2QGVj +bGlwc2Uub3JnMB4XDTIzMDYxNjEwMDQxNloXDTI0MDYxNTEwMDQxNlowgY4xCzAJ +BgNVBAYTAkNBMRAwDgYDVQQIDAdPbnRhcmlvMQ8wDQYDVQQHDAZPdHRhd2ExJTAj +BgNVBAoMHEVjbGlwc2Uub3JnIEZvdW5kYXRpb24sIEluYy4xDzANBgNVBAMMBlNl +cnZlcjEkMCIGCSqGSIb3DQEJARYVa3Vrc2EtZGV2QGVjbGlwc2Uub3JnMIIBIjAN +BgkqhkiG9w0BAQEFAAOCAQ8AMIIBCgKCAQEA0FXTgl4B1nXO0AhHGwPWv/nD73sA +nyhSUqUHKUq9WFg/u2MLYX9OFjAgYvXTQGkEdGYJc6e/8TtNae8hQYsX6JVT669y +g5vatIf3/GVhvxhngNwparLGYXyRV7QCgoBrdzpbTuUhhw51kWzgj1oUw+/XUTre +pAvzZa6dh1ieKQSfmJUi6OdcFCagIC1gVAXBv8fPQKjoznmtXs4rmryKfstgtU4Z +XHx+wDQaafoUiveZbfVntoi4UW926+IUI6xD4y3wVVoJyiit//T96eJ7QQsfeSvQ +GufTSTr3jS+w3oUpG///UZCLdFWfitZk5z3ERGY/ysmGXuRrDKOI1lJsaQIDAQAB +o4IBCDCCAQQwIgYDVR0RBBswGYIGU2VydmVygglsb2NhbGhvc3SHBH8AAAEwHQYD +VR0OBBYEFGC69Htc/kDNUmsFH8hPB0ZGdkQHMIG+BgNVHSMEgbYwgbOhgZqkgZcw +gZQxCzAJBgNVBAYTAkNBMRAwDgYDVQQIDAdPbnRhcmlvMQ8wDQYDVQQHDAZPdHRh +d2ExJTAjBgNVBAoMHEVjbGlwc2Uub3JnIEZvdW5kYXRpb24sIEluYy4xFTATBgNV +BAMMDGxvY2FsaG9zdC1jYTEkMCIGCSqGSIb3DQEJARYVa3Vrc2EtZGV2QGVjbGlw +c2Uub3JnghR92Q2Epkh+nIVseTvM7XpxzB0XwjANBgkqhkiG9w0BAQsFAAOCAQEA +pGgKVbE6AwnSBmCy/Q1tCSqKQj2ILpOZBNBBpAqbu/vUUy9XL4DKKX/Oi5FnRtJk +VgsBg3jVK5NL00FR7bBb/8WLJHr+lTARk8SFOpReP+8vJap4G6vIJYvmJhvLurvH +axJauc81YnookPMjocjm9WW570UKNR3Yac72lwGcoOUnhLkkGzMvW0/xBoVhaVGh +dPCFSuS65ulpB8TYATyEasrtaC9wYlmOxX/FhkgM7MRsHc4XJyEMmPVxJNBgESL4 +Ca/J0lUvnELAP6GndfXaiJ7VaLp7RbRcHfMVfHsy3buF2pVwr2aMteBOl2WBv7NS +divXduydfcBGtdJ1/a70CA== +-----END CERTIFICATE----- diff --git a/kuksa-sdk/src/testDebug/kotlin/org/eclipse/kuksa/connectivity/databroker/DebugDataBrokerConfig.kt b/kuksa-sdk/src/testDebug/kotlin/org/eclipse/kuksa/connectivity/databroker/DebugDataBrokerConfig.kt new file mode 100644 index 00000000..3790bea6 --- /dev/null +++ b/kuksa-sdk/src/testDebug/kotlin/org/eclipse/kuksa/connectivity/databroker/DebugDataBrokerConfig.kt @@ -0,0 +1,23 @@ +/* + * Copyright (c) 2023 - 2024 Contributors to the Eclipse Foundation + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + * + * SPDX-License-Identifier: Apache-2.0 + * + */ + +package org.eclipse.kuksa.connectivity.databroker + +const val DATABROKER_PORT = 55557 +const val DATABROKER_CONTAINER_NAME = "databroker_testDebug" diff --git a/kuksa-sdk/src/testRelease/kotlin/org/eclipse/kuksa/connectivity/databroker/ReleaseDataBrokerConfig.kt b/kuksa-sdk/src/testRelease/kotlin/org/eclipse/kuksa/connectivity/databroker/ReleaseDataBrokerConfig.kt new file mode 100644 index 00000000..f711c133 --- /dev/null +++ b/kuksa-sdk/src/testRelease/kotlin/org/eclipse/kuksa/connectivity/databroker/ReleaseDataBrokerConfig.kt @@ -0,0 +1,22 @@ +/* + * Copyright (c) 2023 Contributors to the Eclipse Foundation + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + * + * SPDX-License-Identifier: Apache-2.0 + */ + +package org.eclipse.kuksa.connectivity.databroker + +const val DATABROKER_PORT = 55558 +const val DATABROKER_CONTAINER_NAME = "databroker_testRelease" From f1fe5fbddae5d9165f2f17380cc210e73bf6369a Mon Sep 17 00:00:00 2001 From: Andre Weber Date: Fri, 15 Mar 2024 09:19:34 +0100 Subject: [PATCH 2/8] chore: Move JwtType to separate class --- .../DataBrokerConnectorAuthenticationTest.kt | 15 -------- .../connectivity/authentication/JwtType.kt | 38 +++++++++++++++++++ 2 files changed, 38 insertions(+), 15 deletions(-) create mode 100644 kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/authentication/JwtType.kt diff --git a/kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/authentication/DataBrokerConnectorAuthenticationTest.kt b/kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/authentication/DataBrokerConnectorAuthenticationTest.kt index b8c1558a..8a4ef959 100644 --- a/kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/authentication/DataBrokerConnectorAuthenticationTest.kt +++ b/kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/authentication/DataBrokerConnectorAuthenticationTest.kt @@ -184,18 +184,3 @@ class DataBrokerConnectorAuthenticationTest : BehaviorSpec({ } } }) - -// The tokens provided here might need to be updated irregularly -// see: https://github.com/eclipse/kuksa.val/tree/master/jwt -// The tokens only work when the Databroker is started using the correct public key: jwt.key.pub -enum class JwtType(private val fileName: String) { - READ_WRITE_ALL("actuate-provide-all.token"), // ACTUATOR_TARGET and VALUE - READ_WRITE_ALL_VALUES_ONLY("provide-all.token"), // VALUE - READ_ALL("read-all.token"), - ; - - fun asInputStream(): InputStream { - val resourceFile = TestResourceFile(fileName) - return resourceFile.inputStream() - } -} diff --git a/kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/authentication/JwtType.kt b/kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/authentication/JwtType.kt new file mode 100644 index 00000000..e87f9d65 --- /dev/null +++ b/kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/authentication/JwtType.kt @@ -0,0 +1,38 @@ +/* + * Copyright (c) 2023 - 2024 Contributors to the Eclipse Foundation + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + * + * SPDX-License-Identifier: Apache-2.0 + * + */ + +package org.eclipse.kuksa.connectivity.authentication + +import org.eclipse.kuksa.test.TestResourceFile +import java.io.InputStream + +// The tokens provided here might need to be updated irregularly +// see: https://github.com/eclipse/kuksa.val/tree/master/jwt +// The tokens only work when the Databroker is started using the correct public key: jwt.key.pub +enum class JwtType(private val fileName: String) { + READ_WRITE_ALL("authentication/actuate-provide-all.token"), // ACTUATOR_TARGET and VALUE + READ_WRITE_ALL_VALUES_ONLY("authentication/provide-all.token"), // VALUE + READ_ALL("authentication/read-all.token"), + ; + + fun asInputStream(): InputStream { + val resourceFile = TestResourceFile(fileName) + return resourceFile.inputStream() + } +} From 0083330dfe6a2785b8a601a994c06772527e522f Mon Sep 17 00:00:00 2001 From: Andre Weber Date: Fri, 15 Mar 2024 09:21:38 +0100 Subject: [PATCH 3/8] test: Restructure Kotest Test Tags --- .../DataBrokerConnectorAuthenticationTest.kt | 6 +++--- .../databroker/DataBrokerConnectionTest.kt | 5 +++-- .../databroker/DataBrokerConnectorSecureTest.kt | 5 +++-- .../databroker/DataBrokerConnectorTest.kt | 4 ++-- .../databroker/DataBrokerTransporterTest.kt | 4 ++-- .../subscription/DataBrokerSubscriberTest.kt | 4 ++-- .../main/java/org/eclipse/kuksa/test/kotest/Tag.kt | 13 ++++++++----- 7 files changed, 23 insertions(+), 18 deletions(-) diff --git a/kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/authentication/DataBrokerConnectorAuthenticationTest.kt b/kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/authentication/DataBrokerConnectorAuthenticationTest.kt index 8a4ef959..dd251e7b 100644 --- a/kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/authentication/DataBrokerConnectorAuthenticationTest.kt +++ b/kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/authentication/DataBrokerConnectorAuthenticationTest.kt @@ -29,9 +29,9 @@ import org.eclipse.kuksa.connectivity.databroker.request.UpdateRequest import org.eclipse.kuksa.proto.v1.Types import org.eclipse.kuksa.test.TestResourceFile import org.eclipse.kuksa.test.kotest.Authentication -import org.eclipse.kuksa.test.kotest.CustomDatabroker -import org.eclipse.kuksa.test.kotest.Insecure import org.eclipse.kuksa.test.kotest.Integration +import org.eclipse.kuksa.test.kotest.Secure +import org.eclipse.kuksa.test.kotest.SecureDatabroker import kotlin.random.Random import kotlin.random.nextInt @@ -40,7 +40,7 @@ import kotlin.random.nextInt // ./gradlew clean test -Dkotest.tags="Authentication" class DataBrokerConnectorAuthenticationTest : BehaviorSpec({ - tags(Integration, Authentication, Insecure, CustomDatabroker) + tags(Integration, Authentication, Secure, SecureDatabroker) var databrokerContainer: DockerDatabrokerContainer? = null beforeSpec { diff --git a/kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/databroker/DataBrokerConnectionTest.kt b/kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/databroker/DataBrokerConnectionTest.kt index e85d78f3..7f8a7da4 100644 --- a/kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/databroker/DataBrokerConnectionTest.kt +++ b/kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/databroker/DataBrokerConnectionTest.kt @@ -42,7 +42,8 @@ import org.eclipse.kuksa.connectivity.databroker.request.VssNodeSubscribeRequest import org.eclipse.kuksa.mocking.FriendlyVssNodeListener import org.eclipse.kuksa.proto.v1.KuksaValV1 import org.eclipse.kuksa.proto.v1.Types -import org.eclipse.kuksa.test.kotest.DefaultDatabroker +import org.eclipse.kuksa.test.kotest.Insecure +import org.eclipse.kuksa.test.kotest.InsecureDatabroker import org.eclipse.kuksa.test.kotest.Integration import org.eclipse.kuksa.vssNode.VssDriver import org.junit.jupiter.api.Assertions @@ -50,7 +51,7 @@ import kotlin.random.Random import kotlin.time.Duration.Companion.seconds class DataBrokerConnectionTest : BehaviorSpec({ - tags(Integration, DefaultDatabroker) + tags(Integration, Insecure, InsecureDatabroker) var databrokerContainer: DockerDatabrokerContainer? = null beforeSpec { diff --git a/kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/databroker/DataBrokerConnectorSecureTest.kt b/kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/databroker/DataBrokerConnectorSecureTest.kt index 3995c046..e35b4207 100644 --- a/kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/databroker/DataBrokerConnectorSecureTest.kt +++ b/kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/databroker/DataBrokerConnectorSecureTest.kt @@ -22,14 +22,15 @@ import io.kotest.core.spec.style.BehaviorSpec import org.eclipse.kuksa.connectivity.databroker.docker.DockerDatabrokerContainer import org.eclipse.kuksa.connectivity.databroker.docker.DockerSecureDatabrokerContainer import org.eclipse.kuksa.test.TestResourceFile -import org.eclipse.kuksa.test.kotest.CustomDatabroker import org.eclipse.kuksa.test.kotest.Integration import org.eclipse.kuksa.test.kotest.Secure +import org.eclipse.kuksa.test.kotest.SecureDatabroker +import org.eclipse.kuksa.test.kotest.Tls import org.junit.jupiter.api.Assertions // run command: ./gradlew clean test -Dkotest.tags="Secure" class DataBrokerConnectorSecureTest : BehaviorSpec({ - tags(Integration, Secure, CustomDatabroker) + tags(Integration, Secure, Tls, SecureDatabroker) var databrokerContainer: DockerDatabrokerContainer? = null beforeSpec { diff --git a/kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/databroker/DataBrokerConnectorTest.kt b/kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/databroker/DataBrokerConnectorTest.kt index b9eb5d73..79c67d88 100644 --- a/kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/databroker/DataBrokerConnectorTest.kt +++ b/kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/databroker/DataBrokerConnectorTest.kt @@ -21,14 +21,14 @@ package org.eclipse.kuksa.connectivity.databroker import io.kotest.assertions.throwables.shouldThrow import io.kotest.core.spec.style.BehaviorSpec import io.kotest.matchers.shouldNotBe -import org.eclipse.kuksa.test.kotest.DefaultDatabroker import org.eclipse.kuksa.connectivity.databroker.docker.DockerDatabrokerContainer import org.eclipse.kuksa.connectivity.databroker.docker.DockerInsecureDatabrokerContainer import org.eclipse.kuksa.test.kotest.Insecure +import org.eclipse.kuksa.test.kotest.InsecureDatabroker import org.eclipse.kuksa.test.kotest.Integration class DataBrokerConnectorTest : BehaviorSpec({ - tags(Integration, Insecure, DefaultDatabroker) + tags(Integration, Insecure, InsecureDatabroker) var databrokerContainer: DockerDatabrokerContainer? = null beforeSpec { diff --git a/kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/databroker/DataBrokerTransporterTest.kt b/kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/databroker/DataBrokerTransporterTest.kt index 6a053967..bdceda3c 100644 --- a/kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/databroker/DataBrokerTransporterTest.kt +++ b/kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/databroker/DataBrokerTransporterTest.kt @@ -31,13 +31,13 @@ import org.eclipse.kuksa.connectivity.databroker.listener.VssPathListener import org.eclipse.kuksa.extensions.updateRandomFloatValue import org.eclipse.kuksa.proto.v1.KuksaValV1 import org.eclipse.kuksa.proto.v1.Types -import org.eclipse.kuksa.test.kotest.DefaultDatabroker import org.eclipse.kuksa.test.kotest.Insecure +import org.eclipse.kuksa.test.kotest.InsecureDatabroker import org.eclipse.kuksa.test.kotest.Integration import kotlin.random.Random class DataBrokerTransporterTest : BehaviorSpec({ - tags(Integration, Insecure, DefaultDatabroker) + tags(Integration, Insecure, InsecureDatabroker) var databrokerContainer: DockerDatabrokerContainer? = null beforeSpec { diff --git a/kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/databroker/subscription/DataBrokerSubscriberTest.kt b/kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/databroker/subscription/DataBrokerSubscriberTest.kt index 3fb4f93b..1b87904a 100644 --- a/kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/databroker/subscription/DataBrokerSubscriberTest.kt +++ b/kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/databroker/subscription/DataBrokerSubscriberTest.kt @@ -40,15 +40,15 @@ import org.eclipse.kuksa.mocking.FriendlyVssPathListener import org.eclipse.kuksa.pattern.listener.MultiListener import org.eclipse.kuksa.pattern.listener.count import org.eclipse.kuksa.proto.v1.Types -import org.eclipse.kuksa.test.kotest.DefaultDatabroker import org.eclipse.kuksa.test.kotest.Insecure +import org.eclipse.kuksa.test.kotest.InsecureDatabroker import org.eclipse.kuksa.test.kotest.Integration import org.eclipse.kuksa.vssNode.VssDriver import kotlin.time.Duration.Companion.milliseconds import kotlin.time.Duration.Companion.seconds class DataBrokerSubscriberTest : BehaviorSpec({ - tags(Integration, Insecure, DefaultDatabroker) + tags(Integration, Insecure, InsecureDatabroker) var databrokerContainer: DockerDatabrokerContainer? = null beforeSpec { diff --git a/test/src/main/java/org/eclipse/kuksa/test/kotest/Tag.kt b/test/src/main/java/org/eclipse/kuksa/test/kotest/Tag.kt index 01ec66ad..a9dee663 100644 --- a/test/src/main/java/org/eclipse/kuksa/test/kotest/Tag.kt +++ b/test/src/main/java/org/eclipse/kuksa/test/kotest/Tag.kt @@ -22,10 +22,13 @@ package org.eclipse.kuksa.test.kotest import io.kotest.core.NamedTag val Integration = NamedTag("Integration") -val DefaultDatabroker = NamedTag("DefaultDatabroker") // unsecure => no tls, no authentication -val CustomDatabroker = NamedTag("CustomDatabroker") -val Secure = NamedTag("Secure") // tls enabled, no authentication -val Authentication = NamedTag("Authentication") // no tls, authentication enabled -val Unit = NamedTag("Unit") +val InsecureDatabroker = NamedTag("InsecureDatabroker") // no tls, no authentication val Insecure = NamedTag("Insecure") + +val SecureDatabroker = NamedTag("SecureDatabroker") +val Secure = NamedTag("Secure") // tls enabled, authentication enabled +val Authentication = NamedTag("Authentication") +val Tls = NamedTag("Tls") + +val Unit = NamedTag("Unit") From 1c5b0ee77d96bc94b3105ef6db1891684af1651b Mon Sep 17 00:00:00 2001 From: Andre Weber Date: Fri, 15 Mar 2024 09:24:47 +0100 Subject: [PATCH 4/8] test: Make Tests more robust --- .../DataBrokerConnectorAuthenticationTest.kt | 86 +++++++++- .../databroker/DataBrokerConnectionTest.kt | 112 +++++------- .../databroker/DataBrokerConnectorTest.kt | 2 +- .../databroker/DataBrokerTransporterTest.kt | 11 +- .../subscription/DataBrokerSubscriberTest.kt | 159 +++++++++--------- .../kuksa/test/extension/FloatExtension.kt | 26 +++ .../kuksa/test/kotest/KotestProjectConfig.kt | 11 ++ 7 files changed, 248 insertions(+), 159 deletions(-) create mode 100644 test/src/main/java/org/eclipse/kuksa/test/extension/FloatExtension.kt diff --git a/kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/authentication/DataBrokerConnectorAuthenticationTest.kt b/kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/authentication/DataBrokerConnectorAuthenticationTest.kt index dd251e7b..0b5b14c8 100644 --- a/kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/authentication/DataBrokerConnectorAuthenticationTest.kt +++ b/kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/authentication/DataBrokerConnectorAuthenticationTest.kt @@ -19,19 +19,27 @@ package org.eclipse.kuksa.connectivity.authentication +import io.grpc.StatusRuntimeException +import io.kotest.assertions.nondeterministic.eventually import io.kotest.core.spec.style.BehaviorSpec import io.kotest.matchers.shouldBe +import io.kotest.matchers.shouldNotBe +import io.kotest.matchers.string.shouldContain +import io.kotest.matchers.types.instanceOf import org.eclipse.kuksa.connectivity.databroker.DataBrokerConnectorProvider +import org.eclipse.kuksa.connectivity.databroker.DataBrokerException import org.eclipse.kuksa.connectivity.databroker.docker.DockerDatabrokerContainer import org.eclipse.kuksa.connectivity.databroker.docker.DockerSecureDatabrokerContainer import org.eclipse.kuksa.connectivity.databroker.request.FetchRequest +import org.eclipse.kuksa.connectivity.databroker.request.SubscribeRequest import org.eclipse.kuksa.connectivity.databroker.request.UpdateRequest +import org.eclipse.kuksa.mocking.FriendlyVssPathListener import org.eclipse.kuksa.proto.v1.Types -import org.eclipse.kuksa.test.TestResourceFile import org.eclipse.kuksa.test.kotest.Authentication import org.eclipse.kuksa.test.kotest.Integration import org.eclipse.kuksa.test.kotest.Secure import org.eclipse.kuksa.test.kotest.SecureDatabroker +import org.eclipse.kuksa.test.kotest.eventuallyConfiguration import kotlin.random.Random import kotlin.random.nextInt @@ -55,17 +63,15 @@ class DataBrokerConnectorAuthenticationTest : BehaviorSpec({ } val random = Random(System.nanoTime()) - val tlsCertificate = TestResourceFile("tls/CA.pem") given("A DataBrokerConnectorProvider") { val dataBrokerConnectorProvider = DataBrokerConnectorProvider() val speedVssPath = "Vehicle.Speed" - and("an insecure DataBrokerConnector with a READ_WRITE_ALL JWT") { + and("a secure DataBrokerConnector with a READ_WRITE_ALL JWT") { val jwtFile = JwtType.READ_WRITE_ALL val dataBrokerConnector = dataBrokerConnectorProvider.createSecure( - rootCertFileStream = tlsCertificate.inputStream(), jwtFileStream = jwtFile.asInputStream(), ) @@ -95,10 +101,9 @@ class DataBrokerConnectorAuthenticationTest : BehaviorSpec({ } } - and("an insecure DataBrokerConnector with a READ_ALL JWT") { + and("a secure DataBrokerConnector with a READ_ALL JWT") { val jwtFile = JwtType.READ_ALL val dataBrokerConnector = dataBrokerConnectorProvider.createSecure( - rootCertFileStream = tlsCertificate.inputStream(), jwtFileStream = jwtFile.asInputStream(), ) @@ -127,10 +132,9 @@ class DataBrokerConnectorAuthenticationTest : BehaviorSpec({ } } - and("an insecure DataBrokerConnector with a READ_WRITE_ALL_VALUES_ONLY JWT") { + and("a secure DataBrokerConnector with a READ_WRITE_ALL_VALUES_ONLY JWT") { val jwtFile = JwtType.READ_WRITE_ALL_VALUES_ONLY val dataBrokerConnector = dataBrokerConnectorProvider.createSecure( - rootCertFileStream = tlsCertificate.inputStream(), jwtFileStream = jwtFile.asInputStream(), ) @@ -182,5 +186,71 @@ class DataBrokerConnectorAuthenticationTest : BehaviorSpec({ } } } + + and("a secure DataBrokerConnector with no JWT") { + val dataBrokerConnector = dataBrokerConnectorProvider.createSecure( + jwtFileStream = null, + ) + + `when`("Trying to connect") { + val result = runCatching { + dataBrokerConnector.connect() + } + + then("The connection should be successful") { + result.getOrNull() shouldNotBe null + } + + val connection = result.getOrNull()!! + + `when`("Reading the VALUE of Vehicle.Speed") { + val fetchRequest = FetchRequest(speedVssPath) + val fetchResult = runCatching { + connection.fetch(fetchRequest) + } + + then("An error should occur") { + val exception = fetchResult.exceptionOrNull() + exception shouldNotBe null + exception shouldBe instanceOf(DataBrokerException::class) + exception!!.message shouldContain "UNAUTHENTICATED" + } + } + + `when`("Writing the VALUE of Vehicle.Speed") { + val nextFloat = random.nextFloat() * 100F + val datapoint = Types.Datapoint.newBuilder().setFloat(nextFloat).build() + val updateRequest = UpdateRequest(speedVssPath, datapoint) + + val updateResult = runCatching { + connection.update(updateRequest) + } + + then("An error should occur") { + val exception = updateResult.exceptionOrNull() + exception shouldNotBe null + exception shouldBe instanceOf(DataBrokerException::class) + exception!!.message shouldContain "UNAUTHENTICATED" + } + } + + `when`("Subscribing to the VALUE of Vehicle.Speed") { + val subscribeRequest = SubscribeRequest(speedVssPath) + val vssPathListener = FriendlyVssPathListener() + + connection.subscribe(subscribeRequest, vssPathListener) + + then("An error should occur") { + eventually(eventuallyConfiguration) { + vssPathListener.errors.size shouldBe 1 + + val exception = vssPathListener.errors.first() + exception shouldBe instanceOf(StatusRuntimeException::class) + exception.message shouldContain "UNAUTHENTICATED" + } + } + } + } + } } }) diff --git a/kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/databroker/DataBrokerConnectionTest.kt b/kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/databroker/DataBrokerConnectionTest.kt index 7f8a7da4..b5682ad8 100644 --- a/kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/databroker/DataBrokerConnectionTest.kt +++ b/kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/databroker/DataBrokerConnectionTest.kt @@ -23,32 +23,29 @@ import io.grpc.ManagedChannel import io.kotest.assertions.nondeterministic.eventually import io.kotest.core.spec.style.BehaviorSpec import io.kotest.matchers.shouldBe -import io.kotest.matchers.string.shouldContain -import io.mockk.clearMocks import io.mockk.every import io.mockk.mockk -import io.mockk.slot import io.mockk.verify -import kotlinx.coroutines.runBlocking import org.eclipse.kuksa.connectivity.databroker.docker.DockerDatabrokerContainer import org.eclipse.kuksa.connectivity.databroker.docker.DockerInsecureDatabrokerContainer import org.eclipse.kuksa.connectivity.databroker.listener.DisconnectListener -import org.eclipse.kuksa.connectivity.databroker.listener.VssPathListener import org.eclipse.kuksa.connectivity.databroker.request.FetchRequest import org.eclipse.kuksa.connectivity.databroker.request.SubscribeRequest import org.eclipse.kuksa.connectivity.databroker.request.UpdateRequest import org.eclipse.kuksa.connectivity.databroker.request.VssNodeFetchRequest import org.eclipse.kuksa.connectivity.databroker.request.VssNodeSubscribeRequest +import org.eclipse.kuksa.extensions.updateRandomFloatValue import org.eclipse.kuksa.mocking.FriendlyVssNodeListener -import org.eclipse.kuksa.proto.v1.KuksaValV1 +import org.eclipse.kuksa.mocking.FriendlyVssPathListener import org.eclipse.kuksa.proto.v1.Types +import org.eclipse.kuksa.test.extension.equals import org.eclipse.kuksa.test.kotest.Insecure import org.eclipse.kuksa.test.kotest.InsecureDatabroker import org.eclipse.kuksa.test.kotest.Integration +import org.eclipse.kuksa.test.kotest.eventuallyConfiguration import org.eclipse.kuksa.vssNode.VssDriver import org.junit.jupiter.api.Assertions import kotlin.random.Random -import kotlin.time.Duration.Companion.seconds class DataBrokerConnectionTest : BehaviorSpec({ tags(Integration, Insecure, InsecureDatabroker) @@ -66,49 +63,50 @@ class DataBrokerConnectionTest : BehaviorSpec({ } given("A successfully established connection to the DataBroker") { - val dataBrokerConnection = connectToDataBrokerBlocking() + val dataBrokerConnectorProvider = DataBrokerConnectorProvider() + val connector = dataBrokerConnectorProvider.createInsecure() + val dataBrokerConnection = connector.connect() + + val dataBrokerTransporter = DataBrokerTransporter(dataBrokerConnectorProvider.managedChannel) and("A request with a valid VSS Path") { val vssPath = "Vehicle.Acceleration.Lateral" val field = Types.Field.FIELD_VALUE - val subscribeRequest = SubscribeRequest(vssPath, field) + val initialValue = dataBrokerTransporter.updateRandomFloatValue(vssPath) + + val subscribeRequest = SubscribeRequest(vssPath, field) `when`("Subscribing to the VSS path") { - val vssPathListener = mockk(relaxed = true) + val vssPathListener = FriendlyVssPathListener() dataBrokerConnection.subscribe(subscribeRequest, vssPathListener) then("The #onEntryChanged method is triggered") { - val capturingSlot = slot>() - verify(timeout = 100L) { - vssPathListener.onEntryChanged(capture(capturingSlot)) + eventually(eventuallyConfiguration) { + vssPathListener.updates.flatten().count { + val entry = it.entry + val value = entry.value + entry.path == vssPath && value.float.equals(initialValue, 0.0001f) + } shouldBe 1 } - - val entryUpdates = capturingSlot.captured - entryUpdates.size shouldBe 1 - entryUpdates[0].entry.path shouldBe vssPath } `when`("The observed VSS path changes") { - clearMocks(vssPathListener) + vssPathListener.reset() val random = Random(System.currentTimeMillis()) - val newValue = random.nextFloat() - val datapoint = Types.Datapoint.newBuilder().setFloat(newValue).build() + val updatedValue = random.nextFloat() + val datapoint = Types.Datapoint.newBuilder().setFloat(updatedValue).build() val updateRequest = UpdateRequest(vssPath, datapoint, field) dataBrokerConnection.update(updateRequest) then("The #onEntryChanged callback is triggered with the new value") { - val capturingSlot = slot>() - - verify(timeout = 100) { - vssPathListener.onEntryChanged(capture(capturingSlot)) + eventually(eventuallyConfiguration) { + vssPathListener.updates.flatten().count { + val entry = it.entry + val value = entry.value + entry.path == vssPath && value.float.equals(updatedValue, 0.0001f) + } shouldBe 1 } - - val entryUpdates = capturingSlot.captured - val capturedDatapoint = entryUpdates[0].entry.value - val float = capturedDatapoint.float - - Assertions.assertEquals(newValue, float, 0.0001f) } } } @@ -193,7 +191,7 @@ class DataBrokerConnectionTest : BehaviorSpec({ dataBrokerConnection.subscribe(subscribeRequest, listener = vssNodeListener) then("The #onNodeChanged method is triggered") { - eventually(1.seconds) { + eventually(eventuallyConfiguration) { vssNodeListener.updatedVssNodes.size shouldBe 1 } } @@ -206,14 +204,12 @@ class DataBrokerConnectionTest : BehaviorSpec({ dataBrokerConnection.update(updateRequest) then("Every child node has been updated with the correct value") { - eventually(1.seconds) { - vssNodeListener.updatedVssNodes.size shouldBe 2 + eventually(eventuallyConfiguration) { + vssNodeListener.updatedVssNodes.count { + val heartRate = it.heartRate + heartRate.value == newHeartRateValue + } shouldBe 1 } - - val updatedDriver = vssNodeListener.updatedVssNodes.last() - val heartRate = updatedDriver.heartRate - - heartRate.value shouldBe newHeartRateValue } } @@ -225,14 +221,12 @@ class DataBrokerConnectionTest : BehaviorSpec({ dataBrokerConnection.update(updateRequest) then("The subscribed vssNode should be updated") { - eventually(1.seconds) { - vssNodeListener.updatedVssNodes.size shouldBe 3 + eventually(eventuallyConfiguration) { + vssNodeListener.updatedVssNodes.count { + val heartRate = it.heartRate + heartRate.value == newHeartRateValue + } shouldBe 1 } - - val updatedDriver = vssNodeListener.updatedVssNodes.last() - val heartRate = updatedDriver.heartRate - - heartRate.value shouldBe newHeartRateValue } } } @@ -242,15 +236,16 @@ class DataBrokerConnectionTest : BehaviorSpec({ val invalidVssPath = "Vehicle.Some.Unknown.Path" `when`("Trying to subscribe to the INVALID VSS path") { - val vssPathListener = mockk(relaxed = true) + val vssPathListener = FriendlyVssPathListener() val subscribeRequest = SubscribeRequest(invalidVssPath) dataBrokerConnection.subscribe(subscribeRequest, vssPathListener) then("The VssPathListener#onError method should be triggered with 'NOT_FOUND' (Path not found)") { - val capturingSlot = slot() - verify(timeout = 100L) { vssPathListener.onError(capture(capturingSlot)) } - val capturedThrowable = capturingSlot.captured - capturedThrowable.message shouldContain "NOT_FOUND" + eventually(eventuallyConfiguration) { + vssPathListener.errors.count { + it.message?.contains("NOT_FOUND") == true + } shouldBe 1 + } } } @@ -331,20 +326,3 @@ private fun createRandomIntDatapoint(): Types.Datapoint { val newValue = random.nextInt() return Types.Datapoint.newBuilder().setInt32(newValue).build() } - -private fun connectToDataBrokerBlocking(): DataBrokerConnection { - var connection: DataBrokerConnection - - runBlocking { - val connector = DataBrokerConnectorProvider().createInsecure() - try { - connection = connector.connect() - } catch (ignored: DataBrokerException) { - val errorMessage = "Could not establish a connection to the DataBroker. " + - "Check if the DataBroker is running and correctly configured in DataBrokerConfig." - throw IllegalStateException(errorMessage) - } - } - - return connection -} diff --git a/kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/databroker/DataBrokerConnectorTest.kt b/kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/databroker/DataBrokerConnectorTest.kt index 79c67d88..6d56a320 100644 --- a/kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/databroker/DataBrokerConnectorTest.kt +++ b/kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/databroker/DataBrokerConnectorTest.kt @@ -67,7 +67,7 @@ class DataBrokerConnectorTest : BehaviorSpec({ dataBrokerConnector.connect() } - then("It should throw an exception") { + then("It should throw a DataBrokerException") { exception shouldNotBe null } } diff --git a/kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/databroker/DataBrokerTransporterTest.kt b/kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/databroker/DataBrokerTransporterTest.kt index bdceda3c..4544c6c1 100644 --- a/kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/databroker/DataBrokerTransporterTest.kt +++ b/kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/databroker/DataBrokerTransporterTest.kt @@ -19,6 +19,7 @@ package org.eclipse.kuksa.connectivity.databroker import io.grpc.ManagedChannelBuilder +import io.kotest.assertions.nondeterministic.eventually import io.kotest.core.spec.style.BehaviorSpec import io.kotest.matchers.shouldBe import io.kotest.matchers.shouldNotBe @@ -29,11 +30,13 @@ import org.eclipse.kuksa.connectivity.databroker.docker.DockerDatabrokerContaine import org.eclipse.kuksa.connectivity.databroker.docker.DockerInsecureDatabrokerContainer import org.eclipse.kuksa.connectivity.databroker.listener.VssPathListener import org.eclipse.kuksa.extensions.updateRandomFloatValue +import org.eclipse.kuksa.mocking.FriendlyVssPathListener import org.eclipse.kuksa.proto.v1.KuksaValV1 import org.eclipse.kuksa.proto.v1.Types import org.eclipse.kuksa.test.kotest.Insecure import org.eclipse.kuksa.test.kotest.InsecureDatabroker import org.eclipse.kuksa.test.kotest.Integration +import org.eclipse.kuksa.test.kotest.eventuallyConfiguration import kotlin.random.Random class DataBrokerTransporterTest : BehaviorSpec({ @@ -135,12 +138,14 @@ class DataBrokerTransporterTest : BehaviorSpec({ Types.Field.FIELD_VALUE, ) - val vssPathListener = mockk(relaxed = true) + val vssPathListener = FriendlyVssPathListener() subscription.listeners.register(vssPathListener) then("An Error should be triggered") { - verify(timeout = 100L) { - vssPathListener.onError(any()) + eventually(eventuallyConfiguration) { + vssPathListener.errors.count { + it.message?.contains("NOT_FOUND") == true + } shouldBe 1 } } } diff --git a/kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/databroker/subscription/DataBrokerSubscriberTest.kt b/kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/databroker/subscription/DataBrokerSubscriberTest.kt index 1b87904a..14b47d78 100644 --- a/kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/databroker/subscription/DataBrokerSubscriberTest.kt +++ b/kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/databroker/subscription/DataBrokerSubscriberTest.kt @@ -22,11 +22,11 @@ import io.kotest.assertions.nondeterministic.continually import io.kotest.assertions.nondeterministic.eventually import io.kotest.core.spec.style.BehaviorSpec import io.kotest.matchers.shouldBe +import io.kotest.matchers.shouldNotBe import io.mockk.clearMocks import io.mockk.every import io.mockk.mockk import io.mockk.verify -import kotlinx.coroutines.delay import org.eclipse.kuksa.connectivity.databroker.DataBrokerConnectorProvider import org.eclipse.kuksa.connectivity.databroker.DataBrokerTransporter import org.eclipse.kuksa.connectivity.databroker.docker.DockerDatabrokerContainer @@ -43,9 +43,9 @@ import org.eclipse.kuksa.proto.v1.Types import org.eclipse.kuksa.test.kotest.Insecure import org.eclipse.kuksa.test.kotest.InsecureDatabroker import org.eclipse.kuksa.test.kotest.Integration +import org.eclipse.kuksa.test.kotest.continuallyConfiguration +import org.eclipse.kuksa.test.kotest.eventuallyConfiguration import org.eclipse.kuksa.vssNode.VssDriver -import kotlin.time.Duration.Companion.milliseconds -import kotlin.time.Duration.Companion.seconds class DataBrokerSubscriberTest : BehaviorSpec({ tags(Integration, Insecure, InsecureDatabroker) @@ -78,15 +78,18 @@ class DataBrokerSubscriberTest : BehaviorSpec({ val vssPathListener = FriendlyVssPathListener() classUnderTest.subscribe(vssPath, fieldValue, vssPathListener) - then("The VssPathListener should send out ONE update containing ALL children") { - eventually(1.seconds) { - vssPathListener.updates.size shouldBe 1 - } + then("The VssPathListener should send out an update containing ALL children") { + eventually(eventuallyConfiguration) { + val foundUpdate = vssPathListener.updates.find { entryUpdates -> + entryUpdates.size == 3 // all children: IsEnabled, IsEngaged, IsError + } - val entryUpdates = vssPathListener.updates[0] - vssPathListener.updates.size shouldBe 1 // ONE update - entryUpdates.size shouldBe 3 // all children: IsEnabled, IsEngaged, IsError - entryUpdates.all { it.entry.path.startsWith(vssPath) } shouldBe true + foundUpdate shouldNotBe null + foundUpdate?.size shouldBe 3 + foundUpdate?.count { it.entry.path.endsWith(".IsEnabled") } shouldBe 1 + foundUpdate?.count { it.entry.path.endsWith(".IsEngaged") } shouldBe 1 + foundUpdate?.count { it.entry.path.endsWith(".IsError") } shouldBe 1 + } } `when`("Any child changes it's value") { @@ -98,23 +101,21 @@ class DataBrokerSubscriberTest : BehaviorSpec({ val newValueIsEngaged = databrokerTransporter.toggleBoolean(vssPathIsEngaged) then("The VssPathListener should be notified about it") { - eventually(1.seconds) { - vssPathListener.updates.size shouldBe 2 + eventually(eventuallyConfiguration) { + val entryUpdates = vssPathListener.updates.flatten() + entryUpdates.count { + val path = it.entry.path + val entry = it.entry + val value = entry.value + path == vssPathIsError && value.bool == newValueIsError + } shouldBe 1 + entryUpdates.count { + val path = it.entry.path + val entry = it.entry + val value = entry.value + path == vssPathIsEngaged && value.bool == newValueIsEngaged + } shouldBe 1 } - - val entryUpdates = vssPathListener.updates.flatten() - entryUpdates.count { - val path = it.entry.path - val entry = it.entry - val value = entry.value - path == vssPathIsError && value.bool == newValueIsError - } shouldBe 1 - entryUpdates.count { - val path = it.entry.path - val entry = it.entry - val value = entry.value - path == vssPathIsEngaged && value.bool == newValueIsEngaged - } shouldBe 1 } } } @@ -129,21 +130,19 @@ class DataBrokerSubscriberTest : BehaviorSpec({ val updateRandomFloatValue = databrokerTransporter.updateRandomFloatValue(vssPath) then("The VssPathListener is notified about the change") { - eventually(1.seconds) { - vssPathListener.updates.size shouldBe 2 + eventually(eventuallyConfiguration) { + vssPathListener.updates.flatten() + .count { + val dataEntry = it.entry + val datapoint = dataEntry.value + dataEntry.path == vssPath && datapoint.float == updateRandomFloatValue + } shouldBe 1 } - vssPathListener.updates.flatten() - .count { - val dataEntry = it.entry - val datapoint = dataEntry.value - dataEntry.path == vssPath && datapoint.float == updateRandomFloatValue - } shouldBe 1 - - vssPathListener.updates.clear() } } `when`("Subscribing the same VssPathListener to a different vssPath") { + vssPathListener.reset() val otherVssPath = "Vehicle.ADAS.CruiseControl.SpeedSet" classUnderTest.subscribe(otherVssPath, fieldValue, vssPathListener) @@ -153,25 +152,22 @@ class DataBrokerSubscriberTest : BehaviorSpec({ val updatedValueOtherVssPath = databrokerTransporter.updateRandomFloatValue(otherVssPath) then("The Observer is notified about both changes") { - eventually(1.seconds) { - vssPathListener.updates.size shouldBe 3 // 1 from subscribe(otherVssPath) + 2 updates + eventually(eventuallyConfiguration) { + val entryUpdates = vssPathListener.updates.flatten() + entryUpdates + .count { + val path = it.entry.path + val value = it.entry.value + path == vssPath && value.float == updatedValueVssPath + } shouldBe 1 + + entryUpdates + .count { + val path = it.entry.path + val value = it.entry.value + path == otherVssPath && value.float == updatedValueOtherVssPath + } shouldBe 1 } - - val entryUpdates = vssPathListener.updates.flatten() - entryUpdates - .count { - val path = it.entry.path - val entry = it.entry - val value = entry.value - path == vssPath && value.float == updatedValueVssPath - } shouldBe 1 - entryUpdates - .count { - val path = it.entry.path - val entry = it.entry - val value = entry.value - path == otherVssPath && value.float == updatedValueOtherVssPath - } shouldBe 1 } } } @@ -190,13 +186,14 @@ class DataBrokerSubscriberTest : BehaviorSpec({ then("Each VssPathListener is only notified once") { friendlyVssPathListeners.forEach { listener -> - eventually(1.seconds) { - listener.updates.size shouldBe 2 + eventually(eventuallyConfiguration) { + listener.updates.flatten() + .count { + val path = it.entry.path + val value = it.entry.value + path == vssPath && value.float == randomFloatValue + } shouldBe 1 } - - val count = listener.updates - .count { it[0].entry.value.float == randomFloatValue } - count shouldBe 1 } } } @@ -208,10 +205,9 @@ class DataBrokerSubscriberTest : BehaviorSpec({ and("When the FIELD_VALUE of Vehicle.Speed is updated") { databrokerTransporter.updateRandomFloatValue(vssPath) - delay(100) then("The VssPathListener is not notified") { - continually(100.milliseconds) { + continually(continuallyConfiguration) { vssPathListener.updates.size shouldBe 0 } } @@ -230,13 +226,15 @@ class DataBrokerSubscriberTest : BehaviorSpec({ val randomFloatValue = databrokerTransporter.updateRandomFloatValue(vssPath) then("The VssPathListener is only notified once") { - eventually(1.seconds) { - vssPathListener.updates.size shouldBe 2 + eventually(eventuallyConfiguration) { + val count = vssPathListener.updates.flatten() + .count { + val path = it.entry.path + val value = it.entry.value + path == vssPath && value.float == randomFloatValue + } + count shouldBe 1 } - - val count = vssPathListener.updates - .count { it[0].entry.value.float == randomFloatValue } - count shouldBe 1 } } } @@ -256,13 +254,13 @@ class DataBrokerSubscriberTest : BehaviorSpec({ databrokerTransporter.updateRandomUint32Value(vssHeartRate.vssPath) then("The Observer should be triggered") { - eventually(1.seconds) { - friendlyVssNodeListener.updatedVssNodes.size shouldBe 2 + eventually(eventuallyConfiguration) { + friendlyVssNodeListener.updatedVssNodes.count { + val vssPath = it.vssPath + val value = it.value + vssPath == vssHeartRate.vssPath && value == randomIntValue + } shouldBe 1 } - - val count = friendlyVssNodeListener.updatedVssNodes - .count { it.value == randomIntValue } - count shouldBe 1 } } } @@ -285,12 +283,13 @@ class DataBrokerSubscriberTest : BehaviorSpec({ databrokerTransporter.updateRandomUint32Value(vssHeartRate.vssPath) then("The Observer is only notified once") { - eventually(1.seconds) { - nodeListenerMock.updatedVssNodes.size shouldBe 2 + eventually(eventuallyConfiguration) { + nodeListenerMock.updatedVssNodes.count { + val vssPath = it.vssPath + val value = it.value + vssPath == vssHeartRate.vssPath && value == randomIntValue + } shouldBe 1 } - - val count = nodeListenerMock.updatedVssNodes.count { it.value == randomIntValue } - count shouldBe 1 } } } diff --git a/test/src/main/java/org/eclipse/kuksa/test/extension/FloatExtension.kt b/test/src/main/java/org/eclipse/kuksa/test/extension/FloatExtension.kt new file mode 100644 index 00000000..87919705 --- /dev/null +++ b/test/src/main/java/org/eclipse/kuksa/test/extension/FloatExtension.kt @@ -0,0 +1,26 @@ +/* + * Copyright (c) 2023 - 2024 Contributors to the Eclipse Foundation + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + * + * SPDX-License-Identifier: Apache-2.0 + * + */ + +package org.eclipse.kuksa.test.extension + +import kotlin.math.abs + +fun Float.equals(other: Float, epsilon: Float): Boolean { + return abs(this - other) <= epsilon +} diff --git a/test/src/main/java/org/eclipse/kuksa/test/kotest/KotestProjectConfig.kt b/test/src/main/java/org/eclipse/kuksa/test/kotest/KotestProjectConfig.kt index 3e27f46b..310ffffe 100644 --- a/test/src/main/java/org/eclipse/kuksa/test/kotest/KotestProjectConfig.kt +++ b/test/src/main/java/org/eclipse/kuksa/test/kotest/KotestProjectConfig.kt @@ -19,7 +19,18 @@ package org.eclipse.kuksa.test.kotest +import io.kotest.assertions.nondeterministic.continuallyConfig +import io.kotest.assertions.nondeterministic.eventuallyConfig import io.kotest.core.config.AbstractProjectConfig +import kotlin.time.Duration.Companion.seconds + +val eventuallyConfiguration = eventuallyConfig { + duration = 1.seconds +} + +val continuallyConfiguration = continuallyConfig { + duration = 1.seconds +} // https://kotest.io/docs/framework/project-config.html object KotestProjectConfig : AbstractProjectConfig() { From cb5b88cba36a08b44c4dc0b450f92cc756bb4097 Mon Sep 17 00:00:00 2001 From: Andre Weber Date: Fri, 15 Mar 2024 09:22:21 +0100 Subject: [PATCH 5/8] chore: Execute all IntegrationTests on CI --- .github/actions/run-tests/action.yml | 16 +++------------- .../workflows/daily_integration_main-master.yaml | 2 +- 2 files changed, 4 insertions(+), 14 deletions(-) diff --git a/.github/actions/run-tests/action.yml b/.github/actions/run-tests/action.yml index 6b2a2b5d..07f69d89 100644 --- a/.github/actions/run-tests/action.yml +++ b/.github/actions/run-tests/action.yml @@ -14,19 +14,14 @@ inputs: kotest-tag: description: > The Kotest Tag to use when executing the tests. Check Tag.kt for valid options. Different Tags might require the - Databroker to be started in a different mode. Currently only unsecured mode (no tls, no authentication) is - supported. - default: '!CustomDatabroker' + Databroker to be started in a different mode. + default: '' runs: using: "composite" steps: - - name: "Run Docker Container of Databroker in detached mode" - run: docker run --pull=always --rm --publish 55556:55556/tcp --detach --name databroker ghcr.io/eclipse/kuksa.val/databroker:${{ inputs.databroker-version }} --port 55556 --insecure - shell: bash - - name: Run 'test' with Gradle Wrapper - run: ./gradlew test -Dkotest.tags="${{ inputs.kotest-tag}}" + run: ./gradlew test -Ddatabroker.tag="${{ inputs.databroker-version }}" -Dkotest.tags="${{ inputs.kotest-tag}}" shell: bash - name: Upload Test Reports @@ -51,8 +46,3 @@ runs: path: ${{ github.workspace }}/build/reports/jacoco/jacocoRootReport/html/* if-no-files-found: error retention-days: 14 - - - name: "Stop Docker Container of Databroker" - if: always() - run: docker stop databroker - shell: bash diff --git a/.github/workflows/daily_integration_main-master.yaml b/.github/workflows/daily_integration_main-master.yaml index e2edaf51..e6cf1179 100644 --- a/.github/workflows/daily_integration_main-master.yaml +++ b/.github/workflows/daily_integration_main-master.yaml @@ -21,4 +21,4 @@ jobs: with: upload-test-reports: true databroker-version: master - kotest-tag: "Integration & DefaultDatabroker" + kotest-tag: "Integration" From 9f7728c2037553a6058c703f557bb21ca3804b84 Mon Sep 17 00:00:00 2001 From: Andre Weber Date: Mon, 18 Mar 2024 14:39:11 +0100 Subject: [PATCH 6/8] chore: Align Class Names --- .../DataBrokerConnectorAuthenticationTest.kt | 12 ++++++------ .../databroker/DataBrokerConnectionTest.kt | 12 ++++++------ .../databroker/DataBrokerConnectorSecureTest.kt | 12 ++++++------ .../databroker/DataBrokerConnectorTest.kt | 12 ++++++------ .../databroker/DataBrokerTransporterTest.kt | 12 ++++++------ ...okerContainer.kt => DataBrokerDockerContainer.kt} | 10 +++++----- ...ainer.kt => InsecureDataBrokerDockerContainer.kt} | 6 +++--- ...ntainer.kt => SecureDataBrokerDockerContainer.kt} | 6 +++--- .../subscription/DataBrokerSubscriberTest.kt | 12 ++++++------ .../main/java/org/eclipse/kuksa/test/kotest/Tag.kt | 4 ++-- 10 files changed, 49 insertions(+), 49 deletions(-) rename kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/databroker/docker/{DockerDatabrokerContainer.kt => DataBrokerDockerContainer.kt} (93%) rename kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/databroker/docker/{DockerInsecureDatabrokerContainer.kt => InsecureDataBrokerDockerContainer.kt} (89%) rename kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/databroker/docker/{DockerSecureDatabrokerContainer.kt => SecureDataBrokerDockerContainer.kt} (93%) diff --git a/kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/authentication/DataBrokerConnectorAuthenticationTest.kt b/kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/authentication/DataBrokerConnectorAuthenticationTest.kt index 0b5b14c8..232d291a 100644 --- a/kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/authentication/DataBrokerConnectorAuthenticationTest.kt +++ b/kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/authentication/DataBrokerConnectorAuthenticationTest.kt @@ -28,8 +28,8 @@ import io.kotest.matchers.string.shouldContain import io.kotest.matchers.types.instanceOf import org.eclipse.kuksa.connectivity.databroker.DataBrokerConnectorProvider import org.eclipse.kuksa.connectivity.databroker.DataBrokerException -import org.eclipse.kuksa.connectivity.databroker.docker.DockerDatabrokerContainer -import org.eclipse.kuksa.connectivity.databroker.docker.DockerSecureDatabrokerContainer +import org.eclipse.kuksa.connectivity.databroker.docker.DataBrokerDockerContainer +import org.eclipse.kuksa.connectivity.databroker.docker.SecureDataBrokerDockerContainer import org.eclipse.kuksa.connectivity.databroker.request.FetchRequest import org.eclipse.kuksa.connectivity.databroker.request.SubscribeRequest import org.eclipse.kuksa.connectivity.databroker.request.UpdateRequest @@ -38,7 +38,7 @@ import org.eclipse.kuksa.proto.v1.Types import org.eclipse.kuksa.test.kotest.Authentication import org.eclipse.kuksa.test.kotest.Integration import org.eclipse.kuksa.test.kotest.Secure -import org.eclipse.kuksa.test.kotest.SecureDatabroker +import org.eclipse.kuksa.test.kotest.SecureDataBroker import org.eclipse.kuksa.test.kotest.eventuallyConfiguration import kotlin.random.Random import kotlin.random.nextInt @@ -48,11 +48,11 @@ import kotlin.random.nextInt // ./gradlew clean test -Dkotest.tags="Authentication" class DataBrokerConnectorAuthenticationTest : BehaviorSpec({ - tags(Integration, Authentication, Secure, SecureDatabroker) + tags(Integration, Authentication, Secure, SecureDataBroker) - var databrokerContainer: DockerDatabrokerContainer? = null + var databrokerContainer: DataBrokerDockerContainer? = null beforeSpec { - databrokerContainer = DockerSecureDatabrokerContainer() + databrokerContainer = SecureDataBrokerDockerContainer() .apply { start() } diff --git a/kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/databroker/DataBrokerConnectionTest.kt b/kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/databroker/DataBrokerConnectionTest.kt index b5682ad8..a0b85868 100644 --- a/kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/databroker/DataBrokerConnectionTest.kt +++ b/kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/databroker/DataBrokerConnectionTest.kt @@ -26,8 +26,8 @@ import io.kotest.matchers.shouldBe import io.mockk.every import io.mockk.mockk import io.mockk.verify -import org.eclipse.kuksa.connectivity.databroker.docker.DockerDatabrokerContainer -import org.eclipse.kuksa.connectivity.databroker.docker.DockerInsecureDatabrokerContainer +import org.eclipse.kuksa.connectivity.databroker.docker.DataBrokerDockerContainer +import org.eclipse.kuksa.connectivity.databroker.docker.InsecureDataBrokerDockerContainer import org.eclipse.kuksa.connectivity.databroker.listener.DisconnectListener import org.eclipse.kuksa.connectivity.databroker.request.FetchRequest import org.eclipse.kuksa.connectivity.databroker.request.SubscribeRequest @@ -40,7 +40,7 @@ import org.eclipse.kuksa.mocking.FriendlyVssPathListener import org.eclipse.kuksa.proto.v1.Types import org.eclipse.kuksa.test.extension.equals import org.eclipse.kuksa.test.kotest.Insecure -import org.eclipse.kuksa.test.kotest.InsecureDatabroker +import org.eclipse.kuksa.test.kotest.InsecureDataBroker import org.eclipse.kuksa.test.kotest.Integration import org.eclipse.kuksa.test.kotest.eventuallyConfiguration import org.eclipse.kuksa.vssNode.VssDriver @@ -48,11 +48,11 @@ import org.junit.jupiter.api.Assertions import kotlin.random.Random class DataBrokerConnectionTest : BehaviorSpec({ - tags(Integration, Insecure, InsecureDatabroker) + tags(Integration, Insecure, InsecureDataBroker) - var databrokerContainer: DockerDatabrokerContainer? = null + var databrokerContainer: DataBrokerDockerContainer? = null beforeSpec { - databrokerContainer = DockerInsecureDatabrokerContainer() + databrokerContainer = InsecureDataBrokerDockerContainer() .apply { start() } diff --git a/kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/databroker/DataBrokerConnectorSecureTest.kt b/kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/databroker/DataBrokerConnectorSecureTest.kt index e35b4207..02b60192 100644 --- a/kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/databroker/DataBrokerConnectorSecureTest.kt +++ b/kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/databroker/DataBrokerConnectorSecureTest.kt @@ -19,22 +19,22 @@ package org.eclipse.kuksa.connectivity.databroker import io.kotest.core.spec.style.BehaviorSpec -import org.eclipse.kuksa.connectivity.databroker.docker.DockerDatabrokerContainer -import org.eclipse.kuksa.connectivity.databroker.docker.DockerSecureDatabrokerContainer +import org.eclipse.kuksa.connectivity.databroker.docker.DataBrokerDockerContainer +import org.eclipse.kuksa.connectivity.databroker.docker.SecureDataBrokerDockerContainer import org.eclipse.kuksa.test.TestResourceFile import org.eclipse.kuksa.test.kotest.Integration import org.eclipse.kuksa.test.kotest.Secure -import org.eclipse.kuksa.test.kotest.SecureDatabroker +import org.eclipse.kuksa.test.kotest.SecureDataBroker import org.eclipse.kuksa.test.kotest.Tls import org.junit.jupiter.api.Assertions // run command: ./gradlew clean test -Dkotest.tags="Secure" class DataBrokerConnectorSecureTest : BehaviorSpec({ - tags(Integration, Secure, Tls, SecureDatabroker) + tags(Integration, Secure, Tls, SecureDataBroker) - var databrokerContainer: DockerDatabrokerContainer? = null + var databrokerContainer: DataBrokerDockerContainer? = null beforeSpec { - databrokerContainer = DockerSecureDatabrokerContainer() + databrokerContainer = SecureDataBrokerDockerContainer() .apply { start() } diff --git a/kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/databroker/DataBrokerConnectorTest.kt b/kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/databroker/DataBrokerConnectorTest.kt index 6d56a320..82ce9e59 100644 --- a/kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/databroker/DataBrokerConnectorTest.kt +++ b/kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/databroker/DataBrokerConnectorTest.kt @@ -21,18 +21,18 @@ package org.eclipse.kuksa.connectivity.databroker import io.kotest.assertions.throwables.shouldThrow import io.kotest.core.spec.style.BehaviorSpec import io.kotest.matchers.shouldNotBe -import org.eclipse.kuksa.connectivity.databroker.docker.DockerDatabrokerContainer -import org.eclipse.kuksa.connectivity.databroker.docker.DockerInsecureDatabrokerContainer +import org.eclipse.kuksa.connectivity.databroker.docker.DataBrokerDockerContainer +import org.eclipse.kuksa.connectivity.databroker.docker.InsecureDataBrokerDockerContainer import org.eclipse.kuksa.test.kotest.Insecure -import org.eclipse.kuksa.test.kotest.InsecureDatabroker +import org.eclipse.kuksa.test.kotest.InsecureDataBroker import org.eclipse.kuksa.test.kotest.Integration class DataBrokerConnectorTest : BehaviorSpec({ - tags(Integration, Insecure, InsecureDatabroker) + tags(Integration, Insecure, InsecureDataBroker) - var databrokerContainer: DockerDatabrokerContainer? = null + var databrokerContainer: DataBrokerDockerContainer? = null beforeSpec { - databrokerContainer = DockerInsecureDatabrokerContainer() + databrokerContainer = InsecureDataBrokerDockerContainer() .apply { start() } diff --git a/kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/databroker/DataBrokerTransporterTest.kt b/kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/databroker/DataBrokerTransporterTest.kt index 4544c6c1..733b868b 100644 --- a/kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/databroker/DataBrokerTransporterTest.kt +++ b/kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/databroker/DataBrokerTransporterTest.kt @@ -26,25 +26,25 @@ import io.kotest.matchers.shouldNotBe import io.kotest.matchers.types.instanceOf import io.mockk.mockk import io.mockk.verify -import org.eclipse.kuksa.connectivity.databroker.docker.DockerDatabrokerContainer -import org.eclipse.kuksa.connectivity.databroker.docker.DockerInsecureDatabrokerContainer +import org.eclipse.kuksa.connectivity.databroker.docker.DataBrokerDockerContainer +import org.eclipse.kuksa.connectivity.databroker.docker.InsecureDataBrokerDockerContainer import org.eclipse.kuksa.connectivity.databroker.listener.VssPathListener import org.eclipse.kuksa.extensions.updateRandomFloatValue import org.eclipse.kuksa.mocking.FriendlyVssPathListener import org.eclipse.kuksa.proto.v1.KuksaValV1 import org.eclipse.kuksa.proto.v1.Types import org.eclipse.kuksa.test.kotest.Insecure -import org.eclipse.kuksa.test.kotest.InsecureDatabroker +import org.eclipse.kuksa.test.kotest.InsecureDataBroker import org.eclipse.kuksa.test.kotest.Integration import org.eclipse.kuksa.test.kotest.eventuallyConfiguration import kotlin.random.Random class DataBrokerTransporterTest : BehaviorSpec({ - tags(Integration, Insecure, InsecureDatabroker) + tags(Integration, Insecure, InsecureDataBroker) - var databrokerContainer: DockerDatabrokerContainer? = null + var databrokerContainer: DataBrokerDockerContainer? = null beforeSpec { - databrokerContainer = DockerInsecureDatabrokerContainer() + databrokerContainer = InsecureDataBrokerDockerContainer() .apply { start() } diff --git a/kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/databroker/docker/DockerDatabrokerContainer.kt b/kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/databroker/docker/DataBrokerDockerContainer.kt similarity index 93% rename from kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/databroker/docker/DockerDatabrokerContainer.kt rename to kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/databroker/docker/DataBrokerDockerContainer.kt index 6f3a911a..49f05b22 100644 --- a/kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/databroker/docker/DockerDatabrokerContainer.kt +++ b/kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/databroker/docker/DataBrokerDockerContainer.kt @@ -60,7 +60,7 @@ private const val DEFAULT_DATABROKER_TAG = "master" * DATABROKER_TAG="0.4.1" ./gradlew clean build * ``` */ -abstract class DockerDatabrokerContainer( +abstract class DataBrokerDockerContainer( protected val containerName: String, protected val port: Int, ) { @@ -99,8 +99,8 @@ abstract class DockerDatabrokerContainer( fun start() { stop() - pullDatabrokerImage(databrokerTag) - val databrokerContainer = createDatabrokerContainer(databrokerTag) + pullImage(databrokerTag) + val databrokerContainer = createContainer(databrokerTag) startContainer(databrokerContainer.id) } @@ -119,14 +119,14 @@ abstract class DockerDatabrokerContainer( } } - private fun pullDatabrokerImage(tag: String) { + private fun pullImage(tag: String) { dockerClient.pullImageCmd(repository) .withTag(tag) .exec(PullImageResultCallback()) .awaitCompletion(30, TimeUnit.SECONDS) } - protected abstract fun createDatabrokerContainer(tag: String): CreateContainerResponse + protected abstract fun createContainer(tag: String): CreateContainerResponse private fun startContainer(containerId: String) { try { diff --git a/kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/databroker/docker/DockerInsecureDatabrokerContainer.kt b/kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/databroker/docker/InsecureDataBrokerDockerContainer.kt similarity index 89% rename from kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/databroker/docker/DockerInsecureDatabrokerContainer.kt rename to kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/databroker/docker/InsecureDataBrokerDockerContainer.kt index a9f9a5e1..c6e9ab1e 100644 --- a/kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/databroker/docker/DockerInsecureDatabrokerContainer.kt +++ b/kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/databroker/docker/InsecureDataBrokerDockerContainer.kt @@ -24,13 +24,13 @@ import org.eclipse.kuksa.connectivity.databroker.DATABROKER_CONTAINER_NAME import org.eclipse.kuksa.connectivity.databroker.DATABROKER_PORT // no tls, no authentication -class DockerInsecureDatabrokerContainer( +class InsecureDataBrokerDockerContainer( containerName: String = DATABROKER_CONTAINER_NAME, port: Int = DATABROKER_PORT, -) : DockerDatabrokerContainer(containerName, port) { +) : DataBrokerDockerContainer(containerName, port) { @Suppress("ArgumentListWrapping", "ktlint:standard:argument-list-wrapping") // better key-value pair readability - override fun createDatabrokerContainer(tag: String): CreateContainerResponse { + override fun createContainer(tag: String): CreateContainerResponse { return dockerClient.createContainerCmd("$repository:$tag") .withName(containerName) .withHostConfig(hostConfig) diff --git a/kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/databroker/docker/DockerSecureDatabrokerContainer.kt b/kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/databroker/docker/SecureDataBrokerDockerContainer.kt similarity index 93% rename from kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/databroker/docker/DockerSecureDatabrokerContainer.kt rename to kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/databroker/docker/SecureDataBrokerDockerContainer.kt index 634c749c..733c0121 100644 --- a/kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/databroker/docker/DockerSecureDatabrokerContainer.kt +++ b/kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/databroker/docker/SecureDataBrokerDockerContainer.kt @@ -29,10 +29,10 @@ import org.eclipse.kuksa.connectivity.databroker.DATABROKER_PORT import org.eclipse.kuksa.test.TestResourceFile // tls enabled, authentication enabled -class DockerSecureDatabrokerContainer( +class SecureDataBrokerDockerContainer( containerName: String = DATABROKER_CONTAINER_NAME, port: Int = DATABROKER_PORT, -) : DockerDatabrokerContainer(containerName, port) { +) : DataBrokerDockerContainer(containerName, port) { private val authenticationFolder = TestResourceFile("authentication").toString() private val authenticationMount = "/resources/authentication" @@ -47,7 +47,7 @@ class DockerSecureDatabrokerContainer( ) @Suppress("ArgumentListWrapping", "ktlint:standard:argument-list-wrapping") // better key-value pair readability - override fun createDatabrokerContainer(tag: String): CreateContainerResponse { + override fun createContainer(tag: String): CreateContainerResponse { return dockerClient.createContainerCmd("$repository:$tag") .withName(containerName) .withHostConfig(hostConfig) diff --git a/kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/databroker/subscription/DataBrokerSubscriberTest.kt b/kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/databroker/subscription/DataBrokerSubscriberTest.kt index 14b47d78..d065c32a 100644 --- a/kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/databroker/subscription/DataBrokerSubscriberTest.kt +++ b/kuksa-sdk/src/test/kotlin/org/eclipse/kuksa/connectivity/databroker/subscription/DataBrokerSubscriberTest.kt @@ -29,8 +29,8 @@ import io.mockk.mockk import io.mockk.verify import org.eclipse.kuksa.connectivity.databroker.DataBrokerConnectorProvider import org.eclipse.kuksa.connectivity.databroker.DataBrokerTransporter -import org.eclipse.kuksa.connectivity.databroker.docker.DockerDatabrokerContainer -import org.eclipse.kuksa.connectivity.databroker.docker.DockerInsecureDatabrokerContainer +import org.eclipse.kuksa.connectivity.databroker.docker.DataBrokerDockerContainer +import org.eclipse.kuksa.connectivity.databroker.docker.InsecureDataBrokerDockerContainer import org.eclipse.kuksa.connectivity.databroker.listener.VssPathListener import org.eclipse.kuksa.extensions.toggleBoolean import org.eclipse.kuksa.extensions.updateRandomFloatValue @@ -41,18 +41,18 @@ import org.eclipse.kuksa.pattern.listener.MultiListener import org.eclipse.kuksa.pattern.listener.count import org.eclipse.kuksa.proto.v1.Types import org.eclipse.kuksa.test.kotest.Insecure -import org.eclipse.kuksa.test.kotest.InsecureDatabroker +import org.eclipse.kuksa.test.kotest.InsecureDataBroker import org.eclipse.kuksa.test.kotest.Integration import org.eclipse.kuksa.test.kotest.continuallyConfiguration import org.eclipse.kuksa.test.kotest.eventuallyConfiguration import org.eclipse.kuksa.vssNode.VssDriver class DataBrokerSubscriberTest : BehaviorSpec({ - tags(Integration, Insecure, InsecureDatabroker) + tags(Integration, Insecure, InsecureDataBroker) - var databrokerContainer: DockerDatabrokerContainer? = null + var databrokerContainer: DataBrokerDockerContainer? = null beforeSpec { - databrokerContainer = DockerInsecureDatabrokerContainer() + databrokerContainer = InsecureDataBrokerDockerContainer() .apply { start() } diff --git a/test/src/main/java/org/eclipse/kuksa/test/kotest/Tag.kt b/test/src/main/java/org/eclipse/kuksa/test/kotest/Tag.kt index a9dee663..e4906db1 100644 --- a/test/src/main/java/org/eclipse/kuksa/test/kotest/Tag.kt +++ b/test/src/main/java/org/eclipse/kuksa/test/kotest/Tag.kt @@ -23,10 +23,10 @@ import io.kotest.core.NamedTag val Integration = NamedTag("Integration") -val InsecureDatabroker = NamedTag("InsecureDatabroker") // no tls, no authentication +val InsecureDataBroker = NamedTag("InsecureDataBroker") // no tls, no authentication val Insecure = NamedTag("Insecure") -val SecureDatabroker = NamedTag("SecureDatabroker") +val SecureDataBroker = NamedTag("SecureDataBroker") val Secure = NamedTag("Secure") // tls enabled, authentication enabled val Authentication = NamedTag("Authentication") val Tls = NamedTag("Tls") From 3b615666646601fb9ac5d40fb76a13982fd6581c Mon Sep 17 00:00:00 2001 From: Andre Weber Date: Mon, 18 Mar 2024 14:39:33 +0100 Subject: [PATCH 7/8] chore: Adapt Shared Run Configuration --- .run/kuksa-sdk_IntegrationTests.run.xml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/.run/kuksa-sdk_IntegrationTests.run.xml b/.run/kuksa-sdk_IntegrationTests.run.xml index 12bc49be..0bad6f77 100644 --- a/.run/kuksa-sdk_IntegrationTests.run.xml +++ b/.run/kuksa-sdk_IntegrationTests.run.xml @@ -13,7 +13,7 @@ -