This is an automated email from the ASF dual-hosted git repository.

adamsaghy pushed a commit to branch develop
in repository https://gitbox.apache.org/repos/asf/fineract.git


The following commit(s) were added to refs/heads/develop by this push:
     new 221f80699f FINERACT-2418: fetch loan originators by loan
221f80699f is described below

commit 221f80699fbb54eabff46361615a28a5787f5d08
Author: Attila Budai <[email protected]>
AuthorDate: Sun Jan 25 16:55:47 2026 +0100

    FINERACT-2418: fetch loan originators by loan
---
 fineract-loan-origination/dependencies.gradle      |  1 +
 .../api/LoanOriginatorsApiResource.java            | 93 ++++++++++++++++++++++
 .../LoanOriginatorsResponse.java}                  | 24 ++++--
 .../domain/LoanOriginatorMappingRepository.java    |  9 +++
 .../service/LoanOriginatorReadPlatformService.java |  2 +
 .../LoanOriginatorReadPlatformServiceImpl.java     | 13 +++
 6 files changed, 134 insertions(+), 8 deletions(-)

diff --git a/fineract-loan-origination/dependencies.gradle 
b/fineract-loan-origination/dependencies.gradle
index bd2a15867e..dee9b9db49 100644
--- a/fineract-loan-origination/dependencies.gradle
+++ b/fineract-loan-origination/dependencies.gradle
@@ -25,6 +25,7 @@ dependencies {
     // implementation dependencies are directly used (compiled against) in 
src/main (and src/test)
     //
     implementation(project(path: ':fineract-core'))
+    implementation(project(path: ':fineract-loan'))
 
     implementation(
             'org.springframework.boot:spring-boot-starter-web',
diff --git 
a/fineract-loan-origination/src/main/java/org/apache/fineract/portfolio/loanorigination/api/LoanOriginatorsApiResource.java
 
b/fineract-loan-origination/src/main/java/org/apache/fineract/portfolio/loanorigination/api/LoanOriginatorsApiResource.java
new file mode 100644
index 0000000000..4169e60d0c
--- /dev/null
+++ 
b/fineract-loan-origination/src/main/java/org/apache/fineract/portfolio/loanorigination/api/LoanOriginatorsApiResource.java
@@ -0,0 +1,93 @@
+/**
+ * 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.loanorigination.api;
+
+import io.swagger.v3.oas.annotations.Operation;
+import io.swagger.v3.oas.annotations.Parameter;
+import io.swagger.v3.oas.annotations.responses.ApiResponse;
+import io.swagger.v3.oas.annotations.tags.Tag;
+import jakarta.ws.rs.Consumes;
+import jakarta.ws.rs.GET;
+import jakarta.ws.rs.Path;
+import jakarta.ws.rs.PathParam;
+import jakarta.ws.rs.Produces;
+import jakarta.ws.rs.core.MediaType;
+import lombok.RequiredArgsConstructor;
+import org.apache.fineract.infrastructure.core.domain.ExternalId;
+import org.apache.fineract.infrastructure.core.service.ExternalIdFactory;
+import 
org.apache.fineract.infrastructure.security.service.PlatformSecurityContext;
+import 
org.apache.fineract.portfolio.loanaccount.exception.LoanNotFoundException;
+import 
org.apache.fineract.portfolio.loanaccount.service.LoanReadPlatformService;
+import 
org.apache.fineract.portfolio.loanorigination.data.LoanOriginatorsResponse;
+import 
org.apache.fineract.portfolio.loanorigination.service.LoanOriginatorReadPlatformService;
+import org.springframework.boot.autoconfigure.condition.ConditionalOnProperty;
+import org.springframework.stereotype.Component;
+
+@Path("/v1/loans")
+@Component
+@ConditionalOnProperty(value = "fineract.module.loan-origination.enabled", 
havingValue = "true")
+@Tag(name = "Loan Originators", description = "Fetch loan originator details 
for a specific loan")
+@RequiredArgsConstructor
+public class LoanOriginatorsApiResource {
+
+    private static final String LOAN_RESOURCE_NAME = "LOAN";
+
+    private final PlatformSecurityContext context;
+    private final LoanReadPlatformService loanReadPlatformService;
+    private final LoanOriginatorReadPlatformService 
loanOriginatorReadPlatformService;
+
+    @GET
+    @Path("{loanId}/originators")
+    @Consumes({ MediaType.APPLICATION_JSON })
+    @Produces({ MediaType.APPLICATION_JSON })
+    @Operation(summary = "Retrieve originators for a loan by loan ID", 
description = "Retrieves all originators attached to a specific loan. Requires 
READ_LOAN permission.")
+    @ApiResponse(responseCode = "200", description = "OK - Returns wrapped 
list of originators (may be empty)")
+    @ApiResponse(responseCode = "403", description = "Insufficient 
permissions")
+    @ApiResponse(responseCode = "404", description = "Loan not found")
+    public LoanOriginatorsResponse 
retrieveOriginatorsByLoanId(@PathParam("loanId") @Parameter(description = 
"loanId") final Long loanId) {
+        
this.context.authenticatedUser().validateHasReadPermission(LOAN_RESOURCE_NAME);
+
+        if (!this.loanReadPlatformService.existsByLoanId(loanId)) {
+            throw new LoanNotFoundException(loanId);
+        }
+
+        return 
LoanOriginatorsResponse.of(this.loanOriginatorReadPlatformService.retrieveByLoanId(loanId));
+    }
+
+    @GET
+    @Path("external-id/{loanExternalId}/originators")
+    @Consumes({ MediaType.APPLICATION_JSON })
+    @Produces({ MediaType.APPLICATION_JSON })
+    @Operation(summary = "Retrieve originators for a loan by loan external 
ID", description = "Retrieves all originators attached to a specific loan using 
loan external ID. Requires READ_LOAN permission.")
+    @ApiResponse(responseCode = "200", description = "OK - Returns wrapped 
list of originators (may be empty)")
+    @ApiResponse(responseCode = "403", description = "Insufficient 
permissions")
+    @ApiResponse(responseCode = "404", description = "Loan not found")
+    public LoanOriginatorsResponse retrieveOriginatorsByLoanExternalId(
+            @PathParam("loanExternalId") @Parameter(description = 
"loanExternalId") final String loanExternalId) {
+        
this.context.authenticatedUser().validateHasReadPermission(LOAN_RESOURCE_NAME);
+
+        final ExternalId externalId = 
ExternalIdFactory.produce(loanExternalId);
+        final Long loanId = 
this.loanReadPlatformService.retrieveLoanIdByExternalId(externalId);
+        if (loanId == null) {
+            throw new LoanNotFoundException(externalId);
+        }
+
+        return 
LoanOriginatorsResponse.of(this.loanOriginatorReadPlatformService.retrieveByLoanId(loanId));
+    }
+}
diff --git 
a/fineract-loan-origination/src/main/java/org/apache/fineract/portfolio/loanorigination/service/LoanOriginatorReadPlatformService.java
 
b/fineract-loan-origination/src/main/java/org/apache/fineract/portfolio/loanorigination/data/LoanOriginatorsResponse.java
similarity index 59%
copy from 
fineract-loan-origination/src/main/java/org/apache/fineract/portfolio/loanorigination/service/LoanOriginatorReadPlatformService.java
copy to 
fineract-loan-origination/src/main/java/org/apache/fineract/portfolio/loanorigination/data/LoanOriginatorsResponse.java
index e7a169bf85..9f0f1b2812 100644
--- 
a/fineract-loan-origination/src/main/java/org/apache/fineract/portfolio/loanorigination/service/LoanOriginatorReadPlatformService.java
+++ 
b/fineract-loan-origination/src/main/java/org/apache/fineract/portfolio/loanorigination/data/LoanOriginatorsResponse.java
@@ -16,18 +16,26 @@
  * specific language governing permissions and limitations
  * under the License.
  */
-package org.apache.fineract.portfolio.loanorigination.service;
+package org.apache.fineract.portfolio.loanorigination.data;
 
+import java.io.Serial;
+import java.io.Serializable;
 import java.util.List;
-import org.apache.fineract.portfolio.loanorigination.data.LoanOriginatorData;
+import lombok.AllArgsConstructor;
+import lombok.Data;
+import lombok.NoArgsConstructor;
 
-public interface LoanOriginatorReadPlatformService {
+@Data
+@NoArgsConstructor
+@AllArgsConstructor
+public class LoanOriginatorsResponse implements Serializable {
 
-    List<LoanOriginatorData> retrieveAll();
+    @Serial
+    private static final long serialVersionUID = 1L;
 
-    LoanOriginatorData retrieveById(Long id);
+    private List<LoanOriginatorData> originators;
 
-    LoanOriginatorData retrieveByExternalId(String externalId);
-
-    Long resolveIdByExternalId(String externalId);
+    public static LoanOriginatorsResponse of(List<LoanOriginatorData> 
originators) {
+        return new LoanOriginatorsResponse(originators);
+    }
 }
diff --git 
a/fineract-loan-origination/src/main/java/org/apache/fineract/portfolio/loanorigination/domain/LoanOriginatorMappingRepository.java
 
b/fineract-loan-origination/src/main/java/org/apache/fineract/portfolio/loanorigination/domain/LoanOriginatorMappingRepository.java
index c3e6e1af3b..257029fc12 100644
--- 
a/fineract-loan-origination/src/main/java/org/apache/fineract/portfolio/loanorigination/domain/LoanOriginatorMappingRepository.java
+++ 
b/fineract-loan-origination/src/main/java/org/apache/fineract/portfolio/loanorigination/domain/LoanOriginatorMappingRepository.java
@@ -39,4 +39,13 @@ public interface LoanOriginatorMappingRepository
     boolean existsByLoanIdAndOriginatorId(Long loanId, Long originatorId);
 
     void deleteByLoanIdAndOriginatorId(Long loanId, Long originatorId);
+
+    @org.springframework.data.jpa.repository.Query("""
+            SELECT m FROM LoanOriginatorMapping m
+            JOIN FETCH m.originator o
+            LEFT JOIN FETCH o.originatorType
+            LEFT JOIN FETCH o.channelType
+            WHERE m.loanId = :loanId
+            """)
+    List<LoanOriginatorMapping> 
findByLoanIdWithOriginator(@org.springframework.data.repository.query.Param("loanId")
 Long loanId);
 }
diff --git 
a/fineract-loan-origination/src/main/java/org/apache/fineract/portfolio/loanorigination/service/LoanOriginatorReadPlatformService.java
 
b/fineract-loan-origination/src/main/java/org/apache/fineract/portfolio/loanorigination/service/LoanOriginatorReadPlatformService.java
index e7a169bf85..cace22688e 100644
--- 
a/fineract-loan-origination/src/main/java/org/apache/fineract/portfolio/loanorigination/service/LoanOriginatorReadPlatformService.java
+++ 
b/fineract-loan-origination/src/main/java/org/apache/fineract/portfolio/loanorigination/service/LoanOriginatorReadPlatformService.java
@@ -30,4 +30,6 @@ public interface LoanOriginatorReadPlatformService {
     LoanOriginatorData retrieveByExternalId(String externalId);
 
     Long resolveIdByExternalId(String externalId);
+
+    List<LoanOriginatorData> retrieveByLoanId(Long loanId);
 }
diff --git 
a/fineract-loan-origination/src/main/java/org/apache/fineract/portfolio/loanorigination/service/LoanOriginatorReadPlatformServiceImpl.java
 
b/fineract-loan-origination/src/main/java/org/apache/fineract/portfolio/loanorigination/service/LoanOriginatorReadPlatformServiceImpl.java
index 9913858760..b2dd31ff35 100644
--- 
a/fineract-loan-origination/src/main/java/org/apache/fineract/portfolio/loanorigination/service/LoanOriginatorReadPlatformServiceImpl.java
+++ 
b/fineract-loan-origination/src/main/java/org/apache/fineract/portfolio/loanorigination/service/LoanOriginatorReadPlatformServiceImpl.java
@@ -18,11 +18,14 @@
  */
 package org.apache.fineract.portfolio.loanorigination.service;
 
+import java.util.Collections;
 import java.util.List;
 import lombok.RequiredArgsConstructor;
 import org.apache.fineract.infrastructure.core.domain.ExternalId;
 import org.apache.fineract.portfolio.loanorigination.data.LoanOriginatorData;
 import org.apache.fineract.portfolio.loanorigination.domain.LoanOriginator;
+import 
org.apache.fineract.portfolio.loanorigination.domain.LoanOriginatorMapping;
+import 
org.apache.fineract.portfolio.loanorigination.domain.LoanOriginatorMappingRepository;
 import 
org.apache.fineract.portfolio.loanorigination.domain.LoanOriginatorRepository;
 import 
org.apache.fineract.portfolio.loanorigination.exception.LoanOriginatorNotFoundException;
 import 
org.apache.fineract.portfolio.loanorigination.mapper.LoanOriginatorMapper;
@@ -37,6 +40,7 @@ import 
org.springframework.transaction.annotation.Transactional;
 public class LoanOriginatorReadPlatformServiceImpl implements 
LoanOriginatorReadPlatformService {
 
     private final LoanOriginatorRepository loanOriginatorRepository;
+    private final LoanOriginatorMappingRepository 
loanOriginatorMappingRepository;
     private final LoanOriginatorMapper loanOriginatorMapper;
 
     @Override
@@ -65,4 +69,13 @@ public class LoanOriginatorReadPlatformServiceImpl 
implements LoanOriginatorRead
                 .orElseThrow(() -> new 
LoanOriginatorNotFoundException(externalId));
         return originator.getId();
     }
+
+    @Override
+    public List<LoanOriginatorData> retrieveByLoanId(final Long loanId) {
+        final List<LoanOriginatorMapping> mappings = 
this.loanOriginatorMappingRepository.findByLoanIdWithOriginator(loanId);
+        if (mappings.isEmpty()) {
+            return Collections.emptyList();
+        }
+        return 
mappings.stream().map(LoanOriginatorMapping::getOriginator).map(this.loanOriginatorMapper::toData).toList();
+    }
 }

Reply via email to