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 332da5273 FINERACT-1754: Fix Search API
332da5273 is described below

commit 332da52730c27d6d8abce168c504824459c5ca0e
Author: Jose Alberto Hernandez <[email protected]>
AuthorDate: Tue Jul 18 00:59:24 2023 -0600

    FINERACT-1754: Fix Search API
---
 .../service/SearchReadPlatformServiceImpl.java     |  28 +++---
 .../integrationtests/SearchResourcesTest.java      | 109 +++++++++++++++++++++
 .../integrationtests/common/SearchHelper.java      |  52 ++++++++++
 3 files changed, 175 insertions(+), 14 deletions(-)

diff --git 
a/fineract-provider/src/main/java/org/apache/fineract/portfolio/search/service/SearchReadPlatformServiceImpl.java
 
b/fineract-provider/src/main/java/org/apache/fineract/portfolio/search/service/SearchReadPlatformServiceImpl.java
index cc697e5ca..f48aa46f9 100644
--- 
a/fineract-provider/src/main/java/org/apache/fineract/portfolio/search/service/SearchReadPlatformServiceImpl.java
+++ 
b/fineract-provider/src/main/java/org/apache/fineract/portfolio/search/service/SearchReadPlatformServiceImpl.java
@@ -79,37 +79,37 @@ public class SearchReadPlatformServiceImpl implements 
SearchReadPlatformService
     public String searchSchema(final SearchConditions searchConditions) {
 
         final String union = " union ";
-        final String clientMatchSql = " (select 'CLIENT' as entityType, c.id 
as entityId, c.display_name as entityName, c.external_id as entityExternalId, 
c.account_no as entityAccountNo "
+        final String clientMatchSql = "( (select 'CLIENT' as entityType, c.id 
as entityId, c.display_name as entityName, c.external_id as entityExternalId, 
c.account_no as entityAccountNo "
                 + " , c.office_id as parentId, o.name as parentName, 
c.mobile_no as entityMobileNo,c.status_enum as entityStatusEnum, null as 
subEntityType, null as parentType "
                 + " from m_client c join m_office o on o.id = c.office_id 
where o.hierarchy like :hierarchy and (c.account_no like :search or 
c.display_name like :search or c.external_id like :search or c.mobile_no like 
:search)) "
-                + " order by c.id desc";
+                + " order by c.id desc)";
 
-        final String loanMatchSql = " (select 'LOAN' as entityType, l.id as 
entityId, pl.name as entityName, l.external_id as entityExternalId, 
l.account_no as entityAccountNo "
+        final String loanMatchSql = "( (select 'LOAN' as entityType, l.id as 
entityId, pl.name as entityName, l.external_id as entityExternalId, 
l.account_no as entityAccountNo "
                 + " , coalesce(c.id,g.id) as parentId, 
coalesce(c.display_name,g.display_name) as parentName, null as entityMobileNo, 
l.loan_status_id as entityStatusEnum, null as subEntityType, CASE WHEN g.id is 
null THEN 'client' ELSE 'group' END as parentType "
                 + " from m_loan l left join m_client c on l.client_id = c.id 
left join m_group g ON l.group_id = g.id left join m_office o on o.id = 
c.office_id left join m_product_loan pl on pl.id=l.product_id where 
(o.hierarchy IS NULL OR o.hierarchy like :hierarchy) and (l.account_no like 
:search or l.external_id like :search)) "
-                + " order by l.id desc";
+                + " order by l.id desc)";
 
-        final String savingMatchSql = " (select 'SAVING' as entityType, s.id 
as entityId, sp.name as entityName, s.external_id as entityExternalId, 
s.account_no as entityAccountNo "
-                + " , coalesce(c.id,g.id) as parentId, 
coalesce(c.display_name,g.display_name) as parentName, null as entityMobileNo, 
s.status_enum as entityStatusEnum, s.deposit_type_enum as subEntityType, CASE 
WHEN g.id is null THEN 'client' ELSE 'group' END as parentType "
+        final String savingMatchSql = "( (select 'SAVING' as entityType, s.id 
as entityId, sp.name as entityName, s.external_id as entityExternalId, 
s.account_no as entityAccountNo "
+                + " , coalesce(c.id,g.id) as parentId, 
coalesce(c.display_name, g.display_name) as parentName, null as entityMobileNo, 
s.status_enum as entityStatusEnum, concat(s.deposit_type_enum, '') as 
subEntityType, CASE WHEN g.id is null THEN 'client' ELSE 'group' END as 
parentType "
                 + " from m_savings_account s left join m_client c on 
s.client_id = c.id left join m_group g ON s.group_id = g.id left join m_office 
o on o.id = c.office_id left join m_savings_product sp on sp.id=s.product_id "
                 + " where (o.hierarchy IS NULL OR o.hierarchy like :hierarchy) 
and (s.account_no like :search or s.external_id like :search)) "
-                + " order by s.id desc";
+                + " order by s.id desc)";
 
-        final String shareMatchSql = " (select 'SHARE' as entityType, s.id as 
entityId, sp.name as entityName, s.external_id as entityExternalId, 
s.account_no as entityAccountNo "
+        final String shareMatchSql = "( (select 'SHARE' as entityType, s.id as 
entityId, sp.name as entityName, s.external_id as entityExternalId, 
s.account_no as entityAccountNo "
                 + " , c.id as parentId, c.display_name as parentName, null as 
entityMobileNo, s.status_enum as entityStatusEnum, null as subEntityType, 
'client' as parentType "
                 + " from m_share_account s left join m_client c on s.client_id 
= c.id left join m_office o on o.id = c.office_id left join m_share_product sp 
on sp.id=s.product_id "
                 + " where (o.hierarchy IS NULL OR o.hierarchy like :hierarchy) 
and (s.account_no like :search or s.external_id like :search)) "
-                + " order by s.id desc";
+                + " order by s.id desc)";
 
-        final String clientIdentifierMatchSql = " (select 'CLIENTIDENTIFIER' 
as entityType, ci.id as entityId, ci.document_key as entityName, "
+        final String clientIdentifierMatchSql = "( (select 'CLIENTIDENTIFIER' 
as entityType, ci.id as entityId, ci.document_key as entityName, "
                 + " null as entityExternalId, null as entityAccountNo, c.id as 
parentId, c.display_name as parentName,null as entityMobileNo, c.status_enum as 
entityStatusEnum, null as subEntityType, null as parentType "
                 + " from m_client_identifier ci join m_client c on 
ci.client_id=c.id join m_office o on o.id = c.office_id "
-                + " where o.hierarchy like :hierarchy and ci.document_key like 
:search ) " + " order by ci.id desc";
+                + " where o.hierarchy like :hierarchy and ci.document_key like 
:search ) " + " order by ci.id desc)";
 
-        final String groupMatchSql = " (select CASE WHEN g.level_id=1 THEN 
'CENTER' ELSE 'GROUP' END as entityType, g.id as entityId, g.display_name as 
entityName, g.external_id as entityExternalId, g.account_no as entityAccountNo, 
"
+        final String groupMatchSql = "( (select CASE WHEN g.level_id=1 THEN 
'CENTER' ELSE 'GROUP' END as entityType, g.id as entityId, g.display_name as 
entityName, g.external_id as entityExternalId, g.account_no as entityAccountNo, 
"
                 + " g.office_id as parentId, o.name as parentName, null as 
entityMobileNo, g.status_enum as entityStatusEnum, null as subEntityType, null 
as parentType "
-                + " from m_group g join m_office o on o.id = g.office_id where 
o.hierarchy like :hierarchy and (g.account_no like :search or g.display_name 
like :search or g.external_id like :search or g.id like :search )) "
-                + " order by g.id desc";
+                + " from m_group g join m_office o on o.id = g.office_id where 
o.hierarchy like :hierarchy and (g.account_no like :search or g.display_name 
like :search or g.external_id like :search )) "
+                + " order by g.id desc)";
 
         final StringBuilder sql = new StringBuilder();
 
diff --git 
a/integration-tests/src/test/java/org/apache/fineract/integrationtests/SearchResourcesTest.java
 
b/integration-tests/src/test/java/org/apache/fineract/integrationtests/SearchResourcesTest.java
new file mode 100644
index 000000000..39355ea35
--- /dev/null
+++ 
b/integration-tests/src/test/java/org/apache/fineract/integrationtests/SearchResourcesTest.java
@@ -0,0 +1,109 @@
+/**
+ * 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.integrationtests;
+
+import static org.junit.Assert.assertEquals;
+import static org.junit.Assert.assertNotNull;
+
+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.util.ArrayList;
+import java.util.Arrays;
+import java.util.List;
+import org.apache.fineract.client.models.GetClientsClientIdResponse;
+import org.apache.fineract.client.models.GetSearchResponse;
+import org.apache.fineract.client.models.PostClientsResponse;
+import org.apache.fineract.integrationtests.common.ClientHelper;
+import org.apache.fineract.integrationtests.common.SearchHelper;
+import org.apache.fineract.integrationtests.common.Utils;
+import org.junit.jupiter.api.BeforeEach;
+import org.junit.jupiter.api.Test;
+
+public class SearchResourcesTest {
+
+    private ResponseSpecification responseSpec;
+    private RequestSpecification requestSpec;
+
+    @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();
+    }
+
+    @Test
+    public void searchAnyValueOverAllResources() {
+        final List<String> resources = Arrays.asList("clients", 
"clientIdentifiers", "groups", "savings", "shares", "loans");
+
+        final String query = Utils.randomStringGenerator("C", 12);
+        final ArrayList<GetSearchResponse> searchResponse = 
SearchHelper.getSearch(requestSpec, responseSpec, query, Boolean.TRUE,
+                resources.toString());
+        assertNotNull(searchResponse);
+        assertEquals(0, searchResponse.size());
+    }
+
+    @Test
+    public void searchAnyValueOverClientResources() {
+        final List<String> resources = Arrays.asList("clients");
+
+        final String query = Utils.randomStringGenerator("C", 12);
+        final ArrayList<GetSearchResponse> searchResponse = 
SearchHelper.getSearch(requestSpec, responseSpec, query, Boolean.TRUE,
+                getResources(resources));
+        assertNotNull(searchResponse);
+        assertEquals(0, searchResponse.size());
+    }
+
+    @Test
+    public void searchOverClientResources() {
+        final List<String> resources = Arrays.asList("clients");
+
+        // Client and Loan account creation
+        String jsonPayload = 
ClientHelper.getBasicClientAsJSON(ClientHelper.DEFAULT_OFFICE_ID, 
ClientHelper.LEGALFORM_ID_PERSON, null);
+        final PostClientsResponse clientResponse = 
ClientHelper.addClientAsPerson(requestSpec, responseSpec, jsonPayload);
+        final Long clientId = clientResponse.getClientId();
+        final GetClientsClientIdResponse getClientResponse = 
ClientHelper.getClient(requestSpec, responseSpec, clientId.intValue());
+        final String query = getClientResponse.getAccountNo();
+
+        final ArrayList<GetSearchResponse> searchResponse = 
SearchHelper.getSearch(requestSpec, responseSpec, query, Boolean.FALSE,
+                getResources(resources));
+        assertNotNull(searchResponse);
+        assertEquals(1, searchResponse.size());
+        assertEquals("Client name comparation", 
getClientResponse.getDisplayName(), searchResponse.get(0).getEntityName());
+    }
+
+    @Test
+    public void searchAnyValueOverLoanResources() {
+        final List<String> resources = Arrays.asList("loans");
+
+        final String query = Utils.randomStringGenerator("L", 12);
+        final ArrayList<GetSearchResponse> searchResponse = 
SearchHelper.getSearch(requestSpec, responseSpec, query, Boolean.TRUE,
+                getResources(resources));
+        assertNotNull(searchResponse);
+        assertEquals(0, searchResponse.size());
+    }
+
+    private String getResources(final List<String> resources) {
+        return String.join(",", resources);
+    }
+
+}
diff --git 
a/integration-tests/src/test/java/org/apache/fineract/integrationtests/common/SearchHelper.java
 
b/integration-tests/src/test/java/org/apache/fineract/integrationtests/common/SearchHelper.java
new file mode 100644
index 000000000..c0c6db7ec
--- /dev/null
+++ 
b/integration-tests/src/test/java/org/apache/fineract/integrationtests/common/SearchHelper.java
@@ -0,0 +1,52 @@
+/**
+ * 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.integrationtests.common;
+
+import com.google.gson.Gson;
+import com.linecorp.armeria.internal.shaded.guava.reflect.TypeToken;
+import io.restassured.specification.RequestSpecification;
+import io.restassured.specification.ResponseSpecification;
+import java.lang.reflect.Type;
+import java.util.ArrayList;
+import lombok.extern.slf4j.Slf4j;
+import org.apache.fineract.client.models.GetSearchResponse;
+import org.apache.fineract.client.util.JSON;
+
+@Slf4j
+public final class SearchHelper {
+
+    private static final Gson GSON = new JSON().getGson();
+
+    private SearchHelper() {
+
+    }
+
+    private static final String SEARCH_URL = 
"/fineract-provider/api/v1/search?" + Utils.TENANT_IDENTIFIER;
+
+    public static ArrayList<GetSearchResponse> getSearch(final 
RequestSpecification requestSpec, final ResponseSpecification responseSpec,
+            final String query, final Boolean exactMatch, final String 
resources) {
+        final String urlSearch = SEARCH_URL + "&exactMatch=" + 
exactMatch.toString() + "&query=" + query + "&resource=" + resources;
+        log.info("URL to search: {}", urlSearch);
+        final String response = Utils.performServerGet(requestSpec, 
responseSpec, urlSearch);
+        log.info("Result: {}", response);
+        Type searchResourcesListType = new 
TypeToken<ArrayList<GetSearchResponse>>() {}.getType();
+        return GSON.fromJson(response, searchResourcesListType);
+    }
+
+}

Reply via email to