From 4972ff44a1e6eb9eca972f6592a8f9efdaadd202 Mon Sep 17 00:00:00 2001 From: Harshit Kumar Date: Fri, 23 Aug 2024 11:04:26 +0530 Subject: [PATCH] filter builder fix and store changes --- .../config/service/store/FilterBuilder.java | 71 +++++++++++++------ .../service/store/FilterBuilderTest.java | 20 +++--- .../objectstore/IdentifiedObjectStore.java | 38 +++++++--- 3 files changed, 86 insertions(+), 43 deletions(-) diff --git a/config-service-impl/src/main/java/org/hypertrace/config/service/store/FilterBuilder.java b/config-service-impl/src/main/java/org/hypertrace/config/service/store/FilterBuilder.java index 51ed2807..1856db88 100644 --- a/config-service-impl/src/main/java/org/hypertrace/config/service/store/FilterBuilder.java +++ b/config-service-impl/src/main/java/org/hypertrace/config/service/store/FilterBuilder.java @@ -2,7 +2,14 @@ import static org.hypertrace.config.service.store.ConfigDocument.CONFIG_FIELD_NAME; +import com.google.protobuf.ListValue; +import com.google.protobuf.Struct; +import com.google.protobuf.Value; import io.grpc.Status; +import java.util.ArrayList; +import java.util.HashMap; +import java.util.List; +import java.util.Map; import org.hypertrace.config.service.v1.LogicalFilter; import org.hypertrace.config.service.v1.RelationalFilter; import org.hypertrace.core.documentstore.Filter; @@ -54,12 +61,11 @@ private Filter evaluateCompositeExpression(LogicalFilter logicalFilter) { } private Filter evaluateLeafExpression(RelationalFilter relationalFilter) { + Object value = this.convertValueToObject(relationalFilter.getValue()); switch (relationalFilter.getOperator()) { case RELATIONAL_OPERATOR_EQ: return new Filter( - Filter.Op.EQ, - buildConfigFieldPath(relationalFilter.getConfigJsonPath()), - relationalFilter.getValue()); + Filter.Op.EQ, buildConfigFieldPath(relationalFilter.getConfigJsonPath()), value); case RELATIONAL_OPERATOR_NEQ: return new Filter( Filter.Op.NEQ, @@ -67,34 +73,22 @@ private Filter evaluateLeafExpression(RelationalFilter relationalFilter) { relationalFilter.getValue()); case RELATIONAL_OPERATOR_IN: return new Filter( - Filter.Op.IN, - buildConfigFieldPath(relationalFilter.getConfigJsonPath()), - relationalFilter.getValue()); + Filter.Op.IN, buildConfigFieldPath(relationalFilter.getConfigJsonPath()), value); case RELATIONAL_OPERATOR_NOT_IN: return new Filter( - Filter.Op.NOT_IN, - buildConfigFieldPath(relationalFilter.getConfigJsonPath()), - relationalFilter.getValue()); + Filter.Op.NOT_IN, buildConfigFieldPath(relationalFilter.getConfigJsonPath()), value); case RELATIONAL_OPERATOR_LT: return new Filter( - Filter.Op.LT, - buildConfigFieldPath(relationalFilter.getConfigJsonPath()), - relationalFilter.getValue()); + Filter.Op.LT, buildConfigFieldPath(relationalFilter.getConfigJsonPath()), value); case RELATIONAL_OPERATOR_GT: return new Filter( - Filter.Op.GT, - buildConfigFieldPath(relationalFilter.getConfigJsonPath()), - relationalFilter.getValue()); + Filter.Op.GT, buildConfigFieldPath(relationalFilter.getConfigJsonPath()), value); case RELATIONAL_OPERATOR_LTE: return new Filter( - Filter.Op.LTE, - buildConfigFieldPath(relationalFilter.getConfigJsonPath()), - relationalFilter.getValue()); + Filter.Op.LTE, buildConfigFieldPath(relationalFilter.getConfigJsonPath()), value); case RELATIONAL_OPERATOR_GTE: return new Filter( - Filter.Op.GTE, - buildConfigFieldPath(relationalFilter.getConfigJsonPath()), - relationalFilter.getValue()); + Filter.Op.GTE, buildConfigFieldPath(relationalFilter.getConfigJsonPath()), value); case UNRECOGNIZED: default: throw Status.INVALID_ARGUMENT @@ -106,4 +100,39 @@ private Filter evaluateLeafExpression(RelationalFilter relationalFilter) { private String buildConfigFieldPath(String configJsonPath) { return String.format("%s.%s", CONFIG_FIELD_NAME, configJsonPath); } + + public Object convertValueToObject(Value value) { + switch (value.getKindCase()) { + case NULL_VALUE: + return null; + case BOOL_VALUE: + return value.getBoolValue(); + case NUMBER_VALUE: + return value.getNumberValue(); + case STRING_VALUE: + return value.getStringValue(); + case STRUCT_VALUE: + return convertStructToMap(value.getStructValue()); + case LIST_VALUE: + return convertListValueToList(value.getListValue()); + default: + throw new IllegalArgumentException("Unknown value type: " + value.getKindCase()); + } + } + + private Map convertStructToMap(Struct struct) { + Map map = new HashMap<>(); + for (Map.Entry entry : struct.getFieldsMap().entrySet()) { + map.put(entry.getKey(), convertValueToObject(entry.getValue())); + } + return map; + } + + private List convertListValueToList(ListValue listValue) { + List list = new ArrayList<>(); + for (Value value : listValue.getValuesList()) { + list.add(convertValueToObject(value)); + } + return list; + } } diff --git a/config-service-impl/src/test/java/org/hypertrace/config/service/store/FilterBuilderTest.java b/config-service-impl/src/test/java/org/hypertrace/config/service/store/FilterBuilderTest.java index 76e36281..62bbfcad 100644 --- a/config-service-impl/src/test/java/org/hypertrace/config/service/store/FilterBuilderTest.java +++ b/config-service-impl/src/test/java/org/hypertrace/config/service/store/FilterBuilderTest.java @@ -31,11 +31,11 @@ void buildDocStoreFilter1() { RelationalFilter.newBuilder() .setConfigJsonPath("key1") .setOperator(RelationalOperator.RELATIONAL_OPERATOR_EQ) - .setValue(Value.newBuilder().setStringValue("value1"))) + .setValue(Values.of("value1"))) .build(); org.hypertrace.core.documentstore.Filter docFilter = - org.hypertrace.core.documentstore.Filter.eq("config.key1", Values.of("value1")); + org.hypertrace.core.documentstore.Filter.eq("config.key1", "value1"); assertEquals(docFilter.toString(), filterBuilder.buildDocStoreFilter(filter).toString()); Filter filter2 = @@ -44,11 +44,11 @@ void buildDocStoreFilter1() { RelationalFilter.newBuilder() .setConfigJsonPath("key2") .setOperator(RelationalOperator.RELATIONAL_OPERATOR_EQ) - .setValue(Value.newBuilder().setNumberValue(300))) + .setValue(Values.of(300))) .build(); org.hypertrace.core.documentstore.Filter docFilter2 = - org.hypertrace.core.documentstore.Filter.eq("config.key2", Values.of(300)); + org.hypertrace.core.documentstore.Filter.eq("config.key2", 300.0); assertEquals(docFilter2.toString(), filterBuilder.buildDocStoreFilter(filter2).toString()); } @@ -76,12 +76,10 @@ void buildDocStoreFilter2() { .build(); org.hypertrace.core.documentstore.Filter docFilter = - org.hypertrace.core.documentstore.Filter.eq("config.key3", Values.of(true)) + org.hypertrace.core.documentstore.Filter.eq("config.key3", true) .and( new org.hypertrace.core.documentstore.Filter( - org.hypertrace.core.documentstore.Filter.Op.LTE, - "config.key4", - Values.of(100))); + org.hypertrace.core.documentstore.Filter.Op.LTE, "config.key4", 100.0)); assertEquals(docFilter.toString(), filterBuilder.buildDocStoreFilter(filter).toString()); } @@ -133,11 +131,11 @@ void buildDocStoreFilter3() { new org.hypertrace.core.documentstore.Filter( org.hypertrace.core.documentstore.Filter.Op.IN, "config.key5", - Values.of(List.of(Values.of("listValue1"), Values.of("listValue2")))) + List.of("listValue1", "listValue2")) .and( new org.hypertrace.core.documentstore.Filter( - org.hypertrace.core.documentstore.Filter.Op.GTE, "config.key6", Values.of(100))) - .or(org.hypertrace.core.documentstore.Filter.eq("config.key7", Values.of("value7"))); + org.hypertrace.core.documentstore.Filter.Op.GTE, "config.key6", 100.0)) + .or(org.hypertrace.core.documentstore.Filter.eq("config.key7", "value7")); assertEquals(docFilter.toString(), filterBuilder.buildDocStoreFilter(filter).toString()); } diff --git a/object-store/src/main/java/org/hypertrace/config/objectstore/IdentifiedObjectStore.java b/object-store/src/main/java/org/hypertrace/config/objectstore/IdentifiedObjectStore.java index 72fdcec0..e8c62544 100644 --- a/object-store/src/main/java/org/hypertrace/config/objectstore/IdentifiedObjectStore.java +++ b/object-store/src/main/java/org/hypertrace/config/objectstore/IdentifiedObjectStore.java @@ -14,6 +14,7 @@ import org.hypertrace.config.service.v1.DeleteConfigRequest; import org.hypertrace.config.service.v1.DeleteConfigsRequest; import org.hypertrace.config.service.v1.DeleteConfigsRequest.ConfigToDelete; +import org.hypertrace.config.service.v1.Filter; import org.hypertrace.config.service.v1.GetAllConfigsRequest; import org.hypertrace.config.service.v1.GetConfigRequest; import org.hypertrace.config.service.v1.GetConfigResponse; @@ -153,24 +154,39 @@ public Optional getData(RequestContext context, String id) { return getObject(context, id).map(ConfigObject::getData); } - public ContextualConfigObject upsertObject(RequestContext context, T data) { + private ContextualConfigObject upsertObject( + RequestContext context, UpsertConfigRequest request) { UpsertConfigResponse response = context.call( - () -> - this.configServiceBlockingStub - .withDeadline(getDeadline()) - .upsertConfig( - UpsertConfigRequest.newBuilder() - .setResourceName(this.resourceName) - .setResourceNamespace(this.resourceNamespace) - .setContext(this.getContextFromData(data)) - .setConfig(this.buildValueFromData(data)) - .build())); + () -> this.configServiceBlockingStub.withDeadline(getDeadline()).upsertConfig(request)); return this.processUpsertResult(context, response) .orElseThrow(Status.INTERNAL::asRuntimeException); } + public ContextualConfigObject upsertObject(RequestContext context, T data) { + return this.upsertObject( + context, + UpsertConfigRequest.newBuilder() + .setResourceName(this.resourceName) + .setResourceNamespace(this.resourceNamespace) + .setContext(this.getContextFromData(data)) + .setConfig(this.buildValueFromData(data)) + .build()); + } + + public ContextualConfigObject upsertObject(RequestContext context, T data, Filter filter) { + return this.upsertObject( + context, + UpsertConfigRequest.newBuilder() + .setResourceName(this.resourceName) + .setResourceNamespace(this.resourceNamespace) + .setContext(this.getContextFromData(data)) + .setConfig(this.buildValueFromData(data)) + .setUpsertCondition(filter) + .build()); + } + public Optional> deleteObject( RequestContext requestContext, String context) { try {