# 調用鏈追蹤 在微服務場景下,我們會拆分出來很多的服務,也就意味着一個業務請求,少則跨越 3-4 個服務,多則幾十個甚至更多,在這種架構下我們需要對某一個問題進行 Debug 的時候是極其困難的一件事情,那麼我們就需要一個調用鏈追蹤系統來幫助我們動態地展示服務調用的鏈路,以便我們可以快速地對問題點進行定位,亦可根據鏈路信息對服務進行調優。 在 `Hyperf` 裏我們提供了 [hyperf/tracer](https://github.com/hyperf/tracer) 組件來對各個跨網絡請求來進行調用的追蹤以及分析,目前根據 [OpenTracing](https://opentracing.io) 協議對接了 [Zipkin](https://zipkin.io/) 系統和 [Jaeger](https://www.jaegertracing.io/) 系統,用户也可以根據 OpenTracing 協議定製實現。 ## 安裝 ### 通過 Composer 安裝組件 ```bash composer require hyperf/tracer ``` [hyperf/tracer](https://github.com/hyperf/tracer) 組件默認安裝了 [Zipkin](https://zipkin.io/) 相關依賴。如果要使用 [Jaeger](https://www.jaegertracing.io/),還需要執行下面的命令安裝對應的依賴: ```bash composer require jonahgeorge/jaeger-client-php ``` ### 增加組件配置 如文件不存在,可執行下面的命令增加 `config/autoload/opentracing.php` 配置文件: ```bash php bin/hyperf.php vendor:publish hyperf/tracer ``` ## 使用 ### 配置 #### 配置追蹤開關 默認提供了對 `Guzzle HTTP` 調用、`Redis` 調用、`DB` 調用進行了監聽或 `AOP` 切面處理,以實現對調用鏈的傳播與追蹤,默認情況下這些追蹤不會打開,您需要通過更改 `config/autoload/opentracing.php` 配置文件內的 `enable` 項內的開關來打開對某些遠程調用的追蹤。 ```php [ // 打開或關閉對 Guzzle HTTP 調用的追蹤 'guzzle' => false, // 打開或關閉對 Redis 調用的追蹤 'redis' => false, // 打開或關閉對 DB 調用的追蹤 'db' => false, ], ]; ``` 在開始追蹤之前,我們還需要選擇所使用的 Tracer 驅動,並對 Tracer 進行配置。 #### 選擇追蹤器驅動 配置文件內的 `default` 對應的值則為使用的驅動名稱。驅動的具體配置在 `tracer` 項下定義,使用與 `key` 相同的驅動。 ```php env('TRACER_DRIVER', 'staging_zipkin'), // 這裏暫時省略其他配置 'enable' => [], 'tracer' => [ // Zipkin 配置 'staging_zipkin' => [ 'driver' => \Hyperf\Tracer\Adapter\ZipkinTracerFactory::class, ], // 另一套 Zipkin 配置 'producton_zipkin' => [ 'driver' => \Hyperf\Tracer\Adapter\ZipkinTracerFactory::class, ], // Jaeger 配置 'jaeger' => [ 'driver' => \Hyperf\Tracer\Adapter\JaegerTracerFactory::class, ], ] ]; ``` 注意,如示例配置所示,您可以配置多套 Zipkin 驅動或 Jaeger 驅動。雖然採用的底層系統一樣,但他們的具體配置可以不同。一種常見場景是我們希望測試環境 100% 採樣,但在生產環境下 1% 採樣,就可以配置兩套驅動,然後根據 `default` 項下的環境變量來選擇不同的驅動。 #### 配置 Zipkin 使用 Zipkin 時,在配置文件中的 `tracer` 項增加 Zipkin 的具體配置。 ```php env('TRACER_DRIVER', 'zipkin'), // 這裏的代碼演示不對 enable 內的配置進行展開 'enable' => [], 'tracer' => [ // Zipkin 驅動配置 'zipkin' => [ // 當前應用的配置 'app' => [ 'name' => env('APP_NAME', 'skeleton'), // 如果 ipv6 和 ipv6 為空組件會自動從 Server 中檢測 'ipv4' => '127.0.0.1', 'ipv6' => null, 'port' => 9501, ], 'driver' => \Hyperf\Tracer\Adapter\ZipkinTracerFactory::class, 'options' => [ // Zipkin 服務的 endpoint 地址 'endpoint_url' => env('ZIPKIN_ENDPOINT_URL', 'http://localhost:9411/api/v2/spans'), // 請求超時秒數 'timeout' => env('ZIPKIN_TIMEOUT', 1), ], // 採樣器,默認為所有請求的都追蹤 'sampler' => BinarySampler::createAsAlwaysSample(), ], ], ]; ``` #### 配置 Jaeger 使用 Jaeger 時,在配置文件中的 `tracer` 項增加 Jaeger 的具體配置。 ```php env('TRACER_DRIVER', 'jaeger'), // 這裏的代碼演示不對 enable 內的配置進行展開 'enable' => [], 'tracer' => [ // Jaeger 驅動配置 'jaeger' => [ 'driver' => JaegerTracerFactory::class, // 項目名稱 'name' => env('APP_NAME', 'skeleton'), 'options' => [ // 採樣器,默認為所有請求的都追蹤 'sampler' => [ 'type' => SAMPLER_TYPE_CONST, 'param' => true, ], // 上報地址 'local_agent' => [ 'reporting_host' => env('JAEGER_REPORTING_HOST', 'localhost'), 'reporting_port' => env('JAEGER_REPORTING_PORT', 5775), ], ], ], ], ]; ``` 關於 Jaeger 的更多配置可以在 [[這裏](https://github.com/jonahgeorge/jaeger-client-php)] 查看。 #### 配置 JsonRPC 追蹤開關 JsonRPC 的鏈路追蹤並不在統一配置當中,暫時還屬於 `Beta` 版本的功能。 我們只需要配置 `aspects.php`,加入以下 `Aspect` 即可開啟。 > 提示:不要忘了在對端,添加對應的 TraceMiddleware ```php [ \Hyperf\Tracer\Middleware\TraceMiddleware::class, ], ]; ``` ### 配置 Span tag 對於一些 Hyperf 自動收集追蹤信息的 Span Tag 名稱,可以通過更改 Span Tag 配置來更改對應的名稱,只需在配置文件 `config/autolaod/opentracing.php` 內增加 `tags` 配置即可,參考配置如下。如配置項存在,則以配置項的值為準,如配置項不存在,則以組件的默認值為準。 ```php return [ 'tags' => [ // HTTP 客户端 (Guzzle) 'http_client' => [ 'http.url' => 'http.url', 'http.method' => 'http.method', 'http.status_code' => 'http.status_code', ], // Redis 客户端 'redis' => [ 'arguments' => 'arguments', 'result' => 'result', ], // 數據庫客户端 (hyperf/database) 'db' => [ 'db.query' => 'db.query', 'db.statement' => 'db.statement', 'db.query_time' => 'db.query_time', ], ] ]; ``` ### 更換採樣器 默認的採樣器為所有請求都記錄調用鏈,這對性能會存在一定程度上的影響,尤其是內存的佔用,所以我們只需要在我們希望的時候才對調用鏈進行追蹤,那麼我們就需要對採樣器進行更換,更換也很簡單,以 Zipkin 為例,只需對配置項 `opentracing.zipkin.sampler` 對應的值改為您的採樣器對象實例即可,只要您的採樣器對象實現了 `Zipkin\Sampler` 接口類即可。 ### 接入阿里雲鏈路追蹤服務 當我們在使用阿里雲的鏈路追蹤服務時,由於對端也是支持 `Zipkin` 的協議的,故可以直接通過在 `condif/autoload/opentracing.php` 配置文件內修改 `endpoint_url` 的值為您對應的阿里雲 `region` 的地址,具體地址可在阿里雲的鏈路追蹤服務內得到,更多細節可參考 [阿里雲鏈路追蹤服務幫助文檔](https://help.aliyun.com/document_detail/100031.html?spm=a2c4g.11186623.6.547.68f974dcZlg4Mv)。 ### 使用其他 Tracer 驅動 您也可以使用其他任意符合 OpenTracing 協議的 Tracer 驅動。在 Driver 項中,填寫任意實現了 `Hyperf\Tracer\Contract\NamedFactoryInterface` 的類就可以了。該接口只有一個 make 函數,參數為驅動名稱,需返回一個實現了 OpenTracing\Tracer 的實例。 ## Reference - [Opentracing](https://opentracing.io) - [Zipkin](https://zipkin.io/) - [Jaeger](https://www.jaegertracing.io/) - [Dapper, 大規模分佈式系統的跟蹤系統](https://bigbully.github.io/Dapper-translation/)