为什么这会引发 PK 重复错误?
Why does this throw a PK duplicate error?
这是我的测试用例,我希望它是不言自明的。
[Fact]
public async Task ShouldAllowEntityUpdate()
{
var _tenantRepo = LocalIocManager.Resolve<IRepository<Tenant>>();
Tenant tenant = _tenantRepo.Insert(new Tenant("Tname", "name"));
var _userRepo = LocalIocManager.Resolve<IRepository<User,long>>();
_userRepo.Insert(new User()
{
Tenant = tenant,
Name = "jojo"
});
}
这会引发错误,因为它会在 _userRepo.Insert
期间尝试(再次)创建新租户。这是应该被 EF 跟踪,从而知道它是一个存在的实体吗?
快速回购的副本在这里:
https://github.com/Worthy7/AspAbpSpa/commit/d40f238480934a472a122d6ca6f5cda7ed8e21cd
Test Name: AspAbpSPAMay.Tests.PkDupBug.ShouldAllowEntityUpdate
Test FullName: AspAbpSPAMay.Tests.PkDupBug.ShouldAllowEntityUpdate
Test Source: C:\Users\Ian\Source\Repos\AspAbpSPAMay.6.0\test\AspAbpSPAMay.Tests\PkDupBug.cs : line 13
Test Outcome: Failed
Test Duration: 0:00:04.126
Result StackTrace:
at System.Collections.Generic.Dictionary`2.TryInsert(TKey key, TValue value, InsertionBehavior behavior)
at Microsoft.EntityFrameworkCore.InMemory.Storage.Internal.InMemoryTable`1.Create(IUpdateEntry entry)
at Microsoft.EntityFrameworkCore.InMemory.Storage.Internal.InMemoryStore.ExecuteTransaction(IReadOnlyList`1 entries, IDiagnosticsLogger`1 updateLogger)
at Microsoft.EntityFrameworkCore.ChangeTracking.Internal.StateManager.SaveChanges(IReadOnlyList`1 entriesToSave)
at Microsoft.EntityFrameworkCore.ChangeTracking.Internal.StateManager.SaveChanges(Boolean acceptAllChangesOnSuccess)
at Microsoft.EntityFrameworkCore.DbContext.SaveChanges(Boolean acceptAllChangesOnSuccess)
at Abp.EntityFrameworkCore.AbpDbContext.SaveChanges() in D:\Github\aspnetboilerplate\src\Abp.EntityFrameworkCore\EntityFrameworkCore\AbpDbContext.cs:line 208
at Abp.Zero.EntityFrameworkCore.AbpZeroCommonDbContext`3.SaveChanges() in D:\Github\aspnetboilerplate\src\Abp.ZeroCore.EntityFrameworkCore\Zero\EntityFrameworkCore\AbpZeroCommonDbContext.cs:line 159
at Abp.EntityFrameworkCore.Uow.EfCoreUnitOfWork.SaveChangesInDbContext(DbContext dbContext) in D:\Github\aspnetboilerplate\src\Abp.EntityFrameworkCore\EntityFrameworkCore\Uow\EfCoreUnitOfWork.cs:line 163
at Abp.EntityFrameworkCore.Uow.EfCoreUnitOfWork.SaveChanges() in D:\Github\aspnetboilerplate\src\Abp.EntityFrameworkCore\EntityFrameworkCore\Uow\EfCoreUnitOfWork.cs:line 60
at Abp.EntityFrameworkCore.Uow.EfCoreUnitOfWork.CompleteUow() in D:\Github\aspnetboilerplate\src\Abp.EntityFrameworkCore\EntityFrameworkCore\Uow\EfCoreUnitOfWork.cs:line 77
at Abp.Domain.Uow.UnitOfWorkBase.Complete() in D:\Github\aspnetboilerplate\src\Abp\Domain\Uow\UnitOfWorkBase.cs:line 256
at Abp.Domain.Uow.UnitOfWorkInterceptor.PerformSyncUow(IInvocation invocation, UnitOfWorkOptions options) in D:\Github\aspnetboilerplate\src\Abp\Domain\Uow\UnitOfWorkInterceptor.cs:line 68
at Castle.DynamicProxy.AbstractInvocation.Proceed()
at Castle.Proxies.IRepository`2Proxy_4.InsertOrUpdate(User entity)
at AspAbpSPAMay.Tests.PkDupBug.ShouldAllowEntityUpdate() in C:\Users\Ian\Source\Repos\AspAbpSPAMay.6.0\test\AspAbpSPAMay.Tests\PkDupBug.cs:line 23
--- End of stack trace from previous location where exception was thrown ---
Result Message: System.ArgumentException : An item with the same key has already been added. Key: 2
如果这是错误的方法,那么正确的方法是什么?
查看 似乎我的租户实体由于某种原因正在分离。是因为它需要在 UOW 中才能继续跟踪吗?
正确版本:
[Fact]
public async Task ShouldAllowEntityUpdate()
{
var unitOfWorkManager = Resolve<IUnitOfWorkManager>();
var _tenantRepo = LocalIocManager.Resolve<IRepository<Tenant>>();
Tenant tenant = _tenantRepo.Insert(new Tenant("Tname", "name"));
//COMMIT CHANGES to GET the new tenant's Id
_unitOfWorkManager.Current.SaveChanges();
var _userRepo = LocalIocManager.Resolve<IRepository<User,long>>();
//Use TenantId to set the tenant.
_userRepo.Insert(new User()
{
TenantId = tenant.Id,
Name = "jojo"
});
}
这是我的测试用例,我希望它是不言自明的。
[Fact]
public async Task ShouldAllowEntityUpdate()
{
var _tenantRepo = LocalIocManager.Resolve<IRepository<Tenant>>();
Tenant tenant = _tenantRepo.Insert(new Tenant("Tname", "name"));
var _userRepo = LocalIocManager.Resolve<IRepository<User,long>>();
_userRepo.Insert(new User()
{
Tenant = tenant,
Name = "jojo"
});
}
这会引发错误,因为它会在 _userRepo.Insert
期间尝试(再次)创建新租户。这是应该被 EF 跟踪,从而知道它是一个存在的实体吗?
快速回购的副本在这里: https://github.com/Worthy7/AspAbpSpa/commit/d40f238480934a472a122d6ca6f5cda7ed8e21cd
Test Name: AspAbpSPAMay.Tests.PkDupBug.ShouldAllowEntityUpdate
Test FullName: AspAbpSPAMay.Tests.PkDupBug.ShouldAllowEntityUpdate
Test Source: C:\Users\Ian\Source\Repos\AspAbpSPAMay.6.0\test\AspAbpSPAMay.Tests\PkDupBug.cs : line 13
Test Outcome: Failed
Test Duration: 0:00:04.126
Result StackTrace:
at System.Collections.Generic.Dictionary`2.TryInsert(TKey key, TValue value, InsertionBehavior behavior)
at Microsoft.EntityFrameworkCore.InMemory.Storage.Internal.InMemoryTable`1.Create(IUpdateEntry entry)
at Microsoft.EntityFrameworkCore.InMemory.Storage.Internal.InMemoryStore.ExecuteTransaction(IReadOnlyList`1 entries, IDiagnosticsLogger`1 updateLogger)
at Microsoft.EntityFrameworkCore.ChangeTracking.Internal.StateManager.SaveChanges(IReadOnlyList`1 entriesToSave)
at Microsoft.EntityFrameworkCore.ChangeTracking.Internal.StateManager.SaveChanges(Boolean acceptAllChangesOnSuccess)
at Microsoft.EntityFrameworkCore.DbContext.SaveChanges(Boolean acceptAllChangesOnSuccess)
at Abp.EntityFrameworkCore.AbpDbContext.SaveChanges() in D:\Github\aspnetboilerplate\src\Abp.EntityFrameworkCore\EntityFrameworkCore\AbpDbContext.cs:line 208
at Abp.Zero.EntityFrameworkCore.AbpZeroCommonDbContext`3.SaveChanges() in D:\Github\aspnetboilerplate\src\Abp.ZeroCore.EntityFrameworkCore\Zero\EntityFrameworkCore\AbpZeroCommonDbContext.cs:line 159
at Abp.EntityFrameworkCore.Uow.EfCoreUnitOfWork.SaveChangesInDbContext(DbContext dbContext) in D:\Github\aspnetboilerplate\src\Abp.EntityFrameworkCore\EntityFrameworkCore\Uow\EfCoreUnitOfWork.cs:line 163
at Abp.EntityFrameworkCore.Uow.EfCoreUnitOfWork.SaveChanges() in D:\Github\aspnetboilerplate\src\Abp.EntityFrameworkCore\EntityFrameworkCore\Uow\EfCoreUnitOfWork.cs:line 60
at Abp.EntityFrameworkCore.Uow.EfCoreUnitOfWork.CompleteUow() in D:\Github\aspnetboilerplate\src\Abp.EntityFrameworkCore\EntityFrameworkCore\Uow\EfCoreUnitOfWork.cs:line 77
at Abp.Domain.Uow.UnitOfWorkBase.Complete() in D:\Github\aspnetboilerplate\src\Abp\Domain\Uow\UnitOfWorkBase.cs:line 256
at Abp.Domain.Uow.UnitOfWorkInterceptor.PerformSyncUow(IInvocation invocation, UnitOfWorkOptions options) in D:\Github\aspnetboilerplate\src\Abp\Domain\Uow\UnitOfWorkInterceptor.cs:line 68
at Castle.DynamicProxy.AbstractInvocation.Proceed()
at Castle.Proxies.IRepository`2Proxy_4.InsertOrUpdate(User entity)
at AspAbpSPAMay.Tests.PkDupBug.ShouldAllowEntityUpdate() in C:\Users\Ian\Source\Repos\AspAbpSPAMay.6.0\test\AspAbpSPAMay.Tests\PkDupBug.cs:line 23
--- End of stack trace from previous location where exception was thrown ---
Result Message: System.ArgumentException : An item with the same key has already been added. Key: 2
如果这是错误的方法,那么正确的方法是什么?
查看
正确版本:
[Fact]
public async Task ShouldAllowEntityUpdate()
{
var unitOfWorkManager = Resolve<IUnitOfWorkManager>();
var _tenantRepo = LocalIocManager.Resolve<IRepository<Tenant>>();
Tenant tenant = _tenantRepo.Insert(new Tenant("Tname", "name"));
//COMMIT CHANGES to GET the new tenant's Id
_unitOfWorkManager.Current.SaveChanges();
var _userRepo = LocalIocManager.Resolve<IRepository<User,long>>();
//Use TenantId to set the tenant.
_userRepo.Insert(new User()
{
TenantId = tenant.Id,
Name = "jojo"
});
}