Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

MODORDERS -1087 Delete received pieces in bulk #905

Open
wants to merge 35 commits into
base: master
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from 11 commits
Commits
Show all changes
35 commits
Select commit Hold shift + click to select a range
eda88e0
draft
yuntianhu Apr 19, 2024
e64274c
remove the overide
yuntianhu Apr 19, 2024
28f1b6c
coding format correct
yuntianhu Apr 19, 2024
72ac005
modify pieces.rml, PiecesAPI.java
yuntianhu Apr 25, 2024
68d0484
Merge branch 'master' into MODORDERS-1087
yuntianhu Apr 25, 2024
f01b393
modify pieces.rml, PiecesAPI.java, add unit test, add batch delete in…
yuntianhu Apr 30, 2024
525c527
Merge remote-tracking branch 'origin/MODORDERS-1087' into MODORDERS-1087
yuntianhu Apr 30, 2024
6f668d5
Merge branch 'master' into MODORDERS-1087
yuntianhu Apr 30, 2024
a901c73
modify pieces.rml, PiecesAPI.java, add unit test, add batch delete in…
yuntianhu Apr 30, 2024
d0af7ee
modify add batch delete API test, remove debug comments, change refin…
yuntianhu Apr 30, 2024
e8f7699
modify correct batch delete API test,
yuntianhu Apr 30, 2024
21afa09
modify PiecesDeleteFlowManager.java, pieces.raml, code format correct…
yuntianhu May 7, 2024
2757e77
Merge branch 'master' into MODORDERS-1087
yuntianhu May 7, 2024
be962f1
remove the api bach test for now
yuntianhu May 7, 2024
2b281ef
Merge remote-tracking branch 'origin/MODORDERS-1087' into MODORDERS-1087
yuntianhu May 7, 2024
f5c448a
delete pieces manager test
yuntianhu May 7, 2024
87c595a
Merge branch 'master' into MODORDERS-1087
yuntianhu May 7, 2024
bceb008
remodify the batch delete and add test
yuntianhu May 7, 2024
bf581aa
Merge remote-tracking branch 'origin/MODORDERS-1087' into MODORDERS-1087
yuntianhu May 7, 2024
1cf716a
Merge branch 'master' into MODORDERS-1087
yuntianhu May 8, 2024
3bb8489
remodify test
yuntianhu May 8, 2024
7a2add6
Merge remote-tracking branch 'origin/MODORDERS-1087' into MODORDERS-1087
yuntianhu May 8, 2024
b20df9d
remodify test
yuntianhu May 8, 2024
d32be26
remodify test
yuntianhu May 8, 2024
cf510ae
remodify test with name
yuntianhu May 8, 2024
472c0ee
remove issues sonarcloud
yuntianhu May 8, 2024
8629560
remove sonarcloud issues
yuntianhu May 8, 2024
fca8ab8
Merge branch 'master' into MODORDERS-1087
yuntianhu May 9, 2024
a632499
Merge branch 'master' into MODORDERS-1087
yuntianhu May 10, 2024
eb142cc
Merge branch 'master' into MODORDERS-1087
yuntianhu May 28, 2024
01286f9
Update PieceDeleteFlowManager.java
yuntianhu May 28, 2024
3205ba6
Update PieceDeleteFlowManager.java
yuntianhu May 28, 2024
ac5d118
Update PieceDeleteFlowManager.java
yuntianhu May 28, 2024
6d1fd0d
try to resolve the conflict
yuntianhu May 28, 2024
b8d16d0
Merge remote-tracking branch 'origin/MODORDERS-1087' into MODORDERS-1087
yuntianhu May 28, 2024
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
36 changes: 35 additions & 1 deletion ramls/pieces.raml
Original file line number Diff line number Diff line change
Expand Up @@ -6,7 +6,7 @@ protocols: [ HTTP, HTTPS ]

documentation:
- title: Orders Business Logic API
content: <b>API for managing pieces</b>
content: <b>API for managing pieces including batch operations for deletion.</b>

types:
piece: !include acq-models/mod-orders-storage/schemas/piece.json
Expand Down Expand Up @@ -47,6 +47,40 @@ resourceTypes:
example: true
required: false
default: false
/batch:
delete:
description: Batch delete pieces
body:
application/json:
type: piece-collection
example: !include acq-models/mod-orders-storage/examples/piece_collection.sample
responses:
204:
description: "Batch delete of pieces successfully completed"
body:
application/json:
example: "All pieces successfully deleted."
400:
description: "Bad request, e.g., malformed JSON or validation failed"
body:
application/json:
type: errors
text/plain:
example: "Bad request - malformed JSON or validation failed."
404:
description: "One or more pieces not found"
body:
application/json:
type: errors
text/plain:
example: "Error - one or more pieces not found."
500:
description: "Internal server error, e.g. due to misconfiguration"
body:
application/json:
type: errors
text/plain:
example: "Internal server error - due to misconfiguration."
/{id}:
uriParameters:
id:
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -23,6 +23,7 @@ private ResourcePathResolver() {
public static final String REPORTING_CODES = "reportingCodes";
public static final String PURCHASE_ORDER_STORAGE = "purchaseOrder";
public static final String PIECES_STORAGE = "pieces";
public static final String PIECES_COLLECTION_STORAGE = "PiecesCollection";
public static final String RECEIVING_HISTORY = "receiving-history";
public static final String RECEIPT_STATUS = "receiptStatus";
public static final String PAYMENT_STATUS = "paymentStatus";
Expand Down Expand Up @@ -94,6 +95,7 @@ private ResourcePathResolver() {
apis.put(USERS, "/users");
apis.put(ORDER_SETTINGS, "/orders-storage/settings");
apis.put(ROUTING_LISTS, "/orders-storage/routing-lists");
apis.put(PIECES_COLLECTION_STORAGE, "/orders-storage/pieces-collecton");

SUB_OBJECT_COLLECTION_APIS = Collections.unmodifiableMap(apis);
SUB_OBJECT_ITEM_APIS = Collections.unmodifiableMap(
Expand Down
22 changes: 21 additions & 1 deletion src/main/java/org/folio/rest/impl/PiecesAPI.java
Original file line number Diff line number Diff line change
Expand Up @@ -2,16 +2,21 @@

import static io.vertx.core.Future.succeededFuture;

import java.util.ArrayList;
import java.util.List;
import java.util.Map;

import javax.ws.rs.core.Response;

import io.vertx.core.CompositeFuture;
import io.vertx.core.Future;
import org.apache.commons.lang3.StringUtils;
import org.apache.logging.log4j.LogManager;
import org.apache.logging.log4j.Logger;
import org.folio.rest.annotations.Validate;
import org.folio.rest.core.models.RequestContext;
import org.folio.rest.jaxrs.model.Piece;
import org.folio.rest.jaxrs.model.PieceCollection;
import org.folio.rest.jaxrs.resource.OrdersPieces;
import org.folio.service.pieces.PieceStorageService;
import org.folio.service.pieces.flows.create.PieceCreateFlowManager;
Expand All @@ -38,6 +43,7 @@ public class PiecesAPI extends BaseApi implements OrdersPieces {
@Autowired
private PieceUpdateFlowManager pieceUpdateFlowManager;


public PiecesAPI() {
SpringContextUtil.autowireDependencies(this, Vertx.currentContext());
}
Expand Down Expand Up @@ -66,6 +72,8 @@ public void postOrdersPieces(boolean createItem, Piece entity, Map<String, Strin
.onFailure(t -> handleErrorResponse(asyncResultHandler, t));
}



Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

please reformat this file one more time

@Override
@Validate
public void getOrdersPiecesById(String id, Map<String, String> okapiHeaders,
Expand All @@ -91,9 +99,21 @@ public void putOrdersPiecesById(String pieceId, boolean createItem, boolean dele
@Override
@Validate
public void deleteOrdersPiecesById(String pieceId, boolean deleteHolding, Map<String, String> okapiHeaders,
Handler<AsyncResult<Response>> asyncResultHandler, Context vertxContext) {
Handler<AsyncResult<Response>> asyncResultHandler, Context vertxContext) {
pieceDeleteFlowManager.deletePiece(pieceId, deleteHolding, new RequestContext(vertxContext, okapiHeaders))
.onSuccess(ok -> asyncResultHandler.handle(succeededFuture(buildNoContentResponse())))
.onFailure(fail -> handleErrorResponse(asyncResultHandler, fail));

}

@Override
@Validate
public void deleteOrdersPiecesBatch(PieceCollection entity, Map<String, String> okapiHeaders, Handler<AsyncResult<Response>> asyncResultHandler, Context vertxContext) {
List<String> pieceIds = new ArrayList<>();
for (Piece piece : entity.getPieces()) {
pieceIds.add(piece.getId());
}
azizbekxm marked this conversation as resolved.
Show resolved Hide resolved
pieceDeleteFlowManager.batchDeletePiece(pieceIds, new RequestContext(vertxContext, okapiHeaders));
}
}

Original file line number Diff line number Diff line change
Expand Up @@ -27,7 +27,6 @@

public class PieceStorageService {
private static final Logger logger = LogManager.getLogger(PieceStorageService.class);

private static final String PIECES_BY_POL_ID_AND_STATUS_QUERY = "poLineId==%s and receivingStatus==%s";
private static final String PIECES_BY_HOLDING_ID_QUERY = "holdingId==%s";
private static final String PIECE_STORAGE_ENDPOINT = resourcesPath(PIECES_STORAGE);
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -7,7 +7,9 @@
import java.util.ArrayList;
import java.util.List;
import java.util.Optional;
import java.util.stream.Collectors;

import io.vertx.core.CompositeFuture;
import org.apache.commons.collections4.CollectionUtils;
import org.apache.commons.lang3.StringUtils;
import org.apache.commons.lang3.tuple.Pair;
Expand Down Expand Up @@ -54,6 +56,7 @@ public PieceDeleteFlowManager(PieceStorageService pieceStorageService, Protectio

public Future<Void> deletePiece(String pieceId, boolean deleteHolding, RequestContext requestContext) {
PieceDeletionHolder holder = new PieceDeletionHolder().withDeleteHolding(deleteHolding);

return pieceStorageService.getPieceById(pieceId, requestContext)
.map(pieceToDelete -> {
holder.withPieceToDelete(pieceToDelete); return null;
Expand Down Expand Up @@ -140,4 +143,26 @@ private Future<JsonObject> getOnOrderItemForPiece(Piece piece, RequestContext re
return Future.succeededFuture();
}
}

public Future<Void> batchDeletePiece(List<String> ids, RequestContext requestContext) {
PieceDeletionHolder holder = new PieceDeletionHolder().withDeleteHolding(false);

List<Future> deleteFutures = ids.stream()
.map(id -> pieceStorageService.getPieceById(id, requestContext)
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

instead of fetching piece one by one, can we get as collections? like getPieces(List pieceIds)..

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I will try it. I see there is a max number of get pieces in batch is 15. I am not sure why. but, I will try it first, and see how it goes.

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

it is some constant to prevent long url query size, the max size of url length - 4 kb, historically we use chunks by 15 elems to avoid this issue.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Thank you for the info, Serhii

.map(pieceToDelete -> {
holder.withPieceToDelete(pieceToDelete);
return null;
})
.compose(aHolder -> basePieceFlowHolderBuilder.updateHolderWithOrderInformation(holder, requestContext))
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

The same logic also is in deletePiece method. We should avoid code duplication, logic should be in 1 places

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

thank you Serhii, I will rewrite this part

.compose(aHolder -> basePieceFlowHolderBuilder.updateHolderWithTitleInformation(holder, requestContext))
.compose(aVoid -> protectionService.isOperationRestricted(holder.getTitle().getAcqUnitIds(), DELETE, requestContext))
.compose(aVoid -> isDeletePieceRequestValid(holder, requestContext))
.compose(aVoid -> processInventory(holder, requestContext))
.compose(pair -> updatePoLine(holder, requestContext))
.compose(aVoid -> pieceStorageService.deletePiece(holder.getPieceToDelete().getId(), true, requestContext))
)
.collect(Collectors.toList());

return CompositeFuture.all(deleteFutures).mapEmpty();
}
}
21 changes: 21 additions & 0 deletions src/test/java/org/folio/RestTestUtils.java
Original file line number Diff line number Diff line change
Expand Up @@ -18,10 +18,12 @@

import java.util.Map;

import jakarta.validation.Payload;
import org.apache.commons.lang3.StringUtils;
import org.folio.orders.events.handlers.HandlersTestHelper;
import org.folio.rest.jaxrs.model.Error;
import org.folio.rest.jaxrs.model.Errors;
import org.folio.rest.jaxrs.model.PieceCollection;
import org.folio.rest.tools.parser.JsonPathParser;
import org.hamcrest.Matchers;

Expand Down Expand Up @@ -171,6 +173,25 @@ public static Response verifyDeleteResponse(String url, Headers headers, String
return response;
}

public static Response verifyBatchDeleteResponse(String url, String body, Headers headers, String expectedContentType, int expectedCode) {
Response response = RestAssured
.with()
.header(X_OKAPI_URL)
.header(X_OKAPI_TOKEN)
.headers(headers)
.contentType(APPLICATION_JSON)
.body(body)
.when()
.delete(url)
.then()
.statusCode(expectedCode)
.contentType(expectedContentType)
.extract()
.response();
HandlersTestHelper.verifyOrderStatusUpdateEvent(0);
return response;
}

public static void checkPreventProtectedFieldsModificationRule(String path, JsonObject compPO, Map<String, Object> updatedFields) {
JsonObject compPOJson = JsonObject.mapFrom(compPO);
JsonPathParser compPOParser = new JsonPathParser(compPOJson);
Expand Down
43 changes: 43 additions & 0 deletions src/test/java/org/folio/rest/impl/PieceApiTest.java
Original file line number Diff line number Diff line change
@@ -1,11 +1,14 @@
package org.folio.rest.impl;

import static io.restassured.RestAssured.given;
import static javax.ws.rs.core.MediaType.APPLICATION_JSON;
import static javax.ws.rs.core.MediaType.TEXT_PLAIN;
import static org.folio.RestTestUtils.prepareHeaders;
import static org.folio.RestTestUtils.verifyBatchDeleteResponse;
import static org.folio.RestTestUtils.verifyDeleteResponse;
import static org.folio.RestTestUtils.verifyPostResponse;
import static org.folio.RestTestUtils.verifyPut;
import static org.folio.TestConfig.X_OKAPI_URL;
import static org.folio.TestConfig.clearServiceInteractions;
import static org.folio.TestConfig.initSpringContext;
import static org.folio.TestConfig.isVerticleNotDeployed;
Expand All @@ -15,9 +18,12 @@
import static org.folio.TestConstants.ID_DOES_NOT_EXIST;
import static org.folio.TestConstants.ID_FOR_INTERNAL_SERVER_ERROR;
import static org.folio.TestConstants.X_ECHO_STATUS;
import static org.folio.TestConstants.X_OKAPI_TOKEN;
import static org.folio.TestConstants.X_OKAPI_USER_ID;
import static org.folio.TestConstants.X_OKAPI_USER_ID_WITH_ACQ_UNITS;
import static org.folio.TestUtils.getMockAsJson;
import static org.folio.TestUtils.getMockData;
import static org.folio.orders.utils.ResourcePathResolver.PIECES_COLLECTION_STORAGE;
import static org.folio.orders.utils.ResourcePathResolver.PIECES_STORAGE;
import static org.folio.orders.utils.ResourcePathResolver.PO_LINES_STORAGE;
import static org.folio.orders.utils.ResourcePathResolver.PURCHASE_ORDER_STORAGE;
Expand All @@ -31,15 +37,20 @@
import static org.hamcrest.Matchers.is;
import static org.hamcrest.Matchers.notNullValue;
import static org.hamcrest.Matchers.nullValue;
import static org.junit.Assert.assertEquals;
import static org.junit.jupiter.api.Assertions.assertNull;

import javax.ws.rs.core.Response;
import java.util.ArrayList;
import java.util.Collections;
import java.util.List;
import java.util.Map;
import java.util.UUID;
import java.util.concurrent.ExecutionException;
import java.util.concurrent.TimeoutException;

import io.restassured.http.Header;
import io.restassured.http.Headers;
import io.vertx.core.json.JsonObject;
import org.apache.logging.log4j.LogManager;
import org.apache.logging.log4j.Logger;
Expand All @@ -55,12 +66,15 @@
import org.folio.rest.jaxrs.model.Location;
import org.folio.rest.jaxrs.model.Physical;
import org.folio.rest.jaxrs.model.Piece;
import org.folio.rest.jaxrs.model.PieceCollection;
import org.folio.rest.jaxrs.model.PurchaseOrder;
import org.folio.rest.jaxrs.model.Title;
import org.junit.jupiter.api.AfterAll;
import org.junit.jupiter.api.AfterEach;
import org.junit.jupiter.api.BeforeAll;
import org.junit.jupiter.api.Test;
import org.junit.jupiter.api.Timeout;
import org.testcontainers.shaded.com.fasterxml.jackson.core.JsonProcessingException;

public class PieceApiTest {

Expand All @@ -69,6 +83,7 @@ public class PieceApiTest {
public static final String PIECES_ENDPOINT = "/orders/pieces";
private static final String PIECES_ID_PATH = PIECES_ENDPOINT + "/%s";
static final String CONSISTENT_RECEIVED_STATUS_PIECE_UUID = "7d0aa803-a659-49f0-8a95-968f277c87d7";
private static final String PIECES_BATCH_PATH = PIECES_ENDPOINT+"/batch";
private JsonObject pieceJsonReqData = getMockAsJson(PIECE_RECORDS_MOCK_DATA_PATH + "pieceRecord.json");

private static boolean runningOnOwn;
Expand Down Expand Up @@ -356,4 +371,32 @@ void deletePieceInternalErrorOnStorageTest() {
logger.info("=== Test delete piece by id - internal error from storage 500 ===");
verifyDeleteResponse(String.format(PIECES_ID_PATH, ID_FOR_INTERNAL_SERVER_ERROR), APPLICATION_JSON, 500);
}

@Test
@Timeout(5)
void deletePiecesByBatchWithItemDeletionTest() throws JsonProcessingException {
logger.info("=== Test delete pieces in batch - items deleted ===");
Piece postPieceRq = pieceJsonReqData.mapTo(Piece.class);
postPieceRq.withId("2bafc9e1-9dd3-4ede-9f23-c4a03f8bb205");
postPieceRq.setPoLineId("2bafc9e1-9dd3-4ede-9f23-c4a03f8bb2d5");
postPieceRq.setReceiptDate(null);
String orderId = UUID.randomUUID().toString();
CompositePurchaseOrder order = new CompositePurchaseOrder().withId(orderId);
verifyPostResponse(PIECES_ENDPOINT, JsonObject.mapFrom(postPieceRq).encode(),
prepareHeaders(EXIST_CONFIG_X_OKAPI_TENANT_LIMIT_10, X_OKAPI_USER_ID, X_OKAPI_USER_ID_WITH_ACQ_UNITS), APPLICATION_JSON, 201).as(Piece.class);

List<Piece> pieces = new ArrayList<>();
pieces.add(postPieceRq);
PieceCollection pieceCollection = new PieceCollection().withPieces(pieces);
MockServer.addMockEntry(PO_LINES_STORAGE, JsonObject.mapFrom(postPieceRq));
MockServer.addMockEntry(PURCHASE_ORDER_STORAGE, JsonObject.mapFrom(order));
JsonObject jsonPieceCollection = JsonObject.mapFrom(pieceCollection);
MockServer.addMockEntry(PIECES_STORAGE, JsonObject.mapFrom(postPieceRq));
MockServer.addMockEntry(PIECES_COLLECTION_STORAGE,JsonObject.mapFrom(pieceCollection));
int status204 = HttpStatus.HTTP_NO_CONTENT.toInt();

verifyBatchDeleteResponse(PIECES_BATCH_PATH, jsonPieceCollection.encode(),
prepareHeaders(EXIST_CONFIG_X_OKAPI_TENANT_LIMIT_10,X_OKAPI_USER_ID_WITH_ACQ_UNITS ,X_OKAPI_TOKEN, X_OKAPI_USER_ID), APPLICATION_JSON,204);
}

}
Original file line number Diff line number Diff line change
Expand Up @@ -356,6 +356,68 @@ void shouldUpdateLineQuantityIfPoLineIsNotPackageAndManualPieceCreateFalseAndInv
verify(basePieceFlowHolderBuilder).updateHolderWithOrderInformation(holder, requestContext);
}

@Test
void shouldUpdateLineQuantityIfPoLineIsNotPackageAndManualPieceCreateFalseAndInventoryInstanceVsHoldingAndDeleteHoldingAndPieceInBatch() {
String orderId = UUID.randomUUID().toString();
String holdingId = UUID.randomUUID().toString();
String lineId = UUID.randomUUID().toString();
String itemId = UUID.randomUUID().toString();
String locationId = UUID.randomUUID().toString();
String titleId = UUID.randomUUID().toString();
JsonObject holding = new JsonObject();
holding.put(ID, holdingId);
holding.put(HOLDING_PERMANENT_LOCATION_ID, locationId);
JsonObject item = new JsonObject().put(ID, itemId);
item.put(ITEM_STATUS, new JsonObject().put(ITEM_STATUS_NAME, ItemStatus.ON_ORDER.value()));
Piece piece = new Piece().withId(UUID.randomUUID().toString()).withPoLineId(lineId)
.withHoldingId(holdingId).withFormat(Piece.Format.ELECTRONIC);
Location loc = new Location().withHoldingId(holdingId).withQuantityElectronic(1).withQuantity(1);
Cost cost = new Cost().withQuantityElectronic(1)
.withListUnitPriceElectronic(1d).withExchangeRate(1d).withCurrency("USD")
.withPoLineEstimatedPrice(1d);
PoLine poLine = new PoLine().withIsPackage(false).withCheckinItems(false).withOrderFormat(PoLine.OrderFormat.ELECTRONIC_RESOURCE)
.withEresource(new Eresource().withCreateInventory(Eresource.CreateInventory.INSTANCE_HOLDING))
.withPurchaseOrderId(orderId).withId(lineId).withLocations(List.of(loc)).withCost(cost);
PurchaseOrder purchaseOrder = new PurchaseOrder().withId(orderId).withWorkflowStatus(PurchaseOrder.WorkflowStatus.OPEN);
Title title = new Title().withId(titleId);
List<String> ids = new ArrayList<>();
ids.add(piece.getId());

doReturn(succeededFuture(piece)).when(pieceStorageService).getPieceById(piece.getId(), requestContext);
doReturn(succeededFuture(null)).when(protectionService).isOperationRestricted(any(), any(ProtectedOperationType.class), eq(requestContext));
doReturn(succeededFuture(null)).when(pieceStorageService).deletePiece(eq(piece.getId()), eq(true), eq(requestContext));
doReturn(succeededFuture(null)).when(inventoryItemManager).getNumberOfRequestsByItemId(eq(piece.getItemId()), eq(requestContext));
doReturn(succeededFuture(holding)).when(inventoryHoldingManager).getHoldingById(holdingId, false, requestContext);
doReturn(succeededFuture(null)).when(inventoryItemManager).getItemsByHoldingId(holdingId, requestContext);
doReturn(succeededFuture(null)).when(inventoryHoldingManager).deleteHoldingById(piece.getHoldingId(), true, requestContext);
doReturn(succeededFuture(null)).when(inventoryItemManager).getItemRecordById(itemId, true, requestContext);
doReturn(succeededFuture(null)).when(inventoryItemManager).deleteItem(itemId, true, requestContext);
doReturn(succeededFuture(holding)).when(inventoryHoldingManager).getHoldingById(holdingId, true, requestContext);
doReturn(succeededFuture(null)).when(pieceUpdateInventoryService).deleteHoldingConnectedToPiece(piece, requestContext);
doReturn(succeededFuture(new ArrayList<JsonObject>())).when(inventoryItemManager).getItemsByHoldingId(holdingId, requestContext);
final ArgumentCaptor<PieceDeletionHolder> PieceDeletionHolderCapture = ArgumentCaptor.forClass(PieceDeletionHolder.class);
doAnswer((Answer<Future<Void>>) invocation -> {
PieceDeletionHolder answerHolder = invocation.getArgument(0);
answerHolder.withOrderInformation(purchaseOrder, poLine);
return succeededFuture(null);
}).when(basePieceFlowHolderBuilder).updateHolderWithOrderInformation(PieceDeletionHolderCapture.capture(), eq(requestContext));
doAnswer((Answer<Future<Void>>) invocation -> {
PieceDeletionHolder answerHolder = invocation.getArgument(0);
answerHolder.withTitleInformation(title);
return succeededFuture(null);
}).when(basePieceFlowHolderBuilder).updateHolderWithTitleInformation(PieceDeletionHolderCapture.capture(), eq(requestContext));

final ArgumentCaptor<PieceDeletionHolder> pieceDeletionHolderCapture = ArgumentCaptor.forClass(PieceDeletionHolder.class);
doReturn(succeededFuture(null)).when(pieceDeleteFlowPoLineService).updatePoLine(pieceDeletionHolderCapture.capture(), eq(requestContext));
//When
pieceDeleteFlowManager.batchDeletePiece(ids, requestContext).result();
verify(pieceStorageService).deletePiece(eq(piece.getId()), eq(true), eq(requestContext));
verify(inventoryItemManager, times(0)).deleteItem(itemId, true, requestContext);
verify(pieceStorageService, times(1)).deletePiece(eq(piece.getId()), eq(true), eq(requestContext));
verify(pieceDeleteFlowPoLineService).updatePoLine(pieceDeletionHolderCapture.capture(), eq(requestContext));

}

private static class ContextConfiguration {
@Bean PieceStorageService pieceStorageService() {
return mock(PieceStorageService.class);
Expand Down
Loading