Gren Effectful Test Runner
Run Gren integration tests that depend on the actual results of tasks.
See the full API on the package site.
Quick Start
Create a directory for your tests and initialize a gren program targeting node:
mkdir tests
cd tests
gren init --platform=node
Install the package:
gren package install blaix/gren-effectful-tests
Create a src/Main.gren with your tests.
See the examples below,
the working example in the repo,
or the package docs
for how to do that.
Then compile and run your tests:
gren make src/Main.gren
node app
Examples
Your test runner should be a Node.SimpleProgram
that calls run on your test suite.
This package provides wrappers for the normal gren-lang/test
functions to define your test suite.
Plus additional functions like await that let you resolve tasks.
Basic test
A basic test that awaits a task:
import Expect
import Node
import Test.Runner.Effectful exposing (run, test, await)
import Time
main : Effectful.Program a
main = 
    Node.defineSimpleProgram <| \env ->
        run env <|
            await "the current time" Time.now <| \now ->
                test "is not Jan 1, 1970" <| \_ ->
                    Expect.notEqual (Time.millisToPosix 0) now
Errors
If you try to await a task that fails, the test run will fail.
If you want to explicitly test the failure condition, use awaitError:
import Expect
import Test.Runner.Effectful exposing (run, test, awaitError)
main : Effectful.Program a
main = 
    Node.defineSimpleProgram <| \env ->
        run env <|
            awaitError "expected failure" (Task.fail "oopsy") <| \error ->
                test "is an oopsy" <| \_ ->
                    Expect.equal "oopsy" error
Nested awaits
You can nest awaits as deep as you need:
run env <|
    await "task a" (Task.succeed "a") <| \a ->
    await "task b" (Task.succeed "b") <| \b ->
    awaitError "failed task" (Task.fail "failure") <| \error ->
        test "nested tasks" <| \_ ->
            Expect.equalArrays
                [ "a", "b", "error" ]
                [ a, b, error ]
Subsystems
Because your runner is a normal gren node program, you have access to subsystem permissions if you need them:
import Bytes
import Expect
import FileSystem
import Test.Runner.Effectful exposing (run, describe, test, await)
main : Effectful.Program a
main = 
    Node.defineSimpleProgram <| \env ->
        Init.await FileSystem.initialize <| \fsPerm->
        Init.await ChildProcess.initialize <| \processPerm ->
            run env <|
                describe "my effectful tests"
                    [ await "reading test.txt" (readTestFile fsPerm) <| \contents ->
                        test "resolves to contents of file" <| \_ ->
                            Expect.equal (Just "some text\n") content
                    , await "cat test.txt" (catTestFile processPerm) <| \contents ->
                        test "also resolves to contents of file" <| \_ ->
                            Expect.equal (Just "some text\n") content
                    ]
readTestFile : FileSystem.Permission -> Task FileSystem.Error (Maybe String)
readTestFile permission =
    "./test.txt"
        |> Path.fromPosixString
        |> FileSystem.readFile permission
        |> Task.map Bytes.toString
catTestFile : ChildProcess.Permission -> Task ChildProcess.FailedRun (Maybe String)
catTestFile permission =
    ChildProcess.run permission "cat" [ "test.txt" ] ChildProcess.defaultRunOptions
        |> Task.map .stdout
        |> Task.map Bytes.toString
Contact
Found a problem? Need some help? You can open an issue, find me on mastodon, or on the Gren Discord.