From 2da391ea02943f9e44b318ddc9a678bde5ed36a2 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Lars=20Helge=20=C3=98verland?= Date: Sun, 15 Dec 2024 11:49:11 +0100 Subject: [PATCH 1/2] [DRAFT] fix: Analytics tables for ClickHouse [DHIS2-18417] (#19445) --- .../table/AbstractEventJdbcTableManager.java | 6 +- .../table/DefaultAnalyticsTableGenerator.java | 19 ++++++- .../table/JdbcEventAnalyticsTableManager.java | 14 ++++- ...dbcTrackedEntityAnalyticsTableManager.java | 4 +- .../table/setting/AnalyticsTableSettings.java | 2 +- .../DefaultTableReplicationService.java | 56 +++++++++++++++++++ .../TableReplicationService.java | 32 +++++++++++ .../{ => jdbc}/JdbcTableReplicationStore.java | 3 +- ...rackedEntityAnalyticsTableManagerTest.java | 4 -- 9 files changed, 127 insertions(+), 13 deletions(-) create mode 100644 dhis-2/dhis-services/dhis-service-analytics/src/main/java/org/hisp/dhis/tablereplication/DefaultTableReplicationService.java create mode 100644 dhis-2/dhis-services/dhis-service-analytics/src/main/java/org/hisp/dhis/tablereplication/TableReplicationService.java rename dhis-2/dhis-services/dhis-service-analytics/src/main/java/org/hisp/dhis/tablereplication/{ => jdbc}/JdbcTableReplicationStore.java (97%) diff --git a/dhis-2/dhis-services/dhis-service-analytics/src/main/java/org/hisp/dhis/analytics/table/AbstractEventJdbcTableManager.java b/dhis-2/dhis-services/dhis-service-analytics/src/main/java/org/hisp/dhis/analytics/table/AbstractEventJdbcTableManager.java index 31bd476ab638..508761d83d12 100644 --- a/dhis-2/dhis-services/dhis-service-analytics/src/main/java/org/hisp/dhis/analytics/table/AbstractEventJdbcTableManager.java +++ b/dhis-2/dhis-services/dhis-service-analytics/src/main/java/org/hisp/dhis/analytics/table/AbstractEventJdbcTableManager.java @@ -214,6 +214,10 @@ protected List getColumnForAttribute(TrackedEntityAttribut */ private List getColumnForOrgUnitAttribute( TrackedEntityAttribute attribute) { + if (!sqlBuilder.supportsCorrelatedSubquery()) { + return List.of(); + } + Validate.isTrue(attribute.getValueType().isOrganisationUnit()); List columns = new ArrayList<>(); @@ -283,7 +287,7 @@ private String getOrgUnitSelectSubquery(TrackedEntityAttribute attribute, String protected String getAttributeValueJoinClause(Program program) { String template = """ - left join ${trackedentityattributevalue} as ${uid} \ + left join trackedentityattributevalue as ${uid} \ on en.trackedentityid=${uid}.trackedentityid \ and ${uid}.trackedentityattributeid = ${id}\s"""; diff --git a/dhis-2/dhis-services/dhis-service-analytics/src/main/java/org/hisp/dhis/analytics/table/DefaultAnalyticsTableGenerator.java b/dhis-2/dhis-services/dhis-service-analytics/src/main/java/org/hisp/dhis/analytics/table/DefaultAnalyticsTableGenerator.java index 4abc5d9b0ae5..7bd45ff337b3 100644 --- a/dhis-2/dhis-services/dhis-service-analytics/src/main/java/org/hisp/dhis/analytics/table/DefaultAnalyticsTableGenerator.java +++ b/dhis-2/dhis-services/dhis-service-analytics/src/main/java/org/hisp/dhis/analytics/table/DefaultAnalyticsTableGenerator.java @@ -27,6 +27,9 @@ */ package org.hisp.dhis.analytics.table; +import static org.hisp.dhis.analytics.AnalyticsTableType.ENROLLMENT; +import static org.hisp.dhis.analytics.AnalyticsTableType.EVENT; +import static org.hisp.dhis.analytics.AnalyticsTableType.TRACKED_ENTITY_INSTANCE; import static org.hisp.dhis.common.collection.CollectionUtils.emptyIfNull; import static org.hisp.dhis.scheduling.JobProgress.FailurePolicy.SKIP_STAGE; import static org.hisp.dhis.util.DateUtils.toLongDate; @@ -51,6 +54,7 @@ import org.hisp.dhis.setting.SystemSettings; import org.hisp.dhis.setting.SystemSettingsService; import org.hisp.dhis.system.util.Clock; +import org.hisp.dhis.tablereplication.TableReplicationService; import org.springframework.stereotype.Service; /** @@ -64,6 +68,8 @@ public class DefaultAnalyticsTableGenerator implements AnalyticsTableGenerator { private final ResourceTableService resourceTableService; + private final TableReplicationService tableReplicationService; + private final SystemSettingsService settingsService; private final AnalyticsTableSettings settings; @@ -84,7 +90,8 @@ public void generateAnalyticsTables(AnalyticsTableUpdateParams params0, JobProgr log.info("Found analytics table types: {}", getAvailableTableTypes()); log.info("Analytics table update params: {}", params); log.info("Last successful analytics table update: {}", toLongDate(lastSuccessfulUpdate)); - log.debug("Skipping table types: {}", skipTypes); + log.info("Analytics database: {}", settings.isAnalyticsDatabase()); + log.info("Skipping table types: {}", skipTypes); progress.startingProcess( "Analytics table update process{}", (params.isLatestUpdate() ? " (latest partition)" : "")); @@ -92,15 +99,21 @@ public void generateAnalyticsTables(AnalyticsTableUpdateParams params0, JobProgr if (!params.isSkipResourceTables() && !params.isLatestUpdate()) { generateResourceTablesInternal(progress); - if (settings.isAnalyticsDatabaseConfigured()) { + if (settings.isAnalyticsDatabase()) { log.info("Replicating resource tables in analytics database"); resourceTableService.replicateAnalyticsResourceTables(); } } + if (!params.isLatestUpdate() && settings.isAnalyticsDatabase()) { + if (!skipTypes.containsAll(Set.of(EVENT, ENROLLMENT, TRACKED_ENTITY_INSTANCE))) { + log.info("Replicating tracked entity attribute value table"); + tableReplicationService.replicateTrackedEntityAttributeValue(); + } + } + for (AnalyticsTableService service : analyticsTableServices) { AnalyticsTableType tableType = service.getAnalyticsTableType(); - if (!skipTypes.contains(tableType)) { service.create(params, progress); } diff --git a/dhis-2/dhis-services/dhis-service-analytics/src/main/java/org/hisp/dhis/analytics/table/JdbcEventAnalyticsTableManager.java b/dhis-2/dhis-services/dhis-service-analytics/src/main/java/org/hisp/dhis/analytics/table/JdbcEventAnalyticsTableManager.java index 2d22956aaf24..577ca3925f8a 100644 --- a/dhis-2/dhis-services/dhis-service-analytics/src/main/java/org/hisp/dhis/analytics/table/JdbcEventAnalyticsTableManager.java +++ b/dhis-2/dhis-services/dhis-service-analytics/src/main/java/org/hisp/dhis/analytics/table/JdbcEventAnalyticsTableManager.java @@ -534,6 +534,10 @@ private String getSelectExpression(DataElement dataElement, String columnExpress * @return a list of {@link AnalyticsTableColumn}. */ private List getColumnForOrgUnitDataElement(DataElement dataElement) { + if (!sqlBuilder.supportsCorrelatedSubquery()) { + return List.of(); + } + List columns = new ArrayList<>(); if (isSpatialSupport()) { @@ -612,12 +616,16 @@ private List getAttributeColumns(Program program) { */ private List getColumnForAttributeWithLegendSet( TrackedEntityAttribute attribute) { + if (!sqlBuilder.supportsCorrelatedSubquery()) { + return List.of(); + } + String columnExpression = getColumnExpression(attribute.getValueType(), "value"); String numericClause = getNumericClause("value"); String query = """ \s(select l.uid from ${maplegend} l \ - inner join ${trackedentityattributevalue} av on l.startvalue <= ${selectClause} \ + inner join trackedentityattributevalue av on l.startvalue <= ${selectClause} \ and l.endvalue > ${selectClause} \ and l.maplegendsetid=${legendSetId} \ and av.trackedentityid=en.trackedentityid \ @@ -656,6 +664,10 @@ private List getColumnForAttributeWithLegendSet( */ private List getColumnFromDataElementWithLegendSet( DataElement dataElement, String selectExpression, String dataFilterClause) { + if (!sqlBuilder.supportsCorrelatedSubquery()) { + return List.of(); + } + String query = """ (select l.uid from ${maplegend} l \ diff --git a/dhis-2/dhis-services/dhis-service-analytics/src/main/java/org/hisp/dhis/analytics/table/JdbcTrackedEntityAnalyticsTableManager.java b/dhis-2/dhis-services/dhis-service-analytics/src/main/java/org/hisp/dhis/analytics/table/JdbcTrackedEntityAnalyticsTableManager.java index 8d6fc1f302d6..ceecc38220db 100644 --- a/dhis-2/dhis-services/dhis-service-analytics/src/main/java/org/hisp/dhis/analytics/table/JdbcTrackedEntityAnalyticsTableManager.java +++ b/dhis-2/dhis-services/dhis-service-analytics/src/main/java/org/hisp/dhis/analytics/table/JdbcTrackedEntityAnalyticsTableManager.java @@ -179,7 +179,7 @@ private List getColumns( List columns = new ArrayList<>(getFixedColumns()); List programs = programsByTetUid.get(trackedEntityType.getUid()); - if (isNotEmpty(programs)) { + if (isNotEmpty(programs) && sqlBuilder.supportsCorrelatedSubquery()) { String enrolledInProgramExpression = """ \s exists(select 1 from ${enrollment} en_0 \ @@ -320,7 +320,7 @@ public void populateTable(AnalyticsTableUpdateParams params, AnalyticsTableParti sql.append( replaceQualify( """ - \s left join ${trackedentityattributevalue} ${teaUid} on ${teaUid}.trackedentityid=te.trackedentityid \ + \s left join trackedentityattributevalue ${teaUid} on ${teaUid}.trackedentityid=te.trackedentityid \ and ${teaUid}.trackedentityattributeid = ${teaId}""", Map.of( "teaUid", quote(tea.getUid()), diff --git a/dhis-2/dhis-services/dhis-service-analytics/src/main/java/org/hisp/dhis/analytics/table/setting/AnalyticsTableSettings.java b/dhis-2/dhis-services/dhis-service-analytics/src/main/java/org/hisp/dhis/analytics/table/setting/AnalyticsTableSettings.java index 645b1aef8154..e77f44a4ca82 100644 --- a/dhis-2/dhis-services/dhis-service-analytics/src/main/java/org/hisp/dhis/analytics/table/setting/AnalyticsTableSettings.java +++ b/dhis-2/dhis-services/dhis-service-analytics/src/main/java/org/hisp/dhis/analytics/table/setting/AnalyticsTableSettings.java @@ -111,7 +111,7 @@ public PeriodSource getPeriodSource() { * * @return true if an analytics database instance is configured. */ - public boolean isAnalyticsDatabaseConfigured() { + public boolean isAnalyticsDatabase() { return config.isAnalyticsDatabaseConfigured(); } diff --git a/dhis-2/dhis-services/dhis-service-analytics/src/main/java/org/hisp/dhis/tablereplication/DefaultTableReplicationService.java b/dhis-2/dhis-services/dhis-service-analytics/src/main/java/org/hisp/dhis/tablereplication/DefaultTableReplicationService.java new file mode 100644 index 000000000000..b03948fa73da --- /dev/null +++ b/dhis-2/dhis-services/dhis-service-analytics/src/main/java/org/hisp/dhis/tablereplication/DefaultTableReplicationService.java @@ -0,0 +1,56 @@ +/* + * Copyright (c) 2004-2024, University of Oslo + * All rights reserved. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions are met: + * Redistributions of source code must retain the above copyright notice, this + * list of conditions and the following disclaimer. + * + * Redistributions in binary form must reproduce the above copyright notice, + * this list of conditions and the following disclaimer in the documentation + * and/or other materials provided with the distribution. + * Neither the name of the HISP project nor the names of its contributors may + * be used to endorse or promote products derived from this software without + * specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND + * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED + * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE + * DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR + * ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES + * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; + * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON + * ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT + * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS + * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. + */ +package org.hisp.dhis.tablereplication; + +import java.util.List; +import lombok.RequiredArgsConstructor; +import org.hisp.dhis.db.model.Column; +import org.hisp.dhis.db.model.DataType; +import org.hisp.dhis.db.model.Table; +import org.hisp.dhis.db.model.constraint.Nullable; +import org.springframework.stereotype.Service; + +@Service +@RequiredArgsConstructor +public class DefaultTableReplicationService implements TableReplicationService { + private static final Table TABLE_TRACKED_ENTIY_ATTRIBUTE_VALUE = + new Table( + "trackedentityattributevalue", + List.of( + new Column("trackedentityid", DataType.BIGINT, Nullable.NOT_NULL), + new Column("trackedentityattributeid", DataType.BIGINT, Nullable.NOT_NULL), + new Column("value", DataType.TEXT, Nullable.NULL)), + List.of("trackedentityid", "trackedentityattributeid")); + + private final TableReplicationStore store; + + @Override + public void replicateTrackedEntityAttributeValue() { + store.replicateAnalyticsDatabaseTable(TABLE_TRACKED_ENTIY_ATTRIBUTE_VALUE); + } +} diff --git a/dhis-2/dhis-services/dhis-service-analytics/src/main/java/org/hisp/dhis/tablereplication/TableReplicationService.java b/dhis-2/dhis-services/dhis-service-analytics/src/main/java/org/hisp/dhis/tablereplication/TableReplicationService.java new file mode 100644 index 000000000000..bdc514c82b7d --- /dev/null +++ b/dhis-2/dhis-services/dhis-service-analytics/src/main/java/org/hisp/dhis/tablereplication/TableReplicationService.java @@ -0,0 +1,32 @@ +/* + * Copyright (c) 2004-2024, University of Oslo + * All rights reserved. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions are met: + * Redistributions of source code must retain the above copyright notice, this + * list of conditions and the following disclaimer. + * + * Redistributions in binary form must reproduce the above copyright notice, + * this list of conditions and the following disclaimer in the documentation + * and/or other materials provided with the distribution. + * Neither the name of the HISP project nor the names of its contributors may + * be used to endorse or promote products derived from this software without + * specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND + * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED + * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE + * DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR + * ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES + * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; + * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON + * ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT + * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS + * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. + */ +package org.hisp.dhis.tablereplication; + +public interface TableReplicationService { + void replicateTrackedEntityAttributeValue(); +} diff --git a/dhis-2/dhis-services/dhis-service-analytics/src/main/java/org/hisp/dhis/tablereplication/JdbcTableReplicationStore.java b/dhis-2/dhis-services/dhis-service-analytics/src/main/java/org/hisp/dhis/tablereplication/jdbc/JdbcTableReplicationStore.java similarity index 97% rename from dhis-2/dhis-services/dhis-service-analytics/src/main/java/org/hisp/dhis/tablereplication/JdbcTableReplicationStore.java rename to dhis-2/dhis-services/dhis-service-analytics/src/main/java/org/hisp/dhis/tablereplication/jdbc/JdbcTableReplicationStore.java index aa5b275c879e..4bddfa16eb33 100644 --- a/dhis-2/dhis-services/dhis-service-analytics/src/main/java/org/hisp/dhis/tablereplication/JdbcTableReplicationStore.java +++ b/dhis-2/dhis-services/dhis-service-analytics/src/main/java/org/hisp/dhis/tablereplication/jdbc/JdbcTableReplicationStore.java @@ -25,13 +25,14 @@ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. */ -package org.hisp.dhis.tablereplication; +package org.hisp.dhis.tablereplication.jdbc; import lombok.RequiredArgsConstructor; import lombok.extern.slf4j.Slf4j; import org.hisp.dhis.db.model.Table; import org.hisp.dhis.db.sql.SqlBuilder; import org.hisp.dhis.system.util.Clock; +import org.hisp.dhis.tablereplication.TableReplicationStore; import org.springframework.beans.factory.annotation.Qualifier; import org.springframework.jdbc.core.JdbcTemplate; import org.springframework.stereotype.Service; diff --git a/dhis-2/dhis-services/dhis-service-analytics/src/test/java/org/hisp/dhis/analytics/table/JdbcTrackedEntityAnalyticsTableManagerTest.java b/dhis-2/dhis-services/dhis-service-analytics/src/test/java/org/hisp/dhis/analytics/table/JdbcTrackedEntityAnalyticsTableManagerTest.java index 0d537f2e63da..d70a6e25b7bf 100644 --- a/dhis-2/dhis-services/dhis-service-analytics/src/test/java/org/hisp/dhis/analytics/table/JdbcTrackedEntityAnalyticsTableManagerTest.java +++ b/dhis-2/dhis-services/dhis-service-analytics/src/test/java/org/hisp/dhis/analytics/table/JdbcTrackedEntityAnalyticsTableManagerTest.java @@ -51,7 +51,6 @@ import org.hisp.dhis.program.Program; import org.hisp.dhis.resourcetable.ResourceTableService; import org.hisp.dhis.setting.SystemSettingsProvider; -import org.hisp.dhis.system.util.SqlUtils; import org.hisp.dhis.trackedentity.TrackedEntityAttribute; import org.hisp.dhis.trackedentity.TrackedEntityAttributeService; import org.hisp.dhis.trackedentity.TrackedEntityType; @@ -116,9 +115,6 @@ void verifyNonConfidentialTeasAreSkipped() { Program program = mock(Program.class); - when(sqlBuilder.qualifyTable(anyString())) - .thenAnswer(inv -> SqlUtils.quote(inv.getArgument(0))); - when(sqlBuilder.jsonExtract(anyString(), anyString())).thenReturn("jsonExtract"); when(sqlBuilder.coalesce(anyString(), anyString())) From 9de881bacc7dc61d15b417105c3ef56464d682ad Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Lars=20Helge=20=C3=98verland?= Date: Sun, 15 Dec 2024 16:07:14 +0100 Subject: [PATCH 2/2] refactor: Rename PostgresAnalyticsSqlBuilder [DHIS2-16705] (#19469) --- .../AbstractJdbcEventAnalyticsManager.java | 18 ++++------- .../table/model/AnalyticsTableColumn.java | 4 --- .../query/context/sql/SqlQueryBuilders.java | 32 +++++++++---------- .../dhis/analytics/util/DisplayNameUtils.java | 12 +++---- ...EntityEventsAnalyticsTableManagerTest.java | 4 +-- .../dhis/db/AnalyticsSqlBuilderProvider.java | 4 +-- ...ava => PostgreSqlAnalyticsSqlBuilder.java} | 2 +- .../hisp/dhis/db/sql/PostgreSqlBuilder.java | 4 +-- 8 files changed, 34 insertions(+), 46 deletions(-) rename dhis-2/dhis-support/dhis-support-sql/src/main/java/org/hisp/dhis/db/sql/{PostgresAnalyticsSqlBuilder.java => PostgreSqlAnalyticsSqlBuilder.java} (97%) diff --git a/dhis-2/dhis-services/dhis-service-analytics/src/main/java/org/hisp/dhis/analytics/event/data/AbstractJdbcEventAnalyticsManager.java b/dhis-2/dhis-services/dhis-service-analytics/src/main/java/org/hisp/dhis/analytics/event/data/AbstractJdbcEventAnalyticsManager.java index 9f53bfd51add..55a58883bd63 100644 --- a/dhis-2/dhis-services/dhis-service-analytics/src/main/java/org/hisp/dhis/analytics/event/data/AbstractJdbcEventAnalyticsManager.java +++ b/dhis-2/dhis-services/dhis-service-analytics/src/main/java/org/hisp/dhis/analytics/event/data/AbstractJdbcEventAnalyticsManager.java @@ -49,8 +49,8 @@ import static org.hisp.dhis.analytics.event.data.EnrollmentQueryHelper.getHeaderColumns; import static org.hisp.dhis.analytics.event.data.EnrollmentQueryHelper.getOrgUnitLevelColumns; import static org.hisp.dhis.analytics.event.data.EnrollmentQueryHelper.getPeriodColumns; -import static org.hisp.dhis.analytics.table.JdbcEventAnalyticsTableManager.OU_GEOMETRY_COL_SUFFIX; -import static org.hisp.dhis.analytics.table.JdbcEventAnalyticsTableManager.OU_NAME_COL_SUFFIX; +import static org.hisp.dhis.analytics.table.AbstractEventJdbcTableManager.OU_GEOMETRY_COL_SUFFIX; +import static org.hisp.dhis.analytics.table.AbstractEventJdbcTableManager.OU_NAME_COL_SUFFIX; import static org.hisp.dhis.analytics.util.AnalyticsUtils.replaceStringBetween; import static org.hisp.dhis.analytics.util.AnalyticsUtils.throwIllegalQueryEx; import static org.hisp.dhis.analytics.util.AnalyticsUtils.withExceptionHandling; @@ -375,8 +375,9 @@ private void addDimensionSelectColumns( singleQuote(period.getIsoDate()) + " as " + period.getPeriodType().getName()); } else { throw new IllegalStateException( - "Program indicator non-default boundary query must have " - + "exactly one period, or no periods and a period filter"); + """ + Program indicator non-default boundary query must have \" + exactly one period, or no periods and a period filter"""); } }); } @@ -455,7 +456,6 @@ private ColumnAndAlias getColumnAndAlias( } else if (queryItem.getValueType() == ValueType.NUMBER && !isGroupByClause) { ColumnAndAlias columnAndAlias = getColumnAndAlias(queryItem, isAggregated, queryItem.getItemName()); - return ColumnAndAlias.ofColumnAndAlias( columnAndAlias.getColumn(), defaultIfNull(columnAndAlias.getAlias(), queryItem.getItemName())); @@ -532,14 +532,10 @@ protected Optional getAlias(QueryItem queryItem) { @Transactional(readOnly = true, propagation = REQUIRES_NEW) public Grid getAggregatedEventData(EventQueryParams params, Grid grid, int maxLimit) { String aggregateClause = getAggregateClause(params); + String columns = StringUtils.join(getSelectColumns(params, true), ","); String sql = - TextUtils.removeLastComma( - "select " - + aggregateClause - + " as value," - + StringUtils.join(getSelectColumns(params, true), ",") - + " "); + TextUtils.removeLastComma("select " + aggregateClause + " as value," + columns + " "); // --------------------------------------------------------------------- // Criteria diff --git a/dhis-2/dhis-services/dhis-service-analytics/src/main/java/org/hisp/dhis/analytics/table/model/AnalyticsTableColumn.java b/dhis-2/dhis-services/dhis-service-analytics/src/main/java/org/hisp/dhis/analytics/table/model/AnalyticsTableColumn.java index 537b473493e5..fa49efb4500f 100644 --- a/dhis-2/dhis-services/dhis-service-analytics/src/main/java/org/hisp/dhis/analytics/table/model/AnalyticsTableColumn.java +++ b/dhis-2/dhis-services/dhis-service-analytics/src/main/java/org/hisp/dhis/analytics/table/model/AnalyticsTableColumn.java @@ -80,10 +80,6 @@ public class AnalyticsTableColumn { /** Date of creation of the underlying data dimension. */ private final Date created; - // // ------------------------------------------------------------------------- - // // Logic - // // ------------------------------------------------------------------------- - /** Indicates whether this column is not null. */ public boolean isNotNull() { return Nullable.NOT_NULL == nullable; diff --git a/dhis-2/dhis-services/dhis-service-analytics/src/main/java/org/hisp/dhis/analytics/trackedentity/query/context/sql/SqlQueryBuilders.java b/dhis-2/dhis-services/dhis-service-analytics/src/main/java/org/hisp/dhis/analytics/trackedentity/query/context/sql/SqlQueryBuilders.java index e9768189d49c..23ae04e3abd6 100644 --- a/dhis-2/dhis-services/dhis-service-analytics/src/main/java/org/hisp/dhis/analytics/trackedentity/query/context/sql/SqlQueryBuilders.java +++ b/dhis-2/dhis-services/dhis-service-analytics/src/main/java/org/hisp/dhis/analytics/trackedentity/query/context/sql/SqlQueryBuilders.java @@ -61,22 +61,22 @@ select json_agg(json_build_object( // incidentDate is kept to support a deprecated field, will be removed when FE will only // use occurreddate """ - select json_agg( - json_build_object( - 'programUid', en.program, - 'enrollmentUid', en.enrollment, - 'enrollmentDate', en.enrollmentdate, - 'incidentDate', en.occurreddate, - 'occurredDate', en.occurreddate, - 'completedDate', en.completeddate, - 'orgUnitUid', en.ou, - 'orgUnitName', en.ouname, - 'orgUnitCode', en.oucode, - 'orgUnitNameHierarchy', en.ounamehierarchy, - 'enrollmentStatus', en.enrollmentstatus, - 'events', ${eventQuery})) - from analytics_te_enrollment_${trackedEntityType} en - where en.trackedentity = t_1.trackedentity""", + select json_agg( + json_build_object( + 'programUid', en.program, + 'enrollmentUid', en.enrollment, + 'enrollmentDate', en.enrollmentdate, + 'incidentDate', en.occurreddate, + 'occurredDate', en.occurreddate, + 'completedDate', en.completeddate, + 'orgUnitUid', en.ou, + 'orgUnitName', en.ouname, + 'orgUnitCode', en.oucode, + 'orgUnitNameHierarchy', en.ounamehierarchy, + 'enrollmentStatus', en.enrollmentstatus, + 'events', ${eventQuery})) + from analytics_te_enrollment_${trackedEntityType} en + where en.trackedentity = t_1.trackedentity""", Map.of("eventQuery", coalesceToEmptyArray(EVENT_QUERY))); private static final String JSON_AGGREGATION_QUERY = coalesceToEmptyArray(ENROLLMENT_QUERY); diff --git a/dhis-2/dhis-services/dhis-service-analytics/src/main/java/org/hisp/dhis/analytics/util/DisplayNameUtils.java b/dhis-2/dhis-services/dhis-service-analytics/src/main/java/org/hisp/dhis/analytics/util/DisplayNameUtils.java index 5377f0807253..b7f47fbce85d 100644 --- a/dhis-2/dhis-services/dhis-service-analytics/src/main/java/org/hisp/dhis/analytics/util/DisplayNameUtils.java +++ b/dhis-2/dhis-services/dhis-service-analytics/src/main/java/org/hisp/dhis/analytics/util/DisplayNameUtils.java @@ -124,22 +124,22 @@ public static String getDisplayName( isNotEmpty.apply(surname), isNotEmpty.apply(firstName), isEmpty.apply(username), - formatNames(sqlBuilder, surname, "', '", firstName), + sqlBuilder.concat(surname, "', '", firstName), // FirstName and Username isEmpty.apply(surname), isNotEmpty.apply(firstName), isNotEmpty.apply(username), - formatNames(sqlBuilder, firstName, "' ('", username, "')'"), + sqlBuilder.concat(firstName, "' ('", username, "')'"), // Surname and Username isNotEmpty.apply(surname), isEmpty.apply(firstName), isNotEmpty.apply(username), - formatNames(sqlBuilder, surname, "' ('", username, "')'"), + sqlBuilder.concat(surname, "' ('", username, "')'"), // All fields - formatNames(sqlBuilder, surname, "', '", firstName, "' ('", username, "')'"), + sqlBuilder.concat(surname, "', '", firstName, "' ('", username, "')'"), columnAlias); } @@ -149,8 +149,4 @@ private static String extractJsonValue( String jsonExtracted = sqlBuilder.jsonExtract(json, path); return sqlBuilder.trim(jsonExtracted); } - - private static String formatNames(SqlBuilder sqlBuilder, String... elements) { - return sqlBuilder.concat(elements); - } } diff --git a/dhis-2/dhis-services/dhis-service-analytics/src/test/java/org/hisp/dhis/analytics/table/JdbcTrackedEntityEventsAnalyticsTableManagerTest.java b/dhis-2/dhis-services/dhis-service-analytics/src/test/java/org/hisp/dhis/analytics/table/JdbcTrackedEntityEventsAnalyticsTableManagerTest.java index 57fce6c172ac..bc36af519d40 100644 --- a/dhis-2/dhis-services/dhis-service-analytics/src/test/java/org/hisp/dhis/analytics/table/JdbcTrackedEntityEventsAnalyticsTableManagerTest.java +++ b/dhis-2/dhis-services/dhis-service-analytics/src/test/java/org/hisp/dhis/analytics/table/JdbcTrackedEntityEventsAnalyticsTableManagerTest.java @@ -52,8 +52,8 @@ import org.hisp.dhis.category.CategoryService; import org.hisp.dhis.common.IdentifiableObjectManager; import org.hisp.dhis.dataapproval.DataApprovalLevelService; +import org.hisp.dhis.db.sql.PostgreSqlAnalyticsSqlBuilder; import org.hisp.dhis.db.sql.PostgreSqlBuilder; -import org.hisp.dhis.db.sql.PostgresAnalyticsSqlBuilder; import org.hisp.dhis.organisationunit.OrganisationUnitService; import org.hisp.dhis.period.PeriodDataProvider; import org.hisp.dhis.resourcetable.ResourceTableService; @@ -82,7 +82,7 @@ class JdbcTrackedEntityEventsAnalyticsTableManagerTest { @Spy private PostgreSqlBuilder sqlBuilder; - @Spy private PostgresAnalyticsSqlBuilder analyticsSqlBuilder; + @Spy private PostgreSqlAnalyticsSqlBuilder analyticsSqlBuilder; @Mock private PartitionManager partitionManager; diff --git a/dhis-2/dhis-support/dhis-support-sql/src/main/java/org/hisp/dhis/db/AnalyticsSqlBuilderProvider.java b/dhis-2/dhis-support/dhis-support-sql/src/main/java/org/hisp/dhis/db/AnalyticsSqlBuilderProvider.java index 9f7aac27027c..3c6728876f0d 100644 --- a/dhis-2/dhis-support/dhis-support-sql/src/main/java/org/hisp/dhis/db/AnalyticsSqlBuilderProvider.java +++ b/dhis-2/dhis-support/dhis-support-sql/src/main/java/org/hisp/dhis/db/AnalyticsSqlBuilderProvider.java @@ -33,7 +33,7 @@ import org.hisp.dhis.db.sql.AnalyticsSqlBuilder; import org.hisp.dhis.db.sql.ClickhouseAnalyticsSqlBuilder; import org.hisp.dhis.db.sql.DorisAnalyticsSqlBuilder; -import org.hisp.dhis.db.sql.PostgresAnalyticsSqlBuilder; +import org.hisp.dhis.db.sql.PostgreSqlAnalyticsSqlBuilder; import org.hisp.dhis.external.conf.DhisConfigurationProvider; import org.springframework.stereotype.Service; @@ -70,7 +70,7 @@ private AnalyticsSqlBuilder getSqlBuilder(SqlBuilderSettings config) { return switch (database) { case DORIS -> new DorisAnalyticsSqlBuilder(); case CLICKHOUSE -> new ClickhouseAnalyticsSqlBuilder(); - default -> new PostgresAnalyticsSqlBuilder(); + default -> new PostgreSqlAnalyticsSqlBuilder(); }; } } diff --git a/dhis-2/dhis-support/dhis-support-sql/src/main/java/org/hisp/dhis/db/sql/PostgresAnalyticsSqlBuilder.java b/dhis-2/dhis-support/dhis-support-sql/src/main/java/org/hisp/dhis/db/sql/PostgreSqlAnalyticsSqlBuilder.java similarity index 97% rename from dhis-2/dhis-support/dhis-support-sql/src/main/java/org/hisp/dhis/db/sql/PostgresAnalyticsSqlBuilder.java rename to dhis-2/dhis-support/dhis-support-sql/src/main/java/org/hisp/dhis/db/sql/PostgreSqlAnalyticsSqlBuilder.java index 1e1e5dd5fbea..115b7b022c30 100644 --- a/dhis-2/dhis-support/dhis-support-sql/src/main/java/org/hisp/dhis/db/sql/PostgresAnalyticsSqlBuilder.java +++ b/dhis-2/dhis-support/dhis-support-sql/src/main/java/org/hisp/dhis/db/sql/PostgreSqlAnalyticsSqlBuilder.java @@ -27,7 +27,7 @@ */ package org.hisp.dhis.db.sql; -public class PostgresAnalyticsSqlBuilder implements AnalyticsSqlBuilder { +public class PostgreSqlAnalyticsSqlBuilder implements AnalyticsSqlBuilder { /** * Returns a subquery that expand the event datavalue jsonb with two additional fields: diff --git a/dhis-2/dhis-support/dhis-support-sql/src/main/java/org/hisp/dhis/db/sql/PostgreSqlBuilder.java b/dhis-2/dhis-support/dhis-support-sql/src/main/java/org/hisp/dhis/db/sql/PostgreSqlBuilder.java index 92a14270f8cc..df43166bd281 100644 --- a/dhis-2/dhis-support/dhis-support-sql/src/main/java/org/hisp/dhis/db/sql/PostgreSqlBuilder.java +++ b/dhis-2/dhis-support/dhis-support-sql/src/main/java/org/hisp/dhis/db/sql/PostgreSqlBuilder.java @@ -278,6 +278,8 @@ public String dateDifference(String startDate, String endDate, DateUnit dateUnit String.format( "(extract(epoch from (cast(%s as timestamp) - cast(%s as timestamp))) / 60)", endDate, startDate); + case WEEKS -> + String.format("((cast(%s as date) - cast(%s as date)) / 7)", endDate, startDate); case MONTHS -> String.format( "((date_part('year',age(cast(%s as date), cast(%s as date)))) * 12 + " @@ -286,8 +288,6 @@ public String dateDifference(String startDate, String endDate, DateUnit dateUnit case YEARS -> String.format( "(date_part('year',age(cast(%s as date), cast(%s as date))))", endDate, startDate); - case WEEKS -> - String.format("((cast(%s as date) - cast(%s as date)) / 7)", endDate, startDate); }; }