上下文
我在VBA中写了一个小的unit testing library,所以我可以写这样的单元测试:
'@ExpectedError()
Public Sub TestCannotRegisterLoggerTwice()
Dim logger As ILogger
Set logger = MockLogger.Create("TestLogger", TraceLevel)
LogManager.Register logger
LogManager.Register logger
End Sub
以上测试通过:
2014-10-04 16:42:55 TestCannotRegisterLoggerTwice: [PASS]
如果我删除ExpectedError“属性”,它将失败:
2014-10-04 16:43:49 TestCannotRegisterLoggerTwice: [INCONCLUSIVE] - Test raised an error: Automation error
到现在为止还挺好.好吧,差不多.
测试中的错误工作正常!
现在如果我让测试将除法归零,就像这样:
'@ExpectedError()
Public Sub TestCannotRegisterLoggerTwice()
Dim logger As ILogger
Set logger = MockLogger.Create("TestLogger", TraceLevel)
Dim boom As Integer
boom = 1 / 0
LogManager.Register logger
LogManager.Register logger
End Sub
测试仍然没有结果,但这次错误被正确选取:
2014-10-04 16:45:12 TestCannotRegisterLoggerTwice: [INCONCLUSIVE] - Test raised an error: Division by zero
我不明白的部分是,在两种情况下,错误都是在同一个VBAProject中引发的 – 前者在一个名为LogManager(SUT)的类中,后者在一个名为LogManagerTests的类中.
这是抛出我正在测试的预期错误的部分:
Public Sub Register(ByVal logger As ILogger)
If Not this.Loggers.Exists(logger.Name) Then
this.Loggers.Add logger.Name, logger
Else
Err.Raise LogManagerError.DuplicateLoggerError, "LogManager.Register", "There is already a logger registered with name '" & logger.Name & "'."
End If
End Sub
我想要的是
我希望能够使用’@ExpectedError(-2147220406)(LogManagerError.DuplicateLoggerError的长表示)来装饰我的测试方法,并且能够在TestEngine.ReflectTestMethods方法中清除此部分:
For Each prospect In classMethods
If CanAddTestMethod(prospect, result) Then
Set tMethod = New TestMethod
Set tMethod.OwnerInstance = classInstance
tMethod.MethodName = prospect.name
If prospect.HasAttribute(ExpectedErrorAttribute) Then
If prospect.AttributeParameters(ExpectedErrorAttribute).Count <> 0 Then
Err.Raise vbObjectError + 1055, "TestEngine.ReflectTestMethods", "ExpectedErrorAttribute for test method '" & prospect.name & "' should not have parameters."
End If
'force an "Automation Error" to be expected,
'because VBA "eats" the expected error thrown in the SUT:
tMethod.ExpectedError = 440 ' prospect.AttributeParameters(ExpectedErrorAttribute).First
End If
result.Add tMethod.MethodName, tMethod
End If
Next
是这样吗?我只能指望SUT抛出自动化错误?
我试过的东西
由于错误是在实际测试方法中引发时正确报告的,我试过这个:
'@ExpectedError()
Public Sub TestCannotRegisterLoggerTwice()
On Error GoTo CleanFail
Dim logger As ILogger
Set logger = MockLogger.Create("TestLogger", TraceLevel)
LogManager.Register logger
LogManager.Register logger
Exit Sub
CleanFail:
Dim errNumber As Long
errNumber = Err.Number
Dim errSource As String
errSource = Err.Source
Dim errDescription As String
errDescription = Err.Description
Err.Raise errNumber, errSource, errDescription
End Sub
但即使errDescription包含正确/预期的错误消息,测试库仍会收到编号为440的自动化错误.
为什么我的测试库正确地将除法除以零,而不是自定义错误?
编辑:经过进一步的实验,我注意到我可以成功“预期”任何内置错误,所以这有效:
'@ExpectedError(5)
Public Sub TestCannotRegisterLoggerTwice()
Dim logger As ILogger
Set logger = MockLogger.Create("TestLogger", TraceLevel)
LogManager.Register logger
LogManager.Register logger
End Sub
…提供更改我的TestEngine代码以接受@ExpectedError属性的参数,并且我在SUT中引发错误#5 /“无效的过程调用或参数”,而不是自定义错误;我也无法提供自定义错误消息.这并不理想,因为它强制SUT只抛出内置错误,而我的测试库不能假设它是这种情况.
VBA中的自定义错误代码有什么特别之处?
最佳答案 我不知道VBA中的自定义错误代码有什么特别之处.
但是,如果SUT中抛出的错误号不是推荐的vbObjectError n的形式(可以从您期望的错误代码的-2147220406表示中推断出来),那么任何应用程序定义的或对象定义的错误都可以使用,并“预期”:
Public Enum LogManagerError
DuplicateLoggerError = 100
LoggerNotRegisteredError
End Enum
TestEngine仍然不会“看到”自定义错误消息,但现在您可以这样做:
'@ExpectedError(100)
Public Sub TestCannotRegisterLoggerTwice()
Dim logger As ILogger
Set logger = MockLogger.Create("TestLogger", TraceLevel)
LogManager.Register logger
LogManager.Register logger
End Sub
得到这个结果:
2014-10-04 18:09:56 TestCannotRegisterLoggerTwice: [PASS]
如果删除了ExpectedError属性,则会收到应用程序定义或对象定义的错误:
2014-10-04 18:10:20 TestCannotRegisterLoggerTwice: [INCONCLUSIVE] - Test raised an error: Application-defined or object-defined error
这在技术上是正确的 – 被授予,能够在测试结果中报告自定义错误消息会很好,但这只是必须处理的那些限制之一.
至少你现在可以在SUT中抛出不同的错误,并能够判断是否引发了错误 – 这是在这里实现的目标.
同时,自定义错误消息仍然可见,并以“正常”代码显示:
'@ExpectedError(100)
Public Sub TestCannotRegisterLoggerTwice()
On Error GoTo CleanFail
Dim logger As ILogger
Set logger = MockLogger.Create("TestLogger", TraceLevel)
LogManager.Register logger
LogManager.Register logger
Exit Sub
CleanFail:
MsgBox Err.Description
Err.Raise Err.Number
End Sub
2014-10-04 18:21:24 TestCannotRegisterLoggerTwice: [PASS]