Skip to content

Latest commit

 

History

History
1296 lines (919 loc) · 66.3 KB

index.md

File metadata and controls

1296 lines (919 loc) · 66.3 KB

Rooibos test framework

Simple, mocha-inspired, flexible, fun Brightscript test framework for ROKU apps

Features

TABLE OF CONTENTS

Getting started

Rooibos is a brighterscript compiler and Visual Studio Extension plugin. That does not mean you need to write in brighterscript; the bsc compiler also parses regular brightscript projects.

The advantages of this are:

  • you get warnings and errors for your tests as you write them in the ide.
  • you do not have to install any thing in your projects, or maintain the rooibos libraries; it just works
  • it is very easy to integrate into your CI toolchain.

Your project MUST use Vscode and the brighterscript compiler

Rooibos will not work if you do not use the brighterscript compiler. I have no plans to support any other toolchains.

You MUST configure vscode ide to use your project's version of bsc compiler

Either click the select brighterscript button in the bottom right, or add this setting to your .vscode/settings.json file:

"brightscript.bsdk": "node_modules/brighterscript"

IMPORTANT: Disable rendezvous tracking, when launching from vscode

Roku will perform poorly if roku's rendezvous tracking is enabled, when launching from vscode. Please ensure your launch.json config has this flag set to false. e.g. "rendezvousTracking": false

{
      "type": "brightscript",
      "request": "launch",
      "name": "PROD",
      "stopOnEntry": false,
      "preLaunchTask": "build-prod",
      "envFile": "${workspaceFolder}/.vscode/.env",
      "host": "${env:ROKU_DEV_TARGET}",
      "password": "${env:ROKU_DEVPASSWORD}",
      "rootDir": "${workspaceFolder}/build",
      "files": ["*", "*.*", "**/*.*"],
      "enableDebugProtocol": false,
      "rendezvousTracking": false
    }

Installation

I do not have a project that uses the brighterscript transpiler

You will need to setup a bsc project to use rooibos 4. Here is an example project you can clone.

The easiest thing to do is to clone that project and

  • copy bsconfig.json
  • merge the contents of the files in .vscode folder
  • merge the contents of package.json into your project

You can do the setup from scratch as followes:

  1. Ensure your project is set up for use with npm (npm init, and follow the steps)
  2. Install brighterscript: npm install brighterscript --save-dev
  3. Install rooibos roku: npm install rooibos-roku --save-dev
  4. Add a script to your package.json, e.g.
  "scripts": {
    "build-tests": "npx bsc"
  }
  1. Add a bsconfig.json file. The easiest thing to do is copy the file from the example project
  2. Setup a task to run npm run build-tests, in .vscode/tasks.json
  3. Setup a launch task to run the build-tests task, from the previous step, in .vscode/launch.json
  4. Create some tests

I already have a project using brighterscript

  1. ensure you have a bsconfig.json, as per: https://github.com/rokucommunity/brighterscript
  2. npm install rooibos-roku --save-dev
  3. Add the rooibos plugin to your bsconfig.json file as follows:
  "plugins": [
    "rooibos-roku"
  ]

Rooibos will automatically inject the necessary source files into your project, at build time, and inject the hooks to run.

Note - you will likely use a separate bsconfig.json file for your tests, e.g. bsconfig-test.json, which will be used by your CI/testing vscode launch target.

Delaying start so you can set things up - CURRENTLY UNSUPPORTED

If you need to do some setup, like load some data from a service, before starting your tests:

  • add a field isReadyToStartTests to your tests scene
  • set the value to true, when the tests are ready to start.

NOTE: this is not yet supported in rooibos 4! I will add it again soon.

Configuring Rooibos's runtime behavior

Rooibos's configuration is controlled via the configuration passed into the bsconfig.json via flags on the rooibos json blob

e.g.

  "rooibos": {
    "isRecordingCodeCoverage": false,
    "printTestTimes": true,
    "testsFilePattern": null,
    "tags": ["!integration", "!deprecated", "!fixme"],
    "showOnlyFailures": true,
    "catchCrashes": true,
    "lineWidth": 70
  },

The following options are supported:

Here is the information converted into a Markdown table:

Property Type Description
logLevel? RooibosLogLevel 0-4 (error, warn, info, debug)
showOnlyFailures? boolean If true, then only failed tests are shown; but everything is shown if no failures occurred
printTestTimes? boolean If true, then the time each test took is output
lineWidth? number Width of test output lines in columns
catchCrashes? boolean If true, then any crashes will report CRASH statement, and note halt test execution - very useful for running a whole suite
throwOnFailedAssertion? boolean If true, then any failure will result in a runtime crash. Very useful for inspecting the stack frames and jumping right to the first failed test.
sendHomeOnFinish? boolean If true, then the app will exit upon finish. The default is true. Useful to set to false for local test suites
keepAppOpen? boolean When true, the app will remain open upon test completion. The default is true. Set false to return execution to Main
testsFilePattern? string The pattern to use to find tests. This is a glob. The default is "**/*.spec.bs"
tags? string[] The tags listed here control what is run. You can use !tagname to indicate any tests/suites that are skipped. All other tags are ANDed. This is useful for running specific subsets of your suite.
testSceneName string Test scene to use for the test run. Provide a different name here if you need custom setup in your scene. You should extend or duplicate the RooibosScene component (found in RooibosScene.xml)
isGlobalMethodMockingEnabled boolean Default is false. Enables mocking and stubbing support for global and namespace functions
isGlobalMethodMockingEfficientMode boolean default to true, when set causes rooibos to modify only those functions that were mocked or stubbed
globalMethodMockingExcludedFiles string[] Files that rooibos will not modify when adding global function or namespace function mocking support
reporter? @deprecated 1 string The built-in reporter to use. Defaults to empty. Possible values are console and junit.
reporters? 2 string[] An array of factory functions/classes which implement rooibos.BaseTestReporter. Built-in reporters include console and junit. Defaults to ["console"].

1 This parameter is deprecated, use reporters instead. When specified, the reporter will be appended to the list of reporters. 2 Custom reporters are not currently supported on node-based tests, because rooibos does not know which files it should include in the generated test components. This will be addressed in a future Rooibos version (see issue #266).

Creating test suites

Rooibos has a hierarchy of tests as follows:

  • Suite
    • Describe block
      • it test
        • Parameterized it test

Test suites are defined by:

  • declaring a class inside a namespace
  • which extends rooibos.BaseTestSuite
  • and has a @suite annotation

No special file naming is required. I recommend you call your files thing.spec.bs

Please note that rooibos tests are brighterscript only. You can test regular brs files; but all your tests must be brighterscript files.

Some advice

I find it really handy to have my own BaseTestSuite, that extends rooibos.BaseTestSuite and I use that as the base of all my tests. In this way I can easily use common utilities and use common beforeEach/setup for setting things up.

Simple example

The following is a minimum working example of a Rooibos test suite, named Simple.brs

namespace tests

  @suite("basic tests")
  class BasicTests extends rooibos.BaseTestSuite

    '+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
    @describe("tests the node context is available for a Node scope function")
    '+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++

    @it("does something in node scope")
    function _()
      m.assertNotInvalid(m.node)
      Tests.doSomethingInNodeScope(true)
      m.assertIonvalid(m._isNodeScopeVarSet)
      m.assertTrue(m.node._isNodeScopeVarSet)
    end function

  end class

end namespace

Note, you can also use sub, if you wish

namespace tests

  @suite("basic tests")
  class BasicTests extends rooibos.BaseTestSuite

    '+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
    @describe("tests the node context is available for a Node scope function")
    '+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++

    @it("does something in node scope")
    sub _()
      m.assertNotInvalid(m.node)
      Tests.doSomethingInNodeScope(true)
      m.assertIonvalid(m._isNodeScopeVarSet)
      m.assertTrue(m.node._isNodeScopeVarSet)
    end sub

  end class

end namespace

Simple example Notes

  1. The ++++++++++++'s around the @describe declaration are not required; but I find it makes the file much easier to read, and recommend it (or something similar, such as '***** as a best practice.
  2. If you do not provide names for the @suite it will use the class name
  3. The function name is not required. Rooibos will rename your function to whatever your describe group is (so you can easily navigate in the ide), and as such, the function name can be anything you like. I like calling all tests functions _. Don't worry - the plugin will rename it before you get any duplicate function errors!

Rooibos annotations

Rooibos provides a library of annotations which can be used to define test suites, describe groups, tests cases, Parameterized tests, and more. All annotations are of the following form

NOTE - these are official bsc compiler annotations; not like comments in the previous version of rooibos do not put a ' in front of the, and use @brighterscript("argument1", 2, "argument3", ["syntax])

@annotation(args...)

Where ANNOTATION, is the roku annotation and DATA is the data passed to it. e.g. @it("that it handles an empty collection"), defines a test case, with the title that it handles an empty collection

Some annotations act as modifiers. In these cases, they will affect some other annotation. For example @only, and @ignore will affect the following @suite, @it or @it annotation.

The following annotations are supported.

Annotation Description Data
@suite Indicates a file is a test suite. Required. Name of the test suite. Used in test output
@SGNode Indicates that a test will run in a node. Rooibos plugin will automatically generate the test node and inject all the test code. Note you have to have "autoImportComponentScript": true, in your bsconfig.json, for this style of testing to work. Name of the component to extend to run tests
@setup Run once when the suite, or describe group is executed .
@tearDown Run once when the suite or describe group is finished
@beforeEach Run before each test. Can be specified for the @suite, or for each @it group
@afterEach Run after each test. Can be specified for the @suite, or for each @it group
@describe Indicates a group of tests. Groupings improve readability. A group might denote various tests for a given method, or even an aspect of functionality Name of the describe group, which is used in the test output
@it Indicates a test. Must directly precede a function definition The name of the test case, which will be reported in the test output
@only Precedes a Suite, Describe group, or it test, to indicate that only that item should be executed. This can be used to rapidly filter out tests. Only other @only items will be run.
@ignore Precedes a suite, Describe group or it test, to indicate that that item should be ignored. If an @ignore tag is found before an item, then it will not be executed as part of the test run
@params[p1,p2,...,p6] Indicates a Parameterized test. Must come after a @it annotation. Can accept up to 6 arguments, which are comma separated. When using parameterized tests, the test function signatrue must accept the same number of arguments, and each of params statemens, must also have the same number of params Up to 6 arguments can be any valid brightscript code, which can be parsed with an eval function
@ignoreParams[p1,p2,...,p6] A Convenience tag, which makes it easy to temporarily comment out params tests we do not want to run. As per @params
@onlyParams[p1,p2,...,p6] A Convenience tag, which makes it easy to temporarily run just one set of params, so you can run one or more of the params in a params block. Very useful for focusing on a failing test case As per @params
@tags("one","two"..."n") Allows indicating the tags to apply to the group,test or suite. This is a really effective way to categorise your test suite. These tags can be used to filter tests in your rooibos bsconfig options. List of tag names to apply
@noCatch If present, will not catch errors for the test or suite it is placed on. This is handy when developing, and you want to debug the exact line on which an error occurred. none
@noEarlyexit If present, will not exit a test on an assertion failure, which prevents crashes/skewed results. This annotation is mainly used for testing, such as testing rooibos framework itself. It is recommend that you do not use this annotation. none
@async If present, on a test suite or test case (e.g. @it) indicates that the test will execute asynchronously. This will allow you to use observeField in your tests. The only argument is timeout in ms e.g. @async(2000). Default time out is 2000ms for a test, and 6000 for a test suite max time in ms

Writing tests

Rooibos tests must be placed within a group, which in turn must be placed inside a suite.

Tests then comprise assertions, which can test if values are equal, like (using fuzzy logic), contain values, etc.

An assertion looks like this:

m.assertTrue(myValue)

An example test is as follows.

@it("an instantiate with invalid modelLocator")
function Simpl_Basic_true()
    myValue = false
    m.assertTrue(myValue)
end function

In this case, the test will fail

Simple test output

Observe how the test output indicates in which file the test suite resides, and the line of the failed assert, plus the reason for the failure. If your IDE has integrated brightscript support, such as eclipse, you will find that the locations are clickable. In this example, clicking on the Location link will navigate the IDE to the exact line of code of the assertion.

Rooibos provids many assertions to test your code with:

  • assertFalse
  • assertTrue
  • assertEqual
  • assertLike
  • assertNotEqual
  • assertIonvalid
  • assertNotInvalid
  • assertAAHasKey
  • assertAANotHasKey
  • assertAAHasKeys
  • assertAANotHasKeys
  • assertArrayContains
  • assertArrayNotContains
  • assertArrayContainsSubset
  • assertArrayContainsAAs
  • assertArrayNotContainsSubset
  • assertArrayCount
  • assertArrayNotCount
  • assertEmpty
  • assertNotEmpty
  • assertArrayContainsOnlyValuesOfType
  • assertType
  • assertSubType
  • assertNodeCount
  • assertNodeNotCount
  • assertNodeEmpty
  • assertNodeNotEmpty
  • assertNodeContains
  • assertNodeNotContains
  • assertNodeContainsFields
  • assertNodeNotContainsFields
  • assertAAContainsSubset
  • assertMocks

If an assertion fails, then the next assertions will not run.

Special __rooibosSkipFields value

If an aa has the __rooibosSkipFields value set to an associative array of fields, as follows:

myObj = {
  id: "myId"
  refToSomethingThatWillLoopForever: myOtherObjPointingToMyObj
  __rooibosSkipFields: {
    refToSomethingThatWillLoopForever: true
  }
}

then rooibos will skip comparing myObj.refToSomethingThatWillLoopForever and will skip it when printing that object as as string.

This is useful in some scenarios, such as in maestro framework, where an object, might point to itself (i.e. m and top are the same, and the object has a reference to m.top)

Async tests

Rooibos runs in sync mode. Due to scenegraph limitations, we can't use observefield. We can workaround this though, using assertAsyncField

This assert allows you to wait on a field set on a task, or some other async manipulated object. Use as such:

  netTask = createObject("roSGNode", "NetworkTask")
  m.assertAsyncField(netTask, "output")

the framework will then check if the output field has changed, every 500 ms for 10 tries

You can control the timeout behavior by passing delay and maxAttempts, as follows:

  '2 second time out, 3 tries
  netTask = createObject("roSGNode", "NetworkTask")
  m.assertAsyncField(netTask, "output", 2000, 3)

If the field does not change during the retry period, the assertion will fail.

Setting up and tearing down

You may find that you have data which is common to all of your tests in a suite. In this case you can desginate functions to run, before and after all tests are executed in your suite. To achieve this, simply override the setup and tearDown functions. In our example above, we could do the following:

override function setup()
	m.values = [{index:1,name:"one"},{index:4, name:"four"},{index:12, name:"twelve"}]
	m.ds = CreateDataStore(m.values)
end function

@it("finds the correct index")
function _()
	item = m.ds.GetDataItemWithIndex(12)

	m.assertAAContainsSubset(item, m.values[2])
end function

Setup tearDown, beforeEach and afterEach Scoping

Setup and Teardown, can also be scoped to an describe group. If the annotations appear after an @describe group annotation, then the setup and teardown will apply only to that group. If the annotations appear before the first describe group annotation then they will be applied to all groups, which do not have Setup and Teardown group-level-annotations")

Like setup and tearDown, beforeEach and afterEach can be applied to the whole test by overriding the beforeEach and afterEach functions. You can scope them to @describe blocks, by using the @setUp, @tearDown, @beforeEach and @afterEach annotation above the relevant function. Note you can call back your test suite class's overall setup, tearDown, beforeEach and afterEach.

Using BeforeEach and AfterEach

In addition, we can also use beforeEach and afterEach to run before each and every test.

namespace Tests
  class SampleTest extends rooibos.BaseTestSuite

    override function setup()
      m.values = [{index:1,name:"one"},{index:4, name:"four"},{index:12, name:"twelve"}]
      m.ds = CreateDataStore(m.values)
    end function

    '+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
    @describe("tests alternate data")
    '+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++

    @beforeEach
    function alternateData_beforeEach()
      m.alternateValues = [{index:2,name:"two"},{index:3, name:"three"},{index:20, name:"twenty"}]
      m.alternateDS = CreateDataStore(m.alternateValues)
    end function

    @it("does not find correct index")
    function _()
      item = m.alternateDS.GetDataItemWithIndex(12)

      m.assertIonvalid(item)
    end function


    @it("finds correct index")
    function _()
      item = m.alternateDS.GetDataItemWithIndex(3)

      m.assertAAContainsSubset(item, m.alternateValues[1])
    end function
  end class
end namespace

Note that in the example above, each of the tests in the tests alternate data group, can be run with different values; but we do not need to set the up in each test, or mutate the values used by other tests, which were defined in the Setup method.

BeforeEach and AfterEach Scoping

BeforeEach and AfterEach, can also be scoped to an describe group. If the annotations appear after an `@describe(" group annotation, then they will only apply to that group. If the annotations appear before the first describe group annotation then they will be applied to all groups, which do not have BeforeEach and AfterEach group-level-annotations")

Parameterized tests

Many times, we want to test a broad range of values. Writing identical tests with different values is tedious and unnecessary, using Rooibos's @params mechanism

You can run the same test several times, by adding one or more @params(...) annotations after the test annotation and before the method declaration, as follows:

@it("tests assertTrue")
@params(true, true)
@params(false, false)
@params(invalid, false)
@params(0, false)
@params(1, false)
@params("test", false)
function _(value, expectedassertResult)
...

In this case, the test will be run once for each of the @params annotations. Note that the method signatrue takes parameters which correspond to the arguments in the params arrays. Rooibos will give you a build time error, and diagnostic in the ide if you screw this up to save you scratching your head later.

This makes it easy for us to pass in values to our tests, and expected output values, e.g.

@it("schedule shows correct index for now")
@params("04","10","15",0, 6)
@params("04","11","00",0, 7)
@params("05","15","20",1, 15)
@params("05","16","05",1, 17)
@params("05","18","00",1, 19)
function _(dayOfMonth, hour, minute, expectedDayIndex, expectedShowIndex)
	nowDate = SUT.CreateTestDate(dayOfMonth, hour, minute)

	m.scheduleContent.callFunc("UpdateTime", nowDate.asSeconds())
	m.assertEqual(m.scheduleContent.nowDayIndex, expectedDayIndex)
	m.assertEqual(m.scheduleContent.nowShowIndex, expectedShowIndex)
end function

Parameterized tests accept any valid json. However, the number of top level items in the params array must match the amount of arguments for your test method. If they don't the test will fail.

Advanced parameter directives

#RBSNode

You can instruct rooibos to create nodes as your parameter arguments. To do so, use the special value "#RBSNode", which will create a ContentNode for that value. You can also specify the nod type you wish to create. See the following example for guidance:

@it
@params("#RBSNode", "ContentNode")
@params("#RBSNode|Group", "Group")
@params("#RBSNode|Label", "Label")
function _(node, expectedNodeType)
  m.assertSubType(node, expectedNodeType)
end function

Parameterized test output

The output from parameterized tests shows the test name, and all of param configurations that were executed, making it easy to ascertain which param config results in a failure

Simple test output

Ignoring parameterized test configs

If you have a couple of failing param tests, it can assist debugging to ignore the param configs you are not interested in. In this case, you can place the @params annotation with @ignoreParams, and that config will not execute.

Rooibos 0.2.0, will include an @onlyParams annotation for further convenience.

Parameterized tests and other annotations

If a test case has a @only or @ignore annotation, the params will execute in accordance with their parent test case.

Node specific asserts

Rooibos adds some node specifc asserts, which are fully described in the assertion reference. These are:

  • assertNodeCount
  • assertNodeNotCount
  • assertNodeEmpty
  • assertNodeNotEmpty
  • assertNodeContains
  • assertNodeNotContains
  • assertNodeContainsFields
  • assertNodeNotContainsFields

Full list of asserts

The full list of asserts can be found in the documentation - assertion reference

Understanding test output

Rooibos reports test output in an easy to read hierarchical manner.

Each test suite has it's own tree structure in the output, and in turn has branches for each group, then each test that was run.

Success and Failure output

When a test passes, it is indicated by the presence of Success at the end of the line

Simple test output

When a test fails, it is indicated by a - in the trunk of the test output, and Fail, at the end of the line. In addition, failed tests contain a link to the line of the failed assertion, and the reason for the failure.

Simple test output

End of test report

At the end of the test report, you will get a report telling you how many tests ran, failed, crashed, and the time spent.

In addition, you get a simple to parse result as the last line of output, which makes it easy for you to ascertain the result for your Continuous Integration process.

Simple test output

Controlling which tests run

Rooibos is built to facilitate TDD, and other test-based developer workflows. For this reason, I made it very easy to specify which tests run, so you can simply execute 1 or a few tests while developing, then more as you finish the method you are currently working with.

Ignore annotation

If you place @ignore above a test suite, describe group, or test case, it will ignore it. i.e. it will not be executed.

You can give a reason for ignoring a test, as part of the annotation's data. e.g.

@ignore DataStore is being refactored
@it("that the correct index is NOT found")
function Simpl_Datastore_alternate_failures()
	item = m.alternateDS.GetDataItemWithIndex(12)

	m.assertIonvalid(item)
end function

The log reporter will indicate which tests are ignored, if you have log verbosity set to 2 or greater

Only annotation

If you place @only above a test suite, describe group, or test case, it will run that test in solo mode. In solo mode, execution is limited to those suites, groups or test cases, which also have a `@only' annotation.

A good working practice is to put a @only annotaiton on the suite for the class you are working on, then the group, then the individual test. You can then simply remove the annotation from the test when you have finished, and run the tests again, to see if you caused regression in any of the group's tests, then remove from the group and run the suite, then finally remove the @only annotation from the suite. This will allow you to run the least amount of tests at any time, while you work, giving you the fastest testing turnaround time.

Only show output for failures

In addition to the the @only and @ignore annotations, Rooibos has another mechanism for aiding the TDD process. You are able to execute Rooibos in showOnlyFailures mode. In this mode, all tests are executed (according to the @only and @ignore annotations); but if any failures are encountered, then only the failures are displayed. If all tests pass, then the stanard test output is shown.

This makes it easy to quickly dive into the test suite and see what regressions have been introduced, then you can simply navigate to the failed tests and annotate them with @only annotations (so that subsequent runs are much quicker)

This can be achieved by setting showOnlyFailures to true in the config, or, more conveniently, passing showOnlyFailures=true when launching the tests. An example make file target, might look like this:

testFailures:
	curl -d '' "http://${ROKU_DEV_TARGET}:8060/launch/dev?RunTests=true&showOnlyFailures=true&logLevel=4"

Accessing global scope

Note that all test suites contain the param nodeContext, which contains a reference to the node, global object and scene.

Using mocks and stubs

Rooibos can be used for integration, behavior, unit testing, and TDD. In many cases (particularly TDD and unit testing), we will not want to execute all of the methods invoked by our code; but will instead prefer to mock and stub those method calls.

In other cases, it may simply be impractical to execute code as part of our tests. Examples can be executing method that make network calls, require hardware, or playback, or just simply take to long.

In all of these cases, you can use Rooibos's mocks and stubs in place of real method calls.

Mocks, Stubs and Fakes

Fakes

Fakes are objects, which you can use in place of real methods. They are invoked, receive parameters, and can return values, just like real methods. As far as the invoked function is concerned, it is the real method you intended to be executed in your code. Fakes are what Rooibos uses under the hood, to facilitate it's Stub and Mock implementations.

Stubs

Stubs are Fakes which are not tracked. This means that you can execute your unit tests, using the fake methods; but Rooibos will not fail your tests if things were not as expected (i.e. wrong params, wrong number of invocations). These can be very handy for setting certain test conditions (i.e. you can create a fake network response that will return a specific result, which drives your test)

To create a stub, we use the stubCall method:

function stubCall(invocation, returnValue = invalid) as object
  • The target is the object which will have it's method replaced,
  • Method name is the name of the method to replace
  • expectedInvocations is the number of times we expect the method to be called
  • expectedArgs is an array of values we expect the method to be invoked with
  • returnValue is the value we wish to return
A simple example

Given a ViewModel, named DetailsVM, which has a method LoadDetails, as such:

function loadDetails()
	isNetworkRequestExecuted = m.executeNetRequest("http://my.data.com/get")
	m.isLoading = isNetworkRequestExecuted
	m.isShowingError = not isNetworkRequestExecuted
end function

We can use a stub, to facilitate testing against the network layer

detailsVM = createDetailsVM()
returnJson = {success:false}
m.stubCall(detailsVM.rexecuteNetRequest, returnJson)

detailsVM.loadDetails()

m.assertFalse(detailsVM.isLoading)
m.assertTrue(detailsVM.isShowingError)

In this case, our detailsVM object, will not actually call executeNetRequests's source code; but will instead call a fake (i.e fake method body), which can return predtermined values, or be later checked for invocation arg conformance.

Mocks

Mocks are expected fakes. Your code will invoke the method, as if it is the real method; but the difference is that Rooibos will track the invoction of mocks, and if the method was not invoked in the manner you expected (i.e. with the expected parameters and the expected number of invocations) then a unit test failure will result.

We create mocks by using the methods:

  • expectCalled(invocation, result)
  • expectNotCalled(method)

These are advanced functions, using the rooibos plugin to greatly simplify mocking. Simply include the actual invocation you expect, or the method you don't expect to be called, and the framework does the rest.

  • it will Create the underlying rooibos mock method for you
  • and will wire up the correct expected values and return values
  • it will automatically create the whole chain of objects required for the mock to work. For example, if you do m.expectCalled(m.screen.entitlementService.manager.isEntitled(), true) and screen, entitlementServic or manager do not exist, rooibos will automatically create the chain of objects as simple aa's with the relevant id, and setup the mock call for you.

CallFunc @. nuances

You can expectCalled on both method invocations and @. invocations, however. e.g. m.expectCalled(m.screen.entitlementService.manager.isEntitled(), true) m.expectCalled(m.screen.entitlementService.manager@.isEntitled(), true)

You can also expectNotCalled on both; but there is a slight difference here: m.expectNotCalled(m.screen.entitlementService.manager.isEntitled) m.expectNotCalled(m.screen.entitlementService.manager@.isEntitled())

For the regular class method variation, you can simply pass a pointer to the function, for the @.variation you must use an empty params invocation (to satisfy the brighterscript transpiler),

Legacy mocking methods

Under the hood rooibos leverages these methods; but with the modern syntax you will not typicall interact with these methods.

  • expect - Creates a generic mock
  • expectOnce - Creates a mock, which we expect to be called once or can created individual overloaded calls to the same method
  • expectNone - Creates a mock, which we never expect to be invoked

Mocking global and namespaced functions

Mocking and stubbing of global functions works the same as for class methods. Please note, the functions must be in scope - you cannot mock a global function call inside of a node. So if your call crosses an observer, or callfunc bridge, it will not work. This is by design, and there are no plans to provide intra-node global or namespace mocking, at this point.

To enable this feature, set the isGlobalMethodMockingEnabled rooibos config parameter to true. DO NOT enable this on your ide's bsconfig.json, as it will negatively impact performance and may cause other issues. This setting is not compatible with watch mode, at this moment.

Configuration

In addition to the isGlobalMethodMockingEnabled, you use the following config flags:

  • isGlobalMethodMockingEfficientMode - default to true, when set causes rooibos to modify only those functions that were mocked or stubbed
  • globalMethodMockingExcludedFiles - files that rooibos will not modify, when adding global mocking or stubbing support.

Known limitations:

You must include default values in your expect calls, even if your invoking code does not use them

  • if you mock or stub a global or namespaced method, you will have to expect default parameters in your expectCalled invocation, as rooibos will receive the default values.

e.g.

namespace utils
  function getModel(name as string, asJson = false as boolean)
  end function
end namespace

'in the code
function getBeatlesModel()
  utils.getModel("Ringo")
end function

'in the unit test
@describe("getBeatlesModel")
@it("gets ringo")
function _()
  'This will fail, because rooibos will receive the asJson param with the default value
  m.expectCalled(utils.getModel("ringo"))

  'therefore we need to do this
  m.expectCalled(utils.getModel("ringo", false))
end function

Note - this will be addressed shortly; pr's are welcome in the interim.

There might be some bugs, or unknown issues

Mocking, of global and namespaced functions is supported; but is a relatively new feature. It is possible that there are some scenarios that are not supported. Please raise issues in the rooibos slack channel, if you spot any anomalies.

Asserting mocks

Mocks are asserted by invoking m.assertMocks()

** You will not normally need this information**

As a convenience, Rooibos will automatically assert any mocks for you when your test finishes executing. This saves you from having to manually add this line at the end of your code.

When a mock fails, Rooibos will report to you what caused the failure. The possible reasons are:

  • The method was invoked by your code, with the wrong parameters
  • The method was invoked too many times by your code
  • The method was never invoked by your code; but it should've been

For example, the above test could be written with mocks as follows:

detailsVM = createDetailsVM()
returnJson = {success:false}
m.expectCalled(detailsVM,.executeNetRequest() returnJson)

detailsVM.loadDetails()

m.assertFalse(detailsVM.isLoading)
m.assertTrue(detailsVM.isShowingError)

In addition to the other asserts, Rooibos will automatically check that ExecuteNetRequest was executed exactly once

Manually Checking mock invocations

** You will not normally need this information**

If you wish you, you can store a reference to a mock or a stub, and later check the invocation values.

All mock and stub methods return a reference to the Fake that it wired into your code. e.g.

executeMock = m.expectCalled(detailsVM.executeNetRequest(), returnJson)

detailsVM.loadDetails()

? executeMock.invokedArgs
? executeMock.invocations

Specifying expected invocation arguments

You can save yourself a lot of time, and really think about and kick the tyres of your code, by defining the arguments you expect a function to be invoked with. This is done by passing in an array of the expected invocation arguments via the expectedArgs param. You may also really not care about the args, in which case you can set that value to invalid and the call to m.assertMocks() will skip checking the invoked args.

assertion limitations.

  • Up to 15 arguments are supported on mocked methods
  • You can have up to 24 mocks.

expecting several calls to the same method, with verified invocation params

You may wish to call the same method various times, and return different values, or check different arguments were invoked

In this case, we use overloaded expectCalled calls, as per the following example:

  m.expectCalled(obj.mockMethod(arg1), result1)
  m.expectCalled(obj.mockMethod(arg2), result2)
  m.expectCalled(obj.mockMethod(arg3), result3)

This will now set the framework to expect 3 calls to mockMethod, the first with value of arg1, returning result1; the second with value of arg2, returning result2; and the last with value of arg3, returning result3

The order of calls must match the invocations

Rooibos checks that the invocation values match

If you specify the invoked args, then by default, rooibos will check that the invoked args match the arguments you specify.

So for example, if one had the following mock

m.expectCalled(myObj.myMethod("a", "b"), true)

and we invoked myObj.myMethod("a", "b", "c") this will be a mock failure, because the invocation arguments do not match.

Only checking the call is made, and ignoring the params

If you wish to ignore all params for an invocation, simply use m.expectCalled with a function pointer. e.g. m.expectCalled(myObj.myMethod, true), which will simply check the function is called, and return true.

Note, this feature is NOT available for @. invocations.

Skipping value assertion with m.ignoreValue

If you only care about some arguments, then you can use m.ignoreValue to instruct rooibos to not assert equality for the arguments you've ignored.

In the above example, the assertion will be satisfied with a mock configured thusly:

m.expectCalled(myObj.myMethod("a", "b", m.ignoreValue), true)

This will pass when myMethod is invoked with args: "a", "b", "c", or "a", "b", or "a", "b", 22, etc

Using matchers to assert mock invocation args

In addition to the other basic matchers described above, rooibos provides a set of pre-defined any matchers, and a mechanism for you to provide custom matchers for any mock argument.

Built in matchers

Rooibos has matchers that will pass, if the values are of the specified anyXXXMatcher type. For convenience, the matchers are stored on m.anyXXXMatcher on your test suite. The following are available.

  • anyBoolMatcher
  • anyNumberMatcher
  • anyStringMatcher
  • anyArrayMatcher
  • anyAAMatcher
  • anyNodeMatcher

Simply specify the matcher in your mock definition, as follows:

m.expectCalled(myObj.myMethod(m.anyStringMatcher, m.anyBoolMatcher), true)

In this case, the mock be satisfied if it was called with 2 params, the first one a string, the second a bool, both of any value.

Custom matchers.

It is simple to use a custom matcher to assert your mock arguments. Simply:

  • implement a function that takes one argument, and returns true or false
  • ensure it's in scope
  • return your function (either by pointer, or inline), in an aa as follow: `{"matcher": yourMatcherFunction}

For example, using a function pointer (.brs):

  m.expectCalled(m.myClass.doWork({"matcher": Rooibos_Matcher_anyArray}), returnValue)

For example, using a function pointer (bs):

  m.expectCalled(m.myClass.doWork({"matcher": rooibos.Matcher.anyArray}), returnValue)

And inline:

  m.expectCalled(m.myClass.doWork({ "matcher": function(value)
        return value = true
    end function }), returnValue)

returning values from your mocks

Simply set your return value to a non-invalid value, to specify a return value. You can also specify multiple return values, if you expect your mock to be executed multiple times, and would like to use different values. In that case, return a pobo, with the special key : multiResult. The value will be returned for each invocation - if the mock is invoked more than the number of return values, the last one is returned. e.g.

  m.expectCalled(obj.mockMethod(5), {"multiResult": ["one", 2, invalid, "last"]}, true)

  m.assertEqual(obj.mockMethod(), "one")
  m.assertEqual(obj.mockMethod(), 2)
  m.assertEqual(obj.mockMethod(), invalid)
  m.assertEqual(obj.mockMethod(), "last")
  m.assertEqual(obj.mockMethod(), "last")

Allowing mocking of non-existent functions

By default this is disabled because using invocations means that the brighterscript diagnostics will inform you of any illegal or unknown method calls. This flag should remain disabled.

This is documented for legacy purposes

As a sanity check, rooibos will expect a method to exist on an object before allowing it to be stubbed or mocked. If it is not, it will fail log an error and lead to a failing mock assertion.

This behavior can be disabled by passing in the last argument of the mock/stub/fake function (allowNonExistingMethods) as true. In this case, rooibos will still log a warning; but you will be allowed to create the fake method.

This is handy for quickly creating mock dependencies with simple aa objects. e.g.

videoService = {}
m.expectCalled(videoService.getVideos, someJson, true)

Note, you can also opt to disable the error at the whole test suite level; by setting m.allowNonExistingMethods = true in your test suite code.

Command Line Interface

Rooibos includes a simple CLI that can be used to run the tests on a Roku Device from the command line.

To use the CLI, you call it with references to the bsconfig.json file defining your test project, and the host and password of a Roku device that is in developer mode:

npx rooibos --project=<path_to_bsconfig.json> --host=<host> --password=<password>

The test runner CLI will:

  1. build the app as defined in the given bsconfig.json file
  2. deploy the app the Roku device specified
  3. send the Roku's console output to stdout
  4. exit with status 0 on success, or 1 on failure.

Integrating with your CI

Rooibos CLI can be used directly in your CI process.

An example make target might look like

continuousIntegration: build
	echo "Running Rooibos Unit Tests"
	npx rooibos --project=<test project bsconfig.json> --host=${ROKU_DEV_TARGET} --password=${ROKU_DEV_PASSWORD}

Alternately, you can manually deploy the app after it has been built, and check the output. Because the test output has a convenient status at the end of the output, you can simply parse the last line of output from the telnet session to ascertain if your CI build's test succeeded or failed.

Note that Rooibos doesn't need any special parameters to run. If you follow the standard setup the tests will run. Simply ensure that your build system includes, or does not include rooibosDist.brs (and better still, all of your tests), depending on whether you wish to run the tests or not.

An example make target might look like

#amount of time to wait for unit test execution
ROKU_TEST_WAIT_DURATION=5
ROKU_TEST_ID=[Some identifier for your tests, should be set by CI as the buildId/git hash]

continuousIntegration: build install
	echo "Running Rooibos Unit Tests"
	curl -d '' "http://${ROKU_DEV_TARGET}:8060/keypress/home"
	curl -d '' "http://${ROKU_DEV_TARGET}:8060/launch/dev?"
	-sleep ${ROKU_TEST_WAIT_DURATION} | telnet ${ROKU_DEV_TARGET} 8085 | tee dist/test.log
	echo "=================== CI TESTS FINISHED =================== "

	if tail -2 dist/test.log | head | grep -q "RESULT: Success"; then echo "SUCCESS"; else exit -1; fi

In the above example, we pipe the output of the telnet session to dist/test.log, wait for the test execution to finish, and finally, parse the line of test output to check for the SUCCESS flag.

Testing nodes

Rooibos is unashamedly, predominantly a TDD framework, aimed at facilitating rapid TDD with class based roku development.

However, it is useful to be able to test whole real nodes, with observes, and promises, in integration scenarios. In this case you will leverage rooibos "node tests". These are tests that

  • Extend an existing node, so that you can access their node space functions
  • Run on the SG render thread
  • Can run in asynchronous mode, so that you can observe tasks, and other async fields, in your tests, and wait on the response.
  • Node tests are easy to use. All required code is auto-generated for you. Simply add the SGNode("nodeName") annotation to your test suite, where "nodeName" is the component you wish to test.

Example Node test file

In the following example, the tests will be run in a new (auto-generated) component that extends NodeExample component.

NOTE: you have to have "autoImportComponentScript": true, in your bsconfig.json, for this style of testing to work.

namespace Tests
  @SGNode NodeExample
  @suite [NET] Node Example Tests
  class NodeExampleTests extends rooibos.BaseTestSuite
    override function setup()
      m.setupThing = "something created during setup"
    end function

    '+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
    @describe("tests methods present on the node")
    '+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++

    override function beforeEach()
      m.beforeEachThing = "something created beforeEach"
    end function

    @it("HelloFromNode")
    function helloFromNode_simple()
      'bs:disable-next-line
      text = HelloFromNode("rokucommunity", 12)
      m.assertEqual(text, "HELLO rokucommunity" + " age:" + stri(12))
    end function
...

Nuances of testing nodes

The behavior of your unit tests is identical to unit testing any other class, with 2 caveats:

  1. m does not refer to your node, it refers to the test suite, as per normal testing. The following variables are available to you in your test as a convenience:
  • m.node - codebehind for your xml file (e.g. your brs's m)
  • m.top the node you are testing
  • m.global access to the global node
  1. You cannot stub node methods. If there is no . reference to a method (via an associated array style object), then Rooibos cannot mock it.
  2. Asynchronous code, such as an observer, requires an async test. This is explained in the next section

Async testing

To indicate a test suite will run in async mode you will mark the test suite with the @async annotation or mark one ore more tests with @async. This will cause rooibos to run the tests in async mode. If all tests in the suite do not complete within the timeout, then the suite will fail. The default time out is 60 seconds.

When you mark any test in a suite with the @async annotation this will keep the test running in the background, waiting for a call to m.done(). If the call is not made within the timeout, then the test fails. Again, the default timeout for each test is 2 seconds.

Note: you are not required to add @async to your testSuite, it is implied when you add it one more more tests. However, you may wish to add tge async annotation to your suite, to override the default of 60 seconds (e.g. with the annotation async(12000) we are instructing rooibos to wait up to 2 minutes for the whole suite).

Configuring timeouts

You can configure both the testSuite timeout and each test timeout, by providing the ms duration as the argument for the @async annotation. e.g. @async(5000) sets the timeout to 5 seconds.

Calling done, to indicate async operation is finished.

You must call testSuite.done() on your async test. the done method is available on m.done() if your code is executing in your test suite, and on m.testSuite.done(), if your code is executing in the "node scope", of your component (e.g. on an observe callback).

Calling done multiple times should have no effect; but delayed calls to done, from previous tests, may cause your later async tests to fail. It is best to cleanup your observers and tasks, and other async elements, in your test tearDown/afterEach functions.

IMPORTANT! Cancelling pending calls to done.

You should remove any observers and cancel any tasks, etc, in your test cleanup (e.g. afterEach or tearDown methods)

Example

NOTE: you have to have "autoImportComponentScript": true, in your bsconfig.json, for this style of testing to work.

namespace tests
  @only
  @async
  @SGNode("NodeExample")
  @suite
  class NodeExampleTests extends rooibos.BaseTestSuite
    protected override function setup()
      m.setupThing = "something created during setup"
    end function

    protected override function beforeEach()
      m.beforeEachThing = "something created beforeEach"
    end function

    '+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
    @describe("group1")
    '+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++

    @async
    @it("HelloFromNode")
    function _()
      'bs:disable-next-line
      text = HelloFromNode("rokucommunity", 12)
      m.AssertEqual(text, "HELLO rokucommunity" + " age:" + stri(12))
      m.done()
    end function

    '+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
    @describe("group2")
    '+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++

    @async(10)
    @it("HelloFromNode with params")
    @params("jon", 40)
    @params("ringo", 23)
    @params("ringo", 50)
    @params("ringo", 24)
    @params("george", 40)
    @params("paul", 50)
    function _(name, age)
      'bs:disable-next-line
      text = HelloFromNode(name, age)
      m.AssertEqual(text, "HELLO " + name + " age:" + stri(age))
      m.timer = createObject("roSGNode", "Timer")
      m.timer.duration = 0.5
      m.timer.observeFieldScoped("fire", "OnTimer")
      m.timer.control = "start"
    end function
  end class
end namespace

function OnTimer()
  ? " TIMER IS FIRED"
  m.testSuite.done()
end function

Advanced setup

Global test setup

I find it really handy to have my own BaseTestSuite, that extends rooibos.BaseTestSuite and I use that as the base of all my tests. In this way I can easily use common utilities and use common beforeEach/setup for setting things up.

You can always call the super beforeEach/setup/teardown/etc from your other tests, making it trivial to do global setup functions.

Working with tests

There are many articles on the benefits of TDD, and how to go about TDD, which you can readily find, such as this link which has a good overview, plus a great video with warnings of how to take it too far.

Personally, I get great mileage doing TDD in roku by using MVVM pattern (I'm soon to release a lightweight MVVM framework). You can get more info on that here (for iOS), and ideas on how that benefits roku dev here

I use these practices as much as possible because:

  1. Keeps my view and business logic separated
  2. Tests are super fast to run (I generally run about 400 or so in each of my projects in < 5 seconds, and when running 1 test, turn around time is < 3 seconds)
  3. I can test all of my logic and ideas out in isolation, before I start up my app, navigate to the correct page
  4. Easy to mock my code, so that I can test a broad range of network conditions
  5. I get a Regression test suite as I write my code
  6. It's really the quickest way to develop. In my 20+ years professional experience those who follow TDD (or some less-strict form of it) write far more, and far better code than their counterparts, and their projects end up being way more maintainable over the years.

Personally, I like to have my test in one window, the code I'm testing in another, and a third window with a telnet session, showing my testing output. This is especially handy, as Rooibos hyperlinks to the lines that assertions fail on, so I can easily navigate straight to the problematic tests.

E.g.

Simple test output

I slap an @only annotation on the suite, group and test I'm currently working on, and have a hotkey to run my tests.

Once my test passes, I test for regression, then run the whole suite, then run the app and test out in the UI.

Roobios vscode Extension

I have made an extension that adds rooibos (and maestro framework, which I also maintain) support to vscode. It adds useful features such as:

  • navigate straight to test
  • navigate to code
  • @only, on a suite, group or test, with a hotkey
  • Auto generate a run config, to only run files that are open in vscode right now (saves on compile times, and is excellent for TDD workflows).

Get it here

Backward compatibility

Rooibos is NOT backward compatible with pre version 4 versions of rooibos; and not compatible with any other unit test framework.

Generate code coverage

Rooibos can measure and report the test coverage your unit tests are producing. The output is shown in the ide, when your test execution has finished.

WARNING - Running code coverage is slower and may crash your device

Running a code coverage report is a lot slower than a standard test, due to continual probing of your source. Only run coverage when needed. Also - be aware it will crash very tight loops, like inside of a while waiting on a port, which will probably crash and hard reset your device.

It is advisable that your app has run_as_process=1 in the manifest

Recording coverage

To record coverage, set the isRecordingCodeCoverage to true, in your rooibos config (in your bsconfig.json file). Indicate which files to exclude, using the coverageExcludedFiles: [], array in the rooibos config. e.g.

"rooibos": {
    "isRecordingCodeCoverage": true,
    "coverageExcludedFiles": [
        '**/*.spec.bs',
        '**/some-vendor-library/**.*'
    ]
},

Statement support

The following statements types are supported:

  • variable assignments
  • method calls
  • nested function definitions (i.e. functions inside of arrays, variable assignments, method call args, or associative arrays)
  • if statement conditions
  • blocks of code in if, else if, and else statements
  • for and while loops
  • print statements

The following statements are not supported

  • goto
  • named lines (e.g. myGotoLine:)

Coverage report

When your coverage test run finishes rooibos will print out:

+++++++++++++++++++++++++++++++++++++++++++
Code Coverage Report
+++++++++++++++++++++++++++++++++++++++++++

Total Coverage: 5.066445% (610/12040)
Files:  17/ 100

HIT FILES
---------
pkg:/components/ContentRetriever/ContentRetriever.brs: 2.461539% (8/325)
pkg:/components/Analytics/AnalyticsManager.brs: 3.125% (6/192)
pkg:/components/Service/AuthenticationService.brs: 3.532609% (13/368)
pkg:/components/Container/UserContainer/UserContainer.brs: 3.703704% (1/27)
pkg:/components/MaintenanceCheckTask/AppConfigurationTask.brs: 7.407407% (2/27)
pkg:/components/Storage/PersistentStorage.brs: 15.27778% (11/72)
pkg:/source/Analytics/AnalyticsVideoMixin.brs: 16.92308% (22/130)
pkg:/components/DeveloperService/DeveloperService.brs: 33.33334% (2/6)
pkg:/components/Log/Log.brs: 45% (9/20)
pkg:/source/Modules/BaseModule.brs: 55.55556% (5/9)
pkg:/source/Log.brs: 65% (13/20)
pkg:/source/Modules/DeepLinkingModule.brs: 76.92308% (50/65)
pkg:/components/GlobalInitializer/GlobalInitializer.brs: 79.64601% (90/113)
pkg:/source/Modules/RecommendationModule.brs: 84.9315% (186/219)
pkg:/source/Mixins/ParsingMixin.brs: 86.47343% (179/207)
pkg:/source/Mixins/SMCErrors.brs: 92.30769% (12/13)
pkg:/source/Analytics/AnalyticsConstants.brs: 100% (1/1)

MISSED FILES
------------
pkg:/components/CustomRowListItem/CustomItemGenres/CustomItemGenres.brs: MISS!
pkg:/components/StringUtils.brs: MISS!
pkg:/components/Core/Components/TabComponent/TabComponent.brs: MISS!
pkg:/components/Core/Components/NoKeyPressRowList.brs: MISS!
pkg:/components/Model/TabComponentContent.brs: MISS!

e.g.

coverage output

  • Total coverage - % (num of hit lines/ num of trackable lines)
  • Files: num of hit files / total num of trackable files

Following is a list of all the hit files, and their coverage % and (hit lines/total lines)

Lastly the files that were not hit at all, during test execution.

The current implementation is capable of tracking lcov style statistics; to do this, run rooibos-cli with the arg --printLcov or add "printLcov": true to your gulp build.

In this case the lcov report is printed to the end of the console output. Thanks very much to @Ronen on the slack channel for this contribution!

The report is contained after the LCOV.INFO file. Given that that the console output could be saved, it should be trivial to watch the log output file, and update your lcov file after running your tests.

e.g.

coverage output