Jsonnetunit is a unit test framework for Jsonnet.
[TOC]
example_test.jsonnet
:
local test = import "jsonnetunit/test.libsonnet";
test.suite({
testIdentity: {actual: 1, expect: 1},
testNeg: {actual: "YAML", expectNot: "Markup Language"},
testFact: {
local fact(n) = if n == 0 then 1 else n * fact(n-1),
actual: fact(10),
expect: 3628800
},
})
Then, just evaluate your test file with jsonnet
.
$ jsonnet -J path/to/jsonnetunit example_test.jsonnet
{
"verify": "Passed 3 test cases"
}
On failure, it emits an error report and exits with non-zero status.
$ jsonnet -J path/to/jsonnetunit example_test.jsonnet
RUNTIME ERROR: Failed 11/11 test cases:
testFoo: Expected 1 to be 2
testBar: Expected 1 to satisfy the function
testBaz: Expected 1 to satisfy the condition that the value is between 2 and 3
path/to/jsonnetunit/jsonnetunit/test.libsonnet:40:13-25 object <anonymous>
During manifestation
See jsonnet/test/std_matchers_test.jsonnet
for more examples of other matchers.
Just git clone
.
$ git clone https://github.com/yugui/jsonnetunit.git
-
Create a test file
Test files must be
.jsonnet
files which manifestize a result oftest.suite
function.local test = import "path/to/jsonnetunit/test.libsonnet"; test.suite({ })
-
Add test cases
test.suite
function takes an object which contains fields prefixed withtest
. You can add arbitrary number of such fields.test.suite
does not directly use any other fields.Individual test fields must have at least two fields:
-
actual
field: There must be a field namedactual
. This is the actual value to be verified. -
expectation field: There must be another field which describes an expectation. This expectation is used to verify the
actual
value.test.suite({ testFoo: { actual: std.length('foo'), expect: 3, }, })
The interpretation of the expectation depends on the name of the expectation field. The name
expect
in the example means that it expects thatactual
field is equal to the given value.
-
Expectation Field Name | Description | Example |
---|---|---|
expect |
value equality | {actual: 1+1, expect: 2} |
expectNot |
value inequality | {actual: 1+1, expectNot: 3} |
expectLt |
less than | {actual: 1+1, expectLt: 3} |
expectLe |
less than or equal to | {actual: 1+1, expectLe: 3} |
expectGt |
greater than | {actual: 1+1, expectGt: 1} |
expectGe |
greater than or equal to | {actual: 1+1, expectGe: 1} |
You can describe an abitrary expectation with expectThat
.
This expectation field takes a unary function or an object.
The function must take an actual value and it must return if the value satisfies your expectation in a boolean value.
{
actual: ultimateAnswerToLifeTheUniverseAndEverything(),
expectThat: function(x) x == 42,
}
When you pass an object, the object must have two fields actual
and result
.
The first field actual
is overridden with the actual
value of the test case on evaluation.
The second field result
must be an boolean which describes whether actual
field satisfies your expectation.
In this case, you can optionally specifies a custom description
of the expectation. This is used as a part of error message when the test case fails.
{
actual: ultimateAnswerToLifeTheUniverseAndEverything(),
expectThat: {
actual: error "to be overridden",
result: self.actual == 42,
description: "Expect %d to be equal to the known value" % self.actual,
},
}
You can also define your own expectation matcher.
-
Define a binary function which takes
actual
andexpected
values. This function must return an object derived frommatcher.jsonnet
and must have the following three fields.satisfied
: (boolean) Returns ifself.actual
satisfies your expectationpositiveMessage
: (string) Returns an error message to be returned whenself.actual
does not satisfies your expectation.negativeMessage
: (string) Returns an error message to be returned whenself.actual
does not satisfies the negation of the expectation.
e.g.
local setMatcher(actual, expected) = import "jsonnetunit/matcher.libsonnet" { satisfied: std.set(actual) == std.set(expected), positiveMessage: "Expected " + actual + " to be equal to " + expected + " as a set", negativeMessage: "Expected " + actual + " not to be equal to " + expected + " as a set", };
-
Define your expectation field name in the
matchers
field of the test suite.e.g.
test.suite({ testEq: { actual: [6, 7, 2, 3, 7], expectSetEq: [2, 3, 6, 7], }, testNe: { actual: [6, 7, 2, 3, 7], expectSetNe: [1, 2, 3, 4, 5], } }) { matchers+: { // Define a new expectation field name "expectSetEq" for set equality expectSetEq: { matcher: setMatcher, expectationType: true, }, // Define a new expectation field name "expectSetNe" for set inequality expectSetNe: { matcher: setMatcher, expectationType: false, }, }, }
Running with Bazel
Use jsonnet_test
rule.
-
WORKSPACE
:... git_repository( name = "com_github_yugui_jsonnetunit", remote = "https://github.com/yugui/jsonnetunit.git", tag = "0.2.0", )
-
BUILD
:load("@com_github_yugui_jsonnetunit//jsonnetunit:jsonnetunit.bzl", "jsonnet_test") jsonnet_test( name = "your_test", src = "your_test.jsonnet", )
Copyright 2016 Yuki Yugui Sonoda
Licensed under the Apache License, Version 2.0 (the "License"); you may not use this file except in compliance with the License. You may obtain a copy of the License at
http://www.apache.org/licenses/LICENSE-2.0
Unless required by applicable law or agreed to in writing, software distributed under the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the License for the specific language governing permissions and limitations under the License.