From 40ece0860754ddc58bdfb399dc0671ae97055b4a Mon Sep 17 00:00:00 2001 From: RomanChernetskyi <111740564+RomanChernetskyi@users.noreply.github.com> Date: Tue, 25 Jun 2024 10:55:49 +0300 Subject: [PATCH] [MODINV-1031] Implement endpoint to update ownership of Holdings (#731) * [MODINV-1031] Implement endpoint to update ownership of Holdings * Implement updating ownership of holdings withou updating ownership of underlying items * transfer along with all the Items attached to a selected Holdings * Response with error message * Add description for entityId * Add logs on failures * Add logs for debug * Not set hrid for items and holdings during create * Fix sonar errors * Add HoldingsUpdateOwnership to ApiTestSuite * Fix sonar issues --- pom.xml | 3 + ramls/update_ownership_response.json | 25 ++ .../folio/inventory/InventoryVerticle.java | 2 +- .../folio/inventory/resources/MoveApi.java | 100 +---- .../resources/UpdateOwnershipApi.java | 290 ++++++++++++- .../folio/inventory/support/MoveApiUtil.java | 110 +++++ .../validation/UpdateOwnershipValidator.java | 31 ++ src/test/java/api/ApiTestSuite.java | 5 +- .../HoldingsUpdateOwnershipApiTest.java | 404 +++++++++++++++--- src/test/java/api/support/ApiTests.java | 7 + .../builders/HoldingRequestBuilder.java | 48 ++- ...gsRecordUpdateOwnershipRequestBuilder.java | 4 +- .../fakes/FakeStorageModuleBuilder.java | 2 +- 13 files changed, 862 insertions(+), 169 deletions(-) create mode 100644 ramls/update_ownership_response.json create mode 100644 src/main/java/org/folio/inventory/support/MoveApiUtil.java create mode 100644 src/main/java/org/folio/inventory/validation/UpdateOwnershipValidator.java diff --git a/pom.xml b/pom.xml index 4e7b3104d..228b5b9f5 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/update_ownership_response.json ${basedir}/ramls/instance-ingress-event.json org.folio diff --git a/ramls/update_ownership_response.json b/ramls/update_ownership_response.json new file mode 100644 index 000000000..9ef38d0ef --- /dev/null +++ b/ramls/update_ownership_response.json @@ -0,0 +1,25 @@ +{ + "$schema": "http://json-schema.org/draft-04/schema#", + "description": "Holder for errors during item/holdingsRecord update ownership", + "type": "object", + "properties": { + "notUpdatedEntities": { + "description": "Item/HoldingsRecord errors", + "type": "array", + "items": { + "type": "object", + "properties": { + "entityId": { + "$ref": "uuid.json", + "description": "Id of item/holdingsRecord" + }, + "errorMessage": { + "type": "string", + "description": "Error message" + } + } + } + } + }, + "additionalProperties": false +} diff --git a/src/main/java/org/folio/inventory/InventoryVerticle.java b/src/main/java/org/folio/inventory/InventoryVerticle.java index 26af54824..80f2bdae9 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 14216ddb0..68990f155 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); @@ -176,18 +168,14 @@ private void updateItems(RoutingContext routingContext, WebContext context, List } private List updateInstanceIdForHoldings(String toInstanceId, List jsons) { + jsons.forEach(MoveApiUtil::removeExtraRedundantFields); + return jsons.stream() - .peek(this::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 +192,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 339fb8886..0c295d7ce 100644 --- a/src/main/java/org/folio/inventory/resources/UpdateOwnershipApi.java +++ b/src/main/java/org/folio/inventory/resources/UpdateOwnershipApi.java @@ -1,28 +1,304 @@ 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.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.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; +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; +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.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; public class UpdateOwnershipApi extends AbstractInventoryResource { - public UpdateOwnershipApi(Storage storage, HttpClient client) { + private static final Logger LOGGER = LogManager.getLogger(MethodHandles.lookup().lookupClass()); + 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 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; + + public UpdateOwnershipApi(Storage storage, HttpClient client, ConsortiumService consortiumService) { super(storage, client); + this.consortiumService = 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 processUpdateHoldingsOwnership(RoutingContext routingContext) { + try { + final var context = new WebContext(routingContext); + final var updateOwnershipRequest = routingContext.body().asJsonObject(); + + final var validationError = updateOwnershipHasRequiredFields(context.getTenantId(), updateOwnershipRequest, HoldingsUpdateOwnership.class); + + if (validationError.isPresent()) { + unprocessableEntity(routingContext.response(), validationError.get()); + return; + } + 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()); + + 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())) { + Context targetTenantContext = constructContext(holdingsUpdateOwnership.getTargetTenantId(), context.getToken(), context.getOkapiLocation()); + return updateOwnershipOfHoldingsRecords(holdingsUpdateOwnership, notUpdatedEntities, routingContext, context, targetTenantContext); + } else { + String instanceNotSharedErrorMessage = String.format(INSTANCE_NOT_SHARED, holdingsUpdateOwnership.getToInstanceId()); + 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(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(String.format(LOG_UPDATE_HOLDINGS_OWNERSHIP, notInConsortiaErrorMessage)); + return CompletableFuture.failedFuture(new BadRequestException(notInConsortiaErrorMessage)); + }) + .thenAccept(v -> respond(routingContext, notUpdatedEntities)) + .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("updateHoldingsOwnership:: Error during update ownership of holdings", e); + handleFailure(e, routingContext); + } + } + + private void processUpdateItemsOwnership(RoutingContext routingContext) { + // should be implemented in MODINV-955 + } + + private CompletableFuture> updateOwnershipOfHoldingsRecords(HoldingsUpdateOwnership holdingsUpdateOwnership, + 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); + + HoldingsRecordCollection sourceTenantHoldingsRecordCollection = storage.getHoldingsRecordCollection(context); + HoldingsRecordCollection targetTenantHoldingsRecordCollection = storage.getHoldingsRecordCollection(targetTenantContext); + + return holdingsRecordFetchClient.find(holdingsUpdateOwnership.getHoldingsRecordIds(), MoveApiUtil::fetchByIdCql) + .thenCompose(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 -> { + 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: {}", + holdingsUpdateOwnership.getHoldingsRecordIds(), holdingsUpdateOwnership.getTargetTenantId(), e); + return CompletableFuture.failedFuture(e); + } + } + + 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); + + ItemCollection sourceTenantItemCollection = storage.getItemCollection(context); + ItemCollection targetTenantItemCollection = storage.getItemCollection(targetTenantContext); + + return itemsFetchClient.find(holdingsRecordIds, MoveApiUtil::fetchByHoldingsRecordIdCql) + .thenCompose(jsons -> { + LOGGER.info("transferAttachedItems:: Found items to transfer: {}", jsons); + 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: {}", + holdingsRecordIds, targetTenantContext.getTenantId(), e); + return CompletableFuture.failedFuture(e); + } + } + + private CompletableFuture> createItems(List jsons, List notUpdatedEntities, ItemCollection itemCollection) { + List itemRecordsToUpdateOwnership = jsons.stream() + .map(json -> ItemUtil.fromStoredItemRepresentation(json).withHrid(null)) + .toList(); + + List> createFutures = itemRecordsToUpdateOwnership.stream() + .map(item -> + 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); + })) + .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, List notUpdatedEntities, String instanceId, + HoldingsRecordCollection holdingsRecordCollection) { + jsons.forEach(MoveApiUtil::removeExtraRedundantFields); + + List holdingsRecordsToUpdateOwnership = jsons.stream() + .map(json -> json.mapTo(HoldingsRecord.class).withHrid(null)) + .filter(holdingsRecord -> holdingsRecord.getInstanceId().equals(instanceId)) + .toList(); + + List> createFutures = holdingsRecordsToUpdateOwnership.stream() + .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); + })) + .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 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 -> { + 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()); + }); + 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 items, List notUpdatedEntities, ItemCollection itemCollection) { + List> deleteFutures = items.stream() + .map(item -> { + Promise 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()); + }); + 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 void updateHoldingsOwnership(RoutingContext routingContext) { - // should be implemented in MODINV-1031 + 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(String.format("processNotFoundInstances:: %s", errorMessage)); + notUpdatedEntities.add(new NotUpdatedEntity().withEntityId(id).withErrorMessage(errorMessage)); + }); } - private void updateItemsOwnership(RoutingContext routingContext) { - // should be implemented in MODINV-1031 + 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 new file mode 100644 index 000000000..af6051a21 --- /dev/null +++ b/src/main/java/org/folio/inventory/support/MoveApiUtil.java @@ -0,0 +1,110 @@ +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.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; +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"; + 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() { } + + 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 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<>()); + } + + 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(); + if (nonUpdatedIds.isEmpty()) { + successWithEmptyIds(response); + } else { + successWithIds(response, nonUpdatedIds); + } + } + + public static void respond(RoutingContext routingContext, List notUpdatedEntities) { + HttpServerResponse response = routingContext.response(); + success(response, JsonObject.mapFrom(new UpdateOwnershipResponse().withNotUpdatedEntities(notUpdatedEntities))); + } +} 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..bd5afdb55 --- /dev/null +++ b/src/main/java/org/folio/inventory/validation/UpdateOwnershipValidator.java @@ -0,0 +1,31 @@ +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; + +import static org.folio.inventory.support.MoveApiUtil.TARGET_TENANT_ID; + +public final class UpdateOwnershipValidator { + private UpdateOwnershipValidator() { } + + 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); + if (value == null || (value instanceof JsonArray jsonArray && jsonArray.isEmpty())) { + 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/ApiTestSuite.java b/src/test/java/api/ApiTestSuite.java index 2373dc0fd..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,12 +69,14 @@ BoundWithTests.class, TenantApiTest.class, AdminApiTest.class, - InventoryConfigApiTest.class + InventoryConfigApiTest.class, + HoldingsUpdateOwnershipApiTest.class }) 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..85c48a826 100644 --- a/src/test/java/api/holdings/HoldingsUpdateOwnershipApiTest.java +++ b/src/test/java/api/holdings/HoldingsUpdateOwnershipApiTest.java @@ -6,17 +6,21 @@ 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; 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 +29,11 @@ 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.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; @@ -35,7 +41,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"; @@ -55,32 +60,220 @@ 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); + + 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("notUpdatedEntities").size(), is(0)); + assertThat(postHoldingsUpdateOwnershipResponse.getContentType(), containsString(APPLICATION_JSON)); + + 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(createHoldingsRecord2); + Response targetTenantHoldingsRecord2 = collegeHoldingsStorageClient.getById(createHoldingsRecord2); + + 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())); 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) + .withHrid(itemHrId) + .withBarcode("645398607546") + .withStatus(ItemStatusName.AVAILABLE.value())); + 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); 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 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 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)); + + // 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())); + Assert.assertNotEquals(targetTenantItem2.getJson().getString("hrid"), itemHrId); + } + + @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 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.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())), ApiTestSuite.COLLEGE_TENANT_ID).create(); + + 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(HttpStatus.SC_NOT_FOUND, is(targetTenantItem1.getStatusCode())); + assertThat(sourceTenantItem1.getJson().getString(HOLDINGS_RECORD_ID), is(createHoldingsRecord1.toString())); + } + + @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(); + + Response postHoldingsUpdateOwnershipResponse = updateHoldingsRecordsOwnership(holdingsRecordUpdateOwnershipRequestBody); + + collegeItemsClient.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 @@ -88,8 +281,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(); @@ -97,27 +290,28 @@ 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); 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"), + equalTo(String.format("HoldingsRecord with id: %s not found on tenant: %s", createHoldingsRecord2, ApiTestSuite.TENANT_ID))); 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()); } @@ -125,7 +319,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 +343,23 @@ 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 + )); + } + + @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 )); } @@ -157,7 +367,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,15 +375,29 @@ 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 )); } @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); @@ -185,54 +409,121 @@ 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); - 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 canUpdateHoldingsRecordOwnershipDueToHoldingsRecordUpdateError() throws InterruptedException, MalformedURLException, TimeoutException, ExecutionException { + public void cannotUpdateHoldingsRecordOwnershipOfNonSharedInstance() + throws MalformedURLException, InterruptedException, ExecutionException, TimeoutException { UUID instanceId = UUID.randomUUID(); JsonObject instance = smallAngryPlanet(instanceId); InstanceApiClient.createInstance(okapiClient, instance); - InstanceApiClient.createInstance(consortiumOkapiClient, instance); final UUID createHoldingsRecord1 = createHoldingForInstance(instanceId); - final UUID createHoldingsRecord2 = createHoldingForInstance(ID_FOR_FAILURE, instanceId); + 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); - List nonUpdatedIdsIds = postHoldingsUpdateOwnershipResponse.getJson() - .getJsonArray("nonUpdatedIds") - .getList(); + assertThat(postHoldingsUpdateOwnershipResponse.getStatusCode(), is(400)); - assertThat(nonUpdatedIdsIds.size(), is(1)); - assertThat(nonUpdatedIdsIds.get(0), equalTo(ID_FOR_FAILURE.toString())); + assertThat(postHoldingsUpdateOwnershipResponse.getBody(), containsString(String.format("Instance with id: %s is not shared", instanceId))); + } + + @Test + public void cannotUpdateHoldingsRecordOwnershipDueToHoldingsRecordCreateError() 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 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())), ApiTestSuite.COLLEGE_TENANT_ID).create(); + + Response postHoldingsUpdateOwnershipResponse = updateHoldingsRecordsOwnership(holdingsRecordUpdateOwnershipRequestBody); + + collegeHoldingsStorageClient.disableFailureEmulation(); + + 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.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()); + } + + @Test + public void cannotUpdateHoldingsRecordOwnershipDueToHoldingsRecordDeleteError() throws InterruptedException, MalformedURLException, TimeoutException, ExecutionException { + UUID instanceId = UUID.randomUUID(); + JsonObject instance = smallAngryPlanet(instanceId); - Response sourceTenantHoldingsRecord2 = holdingsStorageClient.getById(createHoldingsRecord1); - Response targetTenantHoldingsRecord2 = consortiumHoldingsStorageClient.getById(createHoldingsRecord1); + InstanceApiClient.createInstance(okapiClient, instance.put("source", CONSORTIUM_FOLIO.getValue())); + InstanceApiClient.createInstance(consortiumOkapiClient, instance.put("source", FOLIO.getValue())); - Assert.assertEquals(instanceId.toString(), sourceTenantHoldingsRecord2.getJson().getString(INSTANCE_ID)); - Assert.assertEquals(HttpStatus.SC_NOT_FOUND, targetTenantHoldingsRecord2.getStatusCode()); + final UUID createHoldingsRecord1 = 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())), ApiTestSuite.COLLEGE_TENANT_ID).create(); + + Response postHoldingsUpdateOwnershipResponse = updateHoldingsRecordsOwnership(holdingsRecordUpdateOwnershipRequestBody); + + collegeHoldingsStorageClient.disableFailureEmulation(); + + 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.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)); } @Test @@ -240,8 +531,8 @@ public void canUpdateHoldingsRecordOwnershipToDifferentInstanceWithExtraRedundan 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 @@ -256,22 +547,22 @@ 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); 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); - 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 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)); @@ -280,18 +571,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)) + return holdingsStorageClient.create(new HoldingRequestBuilder().withHrId("hol0000001").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 b23394342..c1300a26b 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 holdingsSourceStorageClient; protected final ResourceClient itemsStorageClient; @@ -35,6 +36,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; @@ -59,6 +62,9 @@ public ApiTests() { consortiumHoldingsStorageClient = ResourceClient.forHoldingsStorage(consortiumOkapiClient); consortiumItemsClient = ResourceClient.forItemsStorage(consortiumOkapiClient); + + collegeHoldingsStorageClient = ResourceClient.forHoldingsStorage(collegeOkapiClient); + collegeItemsClient = ResourceClient.forItemsStorage(collegeOkapiClient); } @BeforeClass @@ -76,6 +82,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/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); } } diff --git a/src/test/java/api/support/builders/HoldingsRecordUpdateOwnershipRequestBuilder.java b/src/test/java/api/support/builders/HoldingsRecordUpdateOwnershipRequestBuilder.java index 2c4e22e0a..fe8e8009d 100644 --- a/src/test/java/api/support/builders/HoldingsRecordUpdateOwnershipRequestBuilder.java +++ b/src/test/java/api/support/builders/HoldingsRecordUpdateOwnershipRequestBuilder.java @@ -6,9 +6,9 @@ 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.TO_INSTANCE_ID; +import static org.folio.inventory.support.MoveApiUtil.TARGET_TENANT_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; } 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()); }