コンテンツにスキップ

クライアントパッケージ@分散トレース

はじめに

本サイトにつきまして、以下をご認識のほど宜しくお願いいたします。


01. PHP用のクライアントパッケージ

セットアップ

▼ インストール (手動の場合)

採用しているミドルウェアごとに、インストール方法が異なる。

サーバーを冗長化している場合、全てのサーバーに共通した設定のエージェントを組み込めるという点で、IaCツールを使用した方が良い。

# GitHubのバイナリファイルのリリースページから、テキストのURLを取得する。
$ curl -L https://github.com/DataDog/dd-trace-php/releases/download/0.63.0/datadog-php-tracer_0.63.0_amd64.deb

# クライアントパッケージをインストールをする。
$ dpkg -i datadog-php-tracer_0.69_amd64.deb

# 残骸ファイルを削除する。
$ rm datadog-php-tracer.deb

また、PHP-FPMに環境変数を渡せるように、wwwプールに関する設定ファイルを配置し、PHP-FPMを再起動する。

# /etc/php-fpm.d/dd-trace.confファイル
[www]
env[DD_SERVICE] = 'foo'
env[DD_SERVICE_MAPPING] = 'guzzle:foo-guzzle,pdo:foo-pdo'
env[DD_ENV] = 'prd'
env[DD_VERSION] = '<バージョンタグ>'

▼ インストール (Ansibleの場合)

使用しているミドルウェアごとに、インストール方法が異なる。

- tasks:
    - name: Install dd-trace-php
      ansible.builtin.shell: |
        curl -Lo https://github.com/DataDog/dd-trace-php/releases/download/${DD_TRACE_VERSION}/datadog-php-tracer_${DD_TRACE_VERSION}_amd64.deb
        dpkg -i datadog-php-tracer_0.69_amd64.deb
        rm datadog-php-tracer.deb
      environment:
        DD_TRACE_VERSION: 0.63.0
    - name: Upload dd-trace.conf
      ansible.builtin.template: src=dd-trace.conf dest=/etc/php-fpm.d/dd-trace.conf
      notify: restart php-fpm

▼ インストール (コンテナの場合)

アプリコンテナのDockerfileにて、PHP用のクライアントパッケージをインストールする。

また、コンテナの環境変数として、DD_SERVICEDD_ENVDD_VERSIONを渡す。

ENV DD_TRACE_VERSION=0.63.0

# GitHubのバイナリファイルのリリースページから、テキストのURLを取得する。
RUN curl -Lo https://github.com/DataDog/dd-trace-php/releases/download/${DD_TRACE_VERSION}/datadog-php-tracer_${DD_TRACE_VERSION}_amd64.deb \
  `# クライアントパッケージをインストールする。` \
  && dpkg -i datadog-php-tracer.deb \
  `# 残骸ファイルを削除する。` \
  && rm datadog-php-tracer.deb

▼ インストールの動作確認

パッケージが正しく読み込まれているか否かは、php --ri=ddtraceコマンドまたはphpinfoメソッドの結果から確認できる。

# 成功の場合
$ php --ri=ddtrace

ddtrace


Datadog PHP tracer extension
For help, check out the documentation at https://docs.datadoghq.com/tracing/languages/php/
(c) Datadog 2020

...
# 失敗の場合
$ php --ri=ddtrace
Extension 'ddtrace' not present.

▼ パラメーターの動作確認

パラメーターがパッケージに渡されたか否かは、DATADOG TRACER CONFIGURATIONの項目で確認できる。

$ php --ri=ddtrace

Datadog tracing support => enabled
Version => 0.57.0
DATADOG TRACER CONFIGURATION => { ... } # ここに設定のJSONが得られる

# 得られたJSONを整形している
{
    "date": "2021-00-00T09:00:00Z",
    "os_name": "Linux ***** 5.10.25-linuxkit #1 SMP Tue Mar 23 09:27:39 UTC 2021 x86_64",
    "os_version": "5.10.25-linuxkit",
    "version": "0.64.1",
    "lang": "php",
    "lang_version": "8.0.8",
    "env": null,
    "enabled": "true",
    "service": null,
    "enabled_cli": "false",
    "agent_url": "http://localhost:8126", # datadogコンテナのアドレスポート
    "debug": "false",
    "analytics_enabled": "false",
    "sample_rate": 1.000000,
    "sampling_rules": null,
    "tags": {},
    "service_mapping": {},
    "distributed_tracing_enabled": "true",
    "priority_sampling_enabled": "true",
    "dd_version": null,
    "architecture": "x86_64",
    "sapi": "cli",
    "datadog.trace.request_init_hook": "/opt/datadog-php/dd-trace-sources/bridge/dd_wrap_autoloader.php",
    "open_basedir_configured": "false",
    "uri_fragment_regex": null,
    "uri_mapping_incoming": null,
    "uri_mapping_outgoing": null,
    "auto_flush_enabled": "false",
    "generate_root_span": "true",
    "http_client_split_by_domain": "false",
    "measure_compile_time": "true",
    "report_hostname_on_root_span": "false",
    "traced_internal_functions": null,
    "auto_prepend_file_configured": "false",
    "integrations_disabled": "default",
    "enabled_from_env": "true",
    "opcache.file_cache": null,
    "agent_error": "Failed to connect to localhost port 8126: Connection refused", # エラーメッセージ
    "DDTRACE_REQUEST_INIT_HOOK": "'DDTRACE_REQUEST_INIT_HOOK=/opt/datadog-php/dd-trace-sources/bridge/dd_wrap_autoloader.php' is deprecated, use DD_TRACE_REQUEST_INIT_HOOK instead."
}

▼ 受信ログの確認

datadogコンテナが分散トレースを受信している場合は、受信できていることを表すログを確認できる。

2022-01-01 12:00:00 UTC | TRACE | INFO | (pkg/trace/info/stats.go:111 in LogStats) | [lang:php lang_version:8.0.8 interpreter:fpm-fcgi tracer_version:0.64.1 endpoint_version:v0.4] -> traces received: 7, traces filtered: 0, traces amount: 25546 bytes, events extracted: 0, events sampled: 0


02. JavaScript用のクライアントパッケージ

セットアップ

▼ インストール

TypeScriptやモジュールバンドルを採用している場合、パッケージの読み出し処理が巻き上げられ、意図しない読み出しの順番になってしまうことがある。

対策として、dd-traceパッケージのinitメソッドの実行を別ファイルに分割し、これをエントリーポイント (nuxt.config.jsファイル) の先頭で読み込むようにする。

また、フレームワークよりも先に読み込むことになるため、.envファイル参照を使用できない。

そこで、環境変数はインフラ側で設定する。

// datadogTracer.tsファイル
import tracer from "dd-trace";

tracer.init({
  // フレームワークの.envファイル参照を使用できない
  env: DD_ENV,
  service: DD_SERVICE + "-ssr",
  version: DD_VERSION,

  // 検証時のオプション
  debug: "true",
  startupLogs: "true",
});

export default datadogTracer;
// nuxt.config.tsファイル
// 先頭で読み込む
import './datadogTracer'
import { Configuration } from '@nuxt/types'

...

▼ 起動ログの確認

パッケージの起動ログは、initメソッドのstartupLogsオプションを有効化すると確認できる。

DATADOG TRACER CONFIGURATION -
{
    "date": "2022-01-02T00:00:00.541Z",
    "os_name": "Darwin",
    "os_version": "20.6.0",
    "architecture": "arm64",
    "version": "2.0.1",
    "lang": "nodejs",
    "lang_version": "14.18.2",
    "env": "prd",
    "service": "foo",
    "agent_url": "http://127.0.0.1:8126",
    "agent_error": "Network error trying to reach the agent: socket hang up",
    "debug": "false",
    "sample_rate": 1,
    "sampling_rules": [],
    "tags": {
        "service": "foo",
        "env": "prd",
        "version": "<バージョンタグ>",
        "runtime-id": "*****"
    },
    "dd_version": "<バージョンタグ>",
    "log_injection_enabled": "false",
    "runtime_metrics_enabled": "false",
    "profiling_enabled": "false",
    "integrations_loaded": [
        "connect@3.7.0",
        "fs",
        "http",
        "https"
    ],
    "appsec_enabled": "false"
}

WARN  DATADOG TRACER DIAGNOSTIC - Agent Error: Network error trying to reach the agent: socket hang up


環境変数

初期化時に環境変数を設定できる。

APMのマイクロサービスのタグ名に反映される。


03. Go用のクライアントパッケージ

HTTPを使用する場合

▼ 最ダウンストリーム側マイクロサービス

最ダウンストリーム側マイクロサービスでは、親スパンを作成する。

package main

import (
    "log"
    "net/http"

    "gopkg.in/DataDog/dd-trace-go.v1/ddtrace/tracer"
)

func InitTracerProvider(w http.ResponseWriter, req *http.Request) {

    // Tracerを作成する
    // Tracer名はパッケージ名が推奨である
    // @see https://pkg.go.dev/go.opentelemetry.io/otel/trace#TracerProvider
    var tracer = otel.Tracer("計装パッケージ名")

    log.Print("Info: Trace provider is initializing ...")

    // 親スパンを作成する。
    span, ctx := tracer.StartSpanFromContext(
        req.Context(),
        "post.process",
    )

    defer span.Finish()

    req, err := http.NewRequest(
        "GET",
        "https://example.com",
        nil,
    )

    req = req.WithContext(ctx)

    err = tracer.Inject(
        // トレースコンテキストを持つ既存コンテキストを設定する
        span.Context(),
        // Carrierとして使用するメタデータを設定し、トレースコンテキストを注入する
        tracer.HTTPHeadersCarrier(req.Header),
    )

    if err != nil {
        log.Printf("Failed to inject context: %s", err)
    }

    http.DefaultClient.Do(req)
}

▼ アップストリーム側マイクロサービス

アップストリーム側マイクロサービスでは、受信した通信からトレースコンテキストを取得する。

また、子スパンを作成する。

package main

import (
    "log"
    "net/http"

    "gopkg.in/DataDog/dd-trace-go.v1/ddtrace/tracer"
)

func InitTracerProvider(w http.ResponseWriter, r *http.Request) {

    log.Print("Info: Trace provider is initializing ...")

    // Carrierからトレースコンテキストを取得する。
    tracectx, err := tracer.Extract(tracer.HTTPHeadersCarrier(r.Header))

    if err != nil {
        log.Printf("Failed to extract context: %s", err)
    }

    log.Print("Info: Tracer provider initialize successfully")

    // Tracerを作成する
    // Tracer名はパッケージ名が推奨である
    // @see https://pkg.go.dev/go.opentelemetry.io/otel/trace#TracerProvider
    var tracer = otel.Tracer("計装パッケージ名")

    // 子スパンを作成する。
    span := tracer.StartSpan(
        "post.filter",
        tracer.ChildOf(tracectx),
    )

    defer span.Finish()
}


gRPCを使用する場合

▼ Interceptorの実行

事前処理としてスパンの作成などを行うInterceptorを使用する。

otelgrpcパッケージのラッパーであるdd-trace-go.v1パッケージが持っている。

各関数でスパンの作成を実行する必要がなくなる。

▼ gRPCサーバー側

package main

import (
    "fmt"
    "log"
    "net"

    "google.golang.org/grpc"
    "gopkg.in/DataDog/dd-trace-go.v1/ddtrace/tracer"

    grpctracer "gopkg.in/DataDog/dd-trace-go.v1/contrib/google.golang.org/grpc"
)

func main() {

    // Tracerを作成する
    // Tracer名はパッケージ名が推奨である
    // @see https://pkg.go.dev/go.opentelemetry.io/otel/trace#TracerProvider
    var tracer = otel.Tracer("計装パッケージ名")

    // パッケージをセットアップする。
    tracer.Start(tracer.WithEnv("prd"))

    defer tracer.Stop()

    // gRPCサーバーを作成する。
    grpcServer := grpc.NewServer(
        // 単項RPCのインターセプター処理
        grpc.UnaryInterceptor(
            grpctrace.UnaryServerInterceptor(datadogAPMServiceName, tracer.DefaultTracer),
        ),
        // ストリーミングRPCのインターセプター処理
        grpc.StreamInterceptor(
            grpctrace.StreamServerInterceptor(datadogAPMServiceName, tracer.DefaultTracer),
        ),
    )

    ... // pb.goファイルに関する実装は省略している。

    listenPort, _ := net.Listen("tcp", fmt.Sprintf(":%d", 9000))

    if err != nil {
        log.Printf("Failed to listen: %v", err)
    }

    // gRPCサーバーでリクエストを受信する。
    if err := grpcServer.Serve(listenPort); err != nil {
        log.Printf("Failed to serve: %s", err)
    }

    if err != nil {
        log.Printf("Failed to serve: %v", err)
    }
}

▼ gRPCクライアント側

gRPCクライアント側では、gRPCサーバーとのコネクションを作成する必要がある。

package main

import (
    "log"

    "google.golang.org/grpc"

    grpctrace "gopkg.in/DataDog/dd-trace-go.v1/contrib/google.golang.org/grpc"
)

func main() {

    ...

    // gRPCサーバーとのコネクションを作成する
    conn, err := grpc.DialContext(
        ctx,
        ":9000",
        grpc.WithInsecure(),
        grpc.WithBlock(),
        grpc.WithChainUnaryInterceptor(grpctrace.UnaryClientInterceptor(datadogAPMServiceName, tracer.DefaultTracer)),
        grpc.WithChainStreamInterceptor(grpctrace.UnaryStreamInterceptor(datadogAPMServiceName, tracer.DefaultTracer)),
    )

    if err != nil {
        log.Printf("Failed to create gRPC connection: %v", err)
    }

    defer conn.Close()

    ... // pb.goファイルに関する実装は省略している。
}