From e61007140a3b98d7c28d1ae96a94c61941e3556f Mon Sep 17 00:00:00 2001 From: Simon Kelly Date: Tue, 16 May 2023 13:26:36 +0200 Subject: [PATCH 1/5] refactor db optimization tests --- .../tests/CaseDbModelQueryTests.java | 71 ++++++++++------ .../tests/CaseDbOptimizationsTest.java | 80 +++++++++++++------ .../formplayer/utils/TestContext.java | 25 +----- 3 files changed, 106 insertions(+), 70 deletions(-) diff --git a/src/test/java/org/commcare/formplayer/tests/CaseDbModelQueryTests.java b/src/test/java/org/commcare/formplayer/tests/CaseDbModelQueryTests.java index d57b4797f..241bedbb2 100644 --- a/src/test/java/org/commcare/formplayer/tests/CaseDbModelQueryTests.java +++ b/src/test/java/org/commcare/formplayer/tests/CaseDbModelQueryTests.java @@ -2,42 +2,63 @@ import static org.commcare.formplayer.utils.DbTestUtils.evaluate; +import org.commcare.formplayer.application.UtilController; +import org.commcare.formplayer.configuration.CacheConfiguration; +import org.commcare.formplayer.junit.InitializeStaticsExtension; +import org.commcare.formplayer.junit.RestoreFactoryExtension; +import org.commcare.formplayer.junit.StorageFactoryExtension; +import org.commcare.formplayer.junit.request.SyncDbRequest; import org.commcare.formplayer.sandbox.UserSqlSandbox; +import org.commcare.formplayer.services.RestoreFactory; import org.commcare.formplayer.utils.TestContext; import org.commcare.formplayer.utils.TestStorageUtils; import org.javarosa.core.model.condition.EvaluationContext; +import org.javarosa.xpath.parser.XPathSyntaxException; import org.junit.jupiter.api.BeforeEach; import org.junit.jupiter.api.Test; +import org.junit.jupiter.api.extension.ExtendWith; +import org.junit.jupiter.api.extension.RegisterExtension; +import org.springframework.beans.factory.annotation.Autowired; import org.springframework.boot.test.autoconfigure.web.servlet.WebMvcTest; +import org.springframework.context.annotation.Import; import org.springframework.test.context.ContextConfiguration; +import org.springframework.test.web.servlet.MockMvc; + -/** - * @author wspride - */ @WebMvcTest -public class CaseDbModelQueryTests extends BaseTestClass { +@Import({UtilController.class}) +@ContextConfiguration(classes = {TestContext.class, CacheConfiguration.class}) +@ExtendWith(InitializeStaticsExtension.class) +public class CaseDbModelQueryTests { - @Override - @BeforeEach - public void setUp() throws Exception { - super.setUp(); - configureRestoreFactory("synctestdomain", "synctestusername"); - } + @Autowired + private MockMvc mockMvc; + + @Autowired + private RestoreFactory restoreFactory; + + @RegisterExtension + static RestoreFactoryExtension restoreFactoryExt = new RestoreFactoryExtension.builder() + .withUser("synctestusername").withDomain("synctestdomain") + .withRestorePath("restores/dbtests/case_db_model_query.xml") + .build(); - @Override - protected String getMockRestoreFileName() { - return "restores/dbtests/case_db_model_query.xml"; + @RegisterExtension + static StorageFactoryExtension storageExt = new StorageFactoryExtension.builder() + .withUser("back_nav").withDomain("back_nav").build(); + private UserSqlSandbox sandbox; + private EvaluationContext evaluationContext; + + @BeforeEach + public void setUp() { + new SyncDbRequest(mockMvc, restoreFactory).request(); + sandbox = restoreFactory.getSqlSandbox(); + evaluationContext = TestStorageUtils.getEvaluationContextWithoutSession(sandbox); + evaluationContext.setDebugModeOn(); } - /** - * Tests for basic common case database queries - */ @Test - public void testDbModelQueryLookups() throws Exception { - syncDb(); - UserSqlSandbox sandbox = getRestoreSandbox(); - EvaluationContext ec = TestStorageUtils.getEvaluationContextWithoutSession(sandbox); - ec.setDebugModeOn(); + public void testModelQueryLookupDerivations() throws Exception { evaluate( "join(',',instance('casedb')/casedb/case[@case_type='unit_test_child_child" + "'][@status='open'][true() and " @@ -45,7 +66,11 @@ public void testDbModelQueryLookups() throws Exception { "instance('casedb')/casedb/case[@case_id = instance('casedb')" + "/casedb/case[@case_id=current()/index/parent]/index/parent]/test = " + "'true']/@case_id)", - "child_ptwo_one_one,child_one_one", ec); + "child_ptwo_one_one,child_one_one", evaluationContext); + } + + @Test + public void testModelSelfReference() throws XPathSyntaxException { evaluate( "join(',',instance('casedb')/casedb/case[@case_type='unit_test_child'][@status" + "='open'][true() and " @@ -53,7 +78,7 @@ public void testDbModelQueryLookups() throws Exception { "count(instance('casedb')/casedb/case[index/parent = instance('casedb')" + "/casedb/case[@case_id=current()/@case_id]/index/parent][false = " + "'true']) > 0]/@case_id)", - "", ec); + "", evaluationContext); } } diff --git a/src/test/java/org/commcare/formplayer/tests/CaseDbOptimizationsTest.java b/src/test/java/org/commcare/formplayer/tests/CaseDbOptimizationsTest.java index e2f6d1e39..ccf7d65b4 100644 --- a/src/test/java/org/commcare/formplayer/tests/CaseDbOptimizationsTest.java +++ b/src/test/java/org/commcare/formplayer/tests/CaseDbOptimizationsTest.java @@ -1,73 +1,101 @@ package org.commcare.formplayer.tests; -import static org.commcare.formplayer.utils.DbTestUtils.evaluate; - +import org.commcare.formplayer.application.UtilController; +import org.commcare.formplayer.configuration.CacheConfiguration; +import org.commcare.formplayer.junit.InitializeStaticsExtension; +import org.commcare.formplayer.junit.RestoreFactoryExtension; +import org.commcare.formplayer.junit.StorageFactoryExtension; +import org.commcare.formplayer.junit.request.SyncDbRequest; import org.commcare.formplayer.sandbox.UserSqlSandbox; +import org.commcare.formplayer.services.RestoreFactory; import org.commcare.formplayer.utils.TestContext; import org.commcare.formplayer.utils.TestStorageUtils; import org.javarosa.core.model.condition.EvaluationContext; import org.junit.jupiter.api.BeforeEach; import org.junit.jupiter.api.Test; +import org.junit.jupiter.api.extension.ExtendWith; +import org.junit.jupiter.api.extension.RegisterExtension; +import org.springframework.beans.factory.annotation.Autowired; import org.springframework.boot.test.autoconfigure.web.servlet.WebMvcTest; +import org.springframework.context.annotation.Import; import org.springframework.test.context.ContextConfiguration; +import org.springframework.test.web.servlet.MockMvc; + +import static org.commcare.formplayer.utils.DbTestUtils.evaluate; -/** - * @author wspride - */ @WebMvcTest -public class CaseDbOptimizationsTest extends BaseTestClass { +@Import({UtilController.class}) +@ContextConfiguration(classes = {TestContext.class, CacheConfiguration.class}) +@ExtendWith(InitializeStaticsExtension.class) +public class CaseDbOptimizationsTest { - @Override - @BeforeEach - public void setUp() throws Exception { - super.setUp(); - configureRestoreFactory("synctestdomain", "synctestusername"); - } + @Autowired + private MockMvc mockMvc; - @Override - protected String getMockRestoreFileName() { - return "restores/dbtests/case_test_db_optimizations.xml"; - } + @Autowired + private RestoreFactory restoreFactory; + + @RegisterExtension + static RestoreFactoryExtension restoreFactoryExt = new RestoreFactoryExtension.builder() + .withUser("synctestusername").withDomain("synctestdomain") + .withRestorePath("restores/dbtests/case_test_db_optimizations.xml") + .build(); + @RegisterExtension + static StorageFactoryExtension storageExt = new StorageFactoryExtension.builder() + .withUser("back_nav").withDomain("back_nav").build(); + private UserSqlSandbox sandbox; + private EvaluationContext evaluationContext; + + @BeforeEach + public void setUp() { + new SyncDbRequest(mockMvc, restoreFactory).request(); + sandbox = restoreFactory.getSqlSandbox(); + evaluationContext = TestStorageUtils.getEvaluationContextWithoutSession(sandbox); + evaluationContext.setDebugModeOn(); + } /** * Tests for basic common case database queries */ @Test public void testDbOptimizations() throws Exception { - syncDb(); - UserSqlSandbox sandbox = getRestoreSandbox(); EvaluationContext ec = TestStorageUtils.getEvaluationContextWithoutSession(sandbox); ec.setDebugModeOn(); evaluate( "sort(join(' ',instance('casedb')/casedb/case[index/parent = " + "'test_case_parent']/@case_id))", - "child_one child_three child_two", ec); + "child_one child_three child_two", evaluationContext); evaluate( "join(' ',instance('casedb')/casedb/case[index/parent = " + "'test_case_parent'][@case_id = 'child_two']/@case_id)", - "child_two", ec); + "child_two", evaluationContext); evaluate( "sort(join(' ',instance('casedb')/casedb/case[index/parent = " + "'test_case_parent'][@case_id != 'child_two']/@case_id))", - "child_one child_three", ec); + "child_one child_three", evaluationContext); + } + @Test + public void testSelectedOptimization() throws Exception { + EvaluationContext ec = TestStorageUtils.getEvaluationContextWithoutSession(sandbox); + ec.setDebugModeOn(); evaluate( "sort(join(' ',instance('casedb')/casedb/case[selected('test_case_parent', " + "index/parent)]/@case_id))", - "child_one child_three child_two", ec); + "child_one child_three child_two", evaluationContext); evaluate( "sort(join(' ',instance('casedb')/casedb/case[selected('test_case_parent " + "test_case_parent_2', index/parent)]/@case_id))", - "child_one child_three child_two", ec); + "child_one child_three child_two", evaluationContext); evaluate( "sort(join(' ',instance('casedb')/casedb/case[selected('test_case_parent_2 " + "test_case_parent', index/parent)]/@case_id))", - "child_one child_three child_two", ec); + "child_one child_three child_two", evaluationContext); evaluate( "join(' ',instance('casedb')/casedb/case[selected('test_case_parent_2 " + "test_case_parent_3', index/parent)]/@case_id)", - "", ec); + "", evaluationContext); evaluate("join(' ',instance('casedb')/casedb/case[selected('', index/parent)]/@case_id)", - "", ec); + "", evaluationContext); } } diff --git a/src/test/java/org/commcare/formplayer/utils/TestContext.java b/src/test/java/org/commcare/formplayer/utils/TestContext.java index 8a4beb4e5..9100ce383 100644 --- a/src/test/java/org/commcare/formplayer/utils/TestContext.java +++ b/src/test/java/org/commcare/formplayer/utils/TestContext.java @@ -7,22 +7,7 @@ import org.commcare.formplayer.mocks.MockLockRegistry; import org.commcare.formplayer.mocks.TestInstallService; import org.commcare.formplayer.objects.FormVolatilityRecord; -import org.commcare.formplayer.services.CaseSearchHelper; -import org.commcare.formplayer.services.CategoryTimingHelper; -import org.commcare.formplayer.services.FormDefinitionService; -import org.commcare.formplayer.services.FormSessionService; -import org.commcare.formplayer.services.FormplayerFormSendCalloutHandler; -import org.commcare.formplayer.services.FormplayerStorageFactory; -import org.commcare.formplayer.services.HqUserDetailsService; -import org.commcare.formplayer.services.InstallService; -import org.commcare.formplayer.services.MediaMetaDataService; -import org.commcare.formplayer.services.MenuSessionFactory; -import org.commcare.formplayer.services.MenuSessionRunnerService; -import org.commcare.formplayer.services.MenuSessionService; -import org.commcare.formplayer.services.NewFormResponseFactory; -import org.commcare.formplayer.services.RestoreFactory; -import org.commcare.formplayer.services.SubmitService; -import org.commcare.formplayer.services.VirtualDataInstanceService; +import org.commcare.formplayer.services.*; import org.commcare.formplayer.util.Constants; import org.commcare.formplayer.util.FormplayerDatadog; import org.commcare.formplayer.util.NotificationLogger; @@ -103,6 +88,9 @@ public InternalResourceViewResolver viewResolver() { @MockBean public NotificationLogger notificationLogger; + @MockBean + public FormplayerLockRegistry userLockRegistry; + @Bean public ValueOperations redisTemplateLong() { return Mockito.mock(ValueOperations.class); @@ -153,11 +141,6 @@ public FormplayerDatadog datadog() { return Mockito.spy(new FormplayerDatadog(datadogStatsDClient(), new ArrayList())); } - @Bean - public LockRegistry userLockRegistry() { - return Mockito.spy(MockLockRegistry.class); - } - @Bean public NewFormResponseFactory newFormResponseFactory() { return Mockito.spy(NewFormResponseFactory.class); From d9f4e6bbce3d3de37c3c334402bbd7553e7d2726 Mon Sep 17 00:00:00 2001 From: Simon Kelly Date: Wed, 17 May 2023 15:58:36 +0200 Subject: [PATCH 2/5] implement reverse index lookup and tests --- .../models/FormplayerCaseIndexTable.java | 70 +++++++++++-- .../tests/CaseDbModelQueryTests.java | 98 ++++++++++++++++++- 2 files changed, 159 insertions(+), 9 deletions(-) diff --git a/src/main/java/org/commcare/formplayer/database/models/FormplayerCaseIndexTable.java b/src/main/java/org/commcare/formplayer/database/models/FormplayerCaseIndexTable.java index 1a0461894..5a478e1ff 100644 --- a/src/main/java/org/commcare/formplayer/database/models/FormplayerCaseIndexTable.java +++ b/src/main/java/org/commcare/formplayer/database/models/FormplayerCaseIndexTable.java @@ -1,11 +1,12 @@ package org.commcare.formplayer.database.models; -import static org.commcare.formplayer.sandbox.SqlSandboxUtils.execSql; - +import com.google.common.collect.ImmutableMap; +import org.apache.commons.lang3.text.StrSubstitutor; import org.apache.commons.logging.Log; import org.apache.commons.logging.LogFactory; import org.commcare.cases.model.Case; import org.commcare.cases.model.CaseIndex; +import org.commcare.cases.query.queryset.DualTableMultiMatchModelQuerySet; import org.commcare.cases.query.queryset.DualTableSingleMatchModelQuerySet; import org.commcare.formplayer.sandbox.SqlHelper; import org.commcare.formplayer.sandbox.SqlStorage; @@ -21,11 +22,9 @@ import java.sql.PreparedStatement; import java.sql.ResultSet; import java.sql.SQLException; -import java.util.Collection; -import java.util.HashMap; -import java.util.LinkedHashSet; -import java.util.List; -import java.util.Vector; +import java.util.*; + +import static org.commcare.formplayer.sandbox.SqlSandboxUtils.execSql; /** * @author ctsims @@ -334,6 +333,63 @@ public DualTableSingleMatchModelQuerySet bulkReadIndexToCaseIdMatch(String index } } + /** + * Performs a reverse index match on for the provided index name and target (parent) cases. + * + * @param indexName The name of the index e.g. 'parent' + * @param cuedCases Row IDs for the parent cases. + * @return ModelQuerySet with the results + */ + public DualTableMultiMatchModelQuerySet bulkReadIndexToCaseIdMatchReverse(String indexName, + Collection cuedCases) { + DualTableMultiMatchModelQuerySet set = new DualTableMultiMatchModelQuerySet(); + String caseIdIndex = TableBuilder.scrubName(Case.INDEX_CASE_ID); + List> whereParamList = TableBuilder.sqlList(cuedCases, "?"); + try { + for (Pair querySet : whereParamList) { + + StrSubstitutor substitutor = new StrSubstitutor(ImmutableMap.of( + "caseId", caseTableName + "." + DatabaseHelper.ID_COL, + "childId", COL_CASE_RECORD_ID, + "caseTable", caseTableName, + "indexTable", getTableName(), + "targetCol", COL_INDEX_TARGET, + "caseIdCol", caseIdIndex, + "indexNameCol", COL_INDEX_NAME, + "indexName", indexName, + "parentCases", querySet.first + )); + String query = substitutor.replace("SELECT ${caseId}, ${childId} " + + "FROM ${caseTable} JOIN ${indexTable} ON ${targetCol} = ${caseIdCol} " + + "WHERE ${indexNameCol} = '${indexName}' AND ${caseId} IN ${parentCases};"); + + try (PreparedStatement preparedStatement = + connectionHandler.getConnection().prepareStatement(query)) { + int argIndex = 1; + for (String arg : querySet.second) { + preparedStatement.setString(argIndex, arg); + argIndex++; + } + + if (log.isTraceEnabled()) { + SqlHelper.explainSql(connectionHandler.getConnection(), query, querySet.second); + } + + try (ResultSet resultSet = preparedStatement.executeQuery()) { + while (resultSet.next()) { + int caseId = resultSet.getInt(resultSet.findColumn(DatabaseHelper.ID_COL)); + int targetCase = resultSet.getInt(resultSet.findColumn(COL_CASE_RECORD_ID)); + set.loadResult(caseId, targetCase); + } + } + } + } + return set; + } catch (SQLException e) { + throw new RuntimeException(e); + } + } + public static String getArgumentBasedVariableSet(int number) { StringBuffer sb = new StringBuffer(); sb.append("("); diff --git a/src/test/java/org/commcare/formplayer/tests/CaseDbModelQueryTests.java b/src/test/java/org/commcare/formplayer/tests/CaseDbModelQueryTests.java index 241bedbb2..301748269 100644 --- a/src/test/java/org/commcare/formplayer/tests/CaseDbModelQueryTests.java +++ b/src/test/java/org/commcare/formplayer/tests/CaseDbModelQueryTests.java @@ -1,7 +1,8 @@ package org.commcare.formplayer.tests; -import static org.commcare.formplayer.utils.DbTestUtils.evaluate; - +import com.google.common.collect.ImmutableMap; +import org.commcare.cases.query.QueryContext; +import org.commcare.cases.query.queryset.CurrentModelQuerySet; import org.commcare.formplayer.application.UtilController; import org.commcare.formplayer.configuration.CacheConfiguration; import org.commcare.formplayer.junit.InitializeStaticsExtension; @@ -13,7 +14,15 @@ import org.commcare.formplayer.utils.TestContext; import org.commcare.formplayer.utils.TestStorageUtils; import org.javarosa.core.model.condition.EvaluationContext; +import org.javarosa.core.model.instance.TreeReference; +import org.javarosa.core.model.trace.AccumulatingReporter; +import org.javarosa.core.model.trace.EvaluationTraceReporter; +import org.javarosa.core.model.utils.InstrumentationUtils; +import org.javarosa.xpath.XPathLazyNodeset; +import org.javarosa.xpath.XPathParseTool; +import org.javarosa.xpath.expr.FunctionUtils; import org.javarosa.xpath.parser.XPathSyntaxException; +import org.junit.jupiter.api.Assertions; import org.junit.jupiter.api.BeforeEach; import org.junit.jupiter.api.Test; import org.junit.jupiter.api.extension.ExtendWith; @@ -24,6 +33,10 @@ import org.springframework.test.context.ContextConfiguration; import org.springframework.test.web.servlet.MockMvc; +import java.util.function.Predicate; + +import static org.commcare.formplayer.utils.DbTestUtils.evaluate; + @WebMvcTest @Import({UtilController.class}) @@ -81,4 +94,85 @@ public void testModelSelfReference() throws XPathSyntaxException { "", evaluationContext); } + + @Test + public void testModelReverseIndexLookup() throws XPathSyntaxException { + XPathLazyNodeset nodeset = (XPathLazyNodeset) XPathParseTool.parseXPath( + "instance('casedb')/casedb/case[@case_type='unit_test_parent']").eval(evaluationContext); + + ImmutableMap expectedOutputs = ImmutableMap.of( + "test_case_parent", "child_one,child_two,child_three", + "parent_two", "child_ptwo_one" + ); + Assertions.assertEquals(nodeset.getReferences().size(), expectedOutputs.size()); + + EvaluationTraceReporter reporter = new AccumulatingReporter(); + evaluationContext.setDebugModeOn(reporter); + for (TreeReference current : nodeset.getReferences()) { + EvaluationContext subContext = new EvaluationContext(evaluationContext, current); + QueryContext newContext = subContext.getCurrentQueryContext() + .checkForDerivativeContextAndReturn(nodeset.getReferences().size()); + newContext.setHackyOriginalContextBody(new CurrentModelQuerySet(nodeset.getReferences())); + subContext.setQueryContext(newContext); + + String parentCaseId = FunctionUtils.toString( + XPathParseTool.parseXPath("./@case_id").eval(subContext) + ); + + evaluate( + "join(',',instance('casedb')/casedb/case[index/parent = current()/@case_id]/@case_id)", + expectedOutputs.get(parentCaseId), + subContext + ); + + } + Predicate predicate = line -> line.contains("Load Query Set Transform[current]=>[current|reverse index|parent]: Loaded: 4"); + int matchedReverseIndexLoad = InstrumentationUtils.countMatchedTraces(reporter, predicate); + Assertions.assertEquals(1, matchedReverseIndexLoad); + + int matchedLookups = InstrumentationUtils.countMatchedTraces(reporter, line -> line.contains("QuerySetLookup|current|reverse index|parent: Results:")); + Assertions.assertEquals(2, matchedLookups); + } + + @Test + public void testModelIndexLookup() throws XPathSyntaxException { + XPathLazyNodeset nodeset = (XPathLazyNodeset) XPathParseTool.parseXPath( + "instance('casedb')/casedb/case[@case_type='unit_test_child']").eval(evaluationContext); + EvaluationTraceReporter reporter = new AccumulatingReporter(); + evaluationContext.setDebugModeOn(reporter); + + ImmutableMap expectedOutputs = ImmutableMap.of( + "child_one", "test_case_parent", + "child_two", "test_case_parent", + "child_three", "test_case_parent", + "child_ptwo_one", "parent_two" + ); + + Assertions.assertEquals(nodeset.getReferences().size(), expectedOutputs.size()); + + for (TreeReference current : nodeset.getReferences()) { + EvaluationContext subContext = new EvaluationContext(evaluationContext, current); + QueryContext newContext = subContext.getCurrentQueryContext() + .checkForDerivativeContextAndReturn(nodeset.getReferences().size()); + newContext.setHackyOriginalContextBody(new CurrentModelQuerySet(nodeset.getReferences())); + subContext.setQueryContext(newContext); + + String childCaseId = FunctionUtils.toString( + XPathParseTool.parseXPath("./@case_id").eval(subContext) + ); + + evaluate( + "join(',',instance('casedb')/casedb/case[@case_id = current()/index/parent]/@case_id)", + expectedOutputs.get(childCaseId), + subContext + ); + + } + Predicate predicate = line -> line.contains("Load Query Set Transform[current]=>[current|index|parent]: Loaded: 2"); + int matchedReverseIndexLoad = InstrumentationUtils.countMatchedTraces(reporter, predicate); + Assertions.assertEquals(1, matchedReverseIndexLoad); + + int matchedLookups = InstrumentationUtils.countMatchedTraces(reporter, line -> line.contains("QuerySetLookup|current|index|parent: Results: 1")); + Assertions.assertEquals(4, matchedLookups); + } } From e18ba0b763cf29fc54033eac4681f938906bc46d Mon Sep 17 00:00:00 2001 From: Simon Kelly Date: Wed, 17 May 2023 16:42:51 +0200 Subject: [PATCH 3/5] refactor query execution functions --- .../models/FormplayerCaseIndexTable.java | 163 +++++++++--------- 1 file changed, 83 insertions(+), 80 deletions(-) diff --git a/src/main/java/org/commcare/formplayer/database/models/FormplayerCaseIndexTable.java b/src/main/java/org/commcare/formplayer/database/models/FormplayerCaseIndexTable.java index 5a478e1ff..e2c836fe4 100644 --- a/src/main/java/org/commcare/formplayer/database/models/FormplayerCaseIndexTable.java +++ b/src/main/java/org/commcare/formplayer/database/models/FormplayerCaseIndexTable.java @@ -1,5 +1,7 @@ package org.commcare.formplayer.database.models; +import static org.commcare.formplayer.sandbox.SqlSandboxUtils.execSql; + import com.google.common.collect.ImmutableMap; import org.apache.commons.lang3.text.StrSubstitutor; import org.apache.commons.logging.Log; @@ -22,9 +24,13 @@ import java.sql.PreparedStatement; import java.sql.ResultSet; import java.sql.SQLException; -import java.util.*; - -import static org.commcare.formplayer.sandbox.SqlSandboxUtils.execSql; +import java.util.Collection; +import java.util.HashMap; +import java.util.LinkedHashSet; +import java.util.List; +import java.util.Vector; +import java.util.function.Consumer; +import java.util.stream.Collector; /** * @author ctsims @@ -53,7 +59,7 @@ public FormplayerCaseIndexTable(ConnectionHandler connectionHandler) { } public FormplayerCaseIndexTable(ConnectionHandler connectionHandler, String tableName, String caseTableName, - boolean createTable) { + boolean createTable) { this.connectionHandler = connectionHandler; this.tableName = tableName; this.caseTableName = caseTableName; @@ -283,54 +289,40 @@ public int loadIntoIndexTable(HashMap> indexCache, Strin * @return */ public DualTableSingleMatchModelQuerySet bulkReadIndexToCaseIdMatch(String indexName, - Collection cuedCases) { + Collection cuedCases) { DualTableSingleMatchModelQuerySet set = new DualTableSingleMatchModelQuerySet(); String caseIdIndex = TableBuilder.scrubName(Case.INDEX_CASE_ID); List> whereParamList = TableBuilder.sqlList(cuedCases, "?"); - try { - for (Pair querySet : whereParamList) { - - String query = String.format( - "SELECT %s,%s " + - "FROM %s " + - "INNER JOIN %s " + - "ON %s = %s " + - "WHERE %s = '%s' " + - "AND " + - "%s IN %s", - - COL_CASE_RECORD_ID, caseTableName + "." + DatabaseHelper.ID_COL, - getTableName(), - caseTableName, - COL_INDEX_TARGET, caseIdIndex, - COL_INDEX_NAME, indexName, - COL_CASE_RECORD_ID, querySet.first); - - try (PreparedStatement preparedStatement = - connectionHandler.getConnection().prepareStatement(query)) { - int argIndex = 1; - for (String arg : querySet.second) { - preparedStatement.setString(argIndex, arg); - argIndex++; - } - if (log.isTraceEnabled()) { - SqlHelper.explainSql(connectionHandler.getConnection(), query, querySet.second); - } + for (Pair querySet : whereParamList) { - try (ResultSet resultSet = preparedStatement.executeQuery()) { - while (resultSet.next()) { - int caseId = resultSet.getInt(resultSet.findColumn(COL_CASE_RECORD_ID)); - int targetCase = resultSet.getInt(resultSet.findColumn(DatabaseHelper.ID_COL)); - set.loadResult(caseId, targetCase); - } - } + String query = String.format( + "SELECT %s,%s " + + "FROM %s " + + "INNER JOIN %s " + + "ON %s = %s " + + "WHERE %s = '%s' " + + "AND " + + "%s IN %s", + + COL_CASE_RECORD_ID, caseTableName + "." + DatabaseHelper.ID_COL, + getTableName(), + caseTableName, + COL_INDEX_TARGET, caseIdIndex, + COL_INDEX_NAME, indexName, + COL_CASE_RECORD_ID, querySet.first); + + executeQueryCollectResults(query, querySet.second, resultSet -> { + try { + int caseId = resultSet.getInt(resultSet.findColumn(COL_CASE_RECORD_ID)); + int targetCase = resultSet.getInt(resultSet.findColumn(DatabaseHelper.ID_COL)); + set.loadResult(caseId, targetCase); + } catch (SQLException e) { + throw new RuntimeException(e); } - } - return set; - } catch (SQLException e) { - throw new RuntimeException(e); + }); } + return set; } /** @@ -341,50 +333,61 @@ public DualTableSingleMatchModelQuerySet bulkReadIndexToCaseIdMatch(String index * @return ModelQuerySet with the results */ public DualTableMultiMatchModelQuerySet bulkReadIndexToCaseIdMatchReverse(String indexName, - Collection cuedCases) { + Collection cuedCases) { DualTableMultiMatchModelQuerySet set = new DualTableMultiMatchModelQuerySet(); String caseIdIndex = TableBuilder.scrubName(Case.INDEX_CASE_ID); List> whereParamList = TableBuilder.sqlList(cuedCases, "?"); + + for (Pair querySet : whereParamList) { + + StrSubstitutor substitutor = new StrSubstitutor(ImmutableMap.of( + "caseId", caseTableName + "." + DatabaseHelper.ID_COL, + "childId", COL_CASE_RECORD_ID, + "caseTable", caseTableName, + "indexTable", getTableName(), + "targetCol", COL_INDEX_TARGET, + "caseIdCol", caseIdIndex, + "indexNameCol", COL_INDEX_NAME, + "indexName", indexName, + "parentCases", querySet.first + )); + String query = substitutor.replace("SELECT ${caseId}, ${childId} " + + "FROM ${caseTable} JOIN ${indexTable} ON ${targetCol} = ${caseIdCol} " + + "WHERE ${indexNameCol} = '${indexName}' AND ${caseId} IN ${parentCases};"); + + executeQueryCollectResults(query, querySet.second, resultSet -> { + try { + int caseId = resultSet.getInt(resultSet.findColumn(DatabaseHelper.ID_COL)); + int targetCase = resultSet.getInt(resultSet.findColumn(COL_CASE_RECORD_ID)); + set.loadResult(caseId, targetCase); + } catch (SQLException e) { + throw new RuntimeException(e); + } + }); + } + return set; + } + + private void executeQueryCollectResults(String query, String[] args, Consumer collector) { try { - for (Pair querySet : whereParamList) { - - StrSubstitutor substitutor = new StrSubstitutor(ImmutableMap.of( - "caseId", caseTableName + "." + DatabaseHelper.ID_COL, - "childId", COL_CASE_RECORD_ID, - "caseTable", caseTableName, - "indexTable", getTableName(), - "targetCol", COL_INDEX_TARGET, - "caseIdCol", caseIdIndex, - "indexNameCol", COL_INDEX_NAME, - "indexName", indexName, - "parentCases", querySet.first - )); - String query = substitutor.replace("SELECT ${caseId}, ${childId} " + - "FROM ${caseTable} JOIN ${indexTable} ON ${targetCol} = ${caseIdCol} " + - "WHERE ${indexNameCol} = '${indexName}' AND ${caseId} IN ${parentCases};"); - - try (PreparedStatement preparedStatement = - connectionHandler.getConnection().prepareStatement(query)) { - int argIndex = 1; - for (String arg : querySet.second) { - preparedStatement.setString(argIndex, arg); - argIndex++; - } + try (PreparedStatement preparedStatement = + connectionHandler.getConnection().prepareStatement(query)) { + int argIndex = 1; + for (String arg : args) { + preparedStatement.setString(argIndex, arg); + argIndex++; + } - if (log.isTraceEnabled()) { - SqlHelper.explainSql(connectionHandler.getConnection(), query, querySet.second); - } + if (log.isTraceEnabled()) { + SqlHelper.explainSql(connectionHandler.getConnection(), query, args); + } - try (ResultSet resultSet = preparedStatement.executeQuery()) { - while (resultSet.next()) { - int caseId = resultSet.getInt(resultSet.findColumn(DatabaseHelper.ID_COL)); - int targetCase = resultSet.getInt(resultSet.findColumn(COL_CASE_RECORD_ID)); - set.loadResult(caseId, targetCase); - } + try (ResultSet resultSet = preparedStatement.executeQuery()) { + while (resultSet.next()) { + collector.accept(resultSet); } } } - return set; } catch (SQLException e) { throw new RuntimeException(e); } From 1d1ffc6f7cb0cc556c82c972bc51b4a9f1e91ebc Mon Sep 17 00:00:00 2001 From: Simon Kelly Date: Thu, 18 May 2023 09:33:50 +0200 Subject: [PATCH 4/5] rename var and function --- .../formplayer/database/models/FormplayerCaseIndexTable.java | 5 ++--- .../org/commcare/formplayer/tests/CaseDbModelQueryTests.java | 4 ++-- 2 files changed, 4 insertions(+), 5 deletions(-) diff --git a/src/main/java/org/commcare/formplayer/database/models/FormplayerCaseIndexTable.java b/src/main/java/org/commcare/formplayer/database/models/FormplayerCaseIndexTable.java index e2c836fe4..ef3bc7f5c 100644 --- a/src/main/java/org/commcare/formplayer/database/models/FormplayerCaseIndexTable.java +++ b/src/main/java/org/commcare/formplayer/database/models/FormplayerCaseIndexTable.java @@ -30,7 +30,6 @@ import java.util.List; import java.util.Vector; import java.util.function.Consumer; -import java.util.stream.Collector; /** * @author ctsims @@ -332,8 +331,8 @@ public DualTableSingleMatchModelQuerySet bulkReadIndexToCaseIdMatch(String index * @param cuedCases Row IDs for the parent cases. * @return ModelQuerySet with the results */ - public DualTableMultiMatchModelQuerySet bulkReadIndexToCaseIdMatchReverse(String indexName, - Collection cuedCases) { + public DualTableMultiMatchModelQuerySet bulkReadCaseIdToIndexMatch(String indexName, + Collection cuedCases) { DualTableMultiMatchModelQuerySet set = new DualTableMultiMatchModelQuerySet(); String caseIdIndex = TableBuilder.scrubName(Case.INDEX_CASE_ID); List> whereParamList = TableBuilder.sqlList(cuedCases, "?"); diff --git a/src/test/java/org/commcare/formplayer/tests/CaseDbModelQueryTests.java b/src/test/java/org/commcare/formplayer/tests/CaseDbModelQueryTests.java index 301748269..16e10e79f 100644 --- a/src/test/java/org/commcare/formplayer/tests/CaseDbModelQueryTests.java +++ b/src/test/java/org/commcare/formplayer/tests/CaseDbModelQueryTests.java @@ -169,8 +169,8 @@ public void testModelIndexLookup() throws XPathSyntaxException { } Predicate predicate = line -> line.contains("Load Query Set Transform[current]=>[current|index|parent]: Loaded: 2"); - int matchedReverseIndexLoad = InstrumentationUtils.countMatchedTraces(reporter, predicate); - Assertions.assertEquals(1, matchedReverseIndexLoad); + int matchedIndexLoad = InstrumentationUtils.countMatchedTraces(reporter, predicate); + Assertions.assertEquals(1, matchedIndexLoad); int matchedLookups = InstrumentationUtils.countMatchedTraces(reporter, line -> line.contains("QuerySetLookup|current|index|parent: Results: 1")); Assertions.assertEquals(4, matchedLookups); From bee6b6e6554f66eca9e695fce34279f9a00da50e Mon Sep 17 00:00:00 2001 From: Simon Kelly Date: Thu, 18 May 2023 09:34:09 +0200 Subject: [PATCH 5/5] covert field to local var --- .../org/commcare/formplayer/tests/CaseDbModelQueryTests.java | 3 +-- 1 file changed, 1 insertion(+), 2 deletions(-) diff --git a/src/test/java/org/commcare/formplayer/tests/CaseDbModelQueryTests.java b/src/test/java/org/commcare/formplayer/tests/CaseDbModelQueryTests.java index 16e10e79f..90546fc9a 100644 --- a/src/test/java/org/commcare/formplayer/tests/CaseDbModelQueryTests.java +++ b/src/test/java/org/commcare/formplayer/tests/CaseDbModelQueryTests.java @@ -59,13 +59,12 @@ public class CaseDbModelQueryTests { @RegisterExtension static StorageFactoryExtension storageExt = new StorageFactoryExtension.builder() .withUser("back_nav").withDomain("back_nav").build(); - private UserSqlSandbox sandbox; private EvaluationContext evaluationContext; @BeforeEach public void setUp() { new SyncDbRequest(mockMvc, restoreFactory).request(); - sandbox = restoreFactory.getSqlSandbox(); + UserSqlSandbox sandbox = restoreFactory.getSqlSandbox(); evaluationContext = TestStorageUtils.getEvaluationContextWithoutSession(sandbox); evaluationContext.setDebugModeOn(); }