Я снова отправляю этот вопрос по поручению уважаемого г-на Джона Скита, который предложил мне разработать простую тестовую программу, которая изолирует и демонстрирует проблему, с которой я столкнулся, и перепечатываю вопрос.Этот вопрос вырос из этого , поэтому, пожалуйста, прости меня, если все это звучит очень знакомо.Вы можете получить дополнительные сведения об этом вопросе из этого.
Проблема, с которой я сталкиваюсь Assert.Throws<T>
из NUnit 2.5.9.В некоторых случаях он не сможет перехватить любых исключений, вызванных в методе, вызванном TestDelegate.Я закрепил это поведение в воспроизводимом коде ниже.(Хотя, по общему признанию, это может быть случай Fails On My Machine ™.
Чтобы воспроизвести ошибку, я создал решение с двумя проектами C # DLL:
- сначала содержит один класс с одним открытым методом. Этот метод является методом расширения, который инкапсулирует логику, необходимую для создания
SqlCommand
, заполнения его параметров и вызова ExecuteScalar
. Этот проект не содержит других ссылок. - Второй содержит один класс с двумя методами, которые проверяют, работает ли метод в первой DLL должным образом. Этот проект ссылается на первый и содержит ссылку на платформу NUnit. На другие сборки нет ссылок.
Когда я выполняю тесты в отладчике, я наблюдаю следующее:
Assert.Throws
правильно вызывает метод расширения ExecuteScalar<T>
. - Значения параметров, как и ожидалось, равны нулю.
ExecuteScalar<T>
проверяет свои параметры на наличие нулевых значений. - Отладчик действительно запускает и выполняетстрока, содержащая
throw new ArgumentNullException(...)
. - После выполнения
throw
управление приложением не сразу передается Assert.Throws
.Вместо этого он продолжается на следующей строке в ExecuteScalar<T>
. - . Как только выполняется следующая строка кода, отладчик прерывается и отображает ошибку «Аргумент нулевое исключение не было обработано кодом пользователя».
Исходный код, который изолирует это поведение, приведен ниже.
МЕТОД РАСШИРЕНИЯ
namespace NUnit_Anomaly
{
using System;
using System.Data;
using System.Data.SqlClient;
public static class Class1
{
public static T ExecuteScalar<T>(this SqlConnection connection, string sql)
{
if (connection == null)
{
throw new ArgumentNullException("connection");
}
if (sql == null)
{
throw new ArgumentNullException("sql");
}
using (var command = connection.CreateCommand())
{
command.CommandType = CommandType.Text;
command.CommandText = sql;
return (T)command.ExecuteScalar();
}
}
}
}
ТЕСТОВЫЕ СЛУЧАИ
namespace NUnit_Tests
{
using System;
using System.Data.SqlClient;
using System.Diagnostics;
using NUnit.Framework;
using NUnit_Anomaly;
[TestFixture]
public class NUnitAnomalyTest
{
[Test]
public void ExecuteDataSetThrowsForNullConnection()
{
Assert.Throws<ArgumentNullException>(() => ((SqlConnection)null).ExecuteScalar<int>(null));
}
[Test]
public void ExecuteDataSetThrowsForNullSql()
{
const string server = "MY-LOCAL-SQL-SERVER";
const string instance = "staging";
string connectionString = String.Format("Data Source={0};Initial Catalog={1};Integrated Security=True;",
server,
instance);
using (var connection = new SqlConnection(connectionString))
{
Assert.Throws<ArgumentNullException>(() => connection.ExecuteScalar<int>(null));
}
}
}
}
Чистый эффект состоит в том, что тесты не выполняются, когда они не должны.Насколько я понимаю, Assert.Throws<T>
должно поймать мое исключение, и тест должен пройти.
ОБНОВЛЕНИЕ
Я принял совет Гансаи проверил диалог Исключения.Я не нарушал сгенерированных исключений, но я нарушал необработанных пользовательских исключений.По-видимому, именно поэтому отладчик врывается в IDE при возникновении исключения.Снятие флажка устранило проблему, и Assert.Throws<T>
поднял ее.Однако, если я этого не сделал, я не могу просто нажать F5, чтобы продолжить выполнение, или исключение станет NullReferenceException
.
Итак, теперь возникает вопрос: можно ли настроить разрывы исключений для каждого проекта?Я хочу делать это только во время тестирования, но не в целом.