You cannot select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
EntityFrameworkCore.Jet/test/EFCore.Jet.FunctionalTests/Query/AdHocJsonQueryJetTestBase.cs

364 lines
15 KiB
C#

#nullable disable
using EntityFrameworkCore.Jet.Diagnostics.Internal;
using EntityFrameworkCore.Jet.FunctionalTests.TestUtilities;
using Microsoft.EntityFrameworkCore.Diagnostics;
using Microsoft.EntityFrameworkCore.Diagnostics.Internal;
using Microsoft.EntityFrameworkCore.TestUtilities;
using System.Collections.Generic;
using System.Data.Common;
using System.Linq;
using System.Threading.Tasks;
using Xunit;
namespace Microsoft.EntityFrameworkCore.Query;
public abstract class AdHocJsonQueryJetTestBase : AdHocJsonQueryRelationalTestBase
{
protected override ITestStoreFactory TestStoreFactory
=> JetTestStoreFactory.Instance;
protected override void ConfigureWarnings(WarningsConfigurationBuilder builder)
{
base.ConfigureWarnings(builder);
builder.Log(CoreEventId.StringEnumValueInJson);
}
protected override async Task Seed30028(DbContext ctx)
{
// complete
await ctx.Database.ExecuteSqlAsync(
$$$$"""
INSERT INTO [Entities] ([Id], [Json])
VALUES(
1,
'{"RootName":"e1","Collection":[{"BranchName":"e1 c1","Nested":{"LeafName":"e1 c1 l"}},{"BranchName":"e1 c2","Nested":{"LeafName":"e1 c2 l"}}],"OptionalReference":{"BranchName":"e1 or","Nested":{"LeafName":"e1 or l"}},"RequiredReference":{"BranchName":"e1 rr","Nested":{"LeafName":"e1 rr l"}}}')
""");
// missing collection
await ctx.Database.ExecuteSqlAsync(
$$$$"""
INSERT INTO [Entities] ([Id], [Json])
VALUES(
2,
'{"RootName":"e2","OptionalReference":{"BranchName":"e2 or","Nested":{"LeafName":"e2 or l"}},"RequiredReference":{"BranchName":"e2 rr","Nested":{"LeafName":"e2 rr l"}}}')
""");
// missing optional reference
await ctx.Database.ExecuteSqlAsync(
$$$$"""
INSERT INTO [Entities] ([Id], [Json])
VALUES(
3,
'{"RootName":"e3","Collection":[{"BranchName":"e3 c1","Nested":{"LeafName":"e3 c1 l"}},{"BranchName":"e3 c2","Nested":{"LeafName":"e3 c2 l"}}],"RequiredReference":{"BranchName":"e3 rr","Nested":{"LeafName":"e3 rr l"}}}')
""");
// missing required reference
await ctx.Database.ExecuteSqlAsync(
$$$$"""
INSERT INTO [Entities] ([Id], [Json])
VALUES(
4,
'{"RootName":"e4","Collection":[{"BranchName":"e4 c1","Nested":{"LeafName":"e4 c1 l"}},{"BranchName":"e4 c2","Nested":{"LeafName":"e4 c2 l"}}],"OptionalReference":{"BranchName":"e4 or","Nested":{"LeafName":"e4 or l"}}}')
""");
}
protected override Task Seed33046(DbContext ctx)
=> ctx.Database.ExecuteSqlAsync(
$$"""
INSERT INTO [Reviews] ([Rounds], [Id])
VALUES('[{"RoundNumber":11,"SubRounds":[{"SubRoundNumber":111},{"SubRoundNumber":112}]}]', 1)
""");
protected override Task SeedArrayOfPrimitives(DbContext ctx)
{
var entity1 = new MyEntityArrayOfPrimitives
{
Id = 1,
Reference = new MyJsonEntityArrayOfPrimitives
{
IntArray = [1, 2, 3],
ListOfString =
[
"Foo",
"Bar",
"Baz"
]
},
Collection =
[
new MyJsonEntityArrayOfPrimitives { IntArray = [111, 112, 113], ListOfString = ["Foo11", "Bar11"] },
new MyJsonEntityArrayOfPrimitives { IntArray = [211, 212, 213], ListOfString = ["Foo12", "Bar12"] }
]
};
var entity2 = new MyEntityArrayOfPrimitives
{
Id = 2,
Reference = new MyJsonEntityArrayOfPrimitives
{
IntArray = [10, 20, 30],
ListOfString =
[
"A",
"B",
"C"
]
},
Collection =
[
new MyJsonEntityArrayOfPrimitives { IntArray = [110, 120, 130], ListOfString = ["A1", "Z1"] },
new MyJsonEntityArrayOfPrimitives { IntArray = [210, 220, 230], ListOfString = ["A2", "Z2"] }
]
};
ctx.AddRange(entity1, entity2);
return ctx.SaveChangesAsync();
}
protected override Task SeedJunkInJson(DbContext ctx)
=> ctx.Database.ExecuteSqlAsync(
$$$$"""
INSERT INTO [Entities] ([Collection], [CollectionWithCtor], [Reference], [ReferenceWithCtor], [Id])
VALUES(
'[{"JunkReference":{"Something":"SomeValue" },"Name":"c11","JunkProperty1":50,"Number":11.5,"JunkCollection1":[],"JunkCollection2":[{"Foo":"junk value"}],"NestedCollection":[{"DoB":"2002-04-01T00:00:00","DummyProp":"Dummy value"},{"DoB":"2002-04-02T00:00:00","DummyReference":{"Foo":5}}],"NestedReference":{"DoB":"2002-03-01T00:00:00"}},{"Name":"c12","Number":12.5,"NestedCollection":[{"DoB":"2002-06-01T00:00:00"},{"DoB":"2002-06-02T00:00:00"}],"NestedDummy":59,"NestedReference":{"DoB":"2002-05-01T00:00:00"}}]',
'[{"MyBool":true,"Name":"c11 ctor","JunkReference":{"Something":"SomeValue","JunkCollection":[{"Foo":"junk value"}]},"NestedCollection":[{"DoB":"2002-08-01T00:00:00"},{"DoB":"2002-08-02T00:00:00"}],"NestedReference":{"DoB":"2002-07-01T00:00:00"}},{"MyBool":false,"Name":"c12 ctor","NestedCollection":[{"DoB":"2002-10-01T00:00:00"},{"DoB":"2002-10-02T00:00:00"}],"JunkCollection":[{"Foo":"junk value"}],"NestedReference":{"DoB":"2002-09-01T00:00:00"}}]',
'{"Name":"r1","JunkCollection":[{"Foo":"junk value"}],"JunkReference":{"Something":"SomeValue" },"Number":1.5,"NestedCollection":[{"DoB":"2000-02-01T00:00:00","JunkReference":{"Something":"SomeValue"}},{"DoB":"2000-02-02T00:00:00"}],"NestedReference":{"DoB":"2000-01-01T00:00:00"}}',
'{"MyBool":true,"JunkCollection":[{"Foo":"junk value"}],"Name":"r1 ctor","JunkReference":{"Something":"SomeValue" },"NestedCollection":[{"DoB":"2001-02-01T00:00:00"},{"DoB":"2001-02-02T00:00:00"}],"NestedReference":{"JunkCollection":[{"Foo":"junk value"}],"DoB":"2001-01-01T00:00:00"}}',
1)
""");
protected override Task SeedTrickyBuffering(DbContext ctx)
=> ctx.Database.ExecuteSqlAsync(
$$$"""
INSERT INTO [Entities] ([Reference], [Id])
VALUES(
'{"Name": "r1", "Number": 7, "JunkReference":{"Something": "SomeValue" }, "JunkCollection": [{"Foo": "junk value"}], "NestedReference": {"DoB": "2000-01-01T00:00:00"}, "NestedCollection": [{"DoB": "2000-02-01T00:00:00", "JunkReference": {"Something": "SomeValue"}}, {"DoB": "2000-02-02T00:00:00"}]}',1)
""");
protected override Task SeedShadowProperties(DbContext ctx)
=> ctx.Database.ExecuteSqlAsync(
$$"""
INSERT INTO [Entities] ([Collection], [CollectionWithCtor], [Reference], [ReferenceWithCtor], [Id], [Name])
VALUES(
'[{"Name":"e1_c1","ShadowDouble":5.5},{"ShadowDouble":20.5,"Name":"e1_c2"}]',
'[{"Name":"e1_c1 ctor","ShadowNullableByte":6},{"ShadowNullableByte":null,"Name":"e1_c2 ctor"}]',
'{"Name":"e1_r", "ShadowString":"Foo"}',
'{"ShadowInt":143,"Name":"e1_r ctor"}',
1,
'e1')
""");
protected override async Task SeedNotICollection(DbContext ctx)
{
await ctx.Database.ExecuteSqlAsync(
$$"""
INSERT INTO [Entities] ([Json], [Id])
VALUES(
'{"Collection":[{"Bar":11,"Foo":"c11"},{"Bar":12,"Foo":"c12"},{"Bar":13,"Foo":"c13"}]}',
1)
""");
await ctx.Database.ExecuteSqlAsync(
$$$"""
INSERT INTO [Entities] ([Json], [Id])
VALUES(
'{"Collection":[{"Bar":21,"Foo":"c21"},{"Bar":22,"Foo":"c22"}]}',
2)
""");
}
#region EnumLegacyValues
[ConditionalTheory]
[MemberData(nameof(IsAsyncData))]
public virtual async Task Read_enum_property_with_legacy_values(bool async)
{
var contextFactory = await InitializeAsync<DbContext>(
onModelCreating: BuildModelEnumLegacyValues,
onConfiguring: b => b.ConfigureWarnings(ConfigureWarnings),
seed: SeedEnumLegacyValues);
using var context = contextFactory.CreateContext();
var query = context.Set<MyEntityEnumLegacyValues>().Select(
x => new
{
x.Reference.IntEnum,
x.Reference.ByteEnum,
x.Reference.LongEnum,
x.Reference.NullableEnum
});
var exception = async
? await (Assert.ThrowsAsync<DbException>(() => query.ToListAsync()))
: Assert.Throws<DbException>(() => query.ToList());
// Conversion failed when converting the nvarchar value '...' to data type int
//Assert.Equal(245, exception.Number);
}
[ConditionalTheory]
[MemberData(nameof(IsAsyncData))]
public virtual async Task Read_json_entity_with_enum_properties_with_legacy_values(bool async)
{
var contextFactory = await InitializeAsync<DbContext>(
onModelCreating: BuildModelEnumLegacyValues,
onConfiguring: b => b.ConfigureWarnings(ConfigureWarnings),
seed: SeedEnumLegacyValues,
shouldLogCategory: c => c == DbLoggerCategory.Query.Name);
using (var context = contextFactory.CreateContext())
{
var query = context.Set<MyEntityEnumLegacyValues>().Select(x => x.Reference).AsNoTracking();
var result = async
? await query.ToListAsync()
: [.. query];
Assert.Equal(1, result.Count);
Assert.Equal(ByteEnumLegacyValues.Redmond, result[0].ByteEnum);
Assert.Equal(IntEnumLegacyValues.Foo, result[0].IntEnum);
Assert.Equal(LongEnumLegacyValues.Three, result[0].LongEnum);
Assert.Equal(ULongEnumLegacyValues.Three, result[0].ULongEnum);
Assert.Equal(IntEnumLegacyValues.Bar, result[0].NullableEnum);
}
var testLogger = new TestLogger<JetLoggingDefinitions>();
Assert.Single(
ListLoggerFactory.Log.Where(
l => l.Message == CoreResources.LogStringEnumValueInJson(testLogger).GenerateMessage(nameof(ByteEnumLegacyValues))));
Assert.Single(
ListLoggerFactory.Log.Where(
l => l.Message == CoreResources.LogStringEnumValueInJson(testLogger).GenerateMessage(nameof(IntEnumLegacyValues))));
Assert.Single(
ListLoggerFactory.Log.Where(
l => l.Message == CoreResources.LogStringEnumValueInJson(testLogger).GenerateMessage(nameof(LongEnumLegacyValues))));
Assert.Single(
ListLoggerFactory.Log.Where(
l => l.Message == CoreResources.LogStringEnumValueInJson(testLogger).GenerateMessage(nameof(ULongEnumLegacyValues))));
}
[ConditionalTheory]
[MemberData(nameof(IsAsyncData))]
public virtual async Task Read_json_entity_collection_with_enum_properties_with_legacy_values(bool async)
{
var contextFactory = await InitializeAsync<DbContext>(
onModelCreating: BuildModelEnumLegacyValues,
onConfiguring: b => b.ConfigureWarnings(ConfigureWarnings),
seed: SeedEnumLegacyValues,
shouldLogCategory: c => c == DbLoggerCategory.Query.Name);
using (var context = contextFactory.CreateContext())
{
var query = context.Set<MyEntityEnumLegacyValues>().Select(x => x.Collection).AsNoTracking();
var result = async
? await query.ToListAsync()
: [.. query];
Assert.Equal(1, result.Count);
Assert.Equal(2, result[0].Count);
Assert.Equal(ByteEnumLegacyValues.Bellevue, result[0][0].ByteEnum);
Assert.Equal(IntEnumLegacyValues.Foo, result[0][0].IntEnum);
Assert.Equal(LongEnumLegacyValues.One, result[0][0].LongEnum);
Assert.Equal(ULongEnumLegacyValues.One, result[0][0].ULongEnum);
Assert.Equal(IntEnumLegacyValues.Bar, result[0][0].NullableEnum);
Assert.Equal(ByteEnumLegacyValues.Seattle, result[0][1].ByteEnum);
Assert.Equal(IntEnumLegacyValues.Baz, result[0][1].IntEnum);
Assert.Equal(LongEnumLegacyValues.Two, result[0][1].LongEnum);
Assert.Equal(ULongEnumLegacyValues.Two, result[0][1].ULongEnum);
Assert.Null(result[0][1].NullableEnum);
}
var testLogger = new TestLogger<JetLoggingDefinitions>();
Assert.Single(
ListLoggerFactory.Log.Where(
l => l.Message == CoreResources.LogStringEnumValueInJson(testLogger).GenerateMessage(nameof(ByteEnumLegacyValues))));
Assert.Single(
ListLoggerFactory.Log.Where(
l => l.Message == CoreResources.LogStringEnumValueInJson(testLogger).GenerateMessage(nameof(IntEnumLegacyValues))));
Assert.Single(
ListLoggerFactory.Log.Where(
l => l.Message == CoreResources.LogStringEnumValueInJson(testLogger).GenerateMessage(nameof(LongEnumLegacyValues))));
Assert.Single(
ListLoggerFactory.Log.Where(
l => l.Message == CoreResources.LogStringEnumValueInJson(testLogger).GenerateMessage(nameof(ULongEnumLegacyValues))));
}
private Task SeedEnumLegacyValues(DbContext ctx)
=> ctx.Database.ExecuteSqlAsync(
$$"""
INSERT INTO [Entities] ([Collection], [Reference], [Id], [Name])
VALUES(
'[{"ByteEnum":"Bellevue","IntEnum":"Foo","LongEnum":"One","ULongEnum":"One","Name":"e1_c1","NullableEnum":"Bar"},{"ByteEnum":"Seattle","IntEnum":"Baz","LongEnum":"Two","ULongEnum":"Two","Name":"e1_c2","NullableEnum":null}]',
'{"ByteEnum":"Redmond","IntEnum":"Foo","LongEnum":"Three","ULongEnum":"Three","Name":"e1_r","NullableEnum":"Bar"}',
1,
'e1')
""");
protected virtual void BuildModelEnumLegacyValues(ModelBuilder modelBuilder)
=> modelBuilder.Entity<MyEntityEnumLegacyValues>(
b =>
{
b.ToTable("Entities");
b.Property(x => x.Id).ValueGeneratedNever();
b.OwnsOne(x => x.Reference, b => b.ToJson().HasColumnType(JsonColumnType));
b.OwnsMany(x => x.Collection, b => b.ToJson().HasColumnType(JsonColumnType));
});
private class MyEntityEnumLegacyValues
{
public int Id { get; set; }
public string Name { get; set; }
public MyJsonEntityEnumLegacyValues Reference { get; set; }
public List<MyJsonEntityEnumLegacyValues> Collection { get; set; }
}
private class MyJsonEntityEnumLegacyValues
{
public string Name { get; set; }
// ReSharper disable once UnusedAutoPropertyAccessor.Local
public IntEnumLegacyValues IntEnum { get; set; }
// ReSharper disable once UnusedAutoPropertyAccessor.Local
public ByteEnumLegacyValues ByteEnum { get; set; }
// ReSharper disable once UnusedAutoPropertyAccessor.Local
public LongEnumLegacyValues LongEnum { get; set; }
// ReSharper disable once UnusedAutoPropertyAccessor.Local
public ULongEnumLegacyValues ULongEnum { get; set; }
// ReSharper disable once UnusedAutoPropertyAccessor.Local
public IntEnumLegacyValues? NullableEnum { get; set; }
}
private enum IntEnumLegacyValues
{
Foo = int.MinValue,
Bar,
Baz = int.MaxValue,
}
private enum ByteEnumLegacyValues : byte
{
Seattle,
Redmond,
Bellevue = 255,
}
private enum LongEnumLegacyValues : long
{
One = long.MinValue,
Two = 1,
Three = long.MaxValue,
}
private enum ULongEnumLegacyValues : ulong
{
One = ulong.MinValue,
Two = 1,
Three = ulong.MaxValue,
}
#endregion
}