动态测试

动态测试入门

仓颉测试框架支持动态测试,其支持在存在编译期未知的测试数据时构造测试用例。关键场景如下:

  • 创建基于外部数据的测试套。
  • 创建基于参数或配置文件的测试套。

通过与普通测试用例对比,可以看到如何通过 @TestBuilder 来构造动态测试用例。

如下是一个通过 @Test/@TestCase 构造的简单测试套:

@Test
class A {
    @TestCase
    func f() { @Assert(false) }

    @TestCase[x in [1, 2]]
    func g(x: Int) {
        @Assert( x >= 1 )
    }
}

通过 @TestBuilder 可以创建跟上述测试套相同逻辑的动态测试套 :

@TestBuilder
public func buildCustomTestSuite(): TestSuite {
   let suiteBuilder = TestSuite.builder("A")
   let caseConfiguration = Configuration()
   suiteBuilder.add(
       UnitTestCase.create("f", configuration: caseConfiguration) { @Assert(false) })
   suiteBuilder.add(
       UnitTestCase.createParameterized("g", [1, 2]) { value => @Assert( value >= 1 ) })
   suiteBuilder.build()
}

TestSuite 创建了一个 TestSuiteBuilder 对象,该对象支持添加用例。用例通过 UnitTestCase 类中的静态函数构造。该类支持构造简单用例或者参数化用例。 TestSuiteBuilder 被配置完毕后,最终创建生成一个 TestSuite 对象,作为被 @TestBuilder 修饰的函数的返回值。

当上述代码通过 --test 编译后再执行,输出结果与 @Test/@TestCase 构造的测试套一致。

--------------------------------------------------------------------------------------------------
TP: default, time elapsed: 121592 ns, RESULT:
    TCS: A, time elapsed: 121592 ns, RESULT:
    [ PASSED ] CASE: g (13969 ns)
    [ FAILED ] CASE: f (91641 ns)
    Assert Failed: `(false == true)`
       left: false
      right: true

Summary: TOTAL: 2
    PASSED: 1, SKIPPED: 0, ERROR: 0
    FAILED: 1, listed below:
            TCS: A, CASE: f
--------------------------------------------------------------------------------------------------

@TestBuilder 有如下约束:

  • 它只能修饰顶层函数,且该函数不能是 foreign 函数。
  • 它的返回值类型必须被显式指定,且必须为 TestSuite 类型。
  • 它可以与 @Configure / @Timeout / @Parallel 宏组合使用,不允许与 unittest.testmacro 包中其他的宏组合。

运行动态测试并获取输出

如果需要不从单元测试框架运行测试用例,而是从程序内部运行测试,则可以借助 TestSuite runTests() 函数来实现。

例如:

import std.unittest.*

main() {
   let suiteBuilder = TestSuite.builder("A")
   let caseConfiguration = Configuration()
   suiteBuilder.add(
       UnitTestCase.create("f", configuration: caseConfiguration) { @Assert(false) })
   suiteBuilder.add(
       UnitTestCase.createParameterized("g", [1, 2]) { value => @Assert( value >= 1 ) })
   let suite = suiteBuilder.build()
   let report = suite.runTests()
}

runTests 函数返回 Report 类的实例。您可以使用 Report 对象执行以下操作:

  • 使用 ConsoleReporter 类将其打印到控制台: report.reportTo(ConsoleReporter(colored: true))
  • 使用 TextReporter 类打印到任何 PrettyPrinter 实现: report.reportTo(TextReporter(into: PrettyText()))
  • 将结果输出为支持的格式之一,例如 XML( XmlReporter ,仅用于单元测试,不用于性能测试)或 CSV( CsvReporterCsvRawReporter ,仅用于性能测试)。