Classy Test
Meteor package which provides a class-based wrapper around tinytest
.
It has the following features:
- Class-based test cases.
- Common setUp/tearDown methods for test cases, separated for server and client side.
- A single test can interleave client-side and server-side assertions.
- Support for asynchronous tests.
- Compatible with tinytest (all tests are actually registered via tinytest).
Installation
meteor add peerlibrary:classy-test
Test cases
Each test case is a class extending from ClassyTestCase
as follows:
1class SimpleTestCase extends ClassyTestCase 2 # Define the test case name (required). 3 @testName: 'Simple' 4 # Define the timeout in milliseconds (optional). 5 @testTimeout: 200000 6 7 testThatTrueIsTrue: -> 8 @assertTrue true, "True should be true." 9 10 testThatFalseIsFalse: -> 11 @assertFalse false, "False should be false." 12 13# Register the test case. 14ClassyTestCase.addTest new SimpleTestCase()
This simple test case definition will generate two tests via tinytest, the first will be called Simple - ThatTrueIsTrue
and the second one Simple - ThatFalseIsFalse
.
The addTest
method also takes an optional second argument called options
, which should contain an object specifying any custom options. The following options are supported:
mustFail
should be set totrue
in case the test case must fail in order for it to be marked as passed.
Assertions
Classy test assertions are named slightly differently than in tinytest, but are otherwise equivalent with some additional assertions provided by default. Because test cases are class-based, assertions are methods which may be called in test context. The list of assertions is as follows:
assertEqual(actual, expected, message)
asserts thatactual
is equal toexpected
.assertNotEqual(a, b, message)
asserts thata
is not equal tob
.assertInstanceOf(object, class)
asserts thatobject
is an instance ofclass
.assertNotInstanceOf(object, class)
asserts thatobject
is not an instance ofclass
.assertRegexpMatches(string, regexp, message)
asserts thatstring
matches the regular expressionregexp
.assertNotRegexpMatches(string, regexp, message)
asserts thatstring
does not match the regular expressionregexp
.assertThrows(function, exception)
asserts thatfunction
throwsexception
.assertTrue(value, message)
asserts thatvalue
istrue
.assertFalse(value, message)
asserts thatvalue
isfalse
.assertIsNull(value, message)
asserts thatvalue
isnull
.assertIsNotNull(value, message)
asserts thatvalue
is notnull
.assertIsUndefined(value, message)
asserts thatvalue
isundefined
.assertIsNotUndefined(value, message)
asserts thatvalue
is notundefined
.assertIsNaN(value, message)
asserts thatvalue
isNaN
.assertIsNotNaN(value, message)
asserts thatvalue
is notNaN
.assertIn(value, collection)
asserts thatcollection
contains an elementvalue
.assertNotIn(value, collection)
asserts thatcollection
does not contain an elementvalue
.assertItemsEqual(actual, expected)
asserts that arraysactual
andexpected
contain the same elements (disregarding their order).assertObjectContainsSubset(actual, expected)
asserts that the key/value pairs in an objectactual
are a (non-strict) superset of those inexpected
.assertLengthOf(array, length, message)
asserts that the length ofarray
islength
.assertFail({type, message, stack})
asserts a failure.assertSubscribeSuccessful(endpoint, args..., callback)
asserts that subscription to Meteor endpointendpoint
using argumentsargs...
is successful. This is an async assertion wherecallback
is called after evaluation is completed.assertSubscribeFails(endpoint, args..., callback)
asserts that subscription to Meteor endpointendpoint
using argumentsargs...
fails with an error. This is an async assertion wherecallback
is called after evaluation is completed.
As mentioned, all assertions are methods and may be called on this
:
1 testFoo: -> 2 @assertEqual foo, bar, "Foo must be equal to bar." 3 @assertLengthOf [1,1,1], 3 4 # ...
Set up and tear down methods
Usually multiple tests share some common initialization and cleanup code. Using classy tests such code should be placed into set up and tear down methods. There are multiple of each, based on where they are executed:
setUp
runs both on the server and client side before each test.setUpServer
runs only on the server side before each test.setUpClient
runs only on the client side before each test.tearDown
runs both on the server and client side after each test.tearDownServer
runs on the server side after each test.tearDownClient
runs on the client side after each test.
Set up and tear down methods are actually specially named tests, so they may also invoke assertions. If we take the above testFoo
example, the order of executed methods is as follows:
1# Test initialization. 2@setUpServer() 3@setUp() 4@setUpClient() 5# Test body. 6@testFoo() 7# Test cleanup. 8@tearDownClient() 9@tearDownServer() 10@tearDown()
Server-side and client-side tests
By default all tests run both on client and server. It is possible to specify that some should only be executed on either the server-side or the client-side. This is done through a method naming convention which is as follows:
- If a method name begins with
testServer
then the test will only be executed on the server side. - If a method name begins with
testClient
then the test will only be executed on the client side.
Asynchronous tests
Tests can be specified in two ways:
- A single test method as in the above examples. When such a test method finishes, the test is deemed complete and the respective tear down methods will run.
- Test containing multiple steps where each step is only deemed complete after certain callbacks get called. This is similar to
testAsyncMulti
fromtest-helpers
.
In the second case, the test should not be defined as a method, but rather as an array of functions like in the following example:
1 testClientFoo: [ 2 -> 3 # Call the first method. 4 Meteor.call 'first', 'argument', @expect (error, result) => 5 @assertFalse error, "Error while calling first: #{ error }" 6 , 7 -> 8 # Call the second method. 9 Meteor.call 'second', 'argument', @expect (error, result) => 10 @assertFalse error, "Error while calling second: #{ error }" 11 ]
This defines a chain of sub-tests where the next case will only get executed once all the expected callbacks are run. In order to define which callbacks are expected one should use the @expect(fun)
method which takes a function argument and returns a wrapper function that will mark the callback as called. When all expected callbacks are called, the execution will proceed to the next sub-test in the chain.
Often one would like to abort the test early in case an expectation handler does not get called in a specified amount of time. In this case one may use the @expectWithTimeout(timeout, message, fun)
method where the timeout
argument specifies the timeout in milliseconds, the message
specifies what should be displayed when a timeout occurs and fun
is a callback similar to the normal @expect(fun)
call.
Note that in this case set up and tear down methods are only called once for the whole test and not in-between sub-tests.
Interleaving client-side and server-side assertions
Sometimes it can be useful to first run some tests on the client, then after those are done, run some tests on the server to check whether the client calls correctly affected the backend storage. This can be done by interleaving client-side sub-tests with server-side sub-tests. We take the previous async test example and add a server-side sub-test between the existing two using the @runOnServer
decorator:
1 testClientFoo: [ 2 -> 3 # Call the first method. 4 Meteor.call 'first', 'argument', @expect (error, result) => 5 @assertFalse error, "Error while calling first: #{ error }" 6 , 7 @runOnServer -> 8 # Check if the first method really cleared everything in Foo collection. 9 @assertEqual Foo.find().count(), 0 10 , 11 -> 12 # Call the second method. 13 Meteor.call 'second', 'argument', @expect (error, result) => 14 @assertFalse error, "Error while calling second: #{ error }" 15 ]
After the first
method call completes, the second sub-test will be executed on the server and all assertions will be propagated back to the client.
Another similar decorator is @runOnBoth
which will behave the same as @runOnServer
but will additionally also run the code on the client (in client-side tests) once it finishes executing on the server.
Passing variables from server-side tests to client-side tests
Sometimes there is the need of passing variables from server-side tests for use in client-side tests, usually when defining fixtures in setUp
methods. Consider this non-working example:
1 setUpServer: -> 2 # Initialize the database. 3 Foo.remove {} 4 5 # Create a test document. 6 @testDocumentId = Foo.insert 7 bar: true 8 9 testClientRemoval: -> 10 Meteor.call 'remove', @testDocumentId, @expect (error, result) => 11 @assertFalse error, "Error while remove: #{ error }"
So before the test starts we create a test fixture on the server and would then like to reference its _id
on the client. The problem is that this will not work as the test case instance on the server differs from the one on the client and @testDocumentId
will not be available there. In order to address this, classy tests support passing specific variables from server-side tests to client-side tests using @get
and @set
methods. In order to fix the above example we can do:
1 setUpServer: -> 2 # Initialize the database. 3 Foo.remove {} 4 5 # Create a test document. 6 testDocumentId = Foo.insert 7 bar: true 8 9 # Pass variable to client-side tests. 10 @set 'testDocumentId', testDocumentId 11 12 testClientRemoval: -> 13 Meteor.call 'remove', @get('testDocumentId'), @expect (error, result) => 14 @assertFalse error, "Error while remove: #{ error }"
In the background, the test framework will seamlessly transfer the variables between the tests. Note that as variables are transferred via DDP, they must be EJSON serializable.
Variables are automatically transferred in both directions, server-to-client and client-to-server.
Miscellaneous methods
You can use @subscribe
to subscribe to Meteor publish endpoint in a way which automatically unsubscribes on test tear down. You can use @unsubscribeAll
to force unsubscribing all subscriptions immediatelly.