679cb64554
[GB.TEST.TAP] * NEW: Add more diagnostics when Assert methods fail.
387 lines
8.5 KiB
Text
387 lines
8.5 KiB
Text
' Gambas class file
|
|
|
|
''' Assertions which print TAP.
|
|
|
|
Export
|
|
Create Static
|
|
|
|
Public Struct Subtest
|
|
Printer As TapPrinter
|
|
Description As String
|
|
Indent As Integer
|
|
Success As Boolean
|
|
' Directive to apply to the next "ok" line
|
|
Directive As Integer
|
|
Comment As String
|
|
End Struct
|
|
|
|
Property Output As Stream
|
|
Property Read Finished As Boolean
|
|
|
|
Private $aActiveTests As Subtest[]
|
|
Private $hCurrent As New Subtest
|
|
Private $hOutput As Stream
|
|
|
|
Public Sub _new()
|
|
|
|
Reset()
|
|
|
|
End
|
|
|
|
Public Sub Reset()
|
|
|
|
$aActiveTests = New Subtest[]
|
|
$hOutput = File.Out
|
|
With $hCurrent = New Subtest
|
|
.Printer = New TapPrinter As "Printer"
|
|
.Indent = 0
|
|
.Success = True
|
|
.Directive = Tap.NONE
|
|
End With
|
|
|
|
End
|
|
|
|
Public Sub Setup(Optional Tests As Integer, Optional Comment As String, Optional {Output} As Stream)
|
|
|
|
If IsMissing({Output}) Then {Output} = $hOutput
|
|
$hCurrent.Printer.Output = {Output}
|
|
Plan(Tests, Comment)
|
|
|
|
End
|
|
|
|
Public Sub Subtest(Description As String, Optional Tests As Integer)
|
|
|
|
Dim iIndent As Integer = $hCurrent.Indent
|
|
Dim hOutput As Stream = $hCurrent.Printer.Output
|
|
|
|
$aActiveTests.Push($hCurrent)
|
|
With $hCurrent = New Subtest
|
|
.Printer = New TapPrinter As "Printer"
|
|
.Printer.Output = hOutput
|
|
.Description = Description
|
|
.Indent = iIndent + 1
|
|
.Success = True
|
|
.Directive = Tap.NONE
|
|
End With
|
|
If Tests Then Plan(Tests)
|
|
|
|
End
|
|
|
|
Public Sub Finish()
|
|
|
|
Dim hFinished As Subtest
|
|
|
|
With $hCurrent
|
|
.Printer.Finish()
|
|
.Success = .Success And (.Printer.SkippedAll Or (.Printer.Planned > 0 And .Printer.Count = .Printer.Planned))
|
|
End With
|
|
hFinished = $hCurrent
|
|
|
|
Try $hCurrent = $aActiveTests.Pop()
|
|
If Not Error Then
|
|
$hCurrent.Printer.Test(hFinished.Success,, hFinished.Description)
|
|
Endif
|
|
|
|
End
|
|
|
|
Public Sub Printer_Filter()
|
|
|
|
Last.Line = String$($hCurrent.Indent, "\t") & Last.Line
|
|
|
|
End
|
|
|
|
Public Sub Plan(Tests As Integer, Optional Comment As String)
|
|
|
|
$hCurrent.Printer.Plan(Tests, Comment)
|
|
|
|
End
|
|
|
|
Public Sub SkipAll(Optional Comment As String)
|
|
|
|
$hCurrent.Printer.SkipAll(Comment)
|
|
|
|
End
|
|
|
|
Public Sub Ok(Result As Boolean, Optional Description As String) As Boolean
|
|
|
|
With $hCurrent
|
|
If .Directive <> Tap.NONE Then
|
|
.Printer.Test(Result,, Description, .Directive, .Comment)
|
|
Else
|
|
.Printer.Test(Result,, Description)
|
|
.Success = .Success And Result
|
|
Endif
|
|
.Directive = Tap.NONE
|
|
.Comment = Null
|
|
End With
|
|
Return Result
|
|
|
|
End
|
|
|
|
Public Sub Todo(Optional Comment As String)
|
|
|
|
$hCurrent.Directive = Tap.TODO
|
|
$hCurrent.Comment = Comment
|
|
|
|
End
|
|
|
|
Public Sub Skip(Optional Comment As String)
|
|
|
|
$hCurrent.Directive = Tap.SKIP
|
|
$hCurrent.Comment = Comment
|
|
Pass()
|
|
|
|
End
|
|
|
|
Public Sub BailOut(Optional Comment As String)
|
|
|
|
$hCurrent.Printer.BailOut(Comment)
|
|
$hCurrent.Success = False
|
|
|
|
End
|
|
|
|
Public Sub Diagnostic(Comment As String)
|
|
|
|
$hCurrent.Printer.Diagnostic(Comment)
|
|
|
|
End
|
|
|
|
Public Sub Note(Comment As String)
|
|
|
|
Diagnostic(Comment)
|
|
|
|
End
|
|
|
|
Public Sub Print({Line} As String)
|
|
|
|
$hCurrent.Printer.Print({Line})
|
|
|
|
End
|
|
|
|
Private Function Output_Read() As Stream
|
|
|
|
Return $hOutput
|
|
|
|
End
|
|
|
|
Private Sub Output_Write(Value As Stream)
|
|
|
|
$hOutput = Value
|
|
$hCurrent.Printer.Output = Value
|
|
|
|
End
|
|
|
|
Private Function Finished_Read() As Boolean
|
|
|
|
Return $hCurrent.Printer.Finished
|
|
|
|
End
|
|
|
|
' -------------------- High-level test functions --------------------
|
|
|
|
Public Sub Pass(Optional Description As String) As Boolean
|
|
|
|
Return Ok(True, Description)
|
|
|
|
End
|
|
|
|
Public Sub Fail(Optional Description As String) As Boolean
|
|
|
|
Return Ok(False, Description)
|
|
|
|
End
|
|
|
|
Public Sub NotOk(Result As Boolean, Optional Description As String) As Boolean
|
|
|
|
Return Ok(Not Result, Description)
|
|
|
|
End
|
|
|
|
Public Sub Equals(Got As Variant, Expected As Variant, Optional Description As String) As Boolean
|
|
|
|
Dim bRes As Boolean
|
|
|
|
bRes = Ok(Got = Expected, Description)
|
|
If Not bRes Then
|
|
Note(Subst$((" Got: &1"), Got))
|
|
Note(Subst$((" Expected: &1"), Expected))
|
|
Endif
|
|
Return bRes
|
|
|
|
End
|
|
|
|
Public Sub NotEquals(Got As Variant, UnExpected As Variant, Optional Description As String) As Boolean
|
|
|
|
Return Ok(Got <> UnExpected, Description)
|
|
|
|
End
|
|
|
|
Public Sub LessEqual(Got As Variant, Bound As Variant, Optional Description As String) As Boolean
|
|
|
|
Dim bRes As Boolean
|
|
|
|
bRes = Ok(Got <= Bound, Description)
|
|
If Not bRes Then
|
|
Note(Subst$((" Got: &1"), Got))
|
|
Note(Subst$((" Expected: <= &1"), Bound))
|
|
Endif
|
|
Return bRes
|
|
|
|
End
|
|
|
|
Public Sub Less(Got As Variant, Bound As Variant, Optional Description As String) As Boolean
|
|
|
|
Dim bRes As Boolean
|
|
|
|
bRes = Ok(Got < Bound, Description)
|
|
If Not bRes Then
|
|
Note(Subst$((" Got: &1"), Got))
|
|
Note(Subst$((" Expected: < &1"), Bound))
|
|
Endif
|
|
Return bRes
|
|
|
|
End
|
|
|
|
Public Sub GreaterEqual(Got As Variant, Bound As Variant, Optional Description As String) As Boolean
|
|
|
|
Dim bRes As Boolean
|
|
|
|
bRes = Ok(Got >= Bound, Description)
|
|
If Not bRes Then
|
|
Note(Subst$((" Got: &1"), Got))
|
|
Note(Subst$((" Expected: >= &1"), Bound))
|
|
Endif
|
|
Return bRes
|
|
|
|
End
|
|
|
|
Public Sub Greater(Got As Variant, Bound As Variant, Optional Description As String) As Boolean
|
|
|
|
Dim bRes As Boolean
|
|
|
|
bRes = Ok(Got > Bound, Description)
|
|
If Not bRes Then
|
|
Note(Subst$((" Got: &1"), Got))
|
|
Note(Subst$((" Expected: > &1"), Bound))
|
|
Endif
|
|
Return bRes
|
|
|
|
End
|
|
|
|
Public Sub Approximate(Got As Float, Expected As Float, Precision As Float, Optional Description As String) As Boolean
|
|
|
|
Return LessEqual(Abs(Got - Expected), Precision, Description)
|
|
|
|
End
|
|
|
|
Public Sub RelativeApproximate(Got As Float, Expected As Float, RelPrecision As Float, Optional Description As String) As Boolean
|
|
|
|
Return LessEqual(Abs((Got - Expected) / Expected), RelPrecision, Description)
|
|
|
|
End
|
|
|
|
Public Sub IsType(Got As Variant, Type As Integer, Optional Description As String) As Boolean
|
|
|
|
Return Equals(TypeOf(Got), Type, Description)
|
|
|
|
End
|
|
|
|
Public Sub Null(Got As Variant, Optional Description As String) As Boolean
|
|
|
|
Return Equals(Got, Null, Description)
|
|
|
|
End
|
|
|
|
Public Sub NotNull(Got As Variant, Optional Description As String) As Boolean
|
|
|
|
Return NotEquals(Got, Null, Description)
|
|
|
|
End
|
|
|
|
Public Sub Like(Got As String, Pattern As String, Optional Description As String) As Boolean
|
|
|
|
Dim bRes As Boolean
|
|
|
|
bRes = Ok(Got Like Pattern, Description)
|
|
If Not bRes Then
|
|
Note(Subst$((" Got: &1"), Got))
|
|
Note(Subst$((" Expected: &1"), Pattern))
|
|
Endif
|
|
Return bRes
|
|
|
|
End
|
|
|
|
Public Sub Match(Got As String, Pattern As String, Optional Description As String) As Boolean
|
|
|
|
Dim bRes As Boolean
|
|
|
|
bRes = Ok(Got Match Pattern, Description)
|
|
If Not bRes Then
|
|
Note(Subst$((" Got: &1"), Got))
|
|
Note(Subst$((" Expected: &1"), Pattern))
|
|
Endif
|
|
Return bRes
|
|
|
|
End
|
|
|
|
Public Sub StringEquals(Got As String, Expected As String, Optional Description As String) As Boolean
|
|
|
|
Dim bRes As Boolean
|
|
Dim iPos As Integer
|
|
|
|
bRes = Equals(Got, Expected, Description)
|
|
If Not bRes Then
|
|
If Len(Got) <> Len(Expected) Then
|
|
Note(("Strings are of different length."))
|
|
Note(Subst$((" Got: &1"), Len(Got)))
|
|
Note(Subst$((" Expected: &1"), Len(Expected)))
|
|
Endif
|
|
For iPos = 1 To Min(Len(Got), Len(Expected))
|
|
If Mid$(Got, iPos, 1) <> Mid$(Expected, iPos, 1) Then Break
|
|
Next
|
|
Note(Subst$(("Strings differ at position &1"), iPos))
|
|
Note(Subst$((" Got: &1"), Quote$(Mid$(Got, iPos, 20)) & IIf(Len(Got) > iPos + 20, "...", "")))
|
|
Note(Subst$((" Expected: &1"), Quote$(Mid$(Expected, iPos, 20)) & IIf(Len(Expected) > iPos + 20, "...", "")))
|
|
Endif
|
|
Return bRes
|
|
|
|
End
|
|
|
|
Public Sub Error(Optional Description As String) As Boolean
|
|
|
|
Return Ok( Error , Description)
|
|
|
|
End
|
|
|
|
Public Sub ErrorCode(Code As Integer, Optional Description As String) As Boolean
|
|
|
|
Dim bRes As Boolean
|
|
|
|
If Not Error Then
|
|
bRes = Fail(Description)
|
|
Note(("No error happened"))
|
|
Else
|
|
bRes = Equals(Error.Code, Code)
|
|
If Not bRes Then
|
|
Note(Subst$(("Error was: &1 (code: &2) at &3"), Error.Text, Error.Code, Error.Where))
|
|
Endif
|
|
Endif
|
|
|
|
Error.Clear()
|
|
Return bRes
|
|
|
|
End
|
|
|
|
Public Sub NotError(Optional Description As String) As Boolean
|
|
|
|
Dim bRes As Boolean
|
|
|
|
bRes = Ok(Not Error , Description)
|
|
If Not bRes Then
|
|
Note(Subst$(("Error was: &1 (code: &2) at &3"), Error.Text, Error.Code, Error.Where))
|
|
Endif
|
|
|
|
Error.Clear()
|
|
Return bRes
|
|
|
|
End
|