Как я могу привлечь сообщение об исключении с атрибутом ожидаемого экспрессии
-
11-10-2019 - |
Вопрос
Я думал, что эти два теста должны вести себя одинаково, на самом деле я написал тест в своем проекте, используя тест MS только для того, чтобы выяснить, что теперь он не уважает ожидаемое сообщение так же, как NUNIT.
NUNIT (терпит неудачу):
[Test, ExpectedException(typeof(System.FormatException), ExpectedMessage = "blah")]
public void Validate()
{
int.Parse("dfd");
}
МС -тест (проходы):
[TestMethod, ExpectedException(typeof(System.FormatException), "blah")]
public void Validate()
{
int.Parse("dfd");
}
Независимо от того, какое сообщение я даю тест MS, оно пройдет.
Есть ли способ провести провал теста MS, если сообщение не правильно? Могу ли я даже создать свой собственный атрибут исключения? Я бы предпочел, чтобы не пришлось писать блок уловов для каждого теста, где это происходит.
Решение
Этот второй параметр MSTest - это сообщение, которое распечатано при сбое теста. MSTEST будет успешным, если будет брошено формат -индивидуальность. Я нашел этот пост, который может быть полезен
Другие советы
Мы используем этот атрибут повсюду, и мы явно неправильно поняли второй параметр (позор для нас).
Тем не менее, мы определенно использовали его для проверки сообщения об исключении. Следующим было то, что мы использовали с подсказками с этой страницы. Он не обрабатывает глобализацию или унаследованные типы исключений, но делает то, что нам нужно. Опять же, цель состояла в том, чтобы просто RR 'WeareException' и поменять его с этим классом. (Bummer WearedException запечатано.)
public class ExpectedExceptionWithMessageAttribute : ExpectedExceptionBaseAttribute
{
public Type ExceptionType { get; set; }
public string ExpectedMessage { get; set; }
public ExpectedExceptionWithMessageAttribute(Type exceptionType)
{
this.ExceptionType = exceptionType;
}
public ExpectedExceptionWithMessageAttribute(Type exceptionType, string expectedMessage)
{
this.ExceptionType = exceptionType;
this.ExpectedMessage = expectedMessage;
}
protected override void Verify(Exception e)
{
if (e.GetType() != this.ExceptionType)
{
Assert.Fail(String.Format(
"ExpectedExceptionWithMessageAttribute failed. Expected exception type: {0}. Actual exception type: {1}. Exception message: {2}",
this.ExceptionType.FullName,
e.GetType().FullName,
e.Message
)
);
}
var actualMessage = e.Message.Trim();
if (this.ExpectedMessage != null)
{
Assert.AreEqual(this.ExpectedMessage, actualMessage);
}
Console.Write("ExpectedExceptionWithMessageAttribute:" + e.Message);
}
}
@Rcravens верен - второй параметр - это сообщение, которое напечатано, если тест не удается. То, что я сделал, чтобы обойти это, - это создание моих тестов немного по -другому. По общему признанию, я не люблю этот подход, но он работает.
[TestMethod]
public void Validate()
{
try
{
int.Parse("dfd");
// Test fails if it makes it this far
Assert.Fail("Expected exception was not thrown.");
}
catch (Exception ex)
{
Assert.AreEqual("blah", ex.Message);
}
}
Я расширил ответ от BlackJacketMack для нашего проекта, добавив поддержку для содержимого, нечувствительных к случаям и комбинации ресурсов-ресурсов-имен.
Пример использования:
public class Foo
{
public void Bar(string mode)
{
if (string.IsNullOrEmpty(mode)) throw new InvalidOperationException(Resources.InvalidModeSpecified);
}
public void Bar(int port)
{
if (port < 0 || port > Int16.MaxValue) throw new ArgumentOutOfRangeException("port");
}
}
[TestClass]
class ExampleClass
{
[TestMethod]
[ExpectedExceptionWithMessage(typeof(InvalidOperationException), typeof(Samples.Resources), "InvalidModeSpecified")]
public void Raise_Exception_With_Message_Resource()
{
new Foo().Bar(null);
}
[TestMethod]
[ExpectedExceptionWithMessage(typeof(ArgumentOutOfRangeException), "port", true)]
public void Raise_Exeception_Containing_String()
{
new Foo().Bar(-123);
}
}
Вот обновленный класс:
public class ExpectedExceptionWithMessageAttribute : ExpectedExceptionBaseAttribute
{
public Type ExceptionType { get; set; }
public Type ResourcesType { get; set; }
public string ResourceName { get; set; }
public string ExpectedMessage { get; set; }
public bool Containing { get; set; }
public bool IgnoreCase { get; set; }
public ExpectedExceptionWithMessageAttribute(Type exceptionType)
: this(exceptionType, null)
{
}
public ExpectedExceptionWithMessageAttribute(Type exceptionType, string expectedMessage)
: this(exceptionType, expectedMessage, false)
{
}
public ExpectedExceptionWithMessageAttribute(Type exceptionType, string expectedMessage, bool containing)
{
this.ExceptionType = exceptionType;
this.ExpectedMessage = expectedMessage;
this.Containing = containing;
}
public ExpectedExceptionWithMessageAttribute(Type exceptionType, Type resourcesType, string resourceName)
: this(exceptionType, resourcesType, resourceName, false)
{
}
public ExpectedExceptionWithMessageAttribute(Type exceptionType, Type resourcesType, string resourceName, bool containing)
{
this.ExceptionType = exceptionType;
this.ExpectedMessage = ExpectedMessage;
this.ResourcesType = resourcesType;
this.ResourceName = resourceName;
this.Containing = containing;
}
protected override void Verify(Exception e)
{
if (e.GetType() != this.ExceptionType)
{
Assert.Fail(String.Format(
"ExpectedExceptionWithMessageAttribute failed. Expected exception type: <{0}>. Actual exception type: <{1}>. Exception message: <{2}>",
this.ExceptionType.FullName,
e.GetType().FullName,
e.Message
)
);
}
var actualMessage = e.Message.Trim();
var expectedMessage = this.ExpectedMessage;
if (expectedMessage == null)
{
if (this.ResourcesType != null && this.ResourceName != null)
{
PropertyInfo resourceProperty = this.ResourcesType.GetProperty(this.ResourceName, BindingFlags.Static | BindingFlags.NonPublic | BindingFlags.Public);
if (resourceProperty != null)
{
string resourceValue = null;
try
{
resourceValue = resourceProperty.GetMethod.Invoke(null, null) as string;
}
finally
{
if (resourceValue != null)
{
expectedMessage = resourceValue;
}
else
{
Assert.Fail("ExpectedExceptionWithMessageAttribute failed. Could not get resource value. ResourceName: <{0}> ResourcesType<{1}>.",
this.ResourceName,
this.ResourcesType.FullName);
}
}
}
else
{
Assert.Fail("ExpectedExceptionWithMessageAttribute failed. Could not find static resource property on resources type. ResourceName: <{0}> ResourcesType<{1}>.",
this.ResourceName,
this.ResourcesType.FullName);
}
}
else
{
Assert.Fail("ExpectedExceptionWithMessageAttribute failed. Both ResourcesType and ResourceName must be specified.");
}
}
if (expectedMessage != null)
{
StringComparison stringComparison = this.IgnoreCase ? StringComparison.OrdinalIgnoreCase : StringComparison.Ordinal;
if (this.Containing)
{
if (actualMessage == null || actualMessage.IndexOf(expectedMessage, stringComparison) == -1)
{
Assert.Fail(String.Format(
"ExpectedExceptionWithMessageAttribute failed. Expected message: <{0}>. Actual message: <{1}>. Exception type: <{2}>",
expectedMessage,
e.Message,
e.GetType().FullName
)
);
}
}
else
{
if (!string.Equals(expectedMessage, actualMessage, stringComparison))
{
Assert.Fail(String.Format(
"ExpectedExceptionWithMessageAttribute failed. Expected message to contain: <{0}>. Actual message: <{1}>. Exception type: <{2}>",
expectedMessage,
e.Message,
e.GetType().FullName
)
);
}
}
}
}
}
Вы можете использовать код из этот проект объяснил в эта статья Чтобы создать атрибут wederexceptionmessage, который будет работать с MS Test, чтобы получить желаемый результат.
МС -тест (терпит неудачу):
[TestClass]
public class Tests : MsTestExtensionsTestFixture
{
[TestMethod, ExpectedExceptionMessage(typeof(System.FormatException), "blah")]
public void Validate()
{
int.Parse("dfd");
}
}
Я написал это некоторое время назад, так что, возможно, в VS2012 есть лучший способ.
Попробуй это. В этом примере у меня есть приветствие, которое отображает сообщение приветствия для данного имени. Когда параметр имени пуст или нулевая система бросает исключение с сообщением. А ExceptionAssert.Throws
Проверьте оба в MSTest.
[TestMethod]
public void Does_Application_Display_Correct_Exception_Message_For_Empty_String()
{
// Arrange
var oHelloWorld = new HelloWorld();
// Act
// Asset
ExceptionAssert.Throws<ArgumentException>(() =>
oHelloWorld.GreetingMessge(""),"Invalid Name, Name can't be empty");
}
[TestMethod]
public void Does_Application_Display_Correct_Exception_Message_For_Null_String()
{
// Arrange
var oHelloWorld = new HelloWorld();
// Act
// Asset
ExceptionAssert.Throws<ArgumentNullException>(() =>
oHelloWorld.GreetingMessge(null), "Invalid Name, Name can't be null");
}