This is an automated email from the ASF dual-hosted git repository.
gongchao pushed a commit to branch master
in repository https://gitbox.apache.org/repos/asf/hertzbeat.git
The following commit(s) were added to refs/heads/master by this push:
new 1b51c19c97 [fix]fixed sql formatting error (#4008)
1b51c19c97 is described below
commit 1b51c19c972cdccd2970b71e72ed73dad8ff92a9
Author: Duansg <[email protected]>
AuthorDate: Fri Jan 30 00:00:31 2026 +0800
[fix]fixed sql formatting error (#4008)
Co-authored-by: Tomsun28 <[email protected]>
---
.../apache/hertzbeat/common/util/StrBuffer.java | 4 +
.../history/tsdb/tdengine/TdEngineDataStorage.java | 10 +-
.../warehouse/store/TdEngineDataStorageTest.java | 158 ++++++++++++++++++++-
3 files changed, 168 insertions(+), 4 deletions(-)
diff --git
a/hertzbeat-common/src/main/java/org/apache/hertzbeat/common/util/StrBuffer.java
b/hertzbeat-common/src/main/java/org/apache/hertzbeat/common/util/StrBuffer.java
index 21d06bfd12..c15bc33594 100644
---
a/hertzbeat-common/src/main/java/org/apache/hertzbeat/common/util/StrBuffer.java
+++
b/hertzbeat-common/src/main/java/org/apache/hertzbeat/common/util/StrBuffer.java
@@ -159,4 +159,8 @@ public class StrBuffer {
}
return Double.parseDouble(s);
}
+
+ public static String escapeForFormat(String value) {
+ return value.replace("%", "%%");
+ }
}
diff --git
a/hertzbeat-warehouse/src/main/java/org/apache/hertzbeat/warehouse/store/history/tsdb/tdengine/TdEngineDataStorage.java
b/hertzbeat-warehouse/src/main/java/org/apache/hertzbeat/warehouse/store/history/tsdb/tdengine/TdEngineDataStorage.java
index 85e905f31d..361d45dbbb 100644
---
a/hertzbeat-warehouse/src/main/java/org/apache/hertzbeat/warehouse/store/history/tsdb/tdengine/TdEngineDataStorage.java
+++
b/hertzbeat-warehouse/src/main/java/org/apache/hertzbeat/warehouse/store/history/tsdb/tdengine/TdEngineDataStorage.java
@@ -47,6 +47,7 @@ import org.apache.hertzbeat.common.entity.arrow.RowWrapper;
import org.apache.hertzbeat.common.entity.dto.Value;
import org.apache.hertzbeat.common.entity.message.CollectRep;
import org.apache.hertzbeat.common.util.JsonUtil;
+import org.apache.hertzbeat.common.util.StrBuffer;
import
org.apache.hertzbeat.warehouse.store.history.tsdb.AbstractHistoryDataStorage;
import org.jetbrains.annotations.NotNull;
import org.springframework.boot.autoconfigure.condition.ConditionalOnProperty;
@@ -221,7 +222,12 @@ public class TdEngineDataStorage extends
AbstractHistoryDataStorage {
} else {
try {
double number = Double.parseDouble(value);
- sqlRowBuffer.append(number);
+ // TDengine doesn't support NaN or Infinity
literals, convert to NULL
+ if (Double.isNaN(number) ||
Double.isInfinite(number)) {
+ sqlRowBuffer.append("NULL");
+ } else {
+ sqlRowBuffer.append(number);
+ }
} catch (Exception e) {
if (log.isWarnEnabled()) {
@@ -236,7 +242,7 @@ public class TdEngineDataStorage extends
AbstractHistoryDataStorage {
if (CommonConstants.NULL_VALUE.equals(value)) {
sqlRowBuffer.append("NULL");
} else {
-
sqlRowBuffer.append("'").append(formatStringValue(value)).append("'");
+
sqlRowBuffer.append("'").append(StrBuffer.escapeForFormat(formatStringValue(value))).append("'");
}
}
diff --git
a/hertzbeat-warehouse/src/test/java/org/apache/hertzbeat/warehouse/store/TdEngineDataStorageTest.java
b/hertzbeat-warehouse/src/test/java/org/apache/hertzbeat/warehouse/store/TdEngineDataStorageTest.java
index 2326191b63..97f94090c6 100644
---
a/hertzbeat-warehouse/src/test/java/org/apache/hertzbeat/warehouse/store/TdEngineDataStorageTest.java
+++
b/hertzbeat-warehouse/src/test/java/org/apache/hertzbeat/warehouse/store/TdEngineDataStorageTest.java
@@ -17,17 +17,67 @@
package org.apache.hertzbeat.warehouse.store;
+import com.zaxxer.hikari.HikariDataSource;
+import org.apache.arrow.vector.types.pojo.ArrowType;
+import org.apache.arrow.vector.types.pojo.Field;
+import org.apache.arrow.vector.types.pojo.FieldType;
+import org.apache.hertzbeat.common.constants.CommonConstants;
+import org.apache.hertzbeat.common.constants.MetricDataConstants;
+import org.apache.hertzbeat.common.entity.arrow.ArrowCell;
+import org.apache.hertzbeat.common.entity.arrow.RowWrapper;
+import org.apache.hertzbeat.common.entity.message.CollectRep;
import
org.apache.hertzbeat.warehouse.store.history.tsdb.tdengine.TdEngineDataStorage;
+import
org.apache.hertzbeat.warehouse.store.history.tsdb.tdengine.TdEngineProperties;
import org.junit.jupiter.api.BeforeEach;
import org.junit.jupiter.api.Test;
+import org.junit.jupiter.api.extension.ExtendWith;
+import org.mockito.ArgumentCaptor;
+import org.mockito.Mock;
+import org.mockito.Mockito;
+import org.mockito.junit.jupiter.MockitoExtension;
+import org.mockito.junit.jupiter.MockitoSettings;
+import org.mockito.quality.Strictness;
+
+import java.sql.Connection;
+import java.sql.Statement;
+import java.util.List;
+
+import static org.junit.jupiter.api.Assertions.assertNotNull;
+import static org.junit.jupiter.api.Assertions.assertTrue;
+import static org.mockito.Mockito.atLeastOnce;
+import static org.mockito.Mockito.verify;
+import static org.mockito.Mockito.when;
/**
* Test case for {@link TdEngineDataStorage}
*/
+@ExtendWith(MockitoExtension.class)
+@MockitoSettings(strictness = Strictness.LENIENT)
class TdEngineDataStorageTest {
+ @Mock
+ private TdEngineProperties tdEngineProperties;
+
+ @Mock
+ private HikariDataSource mockHikariDataSource;
+
+ @Mock
+ private Connection mockConnection;
+
+ @Mock
+ private Statement mockStatement;
+
+ private TdEngineDataStorage tdEngineDataStorage;
+
@BeforeEach
- void setUp() {
+ void setUp() throws Exception {
+ when(tdEngineProperties.enabled()).thenReturn(true);
+
when(tdEngineProperties.url()).thenReturn("jdbc:TAOS-RS://localhost:6041/demo");
+ when(tdEngineProperties.username()).thenReturn("root");
+ when(tdEngineProperties.password()).thenReturn("root");
+
when(tdEngineProperties.tableStrColumnDefineMaxLength()).thenReturn(200);
+ when(mockHikariDataSource.getConnection()).thenReturn(mockConnection);
+ when(mockConnection.createStatement()).thenReturn(mockStatement);
}
@Test
@@ -35,7 +85,34 @@ class TdEngineDataStorageTest {
}
@Test
- void saveData() {
+ void testSaveData() throws Exception {
+ tdEngineDataStorage = new TdEngineDataStorage(tdEngineProperties);
+ setPrivateField(tdEngineDataStorage, "hikariDataSource",
mockHikariDataSource);
+ setParentPrivateField(tdEngineDataStorage, "serverAvailable", true);
+
+ CollectRep.MetricsData metricsData = generateMockedMetricsData();
+ tdEngineDataStorage.saveData(metricsData);
+
+ ArgumentCaptor<String> sqlCaptor =
ArgumentCaptor.forClass(String.class);
+ verify(mockStatement, atLeastOnce()).execute(sqlCaptor.capture());
+
+ String executedSql = sqlCaptor.getValue();
+
+ // Verify SQL structure
+ assertNotNull(executedSql);
+ assertTrue(executedSql.startsWith("INSERT INTO"), "SQL should start
with INSERT INTO");
+ assertTrue(executedSql.contains("USING"), "SQL should contain USING
clause");
+ assertTrue(executedSql.contains("TAGS"), "SQL should contain TAGS
clause");
+ assertTrue(executedSql.contains("VALUES"), "SQL should contain VALUES
clause");
+
+ // Verify table name format: app_metrics_instance_v2
+ assertTrue(executedSql.contains("app_cpu_test-%server-01_v2"), "Should
contain correct table name");
+ // Verify super table name format: app_metrics_super_v2
+ assertTrue(executedSql.contains("app_cpu_super_v2"), "Should contain
correct super table name");
+ // Verify tags format: test-%server-01
+ assertTrue(executedSql.contains("TAGS ('test-%server-01')"), "Should
contain correct super table name");
+ // Verify VALUES clause structure (timestamp + data values)
+ assertTrue(executedSql.matches(".*VALUES\\s+\\(\\d+.*68\\.7\\)"),
"Should contain timestamp and value 68.7");
}
@Test
@@ -49,4 +126,81 @@ class TdEngineDataStorageTest {
@Test
void getHistoryIntervalMetricData() {
}
+
+ /**
+ * Helper method to set private field using reflection
+ */
+ private void setPrivateField(Object target, String fieldName, Object
value) throws Exception {
+ java.lang.reflect.Field field =
target.getClass().getDeclaredField(fieldName);
+ field.setAccessible(true);
+ field.set(target, value);
+ }
+
+ /**
+ * Helper method to set private field from parent class using reflection
+ */
+ private void setParentPrivateField(Object target, String fieldName, Object
value) throws Exception {
+ java.lang.reflect.Field field =
target.getClass().getSuperclass().getDeclaredField(fieldName);
+ field.setAccessible(true);
+ field.set(target, value);
+ }
+
+ public static CollectRep.MetricsData generateMockedMetricsData() {
+ CollectRep.MetricsData mockMetricsData =
Mockito.mock(CollectRep.MetricsData.class);
+
+ when(mockMetricsData.getId()).thenReturn(0L);
+ when(mockMetricsData.getMetrics()).thenReturn("cpu");
+ when(mockMetricsData.getTime()).thenReturn(System.currentTimeMillis());
+ when(mockMetricsData.getCode()).thenReturn(CollectRep.Code.SUCCESS);
+ when(mockMetricsData.getApp()).thenReturn("app");
+ when(mockMetricsData.getInstance()).thenReturn("test-%server-01");
+
+ CollectRep.ValueRow mockValueRow =
Mockito.mock(CollectRep.ValueRow.class);
+ List<String> columnValues = List.of("test-%server-01", "68.7");
+ when(mockValueRow.getColumnsList()).thenReturn(columnValues);
+ when(mockValueRow.getColumns(0)).thenReturn("test-%server-01");
+ when(mockValueRow.getColumns(1)).thenReturn("68.7");
+ List<CollectRep.ValueRow> mockValueRowsList = List.of(mockValueRow);
+ when(mockMetricsData.getValues()).thenReturn(mockValueRowsList);
+
+ CollectRep.Field instanceField = Mockito.mock(CollectRep.Field.class);
+ when(instanceField.getName()).thenReturn("instance");
+ CollectRep.Field usageField = Mockito.mock(CollectRep.Field.class);
+ when(usageField.getName()).thenReturn("usage");
+ CollectRep.Field systemField = Mockito.mock(CollectRep.Field.class);
+ when(systemField.getName()).thenReturn("system");
+ List<CollectRep.Field> mockFields = List.of(instanceField, usageField,
systemField);
+ when(mockMetricsData.getFields()).thenReturn(mockFields);
+
+ ArrowType instanceArrowType = new ArrowType.Utf8();
+ FieldType instanceFieldType = new FieldType(true, instanceArrowType,
null, null);
+ Field instanceArrowField = new Field("instance", instanceFieldType,
null);
+ ArrowCell instanceCell = Mockito.mock(ArrowCell.class);
+ when(instanceCell.getField()).thenReturn(instanceArrowField);
+ when(instanceCell.getValue()).thenReturn("test-%server-01");
+
when(instanceCell.getMetadataAsBoolean(MetricDataConstants.LABEL)).thenReturn(true);
+
when(instanceCell.getMetadataAsByte(MetricDataConstants.TYPE)).thenReturn(CommonConstants.TYPE_STRING);
+
+ ArrowType usageArrowType = new ArrowType.Utf8();
+ FieldType usageFieldType = new FieldType(true, usageArrowType, null,
null);
+ Field usageArrowField = new Field("usage", usageFieldType, null);
+ ArrowCell usageCell = Mockito.mock(ArrowCell.class);
+ when(usageCell.getField()).thenReturn(usageArrowField);
+ when(usageCell.getValue()).thenReturn("68.7");
+
when(usageCell.getMetadataAsBoolean(MetricDataConstants.LABEL)).thenReturn(false);
+
when(usageCell.getMetadataAsByte(MetricDataConstants.TYPE)).thenReturn(CommonConstants.TYPE_NUMBER);
+ List<ArrowCell> mockCells = List.of(instanceCell, usageCell);
+
+ // Create Arrow Field list for RowWrapper
+ List<org.apache.arrow.vector.types.pojo.Field> arrowFields =
List.of(instanceArrowField, usageArrowField);
+
+ RowWrapper mockRowWrapper = Mockito.mock(RowWrapper.class);
+ when(mockRowWrapper.hasNextRow()).thenReturn(true).thenReturn(false);
+ when(mockRowWrapper.nextRow()).thenReturn(mockRowWrapper);
+ when(mockRowWrapper.cellStream()).thenAnswer(invocation ->
mockCells.stream());
+ when(mockRowWrapper.getFieldList()).thenReturn(arrowFields);
+ when(mockMetricsData.readRow()).thenReturn(mockRowWrapper);
+ return mockMetricsData;
+ }
+
}
---------------------------------------------------------------------
To unsubscribe, e-mail: [email protected]
For additional commands, e-mail: [email protected]