A basic unit test framework for lune.
I developed this mostly for myself at a time where there were no alternatives for Lune. I plan to keep it simple and use it for my projects that dont require a heavy framework.
If you require more than basic test running, assertions, and console reporting, consider using a larger framework (like jest-lua when it gets ported to lune, see issue for tracking)
- It runs tests
- Pretty ergonomic to write (I think so)
- Assertion library, including table & array checks with diff printing, should_error, and a few more!
- Builtin console reporter with colors
- (undocumented) api to implement custom reporters (see console_reporter.luau for sample use)
Here's what a test looks like:
return function()
test.suite("A collection of tests", function()
test.case("A test that passes", function()
local exp = 4
local got = 2 + 2
check.equal(got, exp)
end)
test.case("A test that fails", function()
local exp = 3
local got = 2 + 2
check.equal(got, exp)
end)
test.case("Halt early", function()
local a = nil
req.not_nil(a)
check.falsy(a.b) -- wont run, a failed req stops the test early
end)
test.case("Custom messages!", function()
check.equal(1, 2, msg("my message prints in addition to the expansion"))
end)
test.case("Array assertions", function()
local a = { "a", "b", "c" }
local b = { "a", "z" }
check.array.contains(a, { "z", "x" })
check.array.equal(a, b)
end)
test.case("Table assertions", function()
local a = { a = 1, b = 2, c = { a = 1, b = 2 } }
local b = { a = 1, b = 22, c = { a = -1 } }
check.table.contains(a, "b")
check.table.equal(a, b)
end)
end)
-- you can use test.focus to temporarily focus on a test case/suite:
test.focus.suite("only this suite will run", function() end)
test.focus.case("only this case will run", function() end)
-- you can use test.skip to skip test cases/suites:
test.skip.suite("all tests in this suite will be skipped", function() end)
test.skip.case("this test will be skipped", function() end)
end
And here is the output (theme is Rose Pine):
Installing
frktest is available on wally, pesde, or you can just clone the repo.
using wally
Add the dependency:
in wally.toml
:
[dev-dependencies]
frktest = "itsfrank/[email protected]"
Create alias in .luaurc
:
{
"aliases": {
"frktest": "DevPackages/_Index/[email protected]/frktest/src",
}
}
using pesde
in pesde.toml
:
[dev_dependencies]
frktest = { name = "itsfrank/frktest", version = "^0.0.2" }
Create alias in .luaurc
:
{
"aliases": {
"frktest": "lune_packages/.pesde/itsfrank+frktest/0.0.2/frktest/src/"
}
}
Note: If you want to use the generated luau file in ./lune_packages
, in
the examples, replace require("@frktest/frktest")
with
require("./lune_packages/frktest")
. Reporters will be avilable in the
_reporters
member:
-- from project root
local frktest = require("./lune_packages/frktest")
local console_reporter = frktest._reporters.console_reporter
A sample pesde project using frktest can be found here: https://github.com/itsfrank/frktest-pesde-sample
clone the repo
# somewhere on your machine
git clone https://github.com/itsfrank/frktest.git
Create alias in .luaurc
:
{
"aliases": {
"frktest": "<path to frktest/src>",
}
}
All example assume you created a require
alias "@frktest"
in your project's .luaurc
:
{
"aliases": {
"frktest": "<see install sections above for paths>",
}
}
note: you should also register the alias with your lsp of choice, with luau-lsp
using VSCode I think it would look like this:
"luau-lsp.require.mode": "relativeToFile",
"luau-lsp.require.directoryAliases": {
"@frktest/": "<alias in luaurc>"
}
The framework does not have a cli or test discovery. So you need to make an entrypoint file, see examples/_run.luau for how I made mine. But basically all you need is this:
-- filename: _run.luau
-- require test files and call the returned function
require("./some_test")()
-- initialize a reporter (there is currently only one... this one, but you can make your own!)
local console_reporter = require("@frktest/reporters/console_reporter")
console_reporter.init()
-- run the tests
local frktest = require("@frktest/frktest")
frktest.run()
Then you can run this file with lune like so:
> lune run _run.luau
I suggest you use these requires (but you do what you want!):
local frktest = require("@frktest/frktest")
local test = frktest.test
local check = frktest.assert.check
local req = frktest.assert.require
A test file should return a function, and for any tests to run the function
needs to create tests with test.case
. If you want to group tests together you
can use test.suite
and create tests inside of that. See examples
for how I wrote a bunch of test files.
return function()
test.case("a global test", function()
-- ...
end)
test.suite("a group of tests", function()
test.case("a test within a suite", function()
-- ...
end)
end)
end
Every single assertion has at least one example in the examples folder. If there are any nuances, I added comments.
Really though, they should be self explanatory, your LSP should list them all
out if you type assert.check.|
- Write tests to test the framework (ironic...)
- Encapsulate test state so that I can test the framework using the framework (with 2 separate states, one doing the testing, and one being tested)
- Probably add more assertions
- Fix bugs people might report
- martinfelis/luatablediff for a great table diff inplementation that I adapted for the
table.equal
assertions - kikito/inspect because it's the best pretty printer and I use it to print tables
- catch2 inspired the test output and probably the assertion syntax
My name is Frank, I often use frk
as a namespace/prefix for stuff meant for me :)