diff --git a/src/EFCore.InMemory/Storage/Internal/InMemoryStore.cs b/src/EFCore.InMemory/Storage/Internal/InMemoryStore.cs index d5407798c60..3b971cc1265 100644 --- a/src/EFCore.InMemory/Storage/Internal/InMemoryStore.cs +++ b/src/EFCore.InMemory/Storage/Internal/InMemoryStore.cs @@ -20,31 +20,19 @@ namespace Microsoft.EntityFrameworkCore.InMemory.Storage.Internal public class InMemoryStore : IInMemoryStore { private readonly IInMemoryTableFactory _tableFactory; - private readonly bool _useNameMatching; private readonly object _lock = new object(); - private LazyRef> _tables = CreateTables(); - - /// - /// This API supports the Entity Framework Core infrastructure and is not intended to be used - /// directly from your code. This API may change or be removed in future releases. - /// - public InMemoryStore([NotNull] IInMemoryTableFactory tableFactory) - : this(tableFactory, useNameMatching: false) - { - } - + private LazyRef> _tables = CreateTables(); + /// /// This API supports the Entity Framework Core infrastructure and is not intended to be used /// directly from your code. This API may change or be removed in future releases. /// public InMemoryStore( - [NotNull] IInMemoryTableFactory tableFactory, - bool useNameMatching) + [NotNull] IInMemoryTableFactory tableFactory) { _tableFactory = tableFactory; - _useNameMatching = useNameMatching; } /// @@ -99,8 +87,8 @@ public virtual bool Clear() } } - private static LazyRef> CreateTables() - => new LazyRef>(() => new Dictionary()); + private static LazyRef> CreateTables() + => new LazyRef>(() => new Dictionary()); /// /// This API supports the Entity Framework Core infrastructure and is not intended to be used @@ -115,7 +103,7 @@ public virtual IReadOnlyList GetTables(IEntityType entity { foreach (var et in entityType.GetConcreteTypesInHierarchy()) { - var key = _useNameMatching ? (object)et.Name : et; + var key = GetTableKey(et); if (_tables.Value.TryGetValue(key, out var table)) { data.Add(new InMemoryTableSnapshot(et, table.SnapshotRows())); @@ -147,7 +135,7 @@ public virtual int ExecuteTransaction( Debug.Assert(!entityType.IsAbstract()); - var key = _useNameMatching ? (object)entityType.Name : entityType; + var key = GetTableKey(entityType); if (!_tables.Value.TryGetValue(key, out var table)) { _tables.Value.Add(key, table = _tableFactory.Create(entityType)); @@ -184,5 +172,10 @@ public virtual int ExecuteTransaction( return rowsAffected; } + + private string GetTableKey(IEntityType entityType) + { + return entityType.ToString(); + } } } diff --git a/src/EFCore.InMemory/Storage/Internal/InMemoryStoreCache.cs b/src/EFCore.InMemory/Storage/Internal/InMemoryStoreCache.cs index c7248b17328..d29b8bb2742 100644 --- a/src/EFCore.InMemory/Storage/Internal/InMemoryStoreCache.cs +++ b/src/EFCore.InMemory/Storage/Internal/InMemoryStoreCache.cs @@ -16,7 +16,6 @@ namespace Microsoft.EntityFrameworkCore.InMemory.Storage.Internal public class InMemoryStoreCache : IInMemoryStoreCache { private readonly IInMemoryTableFactory _tableFactory; - private readonly bool _useNameMatching; private readonly ConcurrentDictionary _namedStores; /// @@ -41,8 +40,6 @@ public InMemoryStoreCache( if (options?.DatabaseRoot != null) { - _useNameMatching = true; - LazyInitializer.EnsureInitialized( ref options.DatabaseRoot.Instance, () => new ConcurrentDictionary()); @@ -60,6 +57,6 @@ public InMemoryStoreCache( /// directly from your code. This API may change or be removed in future releases. /// public virtual IInMemoryStore GetStore(string name) - => _namedStores.GetOrAdd(name, _ => new InMemoryStore(_tableFactory, _useNameMatching)); + => _namedStores.GetOrAdd(name, _ => new InMemoryStore(_tableFactory)); } } diff --git a/test/EFCore.InMemory.FunctionalTests/Query/CompiledQueryInMemoryTest.cs b/test/EFCore.InMemory.FunctionalTests/Query/CompiledQueryInMemoryTest.cs index 4a6829e8778..e994b58bf87 100644 --- a/test/EFCore.InMemory.FunctionalTests/Query/CompiledQueryInMemoryTest.cs +++ b/test/EFCore.InMemory.FunctionalTests/Query/CompiledQueryInMemoryTest.cs @@ -1,7 +1,12 @@ // Copyright (c) .NET Foundation. All rights reserved. // Licensed under the Apache License, Version 2.0. See License.txt in the project root for license information. +using System; +using System.Linq; +using Microsoft.EntityFrameworkCore.TestModels.Northwind; using Microsoft.EntityFrameworkCore.TestUtilities; +using Microsoft.Extensions.DependencyInjection; +using Xunit; namespace Microsoft.EntityFrameworkCore.Query { @@ -11,5 +16,69 @@ public CompiledQueryInMemoryTest(NorthwindQueryInMemoryFixture( + o => o.UseInMemoryDatabase(dbName)); + } + } + + [Fact] + public void Query_Different_InMemoryDatabases() + { + var preCompiled = EF.CompileQuery(ctx => ctx.Customers.SingleOrDefault()); + + using (var services = new MyServiceCollection("FOO").BuildServiceProvider()) + { + using (var scope = services.GetRequiredService().CreateScope()) + { + var ctx = scope.ServiceProvider.GetRequiredService(); + + ctx.Database.EnsureDeleted(); + ctx.Database.EnsureCreated(); + + ctx.Add(new Customer { ContactName = "One" }); + + ctx.SaveChanges(); + + var item = preCompiled(ctx); + Assert.Equal("One", item?.ContactName); + } + + using (var scope = services.GetRequiredService().CreateScope()) + { + var ctx = scope.ServiceProvider.GetRequiredService(); + var item = preCompiled(ctx); + Assert.Equal("One", item?.ContactName); + } + } + + using (var services = new MyServiceCollection("BAR").BuildServiceProvider()) + { + using (var scope = services.GetRequiredService().CreateScope()) + { + var ctx = scope.ServiceProvider.GetRequiredService(); + ctx.Database.EnsureDeleted(); + ctx.Database.EnsureCreated(); + + ctx.Add(new Customer { ContactName = "Two" }); + + ctx.SaveChanges(); + + var item = preCompiled(ctx); + Assert.Equal("Two", item?.ContactName); + } + + using (var scope = services.GetRequiredService().CreateScope()) + { + var ctx = scope.ServiceProvider.GetRequiredService(); + var item = preCompiled(ctx); + Assert.Equal("Two", item?.ContactName); + } + } + } } }