From a570a09594daa8d87d4288a2fc491d3e04794283 Mon Sep 17 00:00:00 2001 From: andacata <1506402+andacata@users.noreply.github.com> Date: Tue, 9 Jan 2024 17:52:42 +0100 Subject: [PATCH] feat: Add validation of required endpoints --- .../services/CredentialsClientService.kt | 7 ++- .../credentials/services/CredentialsCommon.kt | 52 +++++++++++++++++++ .../services/CredentialsServerService.kt | 43 +-------------- .../credentials/CredentialsReceiver.kt | 10 ++-- .../samples/credentials/CredentialsSender.kt | 5 +- .../CredentialsIntegrationTests.kt | 42 ++++++++------- 6 files changed, 93 insertions(+), 66 deletions(-) create mode 100644 ocpi-toolkit-2.2.1/src/main/kotlin/com/izivia/ocpi/toolkit/modules/credentials/services/CredentialsCommon.kt diff --git a/ocpi-toolkit-2.2.1/src/main/kotlin/com/izivia/ocpi/toolkit/modules/credentials/services/CredentialsClientService.kt b/ocpi-toolkit-2.2.1/src/main/kotlin/com/izivia/ocpi/toolkit/modules/credentials/services/CredentialsClientService.kt index 1a89c8be..ba85a8f9 100644 --- a/ocpi-toolkit-2.2.1/src/main/kotlin/com/izivia/ocpi/toolkit/modules/credentials/services/CredentialsClientService.kt +++ b/ocpi-toolkit-2.2.1/src/main/kotlin/com/izivia/ocpi/toolkit/modules/credentials/services/CredentialsClientService.kt @@ -8,6 +8,7 @@ import com.izivia.ocpi.toolkit.modules.credentials.repositories.PartnerRepositor import com.izivia.ocpi.toolkit.modules.versions.VersionDetailsClient import com.izivia.ocpi.toolkit.modules.versions.VersionsClient import com.izivia.ocpi.toolkit.modules.versions.domain.Endpoint +import com.izivia.ocpi.toolkit.modules.versions.domain.InterfaceRole import com.izivia.ocpi.toolkit.modules.versions.domain.ModuleID import com.izivia.ocpi.toolkit.modules.versions.domain.parseVersionNumber import com.izivia.ocpi.toolkit.modules.versions.repositories.VersionsRepository @@ -28,6 +29,7 @@ import com.izivia.ocpi.toolkit.transport.TransportClientBuilder * @property clientCredentialsRoleRepository client's repository to retrieve its role * @property serverVersionsEndpointUrl the versions endpoint url of the server (for the client to retrieve endpoints) * @property transportClientBuilder used to build a transport (will be used to create CredentialClient to make calls) + * @property requiredOtherPartEndpointsProvider the endpoints this client expects from the other part to provide */ class CredentialsClientService( private val clientVersionsEndpointUrl: String, @@ -35,7 +37,8 @@ class CredentialsClientService( private val clientVersionsRepository: VersionsRepository, private val clientCredentialsRoleRepository: CredentialsRoleRepository, private val serverVersionsEndpointUrl: String, - private val transportClientBuilder: TransportClientBuilder + private val transportClientBuilder: TransportClientBuilder, + private val requiredOtherPartEndpointsProvider: suspend () -> Map> ) { suspend fun get(): Credentials = clientPartnerRepository .getCredentialsClientToken(partnerUrl = serverVersionsEndpointUrl) @@ -226,6 +229,8 @@ class CredentialsClientService( it.data ?: throw OcpiResponseException(it.status_code, it.status_message ?: "unknown") } + CredentialsCommon.checkRequiredEndpoints(requiredOtherPartEndpointsProvider(), versionDetails.endpoints) + // Store version & endpoint return clientPartnerRepository.saveEndpoints( partnerUrl = serverVersionsEndpointUrl, diff --git a/ocpi-toolkit-2.2.1/src/main/kotlin/com/izivia/ocpi/toolkit/modules/credentials/services/CredentialsCommon.kt b/ocpi-toolkit-2.2.1/src/main/kotlin/com/izivia/ocpi/toolkit/modules/credentials/services/CredentialsCommon.kt new file mode 100644 index 00000000..6b3338cf --- /dev/null +++ b/ocpi-toolkit-2.2.1/src/main/kotlin/com/izivia/ocpi/toolkit/modules/credentials/services/CredentialsCommon.kt @@ -0,0 +1,52 @@ +package com.izivia.ocpi.toolkit.modules.credentials.services + +import com.izivia.ocpi.toolkit.common.OcpiServerNoMatchingEndpointsException +import com.izivia.ocpi.toolkit.modules.versions.domain.Endpoint +import com.izivia.ocpi.toolkit.modules.versions.domain.InterfaceRole +import com.izivia.ocpi.toolkit.modules.versions.domain.ModuleID + +class CredentialsCommon { + companion object { + fun checkRequiredEndpoints( + requiredEndpoints: Map>, + actualEndpoints: List + ) { + if (requiredEndpoints.isEmpty()) { + return + } + + val requiredOtherPartAsReceiverEndpoints = requiredEndpoints[InterfaceRole.RECEIVER] ?: listOf() + val requiredOtherPartAsSenderEndpoints = requiredEndpoints[InterfaceRole.SENDER] ?: listOf() + + actualEndpoints + .find { it.identifier == ModuleID.credentials } + .let { + it ?: throw OcpiServerNoMatchingEndpointsException( + "${ModuleID.credentials} other part endpoint missing" + ) + } + + checkRequiredRoleEndpoints(actualEndpoints, requiredOtherPartAsReceiverEndpoints, InterfaceRole.RECEIVER) + checkRequiredRoleEndpoints(actualEndpoints, requiredOtherPartAsSenderEndpoints, InterfaceRole.SENDER) + } + + private fun checkRequiredRoleEndpoints( + endpoints: List, + requiredEndpoints: List, + role: InterfaceRole + ) { + for (requiredEndpoint in requiredEndpoints) { + endpoints + .find { + requiredEndpoint == ModuleID.credentials || + (it.role == role && it.identifier == requiredEndpoint) + } + .let { + it ?: throw OcpiServerNoMatchingEndpointsException( + "${requiredEndpoint.name} as ${role.name} other part endpoint missing" + ) + } + } + } + } +} diff --git a/ocpi-toolkit-2.2.1/src/main/kotlin/com/izivia/ocpi/toolkit/modules/credentials/services/CredentialsServerService.kt b/ocpi-toolkit-2.2.1/src/main/kotlin/com/izivia/ocpi/toolkit/modules/credentials/services/CredentialsServerService.kt index 91f4d0ec..c5b7218c 100644 --- a/ocpi-toolkit-2.2.1/src/main/kotlin/com/izivia/ocpi/toolkit/modules/credentials/services/CredentialsServerService.kt +++ b/ocpi-toolkit-2.2.1/src/main/kotlin/com/izivia/ocpi/toolkit/modules/credentials/services/CredentialsServerService.kt @@ -16,7 +16,7 @@ class CredentialsServerService( private val credentialsRoleRepository: CredentialsRoleRepository, private val transportClientBuilder: TransportClientBuilder, private val serverVersionsUrlProvider: suspend () -> String, - private val requiredClientEndpointsProvider: suspend () -> Map> + private val requiredOtherPartEndpointsProvider: suspend () -> Map> ) : CredentialsInterface { override suspend fun get( @@ -201,50 +201,11 @@ class CredentialsServerService( ) } - checkRequiredClientEndpoints(versionDetail.endpoints) + CredentialsCommon.checkRequiredEndpoints(requiredOtherPartEndpointsProvider(), versionDetail.endpoints) partnerRepository.saveEndpoints(partnerUrl = credentials.url, endpoints = versionDetail.endpoints) } - private suspend fun checkRequiredClientEndpoints(endpoints: List) { - val requiredClientEndpoints = requiredClientEndpointsProvider() - - if (requiredClientEndpoints.isEmpty()) { - return - } - - val requiredClientAsReceiverEndpoints = requiredClientEndpoints[InterfaceRole.RECEIVER.name] ?: listOf() - val requiredClientAsSenderEndpoints = requiredClientEndpoints[InterfaceRole.SENDER.name] ?: listOf() - - endpoints - .find { it.identifier == ModuleID.credentials } - .let { - it ?: throw OcpiServerNoMatchingEndpointsException("${ModuleID.credentials} client endpoint missing") - } - - checkRequiredRoleEndpoints(endpoints, requiredClientAsReceiverEndpoints, InterfaceRole.RECEIVER) - checkRequiredRoleEndpoints(endpoints, requiredClientAsSenderEndpoints, InterfaceRole.SENDER) - } - - private suspend fun checkRequiredRoleEndpoints( - endpoints: List, - requiredEndpoints: List, - role: InterfaceRole - ) { - for (requiredEndpoint in requiredEndpoints) { - endpoints - .find { - requiredEndpoint == ModuleID.credentials - || (it.role == role && it.identifier == requiredEndpoint) - } - .let { - it ?: throw OcpiServerNoMatchingEndpointsException( - "${requiredEndpoint.name} as ${role.name} client endpoint missing" - ) - } - } - } - private suspend fun getCredentials(serverToken: String): Credentials = Credentials( token = serverToken, url = serverVersionsUrlProvider(), diff --git a/ocpi-toolkit-2.2.1/src/test/kotlin/com/izivia/ocpi/toolkit/samples/credentials/CredentialsReceiver.kt b/ocpi-toolkit-2.2.1/src/test/kotlin/com/izivia/ocpi/toolkit/samples/credentials/CredentialsReceiver.kt index 7ca7eea5..973684cc 100644 --- a/ocpi-toolkit-2.2.1/src/test/kotlin/com/izivia/ocpi/toolkit/samples/credentials/CredentialsReceiver.kt +++ b/ocpi-toolkit-2.2.1/src/test/kotlin/com/izivia/ocpi/toolkit/samples/credentials/CredentialsReceiver.kt @@ -32,6 +32,8 @@ fun main() { secureFilter = receiverPlatformRepository::checkToken ) + val requiredOtherPartEndpoints = mapOf(InterfaceRole.RECEIVER to listOf(ModuleID.credentials)) + runBlocking { CredentialsServer( service = CredentialsServerService( @@ -40,15 +42,15 @@ fun main() { override suspend fun getCredentialsRoles(): List = listOf( CredentialRole( role = Role.EMSP, - business_details = BusinessDetails(name = "Receiver", website = null, logo = null), - party_id = "DEF", - country_code = "FR" + businessDetails = BusinessDetails(name = "Receiver", website = null, logo = null), + partyId = "DEF", + countryCode = "FR" ) ) }, transportClientBuilder = Http4kTransportClientBuilder(), serverVersionsUrlProvider = { receiverVersionsUrl }, - requiredClientEndpointsProvider = { mapOf(InterfaceRole.RECEIVER.name to listOf(ModuleID.credentials)) } + requiredOtherPartEndpointsProvider = { requiredOtherPartEndpoints } ) ).registerOn(receiverServer) VersionsServer( diff --git a/ocpi-toolkit-2.2.1/src/test/kotlin/com/izivia/ocpi/toolkit/samples/credentials/CredentialsSender.kt b/ocpi-toolkit-2.2.1/src/test/kotlin/com/izivia/ocpi/toolkit/samples/credentials/CredentialsSender.kt index c450665d..aaefb81f 100644 --- a/ocpi-toolkit-2.2.1/src/test/kotlin/com/izivia/ocpi/toolkit/samples/credentials/CredentialsSender.kt +++ b/ocpi-toolkit-2.2.1/src/test/kotlin/com/izivia/ocpi/toolkit/samples/credentials/CredentialsSender.kt @@ -8,6 +8,8 @@ import com.izivia.ocpi.toolkit.modules.credentials.services.CredentialsClientSer import com.izivia.ocpi.toolkit.modules.locations.domain.BusinessDetails import com.izivia.ocpi.toolkit.modules.versions.VersionDetailsServer import com.izivia.ocpi.toolkit.modules.versions.VersionsServer +import com.izivia.ocpi.toolkit.modules.versions.domain.InterfaceRole +import com.izivia.ocpi.toolkit.modules.versions.domain.ModuleID import com.izivia.ocpi.toolkit.modules.versions.services.VersionDetailsService import com.izivia.ocpi.toolkit.modules.versions.services.VersionsService import com.izivia.ocpi.toolkit.samples.common.* @@ -61,7 +63,8 @@ fun main() { ) }, serverVersionsEndpointUrl = receiverVersionsUrl, - transportClientBuilder = Http4kTransportClientBuilder() + transportClientBuilder = Http4kTransportClientBuilder(), + requiredOtherPartEndpointsProvider = { mapOf(InterfaceRole.RECEIVER to listOf(ModuleID.credentials)) } ) runBlocking { diff --git a/ocpi-toolkit-2.2.1/src/test/kotlin/com/izivia/ocpi/toolkit/tests/integration/CredentialsIntegrationTests.kt b/ocpi-toolkit-2.2.1/src/test/kotlin/com/izivia/ocpi/toolkit/tests/integration/CredentialsIntegrationTests.kt index 56f2d1ee..c48d81bc 100644 --- a/ocpi-toolkit-2.2.1/src/test/kotlin/com/izivia/ocpi/toolkit/tests/integration/CredentialsIntegrationTests.kt +++ b/ocpi-toolkit-2.2.1/src/test/kotlin/com/izivia/ocpi/toolkit/tests/integration/CredentialsIntegrationTests.kt @@ -45,7 +45,7 @@ class CredentialsIntegrationTests : BaseServerIntegrationTest() { private var database: MongoDatabase? = null - private fun setupReceiver(requiredEndpoints: Map> = mapOf()): ServerSetupResult { + private fun setupReceiver(requiredEndpoints: Map> = mapOf()): ServerSetupResult { if (database == null) database = buildDBClient().getDatabase("ocpi-2-2-1-tests") val receiverPartnerCollection = database!! .getCollection("receiver-server-${UUID.randomUUID()}") @@ -68,15 +68,15 @@ class CredentialsIntegrationTests : BaseServerIntegrationTest() { override suspend fun getCredentialsRoles(): List = listOf( CredentialRole( role = Role.EMSP, - business_details = BusinessDetails(name = "Receiver", website = null, logo = null), - party_id = "DEF", - country_code = "FR" + businessDetails = BusinessDetails(name = "Receiver", website = null, logo = null), + partyId = "DEF", + countryCode = "FR" ) ) }, transportClientBuilder = Http4kTransportClientBuilder(), serverVersionsUrlProvider = { receiverServerVersionsUrl }, - requiredClientEndpointsProvider = { requiredEndpoints } + requiredOtherPartEndpointsProvider = { requiredEndpoints } ) ).registerOn(receiverServer) VersionsServer( @@ -133,25 +133,27 @@ class CredentialsIntegrationTests : BaseServerIntegrationTest() { private fun setupCredentialsSenderClient( senderServerSetupResult: ServerSetupResult, - receiverServerSetupResult: ServerSetupResult + receiverServerSetupResult: ServerSetupResult, + requiredEndpoints: Map> = mapOf() ): CredentialsClientService { // Setup sender (client) return CredentialsClientService( clientVersionsEndpointUrl = senderServerSetupResult.versionsEndpoint, - clientPlatformRepository = PartnerMongoRepository(collection = senderServerSetupResult.partnerCollection), + clientPartnerRepository = PartnerMongoRepository(collection = senderServerSetupResult.partnerCollection), clientVersionsRepository = VersionsCacheRepository(baseUrl = senderServerSetupResult.transport.baseUrl), clientCredentialsRoleRepository = object : CredentialsRoleRepository { override suspend fun getCredentialsRoles(): List = listOf( CredentialRole( role = Role.CPO, - business_details = BusinessDetails(name = "Sender", website = null, logo = null), - party_id = "ABC", - country_code = "FR" + businessDetails = BusinessDetails(name = "Sender", website = null, logo = null), + partyId = "ABC", + countryCode = "FR" ) ) }, serverVersionsEndpointUrl = receiverServerSetupResult.versionsEndpoint, - transportClientBuilder = Http4kTransportClientBuilder() + transportClientBuilder = Http4kTransportClientBuilder(), + requiredOtherPartEndpointsProvider = { requiredEndpoints } ) } @@ -256,11 +258,11 @@ class CredentialsIntegrationTests : BaseServerIntegrationTest() { fun `should not properly run registration process because required endpoints are missing`() { val receiverServer = setupReceiver( mapOf( - InterfaceRole.RECEIVER.name to listOf( + InterfaceRole.RECEIVER to listOf( ModuleID.credentials, ModuleID.locations ), - InterfaceRole.SENDER.name to listOf( + InterfaceRole.SENDER to listOf( ModuleID.chargingprofiles ) ) @@ -292,16 +294,18 @@ class CredentialsIntegrationTests : BaseServerIntegrationTest() { get { statusCode } .isEqualTo(OcpiStatus.SERVER_NO_MATCHING_ENDPOINTS.code) } - } @Test fun `should properly run registration process then correct get credentials from receiver`() { - val receiverServer = setupReceiver(mapOf( - InterfaceRole.RECEIVER.name to listOf( - ModuleID.credentials, - ModuleID.locations) - )) + val receiverServer = setupReceiver( + mapOf( + InterfaceRole.RECEIVER to listOf( + ModuleID.credentials, + ModuleID.locations + ) + ) + ) val senderServer = setupSender() val credentialsClientService = setupCredentialsSenderClient(