generated from TBD54566975/tbd-project-template
-
Notifications
You must be signed in to change notification settings - Fork 8
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
closes #1679 --------- Co-authored-by: Alec Thomas <[email protected]>
- Loading branch information
1 parent
6373a2e
commit 1eb6d84
Showing
1 changed file
with
160 additions
and
0 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,160 @@ | ||
+++ | ||
title = "Unit Tests" | ||
description = "Build unit tests for your modules" | ||
date = 2024-06-13T08:20:00+00:00 | ||
updated = 2024-06-13T08:20:00+00:00 | ||
draft = false | ||
weight = 110 | ||
sort_by = "weight" | ||
template = "docs/page.html" | ||
|
||
[extra] | ||
toc = true | ||
top = false | ||
+++ | ||
|
||
## Create a context | ||
|
||
When writing a unit test, first create a context: | ||
```go | ||
func ExampleTest(t *testing.Test) { | ||
ctx := ftltest.Context( | ||
// options go here | ||
) | ||
} | ||
``` | ||
|
||
FTL will help isolate what you want to test by restricting access to FTL features by default. You can expand what is available to test by adding options to `ftltest.Context(...)`. | ||
|
||
In this default set up, FTL does the following: | ||
- prevents access to `ftl.ConfigValue` and `ftl.SecretValue` ([See options](#project-files-configs-and-secrets)) | ||
- prevents access to `ftl.Database` ([See options](#databases)) | ||
- prevents access to `ftl.MapHandle` ([See options](#maps)) | ||
- prevents calls via `ftl.Call(...)` ([See options](#calls)) | ||
- disables all subscribers ([See options](#pubsub)) | ||
|
||
## Customization | ||
### Project files, configs and secrets | ||
|
||
To enable configs and secrets from the default project file: | ||
```go | ||
ctx := ftltest.Context( | ||
ftltest.WithDefaultProjectFile(), | ||
) | ||
``` | ||
|
||
Or you can specify a specific project file: | ||
```go | ||
ctx := ftltest.Context( | ||
ftltest.WithProjectFile(path), | ||
) | ||
``` | ||
|
||
You can also override specific config and secret values: | ||
```go | ||
ctx := ftltest.Context( | ||
ftltest.WithDefaultProjectFile(), | ||
|
||
ftltest.WithConfig(endpoint, "test"), | ||
ftltest.WithSecret(secret, "..."), | ||
) | ||
``` | ||
|
||
### Databases | ||
By default, calling `Get(ctx)` on a database panics. | ||
|
||
To enable database access in a test, you must first [provide a DSN via a project file](#project-files-configs-and-secrets). You can then set up a test database: | ||
```go | ||
ctx := ftltest.Context( | ||
ftltest.WithDefaultProjectFile(), | ||
ftltest.WithDatabase(db), | ||
) | ||
``` | ||
This will: | ||
- Take the provided DSN and appends `_test` to the database name. Eg: `accounts` becomes `accounts_test` | ||
- Wipe all tables in the database so each test run happens on a clean database | ||
|
||
|
||
### Maps | ||
By default, calling `Get(ctx)` on a map handle will panic. | ||
|
||
You can inject a fake via a map: | ||
```go | ||
ctx := ftltest.Context( | ||
ftltest.WhenMap(exampleMap, func(ctx context.Context) (string, error) { | ||
return "Test Value" | ||
}), | ||
) | ||
``` | ||
|
||
You can also allow the use of all maps: | ||
```go | ||
ctx := ftltest.Context( | ||
ftltest.WithMapsAllowed(), | ||
) | ||
``` | ||
|
||
### Calls | ||
By default, `ftl.Call(...)` will fail. | ||
|
||
You can inject fakes for verbs: | ||
```go | ||
ctx := ftltest.Context( | ||
ftltest.WhenVerb(ExampleVerb, func(ctx context.Context, req Request) (Response, error) { | ||
return Response{Result: "Lorem Ipsum"}, nil | ||
}), | ||
) | ||
``` | ||
If there is no request or response parameters, you can use `WhenSource(...)`, `WhenSink(...)`, or `WhenEmpty(...)`. | ||
|
||
To enable all calls within a module: | ||
```go | ||
ctx := ftltest.Context( | ||
ftltest.WithCallsAllowedWithinModule(), | ||
) | ||
``` | ||
|
||
### PubSub | ||
By default, all subscribers are disabled. | ||
To enable a subscriber: | ||
```go | ||
ctx := ftltest.Context( | ||
ftltest.WithSubscriber(paymentsSubscription, ProcessPayment), | ||
) | ||
``` | ||
|
||
Or you can inject a fake subscriber: | ||
```go | ||
ctx := ftltest.Context( | ||
ftltest.WithSubscriber(paymentsSubscription, func (ctx context.Context, in PaymentEvent) error { | ||
return fmt.Errorf("failed payment: %v", in) | ||
}), | ||
) | ||
``` | ||
|
||
Due to the asynchronous nature of pubsub, your test should wait for subscriptions to consume the published events: | ||
```go | ||
topic.Publish(ctx, Event{Name: "Test"}) | ||
|
||
ftltest.WaitForSubscriptionsToComplete(ctx) | ||
// Event will have been consumed by now | ||
``` | ||
|
||
You can check what events were published to a topic: | ||
```go | ||
events := ftltest.EventsForTopic(ctx, topic) | ||
``` | ||
|
||
You can check what events were consumed by a subscription, and whether a subscriber returned an error: | ||
```go | ||
results := ftltest.ResultsForSubscription(ctx, subscription) | ||
``` | ||
|
||
If all you wanted to check was whether a subscriber returned an error, this function is simpler: | ||
```go | ||
errs := ftltest.ErrorsForSubscription(ctx, subscription) | ||
``` | ||
|
||
PubSub also has these different behaviours while testing: | ||
- Publishing to topics in other modules is allowed | ||
- If a subscriber returns an error, no retries will occur regardless of retry policy. |