问题描述
我有一些测试可以对数据库进行一些写操作。我知道这并不是真正的单元测试,但我们可以把它放在一边。
为了使每个测试都能在干净的工作空间中进行,我回滚了到目前为止的所有事务。但是由于数据库锁无法建立,我随机得到并发错误。
这是我的代码:
Test1.dll
[TestFixture]
class MyTest1
{
[OneTimeSetup]
public void SetupFixture()
{
myworkspace.StartEditing(); // this will establish a lock on the underlying database
}
[OneTimeTearDow]
public void TearDownFixture()
{
myWorkspace.Rollback();
}
}
相同的代码也存在于另一个测试程序集中,我们将其命名为Test2.dll
。现在,当我通过nunit3-console Test1.dll Test2.dll
使用nunit-console-runner时,出现以下错误:
System.Runtime.InteropServices.COMException:表'GDB_DatabaseLocks'无法被锁定;当前由主机“ MyHost”上的用户“ ADMIN”(这是我)使用 在ESRI.ArcGIS.Geodatabase.IWorkspaceEdit.StartEditing(带有UndoRedo的布尔值)处
myWorkspace
是与MS-Access-Database相关的COM对象(Arcobjects-interface IWorkspace
)。我认为这是因为nunit创建了多个线程,这些线程同时输入上述代码。因此,我向两个程序集都添加了NonParalizable
属性,但没有成功。我还尝试将Apartment(ApartmentState.STA)
添加到我的程序集中,以便在单个线程中执行所有操作,这导致控制台无法完成。
让我发疯的是,使用ReSahrpers测试运行器运行测试的过程非常完美。但是我不知道ReSharper如何启动nunit。 ReSharper似乎不使用nunit-console,而是使用nunit-API。
是否有另一种方法可以强制所有测试在单个线程中运行?我使用nunit3.10和ArcGIS 10.8。
解决方法
默认情况下,NUnit控制台将并行运行多个测试程序集。添加--agents=1
,以强制两个程序集在单个代理程序下顺序运行。
只是为了阐明您也尝试过的其他方法...
[NonParallelizable]
用于防止单个程序集中不同测试的并行化。默认情况下,程序集中的测试不会并行运行,因此,如果没有在更高级别上专门添加[Parallelizable]
时添加此属性将无效。
[Apartments(Apartment.STA)]
添加为程序集级属性,而不必为每个测试添加。在此处查看文档:{{3}}