AWS SDK for Go v2 で EMF
はじめに
AWS では、EMF という CloudWatch ログに決まったフォーマットの JSONをパブリッシュするとそれをメトリクスとして扱ってくれる機能がある。 悪名高い AWS SDK for Go v2 で EMF 形式で送信する方法のメモである。
EMF ヘッダを設定する
ドキュメントにある通り、次のヘッダーを付与する必要がある。
x-amzn-logs-format: json/emf
次のように、PutLogEvents の 第 3 引数の optFns で設定できる。
import (
"context"
"github.com/aws/aws-sdk-go-v2/config"
"github.com/aws/aws-sdk-go-v2/service/cloudwatchlogs"
smithyhttp "github.com/aws/smithy-go/transport/http"
)
func newCWLClient(ctx context.Context) (*cloudwatchlogs.Client, error) {
cfg, err := config.LoadDefaultConfig(context.Background())
if err != nil {
return nil, err
}
client := cloudwatchlogs.NewFromConfig(cfg)
return client, nil
}
func publishMetric(ctx context.Context, client *cloudwatchlogs.Client, input *cloudwatchlogs.PutLogEventsInput) error {
_, err := client.PutLogEvents(ctx, input, func(o *cloudwatchlogs.Options) {
o.APIOptions = append(o.APIOptions, smithyhttp.SetHeaderValue("x-amzn-logs-format", "json/emf"))
})
return err
}
JSON を構築する
使ったことはないが、aws-embedded-metrics-golangを使うとチェインメソッドで構築できそう。
Lambda では、デフォルトで x-amzn-logs-format: json/emf
が設定されているとのことで、aws-embedded-metrics-golang を使えば、io.Writer に標準出力・エラー(os.Stdout/os.Stderr)を指定すれば簡単に EMF として扱える。
Docker(moby) の v23.0.0 では、awslogs ドライバーで emf を指定できるようになったので、Lambda 同様に標準出力・エラーに吐き出せば簡単に扱えるようになるらしい。ただし、Fargate では Docker を使っていないため、別途 shim logger 側の対応が必要そう。
上記外では、前述の SDK でヘッダーの設定が必要となり、EMF の構造体を定義し json.Marshal したほうがいいかもしれない。