|
22 | 22 | import static org.junit.jupiter.api.Assertions.assertNotNull; |
23 | 23 | import static org.mockito.ArgumentMatchers.any; |
24 | 24 | import static org.mockito.ArgumentMatchers.anyIterable; |
| 25 | +import static org.mockito.ArgumentMatchers.anyList; |
25 | 26 | import static org.mockito.ArgumentMatchers.anyLong; |
26 | 27 | import static org.mockito.Mockito.times; |
27 | 28 | import static org.mockito.Mockito.verify; |
|
30 | 31 | import java.math.BigDecimal; |
31 | 32 | import java.time.LocalDate; |
32 | 33 | import java.time.ZoneId; |
| 34 | +import java.util.ArrayList; |
33 | 35 | import java.util.Arrays; |
34 | 36 | import java.util.Collections; |
35 | 37 | import java.util.HashMap; |
|
38 | 40 | import java.util.Map; |
39 | 41 | import java.util.Optional; |
40 | 42 | import org.apache.fineract.infrastructure.businessdate.domain.BusinessDateType; |
| 43 | +import org.apache.fineract.infrastructure.core.api.JsonCommand; |
41 | 44 | import org.apache.fineract.infrastructure.core.domain.ActionContext; |
42 | 45 | import org.apache.fineract.infrastructure.core.domain.FineractPlatformTenant; |
43 | 46 | import org.apache.fineract.infrastructure.core.service.DateUtils; |
44 | 47 | import org.apache.fineract.infrastructure.core.service.ThreadLocalContextUtil; |
| 48 | +import org.apache.fineract.infrastructure.event.business.domain.loan.LoanAccountDelinquencyPauseChangedBusinessEvent; |
45 | 49 | import org.apache.fineract.infrastructure.event.business.domain.loan.LoanDelinquencyRangeChangeBusinessEvent; |
46 | 50 | import org.apache.fineract.infrastructure.event.business.service.BusinessEventNotifierService; |
| 51 | +import org.apache.fineract.portfolio.delinquency.domain.DelinquencyAction; |
47 | 52 | import org.apache.fineract.portfolio.delinquency.domain.DelinquencyBucket; |
48 | 53 | import org.apache.fineract.portfolio.delinquency.domain.DelinquencyBucketMappingsRepository; |
49 | 54 | import org.apache.fineract.portfolio.delinquency.domain.DelinquencyBucketRepository; |
50 | 55 | import org.apache.fineract.portfolio.delinquency.domain.DelinquencyRange; |
51 | 56 | import org.apache.fineract.portfolio.delinquency.domain.DelinquencyRangeRepository; |
| 57 | +import org.apache.fineract.portfolio.delinquency.domain.LoanDelinquencyAction; |
| 58 | +import org.apache.fineract.portfolio.delinquency.domain.LoanDelinquencyActionRepository; |
52 | 59 | import org.apache.fineract.portfolio.delinquency.domain.LoanDelinquencyTagHistory; |
53 | 60 | import org.apache.fineract.portfolio.delinquency.domain.LoanDelinquencyTagHistoryRepository; |
54 | 61 | import org.apache.fineract.portfolio.delinquency.domain.LoanInstallmentDelinquencyTag; |
55 | 62 | import org.apache.fineract.portfolio.delinquency.domain.LoanInstallmentDelinquencyTagRepository; |
| 63 | +import org.apache.fineract.portfolio.delinquency.helper.DelinquencyEffectivePauseHelper; |
| 64 | +import org.apache.fineract.portfolio.delinquency.service.DelinquencyReadPlatformService; |
56 | 65 | import org.apache.fineract.portfolio.delinquency.service.DelinquencyWritePlatformServiceImpl; |
57 | 66 | import org.apache.fineract.portfolio.delinquency.service.LoanDelinquencyDomainService; |
| 67 | +import org.apache.fineract.portfolio.delinquency.validator.DelinquencyActionParseAndValidator; |
58 | 68 | import org.apache.fineract.portfolio.delinquency.validator.DelinquencyBucketParseAndValidator; |
59 | 69 | import org.apache.fineract.portfolio.delinquency.validator.DelinquencyRangeParseAndValidator; |
60 | 70 | import org.apache.fineract.portfolio.delinquency.validator.LoanDelinquencyActionData; |
@@ -100,6 +110,14 @@ public class DelinquencyWritePlatformServiceRangeChangeEventTest { |
100 | 110 | private LoanDelinquencyDomainService loanDelinquencyDomainService; |
101 | 111 | @Mock |
102 | 112 | private LoanInstallmentDelinquencyTagRepository loanInstallmentDelinquencyTagRepository; |
| 113 | + @Mock |
| 114 | + private DelinquencyReadPlatformService delinquencyReadPlatformService; |
| 115 | + @Mock |
| 116 | + private DelinquencyActionParseAndValidator delinquencyActionParseAndValidator; |
| 117 | + @Mock |
| 118 | + private LoanDelinquencyActionRepository loanDelinquencyActionRepository; |
| 119 | + @Mock |
| 120 | + private DelinquencyEffectivePauseHelper delinquencyEffectivePauseHelper; |
103 | 121 |
|
104 | 122 | @InjectMocks |
105 | 123 | private DelinquencyWritePlatformServiceImpl underTest; |
@@ -498,4 +516,103 @@ public void givenLoanAccountEnableInstallmentLevelDelinquencyWhenLoanIsOutOfDeli |
498 | 516 | assertEquals(loanForProcessing, loanPayloadForEvent); |
499 | 517 |
|
500 | 518 | } |
| 519 | + |
| 520 | + @Test |
| 521 | + public void givenLoanAccountWhenBackdatedPauseActionThenLoanDelinquencyPauseChangeBusinessEventIsRaisedTest() { |
| 522 | + ArgumentCaptor<LoanAccountDelinquencyPauseChangedBusinessEvent> loanDelinquencyPauseChangeEvent = ArgumentCaptor |
| 523 | + .forClass(LoanAccountDelinquencyPauseChangedBusinessEvent.class); |
| 524 | + // given |
| 525 | + Loan loanForProcessing = Mockito.mock(Loan.class); |
| 526 | + loanForProcessing.setId(1L); |
| 527 | + |
| 528 | + JsonCommand command = Mockito.mock(JsonCommand.class); |
| 529 | + |
| 530 | + // Pause period |
| 531 | + LocalDate startDate = DateUtils.getBusinessLocalDate().minusDays(8); |
| 532 | + LocalDate endDate = DateUtils.getBusinessLocalDate().minusDays(1); |
| 533 | + |
| 534 | + List<LoanDelinquencyAction> delinquencyActions = new ArrayList<>(); |
| 535 | + List<LoanDelinquencyActionData> effectiveDelinquency = new ArrayList<>(); |
| 536 | + CollectionData loanCollectionData = CollectionData.template(); |
| 537 | + |
| 538 | + when(loanRepository.findOneWithNotFoundDetection(anyLong())).thenReturn(loanForProcessing); |
| 539 | + |
| 540 | + when(delinquencyReadPlatformService.retrieveLoanDelinquencyActions(anyLong())).thenReturn(delinquencyActions); |
| 541 | + LoanDelinquencyAction backdatedPauseAction = Mockito.mock(LoanDelinquencyAction.class); |
| 542 | + backdatedPauseAction.setId(1L); |
| 543 | + |
| 544 | + when(delinquencyActionParseAndValidator.validateAndParseUpdate(command, loanForProcessing, delinquencyActions, |
| 545 | + DateUtils.getBusinessLocalDate())).thenReturn(backdatedPauseAction); |
| 546 | + when(backdatedPauseAction.getStartDate()).thenReturn(startDate); |
| 547 | + when(backdatedPauseAction.getEndDate()).thenReturn(endDate); |
| 548 | + when(backdatedPauseAction.getAction()).thenReturn(DelinquencyAction.PAUSE); |
| 549 | + |
| 550 | + when(loanDelinquencyActionRepository.saveAndFlush(backdatedPauseAction)).thenReturn(backdatedPauseAction); |
| 551 | + |
| 552 | + when(delinquencyEffectivePauseHelper.calculateEffectiveDelinquencyList(delinquencyActions)).thenReturn(effectiveDelinquency); |
| 553 | + when(loanDelinquencyDomainService.getOverdueCollectionData(any(), anyList())).thenReturn(loanCollectionData); |
| 554 | + when(loanDelinquencyTagRepository.findByLoanAndLiftedOnDate(any(), any())).thenReturn(Optional.empty()); |
| 555 | + |
| 556 | + // when |
| 557 | + underTest.createDelinquencyAction(loanForProcessing.getId(), command); |
| 558 | + |
| 559 | + // then |
| 560 | + // verify event is raised |
| 561 | + verify(businessEventNotifierService, times(1)).notifyPostBusinessEvent(loanDelinquencyPauseChangeEvent.capture()); |
| 562 | + |
| 563 | + Loan loanPayloadForEvent = loanDelinquencyPauseChangeEvent.getValue().get(); |
| 564 | + assertEquals(loanForProcessing, loanPayloadForEvent); |
| 565 | + |
| 566 | + // verify no range change event for pause flag change as both start and end date are backdated |
| 567 | + verify(businessEventNotifierService, times(0)).notifyPostBusinessEvent(any(LoanDelinquencyRangeChangeBusinessEvent.class)); |
| 568 | + |
| 569 | + } |
| 570 | + |
| 571 | + @Test |
| 572 | + public void givenLoanAccountWhenBackdatedPauseActionThenLoanDelinquencyRangeChangeBusinessEventIsRaisedIfPauseFlagChangeTest() { |
| 573 | + ArgumentCaptor<LoanDelinquencyRangeChangeBusinessEvent> loanDelinquencyRangeChangeEvent = ArgumentCaptor |
| 574 | + .forClass(LoanDelinquencyRangeChangeBusinessEvent.class); |
| 575 | + // given |
| 576 | + Loan loanForProcessing = Mockito.mock(Loan.class); |
| 577 | + loanForProcessing.setId(1L); |
| 578 | + |
| 579 | + JsonCommand command = Mockito.mock(JsonCommand.class); |
| 580 | + |
| 581 | + // Pause period |
| 582 | + LocalDate startDate = DateUtils.getBusinessLocalDate().minusDays(2); |
| 583 | + LocalDate endDate = DateUtils.getBusinessLocalDate().plusDays(10); |
| 584 | + |
| 585 | + List<LoanDelinquencyAction> delinquencyActions = new ArrayList<>(); |
| 586 | + List<LoanDelinquencyActionData> effectiveDelinquency = new ArrayList<>(); |
| 587 | + CollectionData loanCollectionData = CollectionData.template(); |
| 588 | + |
| 589 | + when(loanRepository.findOneWithNotFoundDetection(anyLong())).thenReturn(loanForProcessing); |
| 590 | + |
| 591 | + when(delinquencyReadPlatformService.retrieveLoanDelinquencyActions(anyLong())).thenReturn(delinquencyActions); |
| 592 | + LoanDelinquencyAction backdatedPauseAction = Mockito.mock(LoanDelinquencyAction.class); |
| 593 | + backdatedPauseAction.setId(1L); |
| 594 | + |
| 595 | + when(delinquencyActionParseAndValidator.validateAndParseUpdate(command, loanForProcessing, delinquencyActions, |
| 596 | + DateUtils.getBusinessLocalDate())).thenReturn(backdatedPauseAction); |
| 597 | + when(backdatedPauseAction.getStartDate()).thenReturn(startDate); |
| 598 | + when(backdatedPauseAction.getEndDate()).thenReturn(endDate); |
| 599 | + when(backdatedPauseAction.getAction()).thenReturn(DelinquencyAction.PAUSE); |
| 600 | + |
| 601 | + when(loanDelinquencyActionRepository.saveAndFlush(backdatedPauseAction)).thenReturn(backdatedPauseAction); |
| 602 | + |
| 603 | + when(delinquencyEffectivePauseHelper.calculateEffectiveDelinquencyList(delinquencyActions)).thenReturn(effectiveDelinquency); |
| 604 | + when(loanDelinquencyDomainService.getOverdueCollectionData(any(), anyList())).thenReturn(loanCollectionData); |
| 605 | + when(loanDelinquencyTagRepository.findByLoanAndLiftedOnDate(any(), any())).thenReturn(Optional.empty()); |
| 606 | + |
| 607 | + // when |
| 608 | + underTest.createDelinquencyAction(loanForProcessing.getId(), command); |
| 609 | + |
| 610 | + // then |
| 611 | + // verify event is raised |
| 612 | + verify(businessEventNotifierService, times(1)).notifyPostBusinessEvent(loanDelinquencyRangeChangeEvent.capture()); |
| 613 | + |
| 614 | + Loan loanPayloadForEvent = loanDelinquencyRangeChangeEvent.getValue().get(); |
| 615 | + assertEquals(loanForProcessing, loanPayloadForEvent); |
| 616 | + } |
| 617 | + |
501 | 618 | } |
0 commit comments