トレース - PHPにOpenTelemetryを導入する

MackerelはOpenTelemetryの仕組み (計装)を利用してデータを取得しています。

このページではPHPのデータをMackerelに送信する方法を解説します。

PHP向けOpenTelemetry

OpenTelemetryにはPHP用のSDKが用意されています。

このSDKに加えて、LaravelやPDO用のSDKを使用すると、既存の実装を変更せずに計装することができます。

opentelemetry.io

Collectorを使用するべきか

データをMackerelに送信する際に、SDKから直接送信するだけではなく、Collectorを使うこともできます。

しかし、PHPの多くのランタイムは全ての処理が終わるまでリクエストをブロッキングするため、直接Mackerelにデータを送信すると、Mackerelに送信している時間の分だけリクエストも遅くなってしまいます。

そのため、Collectorを使用して、ユーザーへの返答をできるだけ速くすることを推奨しています。

ただし、fastcgifastcgi_finish_request() を使用すると、Mackerelへの送信を待つことなくユーザーにレスポンスを返すことができます。その場合は、直接Mackerelにデータを送信しても問題になることはないでしょう。

Collectorを使用する方法は、以下のページを参照してください。

mackerel.io

導入方法

PHPには複数のWebフレームワークが存在しますが、このページでは Laravel への導入方法を説明します。

他のフレームワークを使っている場合もほぼ同じ方法で計装することができます。

以下のステップでMackerelトレーシング機能を導入できます。

  1. 拡張ライブラリ (PECL) の追加
  2. パッケージの追加
  3. 環境変数の設定
  4. 独自の計装 (任意)

1. 拡張ライブラリ (PECL) の追加

PECL を使用して OpenTelemetry 用の拡張ライブラリをインストールします。

pecl install opentelemetry

その後、インストールした拡張を php.ini に追加します。

[opentelemetry]
extension=opentelemetry.so

2. パッケージの追加

Laravel や PDO用のパッケージを追加します。

composer require open-telemetry/api \
  open-telemetry/exporter-otlp \
  open-telemetry/opentelemetry-auto-http-async \
  open-telemetry/opentelemetry-auto-io \
  open-telemetry/opentelemetry-auto-laravel \
  open-telemetry/opentelemetry-auto-pdo \
  open-telemetry/opentelemetry-auto-psr3 \
  open-telemetry/opentelemetry-auto-psr15 \
  open-telemetry/opentelemetry-auto-psr18 \
  open-telemetry/sdk

⚠️ 他のフレームワークを使用している場合

このページではLaravelを例にするため opentelemetry-auto-laravel を使用しています。SlimやSymfonyなど別のフレームワークを使用している場合、それぞれにあったパッケージを、以下のページから検索することができます。

opentelemetry.io

3. 環境変数の設定

パッケージのインストール後、Mackerelにデータを送信するための環境変数を設定します。以下はプロジェクト内の.envファイルを利用して設定する例です。

OTEL_PHP_AUTOLOAD_ENABLED=true
OTEL_SERVICE_NAME=your-service-name
OTEL_TRACES_EXPORTER=otlp
# デバッグ時にはconsoleと設定すると便利
# OTEL_TRACES_EXPORTER=console
OTEL_EXPORTER_OTLP_PROTOCOL=http/protobuf
OTEL_EXPORTER_OTLP_TRACES_ENDPOINT=https://otlp-vaxila.mackerelio.com/v1/traces
OTEL_EXPORTER_OTLP_TRACES_HEADERS="Accept=*/*,Mackerel-Api-Key=${MACKEREL_APIKEY}"
OTEL_METRICS_EXPORTER=none
OTEL_LOGS_EXPORTER=none
OTEL_PROPAGATORS=baggage,tracecontext

これらの環境変数を、オートロードが開始する前に設定してください。

OTEL_SERVICE_NAMEOTEL_EXPORTER_OTLP_TRACES_ENDPOINT の値は各環境にあわせた値へ変更してください。 また、OTEL_TRACES_EXPORTERconsole に設定すると、データがログに出力されるようになるので、送信が動かない時のデバッグに便利です。

以上で、インストールが完了し、Collectorを通して、Mackerelにデータが送られるようになりました。

4. 独自の計装 (任意)

独自のSpanを追加することで、任意の範囲を計装することができます。

計装によって、変数の値や処理時間を記録することができるようになります。

具体的には、下のように startSpan を使用すると計装が追加できます。

use OpenTelemetry\API\Globals;

function awesomeFunction(string $arg) {
  $tracer = Globals::tracerProvider()->getTracer("...");
  $childSpan = $tracer->spanBuilder('Awesome span name')->startSpan();
  $childSpan->setAttribute('arg', $arg);
  $scope = $childSpan->activate();

  try {
    // 計装対象の処理
  } finally {
    $childSpan->end();
    $scope->detach();
  }
}

計装の方法は他にも用意されています。詳細はOpenTelemetryのドキュメントを参照してください。

opentelemetry.io