Skip to content

Commit

Permalink
FINERACT-1971: external_id column of accrual activity transaction get…
Browse files Browse the repository at this point in the history
…s updated to null when an accrual activity is reversed
  • Loading branch information
kulminsky committed Jan 8, 2025
1 parent e1d043b commit 881523a
Show file tree
Hide file tree
Showing 3 changed files with 145 additions and 2 deletions.
Original file line number Diff line number Diff line change
Expand Up @@ -458,6 +458,16 @@ public void reverse(final ExternalId reversalExternalId) {
this.reversalExternalId = reversalExternalId;
}

public void reverseWithExternalIdHandling(final boolean isReverseReplayed) {
this.reversed = true;
this.reversedOnDate = DateUtils.getBusinessLocalDate();
this.loanTransactionToRepaymentScheduleMappings.clear();

if (isReverseReplayed) {
this.externalId = null;
}
}

public void resetDerivedComponents() {
this.principalPortion = null;
this.interestPortion = null;
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -780,7 +780,6 @@ private static LoanTransaction useOldTransactionIfApplicable(LoanTransaction old

protected void createNewTransaction(final LoanTransaction oldTransaction, final LoanTransaction newTransaction,
final TransactionCtx ctx) {
oldTransaction.updateExternalId(null);
oldTransaction.getLoanChargesPaid().clear();

if (newTransaction.getTypeOf().isInterestRefund()) {
Expand Down Expand Up @@ -809,7 +808,11 @@ protected void createNewTransaction(final LoanTransaction oldTransaction, final
relations.add(LoanTransactionRelation.linkToTransaction(originalTransaction, newTransaction, CHARGEBACK));
}
loanChargeValidator.validateRepaymentTypeTransactionNotBeforeAChargeRefund(oldTransaction.getLoan(), oldTransaction, "reversed");
oldTransaction.reverse();

boolean isReverseReplayed = oldTransaction.getLoanTransactionRelations().stream()
.anyMatch(relation -> LoanTransactionRelationTypeEnum.REPLAYED.equals(relation.getRelationType()));

oldTransaction.reverseWithExternalIdHandling(isReverseReplayed);
}

private void processSingleCharge(LoanCharge loanCharge, MonetaryCurrency currency, List<LoanRepaymentScheduleInstallment> installments,
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,130 @@
package org.apache.fineract.integrationtests;

import static org.junit.jupiter.api.Assertions.assertEquals;
import static org.junit.jupiter.api.Assertions.assertFalse;
import static org.junit.jupiter.api.Assertions.assertNotNull;
import static org.junit.jupiter.api.Assertions.assertNull;
import static org.junit.jupiter.api.Assertions.assertTrue;

import io.restassured.builder.RequestSpecBuilder;
import io.restassured.builder.ResponseSpecBuilder;
import io.restassured.http.ContentType;
import io.restassured.specification.RequestSpecification;
import io.restassured.specification.ResponseSpecification;
import java.math.BigDecimal;
import java.time.LocalDate;
import java.util.HashMap;
import java.util.UUID;
import org.apache.fineract.client.models.GetLoanProductsProductIdResponse;
import org.apache.fineract.client.models.GetLoansLoanIdResponse;
import org.apache.fineract.client.models.PostLoansLoanIdRequest;
import org.apache.fineract.client.models.PostLoansLoanIdTransactionsRequest;
import org.apache.fineract.client.models.PostLoansLoanIdTransactionsResponse;
import org.apache.fineract.client.models.PutGlobalConfigurationsRequest;
import org.apache.fineract.infrastructure.businessdate.domain.BusinessDateType;
import org.apache.fineract.infrastructure.configuration.api.GlobalConfigurationConstants;
import org.apache.fineract.integrationtests.common.BusinessDateHelper;
import org.apache.fineract.integrationtests.common.ClientHelper;
import org.apache.fineract.integrationtests.common.Utils;
import org.apache.fineract.integrationtests.common.loans.LoanApplicationTestBuilder;
import org.apache.fineract.integrationtests.common.loans.LoanProductTestBuilder;
import org.apache.fineract.integrationtests.common.loans.LoanTransactionHelper;
import org.junit.jupiter.api.BeforeEach;
import org.junit.jupiter.api.Test;

public class RepaymentReverseExternalIdTest extends BaseLoanIntegrationTest {

private static final String loanAmount = "1000";
private static final String startDate = "20 December 2024";
private static final String firstRepaymentDate = "23 December 2024";
private static final String secondRepaymentDate = "26 December 2024";
private static final String reverseDate = "27 December 2024";
private static final Double firstRepaymentAmount = 1000.0;
private static final Double secondRepaymentAmount = 10.0;

private ResponseSpecification responseSpec;
private RequestSpecification requestSpec;
private ClientHelper clientHelper;
private LoanTransactionHelper loanTransactionHelper;

@BeforeEach
public void setup() {
Utils.initializeRESTAssured();
this.requestSpec = new RequestSpecBuilder().setContentType(ContentType.JSON).build();
this.requestSpec.header("Authorization", "Basic " + Utils.loginIntoServerAndGetBase64EncodedAuthenticationKey());
this.responseSpec = new ResponseSpecBuilder().expectStatusCode(200).build();
this.loanTransactionHelper = new LoanTransactionHelper(this.requestSpec, this.responseSpec);
this.clientHelper = new ClientHelper(this.requestSpec, this.responseSpec);
}

@Test
public void testReverseRepaymentUpdatesExternalIdCorrectlyForOverpayment() {
try {
// Set up the business date if required
final LocalDate todaysDate = Utils.getLocalDateOfTenant();
globalConfigurationHelper.updateGlobalConfiguration(GlobalConfigurationConstants.ENABLE_BUSINESS_DATE,
new PutGlobalConfigurationsRequest().enabled(true));
BusinessDateHelper.updateBusinessDate(requestSpec, responseSpec, BusinessDateType.BUSINESS_DATE, todaysDate);

// Create a client and a loan with externalId
final Integer clientId = clientHelper.createClient(ClientHelper.defaultClientCreationRequest()).getClientId().intValue();
final GetLoanProductsProductIdResponse loanProduct = createLoanProduct(loanTransactionHelper, null);
final String loanExternalId = UUID.randomUUID().toString();
final Integer loanId = createLoanAccount(clientId, loanProduct.getId(), loanExternalId);

// First repayment to cover part of the loan
final PostLoansLoanIdTransactionsResponse repaymentTransaction1 = loanTransactionHelper.makeLoanRepayment(loanExternalId,
new PostLoansLoanIdTransactionsRequest().dateFormat("dd MMMM yyyy").transactionDate(firstRepaymentDate).locale("en")
.transactionAmount(firstRepaymentAmount));

// Second repayment to create overpayment
final PostLoansLoanIdTransactionsResponse repaymentTransaction2 = loanTransactionHelper.makeLoanRepayment(loanExternalId,
new PostLoansLoanIdTransactionsRequest().dateFormat("dd MMMM yyyy").transactionDate(secondRepaymentDate).locale("en")
.transactionAmount(secondRepaymentAmount)); // This creates an overpayment as total is now
// 1010

// Verify that the loan is marked as overpaid
GetLoansLoanIdResponse loanDetailsOverpaid = loanTransactionHelper.getLoanDetails((long) loanId);
assertTrue(loanDetailsOverpaid.getStatus().getOverpaid()); // Overpaid status should be true
assertNotNull(loanDetailsOverpaid.getOverpaidOnDate()); // Overpaid date should be set
assertEquals(loanDetailsOverpaid.getExternalId(), loanExternalId); // externalId should match the original

// Reverse the second repayment to remove the overpayment
loanTransactionHelper.reverseRepayment(loanId, repaymentTransaction2.getResourceId().intValue(), reverseDate);

// Verify that the loan is no longer overpaid and overpaid date is reset
GetLoansLoanIdResponse loanDetailsNotOverpaid = loanTransactionHelper.getLoanDetails((long) loanId);
assertFalse(loanDetailsNotOverpaid.getStatus().getOverpaid()); // Overpaid status should be false
assertNull(loanDetailsNotOverpaid.getOverpaidOnDate()); // Overpaid date should be reset
assertEquals(loanDetailsNotOverpaid.getExternalId(), loanExternalId); // externalId should remain unchanged

} finally {
// Disable business date configuration if it was enabled
globalConfigurationHelper.updateGlobalConfiguration(GlobalConfigurationConstants.ENABLE_BUSINESS_DATE,
new PutGlobalConfigurationsRequest().enabled(false));
}
}

private GetLoanProductsProductIdResponse createLoanProduct(final LoanTransactionHelper loanTransactionHelper,
final Integer delinquencyBucketId) {
final HashMap<String, Object> loanProductMap = new LoanProductTestBuilder().build(null, delinquencyBucketId);
final Integer loanProductId = loanTransactionHelper.getLoanProductId(Utils.convertToJson(loanProductMap));
return loanTransactionHelper.getLoanProduct(loanProductId);
}

private Integer createLoanAccount(final Integer clientID, final Long loanProductID, final String externalId) {

String loanApplicationJSON = new LoanApplicationTestBuilder().withPrincipal(loanAmount).withLoanTermFrequency("1")
.withLoanTermFrequencyAsMonths().withNumberOfRepayments("1").withRepaymentEveryAfter("1")
.withRepaymentFrequencyTypeAsMonths().withInterestRatePerPeriod("0").withInterestTypeAsFlatBalance()
.withAmortizationTypeAsEqualPrincipalPayments().withInterestCalculationPeriodTypeSameAsRepaymentPeriod()
.withExpectedDisbursementDate(startDate).withSubmittedOnDate(startDate).withLoanType("individual")
.withExternalId(externalId).build(clientID.toString(), loanProductID.toString(), null);

final Integer loanId = loanTransactionHelper.getLoanId(loanApplicationJSON);
loanTransactionHelper.approveLoan(startDate, loanAmount, loanId, null);
loanTransactionHelper.disburseLoan(Long.valueOf(loanId), new PostLoansLoanIdRequest().actualDisbursementDate(startDate)
.transactionAmount(new BigDecimal(loanAmount)).locale("en").dateFormat("dd MMMM yyyy"));
return loanId;
}
}

0 comments on commit 881523a

Please sign in to comment.