blob: ef594d394df7f8cfb2138ab8f1fe1f4da34fe0db [file] [log] [blame]
/*
* 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.cn.individuallending.internal.service.costcomponent;
import org.apache.fineract.cn.individuallending.api.v1.domain.product.AccountDesignators;
import org.apache.fineract.cn.individuallending.api.v1.domain.workflow.Action;
import org.apache.fineract.cn.individuallending.internal.service.DataContextOfAction;
import org.apache.fineract.cn.individuallending.internal.service.DesignatorToAccountIdentifierMapper;
import org.apache.fineract.cn.portfolio.api.v1.domain.ChargeDefinition;
import org.apache.fineract.cn.portfolio.service.internal.util.AccountingAdapter;
import java.math.BigDecimal;
import java.time.LocalDate;
import java.time.LocalDateTime;
import java.util.Optional;
import java.util.concurrent.TimeUnit;
import net.jodah.expiringmap.ExpirationPolicy;
import net.jodah.expiringmap.ExpiringMap;
import org.apache.fineract.cn.accounting.api.v1.domain.Account;
import org.apache.fineract.cn.accounting.api.v1.domain.AccountType;
/**
* @author Myrle Krantz
*/
public class RealRunningBalances implements RunningBalances {
private final AccountingAdapter accountingAdapter;
private final DesignatorToAccountIdentifierMapper designatorToAccountIdentifierMapper;
private final DataContextOfAction dataContextOfAction;
private final ExpiringMap<String, Optional<Account>> accountCache;
@SuppressWarnings("OptionalUsedAsFieldOrParameterType")
private Optional<LocalDateTime> startOfTerm;
public RealRunningBalances(
final AccountingAdapter accountingAdapter,
final DataContextOfAction dataContextOfAction) {
this.accountingAdapter = accountingAdapter;
this.designatorToAccountIdentifierMapper =
new DesignatorToAccountIdentifierMapper(dataContextOfAction);
this.dataContextOfAction = dataContextOfAction;
this.accountCache = ExpiringMap.builder()
.maxSize(40)
.expirationPolicy(ExpirationPolicy.CREATED)
.expiration(60,TimeUnit.SECONDS)
.entryLoader((String accountDesignator) -> {
final Optional<String> accountIdentifier;
if (accountDesignator.equals(AccountDesignators.ENTRY) || accountDesignator.equals(AccountDesignators.EXPENSE)) {
accountIdentifier = designatorToAccountIdentifierMapper.map(accountDesignator);
}
else {
accountIdentifier = Optional.of(designatorToAccountIdentifierMapper.mapOrThrow(accountDesignator));
}
return accountIdentifier.map(accountingAdapter::getAccount);
})
.build();
this.startOfTerm = Optional.empty();
}
@Override
public BigDecimal getAccountSign(final String accountDesignator) {
return accountCache.get(accountDesignator)
.map(Account::getType)
.map(AccountType::valueOf)
.flatMap(x -> {
switch (x)
{
case LIABILITY:
case REVENUE:
case EQUITY:
return Optional.of(POSITIVE);
default:
case ASSET:
case EXPENSE:
return Optional.of(NEGATIVE);
}
})
.orElseGet(() -> {
switch (accountDesignator) {
case AccountDesignators.EXPENSE:
return NEGATIVE;
case AccountDesignators.ENTRY:
return POSITIVE;
default:
return NEGATIVE;
}}
);
}
@Override
public Optional<BigDecimal> getAccountBalance(final String accountDesignator) {
return accountCache.get(accountDesignator).map(Account::getBalance).map(BigDecimal::valueOf);
}
@Override
public BigDecimal getAccruedBalanceForCharge(final ChargeDefinition chargeDefinition) {
final String accrualAccountIdentifier = designatorToAccountIdentifierMapper.mapOrThrow(chargeDefinition.getAccrualAccountDesignator());
final LocalDate startOfTermLocalDate = getStartOfTermOrThrow(dataContextOfAction).toLocalDate();
final BigDecimal amountAccrued = accountingAdapter.sumMatchingEntriesSinceDate(
accrualAccountIdentifier,
startOfTermLocalDate,
dataContextOfAction.getMessageForCharge(Action.valueOf(chargeDefinition.getAccrueAction())));
final BigDecimal amountApplied = accountingAdapter.sumMatchingEntriesSinceDate(
accrualAccountIdentifier,
startOfTermLocalDate,
dataContextOfAction.getMessageForCharge(Action.valueOf(chargeDefinition.getChargeAction())));
return amountAccrued.subtract(amountApplied);
}
@Override
public Optional<LocalDateTime> getStartOfTerm() {
if (!startOfTerm.isPresent()) {
final LocalDateTime persistedStartOfTerm = dataContextOfAction.getCustomerCaseEntity().getStartOfTerm();
if (persistedStartOfTerm != null) {
this.startOfTerm = Optional.of(persistedStartOfTerm);
return this.startOfTerm;
}
final String customerLoanPrincipalAccountIdentifier = designatorToAccountIdentifierMapper.mapOrThrow(AccountDesignators.CUSTOMER_LOAN_PRINCIPAL);
this.startOfTerm = accountingAdapter.getDateOfOldestEntryContainingMessage(
customerLoanPrincipalAccountIdentifier,
dataContextOfAction.getMessageForCharge(Action.DISBURSE));
//Moving start of term persistence from accounting to the portfolio db. Opportunistic migration only right now.
startOfTerm.ifPresent(startOfTermPersistedInAccounting ->
dataContextOfAction.getCustomerCaseEntity().setStartOfTerm(startOfTermPersistedInAccounting));
}
return this.startOfTerm;
}
public BigDecimal getSumOfChargesForActionSinceDate(
final String accountDesignator,
final Action action,
final LocalDateTime since) {
final String accountIdentifier = designatorToAccountIdentifierMapper.mapOrThrow(accountDesignator);
return accountingAdapter.sumMatchingEntriesSinceDate(
accountIdentifier,
since.toLocalDate(),
dataContextOfAction.getMessageForCharge(action));
}
}