From 8d05d915e152192019993737b61f6e2ebb87add0 Mon Sep 17 00:00:00 2001 From: Steve Lorello <42971704+slorello89@users.noreply.github.com> Date: Tue, 14 Mar 2023 08:01:12 -0400 Subject: [PATCH] mixed embedded object paths (#324) --- src/Redis.OM/Modeling/RedisSchemaField.cs | 56 +++++++++++----- src/Redis.OM/Searching/RedisCollection.cs | 6 +- .../RediSearchTests/SearchFunctionalTests.cs | 30 +++++++++ .../RediSearchTests/SearchTests.cs | 66 +++++++++++++++++++ .../RedisSetupCollection.cs | 2 + .../ComplexObjectWithCascadeAndJsonPath.cs | 24 +++++++ 6 files changed, 166 insertions(+), 18 deletions(-) create mode 100644 test/Redis.OM.Unit.Tests/Serialization/ComplexObjectWithCascadeAndJsonPath.cs diff --git a/src/Redis.OM/Modeling/RedisSchemaField.cs b/src/Redis.OM/Modeling/RedisSchemaField.cs index ee7ab785..b4e2214d 100644 --- a/src/Redis.OM/Modeling/RedisSchemaField.cs +++ b/src/Redis.OM/Modeling/RedisSchemaField.cs @@ -12,6 +12,18 @@ namespace Redis.OM.Modeling /// internal static class RedisSchemaField { + internal static bool IsComplexType(Type type) + { + return !TypeDeterminationUtilities.IsNumeric(type) + && type != typeof(string) + && type != typeof(GeoLoc) + && type != typeof(Ulid) + && type != typeof(bool) + && type != typeof(Guid) + && !type.IsEnum + && !IsTypeIndexableArray(type); + } + /// /// gets the schema field args serialized for json. /// @@ -35,24 +47,17 @@ internal static string[] SerializeArgsJson(this PropertyInfo info, int remaining var ret = new List(); foreach (var attr in attributes) { + int cascadeDepth = remainingDepth == -1 ? attr.CascadeDepth : remainingDepth; if (attr.JsonPath != null) { - ret.AddRange(SerializeIndexFromJsonPaths(info, attr)); + ret.AddRange(SerializeIndexFromJsonPaths(info, attr, pathPrefix, aliasPrefix, cascadeDepth)); } else { var innerType = Nullable.GetUnderlyingType(info.PropertyType) ?? info.PropertyType; - if (!TypeDeterminationUtilities.IsNumeric(innerType) - && innerType != typeof(string) - && innerType != typeof(GeoLoc) - && innerType != typeof(Ulid) - && innerType != typeof(bool) - && innerType != typeof(Guid) - && !innerType.IsEnum - && !IsTypeIndexableArray(innerType)) + if (IsComplexType(innerType)) { - int cascadeDepth = remainingDepth == -1 ? attr.CascadeDepth : remainingDepth; if (cascadeDepth > 0) { foreach (var property in info.PropertyType.GetProperties()) @@ -96,7 +101,7 @@ internal static string[] SerializeArgs(this PropertyInfo info) private static bool IsTypeIndexableArray(Type t) => t == typeof(string[]) || t == typeof(bool[]) || t == typeof(List) || t == typeof(List); - private static IEnumerable SerializeIndexFromJsonPaths(PropertyInfo parentInfo, SearchFieldAttribute attribute, string prefix = "$.") + private static IEnumerable SerializeIndexFromJsonPaths(PropertyInfo parentInfo, SearchFieldAttribute attribute, string prefix = "$.", string aliasPrefix = "", int remainingDepth = -1) { var isCollection = false; var indexArgs = new List(); @@ -128,12 +133,29 @@ private static IEnumerable SerializeIndexFromJsonPaths(PropertyInfo pare type = childProperty.PropertyType; } - var arrayStr = isCollection ? "[*]" : string.Empty; - indexArgs.Add($"{prefix}{parentInfo.Name}{arrayStr}{path.Substring(1)}"); - indexArgs.Add("AS"); - indexArgs.Add($"{parentInfo.Name}_{string.Join("_", propertyNames)}"); - var underlyingType = Nullable.GetUnderlyingType(type); - indexArgs.AddRange(CommonSerialization(attribute, underlyingType ?? type, propertyInfo)); + var underlyingType = Nullable.GetUnderlyingType(type) ?? type; + if (!IsComplexType(underlyingType)) + { + var arrayStr = isCollection ? "[*]" : string.Empty; + indexArgs.Add($"{prefix}{parentInfo.Name}{arrayStr}{path.Substring(1)}"); + indexArgs.Add("AS"); + indexArgs.Add($"{aliasPrefix}{parentInfo.Name}_{string.Join("_", propertyNames)}"); + indexArgs.AddRange(CommonSerialization(attribute, underlyingType, propertyInfo)); + } + else + { + int cascadeDepth = remainingDepth == -1 ? attribute.CascadeDepth : remainingDepth; + if (cascadeDepth > 0) + { + foreach (var property in propertyInfo.PropertyType.GetProperties()) + { + var pathPrefix = $"{prefix}{parentInfo.Name}{path.Substring(1)}."; + var alias = $"{aliasPrefix}{parentInfo.Name}_{string.Join("_", propertyNames)}_"; + indexArgs.AddRange(property.SerializeArgsJson(cascadeDepth - 1, pathPrefix, alias)); + } + } + } + return indexArgs; } diff --git a/src/Redis.OM/Searching/RedisCollection.cs b/src/Redis.OM/Searching/RedisCollection.cs index 6b849c98..e6e4a90b 100644 --- a/src/Redis.OM/Searching/RedisCollection.cs +++ b/src/Redis.OM/Searching/RedisCollection.cs @@ -476,7 +476,11 @@ public T First(Expression> expression) query.Limit = new SearchLimit { Number = 1, Offset = 0 }; var res = _connection.Search(query); var result = res.Documents.FirstOrDefault(); - SaveToStateManager(result.Key, result.Value); + if (result.Key != null) + { + SaveToStateManager(result.Key, result.Value); + } + return result.Value; } diff --git a/test/Redis.OM.Unit.Tests/RediSearchTests/SearchFunctionalTests.cs b/test/Redis.OM.Unit.Tests/RediSearchTests/SearchFunctionalTests.cs index 551ede8f..5134f165 100644 --- a/test/Redis.OM.Unit.Tests/RediSearchTests/SearchFunctionalTests.cs +++ b/test/Redis.OM.Unit.Tests/RediSearchTests/SearchFunctionalTests.cs @@ -1008,6 +1008,36 @@ public void TestIntSelects() collection.Delete(obj); } + [Fact] + public void TestComplexObjectsWithMixedNesting() + { + var obj = new ComplexObjectWithCascadeAndJsonPath + { + InnerCascade = new InnerObject() + { + InnerInnerCascade = new InnerInnerObject() { Arr = new[] { "hello" }, Tag = "World", Num = 42 }, + InnerInnerCollection = new[] { new InnerInnerObject() { Arr = new[] { "hello" }, Tag = "World", Num = 42 } }, + InnerInnerJson = new InnerInnerObject() { Arr = new[] { "hello" }, Tag = "World", Num = 42 } + }, + InnerJson = new InnerObject() + { + InnerInnerCascade = new InnerInnerObject() { Arr = new[] { "hello" }, Tag = "World", Num = 42 }, + InnerInnerCollection = new[] { new InnerInnerObject() { Arr = new[] { "hello" }, Tag = "World", Num = 42 } }, + InnerInnerJson = new InnerInnerObject() { Arr = new[] { "hello" }, Tag = "World", Num = 42 } + } + }; + + var collection = new RedisCollection(_connection); + + collection.Insert(obj); + + Assert.NotNull(collection.FirstOrDefault(x => x.InnerCascade.InnerInnerCascade.Tag == "World")); + Assert.NotNull(collection.FirstOrDefault(x=> x.InnerCascade.InnerInnerCascade.Num == 42)); + Assert.NotNull(collection.FirstOrDefault(x=> x.InnerCascade.InnerInnerCollection.Any(x=>x.Tag == "World"))); + Assert.NotNull(collection.FirstOrDefault(x=>x.InnerJson.InnerInnerCascade.Tag == "World")); + Assert.NotNull(collection.FirstOrDefault(x=>x.InnerJson.InnerInnerCascade.Arr.Contains("hello"))); + } + [Fact] public void TestUpdateWithQuotes() { diff --git a/test/Redis.OM.Unit.Tests/RediSearchTests/SearchTests.cs b/test/Redis.OM.Unit.Tests/RediSearchTests/SearchTests.cs index 55d63a4b..36767f40 100644 --- a/test/Redis.OM.Unit.Tests/RediSearchTests/SearchTests.cs +++ b/test/Redis.OM.Unit.Tests/RediSearchTests/SearchTests.cs @@ -2888,5 +2888,71 @@ public void TestNullableEnumQueries() "100" )); } + + [Fact] + public void TestMixedNestingIndexCreation() + { + _mock.Setup(x => x.Execute(It.IsAny(), It.IsAny())) + .Returns("OK"); + + _mock.Object.CreateIndex(typeof(ComplexObjectWithCascadeAndJsonPath)); + + _mock.Verify(x => x.Execute( + "FT.CREATE", + $"{nameof(ComplexObjectWithCascadeAndJsonPath).ToLower()}-idx", + "ON", + "Json", + "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", "|", + "$.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", "|", + "$.InnerJson.InnerInnerCascade.Num", "AS", "InnerJson_InnerInnerCascade_Num", "NUMERIC", + "$.InnerJson.InnerInnerCascade.Arr[*]", "AS", "InnerJson_InnerInnerCascade_Arr", "TAG", "SEPARATOR", "|")); + } + + [Fact] + public void TestMixedNestingQuerying() + { + _mock.Setup(x => x.Execute(It.IsAny(), It.IsAny())) + .Returns(_mockReply); + + var collection = new RedisCollection(_mock.Object); + + collection.FirstOrDefault(x => x.InnerCascade.InnerInnerCascade.Tag == "hello"); + + _mock.Verify(x => x.Execute( + "FT.SEARCH", + "complexobjectwithcascadeandjsonpath-idx", + "(@InnerCascade_InnerInnerCascade_Tag:{hello})", + "LIMIT", + "0", + "1" + )); + + collection.FirstOrDefault(x => x.InnerCascade.InnerInnerCascade.Num == 5); + _mock.Verify(x => x.Execute( + "FT.SEARCH", + "complexobjectwithcascadeandjsonpath-idx", + "(@InnerCascade_InnerInnerCascade_Num:[5 5])", + "LIMIT", + "0", + "1" + )); + + collection.FirstOrDefault(x => x.InnerCascade.InnerInnerCollection.Any(x=>x.Tag == "hello")); + _mock.Verify(x => x.Execute( + "FT.SEARCH", + "complexobjectwithcascadeandjsonpath-idx", + "(@InnerCascade_InnerInnerCollection_Tag:{hello})", + "LIMIT", + "0", + "1" + )); + } } } \ 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 27e4de6c..30f699a0 100644 --- a/test/Redis.OM.Unit.Tests/RedisSetupCollection.cs +++ b/test/Redis.OM.Unit.Tests/RedisSetupCollection.cs @@ -26,6 +26,7 @@ public RedisSetup() Connection.CreateIndex(typeof(ObjectWithDateTime)); Connection.CreateIndex(typeof(ObjectWithDateTimeHash)); Connection.CreateIndex(typeof(PersonWithNestedArrayOfObject)); + Connection.CreateIndex(typeof(ComplexObjectWithCascadeAndJsonPath)); Connection.CreateIndex(typeof(BasicJsonObjectTestSave)); } @@ -61,6 +62,7 @@ public void Dispose() Connection.DropIndexAndAssociatedRecords(typeof(ObjectWithDateTime)); Connection.DropIndexAndAssociatedRecords(typeof(ObjectWithDateTimeHash)); Connection.DropIndexAndAssociatedRecords(typeof(PersonWithNestedArrayOfObject)); + Connection.DropIndexAndAssociatedRecords(typeof(ComplexObjectWithCascadeAndJsonPath)); Connection.DropIndexAndAssociatedRecords(typeof(BasicJsonObjectTestSave)); } } diff --git a/test/Redis.OM.Unit.Tests/Serialization/ComplexObjectWithCascadeAndJsonPath.cs b/test/Redis.OM.Unit.Tests/Serialization/ComplexObjectWithCascadeAndJsonPath.cs new file mode 100644 index 00000000..7d7319e2 --- /dev/null +++ b/test/Redis.OM.Unit.Tests/Serialization/ComplexObjectWithCascadeAndJsonPath.cs @@ -0,0 +1,24 @@ +using Redis.OM.Modeling; + +namespace Redis.OM.Unit.Tests; + +[Document(StorageType = StorageType.Json)] +public class ComplexObjectWithCascadeAndJsonPath +{ + [Indexed(CascadeDepth = 2)] public InnerObject InnerCascade { get; set; } + [Indexed(JsonPath = "$.InnerInnerCascade", CascadeDepth = 2)] public InnerObject InnerJson { get; set; } +} + +public class InnerObject +{ + [Indexed(JsonPath = "$.Tag")] public InnerInnerObject InnerInnerJson { get; set; } + [Indexed(CascadeDepth = 1)] public InnerInnerObject InnerInnerCascade { get; set; } + [Indexed(JsonPath = "$.Tag")] public InnerInnerObject[] InnerInnerCollection { get; set; } +} + +public class InnerInnerObject +{ + [Indexed] public string Tag { get; set; } + [Indexed] public int Num { get; set; } + [Indexed] public string[] Arr { get; set; } +} \ No newline at end of file