diff --git a/arex-schedule-web-api/pom.xml b/arex-schedule-web-api/pom.xml
index c8bb5bb1..3097f54f 100644
--- a/arex-schedule-web-api/pom.xml
+++ b/arex-schedule-web-api/pom.xml
@@ -136,7 +136,7 @@
arex-schedule-parent
com.arextest
- 1.2.15
+ 1.2.19
@@ -338,5 +338,5 @@
- 1.2.15
+ 1.2.19
\ No newline at end of file
diff --git a/arex-schedule-web-api/src/main/java/com/arextest/schedule/client/HttpWepServiceApiClient.java b/arex-schedule-web-api/src/main/java/com/arextest/schedule/client/HttpWepServiceApiClient.java
index 3e70cbff..99544cf6 100644
--- a/arex-schedule-web-api/src/main/java/com/arextest/schedule/client/HttpWepServiceApiClient.java
+++ b/arex-schedule-web-api/src/main/java/com/arextest/schedule/client/HttpWepServiceApiClient.java
@@ -1,6 +1,7 @@
package com.arextest.schedule.client;
import static com.arextest.schedule.common.CommonConstant.URL;
+import static org.springframework.http.HttpHeaders.CONTENT_TYPE;
import com.arextest.schedule.utils.SSLUtils;
import com.fasterxml.jackson.databind.ObjectMapper;
@@ -74,6 +75,7 @@ public final class HttpWepServiceApiClient {
@Autowired(required = false)
private List clientHttpRequestInterceptors;
+ private static final String ZSTD_JSON_CONTENT_TYPE = "application/zstd-json;charset=UTF-8";
@PostConstruct
private void initTemplate() {
@@ -237,6 +239,18 @@ public TResponse retryJsonPost(String url, TRequest reques
}
}
+ public TResponse retryJsonPost(String url, TRequest request,
+ Class responseType, Map headers) {
+ try {
+ return retryTemplate.execute(retryCallback -> {
+ retryCallback.setAttribute(URL, url);
+ return restTemplate.postForObject(url, wrapJsonContentType(request, headers), responseType);
+ });
+ } catch (Exception e) {
+ return null;
+ }
+ }
+
public ResponseEntity retryJsonPost(String url, TRequest request,
ParameterizedTypeReference responseType) {
try {
@@ -250,6 +264,21 @@ public ResponseEntity retryJsonPost(String url,
}
}
+ public TResponse retryZstdJsonPost(String url, TRequest request,
+ Class responseType) {
+ Map headers = Maps.newHashMapWithExpectedSize(1);
+ headers.put(HttpHeaders.CONTENT_TYPE, ZSTD_JSON_CONTENT_TYPE);
+
+ try {
+ return retryTemplate.execute(retryCallback -> {
+ retryCallback.setAttribute(URL, url);
+ return restTemplate.postForObject(url, wrapJsonContentType(request, headers), responseType);
+ });
+ } catch (Exception e) {
+ return null;
+ }
+ }
+
@SuppressWarnings("unchecked")
private HttpEntity wrapJsonContentType(TRequest request) {
HttpEntity httpJsonEntity;
@@ -271,7 +300,9 @@ private HttpEntity wrapJsonContentType(TRequest request,
httpJsonEntity = (HttpEntity) request;
} else {
HttpHeaders headers = new HttpHeaders();
- headers.setContentType(MediaType.APPLICATION_JSON);
+ if (!extraHeaders.containsKey(CONTENT_TYPE)) {
+ headers.setContentType(MediaType.APPLICATION_JSON);
+ }
headers.setAll(extraHeaders);
httpJsonEntity = new HttpEntity<>(request, headers);
}
diff --git a/arex-schedule-web-api/src/main/java/com/arextest/schedule/client/ZstdJacksonMessageConverter.java b/arex-schedule-web-api/src/main/java/com/arextest/schedule/client/ZstdJacksonMessageConverter.java
index 4c4cb7ce..5db1b764 100644
--- a/arex-schedule-web-api/src/main/java/com/arextest/schedule/client/ZstdJacksonMessageConverter.java
+++ b/arex-schedule-web-api/src/main/java/com/arextest/schedule/client/ZstdJacksonMessageConverter.java
@@ -29,7 +29,7 @@ public ZstdJacksonMessageConverter() {
@Override
protected boolean supports(Class> clazz) {
- return true;
+ return !(clazz == byte[].class || clazz.isPrimitive());
}
@Override
@@ -41,6 +41,6 @@ protected Object readInternal(Class> clazz, HttpInputMessage inputMessage) thr
@Override
protected void writeInternal(Object o, HttpOutputMessage outputMessage) throws IOException,
HttpMessageNotWritableException {
- zstdJacksonSerializer.serializeTo(o, outputMessage.getBody());
+ outputMessage.getBody().write(zstdJacksonSerializer.serialize(o));
}
}
\ No newline at end of file
diff --git a/arex-schedule-web-api/src/main/java/com/arextest/schedule/common/JsonUtils.java b/arex-schedule-web-api/src/main/java/com/arextest/schedule/common/JsonUtils.java
index f9f9ede4..f9396efe 100644
--- a/arex-schedule-web-api/src/main/java/com/arextest/schedule/common/JsonUtils.java
+++ b/arex-schedule-web-api/src/main/java/com/arextest/schedule/common/JsonUtils.java
@@ -13,20 +13,29 @@
public class JsonUtils {
private static final ObjectMapper objectMapper = new ObjectMapper();
- public static T byteToObject(byte[] bytes, Class tClass) {
+ public static T byteToObject(byte[] bytes, Class tClazz) {
try {
- return objectMapper.readValue(bytes, tClass);
+ return objectMapper.readValue(bytes, tClazz);
} catch (IOException e) {
LOGGER.error("byteToObject error:{}", e.getMessage(), e);
}
return null;
}
+ public static T jsonStringToObject(String string, Class tClazz) {
+ try {
+ return objectMapper.readValue(string, tClazz);
+ } catch (IOException e) {
+ LOGGER.error("jsonStringToObject error:{}", e.getMessage(), e);
+ }
+ return null;
+ }
+
public static String objectToJsonString(Object value) {
try {
return objectMapper.writeValueAsString(value);
} catch (IOException e) {
- LOGGER.error("byteToObject error:{}", e.getMessage(), e);
+ LOGGER.error("objectToJsonString error:{}", e.getMessage(), e);
}
return StringUtils.EMPTY;
}
diff --git a/arex-schedule-web-api/src/main/java/com/arextest/schedule/comparer/CategoryComparisonHolder.java b/arex-schedule-web-api/src/main/java/com/arextest/schedule/comparer/CategoryComparisonHolder.java
index 17a1b2b8..c5f2c37c 100644
--- a/arex-schedule-web-api/src/main/java/com/arextest/schedule/comparer/CategoryComparisonHolder.java
+++ b/arex-schedule-web-api/src/main/java/com/arextest/schedule/comparer/CategoryComparisonHolder.java
@@ -1,6 +1,7 @@
package com.arextest.schedule.comparer;
import java.util.List;
+import lombok.AllArgsConstructor;
import lombok.Data;
/**
@@ -11,6 +12,22 @@
public class CategoryComparisonHolder {
private String categoryName;
+ /**
+ * Need to match the comparison relationship
+ */
private List record;
private List replayResult;
+
+ private Boolean needMatch;
+ /**
+ * Not need to match the comparison relationship
+ */
+ private CompareResultItem compareResultItem;
+
+ @Data
+ @AllArgsConstructor
+ public static class CompareResultItem {
+ CompareItem recordItem;
+ CompareItem replayItem;
+ }
}
\ No newline at end of file
diff --git a/arex-schedule-web-api/src/main/java/com/arextest/schedule/comparer/converter/CompareItemConvertFactory.java b/arex-schedule-web-api/src/main/java/com/arextest/schedule/comparer/converter/CompareItemConvertFactory.java
new file mode 100644
index 00000000..4970683d
--- /dev/null
+++ b/arex-schedule-web-api/src/main/java/com/arextest/schedule/comparer/converter/CompareItemConvertFactory.java
@@ -0,0 +1,26 @@
+package com.arextest.schedule.comparer.converter;
+
+import java.util.List;
+import java.util.Map;
+import java.util.function.Function;
+import java.util.stream.Collectors;
+import javax.annotation.Resource;
+import org.apache.commons.lang3.StringUtils;
+import org.springframework.beans.factory.annotation.Autowired;
+import org.springframework.stereotype.Component;
+
+@Component
+public class CompareItemConvertFactory {
+ private final Map categoryConverters;
+ @Resource
+ private CompareItemConverter defaultCompareItemConverterImpl;
+
+ public CompareItemConvertFactory(@Autowired List converters) {
+ this.categoryConverters = converters.stream().filter(c -> StringUtils.isNotBlank(c.getCategoryName()))
+ .collect(Collectors.toMap(CompareItemConverter::getCategoryName, Function.identity()));
+ }
+
+ public CompareItemConverter getConvert(String categoryName) {
+ return categoryConverters.getOrDefault(categoryName, defaultCompareItemConverterImpl);
+ }
+}
diff --git a/arex-schedule-web-api/src/main/java/com/arextest/schedule/comparer/converter/CompareItemConverter.java b/arex-schedule-web-api/src/main/java/com/arextest/schedule/comparer/converter/CompareItemConverter.java
new file mode 100644
index 00000000..e872bf30
--- /dev/null
+++ b/arex-schedule-web-api/src/main/java/com/arextest/schedule/comparer/converter/CompareItemConverter.java
@@ -0,0 +1,29 @@
+package com.arextest.schedule.comparer.converter;
+
+import com.arextest.model.mock.AREXMocker;
+import com.arextest.model.replay.CompareRelationResult;
+import com.arextest.schedule.comparer.CompareItem;
+
+public interface CompareItemConverter {
+ String DEFAULT_CATEGORY_NAME = "default";
+
+ default String getCategoryName() {
+ return DEFAULT_CATEGORY_NAME;
+ }
+
+ /**
+ * Convert mocker to compare item, The agent handles the compatibility needs before the match
+ * @param mocker
+ * @return
+ */
+ CompareItem convert(AREXMocker mocker);
+
+ /**
+ * Convert relation result to compare item
+ * @param relationResult
+ * @param recordCompareItem
+ * @return
+ */
+ CompareItem convert(CompareRelationResult relationResult, boolean recordCompareItem);
+
+}
diff --git a/arex-schedule-web-api/src/main/java/com/arextest/schedule/comparer/converter/impl/DatabaseCompareItemConverterImpl.java b/arex-schedule-web-api/src/main/java/com/arextest/schedule/comparer/converter/impl/DatabaseCompareItemConverterImpl.java
new file mode 100644
index 00000000..db321ba6
--- /dev/null
+++ b/arex-schedule-web-api/src/main/java/com/arextest/schedule/comparer/converter/impl/DatabaseCompareItemConverterImpl.java
@@ -0,0 +1,99 @@
+package com.arextest.schedule.comparer.converter.impl;
+
+import com.arextest.model.constants.MockAttributeNames;
+import com.arextest.model.mock.AREXMocker;
+import com.arextest.model.mock.MockCategoryType;
+import com.arextest.model.mock.Mocker.Target;
+import com.arextest.model.replay.CompareRelationResult;
+import com.arextest.schedule.common.JsonUtils;
+import com.arextest.schedule.comparer.CompareItem;
+import com.arextest.schedule.comparer.converter.CompareItemConverter;
+import com.arextest.schedule.comparer.impl.PrepareCompareItemBuilder.CompareItemImpl;
+import com.fasterxml.jackson.databind.node.JsonNodeFactory;
+import com.fasterxml.jackson.databind.node.ObjectNode;
+import java.util.Map;
+import java.util.Map.Entry;
+import lombok.extern.slf4j.Slf4j;
+import org.apache.commons.lang3.StringUtils;
+import org.springframework.stereotype.Component;
+
+/**
+ * @author xinyuan_wang.
+ * @create 2024/9/2 16:46
+ */
+@Slf4j
+@Component
+public class DatabaseCompareItemConverterImpl implements CompareItemConverter {
+
+ @Override
+ public String getCategoryName() {
+ return MockCategoryType.DATABASE.getName();
+ }
+
+ @Override
+ public CompareItem convert(AREXMocker mocker) {
+ if (mocker == null || mocker.getCategoryType() == null) {
+ return null;
+ }
+
+ String operationKey = getOperationName(mocker.getTargetRequest(), mocker.getOperationName());
+ return new CompareItemImpl(operationKey, buildAttributes(mocker.getTargetRequest()).toString(),
+ mocker.getId(), mocker.getCreationTime(), mocker.getCategoryType().isEntryPoint());
+ }
+
+ @Override
+ public CompareItem convert(CompareRelationResult relationResult, boolean recordCompareItem) {
+ if (relationResult == null || relationResult.getCategoryType() == null) {
+ return null;
+ }
+
+ String message = recordCompareItem ? relationResult.getRecordMessage() : relationResult.getReplayMessage();
+ long createTime = recordCompareItem ? relationResult.getRecordTime() : relationResult.getReplayTime();
+ return new CompareItemImpl(relationResult.getOperationName(), processMessage(message), null,
+ createTime, relationResult.getCategoryType().isEntryPoint());
+ }
+
+ private String processMessage(String message) {
+ if (StringUtils.isBlank(message)) {
+ return message;
+ }
+
+ Target target = JsonUtils.jsonStringToObject(message, Target.class);
+ if (target != null) {
+ message = buildAttributes(target).toString();
+ }
+ return message;
+ }
+
+ private ObjectNode buildAttributes(Target target) {
+ ObjectNode obj = JsonNodeFactory.instance.objectNode();
+ if (target == null) {
+ return obj;
+ }
+ Map attributes = target.getAttributes();
+ if (attributes != null) {
+ for (Entry entry : attributes.entrySet()) {
+ Object value = entry.getValue();
+ if (value instanceof String) {
+ obj.put(entry.getKey(), (String) value);
+ } else {
+ obj.putPOJO(entry.getKey(), value);
+ }
+ }
+ }
+ if (StringUtils.isNotEmpty(target.getBody())) {
+ obj.put("body", target.getBody());
+ }
+ return obj;
+ }
+
+ private String getOperationName(Target target, String operationName) {
+ // The "@" in the operationName of DATABASE indicates that the SQL statement has been parsed and returned directly.
+ String compareOperationName = StringUtils.contains(operationName, "@") ? operationName
+ : target.attributeAsString(MockAttributeNames.DB_NAME);
+ if (StringUtils.isNotBlank(compareOperationName)) {
+ return compareOperationName;
+ }
+ return operationName;
+ }
+}
diff --git a/arex-schedule-web-api/src/main/java/com/arextest/schedule/comparer/converter/impl/DefaultCompareItemConverterImpl.java b/arex-schedule-web-api/src/main/java/com/arextest/schedule/comparer/converter/impl/DefaultCompareItemConverterImpl.java
new file mode 100644
index 00000000..b4a1d508
--- /dev/null
+++ b/arex-schedule-web-api/src/main/java/com/arextest/schedule/comparer/converter/impl/DefaultCompareItemConverterImpl.java
@@ -0,0 +1,54 @@
+package com.arextest.schedule.comparer.converter.impl;
+
+import com.arextest.model.mock.AREXMocker;
+import com.arextest.model.mock.MockCategoryType;
+import com.arextest.model.replay.CompareRelationResult;
+import com.arextest.schedule.comparer.CompareItem;
+import com.arextest.schedule.comparer.converter.CompareItemConverter;
+import com.arextest.schedule.comparer.impl.PrepareCompareItemBuilder.CompareItemImpl;
+import java.util.Objects;
+import lombok.extern.slf4j.Slf4j;
+import org.springframework.stereotype.Component;
+
+/**
+ * @author xinyuan_wang.
+ * @create 2024/9/2 16:46
+ */
+@Slf4j
+@Component
+public class DefaultCompareItemConverterImpl implements CompareItemConverter {
+
+ @Override
+ public CompareItem convert(AREXMocker mocker) {
+ if (mocker == null || mocker.getCategoryType() == null) {
+ return null;
+ }
+
+ MockCategoryType categoryType = mocker.getCategoryType();
+ String operationKey = mocker.getOperationName();
+ long createTime = mocker.getCreationTime();
+ String body;
+ String compareKey = mocker.getId();
+ boolean entryPointCategory = false;
+ if (categoryType.isEntryPoint()) {
+ body = Objects.isNull(mocker.getTargetResponse()) ? null
+ : mocker.getTargetResponse().getBody();
+ compareKey = null;
+ entryPointCategory = true;
+ } else {
+ body = Objects.isNull(mocker.getTargetRequest()) ? null
+ : mocker.getTargetRequest().getBody();
+ }
+ return new CompareItemImpl(operationKey, body, compareKey, createTime, entryPointCategory);
+ }
+
+ @Override
+ public CompareItem convert(CompareRelationResult relationResult, boolean recordCompareItem) {
+ if (relationResult == null) {
+ return null;
+ }
+
+ return new CompareItemImpl(relationResult, recordCompareItem);
+ }
+
+}
diff --git a/arex-schedule-web-api/src/main/java/com/arextest/schedule/comparer/converter/impl/RedisCompareItemConverterImpl.java b/arex-schedule-web-api/src/main/java/com/arextest/schedule/comparer/converter/impl/RedisCompareItemConverterImpl.java
new file mode 100644
index 00000000..58c1a6b8
--- /dev/null
+++ b/arex-schedule-web-api/src/main/java/com/arextest/schedule/comparer/converter/impl/RedisCompareItemConverterImpl.java
@@ -0,0 +1,66 @@
+package com.arextest.schedule.comparer.converter.impl;
+
+import com.arextest.model.constants.MockAttributeNames;
+import com.arextest.model.mock.AREXMocker;
+import com.arextest.model.mock.MockCategoryType;
+import com.arextest.model.mock.Mocker.Target;
+import com.arextest.model.replay.CompareRelationResult;
+import com.arextest.schedule.comparer.CompareItem;
+import com.arextest.schedule.comparer.converter.CompareItemConverter;
+import com.arextest.schedule.comparer.impl.PrepareCompareItemBuilder.CompareItemImpl;
+import java.util.Objects;
+import lombok.extern.slf4j.Slf4j;
+import org.apache.commons.lang3.StringUtils;
+import org.springframework.stereotype.Component;
+
+/**
+ * @author xinyuan_wang.
+ * @create 2024/9/2 16:46
+ */
+@Slf4j
+@Component
+public class RedisCompareItemConverterImpl implements CompareItemConverter {
+
+ @Override
+ public String getCategoryName() {
+ return MockCategoryType.REDIS.getName();
+ }
+
+ @Override
+ public CompareItem convert(AREXMocker mocker) {
+ if (mocker == null || mocker.getCategoryType() == null) {
+ return null;
+ }
+
+ MockCategoryType categoryType = mocker.getCategoryType();
+ String operationKey = getOperationName(mocker.getTargetRequest(), mocker.getOperationName());
+ String compareMessage = Objects.isNull(mocker.getTargetRequest()) ? null
+ : mocker.getTargetRequest().getBody();
+
+ return new CompareItemImpl(operationKey, compareMessage, mocker.getId(),
+ mocker.getCreationTime(), categoryType.isEntryPoint());
+ }
+ @Override
+ public CompareItem convert(CompareRelationResult relationResult, boolean recordCompareItem) {
+ if (relationResult == null || relationResult.getCategoryType() == null) {
+ return null;
+ }
+
+ String message = recordCompareItem ? relationResult.getRecordMessage() : relationResult.getReplayMessage();
+ long createTime = recordCompareItem ? relationResult.getRecordTime() : relationResult.getReplayTime();
+ return new CompareItemImpl(relationResult.getOperationName(), message, null,
+ createTime, relationResult.getCategoryType().isEntryPoint());
+ }
+
+ private String getOperationName(Target target, String operationName) {
+ if (target == null) {
+ return operationName;
+ }
+
+ String compareOperationName = target.attributeAsString(MockAttributeNames.CLUSTER_NAME);
+ if (StringUtils.isNotBlank(compareOperationName)) {
+ return compareOperationName;
+ }
+ return operationName;
+ }
+}
diff --git a/arex-schedule-web-api/src/main/java/com/arextest/schedule/comparer/impl/DefaultReplayResultComparer.java b/arex-schedule-web-api/src/main/java/com/arextest/schedule/comparer/impl/DefaultReplayResultComparer.java
index 9adba651..891c152d 100644
--- a/arex-schedule-web-api/src/main/java/com/arextest/schedule/comparer/impl/DefaultReplayResultComparer.java
+++ b/arex-schedule-web-api/src/main/java/com/arextest/schedule/comparer/impl/DefaultReplayResultComparer.java
@@ -6,6 +6,7 @@
import com.arextest.diff.sdk.CompareSDK;
import com.arextest.model.mock.MockCategoryType;
import com.arextest.schedule.comparer.CategoryComparisonHolder;
+import com.arextest.schedule.comparer.CategoryComparisonHolder.CompareResultItem;
import com.arextest.schedule.comparer.CompareConfigService;
import com.arextest.schedule.comparer.CompareItem;
import com.arextest.schedule.comparer.CompareService;
@@ -162,10 +163,39 @@ public List doContentCompare(ReplayActionCaseItem caseItem,
}
/**
- * compare recording and replay data. 1. record and replay data through compareKey.
+ * compare recording and replay data.
*/
private List compareReplayResult(CategoryComparisonHolder bindHolder,
ReplayActionCaseItem caseItem, ComparisonInterfaceConfig operationConfig) {
+ if (Boolean.TRUE.equals(bindHolder.getNeedMatch())) {
+ return matchCompareReplayResults(bindHolder, caseItem, operationConfig);
+ }
+
+ return getReplayCompareResults(bindHolder, caseItem, operationConfig);
+ }
+
+ private List getReplayCompareResults(CategoryComparisonHolder bindHolder,
+ ReplayActionCaseItem caseItem, ComparisonInterfaceConfig operationConfig) {
+ CompareResultItem item = bindHolder.getCompareResultItem();
+ if (item == null) {
+ return Collections.emptyList();
+ }
+
+ List compareResults = new ArrayList<>();
+ compareResults.add(compareRecordAndResult(operationConfig, caseItem, bindHolder.getCategoryName(),
+ item.getReplayItem(), item.getRecordItem()));
+ return compareResults;
+ }
+
+ /**
+ * record and replay data through compareKey.
+ * @param bindHolder
+ * @param caseItem
+ * @param operationConfig
+ * @return
+ */
+ private List matchCompareReplayResults(CategoryComparisonHolder bindHolder,
+ ReplayActionCaseItem caseItem, ComparisonInterfaceConfig operationConfig) {
List compareResults = new ArrayList<>();
List recordResults = bindHolder.getRecord();
List replayResults = bindHolder.getReplayResult();
diff --git a/arex-schedule-web-api/src/main/java/com/arextest/schedule/comparer/impl/PrepareCompareItemBuilder.java b/arex-schedule-web-api/src/main/java/com/arextest/schedule/comparer/impl/PrepareCompareItemBuilder.java
index c918dbbc..5d679208 100644
--- a/arex-schedule-web-api/src/main/java/com/arextest/schedule/comparer/impl/PrepareCompareItemBuilder.java
+++ b/arex-schedule-web-api/src/main/java/com/arextest/schedule/comparer/impl/PrepareCompareItemBuilder.java
@@ -1,16 +1,11 @@
package com.arextest.schedule.comparer.impl;
-import com.arextest.model.constants.MockAttributeNames;
import com.arextest.model.mock.AREXMocker;
-import com.arextest.model.mock.MockCategoryType;
-import com.arextest.model.mock.Mocker.Target;
+import com.arextest.model.replay.CompareRelationResult;
import com.arextest.schedule.comparer.CompareItem;
-import com.fasterxml.jackson.databind.node.JsonNodeFactory;
-import com.fasterxml.jackson.databind.node.ObjectNode;
-import java.util.Map;
-import java.util.Map.Entry;
-import java.util.Objects;
-import org.apache.commons.lang3.StringUtils;
+import com.arextest.schedule.comparer.converter.CompareItemConvertFactory;
+import com.arextest.schedule.comparer.converter.CompareItemConverter;
+import javax.annotation.Resource;
import org.springframework.stereotype.Component;
/**
@@ -18,67 +13,22 @@
* @since 2021/11/23
*/
@Component
-final class PrepareCompareItemBuilder {
+public class PrepareCompareItemBuilder {
- CompareItem build(AREXMocker instance) {
- MockCategoryType categoryType = instance.getCategoryType();
- String operationKey = operationName(categoryType, instance.getTargetRequest(), instance.getOperationName());
- if (StringUtils.isEmpty(operationKey)) {
- operationKey = instance.getOperationName();
- }
- long createTime = instance.getCreationTime();
- String body;
- String compareKey = instance.getId();
- boolean entryPointCategory = false;
- if (categoryType.isEntryPoint()) {
- body = Objects.isNull(instance.getTargetResponse()) ? null
- : instance.getTargetResponse().getBody();
- compareKey = null;
- entryPointCategory = true;
- } else if (Objects.equals(categoryType.getName(), MockCategoryType.DATABASE.getName())) {
- body = this.buildAttributes(instance.getTargetRequest()).toString();
- } else {
- body = Objects.isNull(instance.getTargetRequest()) ? null
- : instance.getTargetRequest().getBody();
- }
- return new CompareItemImpl(operationKey, body, compareKey, createTime, entryPointCategory);
- }
+ @Resource
+ private CompareItemConvertFactory factory;
- private String operationName(MockCategoryType categoryType, Target target, String operationName) {
- if (Objects.equals(categoryType, MockCategoryType.DATABASE)) {
- // The "@" in the operationName of DATABASE indicates that the SQL statement has been parsed and returned directly.
- return StringUtils.contains(operationName, "@") ? operationName : target.attributeAsString(MockAttributeNames.DB_NAME);
- }
- if (Objects.equals(categoryType, MockCategoryType.REDIS)) {
- return target.attributeAsString(MockAttributeNames.CLUSTER_NAME);
- }
- return operationName;
+ CompareItem build(AREXMocker mocker) {
+ CompareItemConverter converter = factory.getConvert(mocker.getCategoryType().getName());
+ return converter.convert(mocker);
}
- private ObjectNode buildAttributes(Target target) {
- ObjectNode obj = JsonNodeFactory.instance.objectNode();
- if (target == null) {
- return obj;
- }
- Map attributes = target.getAttributes();
- if (attributes != null) {
- for (Entry entry : attributes.entrySet()) {
- Object value = entry.getValue();
- if (value instanceof String) {
- obj.put(entry.getKey(), (String) value);
- } else {
- obj.putPOJO(entry.getKey(), value);
- }
- }
- }
- if (StringUtils.isNotEmpty(target.getBody())) {
- obj.put("body", target.getBody());
- }
- return obj;
+ public CompareItem build(CompareRelationResult result, boolean recordCompareItem) {
+ CompareItemConverter converter = factory.getConvert(result.getCategoryType().getName());
+ return converter.convert(result, recordCompareItem);
}
- private final static class CompareItemImpl implements CompareItem {
-
+ public final static class CompareItemImpl implements CompareItem {
private final String compareMessage;
private final String compareOperation;
private final String compareService;
@@ -86,11 +36,20 @@ private final static class CompareItemImpl implements CompareItem {
private final long createTime;
private final boolean entryPointCategory;
- private CompareItemImpl(String compareOperation, String compareMessage, String compareKey,
+ public CompareItemImpl(String compareOperation, String compareMessage, String compareKey,
long createTime, boolean entryPointCategory) {
this(compareOperation, compareMessage, null, compareKey, createTime, entryPointCategory);
}
+ public CompareItemImpl(CompareRelationResult result, boolean recordCompareItem) {
+ this.compareOperation = result.getOperationName();
+ this.compareService = null;
+ this.compareKey = null;
+ this.entryPointCategory = result.getCategoryType().isEntryPoint();
+ this.compareMessage = recordCompareItem ? result.getRecordMessage() : result.getReplayMessage();
+ this.createTime = recordCompareItem ? result.getRecordTime() : result.getReplayTime();
+ }
+
private CompareItemImpl(String compareOperation, String compareMessage, String compareService,
String compareKey,
long createTime, boolean entryPointCategory) {
diff --git a/arex-schedule-web-api/src/main/java/com/arextest/schedule/comparer/impl/PrepareCompareSourceRemoteLoader.java b/arex-schedule-web-api/src/main/java/com/arextest/schedule/comparer/impl/PrepareCompareSourceRemoteLoader.java
index 844743bc..d128e8b5 100644
--- a/arex-schedule-web-api/src/main/java/com/arextest/schedule/comparer/impl/PrepareCompareSourceRemoteLoader.java
+++ b/arex-schedule-web-api/src/main/java/com/arextest/schedule/comparer/impl/PrepareCompareSourceRemoteLoader.java
@@ -1,13 +1,14 @@
package com.arextest.schedule.comparer.impl;
+import com.arextest.model.replay.CompareRelationResult;
import com.arextest.model.mock.AREXMocker;
import com.arextest.model.mock.MockCategoryType;
import com.arextest.model.replay.QueryReplayResultRequestType;
import com.arextest.model.replay.QueryReplayResultResponseType;
import com.arextest.model.replay.holder.ListResultHolder;
-import com.arextest.model.response.ResponseStatusType;
import com.arextest.schedule.client.HttpWepServiceApiClient;
import com.arextest.schedule.comparer.CategoryComparisonHolder;
+import com.arextest.schedule.comparer.CategoryComparisonHolder.CompareResultItem;
import com.arextest.schedule.comparer.CompareItem;
import com.arextest.schedule.model.ReplayActionCaseItem;
import com.arextest.schedule.serialization.ZstdJacksonSerializer;
@@ -78,38 +79,46 @@ private QueryReplayResultResponseType remoteLoad(String replayId, String resultI
QueryReplayResultRequestType resultRequest = new QueryReplayResultRequestType();
resultRequest.setRecordId(replayId);
resultRequest.setReplayResultId(resultId);
- return httpWepServiceApiClient.retryJsonPost(replayResultUrl, resultRequest,
- QueryReplayResultResponseType.class);
+ return httpWepServiceApiClient.retryZstdJsonPost(replayResultUrl,
+ resultRequest, QueryReplayResultResponseType.class);
}
+ /**
+ * Compatible logic.
+ * The subsequent agent will match the packets and does not need to be processed in the schedule.
+ * @param replayResultResponseType
+ * @return
+ */
private List decodeResult(
QueryReplayResultResponseType replayResultResponseType) {
- if (replayResultResponseType == null) {
+ if (replayResultResponseType == null ||
+ replayResultResponseType.getResponseStatusType() == null ||
+ replayResultResponseType.getResponseStatusType().hasError() ||
+ Boolean.TRUE.equals(replayResultResponseType.getInvalidResult())) {
+ LOGGER.warn("failed to get replay result because of invalid case");
return Collections.emptyList();
}
- ResponseStatusType responseStatusType = replayResultResponseType.getResponseStatusType();
- if (responseStatusType == null) {
- return Collections.emptyList();
- }
- if (responseStatusType.hasError()) {
- LOGGER.warn("query replay result has error response : {}", responseStatusType);
- return Collections.emptyList();
- }
- if (Boolean.TRUE.equals(replayResultResponseType.getInvalidResult())) {
- LOGGER.warn("query replay result has invalid result: get data failed from storage");
- return Collections.emptyList();
- }
+
+ // Use needMatch to determine whether to adopt new logic
+ return Boolean.TRUE.equals(replayResultResponseType.getNeedMatch()) ?
+ processMatchNeeded(replayResultResponseType) : processMatchNotNeeded(replayResultResponseType);
+ }
+
+ /**
+ * Processing of messages that require matching relationships that need to be compared
+ * @param replayResultResponseType
+ * @return
+ */
+ private List processMatchNeeded(QueryReplayResultResponseType replayResultResponseType) {
List resultHolderList = replayResultResponseType.getResultHolderList();
if (CollectionUtils.isEmpty(resultHolderList)) {
LOGGER.warn("query replay result has empty size");
return Collections.emptyList();
}
- List decodedListResult = new ArrayList<>(resultHolderList.size());
- MockCategoryType categoryType;
- for (int i = 0; i < resultHolderList.size(); i++) {
- ListResultHolder stringListResultHolder = resultHolderList.get(i);
- categoryType = stringListResultHolder.getCategoryType();
+ List decodedListResult = new ArrayList<>(resultHolderList.size());
+ for (ListResultHolder stringListResultHolder : resultHolderList) {
+ MockCategoryType categoryType = stringListResultHolder.getCategoryType();
if (categoryType == null || categoryType.isSkipComparison()) {
continue;
}
@@ -117,19 +126,56 @@ private List decodeResult(
CategoryComparisonHolder resultHolder = new CategoryComparisonHolder();
resultHolder.setCategoryName(categoryType.getName());
decodedListResult.add(resultHolder);
+
List recordList = zstdDeserialize(stringListResultHolder.getRecord());
- List replayResultList = zstdDeserialize(
- stringListResultHolder.getReplayResult());
- if (categoryType.isEntryPoint()) {
- boolean recordEmpty = CollectionUtils.isEmpty(recordList);
- boolean replayResultEmpty = CollectionUtils.isEmpty(replayResultList);
+ List replayResultList = zstdDeserialize(stringListResultHolder.getReplayResult());
+
+ if (categoryType.isEntryPoint() && (CollectionUtils.isEmpty(recordList) || CollectionUtils.isEmpty(replayResultList))) {
// call missing or new call
- if (recordEmpty || replayResultEmpty) {
- return Collections.emptyList();
- }
+ return Collections.emptyList();
}
+
resultHolder.setRecord(recordList);
resultHolder.setReplayResult(replayResultList);
+ resultHolder.setNeedMatch(true);
+ }
+ return decodedListResult;
+ }
+
+ /**
+ * Processing of packets that do not require matching
+ * @param replayResultResponseType
+ * @return
+ */
+ private List processMatchNotNeeded(QueryReplayResultResponseType replayResultResponseType) {
+ List replayResults = replayResultResponseType.getReplayResults();
+ if (CollectionUtils.isEmpty(replayResults)) {
+ LOGGER.warn("query replay result has empty size");
+ return Collections.emptyList();
+ }
+
+ List decodedListResult = new ArrayList<>(replayResults.size());
+ for (CompareRelationResult result : replayResults) {
+ MockCategoryType categoryType = result.getCategoryType();
+ if (categoryType == null || (!categoryType.isEntryPoint() && categoryType.isSkipComparison())) {
+ continue;
+ }
+
+ if (categoryType.isEntryPoint() && !categoryType.isSkipComparison() &&
+ (StringUtils.isEmpty(result.getRecordMessage()) || StringUtils.isEmpty(result.getReplayMessage()))) {
+ // The main category is missing
+ return Collections.emptyList();
+ }
+
+ CategoryComparisonHolder resultHolder = new CategoryComparisonHolder();
+ resultHolder.setCategoryName(categoryType.getName());
+
+ CompareItem recordCompareItem = prepareCompareItemBuilder.build(result, true);
+ CompareItem replayCompareItem = prepareCompareItemBuilder.build(result, false);
+
+ resultHolder.setCompareResultItem(new CompareResultItem(recordCompareItem, replayCompareItem));
+ resultHolder.setNeedMatch(false);
+ decodedListResult.add(resultHolder);
}
return decodedListResult;
}
diff --git a/arex-schedule-web-api/src/main/java/com/arextest/schedule/serialization/ZstdJacksonSerializer.java b/arex-schedule-web-api/src/main/java/com/arextest/schedule/serialization/ZstdJacksonSerializer.java
index ba362219..026eff42 100644
--- a/arex-schedule-web-api/src/main/java/com/arextest/schedule/serialization/ZstdJacksonSerializer.java
+++ b/arex-schedule-web-api/src/main/java/com/arextest/schedule/serialization/ZstdJacksonSerializer.java
@@ -5,6 +5,8 @@
import com.arextest.common.serialization.SerializationProviders;
import com.arextest.common.utils.SerializationUtils;
import com.fasterxml.jackson.databind.ObjectMapper;
+import java.io.ByteArrayOutputStream;
+import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import java.util.Base64;
@@ -25,12 +27,26 @@ public final class ZstdJacksonSerializer {
@Resource
private ObjectMapper objectMapper;
private SerializationProvider serializationProvider;
+ public static final byte[] EMPTY_BYTE = new byte[]{};
@PostConstruct
void initSerializationProvider() {
this.serializationProvider = SerializationProviders.jacksonProvider(this.objectMapper);
}
+ public byte[] serialize(T value) {
+ if (value == null) {
+ return EMPTY_BYTE;
+ }
+ try (ByteArrayOutputStream out = new ByteArrayOutputStream()) {
+ serializeTo(value, out);
+ return out.toByteArray();
+ } catch (IOException e) {
+ LOGGER.error("serialize error:{}", e.getMessage(), e);
+ }
+ return EMPTY_BYTE;
+ }
+
public void serializeTo(T value, OutputStream outputStream) {
if (value == null) {
return;
diff --git a/arex-schedule-web-api/src/main/java/com/arextest/schedule/service/LocalReplayService.java b/arex-schedule-web-api/src/main/java/com/arextest/schedule/service/LocalReplayService.java
index a6ef8276..1e297bf2 100644
--- a/arex-schedule-web-api/src/main/java/com/arextest/schedule/service/LocalReplayService.java
+++ b/arex-schedule-web-api/src/main/java/com/arextest/schedule/service/LocalReplayService.java
@@ -198,6 +198,7 @@ public CommonResponse queryReRunCaseId(ReRunReplayPlanRequest request) {
progressEvent.onReplayPlanReRun(replayPlan);
progressEvent.onUpdateFailedCases(replayPlan, failedCaseList);
planConsumePrepareService.updateFailedActionAndCase(replayPlan, failedCaseList);
+ compareConfigService.preload(replayPlan);
cacheReplayPlan(replayPlan);
if (CollectionUtils.isEmpty(replayPlan.getReplayActionItemList())) {
throw new RuntimeException("no replayActionItem!");
@@ -262,6 +263,7 @@ private void cacheReplayPlan(ReplayPlan replayPlan) {
JsonUtils.objectToJsonString(replayPlanForCache).getBytes(StandardCharsets.UTF_8));
}
+ @SuppressWarnings("java:S1181")
private ReplayPlanForCache loadReplayPlanCache(String planId) {
try {
byte[] json = doWithRetry(
@@ -285,6 +287,7 @@ private byte[] doWithRetry(Supplier action) {
}
}
+ @SuppressWarnings("java:S1181")
private ReplayActionItemForCache loadReplayActionItemCache(String planItemId) {
try {
byte[] json = doWithRetry(
diff --git a/pom.xml b/pom.xml
index c6106995..d07d1341 100644
--- a/pom.xml
+++ b/pom.xml
@@ -307,7 +307,7 @@
**/model/**,
**/mapping/**
- 1.2.19
+ 1.3.0
0.6.5.2
0.2.3
3.20.1
@@ -320,5 +320,5 @@
https://github.com/arextest/arex-replay-schedule
https://github.com/arextest/arex-replay-schedule
- 1.2.16
+ 1.2.20
\ No newline at end of file