Skip to content

Commit

Permalink
feat(pool): #658 consider in validation that if the BPN request ident…
Browse files Browse the repository at this point in the history
…ifier is known to the pool, the task has to be considered as update,

- validate duplicates over all addresses and all legal entities
- move function to create and compare business partner in own files
  • Loading branch information
rainer-exxcellent authored and nicoprow committed Jan 11, 2024
1 parent 3d5a9c6 commit 6ed7859
Show file tree
Hide file tree
Showing 9 changed files with 942 additions and 289 deletions.
Original file line number Diff line number Diff line change
Expand Up @@ -81,3 +81,10 @@ enum class AddressUpdateError : ErrorCode {
IdentifierNotFound,
AddressDuplicateIdentifier,
}

@Schema(description = "OrchestratorError")
enum class OrchestratorError : ErrorCode {
AddressRegionNotFound,
AddressIdentifierNotFound,
AddressDuplicateIdentifier,
}
Original file line number Diff line number Diff line change
Expand Up @@ -66,7 +66,6 @@ class BusinessPartnerEquivalenceService {
}

private fun isEquivalent(original: LegalEntityState, updated: LegalEntityState): Boolean =
original.description != updated.description ||
original.validFrom != updated.validFrom ||
original.validTo != updated.validTo ||
original.type != updated.type
Expand Down

Large diffs are not rendered by default.

Original file line number Diff line number Diff line change
Expand Up @@ -37,27 +37,37 @@ class TaskEntryBpnMapping(taskEntries: List<TaskStepReservationEntryDto>, bpnReq
private fun readRequestMappings(taskEntries: List<TaskStepReservationEntryDto>, bpnRequestIdentifierRepository: BpnRequestIdentifierRepository ): MutableMap<String, String> {

val references = taskEntries.mapNotNull { it.businessPartner.legalEntity?.bpnLReference } +
taskEntries.mapNotNull { it.businessPartner.legalEntity?.legalAddress?.bpnAReference } +
taskEntries.mapNotNull { it.businessPartner.site?.bpnSReference } +
taskEntries.mapNotNull { it.businessPartner.site?.mainAddress?.bpnAReference } +
taskEntries.mapNotNull { it.businessPartner.address?.bpnAReference }

val usedRequestIdentifiers: Collection<String> = references.map { it.referenceValue }

val mappings = bpnRequestIdentifierRepository.findDistinctByRequestIdentifierIn(usedRequestIdentifiers)
val bpnByRequestIdentifier = mappings
.map { it.requestIdentifier to it.bpn }
.toMap()
val bpnByRequestIdentifier = mappings.associate { it.requestIdentifier to it.bpn }
return bpnByRequestIdentifier.toMutableMap()
}

fun getBpn(bpnLReference: BpnReferenceDto): String? {
fun getBpn(bpnReference: BpnReferenceDto?): String? {

return if (bpnLReference.referenceType == BpnReferenceType.BpnRequestIdentifier) {
bpnByRequestIdentifier[bpnLReference.referenceValue]
return if(bpnReference == null) {
null
} else if (bpnReference.referenceType == BpnReferenceType.BpnRequestIdentifier) {
bpnByRequestIdentifier[bpnReference.referenceValue]
} else {
bpnLReference.referenceValue
bpnReference.referenceValue
}
}

fun hasBpnFor(bpnReference: BpnReferenceDto?): Boolean {

return bpnReference != null && (bpnReference.referenceType == BpnReferenceType.Bpn
|| (bpnReference.referenceType == BpnReferenceType.BpnRequestIdentifier
&& bpnByRequestIdentifier.containsKey(bpnReference.referenceValue)))
}


fun addMapping(bpnLReference: BpnReferenceDto, bpn: String) {

createdBpnByRequestIdentifier[bpnLReference.referenceValue] = bpn
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -77,7 +77,10 @@ class TaskStepBuildService(
if (businessPartnerDto.site != null) {
siteEntity = upsertSite(businessPartnerDto.site, legalEntity, taskEntryBpnMapping, businessPartnerDto)
siteResult = businessPartnerDto.site!!.copy(
bpnSReference = BpnReferenceDto(referenceValue = siteEntity.bpn, referenceType = BpnReferenceType.Bpn)
bpnSReference = BpnReferenceDto(referenceValue = siteEntity.bpn, referenceType = BpnReferenceType.Bpn),
mainAddress = businessPartnerDto.site!!.mainAddress!!.copy(
bpnAReference = BpnReferenceDto(referenceValue = siteEntity.mainAddress.bpn, referenceType = BpnReferenceType.Bpn)
)
)
genericBpnA = siteEntity.mainAddress.bpn
}
Expand Down Expand Up @@ -344,7 +347,7 @@ class TaskStepBuildService(
if (siteDto.hasChanged == true) {
val createSite = BusinessPartnerBuildService.createSite(siteDto, updateSite.bpn, legalEntity)
val siteMainAddress =
if (genericBusinessPartner.generic.postalAddress.addressType == AddressType.LegalAndSiteMainAddress)
if (genericBusinessPartner.generic.address.addressType == AddressType.LegalAndSiteMainAddress)
legalEntity.legalAddress
else
createLogisticAddress(mainAddress, taskEntryBpnMapping)
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -63,7 +63,7 @@ class TaskStepFetchAndReserveService(

val taskEntryBpnMapping = TaskEntryBpnMapping(taskEntries, bpnRequestIdentifierRepository)

val invalidTaskResultsByTaskEntry = validateTasks(taskEntries)
val invalidTaskResultsByTaskEntry = validateTasks(taskEntries, taskEntryBpnMapping)

val taskResults = taskEntries.map {

Expand Down Expand Up @@ -91,23 +91,10 @@ class TaskStepFetchAndReserveService(
}
}

private fun validateTasks(
taskEntries: List<TaskStepReservationEntryDto>
private fun validateTasks( taskEntries: List<TaskStepReservationEntryDto>, requestMappings: TaskEntryBpnMapping
): Map<TaskStepReservationEntryDto, TaskStepResultEntryDto?> {

val legalEntitiesToCreate = taskEntries
.filter { it.businessPartner.legalEntity?.bpnLReference?.referenceType == BpnReferenceType.BpnRequestIdentifier }
val legalEntitiesToUpdate = taskEntries
.filter { it.businessPartner.legalEntity?.bpnLReference?.referenceType == BpnReferenceType.Bpn }
val sitesToCreate = taskEntries
.filter { it.businessPartner.site?.bpnSReference?.referenceType == BpnReferenceType.BpnRequestIdentifier }
val sitesToUpdate = taskEntries
.filter { it.businessPartner.site?.bpnSReference?.referenceType == BpnReferenceType.Bpn }

val validationErrorsByTaskEntry = requestValidationService.validateLegalEntitiesToCreateFromOrchestrator(legalEntitiesToCreate) +
requestValidationService.validateLegalEntitiesToUpdateFromOrchestrator(legalEntitiesToUpdate)+
requestValidationService.validateSitesToCreateFromOrchestrator(sitesToCreate) +
requestValidationService.validateSitesToUpdateFromOrchestrator(sitesToUpdate)
val validationErrorsByTaskEntry = requestValidationService.validateTasksFromOrchestrator(taskEntries, requestMappings)

val taskResultsByTaskEntry = taskEntries
.associateWith { taskEntry ->
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,183 @@
/*******************************************************************************
* Copyright (c) 2021,2023 Contributors to the Eclipse Foundation
*
* See the NOTICE file(s) distributed with this work for additional
* information regarding copyright ownership.
*
* This program and the accompanying materials are made available under the
* terms of the Apache License, Version 2.0 which is available at
* https://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.tractusx.bpdm.pool.service

import org.assertj.core.api.Assertions
import org.eclipse.tractusx.bpdm.common.dto.ILegalEntityStateDto
import org.eclipse.tractusx.bpdm.common.dto.ISiteStateDto
import org.eclipse.tractusx.bpdm.pool.api.model.*
import org.eclipse.tractusx.bpdm.pool.api.model.response.LegalEntityWithLegalAddressVerboseDto
import org.eclipse.tractusx.bpdm.pool.api.model.response.SiteWithMainAddressVerboseDto
import org.eclipse.tractusx.orchestrator.api.model.AddressIdentifierDto
import org.eclipse.tractusx.orchestrator.api.model.AddressStateDto
import org.eclipse.tractusx.orchestrator.api.model.LegalEntityClassificationDto
import org.eclipse.tractusx.orchestrator.api.model.LegalEntityDto
import org.eclipse.tractusx.orchestrator.api.model.LegalEntityIdentifierDto
import org.eclipse.tractusx.orchestrator.api.model.LogisticAddressDto
import org.eclipse.tractusx.orchestrator.api.model.SiteDto
import java.time.LocalDateTime
import java.time.temporal.ChronoUnit
import java.util.function.BiPredicate

fun compareLegalEntity(verboseRequest: LegalEntityWithLegalAddressVerboseDto, legalEntity: LegalEntityDto?) {

val verboseLegalEntity = verboseRequest.legalEntity

Assertions.assertThat(verboseLegalEntity.legalShortName).isEqualTo(legalEntity?.legalShortName)
Assertions.assertThat(verboseLegalEntity.legalFormVerbose?.technicalKey).isEqualTo(legalEntity?.legalForm)
compareStates(verboseLegalEntity.states, legalEntity?.states)
compareClassifications(verboseLegalEntity.classifications, legalEntity?.classifications)
compareIdentifiers(verboseLegalEntity.identifiers, legalEntity?.identifiers)

val verboseLegalAddress = verboseRequest.legalAddress
Assertions.assertThat(verboseLegalAddress.bpnLegalEntity).isEqualTo(legalEntity?.bpnLReference?.referenceValue)
Assertions.assertThat(verboseLegalAddress.isLegalAddress).isTrue()
compareLogisticAddress(verboseLegalAddress, legalEntity?.legalAddress)
}

fun compareSite(verboseRequest: SiteWithMainAddressVerboseDto, site: SiteDto?) {

val verboseSite = verboseRequest.site

Assertions.assertThat(verboseSite.name).isEqualTo(site?.name)
Assertions.assertThat(verboseSite.bpns).isEqualTo(site?.bpnSReference?.referenceValue)
compareSiteStates(verboseSite.states, site?.states)

val verboseMainAddress = verboseRequest.mainAddress
Assertions.assertThat(verboseMainAddress.bpnSite).isEqualTo(site?.bpnSReference?.referenceValue)
val mainAddress = site?.mainAddress
Assertions.assertThat(verboseMainAddress.isMainAddress).isTrue()
compareLogisticAddress(verboseMainAddress, mainAddress)
}

fun compareLogisticAddress(verboseAddress: LogisticAddressVerboseDto, address: LogisticAddressDto?) {

Assertions.assertThat(verboseAddress.name).isEqualTo(address?.name)
compareAddressStates(verboseAddress.states, address?.states)
compareAddressIdentifiers(verboseAddress.identifiers, address?.identifiers)


val verbosePhysicalAddress = verboseAddress.physicalPostalAddress
val physicalAddress = address?.physicalPostalAddress
Assertions.assertThat(verbosePhysicalAddress).usingRecursiveComparison()
.ignoringFields(PhysicalPostalAddressVerboseDto::countryVerbose.name, PhysicalPostalAddressVerboseDto::administrativeAreaLevel1Verbose.name)
.isEqualTo(physicalAddress)
Assertions.assertThat(verbosePhysicalAddress.country.name).isEqualTo(physicalAddress?.country?.name)
Assertions.assertThat(verbosePhysicalAddress.administrativeAreaLevel1).isEqualTo(physicalAddress?.administrativeAreaLevel1)
val verboseAlternAddress = verboseAddress.alternativePostalAddress
val alternAddress = address?.alternativePostalAddress
Assertions.assertThat(verboseAlternAddress).usingRecursiveComparison()
.ignoringFields(AlternativePostalAddressVerboseDto::countryVerbose.name, AlternativePostalAddressVerboseDto::administrativeAreaLevel1Verbose.name)
.isEqualTo(alternAddress)
Assertions.assertThat(verboseAlternAddress?.country?.name).isEqualTo(alternAddress?.country?.name)
Assertions.assertThat(verboseAlternAddress?.administrativeAreaLevel1).isEqualTo(alternAddress?.administrativeAreaLevel1)
}

fun compareAddressStates(statesVerbose: Collection<AddressStateVerboseDto>, states: Collection<AddressStateDto>?) {

Assertions.assertThat(statesVerbose.size).isEqualTo(states?.size ?: 0)
val sortedVerboseStates = statesVerbose.sortedBy { it.validFrom }
val sortedStates = states?.sortedBy { it.validFrom }
sortedVerboseStates.indices.forEach {
Assertions.assertThat(sortedVerboseStates[it].typeVerbose.technicalKey.name).isEqualTo(sortedStates!![it].type.name)
Assertions.assertThat(sortedVerboseStates[it]).usingRecursiveComparison()
.withEqualsForFields(isEqualToIgnoringMilliseconds(), AddressStateVerboseDto::validTo.name)
.withEqualsForFields(isEqualToIgnoringMilliseconds(), AddressStateVerboseDto::validFrom.name)
.ignoringFields(AddressStateVerboseDto::typeVerbose.name).isEqualTo(sortedStates[it])
}
}

fun compareAddressIdentifiers(identifiersVerbose: Collection<AddressIdentifierVerboseDto>, identifiers: Collection<AddressIdentifierDto>?) {

Assertions.assertThat(identifiersVerbose.size).isEqualTo(identifiers?.size ?: 0)
val sortedVerboseIdentifiers = identifiersVerbose.sortedBy { it.typeVerbose.name }
val sortedIdentifiers = identifiers!!.sortedBy { it.type }
sortedVerboseIdentifiers.indices.forEach {
Assertions.assertThat(sortedVerboseIdentifiers[it].typeVerbose.technicalKey).isEqualTo(sortedIdentifiers[it].type)
Assertions.assertThat(sortedVerboseIdentifiers[it]).usingRecursiveComparison()
.ignoringFields(AddressIdentifierVerboseDto::typeVerbose.name)
.isEqualTo(sortedIdentifiers[it])
}
}

fun compareStates(statesVerbose: Collection<LegalEntityStateVerboseDto>, states: Collection<ILegalEntityStateDto>?) {

Assertions.assertThat(statesVerbose.size).isEqualTo(states?.size ?: 0)
val sortedVerboseStates = statesVerbose.sortedBy { it.validFrom }
val sortedStates = states!!.sortedBy { it.validFrom }
sortedVerboseStates.indices.forEach {
Assertions.assertThat(sortedVerboseStates[it].typeVerbose.technicalKey.name).isEqualTo(sortedStates[it].type.name)
Assertions.assertThat(sortedVerboseStates[it]).usingRecursiveComparison()
.withEqualsForFields(isEqualToIgnoringMilliseconds(), LegalEntityStateVerboseDto::validTo.name )
.withEqualsForFields(isEqualToIgnoringMilliseconds(), LegalEntityStateVerboseDto::validFrom.name)
.ignoringFields(LegalEntityStateVerboseDto::typeVerbose.name)
.isEqualTo(sortedStates[it])
}
}

fun compareSiteStates(statesVerbose: Collection<SiteStateVerboseDto>, states: Collection<ISiteStateDto>?) {

Assertions.assertThat(statesVerbose.size).isEqualTo(states?.size ?: 0)
val sortedVerboseStates = statesVerbose.sortedBy { it.validFrom }
val sortedStates = states!!.sortedBy { it.validFrom }
sortedVerboseStates.indices.forEach {
Assertions.assertThat(sortedVerboseStates[it].typeVerbose.technicalKey.name).isEqualTo(sortedStates[it].type.name)
Assertions.assertThat(sortedVerboseStates[it]).usingRecursiveComparison()
.withEqualsForFields(isEqualToIgnoringMilliseconds(), SiteStateVerboseDto::validTo.name)
.withEqualsForFields(isEqualToIgnoringMilliseconds(), SiteStateVerboseDto::validFrom.name)
.ignoringFields(SiteStateVerboseDto::typeVerbose.name)
.isEqualTo(sortedStates[it])
}
}

fun isEqualToIgnoringMilliseconds(): BiPredicate<LocalDateTime?, LocalDateTime?> {
return BiPredicate<LocalDateTime?, LocalDateTime?> { d1, d2 ->
(d1 == null && d2 == null)
|| d1.truncatedTo(ChronoUnit.SECONDS).equals(d2.truncatedTo(ChronoUnit.SECONDS))
}
}

fun compareClassifications(
classificationsVerbose: Collection<LegalEntityClassificationVerboseDto>,
classifications: Collection<LegalEntityClassificationDto>?
) {

Assertions.assertThat(classificationsVerbose.size).isEqualTo(classifications?.size ?: 0)
val sortedVerboseClassifications = classificationsVerbose.sortedBy { it.typeVerbose.name }
val sortedClassifications = classifications!!.sortedBy { it.type.name }
sortedVerboseClassifications.indices.forEach {
Assertions.assertThat(sortedVerboseClassifications[it].typeVerbose.technicalKey.name).isEqualTo(sortedClassifications[it].type.name)
Assertions.assertThat(sortedVerboseClassifications[it]).usingRecursiveComparison()
.ignoringFields(LegalEntityClassificationVerboseDto::typeVerbose.name)
.isEqualTo(sortedClassifications[it])
}
}

fun compareIdentifiers(identifiersVerbose: Collection<LegalEntityIdentifierVerboseDto>, identifiers: Collection<LegalEntityIdentifierDto>?) {

Assertions.assertThat(identifiersVerbose.size).isEqualTo(identifiers?.size ?: 0)
val sortedVerboseIdentifiers = identifiersVerbose.sortedBy { it.typeVerbose.name }
val sortedIdentifiers = identifiers!!.sortedBy { it.type }
sortedVerboseIdentifiers.indices.forEach {
Assertions.assertThat(sortedVerboseIdentifiers[it].typeVerbose.technicalKey).isEqualTo(sortedIdentifiers[it].type)
Assertions.assertThat(sortedVerboseIdentifiers[it]).usingRecursiveComparison()
.ignoringFields(LegalEntityIdentifierVerboseDto::typeVerbose.name).isEqualTo(sortedIdentifiers[it])
}
}
Loading

0 comments on commit 6ed7859

Please sign in to comment.