Skip to content

Commit

Permalink
FINERACT-1981: Function Value using simple interest for EMI calculation
Browse files Browse the repository at this point in the history
  • Loading branch information
Jose Alberto Hernandez authored and adamsaghy committed May 29, 2024
1 parent 2092a9a commit adcfc3c
Show file tree
Hide file tree
Showing 3 changed files with 76 additions and 3 deletions.
Original file line number Diff line number Diff line change
@@ -0,0 +1,44 @@
/**
* Licensed to the Apache Software Foundation (ASF) under one
* or more contributor license agreements. See the NOTICE file
* distributed with this work for additional information
* regarding copyright ownership. The ASF licenses this file
* to you under the Apache License, Version 2.0 (the
* "License"); you may not use this file except in compliance
* with the License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing,
* software distributed under the License is distributed on an
* "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
* KIND, either express or implied. See the License for the
* specific language governing permissions and limitations
* under the License.
*/
package org.apache.fineract.portfolio.loanproduct.calc.emi;

import java.math.BigDecimal;
import java.math.MathContext;

public class FnValueFunctions {

protected FnValueFunctions() {}

/**
* To calculate the function value for each period, we are going to use the next formula:
*
* fn = 1 + fnValueFrom * rateFactorEnd
*
* @param previousFnValue
*
* @param currentRateFactor
*
* @param mathContext
*
*/
public static BigDecimal fnValue(final BigDecimal previousFnValue, final BigDecimal currentRateFactor, final MathContext mc) {
return BigDecimal.ONE.add(previousFnValue.multiply(currentRateFactor, mc));
}

}
Original file line number Diff line number Diff line change
Expand Up @@ -16,7 +16,7 @@
* specific language governing permissions and limitations
* under the License.
*/
package org.apache.fineract.portfolio.loanproduct.ratefactor;
package org.apache.fineract.portfolio.loanproduct.calc.ratefactor;

import java.math.BigDecimal;
import java.math.MathContext;
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -16,7 +16,7 @@
* specific language governing permissions and limitations
* under the License.
*/
package org.apache.fineract.portfolio.loanproduct.ratefactor;
package org.apache.fineract.portfolio.loanproduct.calc;

import java.math.BigDecimal;
import java.math.MathContext;
Expand All @@ -28,6 +28,8 @@
import org.apache.fineract.organisation.monetary.domain.MoneyHelper;
import org.apache.fineract.portfolio.common.domain.DaysInYearType;
import org.apache.fineract.portfolio.loanaccount.domain.LoanRepaymentScheduleInstallment;
import org.apache.fineract.portfolio.loanproduct.calc.emi.FnValueFunctions;
import org.apache.fineract.portfolio.loanproduct.calc.ratefactor.RateFactorFunctions;
import org.jetbrains.annotations.NotNull;
import org.junit.jupiter.api.Assertions;
import org.junit.jupiter.api.BeforeAll;
Expand All @@ -38,7 +40,7 @@
import org.mockito.junit.jupiter.MockitoExtension;

@ExtendWith(MockitoExtension.class)
class RateFactorFunctionsTest {
class EMICalculationFunctionsTest {

private static MockedStatic<MoneyHelper> moneyHelper = Mockito.mockStatic(MoneyHelper.class);

Expand Down Expand Up @@ -94,6 +96,33 @@ public void testRateFactorFunctionActual() {
}
}

@Test
public void testFnValueFunctionDay365() {
// Given
final DaysInYearType daysInYearType = DaysInYearType.DAYS_365;
final MathContext mc = MoneyHelper.getMathContext();
final String[] expectedValues = new String[] { "1.0000000", "2.0075336", "3.0237007", "4.0472656", "5.0798590", "6.1194484" };

final List<BigDecimal> fnValuesCalculated = new ArrayList<>();
BigDecimal previousFnValue = BigDecimal.ZERO;
for (LoanRepaymentScheduleInstallment period : periods) {
final Long daysInPeriod = DateUtils.getDifferenceInDays(period.getFromDate(), period.getDueDate());
final Integer daysInYear = DateUtils.daysInYear(daysInYearType, period.getFromDate());
final BigDecimal rateFactor = RateFactorFunctions.rateFactor(interestRate, daysInPeriod, daysInYear,
MoneyHelper.getMathContext());

final BigDecimal currentFnValue = FnValueFunctions.fnValue(previousFnValue, rateFactor, mc);
fnValuesCalculated.add(currentFnValue);

previousFnValue = currentFnValue;
}

int idx = 0;
for (BigDecimal fnValue : fnValuesCalculated) {
Assertions.assertEquals(expectedValues[idx++], fnValue.toString());
}
}

@NotNull
private static LoanRepaymentScheduleInstallment createPeriod(int periodId, LocalDate start, LocalDate end) {
LoanRepaymentScheduleInstallment period = Mockito.mock(LoanRepaymentScheduleInstallment.class);
Expand Down

0 comments on commit adcfc3c

Please sign in to comment.