gambas-source-code/comp/src/gb.test.tap/.src/Assert.class
Tobias Boege 679cb64554 gb.test.tap: Add more diagnostics
[GB.TEST.TAP]
* NEW: Add more diagnostics when Assert methods fail.
2020-02-27 20:38:55 +01:00

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