xxxxxxxxxx
import "context"
import "github.com/aws/aws-sdk-go-v2/service/s3"
import "github.com/aws/smithy-go/middleware"
// ...
type customKey struct{}
func GetCustomKey(metadata middleware.Metadata) (v string) {
v, _ = metadata.Get(customKey{}).(string)
return v
}
func SetCustomKey(metadata *middleware.Metadata, value string) {
metadata.Set(customKey{}, value)
}
// ...
var customInitalize = middleware.InitializeMiddlewareFunc("customInitialize", func (
ctx context.Context, in middleware.InitializeInput, next middleware.InitializeHandler,
) (
out middleware.InitializeOutput, metadata middleware.Metadata, err error,
) {
out, metadata, err = next.HandleInitialize(ctx, in)
if err != nil {
return out, metadata, err
}
SetCustomKey(&metadata, "my-custom-value")
return out, metadata, nil
})
// ...
client := s3.NewFromConfig(cfg, func (options *s3.Options) {
options.APIOptions = append(options.APIOptions, func(stack *middleware.Stack) error {
return stack.Initialize.Add(customInitalize, middleware.After)
})
})
out, err := client.GetObject(context.TODO(), &s3.GetObjectInput{
// input parameters
})
if err != nil {
// handle error
}
customValue := GetCustomKey(out.ResponseMetadata)
You can pass metadata up through the stack by adding metadata key and value pairs using the [middleware.Metadata](https://pkg.go.dev/github.com/aws/smithy-go/middleware#Metadata). Each middleware step returns an output structure, metadata, and an error. Your custom middleware must return the metadata received from calling the next handler in the step. This ensures that metadata added by downstream middleware propagates to the application invoking the service operation. The resulting metadata is accessible to the invoking application by either the operation’s output shape via the `ResultMetadata` structure member.
The following examples shows how a custom middleware can add metadata that is returned as part of the operation output.
xxxxxxxxxx
import "context"
import "github.com/aws/smithy-go/middleware"
// ...
type customKey struct {}
func GetCustomKey(ctx context.Context) (v string) {
v, _ = middleware.GetStackValue(ctx, customKey{}).(string)
return v
}
func SetCustomKey(ctx context.Context, value string) context.Context {
return middleware.WithStackValue(ctx, customkey{}, value)
}
// ...
var customInitalize = middleware.InitializeMiddlewareFunc("customInitialize", func(
ctx context.Context, in middleware.InitializeInput, next middleware.InitializeHandler,
) (
out middleware.InitializeOutput, metadata middleware.Metadata, err error,
) {
ctx = SetCustomKey(ctx, "my-custom-value")
return next.HandleInitialize(ctx, in)
})
var customBuild = middleware.BuildMiddlewareFunc("customBuild", func(
ctx context.Context, in middleware.BuildInput, next middleware.BuildHandler,
) (
out middleware.BuildOutput, metadata Metadata, err error,
) {
customValue := GetCustomKey(ctx)
// use customValue
return next.HandleBuild(ctx, in)
})
In certain situations, you may find that you require two or more middleware to function in tandem by sharing information or state. You can use [context.Context](https://golang.org/pkg/context/#Context) to pass this metadata by using [middleware.WithStackValue](https://pkg.go.dev/github.com/aws/smithy-go/middleware#WithStackValue). `middleware.WithStackValue` attaches the given key-value pair to the provided context, and safely limits the scope to the currently executing stack. These stack-scoped values can be retrieved from a context using [middleware.GetStackValue](https://pkg.go.dev/github.com/aws/smithy-go/middleware#GetStackValue) and providing the key used to stored the corresponding value. Keys must be comparable, and you must define your own types as context keys to avoid collisions. The following examples shows how two middleware can use `context.Context` to pass information down the stack.