5
2

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?

More than 5 years have passed since last update.

Go4Advent Calendar 2019

Day 20

opentelemetry-go移行を試みた話

Last updated at Posted at 2019-12-20

1年ほど前からOpenCensusとStackdriverという構成でシステムを運用しています。OpenTelemetryが発表されたとき、一度検証してみようと思ったのですが、当時(2019年6月くらい)はまだSpecが活発に議論されている段階でスルーしていました。しかし11月のKubeCon NA 2019でも大々的に取り上げられていたので、このタイミングで一度移行出来るか試してみた話です。

OpenCensus/OpenTelemetryともにトレースとメトリクスを出力することが出来ますが、ここではトレースに絞って書きます

OpenCensusでどんな感じでやってるかについては過去いくつか資料があるのでもし興味があればそちらをご覧ください :pray:

現状のシステム構成

実際はより多くのシステム間で使用していますが、最低限検証のためにgRPC propagationとStackdriver exporter、さらにTracingにstatusを付ける状態を再現出来るかどうかを試しています。

image.png

またこの状態でOpenCensus実装のトレーシングはこのようなイメージです。

Screen Shot 2019-12-20 at 22.21.41.png

OpenTelemetryを入れてみる

では実際にOpenCensusからOpenTelemetryに載せ替えたときにどういう変更を行ったかを見ていきましょう。

トレース(スパン)を開始、出力する

StartSpanとspan.Endですね。ここは使用するパッケージが変わる程度でdiffはあまり多くありません。

src/gclient/main.go
- ctx, span := trace.StartSpan(ctx, "gclient/greeting")
+ tr := global.TraceProvider().Tracer("example/grpc")
+ ctx, span := tr.Start(ctx, "gclient/greeting")

スパンにアトリビュートやらを付ける

アトリビュート、ステータスなどを見ていきます。
ここではOpenCensus版でstatusだけにしているところを、OpenTelemetryの方ではstatusがcodeだけを取るようになっているのでattributesも合わせて設定するようにしています。

src/gclient/main.go
- span.SetStatus(trace.Status{
-         Code:    1,
-         Message: err.Error(),
- })
+ span.SetStatus(1)
+ span.SetAttributes(key.String("message", err.Error()))

Exporterの設定

Stackdriver exporterの設定です。opentelemetry-goに実装が用意されていて助かりました。
ここも使用するパッケージを変更する程度で大丈夫そうです。

src/gclient/main.go
 func initStackdriver() error {
-       exporter, err := stackdriver.NewExporter(stackdriver.Options{})
+       exporter, err := stackdriver.NewExporter()
        if err != nil {
                return err
        }
 
-       trace.RegisterExporter(exporter)
-       trace.ApplyConfig(trace.Config{
-               DefaultSampler: trace.AlwaysSample(),
-       })
+       tp, err := sdktrace.NewProvider(
+               sdktrace.WithConfig(sdktrace.Config{
+                       DefaultSampler: sdktrace.AlwaysSample()},
+               ),
+               sdktrace.WithSyncer(exporter),
+       )
+       if err != nil {
+               return err
+       }
+       global.SetTraceProvider(tp)
        return nil
 }

gRPC client/server にトレースを仕込む

OpenCensusではgRPC pluginが担保してくれていた部分です。こちらはまだ(?)pluginとしての実装はないので自前でInterceptorを書く必要があります。

src/gclient/main.go
   ctx,
   fmt.Sprintf("%s", os.Getenv("ADDR")),
   grpc.WithInsecure(),
-  grpc.WithStatsHandler(&ocgrpc.ClientHandler{}),
+  grpc.WithUnaryInterceptor(UnaryClientInterceptor),

...

+
+func UnaryClientInterceptor(ctx context.Context, method string, req, reply interface{}, cc *grpc.ClientConn, invoker grpc.UnaryInvoker, opts ...grpc.CallOption) error {
+       requestMetadata, _ := metadata.FromOutgoingContext(ctx)
+       metadataCopy := requestMetadata.Copy()
+
+       err := trace.CurrentSpan(ctx).Tracer().WithSpan(ctx, "send",
+               func(ctx context.Context) error {
+                       grpctrace.Inject(ctx, &metadataCopy)
+                       ctx = metadata.NewOutgoingContext(ctx, metadataCopy)
+
+                       err := invoker(ctx, method, req, reply, cc, opts...)
+                       return err
+               })
+       return err
+}
src/gserver/main.go
  srv := grpc.NewServer(
-   grpc.StatsHandler(&ocgrpc.ServerHandler{}),
+   grpc.UnaryInterceptor(UnaryServerInterceptor),


+}
+
+func UnaryServerInterceptor(ctx context.Context, req interface{}, info *grpc.UnaryServerInfo, handler grpc.UnaryHandler) (resp interface{}, err error) {
+       requestMetadata, _ := metadata.FromIncomingContext(ctx)
+       metadataCopy := requestMetadata.Copy()
+
+       entries, spanCtx := grpctrace.Extract(ctx, &metadataCopy)
+       ctx = distributedcontext.WithMap(ctx, distributedcontext.NewMap(distributedcontext.MapUpdate{
+               MultiKV: entries,
+       }))
+
+       tr := global.TraceProvider().Tracer("recv")
+       ctx, span := tr.Start(
+               ctx,
+               info.FullMethod,
+               trace.ChildOf(spanCtx),
+               trace.WithSpanKind(trace.SpanKindServer),
+       )
+       defer span.End()
+
+       return handler(ctx, req)
 }

Stackdriver Trace でトレースを見てみる

ここまででOpenTelemetry実装のトレースが実際にexportされていることが確認出来ました。

Screen Shot 2019-12-20 at 22.22.01.png

おわりに

OpenCensusで実現していた最低限の実装をOpenTelemetryに移行出来るか試してみました。
結果的にまだAlphaなので未実装の部分が多いですが、着実に実装が進んできているという感触を得ました。

今回は言及しなかったですが、この他にも

  • Stackdriver exporterのmetricsは未実装
  • OpenCensus用のbridge APIは未実装
    • OpenTracing用のbridge APIは現時点で実装がある
  • GCPクライアントライブラリに組み込まれているのがOpenCensusなので、OpenTelemetryのspanを入れたcontextを渡しても良い感じにトレースが出ない

といったところがあり、特に最後のライブラリの件は自分でラッパー書けば良いとはいえ面倒なのでしばらくはOpenCensusのままかなぁという感じでした。
とはいえOpenTelemetryはこの界隈のデファクトになると思いますし、もしまだ分散トレーシングはじめ特に何もやってないという場合は、bridge APIも実装予定なのでOpenCensusから初めてみるのも良いのではないでしょうか。

今回使ったコードは https://github.com/takashabe/ot-sandbox にあります。またOpenCensusからOpenTelemetryへのdiffは https://github.com/takashabe/ot-sandbox/pull/1 にまとまっています。

5
2
0

Register as a new user and use Qiita more conveniently

  1. You get articles that match your needs
  2. You can efficiently read back useful information
  3. You can use dark theme
What you can do with signing up
5
2

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?