From dcd1fe0daf23288095d1048fc2d1eff91d921177 Mon Sep 17 00:00:00 2001 From: slorello89 Date: Wed, 20 Nov 2024 07:43:16 -0500 Subject: [PATCH 1/4] storing what we have so far --- .../Common/ExpressionParserUtilities.cs | 38 +++++++++++++ src/Redis.OM/Common/ExpressionTranslator.cs | 7 +++ .../ObjectWithNullableStrings.cs | 17 ++++++ .../RediSearchTests/SearchTests.cs | 53 +++++++++++++++++++ 4 files changed, 115 insertions(+) create mode 100644 test/Redis.OM.Unit.Tests/RediSearchTests/ObjectWithNullableStrings.cs diff --git a/src/Redis.OM/Common/ExpressionParserUtilities.cs b/src/Redis.OM/Common/ExpressionParserUtilities.cs index 27a4921..1dda74a 100644 --- a/src/Redis.OM/Common/ExpressionParserUtilities.cs +++ b/src/Redis.OM/Common/ExpressionParserUtilities.cs @@ -316,6 +316,44 @@ internal static string EscapeTagField(string text) return sb.ToString(); } + /// + /// Checks the expression if it resolves to null. + /// + /// The expression to check. + /// whether it resolves to null. + internal static bool ExpressionResolvesToNull(Expression expression) + { + if (expression.NodeType is ExpressionType.Constant && ((ConstantExpression)expression).Value is null) + { + return true; + } + + if (expression.NodeType is ExpressionType.MemberAccess) + { + var parentExpression = expression; + var memberInfos = new Stack(); + while (parentExpression is MemberExpression parentMember) + { + var info = ((MemberExpression)parentExpression).Member; + memberInfos.Push(info); + parentExpression = parentMember.Expression; + } + + if (parentExpression is ConstantExpression c) + { + var resolved = c.Value; + foreach (var info in memberInfos) + { + resolved = GetValue(info, resolved); + } + + return resolved is null; + } + } + + return false; + } + private static string GetOperandStringForMember(MemberExpression member, bool treatEnumsAsInt = false, bool negate = false, bool treatBooleanMemberAsUnary = false) { var memberPath = new List(); diff --git a/src/Redis.OM/Common/ExpressionTranslator.cs b/src/Redis.OM/Common/ExpressionTranslator.cs index f2f2be8..330f963 100644 --- a/src/Redis.OM/Common/ExpressionTranslator.cs +++ b/src/Redis.OM/Common/ExpressionTranslator.cs @@ -364,6 +364,13 @@ internal static string TranslateBinaryExpression(BinaryExpression binExpression, { var leftContent = ExpressionParserUtilities.GetOperandStringForQueryArgs(binExpression.Left, parameters, treatBooleanMemberAsUnary: true); + var rightResolvesToNull = ExpressionParserUtilities.ExpressionResolvesToNull(binExpression.Right); + + if (rightResolvesToNull) + { + return $"(ismissing({leftContent}))"; + } + var rightContent = ExpressionParserUtilities.GetOperandStringForQueryArgs(binExpression.Right, parameters, treatBooleanMemberAsUnary: true); if (binExpression.Left is MemberExpression member) diff --git a/test/Redis.OM.Unit.Tests/RediSearchTests/ObjectWithNullableStrings.cs b/test/Redis.OM.Unit.Tests/RediSearchTests/ObjectWithNullableStrings.cs new file mode 100644 index 0000000..f04bf60 --- /dev/null +++ b/test/Redis.OM.Unit.Tests/RediSearchTests/ObjectWithNullableStrings.cs @@ -0,0 +1,17 @@ +using Redis.OM.Modeling; + +namespace Redis.OM.Unit.Tests.RediSearchTests; + +[Document(StorageType = StorageType.Json)] +public class ObjectWithNullableStrings +{ + [RedisIdField] + [Indexed] + public string Id { get; set; } + + [Indexed] + public string? String1 { get; set; } + + [Indexed] + public string? String2 { get; set; } +} \ No newline at end of file diff --git a/test/Redis.OM.Unit.Tests/RediSearchTests/SearchTests.cs b/test/Redis.OM.Unit.Tests/RediSearchTests/SearchTests.cs index 6eb09b9..236148f 100644 --- a/test/Redis.OM.Unit.Tests/RediSearchTests/SearchTests.cs +++ b/test/Redis.OM.Unit.Tests/RediSearchTests/SearchTests.cs @@ -4026,5 +4026,58 @@ public void TestToQueryString() Assert.Equal(command, queryString); } + + [Fact] + public async Task TestQueryForNull() + { + _substitute.ClearSubstitute(); + _substitute.ExecuteAsync(Arg.Any(), Arg.Any()).Returns(_mockReply); + var collection = new RedisCollection(_substitute); + string? val = null; + await collection.Where(x => x.String1 == null).ToListAsync(); + await _substitute.Received().ExecuteAsync("FT.SEARCH", + $"{nameof(ObjectWithNullableStrings).ToLower()}-idx", + "(ismissing(@String1))", + "LIMIT", + "DIALECT", + 2, + "0", + "100"); + + _substitute.ClearSubstitute(); + _substitute.ExecuteAsync(Arg.Any(), Arg.Any()).Returns(_mockReply); + await collection.Where(x => x.String1 == val).ToListAsync(); + await _substitute.Received().ExecuteAsync("FT.SEARCH", + $"{nameof(ObjectWithNullableStrings).ToLower()}-idx", + "(ismissing(@String1))", + "LIMIT", + "DIALECT", + 2, + "0", + "100"); + + _substitute.ClearSubstitute(); + + var obj = new + { + inner = new + { + val = (string?)null + } + }; + + _substitute.ExecuteAsync(Arg.Any(), Arg.Any()).Returns(_mockReply); + await collection.Where(x => x.String2 == obj.inner.val).ToListAsync(); + await _substitute.Received().ExecuteAsync("FT.SEARCH", + $"{nameof(ObjectWithNullableStrings).ToLower()}-idx", + "(ismissing(@String2))", + "LIMIT", + "DIALECT", + 2, + "0", + "100"); + + } + } } \ No newline at end of file From 612930b74b562776b70f7563227432191b190539 Mon Sep 17 00:00:00 2001 From: slorello89 Date: Tue, 26 Nov 2024 08:51:25 -0500 Subject: [PATCH 2/4] adding queryable null and empty strings feature --- .../AggregationPredicates/QueryPredicate.cs | 5 +- .../Common/ExpressionParserUtilities.cs | 40 ++++--- src/Redis.OM/Common/ExpressionTranslator.cs | 83 +++++++------- src/Redis.OM/Modeling/RedisIndex.cs | 20 ++++ src/Redis.OM/Modeling/RedisSchemaField.cs | 12 +++ src/Redis.OM/Modeling/SearchFieldAttribute.cs | 6 ++ src/Redis.OM/RedisIndexInfo.cs | 16 ++- src/Redis.OM/Searching/Query/RedisQuery.cs | 23 +++- .../ObjectWithNullableStrings.cs | 33 ++++++ .../RediSearchTests/RedisIndexTests.cs | 4 +- .../RediSearchTests/SearchFunctionalTests.cs | 63 +++++++++++ .../RediSearchTests/SearchTests.cs | 101 ++++++++++++------ .../RedisSetupCollection.cs | 4 + 13 files changed, 310 insertions(+), 100 deletions(-) diff --git a/src/Redis.OM/Aggregation/AggregationPredicates/QueryPredicate.cs b/src/Redis.OM/Aggregation/AggregationPredicates/QueryPredicate.cs index c6f30f0..3e9773e 100644 --- a/src/Redis.OM/Aggregation/AggregationPredicates/QueryPredicate.cs +++ b/src/Redis.OM/Aggregation/AggregationPredicates/QueryPredicate.cs @@ -40,6 +40,7 @@ public IEnumerable Serialize() /// protected override void ValidateAndPushOperand(Expression expression, Stack stack) { + var dialect = 1; if (expression is BinaryExpression binaryExpression) { var memberExpression = binaryExpression.Left as MemberExpression; @@ -81,7 +82,7 @@ protected override void ValidateAndPushOperand(Expression expression, Stack()); // hack - will need to revisit when integrating vectors into aggregations. + var val = ExpressionParserUtilities.GetOperandStringForQueryArgs(binaryExpression.Right, new List(), ref dialect); // hack - will need to revisit when integrating vectors into aggregations. stack.Push(BuildQueryPredicate(binaryExpression.NodeType, memberExpression, val)); } } @@ -92,7 +93,7 @@ protected override void ValidateAndPushOperand(Expression expression, Stack())); + stack.Push(ExpressionParserUtilities.TranslateMethodExpressions(method, new List(), ref dialect)); } else if (expression is UnaryExpression uni) { diff --git a/src/Redis.OM/Common/ExpressionParserUtilities.cs b/src/Redis.OM/Common/ExpressionParserUtilities.cs index 1dda74a..db8fde4 100644 --- a/src/Redis.OM/Common/ExpressionParserUtilities.cs +++ b/src/Redis.OM/Common/ExpressionParserUtilities.cs @@ -76,19 +76,20 @@ internal static string GetOperandString(MethodCallExpression exp) /// /// expression. /// The parameters. + /// The required dialect by the final query expression. /// Treat enum as an integer. /// Whether or not to negate the result. /// Treats a boolean member expression as unary. /// the operand string. /// thrown if expression is un-parseable. - internal static string GetOperandStringForQueryArgs(Expression exp, List parameters, bool treatEnumsAsInt = false, bool negate = false, bool treatBooleanMemberAsUnary = false) + internal static string GetOperandStringForQueryArgs(Expression exp, List parameters, ref int dialectNeeded, bool treatEnumsAsInt = false, bool negate = false, bool treatBooleanMemberAsUnary = false) { var res = exp switch { ConstantExpression constExp => ValueToString(constExp.Value), MemberExpression member => GetOperandStringForMember(member, treatEnumsAsInt, negate: negate, treatBooleanMemberAsUnary: treatBooleanMemberAsUnary), - MethodCallExpression method => TranslateMethodStandardQuerySyntax(method, parameters), - UnaryExpression unary => GetOperandStringForQueryArgs(unary.Operand, parameters, treatEnumsAsInt, unary.NodeType == ExpressionType.Not, treatBooleanMemberAsUnary: treatBooleanMemberAsUnary), + MethodCallExpression method => TranslateMethodStandardQuerySyntax(method, parameters, ref dialectNeeded), + UnaryExpression unary => GetOperandStringForQueryArgs(unary.Operand, parameters, ref dialectNeeded, treatEnumsAsInt, unary.NodeType == ExpressionType.Not, treatBooleanMemberAsUnary: treatBooleanMemberAsUnary), _ => throw new ArgumentException("Unrecognized Expression type") }; @@ -97,6 +98,12 @@ internal static string GetOperandStringForQueryArgs(Expression exp, List negate = false; } + if (string.IsNullOrEmpty(res)) + { + res = "\"\""; + dialectNeeded |= 2; + } + if (negate) { return $"-{res}"; @@ -182,13 +189,14 @@ internal static string ParseBinaryExpression(BinaryExpression rootBinaryExpressi /// /// the expression. /// The parameters to be passed into the query. + /// The dialect required by the final query expression. /// The expression translated. /// thrown if the method isn't recognized. - internal static string TranslateMethodExpressions(MethodCallExpression exp, List parameters) + internal static string TranslateMethodExpressions(MethodCallExpression exp, List parameters, ref int dialectNeeded) { return exp.Method.Name switch { - "Contains" => TranslateContainsStandardQuerySyntax(exp, parameters), + "Contains" => TranslateContainsStandardQuerySyntax(exp, parameters, ref dialectNeeded), nameof(StringExtension.FuzzyMatch) => TranslateFuzzyMatch(exp), nameof(StringExtension.MatchContains) => TranslateMatchContains(exp), nameof(StringExtension.MatchPattern) => TranslateMatchPattern(exp), @@ -197,7 +205,7 @@ internal static string TranslateMethodExpressions(MethodCallExpression exp, List nameof(VectorExtensions.VectorRange) => TranslateVectorRange(exp, parameters), nameof(string.StartsWith) => TranslateStartsWith(exp), nameof(string.EndsWith) => TranslateEndsWith(exp), - "Any" => TranslateAnyForEmbeddedObjects(exp, parameters), + "Any" => TranslateAnyForEmbeddedObjects(exp, parameters, ref dialectNeeded), _ => throw new ArgumentException($"Unrecognized method for query translation:{exp.Method.Name}") }; } @@ -697,7 +705,7 @@ private static IEnumerable SplitBinaryExpression(BinaryExpress while (true); } - private static string TranslateMethodStandardQuerySyntax(MethodCallExpression exp, List parameters) + private static string TranslateMethodStandardQuerySyntax(MethodCallExpression exp, List parameters, ref int dialectNeeded) { return exp.Method.Name switch { @@ -707,11 +715,11 @@ private static string TranslateMethodStandardQuerySyntax(MethodCallExpression ex nameof(StringExtension.MatchEndsWith) => TranslateEndsWith(exp), nameof(StringExtension.MatchPattern) => TranslateMatchPattern(exp), nameof(string.Format) => TranslateFormatMethodStandardQuerySyntax(exp), - nameof(string.Contains) => TranslateContainsStandardQuerySyntax(exp, parameters), + nameof(string.Contains) => TranslateContainsStandardQuerySyntax(exp, parameters, ref dialectNeeded), nameof(string.StartsWith) => TranslateStartsWith(exp), nameof(string.EndsWith) => TranslateEndsWith(exp), nameof(VectorExtensions.VectorRange) => TranslateVectorRange(exp, parameters), - "Any" => TranslateAnyForEmbeddedObjects(exp, parameters), + "Any" => TranslateAnyForEmbeddedObjects(exp, parameters, ref dialectNeeded), _ => throw new InvalidOperationException($"Unable to parse method {exp.Method.Name}") }; } @@ -855,7 +863,7 @@ private static string TranslateFuzzyMatch(MethodCallExpression exp) }; } - private static string TranslateContainsStandardQuerySyntax(MethodCallExpression exp, List parameters) + private static string TranslateContainsStandardQuerySyntax(MethodCallExpression exp, List parameters, ref int dialectNeeded) { MemberExpression? expression = null; Type type; @@ -866,7 +874,7 @@ private static string TranslateContainsStandardQuerySyntax(MethodCallExpression { var propertyExpression = (MemberExpression)exp.Arguments.Last(); var valuesExpression = (MemberExpression)exp.Arguments.First(); - literal = GetOperandStringForQueryArgs(propertyExpression, parameters); + literal = GetOperandStringForQueryArgs(propertyExpression, parameters, ref dialectNeeded); if (!literal.StartsWith("@")) { if (exp.Arguments.Count == 1 && exp.Object != null) @@ -905,7 +913,7 @@ private static string TranslateContainsStandardQuerySyntax(MethodCallExpression var valueType = Nullable.GetUnderlyingType(valuesExpression.Type) ?? valuesExpression.Type; memberName = GetOperandStringForMember(propertyExpression); var treatEnumsAsInts = type.IsEnum && !(propertyExpression.Member.GetCustomAttributes(typeof(JsonConverterAttribute)).FirstOrDefault() is JsonConverterAttribute converter && converter.ConverterType == typeof(JsonStringEnumConverter)); - literal = GetOperandStringForQueryArgs(valuesExpression, parameters, treatEnumsAsInts); + literal = GetOperandStringForQueryArgs(valuesExpression, parameters, ref dialectNeeded, treatEnumsAsInts); if ((valueType == typeof(List) || valueType == typeof(string[]) || type == typeof(string[]) || type == typeof(List) || type == typeof(Guid) || type == typeof(Guid[]) || type == typeof(List) || type == typeof(Guid[]) || type == typeof(List) || type == typeof(Ulid) || (type.IsEnum && !treatEnumsAsInts)) && attribute is IndexedAttribute) { @@ -956,7 +964,7 @@ private static string TranslateContainsStandardQuerySyntax(MethodCallExpression type = Nullable.GetUnderlyingType(expression.Type) ?? expression.Type; memberName = GetOperandStringForMember(expression); - literal = GetOperandStringForQueryArgs(exp.Arguments.Last(), parameters); + literal = GetOperandStringForQueryArgs(exp.Arguments.Last(), parameters, ref dialectNeeded); if (searchFieldAttribute is not null && searchFieldAttribute is SearchableAttribute) { @@ -1005,7 +1013,7 @@ private static string GetContainsStringForConstantExpression(string propertyName return sb.ToString(); } - private static string TranslateAnyForEmbeddedObjects(MethodCallExpression exp, List parameters) + private static string TranslateAnyForEmbeddedObjects(MethodCallExpression exp, List parameters, ref int dialectNeeded) { var type = exp.Arguments.Last().Type; var prefix = GetOperandString(exp.Arguments[0]); @@ -1013,12 +1021,12 @@ private static string TranslateAnyForEmbeddedObjects(MethodCallExpression exp, L if (lambda.Body is MethodCallExpression methodCall) { - var tempQuery = TranslateMethodExpressions(methodCall, parameters); + var tempQuery = TranslateMethodExpressions(methodCall, parameters, ref dialectNeeded); return tempQuery.Replace("@", $"{prefix}_"); } else { - var tempQuery = ExpressionTranslator.TranslateBinaryExpression((BinaryExpression)lambda.Body, parameters); + var tempQuery = ExpressionTranslator.TranslateBinaryExpression((BinaryExpression)lambda.Body, parameters, ref dialectNeeded); return tempQuery.Replace("@", $"{prefix}_"); } } diff --git a/src/Redis.OM/Common/ExpressionTranslator.cs b/src/Redis.OM/Common/ExpressionTranslator.cs index 330f963..690d58a 100644 --- a/src/Redis.OM/Common/ExpressionTranslator.cs +++ b/src/Redis.OM/Common/ExpressionTranslator.cs @@ -187,6 +187,7 @@ internal static RedisQuery BuildQueryFromExpression(Expression expression, Type var parameters = new List(); var indexName = string.IsNullOrEmpty(attr.IndexName) ? $"{type.Name.ToLower()}-idx" : attr.IndexName; var query = new RedisQuery(indexName!) { QueryText = "*" }; + var dialect = 1; switch (expression) { case MethodCallExpression methodExpression: @@ -229,7 +230,8 @@ internal static RedisQuery BuildQueryFromExpression(Expression expression, Type query.GeoFilter = ExpressionParserUtilities.TranslateGeoFilter(exp); break; case "Where": - query.QueryText = query.QueryText == "*" ? TranslateWhereMethod(exp, parameters) : $"({TranslateWhereMethod(exp, parameters)} {query.QueryText})"; + query.QueryText = query.QueryText == "*" ? TranslateWhereMethod(exp, parameters, ref dialect) : $"({TranslateWhereMethod(exp, parameters, ref dialect)} {query.QueryText})"; + query.Dialect = dialect; break; case "NearestNeighbors": query.NearestNeighbors = ParseNearestNeighborsFromExpression(exp); @@ -241,14 +243,16 @@ internal static RedisQuery BuildQueryFromExpression(Expression expression, Type } case LambdaExpression lambda: - query.QueryText = BuildQueryFromExpression(lambda.Body, parameters); + query.QueryText = BuildQueryFromExpression(lambda.Body, parameters, ref dialect); + query.Dialect = dialect; break; } if (mainBooleanExpression != null) { parameters = new List(); - query.QueryText = BuildQueryFromExpression(((LambdaExpression)mainBooleanExpression).Body, parameters); + query.QueryText = BuildQueryFromExpression(((LambdaExpression)mainBooleanExpression).Body, parameters, ref dialect); + query.Dialect = dialect; } query.Parameters = parameters; @@ -331,51 +335,53 @@ internal static SearchFieldType DetermineIndexFieldsType(MemberInfo member) /// /// The Binary Expression. /// The parameters of the query. + /// The dialect needed for the final query expression. /// The query string formatted from the binary expression. /// Thrown if expression is not parsable because of the arguments passed into it. - internal static string TranslateBinaryExpression(BinaryExpression binExpression, List parameters) + internal static string TranslateBinaryExpression(BinaryExpression binExpression, List parameters, ref int dialectNeeded) { var sb = new StringBuilder(); if (binExpression.Left is BinaryExpression leftBin && binExpression.Right is BinaryExpression rightBin) { sb.Append("("); - sb.Append(TranslateBinaryExpression(leftBin, parameters)); + sb.Append(TranslateBinaryExpression(leftBin, parameters, ref dialectNeeded)); sb.Append(SplitPredicateSeporators(binExpression.NodeType)); - sb.Append(TranslateBinaryExpression(rightBin, parameters)); + sb.Append(TranslateBinaryExpression(rightBin, parameters, ref dialectNeeded)); sb.Append(")"); } else if (binExpression.Left is BinaryExpression left) { sb.Append("("); - sb.Append(TranslateBinaryExpression(left, parameters)); + sb.Append(TranslateBinaryExpression(left, parameters, ref dialectNeeded)); sb.Append(SplitPredicateSeporators(binExpression.NodeType)); - sb.Append(ExpressionParserUtilities.GetOperandStringForQueryArgs(binExpression.Right, parameters, treatBooleanMemberAsUnary: true)); + sb.Append(ExpressionParserUtilities.GetOperandStringForQueryArgs(binExpression.Right, parameters, ref dialectNeeded, treatBooleanMemberAsUnary: true)); sb.Append(")"); } else if (binExpression.Right is BinaryExpression right) { sb.Append("("); - sb.Append(ExpressionParserUtilities.GetOperandStringForQueryArgs(binExpression.Left, parameters, treatBooleanMemberAsUnary: true)); + sb.Append(ExpressionParserUtilities.GetOperandStringForQueryArgs(binExpression.Left, parameters, ref dialectNeeded, treatBooleanMemberAsUnary: true)); sb.Append(SplitPredicateSeporators(binExpression.NodeType)); - sb.Append(TranslateBinaryExpression(right, parameters)); + sb.Append(TranslateBinaryExpression(right, parameters, ref dialectNeeded)); sb.Append(")"); } else { - var leftContent = ExpressionParserUtilities.GetOperandStringForQueryArgs(binExpression.Left, parameters, treatBooleanMemberAsUnary: true); + var leftContent = ExpressionParserUtilities.GetOperandStringForQueryArgs(binExpression.Left, parameters, ref dialectNeeded, treatBooleanMemberAsUnary: true); var rightResolvesToNull = ExpressionParserUtilities.ExpressionResolvesToNull(binExpression.Right); if (rightResolvesToNull) { + dialectNeeded |= 1 << 1; return $"(ismissing({leftContent}))"; } - var rightContent = ExpressionParserUtilities.GetOperandStringForQueryArgs(binExpression.Right, parameters, treatBooleanMemberAsUnary: true); + var rightContent = ExpressionParserUtilities.GetOperandStringForQueryArgs(binExpression.Right, parameters, ref dialectNeeded, treatBooleanMemberAsUnary: true); if (binExpression.Left is MemberExpression member) { - var predicate = BuildQueryPredicate(binExpression.NodeType, leftContent, rightContent, member); + var predicate = BuildQueryPredicate(binExpression.NodeType, leftContent, rightContent, member, dialectNeeded); sb.Append("("); sb.Append(predicate); sb.Append(")"); @@ -403,7 +409,7 @@ internal static string TranslateBinaryExpression(BinaryExpression binExpression, } } - predicate = BuildQueryPredicate(binExpression.NodeType, leftContent, rightContent, member); + predicate = BuildQueryPredicate(binExpression.NodeType, leftContent, rightContent, member, dialectNeeded); } else { @@ -744,34 +750,16 @@ private static RedisSortBy TranslateOrderByMethod(MethodCallExpression expressio return sb; } - private static string TranslateUnaryOrMemberExpressionIntoBooleanQuery(Expression expression, List parameters) - { - if (expression is MemberExpression member && member.Type == typeof(bool)) - { - var propertyName = ExpressionParserUtilities.GetOperandStringForQueryArgs(member, parameters); - return $"{propertyName}:{{true}}"; - } - - if (expression is UnaryExpression uni && uni.Operand is MemberExpression uniMember && uniMember.Type == typeof(bool) && uni.NodeType is ExpressionType.Not) - { - var propertyName = ExpressionParserUtilities.GetOperandStringForQueryArgs(uniMember, parameters); - return $"{propertyName}:{{false}}"; - } - - throw new InvalidOperationException( - $"Could not translate expression of type {expression.Type} to a boolean expression"); - } - - private static string BuildQueryFromExpression(Expression exp, List parameters) + private static string BuildQueryFromExpression(Expression exp, List parameters, ref int dialect) { if (exp is BinaryExpression binExp) { - return TranslateBinaryExpression(binExp, parameters); + return TranslateBinaryExpression(binExp, parameters, ref dialect); } if (exp is MethodCallExpression method) { - return ExpressionParserUtilities.TranslateMethodExpressions(method, parameters); + return ExpressionParserUtilities.TranslateMethodExpressions(method, parameters, ref dialect); } if (exp is UnaryExpression uni) @@ -782,7 +770,7 @@ private static string BuildQueryFromExpression(Expression exp, List para return $"{propertyName}:{{false}}"; } - var operandString = BuildQueryFromExpression(uni.Operand, parameters); + var operandString = BuildQueryFromExpression(uni.Operand, parameters, ref dialect); if (uni.NodeType == ExpressionType.Not) { operandString = $"-{operandString}"; @@ -800,14 +788,14 @@ private static string BuildQueryFromExpression(Expression exp, List para throw new ArgumentException("Unparseable Lambda Body detected"); } - private static string TranslateWhereMethod(MethodCallExpression expression, List parameters) + private static string TranslateWhereMethod(MethodCallExpression expression, List parameters, ref int dialect) { var predicate = (UnaryExpression)expression.Arguments[1]; var lambda = (LambdaExpression)predicate.Operand; - return BuildQueryFromExpression(lambda.Body, parameters); + return BuildQueryFromExpression(lambda.Body, parameters, ref dialect); } - private static string BuildQueryPredicate(ExpressionType expType, string left, string right, MemberExpression memberExpression) + private static string BuildQueryPredicate(ExpressionType expType, string left, string right, MemberExpression memberExpression, int dialect) { var queryPredicate = expType switch { @@ -815,8 +803,8 @@ private static string BuildQueryPredicate(ExpressionType expType, string left, s ExpressionType.LessThan => $"{left}:[-inf ({right}]", ExpressionType.GreaterThanOrEqual => $"{left}:[{right} inf]", ExpressionType.LessThanOrEqual => $"{left}:[-inf {right}]", - ExpressionType.Equal => BuildEqualityPredicate(memberExpression, right), - ExpressionType.NotEqual => BuildEqualityPredicate(memberExpression, right, true), + ExpressionType.Equal => BuildEqualityPredicate(memberExpression, right, dialect), + ExpressionType.NotEqual => BuildEqualityPredicate(memberExpression, right, dialect, true), ExpressionType.And or ExpressionType.AndAlso => $"{left} {right}", ExpressionType.Or or ExpressionType.OrElse => $"{left} | {right}", _ => string.Empty @@ -824,7 +812,7 @@ private static string BuildQueryPredicate(ExpressionType expType, string left, s return queryPredicate; } - private static string BuildEqualityPredicate(MemberExpression member, string right, bool negated = false) + private static string BuildEqualityPredicate(MemberExpression member, string right, int dialect, bool negated = false) { var sb = new StringBuilder(); var fieldAttribute = ExpressionParserUtilities.DetermineSearchAttribute(member); @@ -846,7 +834,16 @@ private static string BuildEqualityPredicate(MemberExpression member, string rig switch (searchFieldType) { case SearchFieldType.TAG: - sb.Append($"{{{ExpressionParserUtilities.EscapeTagField(right)}}}"); + // special case for sending empty string query. + if (dialect > 1 && right.Equals("\"\"")) + { + sb.Append($"{{{right}}}"); + } + else + { + sb.Append($"{{{ExpressionParserUtilities.EscapeTagField(right)}}}"); + } + break; case SearchFieldType.TEXT: sb.Append($"\"{right}\""); diff --git a/src/Redis.OM/Modeling/RedisIndex.cs b/src/Redis.OM/Modeling/RedisIndex.cs index a65f489..8a5bf16 100644 --- a/src/Redis.OM/Modeling/RedisIndex.cs +++ b/src/Redis.OM/Modeling/RedisIndex.cs @@ -132,6 +132,16 @@ public static bool IndexDefinitionEquals(this RedisIndexInfo redisIndexInfo, Typ { attr.Add("SEPARATOR"); attr.Add(a.Separator ?? "|"); + + if (a.IndexMissing == true) + { + attr.Add("INDEXMISSING"); + } + + if (a.IndexEmpty == true) + { + attr.Add("INDEXEMPTY"); + } } if (a.Type == "TEXT") @@ -146,6 +156,16 @@ public static bool IndexDefinitionEquals(this RedisIndexInfo redisIndexInfo, Typ attr.Add("WEIGHT"); attr.Add(a.Weight); } + + if (a.IndexMissing == true) + { + attr.Add("INDEXMISSING"); + } + + if (a.IndexEmpty == true) + { + attr.Add("INDEXEMPTY"); + } } if (a.Type == "VECTOR") diff --git a/src/Redis.OM/Modeling/RedisSchemaField.cs b/src/Redis.OM/Modeling/RedisSchemaField.cs index 4f6d4da..e06ab06 100644 --- a/src/Redis.OM/Modeling/RedisSchemaField.cs +++ b/src/Redis.OM/Modeling/RedisSchemaField.cs @@ -300,6 +300,12 @@ private static string[] CommonSerialization(SearchFieldAttribute attr, Type decl ret.Add("WEIGHT"); ret.Add(text.Weight.ToString(CultureInfo.InvariantCulture)); } + + if (text.IndexEmptyAndMissing) + { + ret.Add("INDEXMISSING"); + ret.Add("INDEXEMPTY"); + } } if (searchFieldType == "TAG" && attr is IndexedAttribute tag) @@ -319,6 +325,12 @@ private static string[] CommonSerialization(SearchFieldAttribute attr, Type decl { ret.Add("CASESENSITIVE"); } + + if (tag.IndexEmptyAndMissing) + { + ret.Add("INDEXMISSING"); + ret.Add("INDEXEMPTY"); + } } if (searchFieldType == "VECTOR" && attr is IndexedAttribute vector) diff --git a/src/Redis.OM/Modeling/SearchFieldAttribute.cs b/src/Redis.OM/Modeling/SearchFieldAttribute.cs index 7b3d4ad..86ac003 100644 --- a/src/Redis.OM/Modeling/SearchFieldAttribute.cs +++ b/src/Redis.OM/Modeling/SearchFieldAttribute.cs @@ -38,6 +38,12 @@ public abstract class SearchFieldAttribute : RedisFieldAttribute /// public int CascadeDepth { get; set; } + /// + /// Gets or sets a value indicating whether to index empty and missing values. + /// If this is true (the default) you will be able to query null and empty string values in Redis. + /// + public bool IndexEmptyAndMissing { get; set; } = true; + /// /// Gets the type of index. /// diff --git a/src/Redis.OM/RedisIndexInfo.cs b/src/Redis.OM/RedisIndexInfo.cs index dfcd095..a295c3e 100644 --- a/src/Redis.OM/RedisIndexInfo.cs +++ b/src/Redis.OM/RedisIndexInfo.cs @@ -273,11 +273,11 @@ public RedisIndexInfoAttribute(RedisReply redisReply) var responseArray = redisReply.ToArray(); var infoIndex = 0; - while (infoIndex < responseArray.Length - 1) + while (infoIndex < responseArray.Length) { var key = responseArray[infoIndex].ToString(CultureInfo.InvariantCulture); infoIndex++; - var value = responseArray[infoIndex].ToString(CultureInfo.InvariantCulture); + var value = infoIndex < responseArray.Length ? responseArray[infoIndex].ToString(CultureInfo.InvariantCulture) : string.Empty; switch (key) { @@ -292,6 +292,8 @@ public RedisIndexInfoAttribute(RedisReply redisReply) case "M": M = value; break; case "ef_construction": EfConstruction = value; break; case "WEIGHT": Weight = value; break; + case "INDEXMISSING": IndexMissing = true; break; + case "INDEXEMPTY": IndexEmpty = true; break; } } @@ -337,6 +339,16 @@ public RedisIndexInfoAttribute(RedisReply redisReply) /// public bool? NoStem { get; } + /// + /// Gets INDEXMISSING. + /// + public bool? IndexMissing { get; } + + /// + /// Gets INDEXEMPTY. + /// + public bool? IndexEmpty { get; } + /// /// Gets weight. /// diff --git a/src/Redis.OM/Searching/Query/RedisQuery.cs b/src/Redis.OM/Searching/Query/RedisQuery.cs index f0e345b..ecde5ae 100644 --- a/src/Redis.OM/Searching/Query/RedisQuery.cs +++ b/src/Redis.OM/Searching/Query/RedisQuery.cs @@ -70,6 +70,11 @@ public RedisQuery(string index) /// public RedisSortBy? SortBy { get; set; } + /// + /// Gets or sets The Dialect to use in the query. + /// + public int Dialect { get; set; } = 1; + /// /// Serializes the query into a set of arguments. /// @@ -107,8 +112,13 @@ internal object[] SerializeQuery() ret.Add(parameters[i]); } + Dialect |= 1 << 1; + } + + if (DialectToSend() > 1) + { ret.Add("DIALECT"); - ret.Add(2); + ret.Add(DialectToSend()); } foreach (var flag in (QueryFlags[])Enum.GetValues(typeof(QueryFlags))) @@ -146,5 +156,16 @@ internal object[] SerializeQuery() return ret.ToArray(); } + + private int DialectToSend() + { + const int DIALECT_TWO_MASK = 0x2; + if ((Dialect & DIALECT_TWO_MASK) == DIALECT_TWO_MASK) + { + return 2; + } + + return 1; + } } } diff --git a/test/Redis.OM.Unit.Tests/RediSearchTests/ObjectWithNullableStrings.cs b/test/Redis.OM.Unit.Tests/RediSearchTests/ObjectWithNullableStrings.cs index f04bf60..8ec1044 100644 --- a/test/Redis.OM.Unit.Tests/RediSearchTests/ObjectWithNullableStrings.cs +++ b/test/Redis.OM.Unit.Tests/RediSearchTests/ObjectWithNullableStrings.cs @@ -1,3 +1,6 @@ +using System; +using System.Runtime.InteropServices; +using System.Text.Json.Serialization; using Redis.OM.Modeling; namespace Redis.OM.Unit.Tests.RediSearchTests; @@ -12,6 +15,36 @@ public class ObjectWithNullableStrings [Indexed] public string? String1 { get; set; } + [Searchable] + public string? String2 { get; set; } + + [Indexed] + public Guid? Guid { get; set; } + + [Indexed] + public bool? Bool { get; set; } + + [Indexed] + [JsonConverter(typeof(JsonStringEnumConverter))] + public AnEnum? Enum { get; set; } +} + +[Document] +public class ObjectWithNullableStringsHash +{ + [RedisIdField] [Indexed] + public string Id { get; set; } + + [Indexed] + public string? String1 { get; set; } + + [Searchable] public string? String2 { get; set; } + + [Indexed] + public Guid? Guid { get; set; } + + [Indexed] + public bool? Bool { get; set; } } \ No newline at end of file diff --git a/test/Redis.OM.Unit.Tests/RediSearchTests/RedisIndexTests.cs b/test/Redis.OM.Unit.Tests/RediSearchTests/RedisIndexTests.cs index 045989e..ca86f6a 100644 --- a/test/Redis.OM.Unit.Tests/RediSearchTests/RedisIndexTests.cs +++ b/test/Redis.OM.Unit.Tests/RediSearchTests/RedisIndexTests.cs @@ -119,7 +119,7 @@ public void TestIndexSerializationHappyPath() { var expected = new[] { "TestPersonClassHappyPath-idx", "ON", "Hash", "PREFIX", "1", "Redis.OM.Unit.Tests.RediSearchTests.RedisIndexTests+TestPersonClassHappyPath:", "SCHEMA", - "Name", "TEXT", "SORTABLE", "Age", "NUMERIC", "SORTABLE" }; + "Name", "TEXT", "INDEXMISSING", "INDEXEMPTY", "SORTABLE", "Age", "NUMERIC", "SORTABLE" }; var indexArr = typeof(TestPersonClassHappyPath).SerializeIndex(); Assert.True(expected.SequenceEqual(indexArr)); @@ -130,7 +130,7 @@ public void TestIndexSerializationOverridenPrefix() { var expected = new[] { "TestPersonClassHappyPath-idx", "ON", "Hash", "PREFIX", "1", "Person:", "SCHEMA", - "Name", "TEXT", "SORTABLE", "Age", "NUMERIC", "SORTABLE" }; + "Name", "TEXT", "INDEXMISSING", "INDEXEMPTY", "SORTABLE", "Age", "NUMERIC", "SORTABLE" }; var indexArr = typeof(TestPersonClassOverridenPrefix).SerializeIndex(); Assert.True(expected.SequenceEqual(indexArr)); diff --git a/test/Redis.OM.Unit.Tests/RediSearchTests/SearchFunctionalTests.cs b/test/Redis.OM.Unit.Tests/RediSearchTests/SearchFunctionalTests.cs index 506aceb..d46542e 100644 --- a/test/Redis.OM.Unit.Tests/RediSearchTests/SearchFunctionalTests.cs +++ b/test/Redis.OM.Unit.Tests/RediSearchTests/SearchFunctionalTests.cs @@ -1311,5 +1311,68 @@ public async Task TestUpdateByteArray() Assert.Equal(new byte[] { 7, 8, 9 }, updated.Bytes1); Assert.Equal(new byte[] { 10, 11, 12 }, updated.Bytes2); } + + [Fact] + public async Task TestQueryNullStrings() + { + var collection = new RedisCollection(_connection); + var obj = new ObjectWithNullableStrings(); + await collection.InsertAsync(obj); + + var res = await collection.FirstAsync(x=>x.Id == obj.Id && x.String1 == null && x.String2 == null && x.Guid == null && x.Bool == null && x.Enum == null); + + Assert.NotNull(res); + Assert.Null(res.String1); + Assert.Null(res.String2); + Assert.Null(res.Guid); + Assert.Null(res.Bool); + Assert.Null(res.Enum); + Assert.Equal(obj.Id, res.Id); + + res.Bool = true; + res.String1 = "Hello"; + res.String2 = "World"; + res.Guid = Guid.NewGuid(); + res.Enum = AnEnum.one; + await collection.UpdateAsync(res); + + var updated = await collection.FirstAsync(x=>x.Id == obj.Id); + Assert.NotNull(updated); + Assert.Equal("Hello", updated.String1); + Assert.Equal("World", updated.String2); + Assert.NotNull(updated.Guid); + Assert.True(updated.Bool); + Assert.Equal(AnEnum.one, updated.Enum); + } + + [Fact] + public async Task TestQueryNullStringsHash() + { + var collection = new RedisCollection(_connection); + var obj = new ObjectWithNullableStringsHash(); + await collection.InsertAsync(obj); + + var res = await collection.FirstAsync(x=>x.Id == obj.Id && x.String1 == null && x.String2 == null && x.Guid == null && x.Bool == null); + + Assert.NotNull(res); + Assert.Null(res.String1); + Assert.Null(res.String2); + Assert.Null(res.Guid); + Assert.Null(res.Bool); + Assert.Equal(obj.Id, res.Id); + + res.Bool = true; + res.String1 = "Hello"; + res.String2 = "World"; + res.Guid = Guid.NewGuid(); + await collection.UpdateAsync(res); + + var updated = await collection.FirstAsync(x=>x.Id == obj.Id); + Assert.NotNull(updated); + Assert.Equal("Hello", updated.String1); + Assert.Equal("World", updated.String2); + Assert.NotNull(updated.Guid); + Assert.True(updated.Bool); + } } } \ No newline at end of file diff --git a/test/Redis.OM.Unit.Tests/RediSearchTests/SearchTests.cs b/test/Redis.OM.Unit.Tests/RediSearchTests/SearchTests.cs index 236148f..2990dac 100644 --- a/test/Redis.OM.Unit.Tests/RediSearchTests/SearchTests.cs +++ b/test/Redis.OM.Unit.Tests/RediSearchTests/SearchTests.cs @@ -1970,7 +1970,7 @@ await _substitute.Received().ExecuteAsync( $"Redis.OM.Unit.Tests.{nameof(ObjectWithZeroStopwords)}:", "STOPWORDS", "0", - "SCHEMA", "Name", "TAG", "SEPARATOR", "|"); + "SCHEMA", "Name", "TAG", "SEPARATOR", "|", "INDEXMISSING", "INDEXEMPTY"); } [Fact] @@ -1990,7 +1990,7 @@ await _substitute.Received().ExecuteAsync( "1", $"Redis.OM.Unit.Tests.{nameof(ObjectWithTwoStopwords)}:", "STOPWORDS", "2", "foo", "bar", - "SCHEMA", "Name", "TAG", "SEPARATOR", "|"); + "SCHEMA", "Name", "TAG", "SEPARATOR", "|", "INDEXMISSING", "INDEXEMPTY"); } [Fact] @@ -2009,12 +2009,12 @@ await _substitute.Received().ExecuteAsync("FT.CREATE", "1", "Redis.OM.Unit.Tests.RediSearchTests.ObjectWithStringLikeValueTypes:", "SCHEMA", - "$.Ulid", "AS", "Ulid", "TAG", "SEPARATOR", "|", - "$.Boolean", "AS", "Boolean", "TAG", "SEPARATOR", "|", - "$.Guid", "AS", "Guid", "TAG", "SEPARATOR", "|", - "$.AnEnum", "AS", "AnEnum", "TAG", + "$.Ulid", "AS", "Ulid", "TAG", "SEPARATOR", "|", "INDEXMISSING", "INDEXEMPTY", + "$.Boolean", "AS", "Boolean", "TAG", "SEPARATOR", "|", "INDEXMISSING", "INDEXEMPTY", + "$.Guid", "AS", "Guid", "TAG", "SEPARATOR", "|", "INDEXMISSING", "INDEXEMPTY", + "$.AnEnum", "AS", "AnEnum", "TAG", "INDEXMISSING", "INDEXEMPTY", "$.AnEnumAsInt", "AS", "AnEnumAsInt", "NUMERIC", - "$.Flags", "AS", "Flags", "TAG", "SEPARATOR", "," + "$.Flags", "AS", "Flags", "TAG", "SEPARATOR", ",", "INDEXMISSING", "INDEXEMPTY" ); } @@ -2035,9 +2035,9 @@ await _substitute.Received().ExecuteAsync("FT.CREATE", "Redis.OM.Unit.Tests.RediSearchTests.ObjectWithStringLikeValueTypesHash:", "SCHEMA", "Ulid", - "TAG", "SEPARATOR", "|", + "TAG", "SEPARATOR", "|", "INDEXMISSING", "INDEXEMPTY", "Boolean", - "TAG", "SEPARATOR", "|", "Guid", "TAG", "SEPARATOR", "|", "AnEnum", "NUMERIC" + "TAG", "SEPARATOR", "|", "INDEXMISSING", "INDEXEMPTY", "Guid", "TAG", "SEPARATOR", "|", "INDEXMISSING", "INDEXEMPTY", "AnEnum", "NUMERIC" ); } @@ -2185,15 +2185,15 @@ await _substitute.Received().ExecuteAsync("FT.CREATE", "1", "Redis.OM.Unit.Tests.RediSearchTests.ObjectWithEmbeddedArrayOfObjects:", "SCHEMA", - "$.Addresses[*].City", "AS", "Addresses_City", "TAG", "SEPARATOR", "|", - "$.Addresses[*].State", "AS", "Addresses_State", "TAG", "SEPARATOR", "|", - "$.Addresses[*].AddressType", "AS", "Addresses_AddressType", "TAG", - "$.Addresses[*].Boolean", "AS", "Addresses_Boolean", "TAG", "SEPARATOR", "|", - "$.Addresses[*].Guid", "AS", "Addresses_Guid", "TAG", "SEPARATOR", "|", - "$.Addresses[*].Ulid", "AS", "Addresses_Ulid", "TAG", "SEPARATOR", "|", - "$.AddressList[*].City", "AS", "AddressList_City", "TAG", "SEPARATOR", "|", - "$.AddressList[*].State", "AS", "AddressList_State", "TAG", "SEPARATOR", "|", - "$.Name", "AS", "Name", "TAG", "SEPARATOR", "|", "$.Numeric", "AS", "Numeric", "NUMERIC"); + "$.Addresses[*].City", "AS", "Addresses_City", "TAG", "SEPARATOR", "|", "INDEXMISSING", "INDEXEMPTY", + "$.Addresses[*].State", "AS", "Addresses_State", "TAG", "SEPARATOR", "|", "INDEXMISSING", "INDEXEMPTY", + "$.Addresses[*].AddressType", "AS", "Addresses_AddressType", "TAG", "INDEXMISSING", "INDEXEMPTY", + "$.Addresses[*].Boolean", "AS", "Addresses_Boolean", "TAG", "SEPARATOR", "|", "INDEXMISSING", "INDEXEMPTY", + "$.Addresses[*].Guid", "AS", "Addresses_Guid", "TAG", "SEPARATOR", "|", "INDEXMISSING", "INDEXEMPTY", + "$.Addresses[*].Ulid", "AS", "Addresses_Ulid", "TAG", "SEPARATOR", "|", "INDEXMISSING", "INDEXEMPTY", + "$.AddressList[*].City", "AS", "AddressList_City", "TAG", "SEPARATOR", "|", "INDEXMISSING", "INDEXEMPTY", + "$.AddressList[*].State", "AS", "AddressList_State", "TAG", "SEPARATOR", "|", "INDEXMISSING", "INDEXEMPTY", + "$.Name", "AS", "Name", "TAG", "SEPARATOR", "|", "INDEXMISSING", "INDEXEMPTY", "$.Numeric", "AS", "Numeric", "NUMERIC"); } [Fact] @@ -3486,14 +3486,14 @@ public void TestMixedNestingIndexCreation() "PREFIX", "1", $"Redis.OM.Unit.Tests.{nameof(ComplexObjectWithCascadeAndJsonPath)}:", - "SCHEMA", "$.InnerCascade.InnerInnerJson.Tag", "AS", "InnerCascade_InnerInnerJson_Tag", "TAG", "SEPARATOR", "|", - "$.InnerCascade.InnerInnerCascade.Tag", "AS", "InnerCascade_InnerInnerCascade_Tag", "TAG", "SEPARATOR", "|", + "SCHEMA", "$.InnerCascade.InnerInnerJson.Tag", "AS", "InnerCascade_InnerInnerJson_Tag", "TAG", "SEPARATOR", "|", "INDEXMISSING", "INDEXEMPTY", + "$.InnerCascade.InnerInnerCascade.Tag", "AS", "InnerCascade_InnerInnerCascade_Tag", "TAG", "SEPARATOR", "|", "INDEXMISSING", "INDEXEMPTY", "$.InnerCascade.InnerInnerCascade.Num", "AS", "InnerCascade_InnerInnerCascade_Num", "NUMERIC", - "$.InnerCascade.InnerInnerCascade.Arr[*]", "AS", "InnerCascade_InnerInnerCascade_Arr", "TAG", "SEPARATOR", "|", - "$.InnerCascade.InnerInnerCollection[*].Tag", "AS", "InnerCascade_InnerInnerCollection_Tag", "TAG", "SEPARATOR", "|", - "$.InnerJson.InnerInnerCascade.Tag", "AS", "InnerJson_InnerInnerCascade_Tag", "TAG", "SEPARATOR", "|", + "$.InnerCascade.InnerInnerCascade.Arr[*]", "AS", "InnerCascade_InnerInnerCascade_Arr", "TAG", "SEPARATOR", "|", "INDEXMISSING", "INDEXEMPTY", + "$.InnerCascade.InnerInnerCollection[*].Tag", "AS", "InnerCascade_InnerInnerCollection_Tag", "TAG", "SEPARATOR", "|", "INDEXMISSING", "INDEXEMPTY", + "$.InnerJson.InnerInnerCascade.Tag", "AS", "InnerJson_InnerInnerCascade_Tag", "TAG", "SEPARATOR", "|", "INDEXMISSING", "INDEXEMPTY", "$.InnerJson.InnerInnerCascade.Num", "AS", "InnerJson_InnerInnerCascade_Num", "NUMERIC", - "$.InnerJson.InnerInnerCascade.Arr[*]", "AS", "InnerJson_InnerInnerCascade_Arr", "TAG", "SEPARATOR", "|"); + "$.InnerJson.InnerInnerCascade.Arr[*]", "AS", "InnerJson_InnerInnerCascade_Arr", "TAG", "SEPARATOR", "|", "INDEXMISSING", "INDEXEMPTY"); } [Fact] @@ -3550,7 +3550,7 @@ await _substitute.Received().ExecuteAsync( "PREFIX", "1", $"Redis.OM.Unit.Tests.{nameof(ObjectWithPropertyNamesDefined)}:", - "SCHEMA", "$.notKey", "AS", "notKey", "TAG", "SEPARATOR", "|"); + "SCHEMA", "$.notKey", "AS", "notKey", "TAG", "SEPARATOR", "|", "INDEXMISSING", "INDEXEMPTY"); } [Fact] @@ -3996,8 +3996,8 @@ await _substitute.Received().ExecuteAsync("FT.CREATE", "1", "Redis.OM.Unit.Tests.RediSearchTests.ObjectWithMultipleSearchableAttributes:", "SCHEMA", - "$.Address.City", "AS", "Address_City", "TEXT", - "$.Address.State", "AS", "Address_State", "TEXT"); + "$.Address.City", "AS", "Address_City", "TEXT", "INDEXMISSING", "INDEXEMPTY", + "$.Address.State", "AS", "Address_State", "TEXT", "INDEXMISSING", "INDEXEMPTY"); } [Fact] @@ -4028,7 +4028,7 @@ public void TestToQueryString() } [Fact] - public async Task TestQueryForNull() + public async Task TestQueryForNullAndEmpty() { _substitute.ClearSubstitute(); _substitute.ExecuteAsync(Arg.Any(), Arg.Any()).Returns(_mockReply); @@ -4038,9 +4038,9 @@ public async Task TestQueryForNull() await _substitute.Received().ExecuteAsync("FT.SEARCH", $"{nameof(ObjectWithNullableStrings).ToLower()}-idx", "(ismissing(@String1))", - "LIMIT", "DIALECT", 2, + "LIMIT", "0", "100"); @@ -4050,13 +4050,11 @@ await _substitute.Received().ExecuteAsync("FT.SEARCH", await _substitute.Received().ExecuteAsync("FT.SEARCH", $"{nameof(ObjectWithNullableStrings).ToLower()}-idx", "(ismissing(@String1))", - "LIMIT", "DIALECT", 2, + "LIMIT", "0", "100"); - - _substitute.ClearSubstitute(); var obj = new { @@ -4065,19 +4063,54 @@ await _substitute.Received().ExecuteAsync("FT.SEARCH", val = (string?)null } }; - + + _substitute.ClearSubstitute(); _substitute.ExecuteAsync(Arg.Any(), Arg.Any()).Returns(_mockReply); await collection.Where(x => x.String2 == obj.inner.val).ToListAsync(); await _substitute.Received().ExecuteAsync("FT.SEARCH", $"{nameof(ObjectWithNullableStrings).ToLower()}-idx", "(ismissing(@String2))", + "DIALECT", + 2, "LIMIT", + "0", + "100"); + + _substitute.ClearSubstitute(); + _substitute.ExecuteAsync(Arg.Any(), Arg.Any()).Returns(_mockReply); + await collection.Where(x => x.String1 == "").ToListAsync(); + await _substitute.Received().ExecuteAsync("FT.SEARCH", + $"{nameof(ObjectWithNullableStrings).ToLower()}-idx", + "(@String1:{\"\"})", "DIALECT", 2, + "LIMIT", "0", "100"); + _substitute.ClearSubstitute(); + _substitute.ExecuteAsync(Arg.Any(), Arg.Any()).Returns(_mockReply); + await collection.Where(x => x.String2 == "").ToListAsync(); + await _substitute.Received().ExecuteAsync("FT.SEARCH", + $"{nameof(ObjectWithNullableStrings).ToLower()}-idx", + "(@String2:\"\"\"\")", + "DIALECT", + 2, + "LIMIT", + "0", + "100"); + + _substitute.ClearSubstitute(); + _substitute.ExecuteAsync(Arg.Any(), Arg.Any()).Returns(_mockReply); + await collection.Where(x => x.Guid == null && x.Enum == null && x.Bool == null).ToListAsync(); + await _substitute.Received().ExecuteAsync("FT.SEARCH", + $"{nameof(ObjectWithNullableStrings).ToLower()}-idx", + "(((ismissing(@Guid)) (ismissing(@Enum))) (ismissing(@Bool)))", + "DIALECT", + 2, + "LIMIT", + "0", + "100"); } - } } \ No newline at end of file diff --git a/test/Redis.OM.Unit.Tests/RedisSetupCollection.cs b/test/Redis.OM.Unit.Tests/RedisSetupCollection.cs index 7a60efb..cf563df 100644 --- a/test/Redis.OM.Unit.Tests/RedisSetupCollection.cs +++ b/test/Redis.OM.Unit.Tests/RedisSetupCollection.cs @@ -32,6 +32,8 @@ public RedisSetup() Connection.CreateIndex(typeof(ObjectWithDateTimeOffsetJson)); Connection.CreateIndex(typeof(ObjectWithMultipleSearchableAttributes)); Connection.CreateIndex(typeof(ObjectWithByteArray)); + Connection.CreateIndex(typeof(ObjectWithNullableStrings)); + Connection.CreateIndex(typeof(ObjectWithNullableStringsHash)); } private IRedisConnectionProvider _provider; @@ -66,6 +68,8 @@ public void Dispose() Connection.DropIndexAndAssociatedRecords(typeof(ObjectWithDateTimeOffsetJson)); Connection.DropIndexAndAssociatedRecords(typeof(ObjectWithMultipleSearchableAttributes)); Connection.DropIndexAndAssociatedRecords(typeof(ObjectWithByteArray)); + Connection.DropIndexAndAssociatedRecords(typeof(ObjectWithNullableStrings)); + Connection.DropIndexAndAssociatedRecords(typeof(ObjectWithNullableStringsHash)); } } } From 760566ad2c976070270a77956f23bd186deb3348 Mon Sep 17 00:00:00 2001 From: slorello89 Date: Tue, 26 Nov 2024 09:13:10 -0500 Subject: [PATCH 3/4] fixing vector test --- .../RediSearchTests/VectorTests/VectorTests.cs | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/test/Redis.OM.Unit.Tests/RediSearchTests/VectorTests/VectorTests.cs b/test/Redis.OM.Unit.Tests/RediSearchTests/VectorTests/VectorTests.cs index 5c03346..7df3064 100644 --- a/test/Redis.OM.Unit.Tests/RediSearchTests/VectorTests/VectorTests.cs +++ b/test/Redis.OM.Unit.Tests/RediSearchTests/VectorTests/VectorTests.cs @@ -34,7 +34,7 @@ public void CreateIndexWithVector() "1", $"Redis.OM.Unit.Tests.{nameof(ObjectWithVector)}:", "SCHEMA", - "$.Name", "AS", "Name", "TAG", "SEPARATOR", "|", + "$.Name", "AS", "Name", "TAG", "SEPARATOR", "|", "INDEXMISSING", "INDEXEMPTY", "$.Num", "AS", "Num", "NUMERIC", "$.SimpleHnswVector", "AS", "SimpleHnswVector", "VECTOR", "HNSW", "6", "TYPE", "FLOAT64", "DIM", "10", "DISTANCE_METRIC", "L2", "$.SimpleVectorizedVector.Vector", "AS","SimpleVectorizedVector", "VECTOR", "FLAT", "6", "TYPE", "FLOAT32", "DIM", "30", "DISTANCE_METRIC", "L2" @@ -51,7 +51,7 @@ public void CreateIndexWithVector() "1", $"Redis.OM.Unit.Tests.{nameof(ObjectWithVectorHash)}:", "SCHEMA", - "Name", "TAG", "SEPARATOR", "|", + "Name", "TAG", "SEPARATOR", "|", "INDEXMISSING", "INDEXEMPTY", "Num", "NUMERIC", "SimpleHnswVector", "VECTOR", "HNSW", "6", "TYPE", "FLOAT64", "DIM", "10", "DISTANCE_METRIC", "L2", "SimpleVectorizedVector.Vector", "AS", "SimpleVectorizedVector", "VECTOR", "FLAT", "6", "TYPE", "FLOAT32", "DIM", "30", "DISTANCE_METRIC", "L2" From a7c22043c6e81664424bda17b3bc36783b0dff9f Mon Sep 17 00:00:00 2001 From: slorello89 Date: Tue, 26 Nov 2024 10:08:41 -0500 Subject: [PATCH 4/4] fixing tests --- .../SearchJsonTests/RedisJsonIndexTests.cs | 66 ++++++++++--------- 1 file changed, 34 insertions(+), 32 deletions(-) diff --git a/test/Redis.OM.Unit.Tests/SearchJsonTests/RedisJsonIndexTests.cs b/test/Redis.OM.Unit.Tests/SearchJsonTests/RedisJsonIndexTests.cs index 5d2a7dc..7ab924f 100644 --- a/test/Redis.OM.Unit.Tests/SearchJsonTests/RedisJsonIndexTests.cs +++ b/test/Redis.OM.Unit.Tests/SearchJsonTests/RedisJsonIndexTests.cs @@ -105,8 +105,10 @@ public void TestIndexSerializationHappyPath() { var expected = new[] { "person-idx", "ON", "Json", "PREFIX", "1", "Redis.OM.Unit.Tests.SearchJsonTests.RedisJsonIndexTests+Person:", "SCHEMA", - "$.Name", "AS", "Name", "TEXT", "SORTABLE", "$.Tag", "AS","Tag","TAG", "SEPARATOR", "|","$.Age", "AS", "Age", - "NUMERIC","SORTABLE", "$.Height", "AS", "Height", "NUMERIC", "SORTABLE" }; + "$.Name", "AS", "Name", "TEXT", "INDEXMISSING", "INDEXEMPTY", "SORTABLE", + "$.Tag", "AS","Tag","TAG", "SEPARATOR", "|", "INDEXMISSING", "INDEXEMPTY", + "$.Age", "AS", "Age", "NUMERIC", "SORTABLE", + "$.Height", "AS", "Height", "NUMERIC", "SORTABLE" }; var indexArr = typeof(Person).SerializeIndex(); for(var i = 0; i < indexArr.Length; i++) @@ -121,21 +123,21 @@ public void TestIndexSerializationNestedObject() { var expected = new[] { "person-idx", "ON", "Json", "PREFIX", "1", "Redis.OM.Unit.Tests.SearchJsonTests.RedisJsonIndexTests+NestedPerson:", "SCHEMA", - "$.Name", "AS", "Name", "TEXT", "SORTABLE", - "$.Tag", "AS","Tag","TAG", "SEPARATOR", "|", + "$.Name", "AS", "Name", "TEXT", "SORTABLE", "INDEXMISSING", "INDEXEMPTY", + "$.Tag", "AS","Tag","TAG", "SEPARATOR", "|", "INDEXMISSING", "INDEXEMPTY", "$.Age", "AS", "Age", "NUMERIC","SORTABLE", "$.Height", "AS", "Height", "NUMERIC", "SORTABLE", - "$.Address.ZipCode", "AS", "Address_ZipCode", "TAG", "SEPARATOR", "|", - "$.Address.City", "AS", "Address_City", "TAG", "SEPARATOR", "|", - "$.Address.StreetName", "AS", "Address_StreetName", "TEXT", "SORTABLE", - "$.WorkAddress.City", "AS","WorkAddress_City", "TAG", "SEPARATOR", "|", - "$.WorkAddress.AddressType", "AS", "WorkAddress_AddressType", "TAG", - "$.WorkAddress.State", "AS", "WorkAddress_State", "TAG", "SEPARATOR", "|", + "$.Address.ZipCode", "AS", "Address_ZipCode", "TAG", "SEPARATOR", "|", "INDEXMISSING", "INDEXEMPTY", + "$.Address.City", "AS", "Address_City", "TAG", "SEPARATOR", "|", "INDEXMISSING", "INDEXEMPTY", + "$.Address.StreetName", "AS", "Address_StreetName", "TEXT", "SORTABLE", "INDEXMISSING", "INDEXEMPTY", + "$.WorkAddress.City", "AS","WorkAddress_City", "TAG", "SEPARATOR", "|", "INDEXMISSING", "INDEXEMPTY", + "$.WorkAddress.AddressType", "AS", "WorkAddress_AddressType", "TAG", "INDEXMISSING", "INDEXEMPTY", + "$.WorkAddress.State", "AS", "WorkAddress_State", "TAG", "SEPARATOR", "|", "INDEXMISSING", "INDEXEMPTY", "$.WorkAddress.Location", "AS", "WorkAddress_Location", "GEO", "$.WorkAddress.HouseNumber", "AS", "WorkAddress_HouseNumber", "NUMERIC", - "$.WorkAddress.Boolean", "AS", "WorkAddress_Boolean", "TAG", "SEPARATOR", "|", - "$.WorkAddress.Ulid", "AS", "WorkAddress_Ulid", "TAG", "SEPARATOR", "|", - "$.WorkAddress.Guid", "AS", "WorkAddress_Guid", "TAG", "SEPARATOR", "|", + "$.WorkAddress.Boolean", "AS", "WorkAddress_Boolean", "TAG", "SEPARATOR", "|", "INDEXMISSING", "INDEXEMPTY", + "$.WorkAddress.Ulid", "AS", "WorkAddress_Ulid", "TAG", "SEPARATOR", "|", "INDEXMISSING", "INDEXEMPTY", + "$.WorkAddress.Guid", "AS", "WorkAddress_Guid", "TAG", "SEPARATOR", "|", "INDEXMISSING", "INDEXEMPTY", }; var indexArr = typeof(NestedPerson).SerializeIndex(); @@ -147,29 +149,29 @@ public void TestIndexSerializationNestedObjectCascade2() { var expected = new[] { "person-idx", "ON", "Json", "PREFIX", "1", "Redis.OM.Unit.Tests.SearchJsonTests.RedisJsonIndexTests+NestedPersonCascade2:", "SCHEMA", - "$.Name", "AS", "Name", "TEXT", "SORTABLE", - "$.Tag", "AS","Tag","TAG", "SEPARATOR", "|", + "$.Name", "AS", "Name", "TEXT", "SORTABLE", "INDEXMISSING", "INDEXEMPTY", + "$.Tag", "AS","Tag","TAG", "SEPARATOR", "|", "INDEXMISSING", "INDEXEMPTY", "$.Age", "AS", "Age", "NUMERIC","SORTABLE", "$.Height", "AS", "Height", "NUMERIC", "SORTABLE", - "$.Address.ZipCode", "AS", "Address_ZipCode", "TAG", "SEPARATOR", "|", - "$.Address.City", "AS", "Address_City", "TAG", "SEPARATOR", "|", - "$.Address.StreetName", "AS", "Address_StreetName", "TEXT", "SORTABLE", - "$.WorkAddress.City", "AS","WorkAddress_City", "TAG", "SEPARATOR", "|", - "$.WorkAddress.AddressType", "AS", "WorkAddress_AddressType", "TAG", - "$.WorkAddress.State", "AS", "WorkAddress_State", "TAG", "SEPARATOR", "|", - "$.WorkAddress.ForwardingAddress.City", "AS", "WorkAddress_ForwardingAddress_City", "TAG", "SEPARATOR", "|", - "$.WorkAddress.ForwardingAddress.AddressType", "AS", "WorkAddress_ForwardingAddress_AddressType", "TAG", - "$.WorkAddress.ForwardingAddress.State", "AS", "WorkAddress_ForwardingAddress_State", "TAG", "SEPARATOR", "|", + "$.Address.ZipCode", "AS", "Address_ZipCode", "TAG", "SEPARATOR", "|", "INDEXMISSING", "INDEXEMPTY", + "$.Address.City", "AS", "Address_City", "TAG", "SEPARATOR", "|", "INDEXMISSING", "INDEXEMPTY", + "$.Address.StreetName", "AS", "Address_StreetName", "TEXT", "SORTABLE", "INDEXMISSING", "INDEXEMPTY", + "$.WorkAddress.City", "AS","WorkAddress_City", "TAG", "SEPARATOR", "|", "INDEXMISSING", "INDEXEMPTY", + "$.WorkAddress.AddressType", "AS", "WorkAddress_AddressType", "TAG", "INDEXMISSING", "INDEXEMPTY", + "$.WorkAddress.State", "AS", "WorkAddress_State", "TAG", "SEPARATOR", "|", "INDEXMISSING", "INDEXEMPTY", + "$.WorkAddress.ForwardingAddress.City", "AS", "WorkAddress_ForwardingAddress_City", "TAG", "SEPARATOR", "|", "INDEXMISSING", "INDEXEMPTY", + "$.WorkAddress.ForwardingAddress.AddressType", "AS", "WorkAddress_ForwardingAddress_AddressType", "TAG", "INDEXMISSING", "INDEXEMPTY", + "$.WorkAddress.ForwardingAddress.State", "AS", "WorkAddress_ForwardingAddress_State", "TAG", "SEPARATOR", "|", "INDEXMISSING", "INDEXEMPTY", "$.WorkAddress.ForwardingAddress.Location", "AS", "WorkAddress_ForwardingAddress_Location", "GEO", "$.WorkAddress.ForwardingAddress.HouseNumber", "AS", "WorkAddress_ForwardingAddress_HouseNumber", "NUMERIC", - "$.WorkAddress.ForwardingAddress.Boolean", "AS", "WorkAddress_ForwardingAddress_Boolean", "TAG", "SEPARATOR", "|", - "$.WorkAddress.ForwardingAddress.Ulid", "AS", "WorkAddress_ForwardingAddress_Ulid", "TAG", "SEPARATOR", "|", - "$.WorkAddress.ForwardingAddress.Guid", "AS", "WorkAddress_ForwardingAddress_Guid", "TAG", "SEPARATOR", "|", + "$.WorkAddress.ForwardingAddress.Boolean", "AS", "WorkAddress_ForwardingAddress_Boolean", "TAG", "SEPARATOR", "|", "INDEXMISSING", "INDEXEMPTY", + "$.WorkAddress.ForwardingAddress.Ulid", "AS", "WorkAddress_ForwardingAddress_Ulid", "TAG", "SEPARATOR", "|", "INDEXMISSING", "INDEXEMPTY", + "$.WorkAddress.ForwardingAddress.Guid", "AS", "WorkAddress_ForwardingAddress_Guid", "TAG", "SEPARATOR", "|", "INDEXMISSING", "INDEXEMPTY", "$.WorkAddress.Location", "AS", "WorkAddress_Location", "GEO", "$.WorkAddress.HouseNumber", "AS", "WorkAddress_HouseNumber", "NUMERIC", - "$.WorkAddress.Boolean", "AS", "WorkAddress_Boolean", "TAG", "SEPARATOR", "|", - "$.WorkAddress.Ulid", "AS", "WorkAddress_Ulid", "TAG", "SEPARATOR", "|", - "$.WorkAddress.Guid", "AS", "WorkAddress_Guid", "TAG", "SEPARATOR", "|", + "$.WorkAddress.Boolean", "AS", "WorkAddress_Boolean", "TAG", "SEPARATOR", "|", "INDEXMISSING", "INDEXEMPTY", + "$.WorkAddress.Ulid", "AS", "WorkAddress_Ulid", "TAG", "SEPARATOR", "|", "INDEXMISSING", "INDEXEMPTY", + "$.WorkAddress.Guid", "AS", "WorkAddress_Guid", "TAG", "SEPARATOR", "|", "INDEXMISSING", "INDEXEMPTY", }; var indexArr = typeof(NestedPersonCascade2).SerializeIndex(); @@ -181,8 +183,8 @@ public void TestIndexSerializationWithNickNames() { var expected = new[] { "person-idx", "ON", "Json", "PREFIX", "1", "Redis.OM.Unit.Tests.SearchJsonTests.RedisJsonIndexTests+PersonWithIndexedNickNames:", "SCHEMA", - "$.Name", "AS", "Name", "TEXT", "SORTABLE", - "$.Tag", "AS","Tag","TAG", "SEPARATOR", "|", + "$.Name", "AS", "Name", "TEXT", "INDEXMISSING", "INDEXEMPTY", "SORTABLE", + "$.Tag", "AS","Tag","TAG", "SEPARATOR", "|", "INDEXMISSING", "INDEXEMPTY", "$.Age", "AS", "Age", "NUMERIC","SORTABLE", "$.Height", "AS", "Height", "NUMERIC", "SORTABLE", "$.NickNames[*]", "AS","NickNames","TAG",