From f7dd55bffefef513a8e35290c743f16922c043f9 Mon Sep 17 00:00:00 2001 From: Roman_Chernetskyi Date: Mon, 10 Jun 2024 17:57:38 +0300 Subject: [PATCH 01/11] [MODINV-1031] Implement endpoint to update ownership of Holdings --- pom.xml | 3 ++ .../folio/inventory/resources/MoveApi.java | 1 + .../resources/UpdateOwnershipApi.java | 54 ++++++++++++++++++- .../validation/UpdateOwnershipValidator.java | 25 +++++++++ src/test/java/api/ApiTestSuite.java | 1 + .../HoldingsUpdateOwnershipApiTest.java | 44 +++++++++++---- ...gsRecordUpdateOwnershipRequestBuilder.java | 4 +- 7 files changed, 119 insertions(+), 13 deletions(-) create mode 100644 src/main/java/org/folio/inventory/validation/UpdateOwnershipValidator.java diff --git a/pom.xml b/pom.xml index 8e36cfbb6..0b3c44ae0 100644 --- a/pom.xml +++ b/pom.xml @@ -445,6 +445,9 @@ ${basedir}/ramls/holdings-record.json ${basedir}/ramls/holdings-records-source.json ${basedir}/ramls/mappingMetadataDto.json + ${basedir}/ramls/holdings_update_ownership.json + ${basedir}/ramls/items_update_ownership.json + ${basedir}/ramls/move_response.json org.folio true diff --git a/src/main/java/org/folio/inventory/resources/MoveApi.java b/src/main/java/org/folio/inventory/resources/MoveApi.java index 14216ddb0..6c32d19c9 100644 --- a/src/main/java/org/folio/inventory/resources/MoveApi.java +++ b/src/main/java/org/folio/inventory/resources/MoveApi.java @@ -44,6 +44,7 @@ public class MoveApi extends AbstractInventoryResource { public static final String TO_INSTANCE_ID = "toInstanceId"; public static final String ITEM_IDS = "itemIds"; public static final String HOLDINGS_RECORD_IDS = "holdingsRecordIds"; + public static final String TARGET_TENANT_ID = "targetTenantId"; public static final String ITEM_STORAGE = "/item-storage/items"; public static final String ITEMS_PROPERTY = "items"; public static final String HOLDINGS_RECORDS_PROPERTY = "holdingsRecords"; diff --git a/src/main/java/org/folio/inventory/resources/UpdateOwnershipApi.java b/src/main/java/org/folio/inventory/resources/UpdateOwnershipApi.java index 339fb8886..86e322b26 100644 --- a/src/main/java/org/folio/inventory/resources/UpdateOwnershipApi.java +++ b/src/main/java/org/folio/inventory/resources/UpdateOwnershipApi.java @@ -3,9 +3,24 @@ import io.vertx.core.http.HttpClient; import io.vertx.ext.web.Router; import io.vertx.ext.web.RoutingContext; +import org.apache.logging.log4j.LogManager; +import org.apache.logging.log4j.Logger; +import org.folio.HoldingsUpdateOwnership; +import org.folio.inventory.common.WebContext; import org.folio.inventory.storage.Storage; +import org.folio.inventory.support.http.server.JsonResponse; +import org.folio.inventory.support.http.server.ServerErrorResponse; + +import java.lang.invoke.MethodHandles; + +import static org.folio.inventory.support.EndpointFailureHandler.handleFailure; +import static org.folio.inventory.support.http.server.JsonResponse.unprocessableEntity; +import static org.folio.inventory.validation.UpdateOwnershipValidator.updateOwnershipHasRequiredFields; public class UpdateOwnershipApi extends AbstractInventoryResource { + private static final Logger LOGGER = LogManager.getLogger(MethodHandles.lookup().lookupClass()); + public static final String CONSORTIUM = "CONSORTIUM"; + public UpdateOwnershipApi(Storage storage, HttpClient client) { super(storage, client); } @@ -19,7 +34,44 @@ public void register(Router router) { } private void updateHoldingsOwnership(RoutingContext routingContext) { - // should be implemented in MODINV-1031 + try { + final var context = new WebContext(routingContext); + final var updateOwnershipRequest = routingContext.body().asJsonObject(); + + final var validationError = updateOwnershipHasRequiredFields(updateOwnershipRequest, HoldingsUpdateOwnership.class); + + if (validationError.isPresent()) { + unprocessableEntity(routingContext.response(), validationError.get()); + return; + } + var holdingsUpdateOwnership = updateOwnershipRequest.mapTo(HoldingsUpdateOwnership.class); + + LOGGER.info("updateHoldingsOwnership:: Started updating ownership of holdings record: {}, to tenant: {}", holdingsUpdateOwnership.getHoldingsRecordIds(), + holdingsUpdateOwnership.getTargetTenantId()); + + storage.getInstanceCollection(context) + .findById(holdingsUpdateOwnership.getToInstanceId()) + .thenAccept(instance -> { + if (instance != null) { + if (instance.getSource().startsWith(CONSORTIUM)) { + handleFailure(new Throwable("not implemented yet"), routingContext); + } else { + JsonResponse.unprocessableEntity(routingContext.response(), + String.format("Instance with id: %s is not shared", holdingsUpdateOwnership.getToInstanceId())); + } + } else { + JsonResponse.unprocessableEntity(routingContext.response(), + String.format("Instance with id: %s not found at source tenant, tenant: %s", holdingsUpdateOwnership.getToInstanceId(), context.getTenantId())); + } + }) + .exceptionally(e -> { + ServerErrorResponse.internalError(routingContext.response(), e); + return null; + }); + } catch (Exception e) { + LOGGER.warn(e); + handleFailure(e, routingContext); + } } private void updateItemsOwnership(RoutingContext routingContext) { diff --git a/src/main/java/org/folio/inventory/validation/UpdateOwnershipValidator.java b/src/main/java/org/folio/inventory/validation/UpdateOwnershipValidator.java new file mode 100644 index 000000000..3c2d48206 --- /dev/null +++ b/src/main/java/org/folio/inventory/validation/UpdateOwnershipValidator.java @@ -0,0 +1,25 @@ +package org.folio.inventory.validation; + +import io.vertx.core.json.JsonArray; +import io.vertx.core.json.JsonObject; +import org.folio.inventory.support.http.server.ValidationError; + +import java.lang.reflect.Field; +import java.util.Arrays; +import java.util.List; +import java.util.Optional; + +public final class UpdateOwnershipValidator { + private UpdateOwnershipValidator() { } + + public static Optional updateOwnershipHasRequiredFields(JsonObject updateOwnershipRequest, Class updateOwnershipClass) { + List requiredFields = Arrays.stream(updateOwnershipClass.getDeclaredFields()).map(Field::getName).toList(); + for (String field: requiredFields) { + var value = updateOwnershipRequest.getValue(field); + if (value == null || (value instanceof JsonArray && ((JsonArray) value).isEmpty())) { + return Optional.of(new ValidationError(field + " is a required field", field, null)); + } + } + return Optional.empty(); + } +} diff --git a/src/test/java/api/ApiTestSuite.java b/src/test/java/api/ApiTestSuite.java index 2373dc0fd..7db9b0e6f 100644 --- a/src/test/java/api/ApiTestSuite.java +++ b/src/test/java/api/ApiTestSuite.java @@ -74,6 +74,7 @@ public class ApiTestSuite { public static final int INVENTORY_VERTICLE_TEST_PORT = 9603; public static final String TENANT_ID = "test_tenant"; public static final String CONSORTIA_TENANT_ID = "consortium"; + public static final String COLLEGE_TENANT_ID = "college"; public static final UUID ID_FOR_FAILURE = UUID.fromString("fa45a95b-38a3-430b-8f34-548ca005a176"); public static final UUID ID_FOR_OPTIMISTIC_LOCKING_FAILURE = UUID.fromString("40900409-0409-4444-8888-409000000409"); diff --git a/src/test/java/api/holdings/HoldingsUpdateOwnershipApiTest.java b/src/test/java/api/holdings/HoldingsUpdateOwnershipApiTest.java index 28f8145e1..cad5da535 100644 --- a/src/test/java/api/holdings/HoldingsUpdateOwnershipApiTest.java +++ b/src/test/java/api/holdings/HoldingsUpdateOwnershipApiTest.java @@ -35,7 +35,7 @@ import static org.hamcrest.Matchers.equalTo; import static support.matchers.ResponseMatchers.hasValidationError; -@Ignore +//@Ignore @RunWith(JUnitParamsRunner.class) public class HoldingsUpdateOwnershipApiTest extends ApiTests { private static final String INSTANCE_ID = "instanceId"; @@ -62,7 +62,7 @@ public void canUpdateHoldingsOwnershipToDifferentTenant() throws MalformedURLExc final UUID createHoldingsRecord2 = createHoldingForInstance(instanceId); JsonObject holdingsRecordUpdateOwnershipRequestBody = new HoldingsRecordUpdateOwnershipRequestBuilder(instanceId, - new JsonArray(List.of(createHoldingsRecord1.toString(), createHoldingsRecord2.toString())), ApiTestSuite.CONSORTIA_TENANT_ID).create(); + new JsonArray(List.of(createHoldingsRecord1.toString(), createHoldingsRecord2.toString())), ApiTestSuite.COLLEGE_TENANT_ID).create(); Response postHoldingsUpdateOwnershipResponse = updateHoldingsRecordsOwnership(holdingsRecordUpdateOwnershipRequestBody); @@ -97,7 +97,7 @@ public void shouldReportErrorsWhenOnlySomeRequestedHoldingsRecordsCouldNotBeUpda Assert.assertNotEquals(createHoldingsRecord1, createHoldingsRecord2); JsonObject holdingsRecordUpdateOwnershipRequestBody = new HoldingsRecordUpdateOwnershipRequestBuilder(instanceId, - new JsonArray(List.of(createHoldingsRecord1.toString(), createHoldingsRecord2.toString())), ApiTestSuite.CONSORTIA_TENANT_ID).create(); + new JsonArray(List.of(createHoldingsRecord1.toString(), createHoldingsRecord2.toString())), ApiTestSuite.COLLEGE_TENANT_ID).create(); Response postHoldingsUpdateOwnershipResponse = updateHoldingsRecordsOwnership(holdingsRecordUpdateOwnershipRequestBody); @@ -125,7 +125,7 @@ public void shouldReportErrorsWhenOnlySomeRequestedHoldingsRecordsCouldNotBeUpda public void cannotUpdateHoldingsRecordsOwnershipToUnspecifiedInstance() throws InterruptedException, MalformedURLException, TimeoutException, ExecutionException { JsonObject holdingsRecordUpdateOwnershipWithoutToInstanceId = new HoldingsRecordUpdateOwnershipRequestBuilder(null, - new JsonArray(List.of(UUID.randomUUID())), ApiTestSuite.CONSORTIA_TENANT_ID).create(); + new JsonArray(List.of(UUID.randomUUID())), ApiTestSuite.COLLEGE_TENANT_ID).create(); Response postHoldingsUpdateOwnershipResponse = updateHoldingsRecordsOwnership(holdingsRecordUpdateOwnershipWithoutToInstanceId); @@ -149,7 +149,7 @@ public void cannotUpdateHoldingsRecordsOwnershipToUnspecifiedTenant() assertThat(postHoldingsUpdateOwnershipResponse.getContentType(), containsString(APPLICATION_JSON)); assertThat(postHoldingsUpdateOwnershipResponse, hasValidationError( - "tenantId is a required field", "toInstanceId", null + "targetTenantId is a required field", "targetTenantId", null )); } @@ -157,7 +157,7 @@ public void cannotUpdateHoldingsRecordsOwnershipToUnspecifiedTenant() public void cannotUpdateUnspecifiedHoldingsRecordsOwnership() throws MalformedURLException, InterruptedException, ExecutionException, TimeoutException { JsonObject holdingsRecordUpdateOwnershipWithoutHoldingsRecordIds = new HoldingsRecordUpdateOwnershipRequestBuilder(UUID.randomUUID(), - new JsonArray(), ApiTestSuite.CONSORTIA_TENANT_ID).create(); + new JsonArray(), ApiTestSuite.COLLEGE_TENANT_ID).create(); Response postHoldingsUpdateOwnershipResponse = updateHoldingsRecordsOwnership(holdingsRecordUpdateOwnershipWithoutHoldingsRecordIds); @@ -165,7 +165,7 @@ public void cannotUpdateUnspecifiedHoldingsRecordsOwnership() assertThat(postHoldingsUpdateOwnershipResponse.getContentType(), containsString(APPLICATION_JSON)); assertThat(postHoldingsUpdateOwnershipResponse, hasValidationError( - "Holdings record ids aren't specified", "holdingsRecordIds", null + "holdingsRecordIds is a required field", "holdingsRecordIds", null )); } @@ -185,7 +185,7 @@ public void cannotUpdateHoldingsRecordOwnershipOfNonExistedInstance() final UUID createHoldingsRecord1 = createHoldingForInstance(instanceId); JsonObject holdingsRecordUpdateOwnershipWithoutHoldingsRecordIds = new HoldingsRecordUpdateOwnershipRequestBuilder(invalidInstanceId, - new JsonArray(List.of(createHoldingsRecord1)), ApiTestSuite.CONSORTIA_TENANT_ID).create(); + new JsonArray(List.of(createHoldingsRecord1)), ApiTestSuite.COLLEGE_TENANT_ID).create(); Response postHoldingsUpdateOwnershipResponse = updateHoldingsRecordsOwnership(holdingsRecordUpdateOwnershipWithoutHoldingsRecordIds); @@ -196,6 +196,30 @@ public void cannotUpdateHoldingsRecordOwnershipOfNonExistedInstance() assertThat(postHoldingsUpdateOwnershipResponse.getBody(), containsString(invalidInstanceId.toString())); } + @Test + public void cannotUpdateHoldingsRecordOwnershipOfNonSharedInstance() + throws MalformedURLException, InterruptedException, ExecutionException, TimeoutException { + createConsortiumTenant(); + + UUID instanceId = UUID.randomUUID(); + JsonObject instance = smallAngryPlanet(instanceId); + + InstanceApiClient.createInstance(okapiClient, instance); + + final UUID createHoldingsRecord1 = createHoldingForInstance(instanceId); + final UUID createHoldingsRecord2 = createHoldingForInstance(instanceId); + + JsonObject holdingsRecordUpdateOwnershipRequestBody = new HoldingsRecordUpdateOwnershipRequestBuilder(instanceId, + new JsonArray(List.of(createHoldingsRecord1.toString(), createHoldingsRecord2.toString())), ApiTestSuite.COLLEGE_TENANT_ID).create(); + + Response postHoldingsUpdateOwnershipResponse = updateHoldingsRecordsOwnership(holdingsRecordUpdateOwnershipRequestBody); + + assertThat(postHoldingsUpdateOwnershipResponse.getStatusCode(), is(422)); + assertThat(postHoldingsUpdateOwnershipResponse.getContentType(), containsString(APPLICATION_JSON)); + + assertThat(postHoldingsUpdateOwnershipResponse.getBody(), containsString(String.format("Instance with id: %s is not shared", instanceId))); + } + @Test public void canUpdateHoldingsRecordOwnershipDueToHoldingsRecordUpdateError() throws InterruptedException, MalformedURLException, TimeoutException, ExecutionException { UUID instanceId = UUID.randomUUID(); @@ -208,7 +232,7 @@ public void canUpdateHoldingsRecordOwnershipDueToHoldingsRecordUpdateError() thr final UUID createHoldingsRecord2 = createHoldingForInstance(ID_FOR_FAILURE, instanceId); JsonObject holdingsRecordUpdateOwnershipRequestBody = new HoldingsRecordUpdateOwnershipRequestBuilder(instanceId, - new JsonArray(List.of(createHoldingsRecord1.toString(), createHoldingsRecord2.toString())), ApiTestSuite.CONSORTIA_TENANT_ID).create(); + new JsonArray(List.of(createHoldingsRecord1.toString(), createHoldingsRecord2.toString())), ApiTestSuite.COLLEGE_TENANT_ID).create(); Response postHoldingsUpdateOwnershipResponse = updateHoldingsRecordsOwnership(holdingsRecordUpdateOwnershipRequestBody); @@ -256,7 +280,7 @@ public void canUpdateHoldingsRecordOwnershipToDifferentInstanceWithExtraRedundan .getId(); JsonObject holdingsRecordUpdateOwnershipRequestBody = new HoldingsRecordUpdateOwnershipRequestBuilder(instanceId, - new JsonArray(List.of(createHoldingsRecord1.toString(), createHoldingsRecord2.toString())), ApiTestSuite.CONSORTIA_TENANT_ID).create(); + new JsonArray(List.of(createHoldingsRecord1.toString(), createHoldingsRecord2.toString())), ApiTestSuite.COLLEGE_TENANT_ID).create(); Response postHoldingsUpdateOwnershipResponse = updateHoldingsRecordsOwnership(holdingsRecordUpdateOwnershipRequestBody); diff --git a/src/test/java/api/support/builders/HoldingsRecordUpdateOwnershipRequestBuilder.java b/src/test/java/api/support/builders/HoldingsRecordUpdateOwnershipRequestBuilder.java index 2c4e22e0a..99adadbff 100644 --- a/src/test/java/api/support/builders/HoldingsRecordUpdateOwnershipRequestBuilder.java +++ b/src/test/java/api/support/builders/HoldingsRecordUpdateOwnershipRequestBuilder.java @@ -6,8 +6,8 @@ import java.util.UUID; -import static api.ApiTestSuite.TENANT_ID; import static org.folio.inventory.resources.MoveApi.HOLDINGS_RECORD_IDS; +import static org.folio.inventory.resources.MoveApi.TARGET_TENANT_ID; import static org.folio.inventory.resources.MoveApi.TO_INSTANCE_ID; public class HoldingsRecordUpdateOwnershipRequestBuilder extends AbstractBuilder { @@ -27,7 +27,7 @@ public JsonObject create() { includeWhenPresent(holdingsRecordUpdateOwnershipRequest, TO_INSTANCE_ID, toInstanceId); includeWhenPresent(holdingsRecordUpdateOwnershipRequest, HOLDINGS_RECORD_IDS, holdingsRecordsIds); - includeWhenPresent(holdingsRecordUpdateOwnershipRequest, TENANT_ID, tenantId); + includeWhenPresent(holdingsRecordUpdateOwnershipRequest, TARGET_TENANT_ID, tenantId); return holdingsRecordUpdateOwnershipRequest; } From bbd960cde74eec56955f3d3125f9b9635c8cda85 Mon Sep 17 00:00:00 2001 From: Roman_Chernetskyi Date: Wed, 12 Jun 2024 19:32:48 +0300 Subject: [PATCH 02/11] Implement updating ownership of holdings withou updating ownership of underlying items --- .../folio/inventory/InventoryVerticle.java | 2 +- .../folio/inventory/resources/MoveApi.java | 1 - .../resources/UpdateOwnershipApi.java | 134 +++++++++++--- .../folio/inventory/support/MoveApiUtil.java | 82 +++++++++ .../validation/UpdateOwnershipValidator.java | 8 +- .../HoldingsUpdateOwnershipApiTest.java | 167 +++++++++++++----- src/test/java/api/support/ApiTests.java | 7 + ...gsRecordUpdateOwnershipRequestBuilder.java | 2 +- .../fakes/FakeStorageModuleBuilder.java | 2 +- 9 files changed, 339 insertions(+), 66 deletions(-) create mode 100644 src/main/java/org/folio/inventory/support/MoveApiUtil.java diff --git a/src/main/java/org/folio/inventory/InventoryVerticle.java b/src/main/java/org/folio/inventory/InventoryVerticle.java index 8079714c6..bbdaac2e5 100644 --- a/src/main/java/org/folio/inventory/InventoryVerticle.java +++ b/src/main/java/org/folio/inventory/InventoryVerticle.java @@ -70,7 +70,7 @@ public void start(Promise started) { new ItemsByHoldingsRecordId(storage, client).register(router); new InventoryConfigApi().register(router); new TenantApi().register(router); - new UpdateOwnershipApi(storage, client).register(router); + new UpdateOwnershipApi(storage, client, consortiumService).register(router); Handler> onHttpServerStart = result -> { if (result.succeeded()) { diff --git a/src/main/java/org/folio/inventory/resources/MoveApi.java b/src/main/java/org/folio/inventory/resources/MoveApi.java index 6c32d19c9..14216ddb0 100644 --- a/src/main/java/org/folio/inventory/resources/MoveApi.java +++ b/src/main/java/org/folio/inventory/resources/MoveApi.java @@ -44,7 +44,6 @@ public class MoveApi extends AbstractInventoryResource { public static final String TO_INSTANCE_ID = "toInstanceId"; public static final String ITEM_IDS = "itemIds"; public static final String HOLDINGS_RECORD_IDS = "holdingsRecordIds"; - public static final String TARGET_TENANT_ID = "targetTenantId"; public static final String ITEM_STORAGE = "/item-storage/items"; public static final String ITEMS_PROPERTY = "items"; public static final String HOLDINGS_RECORDS_PROPERTY = "holdingsRecords"; diff --git a/src/main/java/org/folio/inventory/resources/UpdateOwnershipApi.java b/src/main/java/org/folio/inventory/resources/UpdateOwnershipApi.java index 86e322b26..090126bcf 100644 --- a/src/main/java/org/folio/inventory/resources/UpdateOwnershipApi.java +++ b/src/main/java/org/folio/inventory/resources/UpdateOwnershipApi.java @@ -1,28 +1,50 @@ package org.folio.inventory.resources; +import io.vertx.core.Promise; import io.vertx.core.http.HttpClient; +import io.vertx.core.json.JsonObject; import io.vertx.ext.web.Router; import io.vertx.ext.web.RoutingContext; import org.apache.logging.log4j.LogManager; import org.apache.logging.log4j.Logger; +import org.folio.HoldingsRecord; import org.folio.HoldingsUpdateOwnership; +import org.folio.inventory.common.Context; import org.folio.inventory.common.WebContext; +import org.folio.inventory.consortium.services.ConsortiumService; +import org.folio.inventory.domain.HoldingsRecordCollection; +import org.folio.inventory.exceptions.BadRequestException; +import org.folio.inventory.exceptions.NotFoundException; import org.folio.inventory.storage.Storage; -import org.folio.inventory.support.http.server.JsonResponse; -import org.folio.inventory.support.http.server.ServerErrorResponse; +import org.folio.inventory.storage.external.CollectionResourceClient; +import org.folio.inventory.storage.external.MultipleRecordsFetchClient; +import org.folio.inventory.support.MoveApiUtil; import java.lang.invoke.MethodHandles; +import java.util.List; +import java.util.concurrent.CompletableFuture; +import static org.folio.inventory.dataimport.handlers.matching.util.EventHandlingUtil.constructContext; +import static org.folio.inventory.domain.instances.InstanceSource.CONSORTIUM_FOLIO; +import static org.folio.inventory.domain.instances.InstanceSource.CONSORTIUM_MARC; import static org.folio.inventory.support.EndpointFailureHandler.handleFailure; +import static org.folio.inventory.support.MoveApiUtil.createHoldingsRecordsFetchClient; +import static org.folio.inventory.support.MoveApiUtil.createHoldingsStorageClient; +import static org.folio.inventory.support.MoveApiUtil.createHttpClient; +import static org.folio.inventory.support.MoveApiUtil.respond; import static org.folio.inventory.support.http.server.JsonResponse.unprocessableEntity; import static org.folio.inventory.validation.UpdateOwnershipValidator.updateOwnershipHasRequiredFields; public class UpdateOwnershipApi extends AbstractInventoryResource { private static final Logger LOGGER = LogManager.getLogger(MethodHandles.lookup().lookupClass()); - public static final String CONSORTIUM = "CONSORTIUM"; + public static final String INSTANCE_NOT_SHARED = "Instance with id: %s is not shared"; + public static final String INSTANCE_NOT_FOUND_AT_SOURCE_TENANT = "Instance with id: %s not found at source tenant, tenant: %s"; + public static final String TENANT_NOT_IN_CONSORTIA = "%s tenant is not in consortia"; + private final ConsortiumService consortiumService; - public UpdateOwnershipApi(Storage storage, HttpClient client) { + public UpdateOwnershipApi(Storage storage, HttpClient client, ConsortiumService consortiumService) { super(storage, client); + this.consortiumService = consortiumService; } @Override @@ -38,7 +60,7 @@ private void updateHoldingsOwnership(RoutingContext routingContext) { final var context = new WebContext(routingContext); final var updateOwnershipRequest = routingContext.body().asJsonObject(); - final var validationError = updateOwnershipHasRequiredFields(updateOwnershipRequest, HoldingsUpdateOwnership.class); + final var validationError = updateOwnershipHasRequiredFields(context.getTenantId(), updateOwnershipRequest, HoldingsUpdateOwnership.class); if (validationError.isPresent()) { unprocessableEntity(routingContext.response(), validationError.get()); @@ -49,27 +71,40 @@ private void updateHoldingsOwnership(RoutingContext routingContext) { LOGGER.info("updateHoldingsOwnership:: Started updating ownership of holdings record: {}, to tenant: {}", holdingsUpdateOwnership.getHoldingsRecordIds(), holdingsUpdateOwnership.getTargetTenantId()); - storage.getInstanceCollection(context) - .findById(holdingsUpdateOwnership.getToInstanceId()) - .thenAccept(instance -> { - if (instance != null) { - if (instance.getSource().startsWith(CONSORTIUM)) { - handleFailure(new Throwable("not implemented yet"), routingContext); - } else { - JsonResponse.unprocessableEntity(routingContext.response(), - String.format("Instance with id: %s is not shared", holdingsUpdateOwnership.getToInstanceId())); - } - } else { - JsonResponse.unprocessableEntity(routingContext.response(), - String.format("Instance with id: %s not found at source tenant, tenant: %s", holdingsUpdateOwnership.getToInstanceId(), context.getTenantId())); + consortiumService.getConsortiumConfiguration(context).toCompletionStage().toCompletableFuture() + .thenCompose(consortiumConfigurationOptional -> { + if (consortiumConfigurationOptional.isPresent()) { + return storage.getInstanceCollection(context) + .findById(holdingsUpdateOwnership.getToInstanceId()) + .thenCompose(instance -> { + if (instance != null) { + if (instance.getSource().equals(CONSORTIUM_MARC.getValue()) || instance.getSource().equals(CONSORTIUM_FOLIO.getValue())) { + return updateOwnershipOfHoldingsRecords(holdingsUpdateOwnership, routingContext, context); + } else { + String instanceNotSharedErrorMessage = String.format(INSTANCE_NOT_SHARED, holdingsUpdateOwnership.getToInstanceId()); + LOGGER.warn("updateHoldingsOwnership:: " + instanceNotSharedErrorMessage); + return CompletableFuture.failedFuture(new BadRequestException(instanceNotSharedErrorMessage)); + } + } else { + String instanceNotFoundErrorMessage = String.format(INSTANCE_NOT_FOUND_AT_SOURCE_TENANT, holdingsUpdateOwnership.getToInstanceId(), context.getTenantId()); + LOGGER.warn("updateHoldingsOwnership:: " + instanceNotFoundErrorMessage); + return CompletableFuture.failedFuture(new NotFoundException(instanceNotFoundErrorMessage)); + } + }); } + String notInConsortiaErrorMessage = String.format(TENANT_NOT_IN_CONSORTIA, context.getTenantId()); + LOGGER.warn("updateHoldingsOwnership:: " + notInConsortiaErrorMessage, context); + return CompletableFuture.failedFuture(new BadRequestException(notInConsortiaErrorMessage)); // NEED TEST }) - .exceptionally(e -> { - ServerErrorResponse.internalError(routingContext.response(), e); + .thenAccept(updateHoldingsRecords -> respond(routingContext, holdingsUpdateOwnership.getHoldingsRecordIds(), updateHoldingsRecords)) + .exceptionally(throwable -> { + LOGGER.warn("updateHoldingsOwnership:: Error during update ownership of holdings {}, to tenant: {}", + holdingsUpdateOwnership.getHoldingsRecordIds(), holdingsUpdateOwnership.getTargetTenantId(), throwable); + handleFailure(throwable, routingContext); return null; }); } catch (Exception e) { - LOGGER.warn(e); + LOGGER.warn("updateHoldingsOwnership:: Error during update ownership of holdings", e); handleFailure(e, routingContext); } } @@ -77,4 +112,61 @@ private void updateHoldingsOwnership(RoutingContext routingContext) { private void updateItemsOwnership(RoutingContext routingContext) { // should be implemented in MODINV-1031 } + + private CompletableFuture> updateOwnershipOfHoldingsRecords(HoldingsUpdateOwnership holdingsUpdateOwnership, RoutingContext routingContext, + WebContext context) { + try { + Context targetTenantContext = constructContext(holdingsUpdateOwnership.getTargetTenantId(), context.getToken(), context.getOkapiLocation()); + + CollectionResourceClient holdingsStorageClient = createHoldingsStorageClient(createHttpClient(client, routingContext, context), + context); + MultipleRecordsFetchClient holdingsRecordFetchClient = createHoldingsRecordsFetchClient(holdingsStorageClient); + + HoldingsRecordCollection sourceTenantHoldingsRecordCollection = storage.getHoldingsRecordCollection(context); + HoldingsRecordCollection targetTenantHoldingsRecordCollection = storage.getHoldingsRecordCollection(targetTenantContext); + + return holdingsRecordFetchClient.find(holdingsUpdateOwnership.getHoldingsRecordIds(), MoveApiUtil::fetchByIdCql) + .thenCompose(jsons -> + createHoldings(jsons, holdingsUpdateOwnership.getToInstanceId(), targetTenantHoldingsRecordCollection) + .thenCompose(createdHoldings -> { + List createdHoldingsIds = createdHoldings.stream().map(HoldingsRecord::getId).toList(); + return deleteHoldings(createdHoldingsIds, sourceTenantHoldingsRecordCollection); + })); + } catch (Exception e) { + return CompletableFuture.failedFuture(e); + } + } + + private CompletableFuture> createHoldings(List jsons, String instanceId, + HoldingsRecordCollection holdingsRecordCollection) { + List holdingsRecordsToUpdateOwnership = jsons.stream() + .map(json -> json.mapTo(HoldingsRecord.class)) + .filter(holdingsRecord -> holdingsRecord.getInstanceId().equals(instanceId)) + .toList(); + + List> createFutures = holdingsRecordsToUpdateOwnership.stream() + .map(holdingsRecordCollection::add) + .toList(); + + return CompletableFuture.allOf(createFutures.toArray(new CompletableFuture[0])) + .handle((vVoid, throwable) -> createFutures.stream() + .filter(future -> !future.isCompletedExceptionally()) + .map(CompletableFuture::join) + .toList()); + } + + private CompletableFuture> deleteHoldings(List holdingsRecordIds, HoldingsRecordCollection holdingsRecordCollection) { + List> deleteFutures = holdingsRecordIds.stream() + .map(holdingsRecordId -> { + Promise promise = Promise.promise(); + holdingsRecordCollection.delete(holdingsRecordId, success -> promise.complete(holdingsRecordId), failure -> promise.fail(failure.getReason())); + return promise.future().toCompletionStage().toCompletableFuture(); + }).toList(); + + return CompletableFuture.allOf(deleteFutures.toArray(new CompletableFuture[0])) + .handle((vVoid, throwable) -> deleteFutures.stream() + .filter(future -> !future.isCompletedExceptionally()) + .map(CompletableFuture::join) + .toList()); + } } diff --git a/src/main/java/org/folio/inventory/support/MoveApiUtil.java b/src/main/java/org/folio/inventory/support/MoveApiUtil.java new file mode 100644 index 000000000..c024171e1 --- /dev/null +++ b/src/main/java/org/folio/inventory/support/MoveApiUtil.java @@ -0,0 +1,82 @@ +package org.folio.inventory.support; + +import io.vertx.core.http.HttpClient; +import io.vertx.core.http.HttpServerResponse; +import io.vertx.core.json.JsonObject; +import io.vertx.ext.web.RoutingContext; +import io.vertx.ext.web.client.WebClient; +import org.apache.commons.collections4.ListUtils; +import org.folio.inventory.common.WebContext; +import org.folio.inventory.storage.external.CollectionResourceClient; +import org.folio.inventory.storage.external.CqlQuery; +import org.folio.inventory.storage.external.MultipleRecordsFetchClient; +import org.folio.inventory.support.http.client.OkapiHttpClient; +import org.folio.inventory.support.http.server.ServerErrorResponse; + +import java.net.MalformedURLException; +import java.net.URL; +import java.util.ArrayList; +import java.util.List; + +import static org.folio.inventory.support.http.server.JsonResponse.success; + +public final class MoveApiUtil { + public static final String HOLDINGS_STORAGE = "/holdings-storage/holdings"; + public static final String HOLDINGS_RECORDS_PROPERTY = "holdingsRecords"; + public static final String TARGET_TENANT_ID = "targetTenantId"; + + private MoveApiUtil() { } + + public static OkapiHttpClient createHttpClient(HttpClient client, RoutingContext routingContext, WebContext context) throws MalformedURLException { + return new OkapiHttpClient(WebClient.wrap(client), context, + exception -> ServerErrorResponse.internalError(routingContext.response(), + String.format("Failed to contact storage module: %s", exception.toString()))); + } + + + private static MultipleRecordsFetchClient createFetchClient(CollectionResourceClient client, String propertyName) { + return MultipleRecordsFetchClient.builder() + .withCollectionPropertyName(propertyName) + .withExpectedStatus(200) + .withCollectionResourceClient(client) + .build(); + } + + public static CollectionResourceClient createStorageClient(OkapiHttpClient client, WebContext context, String storageUrl) + throws MalformedURLException { + + return new CollectionResourceClient(client, new URL(context.getOkapiLocation() + storageUrl)); + } + + public static CollectionResourceClient createHoldingsStorageClient(OkapiHttpClient client, WebContext context) + throws MalformedURLException { + return createStorageClient(client, context, HOLDINGS_STORAGE); + } + + public static MultipleRecordsFetchClient createHoldingsRecordsFetchClient(CollectionResourceClient client) { + return createFetchClient(client, HOLDINGS_RECORDS_PROPERTY); + } + + public static CqlQuery fetchByIdCql(List ids) { + return CqlQuery.exactMatchAny("id", ids); + } + + public static void successWithEmptyIds(HttpServerResponse response) { + successWithIds(response, new ArrayList<>()); + } + + public static void successWithIds(HttpServerResponse response, List ids) { + success(response, new JsonObject().put("nonUpdatedIds", ids)); + } + + public static void respond(RoutingContext routingContext, List itemIdsToUpdate, List updatedItemIds) { + List nonUpdatedIds = ListUtils.subtract(itemIdsToUpdate, updatedItemIds); + HttpServerResponse response = routingContext.response(); + if (nonUpdatedIds.isEmpty()) { + successWithEmptyIds(response); + } else { + successWithIds(response, nonUpdatedIds); + } + } + +} diff --git a/src/main/java/org/folio/inventory/validation/UpdateOwnershipValidator.java b/src/main/java/org/folio/inventory/validation/UpdateOwnershipValidator.java index 3c2d48206..c51dfd5bc 100644 --- a/src/main/java/org/folio/inventory/validation/UpdateOwnershipValidator.java +++ b/src/main/java/org/folio/inventory/validation/UpdateOwnershipValidator.java @@ -9,10 +9,12 @@ import java.util.List; import java.util.Optional; +import static org.folio.inventory.support.MoveApiUtil.TARGET_TENANT_ID; + public final class UpdateOwnershipValidator { private UpdateOwnershipValidator() { } - public static Optional updateOwnershipHasRequiredFields(JsonObject updateOwnershipRequest, Class updateOwnershipClass) { + public static Optional updateOwnershipHasRequiredFields(String sourceTenant, JsonObject updateOwnershipRequest, Class updateOwnershipClass) { List requiredFields = Arrays.stream(updateOwnershipClass.getDeclaredFields()).map(Field::getName).toList(); for (String field: requiredFields) { var value = updateOwnershipRequest.getValue(field); @@ -20,6 +22,10 @@ public static Optional updateOwnershipHasRequiredFields(JsonObj return Optional.of(new ValidationError(field + " is a required field", field, null)); } } + String targetTenantId = updateOwnershipRequest.getString(TARGET_TENANT_ID); + if (sourceTenant.equals(targetTenantId)) { + return Optional.of(new ValidationError("targetTenantId field cannot be equal to source tenant id", TARGET_TENANT_ID, targetTenantId)); + } return Optional.empty(); } } diff --git a/src/test/java/api/holdings/HoldingsUpdateOwnershipApiTest.java b/src/test/java/api/holdings/HoldingsUpdateOwnershipApiTest.java index cad5da535..7bc37018c 100644 --- a/src/test/java/api/holdings/HoldingsUpdateOwnershipApiTest.java +++ b/src/test/java/api/holdings/HoldingsUpdateOwnershipApiTest.java @@ -6,17 +6,20 @@ import api.support.InstanceApiClient; import api.support.builders.HoldingRequestBuilder; import api.support.builders.HoldingsRecordUpdateOwnershipRequestBuilder; +import io.vertx.core.http.HttpMethod; import io.vertx.core.json.JsonArray; import io.vertx.core.json.JsonObject; import junitparams.JUnitParamsRunner; import org.apache.http.HttpStatus; import org.folio.inventory.support.http.client.Response; +import org.joda.time.DateTime; import org.junit.After; import org.junit.Assert; import org.junit.Before; import org.junit.Ignore; import org.junit.Test; import org.junit.runner.RunWith; +import support.fakes.EndpointFailureDescriptor; import java.net.MalformedURLException; import java.util.List; @@ -25,9 +28,10 @@ import java.util.concurrent.TimeUnit; import java.util.concurrent.TimeoutException; -import static api.ApiTestSuite.ID_FOR_FAILURE; import static api.ApiTestSuite.createConsortiumTenant; import static api.support.InstanceSamples.smallAngryPlanet; +import static org.folio.inventory.domain.instances.InstanceSource.CONSORTIUM_FOLIO; +import static org.folio.inventory.domain.instances.InstanceSource.FOLIO; import static org.folio.inventory.support.http.ContentType.APPLICATION_JSON; import static org.hamcrest.CoreMatchers.is; import static org.hamcrest.MatcherAssert.assertThat; @@ -55,8 +59,8 @@ public void canUpdateHoldingsOwnershipToDifferentTenant() throws MalformedURLExc UUID instanceId = UUID.randomUUID(); JsonObject instance = smallAngryPlanet(instanceId); - InstanceApiClient.createInstance(okapiClient, instance); - InstanceApiClient.createInstance(consortiumOkapiClient, instance); + InstanceApiClient.createInstance(okapiClient, instance.put("source", CONSORTIUM_FOLIO.getValue())); + InstanceApiClient.createInstance(consortiumOkapiClient, instance.put("source", FOLIO.getValue())); final UUID createHoldingsRecord1 = createHoldingForInstance(instanceId); final UUID createHoldingsRecord2 = createHoldingForInstance(instanceId); @@ -71,13 +75,13 @@ public void canUpdateHoldingsOwnershipToDifferentTenant() throws MalformedURLExc assertThat(postHoldingsUpdateOwnershipResponse.getContentType(), containsString(APPLICATION_JSON)); Response sourceTenantHoldingsRecord1 = holdingsStorageClient.getById(createHoldingsRecord1); - Response targetTenantHoldingsRecord1 = consortiumHoldingsStorageClient.getById(createHoldingsRecord1); + Response targetTenantHoldingsRecord1 = collegeHoldingsStorageClient.getById(createHoldingsRecord1); Assert.assertEquals(HttpStatus.SC_NOT_FOUND, sourceTenantHoldingsRecord1.getStatusCode()); Assert.assertEquals(instanceId.toString(), targetTenantHoldingsRecord1.getJson().getString(INSTANCE_ID)); Response sourceTenantHoldingsRecord2 = holdingsStorageClient.getById(createHoldingsRecord1); - Response targetTenantHoldingsRecord2 = consortiumHoldingsStorageClient.getById(createHoldingsRecord1); + Response targetTenantHoldingsRecord2 = collegeHoldingsStorageClient.getById(createHoldingsRecord1); Assert.assertEquals(HttpStatus.SC_NOT_FOUND, sourceTenantHoldingsRecord2.getStatusCode()); Assert.assertEquals(instanceId.toString(), targetTenantHoldingsRecord2.getJson().getString(INSTANCE_ID)); @@ -88,8 +92,8 @@ public void shouldReportErrorsWhenOnlySomeRequestedHoldingsRecordsCouldNotBeUpda UUID instanceId = UUID.randomUUID(); JsonObject instance = smallAngryPlanet(instanceId); - InstanceApiClient.createInstance(okapiClient, instance); - InstanceApiClient.createInstance(consortiumOkapiClient, instance); + InstanceApiClient.createInstance(okapiClient, instance.put("source", CONSORTIUM_FOLIO.getValue())); + InstanceApiClient.createInstance(consortiumOkapiClient, instance.put("source", FOLIO.getValue())); final UUID createHoldingsRecord1 = createHoldingForInstance(instanceId); final UUID createHoldingsRecord2 = UUID.randomUUID(); @@ -112,12 +116,12 @@ public void shouldReportErrorsWhenOnlySomeRequestedHoldingsRecordsCouldNotBeUpda assertThat(notFoundIds.get(0), equalTo(createHoldingsRecord2.toString())); Response sourceTenantHoldingsRecord1 = holdingsStorageClient.getById(createHoldingsRecord1); - Response targetTenantHoldingsRecord1 = consortiumHoldingsStorageClient.getById(createHoldingsRecord1); + Response targetTenantHoldingsRecord1 = collegeHoldingsStorageClient.getById(createHoldingsRecord1); Assert.assertEquals(HttpStatus.SC_NOT_FOUND, sourceTenantHoldingsRecord1.getStatusCode()); assertThat(instanceId.toString(), equalTo(targetTenantHoldingsRecord1.getJson().getString(INSTANCE_ID))); - Response targetTenantHoldingsRecord2 = consortiumHoldingsStorageClient.getById(createHoldingsRecord1); + Response targetTenantHoldingsRecord2 = collegeHoldingsStorageClient.getById(createHoldingsRecord2); Assert.assertEquals(HttpStatus.SC_NOT_FOUND, targetTenantHoldingsRecord2.getStatusCode()); } @@ -153,6 +157,22 @@ public void cannotUpdateHoldingsRecordsOwnershipToUnspecifiedTenant() )); } + @Test + public void cannotUpdateHoldingsRecordOwnershipToSameTenant() + throws MalformedURLException, InterruptedException, ExecutionException, TimeoutException { + JsonObject holdingsRecordUpdateOwnershipRequestBody = new HoldingsRecordUpdateOwnershipRequestBuilder(UUID.randomUUID(), + new JsonArray(List.of(UUID.randomUUID().toString())), ApiTestSuite.TENANT_ID).create(); + + Response postHoldingsUpdateOwnershipResponse = updateHoldingsRecordsOwnership(holdingsRecordUpdateOwnershipRequestBody); + + assertThat(postHoldingsUpdateOwnershipResponse.getStatusCode(), is(422)); + assertThat(postHoldingsUpdateOwnershipResponse.getContentType(), containsString(APPLICATION_JSON)); + + assertThat(postHoldingsUpdateOwnershipResponse, hasValidationError( + "targetTenantId field cannot be equal to source tenant id", "targetTenantId", ApiTestSuite.TENANT_ID + )); + } + @Test public void cannotUpdateUnspecifiedHoldingsRecordsOwnership() throws MalformedURLException, InterruptedException, ExecutionException, TimeoutException { @@ -170,10 +190,24 @@ public void cannotUpdateUnspecifiedHoldingsRecordsOwnership() } @Test - public void cannotUpdateHoldingsRecordOwnershipOfNonExistedInstance() + public void cannotUpdateHoldingsRecordOwnershipIfTenantNotInConsortium() throws MalformedURLException, InterruptedException, ExecutionException, TimeoutException { + userTenantsClient.deleteAll(); + + JsonObject holdingsRecordUpdateOwnershipRequestBody = new HoldingsRecordUpdateOwnershipRequestBuilder(UUID.randomUUID(), + new JsonArray(List.of(UUID.randomUUID().toString())), ApiTestSuite.COLLEGE_TENANT_ID).create(); + + Response postHoldingsUpdateOwnershipResponse = updateHoldingsRecordsOwnership(holdingsRecordUpdateOwnershipRequestBody); + + assertThat(postHoldingsUpdateOwnershipResponse.getStatusCode(), is(400)); + + assertThat(postHoldingsUpdateOwnershipResponse.getBody(), containsString("tenant is not in consortia")); createConsortiumTenant(); + } + @Test + public void cannotUpdateHoldingsRecordOwnershipOfNonExistedInstance() + throws MalformedURLException, InterruptedException, ExecutionException, TimeoutException { UUID instanceId = UUID.randomUUID(); JsonObject instance = smallAngryPlanet(instanceId); @@ -189,18 +223,15 @@ public void cannotUpdateHoldingsRecordOwnershipOfNonExistedInstance() Response postHoldingsUpdateOwnershipResponse = updateHoldingsRecordsOwnership(holdingsRecordUpdateOwnershipWithoutHoldingsRecordIds); - assertThat(postHoldingsUpdateOwnershipResponse.getStatusCode(), is(422)); - assertThat(postHoldingsUpdateOwnershipResponse.getContentType(), containsString(APPLICATION_JSON)); + assertThat(postHoldingsUpdateOwnershipResponse.getStatusCode(), is(404)); - assertThat(postHoldingsUpdateOwnershipResponse.getBody(), containsString("errors")); + assertThat(postHoldingsUpdateOwnershipResponse.getBody(), containsString("not found")); assertThat(postHoldingsUpdateOwnershipResponse.getBody(), containsString(invalidInstanceId.toString())); } @Test public void cannotUpdateHoldingsRecordOwnershipOfNonSharedInstance() throws MalformedURLException, InterruptedException, ExecutionException, TimeoutException { - createConsortiumTenant(); - UUID instanceId = UUID.randomUUID(); JsonObject instance = smallAngryPlanet(instanceId); @@ -214,22 +245,30 @@ public void cannotUpdateHoldingsRecordOwnershipOfNonSharedInstance() Response postHoldingsUpdateOwnershipResponse = updateHoldingsRecordsOwnership(holdingsRecordUpdateOwnershipRequestBody); - assertThat(postHoldingsUpdateOwnershipResponse.getStatusCode(), is(422)); - assertThat(postHoldingsUpdateOwnershipResponse.getContentType(), containsString(APPLICATION_JSON)); + assertThat(postHoldingsUpdateOwnershipResponse.getStatusCode(), is(400)); assertThat(postHoldingsUpdateOwnershipResponse.getBody(), containsString(String.format("Instance with id: %s is not shared", instanceId))); } @Test - public void canUpdateHoldingsRecordOwnershipDueToHoldingsRecordUpdateError() throws InterruptedException, MalformedURLException, TimeoutException, ExecutionException { + public void cannotUpdateHoldingsRecordOwnershipDueToHoldingsRecordCreateError() throws InterruptedException, MalformedURLException, TimeoutException, ExecutionException { UUID instanceId = UUID.randomUUID(); JsonObject instance = smallAngryPlanet(instanceId); - InstanceApiClient.createInstance(okapiClient, instance); - InstanceApiClient.createInstance(consortiumOkapiClient, instance); + InstanceApiClient.createInstance(okapiClient, instance.put("source", CONSORTIUM_FOLIO.getValue())); + InstanceApiClient.createInstance(consortiumOkapiClient, instance.put("source", FOLIO.getValue())); final UUID createHoldingsRecord1 = createHoldingForInstance(instanceId); - final UUID createHoldingsRecord2 = createHoldingForInstance(ID_FOR_FAILURE, instanceId); + final UUID createHoldingsRecord2 = createHoldingForInstance(instanceId); + + final JsonObject expectedErrorResponse = new JsonObject().put("message", "Server error"); + collegeHoldingsStorageClient.emulateFailure( + new EndpointFailureDescriptor() + .setFailureExpireDate(DateTime.now().plusSeconds(2).toDate()) + .setStatusCode(500) + .setContentType("application/json") + .setBody(expectedErrorResponse.toString()) + .setMethod(HttpMethod.POST.name())); JsonObject holdingsRecordUpdateOwnershipRequestBody = new HoldingsRecordUpdateOwnershipRequestBuilder(instanceId, new JsonArray(List.of(createHoldingsRecord1.toString(), createHoldingsRecord2.toString())), ApiTestSuite.COLLEGE_TENANT_ID).create(); @@ -240,32 +279,87 @@ public void canUpdateHoldingsRecordOwnershipDueToHoldingsRecordUpdateError() thr .getJsonArray("nonUpdatedIds") .getList(); - assertThat(nonUpdatedIdsIds.size(), is(1)); - assertThat(nonUpdatedIdsIds.get(0), equalTo(ID_FOR_FAILURE.toString())); + assertThat(nonUpdatedIdsIds.size(), is(2)); + assertThat(nonUpdatedIdsIds.get(0), equalTo(createHoldingsRecord1.toString())); + assertThat(nonUpdatedIdsIds.get(1), equalTo(createHoldingsRecord2.toString())); assertThat(postHoldingsUpdateOwnershipResponse.getStatusCode(), is(200)); assertThat(postHoldingsUpdateOwnershipResponse.getContentType(), containsString(APPLICATION_JSON)); Response sourceTenantHoldingsRecord1 = holdingsStorageClient.getById(createHoldingsRecord1); - Response targetTenantHoldingsRecord1 = consortiumHoldingsStorageClient.getById(createHoldingsRecord1); + Response targetTenantHoldingsRecord1 = collegeHoldingsStorageClient.getById(createHoldingsRecord1); - Assert.assertEquals(HttpStatus.SC_NOT_FOUND, sourceTenantHoldingsRecord1.getStatusCode()); - Assert.assertEquals(instanceId.toString(), targetTenantHoldingsRecord1.getJson().getString(INSTANCE_ID)); + Assert.assertEquals(instanceId.toString(), sourceTenantHoldingsRecord1.getJson().getString(INSTANCE_ID)); + Assert.assertEquals(HttpStatus.SC_NOT_FOUND, targetTenantHoldingsRecord1.getStatusCode()); - Response sourceTenantHoldingsRecord2 = holdingsStorageClient.getById(createHoldingsRecord1); - Response targetTenantHoldingsRecord2 = consortiumHoldingsStorageClient.getById(createHoldingsRecord1); + Response sourceTenantHoldingsRecord2 = holdingsStorageClient.getById(createHoldingsRecord2); + Response targetTenantHoldingsRecord2 = collegeHoldingsStorageClient.getById(createHoldingsRecord2); Assert.assertEquals(instanceId.toString(), sourceTenantHoldingsRecord2.getJson().getString(INSTANCE_ID)); Assert.assertEquals(HttpStatus.SC_NOT_FOUND, targetTenantHoldingsRecord2.getStatusCode()); + + collegeHoldingsStorageClient.disableFailureEmulation(); + } + + @Test + public void cannotUpdateHoldingsRecordOwnershipDueToHoldingsRecordDeleteError() throws InterruptedException, MalformedURLException, TimeoutException, ExecutionException { + UUID instanceId = UUID.randomUUID(); + JsonObject instance = smallAngryPlanet(instanceId); + + InstanceApiClient.createInstance(okapiClient, instance.put("source", CONSORTIUM_FOLIO.getValue())); + InstanceApiClient.createInstance(consortiumOkapiClient, instance.put("source", FOLIO.getValue())); + + final UUID createHoldingsRecord1 = createHoldingForInstance(instanceId); + final UUID createHoldingsRecord2 = createHoldingForInstance(instanceId); + + final JsonObject expectedErrorResponse = new JsonObject().put("message", "Server error"); + collegeHoldingsStorageClient.emulateFailure( + new EndpointFailureDescriptor() + .setFailureExpireDate(DateTime.now().plusSeconds(2).toDate()) + .setStatusCode(500) + .setContentType("application/json") + .setBody(expectedErrorResponse.toString()) + .setMethod(HttpMethod.DELETE.name())); + + JsonObject holdingsRecordUpdateOwnershipRequestBody = new HoldingsRecordUpdateOwnershipRequestBuilder(instanceId, + new JsonArray(List.of(createHoldingsRecord1.toString(), createHoldingsRecord2.toString())), ApiTestSuite.COLLEGE_TENANT_ID).create(); + + Response postHoldingsUpdateOwnershipResponse = updateHoldingsRecordsOwnership(holdingsRecordUpdateOwnershipRequestBody); + + List nonUpdatedIdsIds = postHoldingsUpdateOwnershipResponse.getJson() + .getJsonArray("nonUpdatedIds") + .getList(); + + assertThat(nonUpdatedIdsIds.size(), is(2)); + assertThat(nonUpdatedIdsIds.get(0), equalTo(createHoldingsRecord1.toString())); + assertThat(nonUpdatedIdsIds.get(1), equalTo(createHoldingsRecord2.toString())); + + assertThat(postHoldingsUpdateOwnershipResponse.getStatusCode(), is(200)); + assertThat(postHoldingsUpdateOwnershipResponse.getContentType(), containsString(APPLICATION_JSON)); + + Response sourceTenantHoldingsRecord1 = holdingsStorageClient.getById(createHoldingsRecord1); + Response targetTenantHoldingsRecord1 = collegeHoldingsStorageClient.getById(createHoldingsRecord1); + + Assert.assertEquals(instanceId.toString(), sourceTenantHoldingsRecord1.getJson().getString(INSTANCE_ID)); + Assert.assertEquals(instanceId.toString(), targetTenantHoldingsRecord1.getJson().getString(INSTANCE_ID)); + + Response sourceTenantHoldingsRecord2 = holdingsStorageClient.getById(createHoldingsRecord2); + Response targetTenantHoldingsRecord2 = collegeHoldingsStorageClient.getById(createHoldingsRecord2); + + Assert.assertEquals(instanceId.toString(), sourceTenantHoldingsRecord2.getJson().getString(INSTANCE_ID)); + Assert.assertEquals(instanceId.toString(), targetTenantHoldingsRecord2.getJson().getString(INSTANCE_ID)); + + collegeHoldingsStorageClient.disableFailureEmulation(); } + @Ignore @Test public void canUpdateHoldingsRecordOwnershipToDifferentInstanceWithExtraRedundantFields() throws InterruptedException, MalformedURLException, TimeoutException, ExecutionException { UUID instanceId = UUID.randomUUID(); JsonObject instance = smallAngryPlanet(instanceId); - InstanceApiClient.createInstance(okapiClient, instance); - InstanceApiClient.createInstance(consortiumOkapiClient, instance); + InstanceApiClient.createInstance(okapiClient, instance.put("source", CONSORTIUM_FOLIO.getValue())); + InstanceApiClient.createInstance(consortiumOkapiClient, instance.put("source", FOLIO.getValue())); JsonObject firstJsonHoldingsAsRequest = new HoldingRequestBuilder().forInstance(instanceId).create(); final UUID createHoldingsRecord1 = holdingsStorageClient.create(firstJsonHoldingsAsRequest @@ -289,13 +383,13 @@ public void canUpdateHoldingsRecordOwnershipToDifferentInstanceWithExtraRedundan assertThat(postHoldingsUpdateOwnershipResponse.getContentType(), containsString(APPLICATION_JSON)); Response sourceTenantHoldingsRecord1 = holdingsStorageClient.getById(createHoldingsRecord1); - Response targetTenantHoldingsRecord1 = consortiumHoldingsStorageClient.getById(createHoldingsRecord1); + Response targetTenantHoldingsRecord1 = collegeHoldingsStorageClient.getById(createHoldingsRecord1); Assert.assertEquals(HttpStatus.SC_NOT_FOUND, sourceTenantHoldingsRecord1.getStatusCode()); Assert.assertEquals(instanceId.toString(), targetTenantHoldingsRecord1.getJson().getString(INSTANCE_ID)); Response sourceTenantHoldingsRecord2 = holdingsStorageClient.getById(createHoldingsRecord1); - Response targetTenantHoldingsRecord2 = consortiumHoldingsStorageClient.getById(createHoldingsRecord1); + Response targetTenantHoldingsRecord2 = collegeHoldingsStorageClient.getById(createHoldingsRecord1); Assert.assertEquals(HttpStatus.SC_NOT_FOUND, sourceTenantHoldingsRecord2.getStatusCode()); Assert.assertEquals(instanceId.toString(), targetTenantHoldingsRecord2.getJson().getString(INSTANCE_ID)); @@ -304,18 +398,11 @@ public void canUpdateHoldingsRecordOwnershipToDifferentInstanceWithExtraRedundan private Response updateHoldingsRecordsOwnership(JsonObject holdingsRecordUpdateOwnershipRequestBody) throws MalformedURLException, InterruptedException, ExecutionException, TimeoutException { final var postHoldingRecordsUpdateOwnershipCompleted = okapiClient.post( ApiRoot.updateHoldingsRecordsOwnership(), holdingsRecordUpdateOwnershipRequestBody); - return postHoldingRecordsUpdateOwnershipCompleted.toCompletableFuture().get(5, TimeUnit.SECONDS); + return postHoldingRecordsUpdateOwnershipCompleted.toCompletableFuture().get(50, TimeUnit.SECONDS); } private UUID createHoldingForInstance(UUID instanceId) { return holdingsStorageClient.create(new HoldingRequestBuilder().forInstance(instanceId)) .getId(); } - - private UUID createHoldingForInstance(UUID id, UUID instanceId) { - JsonObject obj = new HoldingRequestBuilder().forInstance(instanceId).create(); - obj.put("id", id.toString()); - holdingsStorageClient.create(obj); - return id; - } } diff --git a/src/test/java/api/support/ApiTests.java b/src/test/java/api/support/ApiTests.java index 2d079664f..847182cfa 100644 --- a/src/test/java/api/support/ApiTests.java +++ b/src/test/java/api/support/ApiTests.java @@ -19,6 +19,7 @@ public abstract class ApiTests { private static boolean runningOnOwn; protected static OkapiHttpClient okapiClient; protected static OkapiHttpClient consortiumOkapiClient; + protected static OkapiHttpClient collegeOkapiClient; protected final ResourceClient holdingsStorageClient; protected final ResourceClient itemsStorageClient; protected final ResourceClient itemsClient; @@ -34,6 +35,8 @@ public abstract class ApiTests { protected final ResourceClient sourceRecordStorageClient; protected final ResourceClient consortiumItemsClient; protected final ResourceClient consortiumHoldingsStorageClient; + protected final ResourceClient collegeItemsClient; + protected final ResourceClient collegeHoldingsStorageClient; protected final InstanceRelationshipTypeFixture instanceRelationshipTypeFixture; protected final MarkItemFixture markItemFixture; @@ -57,6 +60,9 @@ public ApiTests() { consortiumHoldingsStorageClient = ResourceClient.forHoldingsStorage(consortiumOkapiClient); consortiumItemsClient = ResourceClient.forItemsStorage(consortiumOkapiClient); + + collegeHoldingsStorageClient = ResourceClient.forHoldingsStorage(collegeOkapiClient); + collegeItemsClient = ResourceClient.forItemsStorage(collegeOkapiClient); } @BeforeClass @@ -74,6 +80,7 @@ public static void before() okapiClient = ApiTestSuite.createOkapiHttpClient(); consortiumOkapiClient = ApiTestSuite.createOkapiHttpClient(ApiTestSuite.CONSORTIA_TENANT_ID); + collegeOkapiClient = ApiTestSuite.createOkapiHttpClient(ApiTestSuite.COLLEGE_TENANT_ID); } @AfterClass diff --git a/src/test/java/api/support/builders/HoldingsRecordUpdateOwnershipRequestBuilder.java b/src/test/java/api/support/builders/HoldingsRecordUpdateOwnershipRequestBuilder.java index 99adadbff..fe8e8009d 100644 --- a/src/test/java/api/support/builders/HoldingsRecordUpdateOwnershipRequestBuilder.java +++ b/src/test/java/api/support/builders/HoldingsRecordUpdateOwnershipRequestBuilder.java @@ -7,8 +7,8 @@ import java.util.UUID; import static org.folio.inventory.resources.MoveApi.HOLDINGS_RECORD_IDS; -import static org.folio.inventory.resources.MoveApi.TARGET_TENANT_ID; import static org.folio.inventory.resources.MoveApi.TO_INSTANCE_ID; +import static org.folio.inventory.support.MoveApiUtil.TARGET_TENANT_ID; public class HoldingsRecordUpdateOwnershipRequestBuilder extends AbstractBuilder { diff --git a/src/test/java/support/fakes/FakeStorageModuleBuilder.java b/src/test/java/support/fakes/FakeStorageModuleBuilder.java index 2ed59cf63..9ce8f1a18 100644 --- a/src/test/java/support/fakes/FakeStorageModuleBuilder.java +++ b/src/test/java/support/fakes/FakeStorageModuleBuilder.java @@ -24,7 +24,7 @@ public class FakeStorageModuleBuilder { private final List recordPreProcessors; FakeStorageModuleBuilder() { - this(null, null, List.of(ApiTestSuite.TENANT_ID, ApiTestSuite.CONSORTIA_TENANT_ID), new ArrayList<>(), true, "", + this(null, null, List.of(ApiTestSuite.TENANT_ID, ApiTestSuite.CONSORTIA_TENANT_ID, ApiTestSuite.COLLEGE_TENANT_ID), new ArrayList<>(), true, "", new ArrayList<>(), new HashMap<>(), Collections.emptyList()); } From 02ee8785d8de7baa50b1b14505fe0f5b04145b3b Mon Sep 17 00:00:00 2001 From: Roman_Chernetskyi Date: Fri, 14 Jun 2024 17:41:19 +0300 Subject: [PATCH 03/11] transfer along with all the Items attached to a selected Holdings --- .../folio/inventory/resources/MoveApi.java | 99 ++---------- .../resources/UpdateOwnershipApi.java | 83 +++++++--- .../folio/inventory/support/MoveApiUtil.java | 22 +++ .../HoldingsUpdateOwnershipApiTest.java | 151 +++++++++++++++++- 4 files changed, 246 insertions(+), 109 deletions(-) diff --git a/src/main/java/org/folio/inventory/resources/MoveApi.java b/src/main/java/org/folio/inventory/resources/MoveApi.java index 14216ddb0..3c0cc2687 100644 --- a/src/main/java/org/folio/inventory/resources/MoveApi.java +++ b/src/main/java/org/folio/inventory/resources/MoveApi.java @@ -2,20 +2,21 @@ import static java.util.stream.Collectors.toList; import static org.folio.inventory.support.JsonArrayHelper.toListOfStrings; -import static org.folio.inventory.support.http.server.JsonResponse.success; +import static org.folio.inventory.support.MoveApiUtil.createHoldingsRecordsFetchClient; +import static org.folio.inventory.support.MoveApiUtil.createHoldingsStorageClient; +import static org.folio.inventory.support.MoveApiUtil.createHttpClient; +import static org.folio.inventory.support.MoveApiUtil.createItemStorageClient; +import static org.folio.inventory.support.MoveApiUtil.createItemsFetchClient; +import static org.folio.inventory.support.MoveApiUtil.respond; import static org.folio.inventory.support.http.server.JsonResponse.unprocessableEntity; import static org.folio.inventory.validation.MoveValidator.holdingsMoveHasRequiredFields; import static org.folio.inventory.validation.MoveValidator.itemsMoveHasRequiredFields; -import java.net.MalformedURLException; -import java.net.URL; -import java.util.ArrayList; import java.util.List; import java.util.Optional; import java.util.concurrent.CompletableFuture; import java.util.stream.Collectors; -import org.apache.commons.collections4.ListUtils; import org.folio.HoldingsRecord; import org.folio.inventory.common.WebContext; import org.folio.inventory.domain.HoldingsRecordCollection; @@ -23,20 +24,17 @@ import org.folio.inventory.domain.items.ItemCollection; import org.folio.inventory.storage.Storage; import org.folio.inventory.storage.external.CollectionResourceClient; -import org.folio.inventory.storage.external.CqlQuery; import org.folio.inventory.storage.external.MultipleRecordsFetchClient; import org.folio.inventory.support.ItemUtil; -import org.folio.inventory.support.http.client.OkapiHttpClient; +import org.folio.inventory.support.MoveApiUtil; import org.folio.inventory.support.http.server.JsonResponse; import org.folio.inventory.support.http.server.ServerErrorResponse; import org.folio.inventory.support.http.server.ValidationError; import io.vertx.core.http.HttpClient; -import io.vertx.core.http.HttpServerResponse; import io.vertx.core.json.JsonObject; import io.vertx.ext.web.Router; import io.vertx.ext.web.RoutingContext; -import io.vertx.ext.web.client.WebClient; import io.vertx.ext.web.handler.BodyHandler; public class MoveApi extends AbstractInventoryResource { @@ -44,12 +42,6 @@ public class MoveApi extends AbstractInventoryResource { public static final String TO_INSTANCE_ID = "toInstanceId"; public static final String ITEM_IDS = "itemIds"; public static final String HOLDINGS_RECORD_IDS = "holdingsRecordIds"; - public static final String ITEM_STORAGE = "/item-storage/items"; - public static final String ITEMS_PROPERTY = "items"; - public static final String HOLDINGS_RECORDS_PROPERTY = "holdingsRecords"; - public static final String HOLDINGS_STORAGE = "/holdings-storage/holdings"; - private static final String HOLDINGS_ITEMS_PROPERTY = "holdingsItems"; - private static final String BARE_HOLDINGS_ITEMS_PROPERTY = "bareHoldingsItems"; public MoveApi(final Storage storage, final HttpClient client) { super(storage, client); @@ -84,10 +76,10 @@ private void moveItems(RoutingContext routingContext) { .thenAccept(holding -> { if (holding != null) { try { - final var itemsStorageClient = createItemStorageClient(routingContext, context); + final var itemsStorageClient = createItemStorageClient(createHttpClient(client, routingContext, context), context); final var itemsFetchClient = createItemsFetchClient(itemsStorageClient); - itemsFetchClient.find(itemIdsToUpdate, this::fetchByIdCql) + itemsFetchClient.find(itemIdsToUpdate, MoveApiUtil::fetchByIdCql) .thenAccept(jsons -> { List itemsToUpdate = updateHoldingsRecordIdForItems(toHoldingsRecordId, jsons); updateItems(routingContext, context, itemIdsToUpdate, itemsToUpdate); @@ -129,11 +121,11 @@ private void moveHoldings(RoutingContext routingContext) { return; } try { - CollectionResourceClient holdingsStorageClient = createHoldingsStorageClient(createHttpClient(routingContext, context), + CollectionResourceClient holdingsStorageClient = createHoldingsStorageClient(createHttpClient(client, routingContext, context), context); MultipleRecordsFetchClient holdingsRecordFetchClient = createHoldingsRecordsFetchClient(holdingsStorageClient); - holdingsRecordFetchClient.find(holdingsRecordsIdsToUpdate, this::fetchByIdCql) + holdingsRecordFetchClient.find(holdingsRecordsIdsToUpdate, MoveApiUtil::fetchByIdCql) .thenAccept(jsons -> { List holdingsRecordsToUpdate = updateInstanceIdForHoldings(toInstanceId, jsons); updateHoldings(routingContext, context, holdingsRecordsIdsToUpdate, holdingsRecordsToUpdate); @@ -177,17 +169,12 @@ private void updateItems(RoutingContext routingContext, WebContext context, List private List updateInstanceIdForHoldings(String toInstanceId, List jsons) { return jsons.stream() - .peek(this::removeExtraRedundantFields) + .peek(MoveApiUtil::removeExtraRedundantFields) .map(json -> json.mapTo(HoldingsRecord.class)) .map(holding -> holding.withInstanceId(toInstanceId)) .collect(toList()); } - private void removeExtraRedundantFields(JsonObject json) { - json.remove(HOLDINGS_ITEMS_PROPERTY); - json.remove(BARE_HOLDINGS_ITEMS_PROPERTY); - } - private void updateHoldings(RoutingContext routingContext, WebContext context, List idsToUpdate, List holdingsToUpdate) { HoldingsRecordCollection storageHoldingsRecordsCollection = storage.getHoldingsRecordCollection(context); @@ -204,66 +191,4 @@ private void updateHoldings(RoutingContext routingContext, WebContext context, L .collect(toList())) .thenAccept(updatedIds -> respond(routingContext, idsToUpdate, updatedIds)); } - - private void respond(RoutingContext routingContext, List itemIdsToUpdate, List updatedItemIds) { - List nonUpdatedIds = ListUtils.subtract(itemIdsToUpdate, updatedItemIds); - HttpServerResponse response = routingContext.response(); - if (nonUpdatedIds.isEmpty()) { - successWithEmptyIds(response); - } else { - successWithIds(response, nonUpdatedIds); - } - } - - private OkapiHttpClient createHttpClient(RoutingContext routingContext, WebContext context) throws MalformedURLException { - return new OkapiHttpClient(WebClient.wrap(client), context, - exception -> ServerErrorResponse.internalError(routingContext.response(), - String.format("Failed to contact storage module: %s", exception.toString()))); - } - - private CollectionResourceClient createStorageClient(OkapiHttpClient client, WebContext context, String storageUrl) - throws MalformedURLException { - - return new CollectionResourceClient(client, new URL(context.getOkapiLocation() + storageUrl)); - } - - private CollectionResourceClient createItemStorageClient( - RoutingContext routingContext, WebContext context) throws MalformedURLException { - - return createStorageClient(createHttpClient(routingContext, context), - context, ITEM_STORAGE); - } - - private CollectionResourceClient createHoldingsStorageClient(OkapiHttpClient client, WebContext context) - throws MalformedURLException { - return createStorageClient(client, context, HOLDINGS_STORAGE); - } - - private CqlQuery fetchByIdCql(List ids) { - return CqlQuery.exactMatchAny("id", ids); - } - - private MultipleRecordsFetchClient createFetchClient(CollectionResourceClient client, String propertyName) { - return MultipleRecordsFetchClient.builder() - .withCollectionPropertyName(propertyName) - .withExpectedStatus(200) - .withCollectionResourceClient(client) - .build(); - } - - private MultipleRecordsFetchClient createItemsFetchClient(CollectionResourceClient client) { - return createFetchClient(client, ITEMS_PROPERTY); - } - - private MultipleRecordsFetchClient createHoldingsRecordsFetchClient(CollectionResourceClient client) { - return createFetchClient(client, HOLDINGS_RECORDS_PROPERTY); - } - - private void successWithIds(HttpServerResponse response, List ids) { - success(response, new JsonObject().put("nonUpdatedIds", ids)); - } - - private void successWithEmptyIds(HttpServerResponse response) { - successWithIds(response, new ArrayList<>()); - } } diff --git a/src/main/java/org/folio/inventory/resources/UpdateOwnershipApi.java b/src/main/java/org/folio/inventory/resources/UpdateOwnershipApi.java index 090126bcf..67913d140 100644 --- a/src/main/java/org/folio/inventory/resources/UpdateOwnershipApi.java +++ b/src/main/java/org/folio/inventory/resources/UpdateOwnershipApi.java @@ -12,12 +12,16 @@ import org.folio.inventory.common.Context; import org.folio.inventory.common.WebContext; import org.folio.inventory.consortium.services.ConsortiumService; +import org.folio.inventory.domain.AsynchronousCollection; import org.folio.inventory.domain.HoldingsRecordCollection; +import org.folio.inventory.domain.items.Item; +import org.folio.inventory.domain.items.ItemCollection; import org.folio.inventory.exceptions.BadRequestException; import org.folio.inventory.exceptions.NotFoundException; import org.folio.inventory.storage.Storage; import org.folio.inventory.storage.external.CollectionResourceClient; import org.folio.inventory.storage.external.MultipleRecordsFetchClient; +import org.folio.inventory.support.ItemUtil; import org.folio.inventory.support.MoveApiUtil; import java.lang.invoke.MethodHandles; @@ -31,6 +35,8 @@ import static org.folio.inventory.support.MoveApiUtil.createHoldingsRecordsFetchClient; import static org.folio.inventory.support.MoveApiUtil.createHoldingsStorageClient; import static org.folio.inventory.support.MoveApiUtil.createHttpClient; +import static org.folio.inventory.support.MoveApiUtil.createItemStorageClient; +import static org.folio.inventory.support.MoveApiUtil.createItemsFetchClient; import static org.folio.inventory.support.MoveApiUtil.respond; import static org.folio.inventory.support.http.server.JsonResponse.unprocessableEntity; import static org.folio.inventory.validation.UpdateOwnershipValidator.updateOwnershipHasRequiredFields; @@ -50,12 +56,12 @@ public UpdateOwnershipApi(Storage storage, HttpClient client, ConsortiumService @Override public void register(Router router) { router.post("/inventory/items/update-ownership") - .handler(this::updateItemsOwnership); + .handler(this::processUpdateItemsOwnership); router.post("/inventory/holdings/update-ownership") - .handler(this::updateHoldingsOwnership); + .handler(this::processUpdateHoldingsOwnership); } - private void updateHoldingsOwnership(RoutingContext routingContext) { + private void processUpdateHoldingsOwnership(RoutingContext routingContext) { try { final var context = new WebContext(routingContext); final var updateOwnershipRequest = routingContext.body().asJsonObject(); @@ -79,7 +85,8 @@ private void updateHoldingsOwnership(RoutingContext routingContext) { .thenCompose(instance -> { if (instance != null) { if (instance.getSource().equals(CONSORTIUM_MARC.getValue()) || instance.getSource().equals(CONSORTIUM_FOLIO.getValue())) { - return updateOwnershipOfHoldingsRecords(holdingsUpdateOwnership, routingContext, context); + Context targetTenantContext = constructContext(holdingsUpdateOwnership.getTargetTenantId(), context.getToken(), context.getOkapiLocation()); + return updateOwnershipOfHoldingsRecords(holdingsUpdateOwnership, routingContext, context, targetTenantContext); } else { String instanceNotSharedErrorMessage = String.format(INSTANCE_NOT_SHARED, holdingsUpdateOwnership.getToInstanceId()); LOGGER.warn("updateHoldingsOwnership:: " + instanceNotSharedErrorMessage); @@ -94,7 +101,7 @@ private void updateHoldingsOwnership(RoutingContext routingContext) { } String notInConsortiaErrorMessage = String.format(TENANT_NOT_IN_CONSORTIA, context.getTenantId()); LOGGER.warn("updateHoldingsOwnership:: " + notInConsortiaErrorMessage, context); - return CompletableFuture.failedFuture(new BadRequestException(notInConsortiaErrorMessage)); // NEED TEST + return CompletableFuture.failedFuture(new BadRequestException(notInConsortiaErrorMessage)); }) .thenAccept(updateHoldingsRecords -> respond(routingContext, holdingsUpdateOwnership.getHoldingsRecordIds(), updateHoldingsRecords)) .exceptionally(throwable -> { @@ -109,15 +116,13 @@ private void updateHoldingsOwnership(RoutingContext routingContext) { } } - private void updateItemsOwnership(RoutingContext routingContext) { - // should be implemented in MODINV-1031 + private void processUpdateItemsOwnership(RoutingContext routingContext) { + // should be implemented in MODINV-955 } private CompletableFuture> updateOwnershipOfHoldingsRecords(HoldingsUpdateOwnership holdingsUpdateOwnership, RoutingContext routingContext, - WebContext context) { + WebContext context, Context targetTenantContext) { try { - Context targetTenantContext = constructContext(holdingsUpdateOwnership.getTargetTenantId(), context.getToken(), context.getOkapiLocation()); - CollectionResourceClient holdingsStorageClient = createHoldingsStorageClient(createHttpClient(client, routingContext, context), context); MultipleRecordsFetchClient holdingsRecordFetchClient = createHoldingsRecordsFetchClient(holdingsStorageClient); @@ -126,20 +131,58 @@ private CompletableFuture> updateOwnershipOfHoldingsRecords(Holding HoldingsRecordCollection targetTenantHoldingsRecordCollection = storage.getHoldingsRecordCollection(targetTenantContext); return holdingsRecordFetchClient.find(holdingsUpdateOwnership.getHoldingsRecordIds(), MoveApiUtil::fetchByIdCql) - .thenCompose(jsons -> - createHoldings(jsons, holdingsUpdateOwnership.getToInstanceId(), targetTenantHoldingsRecordCollection) - .thenCompose(createdHoldings -> { - List createdHoldingsIds = createdHoldings.stream().map(HoldingsRecord::getId).toList(); - return deleteHoldings(createdHoldingsIds, sourceTenantHoldingsRecordCollection); - })); + .thenCompose(jsons -> createHoldings(jsons, holdingsUpdateOwnership.getToInstanceId(), targetTenantHoldingsRecordCollection)) + .thenApply(createdHoldings -> createdHoldings.stream().map(HoldingsRecord::getId).toList()) + .thenCompose(createdHoldingsIds -> + transferAttachedItems(createdHoldingsIds, routingContext, context, targetTenantContext) + .thenCompose(items -> deleteCollectionItems(createdHoldingsIds, sourceTenantHoldingsRecordCollection))); + } catch (Exception e) { + LOGGER.warn("updateOwnershipOfHoldingsRecords:: Error during update ownership of holdings {}, to tenant: {}", + holdingsUpdateOwnership.getHoldingsRecordIds(), holdingsUpdateOwnership.getTargetTenantId(), e); + return CompletableFuture.failedFuture(e); + } + } + + private CompletableFuture> transferAttachedItems(List holdingsRecordIds, RoutingContext routingContext, + WebContext context, Context targetTenantContext) { + try { + CollectionResourceClient itemsStorageClient = createItemStorageClient(createHttpClient(client, routingContext, context), context); + MultipleRecordsFetchClient itemsFetchClient = createItemsFetchClient(itemsStorageClient); + + ItemCollection sourceTenantItemCollection = storage.getItemCollection(context); + ItemCollection targetTenantItemCollection = storage.getItemCollection(targetTenantContext); + + return itemsFetchClient.find(holdingsRecordIds, MoveApiUtil::fetchByHoldingsRecordIdCql) + .thenCompose(jsons -> createItems(jsons, targetTenantItemCollection)) + .thenApply(items -> items.stream().map(Item::getId).toList()) + .thenCompose(itemIds -> deleteCollectionItems(itemIds, sourceTenantItemCollection)); } catch (Exception e) { + LOGGER.warn("transferAttachedItems:: Error during transfer attached items for holdings {}, to tenant: {}", + holdingsRecordIds, targetTenantContext.getTenantId(), e); return CompletableFuture.failedFuture(e); } } + private CompletableFuture> createItems(List jsons, ItemCollection itemCollection) { + List itemRecordsToUpdateOwnership = jsons.stream() + .map(ItemUtil::fromStoredItemRepresentation) + .toList(); + + List> createFutures = itemRecordsToUpdateOwnership.stream() + .map(itemCollection::add) + .toList(); + + return CompletableFuture.allOf(createFutures.toArray(new CompletableFuture[0])) + .handle((vVoid, throwable) -> createFutures.stream() + .filter(future -> !future.isCompletedExceptionally()) + .map(CompletableFuture::join) + .toList()); + } + private CompletableFuture> createHoldings(List jsons, String instanceId, HoldingsRecordCollection holdingsRecordCollection) { List holdingsRecordsToUpdateOwnership = jsons.stream() + .peek(MoveApiUtil::removeExtraRedundantFields) .map(json -> json.mapTo(HoldingsRecord.class)) .filter(holdingsRecord -> holdingsRecord.getInstanceId().equals(instanceId)) .toList(); @@ -155,11 +198,11 @@ private CompletableFuture> createHoldings(List .toList()); } - private CompletableFuture> deleteHoldings(List holdingsRecordIds, HoldingsRecordCollection holdingsRecordCollection) { - List> deleteFutures = holdingsRecordIds.stream() - .map(holdingsRecordId -> { + private CompletableFuture> deleteCollectionItems(List collectionItemIds, AsynchronousCollection collection) { + List> deleteFutures = collectionItemIds.stream() + .map(collectionItemId -> { Promise promise = Promise.promise(); - holdingsRecordCollection.delete(holdingsRecordId, success -> promise.complete(holdingsRecordId), failure -> promise.fail(failure.getReason())); + collection.delete(collectionItemId, success -> promise.complete(collectionItemId), failure -> promise.fail(failure.getReason())); return promise.future().toCompletionStage().toCompletableFuture(); }).toList(); diff --git a/src/main/java/org/folio/inventory/support/MoveApiUtil.java b/src/main/java/org/folio/inventory/support/MoveApiUtil.java index c024171e1..f416567a8 100644 --- a/src/main/java/org/folio/inventory/support/MoveApiUtil.java +++ b/src/main/java/org/folio/inventory/support/MoveApiUtil.java @@ -24,6 +24,10 @@ public final class MoveApiUtil { public static final String HOLDINGS_STORAGE = "/holdings-storage/holdings"; public static final String HOLDINGS_RECORDS_PROPERTY = "holdingsRecords"; public static final String TARGET_TENANT_ID = "targetTenantId"; + public static final String ITEM_STORAGE = "/item-storage/items"; + public static final String ITEMS_PROPERTY = "items"; + private static final String HOLDINGS_ITEMS_PROPERTY = "holdingsItems"; + private static final String BARE_HOLDINGS_ITEMS_PROPERTY = "bareHoldingsItems"; private MoveApiUtil() { } @@ -53,14 +57,27 @@ public static CollectionResourceClient createHoldingsStorageClient(OkapiHttpClie return createStorageClient(client, context, HOLDINGS_STORAGE); } + public static CollectionResourceClient createItemStorageClient(OkapiHttpClient client, WebContext context) + throws MalformedURLException { + return createStorageClient(client, context, ITEM_STORAGE); + } + public static MultipleRecordsFetchClient createHoldingsRecordsFetchClient(CollectionResourceClient client) { return createFetchClient(client, HOLDINGS_RECORDS_PROPERTY); } + public static MultipleRecordsFetchClient createItemsFetchClient(CollectionResourceClient client) { + return createFetchClient(client, ITEMS_PROPERTY); + } + public static CqlQuery fetchByIdCql(List ids) { return CqlQuery.exactMatchAny("id", ids); } + public static CqlQuery fetchByHoldingsRecordIdCql(List ids) { + return CqlQuery.exactMatchAny("holdingsRecordId", ids); + } + public static void successWithEmptyIds(HttpServerResponse response) { successWithIds(response, new ArrayList<>()); } @@ -69,6 +86,11 @@ public static void successWithIds(HttpServerResponse response, List ids) success(response, new JsonObject().put("nonUpdatedIds", ids)); } + public static void removeExtraRedundantFields(JsonObject json) { + json.remove(HOLDINGS_ITEMS_PROPERTY); + json.remove(BARE_HOLDINGS_ITEMS_PROPERTY); + } + public static void respond(RoutingContext routingContext, List itemIdsToUpdate, List updatedItemIds) { List nonUpdatedIds = ListUtils.subtract(itemIdsToUpdate, updatedItemIds); HttpServerResponse response = routingContext.response(); diff --git a/src/test/java/api/holdings/HoldingsUpdateOwnershipApiTest.java b/src/test/java/api/holdings/HoldingsUpdateOwnershipApiTest.java index 7bc37018c..a9f5e0935 100644 --- a/src/test/java/api/holdings/HoldingsUpdateOwnershipApiTest.java +++ b/src/test/java/api/holdings/HoldingsUpdateOwnershipApiTest.java @@ -6,11 +6,13 @@ import api.support.InstanceApiClient; import api.support.builders.HoldingRequestBuilder; import api.support.builders.HoldingsRecordUpdateOwnershipRequestBuilder; +import api.support.builders.ItemRequestBuilder; import io.vertx.core.http.HttpMethod; import io.vertx.core.json.JsonArray; import io.vertx.core.json.JsonObject; import junitparams.JUnitParamsRunner; import org.apache.http.HttpStatus; +import org.folio.inventory.domain.items.ItemStatusName; import org.folio.inventory.support.http.client.Response; import org.joda.time.DateTime; import org.junit.After; @@ -32,6 +34,7 @@ import static api.support.InstanceSamples.smallAngryPlanet; import static org.folio.inventory.domain.instances.InstanceSource.CONSORTIUM_FOLIO; import static org.folio.inventory.domain.instances.InstanceSource.FOLIO; +import static org.folio.inventory.support.ItemUtil.HOLDINGS_RECORD_ID; import static org.folio.inventory.support.http.ContentType.APPLICATION_JSON; import static org.hamcrest.CoreMatchers.is; import static org.hamcrest.MatcherAssert.assertThat; @@ -39,7 +42,6 @@ import static org.hamcrest.Matchers.equalTo; import static support.matchers.ResponseMatchers.hasValidationError; -//@Ignore @RunWith(JUnitParamsRunner.class) public class HoldingsUpdateOwnershipApiTest extends ApiTests { private static final String INSTANCE_ID = "instanceId"; @@ -87,6 +89,152 @@ public void canUpdateHoldingsOwnershipToDifferentTenant() throws MalformedURLExc Assert.assertEquals(instanceId.toString(), targetTenantHoldingsRecord2.getJson().getString(INSTANCE_ID)); } + @Test + public void canUpdateHoldingsOwnershipWithRelatedItemsToDifferentTenant() throws MalformedURLException, ExecutionException, InterruptedException, TimeoutException { + UUID instanceId = UUID.randomUUID(); + JsonObject instance = smallAngryPlanet(instanceId); + + InstanceApiClient.createInstance(okapiClient, instance.put("source", CONSORTIUM_FOLIO.getValue())); + InstanceApiClient.createInstance(consortiumOkapiClient, instance.put("source", FOLIO.getValue())); + + final UUID createHoldingsRecord1 = createHoldingForInstance(instanceId); + final UUID createHoldingsRecord2 = createHoldingForInstance(instanceId); + + final var firstItem = itemsClient.create( + new ItemRequestBuilder() + .forHolding(createHoldingsRecord1) + .withBarcode("645398607547") + .withStatus(ItemStatusName.AVAILABLE.value())); + + final var secondItem = itemsClient.create( + new ItemRequestBuilder() + .forHolding(createHoldingsRecord2) + .withBarcode("645398607546") + .withStatus(ItemStatusName.AVAILABLE.value())); + + JsonObject holdingsRecordUpdateOwnershipRequestBody = new HoldingsRecordUpdateOwnershipRequestBuilder(instanceId, + new JsonArray(List.of(createHoldingsRecord1.toString(), createHoldingsRecord2.toString())), ApiTestSuite.COLLEGE_TENANT_ID).create(); + + Response postHoldingsUpdateOwnershipResponse = updateHoldingsRecordsOwnership(holdingsRecordUpdateOwnershipRequestBody); + + assertThat(postHoldingsUpdateOwnershipResponse.getStatusCode(), is(200)); + assertThat(new JsonObject(postHoldingsUpdateOwnershipResponse.getBody()).getJsonArray("nonUpdatedIds").size(), is(0)); + assertThat(postHoldingsUpdateOwnershipResponse.getContentType(), containsString(APPLICATION_JSON)); + + // Verify Holdings ownership updated + Response sourceTenantHoldingsRecord1 = holdingsStorageClient.getById(createHoldingsRecord1); + Response targetTenantHoldingsRecord1 = collegeHoldingsStorageClient.getById(createHoldingsRecord1); + + Assert.assertEquals(HttpStatus.SC_NOT_FOUND, sourceTenantHoldingsRecord1.getStatusCode()); + Assert.assertEquals(instanceId.toString(), targetTenantHoldingsRecord1.getJson().getString(INSTANCE_ID)); + + Response sourceTenantHoldingsRecord2 = holdingsStorageClient.getById(createHoldingsRecord1); + Response targetTenantHoldingsRecord2 = collegeHoldingsStorageClient.getById(createHoldingsRecord1); + + Assert.assertEquals(HttpStatus.SC_NOT_FOUND, sourceTenantHoldingsRecord2.getStatusCode()); + Assert.assertEquals(instanceId.toString(), targetTenantHoldingsRecord2.getJson().getString(INSTANCE_ID)); + + // Verify related Items ownership updated + Response sourceTenantItem1 = itemsClient.getById(firstItem.getId()); + Response targetTenantItem1 = collegeItemsClient.getById(firstItem.getId()); + + assertThat(HttpStatus.SC_NOT_FOUND, is(sourceTenantItem1.getStatusCode())); + assertThat(targetTenantItem1.getJson().getString(HOLDINGS_RECORD_ID), is(createHoldingsRecord1.toString())); + + Response sourceTenantItem2 = itemsClient.getById(secondItem.getId()); + Response targetTenantItem2 = collegeItemsClient.getById(secondItem.getId()); + + assertThat(HttpStatus.SC_NOT_FOUND, is(sourceTenantItem2.getStatusCode())); + assertThat(targetTenantItem2.getJson().getString(HOLDINGS_RECORD_ID), is(createHoldingsRecord2.toString())); + } + + @Test + public void canUpdateHoldingsOwnershipIfErrorUpdatingRelatedItemsToDifferentTenant() throws MalformedURLException, ExecutionException, InterruptedException, TimeoutException { + UUID instanceId = UUID.randomUUID(); + JsonObject instance = smallAngryPlanet(instanceId); + + InstanceApiClient.createInstance(okapiClient, instance.put("source", CONSORTIUM_FOLIO.getValue())); + InstanceApiClient.createInstance(consortiumOkapiClient, instance.put("source", FOLIO.getValue())); + + final UUID createHoldingsRecord1 = createHoldingForInstance(instanceId); + final UUID createHoldingsRecord2 = createHoldingForInstance(instanceId); + + final var firstItem = itemsClient.create( + new ItemRequestBuilder() + .forHolding(createHoldingsRecord1) + .withBarcode("645398607547") + .withStatus(ItemStatusName.AVAILABLE.value())); + + final var secondItem = itemsClient.create( + new ItemRequestBuilder() + .forHolding(createHoldingsRecord2) + .withBarcode("645398607546") + .withStatus(ItemStatusName.AVAILABLE.value())); + + final JsonObject expectedErrorResponse = new JsonObject().put("message", "Server error"); + collegeItemsClient.emulateFailure( + new EndpointFailureDescriptor() + .setFailureExpireDate(DateTime.now().plusSeconds(2).toDate()) + .setStatusCode(500) + .setContentType("application/json") + .setBody(expectedErrorResponse.toString()) + .setMethod(HttpMethod.POST.name())); + + holdingsStorageClient.emulateFailure( + new EndpointFailureDescriptor() + .setFailureExpireDate(DateTime.now().plusSeconds(2).toDate()) + .setStatusCode(500) + .setContentType("application/json") + .setBody(expectedErrorResponse.toString()) + .setMethod(HttpMethod.DELETE.name())); + + JsonObject holdingsRecordUpdateOwnershipRequestBody = new HoldingsRecordUpdateOwnershipRequestBuilder(instanceId, + new JsonArray(List.of(createHoldingsRecord1.toString(), createHoldingsRecord2.toString())), ApiTestSuite.COLLEGE_TENANT_ID).create(); + + Response postHoldingsUpdateOwnershipResponse = updateHoldingsRecordsOwnership(holdingsRecordUpdateOwnershipRequestBody); + + assertThat(postHoldingsUpdateOwnershipResponse.getStatusCode(), is(200)); + + List nonUpdatedIdsIds = postHoldingsUpdateOwnershipResponse.getJson() + .getJsonArray("nonUpdatedIds") + .getList(); + + assertThat(nonUpdatedIdsIds.size(), is(2)); + assertThat(nonUpdatedIdsIds.get(0), equalTo(createHoldingsRecord1.toString())); + assertThat(nonUpdatedIdsIds.get(1), equalTo(createHoldingsRecord2.toString())); + + assertThat(postHoldingsUpdateOwnershipResponse.getContentType(), containsString(APPLICATION_JSON)); + + // Verify Holdings ownership updated + Response sourceTenantHoldingsRecord1 = holdingsStorageClient.getById(createHoldingsRecord1); + Response targetTenantHoldingsRecord1 = collegeHoldingsStorageClient.getById(createHoldingsRecord1); + + Assert.assertEquals(instanceId.toString(), sourceTenantHoldingsRecord1.getJson().getString(INSTANCE_ID)); + Assert.assertEquals(instanceId.toString(), targetTenantHoldingsRecord1.getJson().getString(INSTANCE_ID)); + + Response sourceTenantHoldingsRecord2 = holdingsStorageClient.getById(createHoldingsRecord1); + Response targetTenantHoldingsRecord2 = collegeHoldingsStorageClient.getById(createHoldingsRecord1); + + Assert.assertEquals(instanceId.toString(), sourceTenantHoldingsRecord2.getJson().getString(INSTANCE_ID)); + Assert.assertEquals(instanceId.toString(), targetTenantHoldingsRecord2.getJson().getString(INSTANCE_ID)); + + // Verify related Items ownership updated + Response sourceTenantItem1 = itemsClient.getById(firstItem.getId()); + Response targetTenantItem1 = collegeItemsClient.getById(firstItem.getId()); + + assertThat(HttpStatus.SC_NOT_FOUND, is(targetTenantItem1.getStatusCode())); + assertThat(sourceTenantItem1.getJson().getString(HOLDINGS_RECORD_ID), is(createHoldingsRecord1.toString())); + + Response sourceTenantItem2 = itemsClient.getById(secondItem.getId()); + Response targetTenantItem2 = collegeItemsClient.getById(secondItem.getId()); + + assertThat(HttpStatus.SC_NOT_FOUND, is(targetTenantItem2.getStatusCode())); + assertThat(sourceTenantItem2.getJson().getString(HOLDINGS_RECORD_ID), is(createHoldingsRecord2.toString())); + + collegeItemsClient.disableFailureEmulation(); + holdingsStorageClient.disableFailureEmulation(); + } + @Test public void shouldReportErrorsWhenOnlySomeRequestedHoldingsRecordsCouldNotBeUpdated() throws InterruptedException, MalformedURLException, TimeoutException, ExecutionException { UUID instanceId = UUID.randomUUID(); @@ -352,7 +500,6 @@ public void cannotUpdateHoldingsRecordOwnershipDueToHoldingsRecordDeleteError() collegeHoldingsStorageClient.disableFailureEmulation(); } - @Ignore @Test public void canUpdateHoldingsRecordOwnershipToDifferentInstanceWithExtraRedundantFields() throws InterruptedException, MalformedURLException, TimeoutException, ExecutionException { UUID instanceId = UUID.randomUUID(); From b5765aec945c58169afa11b44949d86192773f5f Mon Sep 17 00:00:00 2001 From: Roman_Chernetskyi Date: Tue, 18 Jun 2024 13:31:24 +0300 Subject: [PATCH 04/11] Response with error message --- pom.xml | 2 +- ramls/update_ownership_response.json | 24 +++ .../resources/UpdateOwnershipApi.java | 103 ++++++++--- .../folio/inventory/support/MoveApiUtil.java | 6 + .../HoldingsUpdateOwnershipApiTest.java | 165 ++++++++++-------- 5 files changed, 205 insertions(+), 95 deletions(-) create mode 100644 ramls/update_ownership_response.json diff --git a/pom.xml b/pom.xml index 0b3c44ae0..ac5ff7c81 100644 --- a/pom.xml +++ b/pom.xml @@ -447,7 +447,7 @@ ${basedir}/ramls/mappingMetadataDto.json ${basedir}/ramls/holdings_update_ownership.json ${basedir}/ramls/items_update_ownership.json - ${basedir}/ramls/move_response.json + ${basedir}/ramls/update_ownership_response.json org.folio true diff --git a/ramls/update_ownership_response.json b/ramls/update_ownership_response.json new file mode 100644 index 000000000..ab603298f --- /dev/null +++ b/ramls/update_ownership_response.json @@ -0,0 +1,24 @@ +{ + "$schema": "http://json-schema.org/draft-04/schema#", + "description": "Holder for errors during holdings/item update ownership", + "type": "object", + "properties": { + "notUpdatedEntities": { + "description": "Items/holdings errors", + "type": "array", + "items": { + "type": "object", + "properties": { + "entityId": { + "$ref": "uuid.json" + }, + "errorMessage": { + "type": "string", + "description": "Error message" + } + } + } + } + }, + "additionalProperties": false +} diff --git a/src/main/java/org/folio/inventory/resources/UpdateOwnershipApi.java b/src/main/java/org/folio/inventory/resources/UpdateOwnershipApi.java index 67913d140..0bca384a7 100644 --- a/src/main/java/org/folio/inventory/resources/UpdateOwnershipApi.java +++ b/src/main/java/org/folio/inventory/resources/UpdateOwnershipApi.java @@ -5,14 +5,15 @@ import io.vertx.core.json.JsonObject; import io.vertx.ext.web.Router; import io.vertx.ext.web.RoutingContext; +import org.apache.commons.collections4.ListUtils; import org.apache.logging.log4j.LogManager; import org.apache.logging.log4j.Logger; import org.folio.HoldingsRecord; import org.folio.HoldingsUpdateOwnership; +import org.folio.NotUpdatedEntity; import org.folio.inventory.common.Context; import org.folio.inventory.common.WebContext; import org.folio.inventory.consortium.services.ConsortiumService; -import org.folio.inventory.domain.AsynchronousCollection; import org.folio.inventory.domain.HoldingsRecordCollection; import org.folio.inventory.domain.items.Item; import org.folio.inventory.domain.items.ItemCollection; @@ -25,8 +26,10 @@ import org.folio.inventory.support.MoveApiUtil; import java.lang.invoke.MethodHandles; +import java.util.ArrayList; import java.util.List; import java.util.concurrent.CompletableFuture; +import java.util.concurrent.CompletionException; import static org.folio.inventory.dataimport.handlers.matching.util.EventHandlingUtil.constructContext; import static org.folio.inventory.domain.instances.InstanceSource.CONSORTIUM_FOLIO; @@ -46,6 +49,8 @@ public class UpdateOwnershipApi extends AbstractInventoryResource { public static final String INSTANCE_NOT_SHARED = "Instance with id: %s is not shared"; public static final String INSTANCE_NOT_FOUND_AT_SOURCE_TENANT = "Instance with id: %s not found at source tenant, tenant: %s"; public static final String TENANT_NOT_IN_CONSORTIA = "%s tenant is not in consortia"; + public static final String INSTANCE_NOT_FOUND = "Instance with id: %s not found on tenant: %s"; + private final ConsortiumService consortiumService; public UpdateOwnershipApi(Storage storage, HttpClient client, ConsortiumService consortiumService) { @@ -74,6 +79,8 @@ private void processUpdateHoldingsOwnership(RoutingContext routingContext) { } var holdingsUpdateOwnership = updateOwnershipRequest.mapTo(HoldingsUpdateOwnership.class); + List notUpdatedEntities = new ArrayList<>(); + LOGGER.info("updateHoldingsOwnership:: Started updating ownership of holdings record: {}, to tenant: {}", holdingsUpdateOwnership.getHoldingsRecordIds(), holdingsUpdateOwnership.getTargetTenantId()); @@ -86,7 +93,7 @@ private void processUpdateHoldingsOwnership(RoutingContext routingContext) { if (instance != null) { if (instance.getSource().equals(CONSORTIUM_MARC.getValue()) || instance.getSource().equals(CONSORTIUM_FOLIO.getValue())) { Context targetTenantContext = constructContext(holdingsUpdateOwnership.getTargetTenantId(), context.getToken(), context.getOkapiLocation()); - return updateOwnershipOfHoldingsRecords(holdingsUpdateOwnership, routingContext, context, targetTenantContext); + return updateOwnershipOfHoldingsRecords(holdingsUpdateOwnership, notUpdatedEntities, routingContext, context, targetTenantContext); } else { String instanceNotSharedErrorMessage = String.format(INSTANCE_NOT_SHARED, holdingsUpdateOwnership.getToInstanceId()); LOGGER.warn("updateHoldingsOwnership:: " + instanceNotSharedErrorMessage); @@ -103,7 +110,7 @@ private void processUpdateHoldingsOwnership(RoutingContext routingContext) { LOGGER.warn("updateHoldingsOwnership:: " + notInConsortiaErrorMessage, context); return CompletableFuture.failedFuture(new BadRequestException(notInConsortiaErrorMessage)); }) - .thenAccept(updateHoldingsRecords -> respond(routingContext, holdingsUpdateOwnership.getHoldingsRecordIds(), updateHoldingsRecords)) + .thenAccept(updateHoldingsRecords -> respond(routingContext, notUpdatedEntities)) .exceptionally(throwable -> { LOGGER.warn("updateHoldingsOwnership:: Error during update ownership of holdings {}, to tenant: {}", holdingsUpdateOwnership.getHoldingsRecordIds(), holdingsUpdateOwnership.getTargetTenantId(), throwable); @@ -120,7 +127,8 @@ private void processUpdateItemsOwnership(RoutingContext routingContext) { // should be implemented in MODINV-955 } - private CompletableFuture> updateOwnershipOfHoldingsRecords(HoldingsUpdateOwnership holdingsUpdateOwnership, RoutingContext routingContext, + private CompletableFuture> updateOwnershipOfHoldingsRecords(HoldingsUpdateOwnership holdingsUpdateOwnership, + List notUpdatedEntities, RoutingContext routingContext, WebContext context, Context targetTenantContext) { try { CollectionResourceClient holdingsStorageClient = createHoldingsStorageClient(createHttpClient(client, routingContext, context), @@ -131,11 +139,17 @@ private CompletableFuture> updateOwnershipOfHoldingsRecords(Holding HoldingsRecordCollection targetTenantHoldingsRecordCollection = storage.getHoldingsRecordCollection(targetTenantContext); return holdingsRecordFetchClient.find(holdingsUpdateOwnership.getHoldingsRecordIds(), MoveApiUtil::fetchByIdCql) - .thenCompose(jsons -> createHoldings(jsons, holdingsUpdateOwnership.getToInstanceId(), targetTenantHoldingsRecordCollection)) - .thenApply(createdHoldings -> createdHoldings.stream().map(HoldingsRecord::getId).toList()) - .thenCompose(createdHoldingsIds -> - transferAttachedItems(createdHoldingsIds, routingContext, context, targetTenantContext) - .thenCompose(items -> deleteCollectionItems(createdHoldingsIds, sourceTenantHoldingsRecordCollection))); + .thenCompose(jsons -> { + processNotFoundedInstances(holdingsUpdateOwnership.getHoldingsRecordIds(), notUpdatedEntities, context, jsons); + return createHoldings(jsons, notUpdatedEntities, holdingsUpdateOwnership.getToInstanceId(), targetTenantHoldingsRecordCollection); + }) + .thenCompose(createdHoldings -> { + List createdHoldingsIds = createdHoldings.stream().map(HoldingsRecord::getId).toList(); + + return transferAttachedItems(createdHoldingsIds, notUpdatedEntities, routingContext, context, targetTenantContext) + .thenCompose(itemIds -> + deleteHoldings(getHoldingsToDelete(notUpdatedEntities, createdHoldings), notUpdatedEntities, sourceTenantHoldingsRecordCollection)); + }); } catch (Exception e) { LOGGER.warn("updateOwnershipOfHoldingsRecords:: Error during update ownership of holdings {}, to tenant: {}", holdingsUpdateOwnership.getHoldingsRecordIds(), holdingsUpdateOwnership.getTargetTenantId(), e); @@ -143,8 +157,8 @@ private CompletableFuture> updateOwnershipOfHoldingsRecords(Holding } } - private CompletableFuture> transferAttachedItems(List holdingsRecordIds, RoutingContext routingContext, - WebContext context, Context targetTenantContext) { + private CompletableFuture> transferAttachedItems(List holdingsRecordIds, List notUpdatedEntities, + RoutingContext routingContext, WebContext context, Context targetTenantContext) { try { CollectionResourceClient itemsStorageClient = createItemStorageClient(createHttpClient(client, routingContext, context), context); MultipleRecordsFetchClient itemsFetchClient = createItemsFetchClient(itemsStorageClient); @@ -153,9 +167,8 @@ private CompletableFuture> transferAttachedItems(List holdi ItemCollection targetTenantItemCollection = storage.getItemCollection(targetTenantContext); return itemsFetchClient.find(holdingsRecordIds, MoveApiUtil::fetchByHoldingsRecordIdCql) - .thenCompose(jsons -> createItems(jsons, targetTenantItemCollection)) - .thenApply(items -> items.stream().map(Item::getId).toList()) - .thenCompose(itemIds -> deleteCollectionItems(itemIds, sourceTenantItemCollection)); + .thenCompose(jsons -> createItems(jsons, notUpdatedEntities, targetTenantItemCollection)) + .thenCompose(items -> deleteItems(items, notUpdatedEntities, sourceTenantItemCollection)); } catch (Exception e) { LOGGER.warn("transferAttachedItems:: Error during transfer attached items for holdings {}, to tenant: {}", holdingsRecordIds, targetTenantContext.getTenantId(), e); @@ -163,13 +176,18 @@ private CompletableFuture> transferAttachedItems(List holdi } } - private CompletableFuture> createItems(List jsons, ItemCollection itemCollection) { + private CompletableFuture> createItems(List jsons, List notUpdatedEntities, ItemCollection itemCollection) { List itemRecordsToUpdateOwnership = jsons.stream() .map(ItemUtil::fromStoredItemRepresentation) .toList(); List> createFutures = itemRecordsToUpdateOwnership.stream() - .map(itemCollection::add) + .map(item -> + itemCollection.add(item) + .exceptionally(e -> { + notUpdatedEntities.add(new NotUpdatedEntity().withEntityId(item.getHoldingId()).withErrorMessage(e.getMessage())); + throw new CompletionException(e); + })) .toList(); return CompletableFuture.allOf(createFutures.toArray(new CompletableFuture[0])) @@ -179,7 +197,7 @@ private CompletableFuture> createItems(List jsons, ItemCo .toList()); } - private CompletableFuture> createHoldings(List jsons, String instanceId, + private CompletableFuture> createHoldings(List jsons, List notUpdatedEntities, String instanceId, HoldingsRecordCollection holdingsRecordCollection) { List holdingsRecordsToUpdateOwnership = jsons.stream() .peek(MoveApiUtil::removeExtraRedundantFields) @@ -188,7 +206,12 @@ private CompletableFuture> createHoldings(List .toList(); List> createFutures = holdingsRecordsToUpdateOwnership.stream() - .map(holdingsRecordCollection::add) + .map(holdingRecord -> + holdingsRecordCollection.add(holdingRecord) + .exceptionally(e -> { + notUpdatedEntities.add(new NotUpdatedEntity().withEntityId(holdingRecord.getId()).withErrorMessage(e.getMessage())); + throw new CompletionException(e); + })) .toList(); return CompletableFuture.allOf(createFutures.toArray(new CompletableFuture[0])) @@ -198,11 +221,35 @@ private CompletableFuture> createHoldings(List .toList()); } - private CompletableFuture> deleteCollectionItems(List collectionItemIds, AsynchronousCollection collection) { - List> deleteFutures = collectionItemIds.stream() - .map(collectionItemId -> { + private CompletableFuture> deleteHoldings(List holdingsRecords, List notUpdatedEntities, + HoldingsRecordCollection holdingsRecordCollection) { + List> deleteFutures = holdingsRecords.stream() + .map(holdingsRecord -> { + Promise promise = Promise.promise(); + holdingsRecordCollection.delete(holdingsRecord.getId(), success -> promise.complete(holdingsRecord.getId()), + failure -> { + notUpdatedEntities.add(new NotUpdatedEntity().withEntityId(holdingsRecord.getId()).withErrorMessage(failure.getReason())); + promise.fail(failure.getReason()); + }); + return promise.future().toCompletionStage().toCompletableFuture(); + }).toList(); + + return CompletableFuture.allOf(deleteFutures.toArray(new CompletableFuture[0])) + .handle((vVoid, throwable) -> deleteFutures.stream() + .filter(future -> !future.isCompletedExceptionally()) + .map(CompletableFuture::join) + .toList()); + } + + private CompletableFuture> deleteItems(List itemIds, List notUpdatedEntities, ItemCollection itemCollection) { + List> deleteFutures = itemIds.stream() + .map(item -> { Promise promise = Promise.promise(); - collection.delete(collectionItemId, success -> promise.complete(collectionItemId), failure -> promise.fail(failure.getReason())); + itemCollection.delete(item.getId(), success -> promise.complete(item.getId()), + failure -> { + notUpdatedEntities.add(new NotUpdatedEntity().withEntityId(item.getHoldingId()).withErrorMessage(failure.getReason())); + promise.fail(failure.getReason()); + }); return promise.future().toCompletionStage().toCompletableFuture(); }).toList(); @@ -212,4 +259,16 @@ private CompletableFuture> deleteCollectionItems(List colle .map(CompletableFuture::join) .toList()); } + + private void processNotFoundedInstances(List holdingsRecordIds, List notUpdatedEntities, WebContext context, List jsons) { + List foundedIds = jsons.stream().map(json -> json.getString("id")).toList(); + List notFoundedIds = ListUtils.subtract(holdingsRecordIds, foundedIds); + notFoundedIds.forEach(id -> + notUpdatedEntities.add(new NotUpdatedEntity().withEntityId(id).withErrorMessage(String.format(INSTANCE_NOT_FOUND, id, context.getTenantId())))); + } + + private List getHoldingsToDelete(List notUpdatedEntities, List createdHoldings) { + List notUpdatedHoldingsIds = notUpdatedEntities.stream().map(NotUpdatedEntity::getEntityId).toList(); + return createdHoldings.stream().filter(holdingsRecord -> !notUpdatedHoldingsIds.contains(holdingsRecord.getId())).toList(); + } } diff --git a/src/main/java/org/folio/inventory/support/MoveApiUtil.java b/src/main/java/org/folio/inventory/support/MoveApiUtil.java index f416567a8..af6051a21 100644 --- a/src/main/java/org/folio/inventory/support/MoveApiUtil.java +++ b/src/main/java/org/folio/inventory/support/MoveApiUtil.java @@ -6,6 +6,8 @@ import io.vertx.ext.web.RoutingContext; import io.vertx.ext.web.client.WebClient; import org.apache.commons.collections4.ListUtils; +import org.folio.NotUpdatedEntity; +import org.folio.UpdateOwnershipResponse; import org.folio.inventory.common.WebContext; import org.folio.inventory.storage.external.CollectionResourceClient; import org.folio.inventory.storage.external.CqlQuery; @@ -101,4 +103,8 @@ public static void respond(RoutingContext routingContext, List itemIdsTo } } + public static void respond(RoutingContext routingContext, List notUpdatedEntities) { + HttpServerResponse response = routingContext.response(); + success(response, JsonObject.mapFrom(new UpdateOwnershipResponse().withNotUpdatedEntities(notUpdatedEntities))); + } } diff --git a/src/test/java/api/holdings/HoldingsUpdateOwnershipApiTest.java b/src/test/java/api/holdings/HoldingsUpdateOwnershipApiTest.java index a9f5e0935..ab8f860c1 100644 --- a/src/test/java/api/holdings/HoldingsUpdateOwnershipApiTest.java +++ b/src/test/java/api/holdings/HoldingsUpdateOwnershipApiTest.java @@ -18,7 +18,6 @@ import org.junit.After; import org.junit.Assert; import org.junit.Before; -import org.junit.Ignore; import org.junit.Test; import org.junit.runner.RunWith; import support.fakes.EndpointFailureDescriptor; @@ -73,7 +72,7 @@ public void canUpdateHoldingsOwnershipToDifferentTenant() throws MalformedURLExc Response postHoldingsUpdateOwnershipResponse = updateHoldingsRecordsOwnership(holdingsRecordUpdateOwnershipRequestBody); assertThat(postHoldingsUpdateOwnershipResponse.getStatusCode(), is(200)); - assertThat(new JsonObject(postHoldingsUpdateOwnershipResponse.getBody()).getJsonArray("nonUpdatedIds").size(), is(0)); + assertThat(new JsonObject(postHoldingsUpdateOwnershipResponse.getBody()).getJsonArray("notUpdatedEntities").size(), is(0)); assertThat(postHoldingsUpdateOwnershipResponse.getContentType(), containsString(APPLICATION_JSON)); Response sourceTenantHoldingsRecord1 = holdingsStorageClient.getById(createHoldingsRecord1); @@ -82,8 +81,8 @@ public void canUpdateHoldingsOwnershipToDifferentTenant() throws MalformedURLExc Assert.assertEquals(HttpStatus.SC_NOT_FOUND, sourceTenantHoldingsRecord1.getStatusCode()); Assert.assertEquals(instanceId.toString(), targetTenantHoldingsRecord1.getJson().getString(INSTANCE_ID)); - Response sourceTenantHoldingsRecord2 = holdingsStorageClient.getById(createHoldingsRecord1); - Response targetTenantHoldingsRecord2 = collegeHoldingsStorageClient.getById(createHoldingsRecord1); + Response sourceTenantHoldingsRecord2 = holdingsStorageClient.getById(createHoldingsRecord2); + Response targetTenantHoldingsRecord2 = collegeHoldingsStorageClient.getById(createHoldingsRecord2); Assert.assertEquals(HttpStatus.SC_NOT_FOUND, sourceTenantHoldingsRecord2.getStatusCode()); Assert.assertEquals(instanceId.toString(), targetTenantHoldingsRecord2.getJson().getString(INSTANCE_ID)); @@ -118,7 +117,7 @@ public void canUpdateHoldingsOwnershipWithRelatedItemsToDifferentTenant() throws Response postHoldingsUpdateOwnershipResponse = updateHoldingsRecordsOwnership(holdingsRecordUpdateOwnershipRequestBody); assertThat(postHoldingsUpdateOwnershipResponse.getStatusCode(), is(200)); - assertThat(new JsonObject(postHoldingsUpdateOwnershipResponse.getBody()).getJsonArray("nonUpdatedIds").size(), is(0)); + assertThat(new JsonObject(postHoldingsUpdateOwnershipResponse.getBody()).getJsonArray("notUpdatedEntities").size(), is(0)); assertThat(postHoldingsUpdateOwnershipResponse.getContentType(), containsString(APPLICATION_JSON)); // Verify Holdings ownership updated @@ -128,8 +127,8 @@ public void canUpdateHoldingsOwnershipWithRelatedItemsToDifferentTenant() throws Assert.assertEquals(HttpStatus.SC_NOT_FOUND, sourceTenantHoldingsRecord1.getStatusCode()); Assert.assertEquals(instanceId.toString(), targetTenantHoldingsRecord1.getJson().getString(INSTANCE_ID)); - Response sourceTenantHoldingsRecord2 = holdingsStorageClient.getById(createHoldingsRecord1); - Response targetTenantHoldingsRecord2 = collegeHoldingsStorageClient.getById(createHoldingsRecord1); + Response sourceTenantHoldingsRecord2 = holdingsStorageClient.getById(createHoldingsRecord2); + Response targetTenantHoldingsRecord2 = collegeHoldingsStorageClient.getById(createHoldingsRecord2); Assert.assertEquals(HttpStatus.SC_NOT_FOUND, sourceTenantHoldingsRecord2.getStatusCode()); Assert.assertEquals(instanceId.toString(), targetTenantHoldingsRecord2.getJson().getString(INSTANCE_ID)); @@ -157,7 +156,6 @@ public void canUpdateHoldingsOwnershipIfErrorUpdatingRelatedItemsToDifferentTena InstanceApiClient.createInstance(consortiumOkapiClient, instance.put("source", FOLIO.getValue())); final UUID createHoldingsRecord1 = createHoldingForInstance(instanceId); - final UUID createHoldingsRecord2 = createHoldingForInstance(instanceId); final var firstItem = itemsClient.create( new ItemRequestBuilder() @@ -165,12 +163,6 @@ public void canUpdateHoldingsOwnershipIfErrorUpdatingRelatedItemsToDifferentTena .withBarcode("645398607547") .withStatus(ItemStatusName.AVAILABLE.value())); - final var secondItem = itemsClient.create( - new ItemRequestBuilder() - .forHolding(createHoldingsRecord2) - .withBarcode("645398607546") - .withStatus(ItemStatusName.AVAILABLE.value())); - final JsonObject expectedErrorResponse = new JsonObject().put("message", "Server error"); collegeItemsClient.emulateFailure( new EndpointFailureDescriptor() @@ -189,19 +181,21 @@ public void canUpdateHoldingsOwnershipIfErrorUpdatingRelatedItemsToDifferentTena .setMethod(HttpMethod.DELETE.name())); JsonObject holdingsRecordUpdateOwnershipRequestBody = new HoldingsRecordUpdateOwnershipRequestBuilder(instanceId, - new JsonArray(List.of(createHoldingsRecord1.toString(), createHoldingsRecord2.toString())), ApiTestSuite.COLLEGE_TENANT_ID).create(); + new JsonArray(List.of(createHoldingsRecord1.toString())), ApiTestSuite.COLLEGE_TENANT_ID).create(); Response postHoldingsUpdateOwnershipResponse = updateHoldingsRecordsOwnership(holdingsRecordUpdateOwnershipRequestBody); + collegeItemsClient.disableFailureEmulation(); + holdingsStorageClient.disableFailureEmulation(); + assertThat(postHoldingsUpdateOwnershipResponse.getStatusCode(), is(200)); - List nonUpdatedIdsIds = postHoldingsUpdateOwnershipResponse.getJson() - .getJsonArray("nonUpdatedIds") - .getList(); + JsonArray notUpdatedEntitiesIds = postHoldingsUpdateOwnershipResponse.getJson() + .getJsonArray("notUpdatedEntities"); - assertThat(nonUpdatedIdsIds.size(), is(2)); - assertThat(nonUpdatedIdsIds.get(0), equalTo(createHoldingsRecord1.toString())); - assertThat(nonUpdatedIdsIds.get(1), equalTo(createHoldingsRecord2.toString())); + assertThat(notUpdatedEntitiesIds.size(), is(1)); + assertThat(notUpdatedEntitiesIds.getJsonObject(0).getString("entityId"), equalTo(createHoldingsRecord1.toString())); + assertThat(notUpdatedEntitiesIds.getJsonObject(0).getString("errorMessage"), containsString(expectedErrorResponse.toString())); assertThat(postHoldingsUpdateOwnershipResponse.getContentType(), containsString(APPLICATION_JSON)); @@ -212,27 +206,70 @@ public void canUpdateHoldingsOwnershipIfErrorUpdatingRelatedItemsToDifferentTena Assert.assertEquals(instanceId.toString(), sourceTenantHoldingsRecord1.getJson().getString(INSTANCE_ID)); Assert.assertEquals(instanceId.toString(), targetTenantHoldingsRecord1.getJson().getString(INSTANCE_ID)); - Response sourceTenantHoldingsRecord2 = holdingsStorageClient.getById(createHoldingsRecord1); - Response targetTenantHoldingsRecord2 = collegeHoldingsStorageClient.getById(createHoldingsRecord1); - - Assert.assertEquals(instanceId.toString(), sourceTenantHoldingsRecord2.getJson().getString(INSTANCE_ID)); - Assert.assertEquals(instanceId.toString(), targetTenantHoldingsRecord2.getJson().getString(INSTANCE_ID)); - - // Verify related Items ownership updated + // Verify related Item ownership not updated Response sourceTenantItem1 = itemsClient.getById(firstItem.getId()); Response targetTenantItem1 = collegeItemsClient.getById(firstItem.getId()); assertThat(HttpStatus.SC_NOT_FOUND, is(targetTenantItem1.getStatusCode())); assertThat(sourceTenantItem1.getJson().getString(HOLDINGS_RECORD_ID), is(createHoldingsRecord1.toString())); + } - Response sourceTenantItem2 = itemsClient.getById(secondItem.getId()); - Response targetTenantItem2 = collegeItemsClient.getById(secondItem.getId()); + @Test + public void canUpdateHoldingsOwnershipIfErrorDeletingRelatedItemsToDifferentTenant() throws MalformedURLException, ExecutionException, InterruptedException, TimeoutException { + UUID instanceId = UUID.randomUUID(); + JsonObject instance = smallAngryPlanet(instanceId); + + InstanceApiClient.createInstance(okapiClient, instance.put("source", CONSORTIUM_FOLIO.getValue())); + InstanceApiClient.createInstance(consortiumOkapiClient, instance.put("source", FOLIO.getValue())); + + final UUID createHoldingsRecord1 = createHoldingForInstance(instanceId); + + final var firstItem = itemsClient.create( + new ItemRequestBuilder() + .forHolding(createHoldingsRecord1) + .withBarcode("645398607547") + .withStatus(ItemStatusName.AVAILABLE.value())); + + final JsonObject expectedErrorResponse = new JsonObject().put("message", "Server error"); + collegeItemsClient.emulateFailure( + new EndpointFailureDescriptor() + .setFailureExpireDate(DateTime.now().plusSeconds(2).toDate()) + .setStatusCode(500) + .setContentType("application/json") + .setBody(expectedErrorResponse.toString()) + .setMethod(HttpMethod.DELETE.name())); + + JsonObject holdingsRecordUpdateOwnershipRequestBody = new HoldingsRecordUpdateOwnershipRequestBuilder(instanceId, + new JsonArray(List.of(createHoldingsRecord1.toString())), ApiTestSuite.COLLEGE_TENANT_ID).create(); - assertThat(HttpStatus.SC_NOT_FOUND, is(targetTenantItem2.getStatusCode())); - assertThat(sourceTenantItem2.getJson().getString(HOLDINGS_RECORD_ID), is(createHoldingsRecord2.toString())); + Response postHoldingsUpdateOwnershipResponse = updateHoldingsRecordsOwnership(holdingsRecordUpdateOwnershipRequestBody); collegeItemsClient.disableFailureEmulation(); - holdingsStorageClient.disableFailureEmulation(); + + assertThat(postHoldingsUpdateOwnershipResponse.getStatusCode(), is(200)); + + JsonArray notUpdatedEntitiesIds = postHoldingsUpdateOwnershipResponse.getJson() + .getJsonArray("notUpdatedEntities"); + + assertThat(notUpdatedEntitiesIds.size(), is(1)); + assertThat(notUpdatedEntitiesIds.getJsonObject(0).getString("entityId"), equalTo(createHoldingsRecord1.toString())); + assertThat(notUpdatedEntitiesIds.getJsonObject(0).getString("errorMessage"), containsString(expectedErrorResponse.toString())); + + assertThat(postHoldingsUpdateOwnershipResponse.getContentType(), containsString(APPLICATION_JSON)); + + // Verify Holdings ownership updated + Response sourceTenantHoldingsRecord1 = holdingsStorageClient.getById(createHoldingsRecord1); + Response targetTenantHoldingsRecord1 = collegeHoldingsStorageClient.getById(createHoldingsRecord1); + + Assert.assertEquals(instanceId.toString(), sourceTenantHoldingsRecord1.getJson().getString(INSTANCE_ID)); + Assert.assertEquals(instanceId.toString(), targetTenantHoldingsRecord1.getJson().getString(INSTANCE_ID)); + + // Verify related Item ownership not updated + Response sourceTenantItem1 = itemsClient.getById(firstItem.getId()); + Response targetTenantItem1 = collegeItemsClient.getById(firstItem.getId()); + + assertThat(targetTenantItem1.getJson().getString(HOLDINGS_RECORD_ID), is(createHoldingsRecord1.toString())); + assertThat(sourceTenantItem1.getJson().getString(HOLDINGS_RECORD_ID), is(createHoldingsRecord1.toString())); } @Test @@ -256,12 +293,12 @@ public void shouldReportErrorsWhenOnlySomeRequestedHoldingsRecordsCouldNotBeUpda assertThat(postHoldingsUpdateOwnershipResponse.getStatusCode(), is(200)); assertThat(postHoldingsUpdateOwnershipResponse.getContentType(), containsString(APPLICATION_JSON)); - List notFoundIds = postHoldingsUpdateOwnershipResponse.getJson() - .getJsonArray("nonUpdatedIds") - .getList(); + JsonArray notFoundIds = postHoldingsUpdateOwnershipResponse.getJson() + .getJsonArray("notUpdatedEntities"); assertThat(notFoundIds.size(), is(1)); - assertThat(notFoundIds.get(0), equalTo(createHoldingsRecord2.toString())); + assertThat(notFoundIds.getJsonObject(0).getString("entityId"), equalTo(createHoldingsRecord2.toString())); + assertThat(notFoundIds.getJsonObject(0).getString("errorMessage"), containsString("not found on tenant")); Response sourceTenantHoldingsRecord1 = holdingsStorageClient.getById(createHoldingsRecord1); Response targetTenantHoldingsRecord1 = collegeHoldingsStorageClient.getById(createHoldingsRecord1); @@ -407,7 +444,6 @@ public void cannotUpdateHoldingsRecordOwnershipDueToHoldingsRecordCreateError() InstanceApiClient.createInstance(consortiumOkapiClient, instance.put("source", FOLIO.getValue())); final UUID createHoldingsRecord1 = createHoldingForInstance(instanceId); - final UUID createHoldingsRecord2 = createHoldingForInstance(instanceId); final JsonObject expectedErrorResponse = new JsonObject().put("message", "Server error"); collegeHoldingsStorageClient.emulateFailure( @@ -419,17 +455,18 @@ public void cannotUpdateHoldingsRecordOwnershipDueToHoldingsRecordCreateError() .setMethod(HttpMethod.POST.name())); JsonObject holdingsRecordUpdateOwnershipRequestBody = new HoldingsRecordUpdateOwnershipRequestBuilder(instanceId, - new JsonArray(List.of(createHoldingsRecord1.toString(), createHoldingsRecord2.toString())), ApiTestSuite.COLLEGE_TENANT_ID).create(); + new JsonArray(List.of(createHoldingsRecord1.toString())), ApiTestSuite.COLLEGE_TENANT_ID).create(); Response postHoldingsUpdateOwnershipResponse = updateHoldingsRecordsOwnership(holdingsRecordUpdateOwnershipRequestBody); - List nonUpdatedIdsIds = postHoldingsUpdateOwnershipResponse.getJson() - .getJsonArray("nonUpdatedIds") - .getList(); + collegeHoldingsStorageClient.disableFailureEmulation(); + + JsonArray notUpdatedEntitiesIds = postHoldingsUpdateOwnershipResponse.getJson() + .getJsonArray("notUpdatedEntities"); - assertThat(nonUpdatedIdsIds.size(), is(2)); - assertThat(nonUpdatedIdsIds.get(0), equalTo(createHoldingsRecord1.toString())); - assertThat(nonUpdatedIdsIds.get(1), equalTo(createHoldingsRecord2.toString())); + assertThat(notUpdatedEntitiesIds.size(), is(1)); + assertThat(notUpdatedEntitiesIds.getJsonObject(0).getString("entityId"), equalTo(createHoldingsRecord1.toString())); + assertThat(notUpdatedEntitiesIds.getJsonObject(0).getString("errorMessage"), containsString(expectedErrorResponse.toString())); assertThat(postHoldingsUpdateOwnershipResponse.getStatusCode(), is(200)); assertThat(postHoldingsUpdateOwnershipResponse.getContentType(), containsString(APPLICATION_JSON)); @@ -439,14 +476,6 @@ public void cannotUpdateHoldingsRecordOwnershipDueToHoldingsRecordCreateError() Assert.assertEquals(instanceId.toString(), sourceTenantHoldingsRecord1.getJson().getString(INSTANCE_ID)); Assert.assertEquals(HttpStatus.SC_NOT_FOUND, targetTenantHoldingsRecord1.getStatusCode()); - - Response sourceTenantHoldingsRecord2 = holdingsStorageClient.getById(createHoldingsRecord2); - Response targetTenantHoldingsRecord2 = collegeHoldingsStorageClient.getById(createHoldingsRecord2); - - Assert.assertEquals(instanceId.toString(), sourceTenantHoldingsRecord2.getJson().getString(INSTANCE_ID)); - Assert.assertEquals(HttpStatus.SC_NOT_FOUND, targetTenantHoldingsRecord2.getStatusCode()); - - collegeHoldingsStorageClient.disableFailureEmulation(); } @Test @@ -458,7 +487,6 @@ public void cannotUpdateHoldingsRecordOwnershipDueToHoldingsRecordDeleteError() InstanceApiClient.createInstance(consortiumOkapiClient, instance.put("source", FOLIO.getValue())); final UUID createHoldingsRecord1 = createHoldingForInstance(instanceId); - final UUID createHoldingsRecord2 = createHoldingForInstance(instanceId); final JsonObject expectedErrorResponse = new JsonObject().put("message", "Server error"); collegeHoldingsStorageClient.emulateFailure( @@ -470,17 +498,18 @@ public void cannotUpdateHoldingsRecordOwnershipDueToHoldingsRecordDeleteError() .setMethod(HttpMethod.DELETE.name())); JsonObject holdingsRecordUpdateOwnershipRequestBody = new HoldingsRecordUpdateOwnershipRequestBuilder(instanceId, - new JsonArray(List.of(createHoldingsRecord1.toString(), createHoldingsRecord2.toString())), ApiTestSuite.COLLEGE_TENANT_ID).create(); + new JsonArray(List.of(createHoldingsRecord1.toString())), ApiTestSuite.COLLEGE_TENANT_ID).create(); Response postHoldingsUpdateOwnershipResponse = updateHoldingsRecordsOwnership(holdingsRecordUpdateOwnershipRequestBody); - List nonUpdatedIdsIds = postHoldingsUpdateOwnershipResponse.getJson() - .getJsonArray("nonUpdatedIds") - .getList(); + collegeHoldingsStorageClient.disableFailureEmulation(); + + JsonArray notUpdatedEntitiesIds = postHoldingsUpdateOwnershipResponse.getJson() + .getJsonArray("notUpdatedEntities"); - assertThat(nonUpdatedIdsIds.size(), is(2)); - assertThat(nonUpdatedIdsIds.get(0), equalTo(createHoldingsRecord1.toString())); - assertThat(nonUpdatedIdsIds.get(1), equalTo(createHoldingsRecord2.toString())); + assertThat(notUpdatedEntitiesIds.size(), is(1)); + assertThat(notUpdatedEntitiesIds.getJsonObject(0).getString("entityId"), equalTo(createHoldingsRecord1.toString())); + assertThat(notUpdatedEntitiesIds.getJsonObject(0).getString("errorMessage"), containsString(expectedErrorResponse.toString())); assertThat(postHoldingsUpdateOwnershipResponse.getStatusCode(), is(200)); assertThat(postHoldingsUpdateOwnershipResponse.getContentType(), containsString(APPLICATION_JSON)); @@ -490,14 +519,6 @@ public void cannotUpdateHoldingsRecordOwnershipDueToHoldingsRecordDeleteError() Assert.assertEquals(instanceId.toString(), sourceTenantHoldingsRecord1.getJson().getString(INSTANCE_ID)); Assert.assertEquals(instanceId.toString(), targetTenantHoldingsRecord1.getJson().getString(INSTANCE_ID)); - - Response sourceTenantHoldingsRecord2 = holdingsStorageClient.getById(createHoldingsRecord2); - Response targetTenantHoldingsRecord2 = collegeHoldingsStorageClient.getById(createHoldingsRecord2); - - Assert.assertEquals(instanceId.toString(), sourceTenantHoldingsRecord2.getJson().getString(INSTANCE_ID)); - Assert.assertEquals(instanceId.toString(), targetTenantHoldingsRecord2.getJson().getString(INSTANCE_ID)); - - collegeHoldingsStorageClient.disableFailureEmulation(); } @Test @@ -526,7 +547,7 @@ public void canUpdateHoldingsRecordOwnershipToDifferentInstanceWithExtraRedundan Response postHoldingsUpdateOwnershipResponse = updateHoldingsRecordsOwnership(holdingsRecordUpdateOwnershipRequestBody); assertThat(postHoldingsUpdateOwnershipResponse.getStatusCode(), is(200)); - assertThat(new JsonObject(postHoldingsUpdateOwnershipResponse.getBody()).getJsonArray("nonUpdatedIds").size(), is(0)); + assertThat(new JsonObject(postHoldingsUpdateOwnershipResponse.getBody()).getJsonArray("notUpdatedEntities").size(), is(0)); assertThat(postHoldingsUpdateOwnershipResponse.getContentType(), containsString(APPLICATION_JSON)); Response sourceTenantHoldingsRecord1 = holdingsStorageClient.getById(createHoldingsRecord1); @@ -535,8 +556,8 @@ public void canUpdateHoldingsRecordOwnershipToDifferentInstanceWithExtraRedundan Assert.assertEquals(HttpStatus.SC_NOT_FOUND, sourceTenantHoldingsRecord1.getStatusCode()); Assert.assertEquals(instanceId.toString(), targetTenantHoldingsRecord1.getJson().getString(INSTANCE_ID)); - Response sourceTenantHoldingsRecord2 = holdingsStorageClient.getById(createHoldingsRecord1); - Response targetTenantHoldingsRecord2 = collegeHoldingsStorageClient.getById(createHoldingsRecord1); + Response sourceTenantHoldingsRecord2 = holdingsStorageClient.getById(createHoldingsRecord2); + Response targetTenantHoldingsRecord2 = collegeHoldingsStorageClient.getById(createHoldingsRecord2); Assert.assertEquals(HttpStatus.SC_NOT_FOUND, sourceTenantHoldingsRecord2.getStatusCode()); Assert.assertEquals(instanceId.toString(), targetTenantHoldingsRecord2.getJson().getString(INSTANCE_ID)); From 88071bcb4e016682aa91f4c3bf5f8ed23ba6c284 Mon Sep 17 00:00:00 2001 From: Roman_Chernetskyi Date: Tue, 18 Jun 2024 13:34:36 +0300 Subject: [PATCH 05/11] Add description for entityId --- ramls/update_ownership_response.json | 7 ++++--- 1 file changed, 4 insertions(+), 3 deletions(-) diff --git a/ramls/update_ownership_response.json b/ramls/update_ownership_response.json index ab603298f..9ef38d0ef 100644 --- a/ramls/update_ownership_response.json +++ b/ramls/update_ownership_response.json @@ -1,16 +1,17 @@ { "$schema": "http://json-schema.org/draft-04/schema#", - "description": "Holder for errors during holdings/item update ownership", + "description": "Holder for errors during item/holdingsRecord update ownership", "type": "object", "properties": { "notUpdatedEntities": { - "description": "Items/holdings errors", + "description": "Item/HoldingsRecord errors", "type": "array", "items": { "type": "object", "properties": { "entityId": { - "$ref": "uuid.json" + "$ref": "uuid.json", + "description": "Id of item/holdingsRecord" }, "errorMessage": { "type": "string", From 8aa13c5b063821bed5a812aef69b10da73775349 Mon Sep 17 00:00:00 2001 From: Roman_Chernetskyi Date: Tue, 18 Jun 2024 14:59:04 +0300 Subject: [PATCH 06/11] Add logs on failures --- .../inventory/resources/UpdateOwnershipApi.java | 15 +++++++++++++-- 1 file changed, 13 insertions(+), 2 deletions(-) diff --git a/src/main/java/org/folio/inventory/resources/UpdateOwnershipApi.java b/src/main/java/org/folio/inventory/resources/UpdateOwnershipApi.java index 0bca384a7..0c7286f3a 100644 --- a/src/main/java/org/folio/inventory/resources/UpdateOwnershipApi.java +++ b/src/main/java/org/folio/inventory/resources/UpdateOwnershipApi.java @@ -185,6 +185,7 @@ private CompletableFuture> createItems(List jsons, List itemCollection.add(item) .exceptionally(e -> { + LOGGER.warn("createHoldings:: Error during creating item with id: {} for holdingsRecord with id: {}", item.getId(), item.getHoldingId(), e); notUpdatedEntities.add(new NotUpdatedEntity().withEntityId(item.getHoldingId()).withErrorMessage(e.getMessage())); throw new CompletionException(e); })) @@ -209,6 +210,7 @@ private CompletableFuture> createHoldings(List .map(holdingRecord -> holdingsRecordCollection.add(holdingRecord) .exceptionally(e -> { + LOGGER.warn("createHoldings:: Error during creating holdingsRecord with id: {}", holdingRecord.getId(), e); notUpdatedEntities.add(new NotUpdatedEntity().withEntityId(holdingRecord.getId()).withErrorMessage(e.getMessage())); throw new CompletionException(e); })) @@ -228,6 +230,9 @@ private CompletableFuture> deleteHoldings(List hold Promise promise = Promise.promise(); holdingsRecordCollection.delete(holdingsRecord.getId(), success -> promise.complete(holdingsRecord.getId()), failure -> { + LOGGER.warn("deleteHoldings:: Error during deleting holdingsRecord with id: {}, status code: {}, reason: {}", + holdingsRecord.getId(), failure.getStatusCode(), failure.getReason()); + notUpdatedEntities.add(new NotUpdatedEntity().withEntityId(holdingsRecord.getId()).withErrorMessage(failure.getReason())); promise.fail(failure.getReason()); }); @@ -247,6 +252,9 @@ private CompletableFuture> deleteItems(List itemIds, List promise = Promise.promise(); itemCollection.delete(item.getId(), success -> promise.complete(item.getId()), failure -> { + LOGGER.warn("deleteItems:: Error during deleting item with id: {} for holdingsRecord with id {}, status code: {}, reason: {}", + item.getId(), item.getHoldingId(), failure.getStatusCode(), failure.getReason()); + notUpdatedEntities.add(new NotUpdatedEntity().withEntityId(item.getHoldingId()).withErrorMessage(failure.getReason())); promise.fail(failure.getReason()); }); @@ -263,8 +271,11 @@ private CompletableFuture> deleteItems(List itemIds, List holdingsRecordIds, List notUpdatedEntities, WebContext context, List jsons) { List foundedIds = jsons.stream().map(json -> json.getString("id")).toList(); List notFoundedIds = ListUtils.subtract(holdingsRecordIds, foundedIds); - notFoundedIds.forEach(id -> - notUpdatedEntities.add(new NotUpdatedEntity().withEntityId(id).withErrorMessage(String.format(INSTANCE_NOT_FOUND, id, context.getTenantId())))); + notFoundedIds.forEach(id -> { + String errorMessage = String.format(INSTANCE_NOT_FOUND, id, context.getTenantId()); + LOGGER.warn("processNotFoundedInstances:: " + errorMessage); + notUpdatedEntities.add(new NotUpdatedEntity().withEntityId(id).withErrorMessage(errorMessage)); + }); } private List getHoldingsToDelete(List notUpdatedEntities, List createdHoldings) { From 1c8b33ce6659128015aa53c69395a551aa546551 Mon Sep 17 00:00:00 2001 From: Roman_Chernetskyi Date: Tue, 18 Jun 2024 15:08:04 +0300 Subject: [PATCH 07/11] Add logs for debug --- .../resources/UpdateOwnershipApi.java | 18 ++++++++++++++++-- 1 file changed, 16 insertions(+), 2 deletions(-) diff --git a/src/main/java/org/folio/inventory/resources/UpdateOwnershipApi.java b/src/main/java/org/folio/inventory/resources/UpdateOwnershipApi.java index 0c7286f3a..aa305dd59 100644 --- a/src/main/java/org/folio/inventory/resources/UpdateOwnershipApi.java +++ b/src/main/java/org/folio/inventory/resources/UpdateOwnershipApi.java @@ -131,6 +131,9 @@ private CompletableFuture> updateOwnershipOfHoldingsRecords(Holding List notUpdatedEntities, RoutingContext routingContext, WebContext context, Context targetTenantContext) { try { + LOGGER.info("updateOwnershipOfHoldingsRecords:: Updating ownership of holdingsRecord: {}, to tenant: {}", + holdingsUpdateOwnership.getHoldingsRecordIds(), targetTenantContext.getTenantId()); + CollectionResourceClient holdingsStorageClient = createHoldingsStorageClient(createHttpClient(client, routingContext, context), context); MultipleRecordsFetchClient holdingsRecordFetchClient = createHoldingsRecordsFetchClient(holdingsStorageClient); @@ -140,10 +143,12 @@ private CompletableFuture> updateOwnershipOfHoldingsRecords(Holding return holdingsRecordFetchClient.find(holdingsUpdateOwnership.getHoldingsRecordIds(), MoveApiUtil::fetchByIdCql) .thenCompose(jsons -> { + LOGGER.info("updateOwnershipOfHoldingsRecords:: Founded holdings to update ownership: {}", jsons); processNotFoundedInstances(holdingsUpdateOwnership.getHoldingsRecordIds(), notUpdatedEntities, context, jsons); return createHoldings(jsons, notUpdatedEntities, holdingsUpdateOwnership.getToInstanceId(), targetTenantHoldingsRecordCollection); }) .thenCompose(createdHoldings -> { + LOGGER.info("updateOwnershipOfHoldingsRecords:: Created holdings: {}, for tenant: {}", createdHoldings, targetTenantContext.getTenantId()); List createdHoldingsIds = createdHoldings.stream().map(HoldingsRecord::getId).toList(); return transferAttachedItems(createdHoldingsIds, notUpdatedEntities, routingContext, context, targetTenantContext) @@ -160,6 +165,9 @@ private CompletableFuture> updateOwnershipOfHoldingsRecords(Holding private CompletableFuture> transferAttachedItems(List holdingsRecordIds, List notUpdatedEntities, RoutingContext routingContext, WebContext context, Context targetTenantContext) { try { + LOGGER.info("transferAttachedItems:: Transfer items of holdingsRecordIds: {}, to tenant: {}", + holdingsRecordIds, targetTenantContext.getTenantId()); + CollectionResourceClient itemsStorageClient = createItemStorageClient(createHttpClient(client, routingContext, context), context); MultipleRecordsFetchClient itemsFetchClient = createItemsFetchClient(itemsStorageClient); @@ -167,8 +175,14 @@ private CompletableFuture> transferAttachedItems(List holdi ItemCollection targetTenantItemCollection = storage.getItemCollection(targetTenantContext); return itemsFetchClient.find(holdingsRecordIds, MoveApiUtil::fetchByHoldingsRecordIdCql) - .thenCompose(jsons -> createItems(jsons, notUpdatedEntities, targetTenantItemCollection)) - .thenCompose(items -> deleteItems(items, notUpdatedEntities, sourceTenantItemCollection)); + .thenCompose(jsons -> { + LOGGER.info("transferAttachedItems:: Found items to transfer: {}", jsons); + return createItems(jsons, notUpdatedEntities, targetTenantItemCollection); + }) + .thenCompose(items -> { + LOGGER.info("transferAttachedItems:: Created items: {}", items); + return deleteItems(items, notUpdatedEntities, sourceTenantItemCollection); + }); } catch (Exception e) { LOGGER.warn("transferAttachedItems:: Error during transfer attached items for holdings {}, to tenant: {}", holdingsRecordIds, targetTenantContext.getTenantId(), e); From 9de14aa708f69153bdb09dd076df800d453d3864 Mon Sep 17 00:00:00 2001 From: Roman_Chernetskyi Date: Wed, 19 Jun 2024 16:22:27 +0300 Subject: [PATCH 08/11] Not set hrid for items and holdings during create --- .../resources/UpdateOwnershipApi.java | 41 ++++++++-------- .../HoldingsUpdateOwnershipApiTest.java | 9 +++- .../builders/HoldingRequestBuilder.java | 48 +++++++++++++++---- 3 files changed, 67 insertions(+), 31 deletions(-) diff --git a/src/main/java/org/folio/inventory/resources/UpdateOwnershipApi.java b/src/main/java/org/folio/inventory/resources/UpdateOwnershipApi.java index aa305dd59..e425dc8f6 100644 --- a/src/main/java/org/folio/inventory/resources/UpdateOwnershipApi.java +++ b/src/main/java/org/folio/inventory/resources/UpdateOwnershipApi.java @@ -49,7 +49,7 @@ public class UpdateOwnershipApi extends AbstractInventoryResource { public static final String INSTANCE_NOT_SHARED = "Instance with id: %s is not shared"; public static final String INSTANCE_NOT_FOUND_AT_SOURCE_TENANT = "Instance with id: %s not found at source tenant, tenant: %s"; public static final String TENANT_NOT_IN_CONSORTIA = "%s tenant is not in consortia"; - public static final String INSTANCE_NOT_FOUND = "Instance with id: %s not found on tenant: %s"; + public static final String HOLDINGS_NOT_FOUND = "HoldingsRecord with id: %s not found on tenant: %s"; private final ConsortiumService consortiumService; @@ -110,7 +110,7 @@ private void processUpdateHoldingsOwnership(RoutingContext routingContext) { LOGGER.warn("updateHoldingsOwnership:: " + notInConsortiaErrorMessage, context); return CompletableFuture.failedFuture(new BadRequestException(notInConsortiaErrorMessage)); }) - .thenAccept(updateHoldingsRecords -> respond(routingContext, notUpdatedEntities)) + .thenAccept(v -> respond(routingContext, notUpdatedEntities)) .exceptionally(throwable -> { LOGGER.warn("updateHoldingsOwnership:: Error during update ownership of holdings {}, to tenant: {}", holdingsUpdateOwnership.getHoldingsRecordIds(), holdingsUpdateOwnership.getTargetTenantId(), throwable); @@ -145,15 +145,18 @@ private CompletableFuture> updateOwnershipOfHoldingsRecords(Holding .thenCompose(jsons -> { LOGGER.info("updateOwnershipOfHoldingsRecords:: Founded holdings to update ownership: {}", jsons); processNotFoundedInstances(holdingsUpdateOwnership.getHoldingsRecordIds(), notUpdatedEntities, context, jsons); - return createHoldings(jsons, notUpdatedEntities, holdingsUpdateOwnership.getToInstanceId(), targetTenantHoldingsRecordCollection); - }) - .thenCompose(createdHoldings -> { - LOGGER.info("updateOwnershipOfHoldingsRecords:: Created holdings: {}, for tenant: {}", createdHoldings, targetTenantContext.getTenantId()); - List createdHoldingsIds = createdHoldings.stream().map(HoldingsRecord::getId).toList(); - - return transferAttachedItems(createdHoldingsIds, notUpdatedEntities, routingContext, context, targetTenantContext) - .thenCompose(itemIds -> - deleteHoldings(getHoldingsToDelete(notUpdatedEntities, createdHoldings), notUpdatedEntities, sourceTenantHoldingsRecordCollection)); + if (!jsons.isEmpty()) { + return createHoldings(jsons, notUpdatedEntities, holdingsUpdateOwnership.getToInstanceId(), targetTenantHoldingsRecordCollection) + .thenCompose(createdHoldings -> { + LOGGER.info("updateOwnershipOfHoldingsRecords:: Created holdings: {}, for tenant: {}", createdHoldings, targetTenantContext.getTenantId()); + List createdHoldingsIds = createdHoldings.stream().map(HoldingsRecord::getId).toList(); + + return transferAttachedItems(createdHoldingsIds, notUpdatedEntities, routingContext, context, targetTenantContext) + .thenCompose(itemIds -> + deleteHoldings(getHoldingsToDelete(notUpdatedEntities, createdHoldings), notUpdatedEntities, sourceTenantHoldingsRecordCollection)); + }); + } + return CompletableFuture.completedFuture(new ArrayList<>()); }); } catch (Exception e) { LOGGER.warn("updateOwnershipOfHoldingsRecords:: Error during update ownership of holdings {}, to tenant: {}", @@ -177,11 +180,11 @@ private CompletableFuture> transferAttachedItems(List holdi return itemsFetchClient.find(holdingsRecordIds, MoveApiUtil::fetchByHoldingsRecordIdCql) .thenCompose(jsons -> { LOGGER.info("transferAttachedItems:: Found items to transfer: {}", jsons); - return createItems(jsons, notUpdatedEntities, targetTenantItemCollection); - }) - .thenCompose(items -> { - LOGGER.info("transferAttachedItems:: Created items: {}", items); - return deleteItems(items, notUpdatedEntities, sourceTenantItemCollection); + if (!jsons.isEmpty()) { + return createItems(jsons, notUpdatedEntities, targetTenantItemCollection) + .thenCompose(items -> deleteItems(items, notUpdatedEntities, sourceTenantItemCollection)); + } + return CompletableFuture.completedFuture(new ArrayList<>()); }); } catch (Exception e) { LOGGER.warn("transferAttachedItems:: Error during transfer attached items for holdings {}, to tenant: {}", @@ -192,7 +195,7 @@ private CompletableFuture> transferAttachedItems(List holdi private CompletableFuture> createItems(List jsons, List notUpdatedEntities, ItemCollection itemCollection) { List itemRecordsToUpdateOwnership = jsons.stream() - .map(ItemUtil::fromStoredItemRepresentation) + .map(json -> ItemUtil.fromStoredItemRepresentation(json).withHrid(null)) .toList(); List> createFutures = itemRecordsToUpdateOwnership.stream() @@ -216,7 +219,7 @@ private CompletableFuture> createHoldings(List HoldingsRecordCollection holdingsRecordCollection) { List holdingsRecordsToUpdateOwnership = jsons.stream() .peek(MoveApiUtil::removeExtraRedundantFields) - .map(json -> json.mapTo(HoldingsRecord.class)) + .map(json -> json.mapTo(HoldingsRecord.class).withHrid(null)) .filter(holdingsRecord -> holdingsRecord.getInstanceId().equals(instanceId)) .toList(); @@ -286,7 +289,7 @@ private void processNotFoundedInstances(List holdingsRecordIds, List foundedIds = jsons.stream().map(json -> json.getString("id")).toList(); List notFoundedIds = ListUtils.subtract(holdingsRecordIds, foundedIds); notFoundedIds.forEach(id -> { - String errorMessage = String.format(INSTANCE_NOT_FOUND, id, context.getTenantId()); + String errorMessage = String.format(HOLDINGS_NOT_FOUND, id, context.getTenantId()); LOGGER.warn("processNotFoundedInstances:: " + errorMessage); notUpdatedEntities.add(new NotUpdatedEntity().withEntityId(id).withErrorMessage(errorMessage)); }); diff --git a/src/test/java/api/holdings/HoldingsUpdateOwnershipApiTest.java b/src/test/java/api/holdings/HoldingsUpdateOwnershipApiTest.java index ab8f860c1..85c48a826 100644 --- a/src/test/java/api/holdings/HoldingsUpdateOwnershipApiTest.java +++ b/src/test/java/api/holdings/HoldingsUpdateOwnershipApiTest.java @@ -86,12 +86,14 @@ public void canUpdateHoldingsOwnershipToDifferentTenant() throws MalformedURLExc Assert.assertEquals(HttpStatus.SC_NOT_FOUND, sourceTenantHoldingsRecord2.getStatusCode()); Assert.assertEquals(instanceId.toString(), targetTenantHoldingsRecord2.getJson().getString(INSTANCE_ID)); + Assert.assertNull(targetTenantHoldingsRecord2.getJson().getString("hrid")); } @Test public void canUpdateHoldingsOwnershipWithRelatedItemsToDifferentTenant() throws MalformedURLException, ExecutionException, InterruptedException, TimeoutException { UUID instanceId = UUID.randomUUID(); JsonObject instance = smallAngryPlanet(instanceId); + String itemHrId = "it0000001"; InstanceApiClient.createInstance(okapiClient, instance.put("source", CONSORTIUM_FOLIO.getValue())); InstanceApiClient.createInstance(consortiumOkapiClient, instance.put("source", FOLIO.getValue())); @@ -108,6 +110,7 @@ public void canUpdateHoldingsOwnershipWithRelatedItemsToDifferentTenant() throws final var secondItem = itemsClient.create( new ItemRequestBuilder() .forHolding(createHoldingsRecord2) + .withHrid(itemHrId) .withBarcode("645398607546") .withStatus(ItemStatusName.AVAILABLE.value())); @@ -145,6 +148,7 @@ public void canUpdateHoldingsOwnershipWithRelatedItemsToDifferentTenant() throws assertThat(HttpStatus.SC_NOT_FOUND, is(sourceTenantItem2.getStatusCode())); assertThat(targetTenantItem2.getJson().getString(HOLDINGS_RECORD_ID), is(createHoldingsRecord2.toString())); + Assert.assertNotEquals(targetTenantItem2.getJson().getString("hrid"), itemHrId); } @Test @@ -298,7 +302,8 @@ public void shouldReportErrorsWhenOnlySomeRequestedHoldingsRecordsCouldNotBeUpda assertThat(notFoundIds.size(), is(1)); assertThat(notFoundIds.getJsonObject(0).getString("entityId"), equalTo(createHoldingsRecord2.toString())); - assertThat(notFoundIds.getJsonObject(0).getString("errorMessage"), containsString("not found on tenant")); + assertThat(notFoundIds.getJsonObject(0).getString("errorMessage"), + equalTo(String.format("HoldingsRecord with id: %s not found on tenant: %s", createHoldingsRecord2, ApiTestSuite.TENANT_ID))); Response sourceTenantHoldingsRecord1 = holdingsStorageClient.getById(createHoldingsRecord1); Response targetTenantHoldingsRecord1 = collegeHoldingsStorageClient.getById(createHoldingsRecord1); @@ -570,7 +575,7 @@ private Response updateHoldingsRecordsOwnership(JsonObject holdingsRecordUpdateO } private UUID createHoldingForInstance(UUID instanceId) { - return holdingsStorageClient.create(new HoldingRequestBuilder().forInstance(instanceId)) + return holdingsStorageClient.create(new HoldingRequestBuilder().withHrId("hol0000001").forInstance(instanceId)) .getId(); } } diff --git a/src/test/java/api/support/builders/HoldingRequestBuilder.java b/src/test/java/api/support/builders/HoldingRequestBuilder.java index cff460332..f8a587bae 100644 --- a/src/test/java/api/support/builders/HoldingRequestBuilder.java +++ b/src/test/java/api/support/builders/HoldingRequestBuilder.java @@ -20,6 +20,7 @@ public class HoldingRequestBuilder extends AbstractBuilder { private final String callNumberTypeId; private final UUID sourceId; private final List administrativeNotes; + private final String hrId; public HoldingRequestBuilder() { this( @@ -31,6 +32,7 @@ public HoldingRequestBuilder() { null, null, FOLIO_SOURCE_HOLDINGS_ID, + null, null); } @@ -43,7 +45,8 @@ private HoldingRequestBuilder( String callNumberPrefix, String callNumberTypeId, UUID sourceId, - List administrativeNotes) { + List administrativeNotes, + String hrId) { this.instanceId = instanceId; this.permanentLocationId = permanentLocationId; @@ -54,6 +57,7 @@ private HoldingRequestBuilder( this.callNumberTypeId = callNumberTypeId; this.sourceId = sourceId; this.administrativeNotes = administrativeNotes; + this.hrId = hrId; } @Override @@ -72,6 +76,7 @@ public JsonObject create() { holding.put("callNumberSuffix", callNumberSuffix); holding.put("callNumberTypeId", callNumberTypeId); holding.put("sourceId", sourceId); + holding.put("hrid", hrId); return holding; } @@ -85,7 +90,8 @@ private HoldingRequestBuilder withPermanentLocation(UUID permanentLocationId) { this.callNumberPrefix, this.callNumberTypeId, this.sourceId, - this.administrativeNotes); + this.administrativeNotes, + this.hrId); } private HoldingRequestBuilder withTemporaryLocation(UUID temporaryLocationId) { @@ -98,7 +104,8 @@ private HoldingRequestBuilder withTemporaryLocation(UUID temporaryLocationId) { this.callNumberPrefix, this.callNumberTypeId, this.sourceId, - this.administrativeNotes); + this.administrativeNotes, + this.hrId); } public JsonObject createFolioHoldingsSource() { @@ -135,7 +142,8 @@ public HoldingRequestBuilder forInstance(UUID instanceId) { this.callNumberPrefix, this.callNumberTypeId, this.sourceId, - this.administrativeNotes); + this.administrativeNotes, + this.hrId); } public HoldingRequestBuilder withCallNumber(String callNumber) { @@ -148,7 +156,8 @@ public HoldingRequestBuilder withCallNumber(String callNumber) { this.callNumberPrefix, this.callNumberTypeId, this.sourceId, - this.administrativeNotes); + this.administrativeNotes, + this.hrId); } public HoldingRequestBuilder withCallNumberSuffix(String suffix) { @@ -161,7 +170,8 @@ public HoldingRequestBuilder withCallNumberSuffix(String suffix) { this.callNumberPrefix, this.callNumberTypeId, this.sourceId, - this.administrativeNotes); + this.administrativeNotes, + this.hrId); } public HoldingRequestBuilder withCallNumberPrefix(String prefix) { @@ -174,7 +184,8 @@ public HoldingRequestBuilder withCallNumberPrefix(String prefix) { prefix, this.callNumberTypeId, this.sourceId, - this.administrativeNotes); + this.administrativeNotes, + this.hrId); } public HoldingRequestBuilder withCallNumberTypeId(String callNumberTypeId) { @@ -187,7 +198,8 @@ public HoldingRequestBuilder withCallNumberTypeId(String callNumberTypeId) { this.callNumberPrefix, callNumberTypeId, this.sourceId, - this.administrativeNotes); + this.administrativeNotes, + this.hrId); } public HoldingRequestBuilder withMarcSource() { @@ -200,7 +212,8 @@ public HoldingRequestBuilder withMarcSource() { this.callNumberPrefix, this.callNumberTypeId, MARC_SOURCE_HOLDINGS_ID, - this.administrativeNotes); + this.administrativeNotes, + this.hrId); } public HoldingRequestBuilder withAdministrativeNotes(List administrativeNotes) { @@ -213,6 +226,21 @@ public HoldingRequestBuilder withAdministrativeNotes(List administrative this.callNumberPrefix, this.callNumberTypeId, this.sourceId, - administrativeNotes); + administrativeNotes, + this.hrId); + } + + public HoldingRequestBuilder withHrId(String hrId) { + return new HoldingRequestBuilder( + this.instanceId, + this.permanentLocationId, + this.temporaryLocationId, + this.callNumber, + this.callNumberSuffix, + this.callNumberPrefix, + this.callNumberTypeId, + this.sourceId, + this.administrativeNotes, + hrId); } } From 22956e79b14064213760bf999489359bf83a96dd Mon Sep 17 00:00:00 2001 From: Roman_Chernetskyi Date: Fri, 21 Jun 2024 15:07:00 +0300 Subject: [PATCH 09/11] Fix sonar errors --- .../resources/UpdateOwnershipApi.java | 28 ++++++++++--------- 1 file changed, 15 insertions(+), 13 deletions(-) diff --git a/src/main/java/org/folio/inventory/resources/UpdateOwnershipApi.java b/src/main/java/org/folio/inventory/resources/UpdateOwnershipApi.java index e425dc8f6..0c295d7ce 100644 --- a/src/main/java/org/folio/inventory/resources/UpdateOwnershipApi.java +++ b/src/main/java/org/folio/inventory/resources/UpdateOwnershipApi.java @@ -50,6 +50,7 @@ public class UpdateOwnershipApi extends AbstractInventoryResource { public static final String INSTANCE_NOT_FOUND_AT_SOURCE_TENANT = "Instance with id: %s not found at source tenant, tenant: %s"; public static final String TENANT_NOT_IN_CONSORTIA = "%s tenant is not in consortia"; public static final String HOLDINGS_NOT_FOUND = "HoldingsRecord with id: %s not found on tenant: %s"; + public static final String LOG_UPDATE_HOLDINGS_OWNERSHIP = "updateHoldingsOwnership:: %s"; private final ConsortiumService consortiumService; @@ -96,18 +97,18 @@ private void processUpdateHoldingsOwnership(RoutingContext routingContext) { return updateOwnershipOfHoldingsRecords(holdingsUpdateOwnership, notUpdatedEntities, routingContext, context, targetTenantContext); } else { String instanceNotSharedErrorMessage = String.format(INSTANCE_NOT_SHARED, holdingsUpdateOwnership.getToInstanceId()); - LOGGER.warn("updateHoldingsOwnership:: " + instanceNotSharedErrorMessage); + LOGGER.warn(String.format(LOG_UPDATE_HOLDINGS_OWNERSHIP, instanceNotSharedErrorMessage)); return CompletableFuture.failedFuture(new BadRequestException(instanceNotSharedErrorMessage)); } } else { String instanceNotFoundErrorMessage = String.format(INSTANCE_NOT_FOUND_AT_SOURCE_TENANT, holdingsUpdateOwnership.getToInstanceId(), context.getTenantId()); - LOGGER.warn("updateHoldingsOwnership:: " + instanceNotFoundErrorMessage); + LOGGER.warn(String.format(LOG_UPDATE_HOLDINGS_OWNERSHIP, instanceNotFoundErrorMessage)); return CompletableFuture.failedFuture(new NotFoundException(instanceNotFoundErrorMessage)); } }); } String notInConsortiaErrorMessage = String.format(TENANT_NOT_IN_CONSORTIA, context.getTenantId()); - LOGGER.warn("updateHoldingsOwnership:: " + notInConsortiaErrorMessage, context); + LOGGER.warn(String.format(LOG_UPDATE_HOLDINGS_OWNERSHIP, notInConsortiaErrorMessage)); return CompletableFuture.failedFuture(new BadRequestException(notInConsortiaErrorMessage)); }) .thenAccept(v -> respond(routingContext, notUpdatedEntities)) @@ -143,8 +144,8 @@ private CompletableFuture> updateOwnershipOfHoldingsRecords(Holding return holdingsRecordFetchClient.find(holdingsUpdateOwnership.getHoldingsRecordIds(), MoveApiUtil::fetchByIdCql) .thenCompose(jsons -> { - LOGGER.info("updateOwnershipOfHoldingsRecords:: Founded holdings to update ownership: {}", jsons); - processNotFoundedInstances(holdingsUpdateOwnership.getHoldingsRecordIds(), notUpdatedEntities, context, jsons); + LOGGER.info("updateOwnershipOfHoldingsRecords:: Found holdings to update ownership: {}", jsons); + processNotFoundInstances(holdingsUpdateOwnership.getHoldingsRecordIds(), notUpdatedEntities, context, jsons); if (!jsons.isEmpty()) { return createHoldings(jsons, notUpdatedEntities, holdingsUpdateOwnership.getToInstanceId(), targetTenantHoldingsRecordCollection) .thenCompose(createdHoldings -> { @@ -217,8 +218,9 @@ private CompletableFuture> createItems(List jsons, List> createHoldings(List jsons, List notUpdatedEntities, String instanceId, HoldingsRecordCollection holdingsRecordCollection) { + jsons.forEach(MoveApiUtil::removeExtraRedundantFields); + List holdingsRecordsToUpdateOwnership = jsons.stream() - .peek(MoveApiUtil::removeExtraRedundantFields) .map(json -> json.mapTo(HoldingsRecord.class).withHrid(null)) .filter(holdingsRecord -> holdingsRecord.getInstanceId().equals(instanceId)) .toList(); @@ -263,8 +265,8 @@ private CompletableFuture> deleteHoldings(List hold .toList()); } - private CompletableFuture> deleteItems(List itemIds, List notUpdatedEntities, ItemCollection itemCollection) { - List> deleteFutures = itemIds.stream() + private CompletableFuture> deleteItems(List items, List notUpdatedEntities, ItemCollection itemCollection) { + List> deleteFutures = items.stream() .map(item -> { Promise promise = Promise.promise(); itemCollection.delete(item.getId(), success -> promise.complete(item.getId()), @@ -285,12 +287,12 @@ private CompletableFuture> deleteItems(List itemIds, List holdingsRecordIds, List notUpdatedEntities, WebContext context, List jsons) { - List foundedIds = jsons.stream().map(json -> json.getString("id")).toList(); - List notFoundedIds = ListUtils.subtract(holdingsRecordIds, foundedIds); - notFoundedIds.forEach(id -> { + private void processNotFoundInstances(List holdingsRecordIds, List notUpdatedEntities, WebContext context, List jsons) { + List foundIds = jsons.stream().map(json -> json.getString("id")).toList(); + List notFoundIds = ListUtils.subtract(holdingsRecordIds, foundIds); + notFoundIds.forEach(id -> { String errorMessage = String.format(HOLDINGS_NOT_FOUND, id, context.getTenantId()); - LOGGER.warn("processNotFoundedInstances:: " + errorMessage); + LOGGER.warn(String.format("processNotFoundInstances:: %s", errorMessage)); notUpdatedEntities.add(new NotUpdatedEntity().withEntityId(id).withErrorMessage(errorMessage)); }); } From a190cede648da4ba68bf5596a54f9bf499e7ecf9 Mon Sep 17 00:00:00 2001 From: Roman_Chernetskyi Date: Sat, 22 Jun 2024 19:47:21 +0300 Subject: [PATCH 10/11] Add HoldingsUpdateOwnership to ApiTestSuite --- src/test/java/api/ApiTestSuite.java | 4 +++- 1 file changed, 3 insertions(+), 1 deletion(-) diff --git a/src/test/java/api/ApiTestSuite.java b/src/test/java/api/ApiTestSuite.java index 7db9b0e6f..4f3c967bc 100644 --- a/src/test/java/api/ApiTestSuite.java +++ b/src/test/java/api/ApiTestSuite.java @@ -11,6 +11,7 @@ import java.util.concurrent.TimeUnit; import java.util.concurrent.TimeoutException; +import api.holdings.HoldingsUpdateOwnershipApiTest; import org.folio.inventory.InventoryVerticle; import org.folio.inventory.common.VertxAssistant; import org.folio.inventory.consortium.util.ConsortiumUtil; @@ -68,7 +69,8 @@ BoundWithTests.class, TenantApiTest.class, AdminApiTest.class, - InventoryConfigApiTest.class + InventoryConfigApiTest.class, + HoldingsUpdateOwnershipApiTest.class }) public class ApiTestSuite { public static final int INVENTORY_VERTICLE_TEST_PORT = 9603; From cadcd08d5ea7ab0ff602b0c19ac548cd7182deab Mon Sep 17 00:00:00 2001 From: Roman_Chernetskyi Date: Mon, 24 Jun 2024 15:14:40 +0300 Subject: [PATCH 11/11] Fix sonar issues --- src/main/java/org/folio/inventory/resources/MoveApi.java | 3 ++- .../folio/inventory/validation/UpdateOwnershipValidator.java | 2 +- 2 files changed, 3 insertions(+), 2 deletions(-) diff --git a/src/main/java/org/folio/inventory/resources/MoveApi.java b/src/main/java/org/folio/inventory/resources/MoveApi.java index 3c0cc2687..68990f155 100644 --- a/src/main/java/org/folio/inventory/resources/MoveApi.java +++ b/src/main/java/org/folio/inventory/resources/MoveApi.java @@ -168,8 +168,9 @@ private void updateItems(RoutingContext routingContext, WebContext context, List } private List updateInstanceIdForHoldings(String toInstanceId, List jsons) { + jsons.forEach(MoveApiUtil::removeExtraRedundantFields); + return jsons.stream() - .peek(MoveApiUtil::removeExtraRedundantFields) .map(json -> json.mapTo(HoldingsRecord.class)) .map(holding -> holding.withInstanceId(toInstanceId)) .collect(toList()); diff --git a/src/main/java/org/folio/inventory/validation/UpdateOwnershipValidator.java b/src/main/java/org/folio/inventory/validation/UpdateOwnershipValidator.java index c51dfd5bc..bd5afdb55 100644 --- a/src/main/java/org/folio/inventory/validation/UpdateOwnershipValidator.java +++ b/src/main/java/org/folio/inventory/validation/UpdateOwnershipValidator.java @@ -18,7 +18,7 @@ public static Optional updateOwnershipHasRequiredFields(String List requiredFields = Arrays.stream(updateOwnershipClass.getDeclaredFields()).map(Field::getName).toList(); for (String field: requiredFields) { var value = updateOwnershipRequest.getValue(field); - if (value == null || (value instanceof JsonArray && ((JsonArray) value).isEmpty())) { + if (value == null || (value instanceof JsonArray jsonArray && jsonArray.isEmpty())) { return Optional.of(new ValidationError(field + " is a required field", field, null)); } }