When using the net/context
package, one should
respect context expiration when performing requests to backing services. Typically, a
context.Context
is created for every incoming request and all calls to other services that happen
during this request have to be cancelled when the request's context expires.
The ctxaws
package includes a helper function ctxaws.InContext
that performs AWS API requests
while respecting a context's cancellation.
Support for the context pattern in the SDK directly was discussed but not implemented.
In Go projects using the AWS SDK, requests or most typically performed like this:
func countItems() (int, error) {
out, err := client.Scan(&dynamodb.ScanInput{
TableName: aws.String("my-table"),
})
if err != nil {
return 0, err
}
return len(out.Items), nil
}
DynamoDB is just used as an example here.
The Problem with such SDK operations is that they lead to HTTP requests that cannot be cancelled and are retried for a (configurable) number of times until they succeed. In most cases, these retries are desired behavior, however it cannot be controlled how long the whole operation takes.
In order to make the same API call while respecting a context's cancellation, one can use the
ctxaws
package like this:
func countItems(ctx context.Context) (int, error) {
req, out := client.ScanRequest(&dynamodb.ScanInput{
TableName: aws.String("my-table"),
})
if err := ctxaws.InContext(ctx, req); err != nil {
return 0, err
}
return len(out.Items), nil
}
By using the special *Request
methods of the AWS service clients, we can delegate the processing
of the request to ctxhttp
which makes sure that the context cancellation is respected while
performing the request.
Besides cancelling requests that exceed the context's deadline, ctxaws
also uses a custom retry
mechanism that stops retrying as soon as the context expires. This way, the context ist honored
during HTTP requests and during backoff periods.
When using SDK operations that return a list of resources, it is almost always a good idea to use
the SDKs *Pages
functions. With this in mind, the example from above becomes slightly more complex
and could be implemented like this:
func countItems() (int, error) {
count := 0
err := client.ScanPages(&dynamodb.ScanInput{
TableName: aws.String("my-table"),
}, func(out *dynamodb.ScanOutput, last bool) bool {
count += len(out.Items)
return !last
})
if err != nil {
return 0, err
}
return count, nil
}
With a context, one can use PaginageInContext
which works very similar to the *Pages
methods. The only difference is the type of the passed closure's first argument. Instead of a
reqular *Output
type, the closure takes an interface{}
which can safely be cased to the
respective *Output
type (*dynamodb.ScanOutput
in the example):
func countItems(ctx context.Context) (int, error) {
count := 0
req, _ := client.ScanRequest(&dynamodb.ScanInput{
TableName: aws.String("my-table"),
})
err := ctxaws.PaginateInContext(ctx, req, func(out interface{}, last bool) bool {
count += len(out.(*dynamodb.ScanOutput).Items)
return !last
})
if err != nil {
return 0, err
}
return count, nil
}
There is a cmd/main
package with an example program that can be used for testing. Please note that
it requires a DynamoDB table named test-table
that needs to be created before running it. When
run it uses the AWS credentials exported on the command line to connect to DynamoDB.
$ go run cmd/main/main.go
Besides that, there are a couple of tests that are run automatically by Travis. To run those tests locally, execute
$ go test