0
  • 聊天消息
  • 系統(tǒng)消息
  • 評論與回復
登錄后你可以
  • 下載海量資料
  • 學習在線課程
  • 觀看技術視頻
  • 寫文章/發(fā)帖/加入社區(qū)
會員中心
創(chuàng)作中心

完善資料讓更多小伙伴認識你,還能領取20積分哦,立即完善>

3天內不再提示

使用Quickwit、Jaeger和Grafana監(jiān)控您的Rust應用程序

jf_wN0SrCdH ? 來源:Rust語言中文社區(qū) ? 作者:Rust語言中文社區(qū) ? 2023-06-27 11:36 ? 次閱讀

使用Quickwit、Jaeger和Grafana監(jiān)控您的Rust應用程序

你可能已經看過了Lucas Palmieri的博客文章Are we observable yet? An introduction to Rust telemetry。如果你還沒有看過,我們建議閱讀一下,因為它提供了一個全面的介紹,介紹了如何處理 Rust 代碼中的日志。

然而,僅僅記錄日志可能是不夠的,特別是在分布式架構中。在 Quickwit 中,我們經常使用跟蹤來理解性能瓶頸并提高速度。當我們遇到 Quickwit 的搜索響應緩慢時,我們經常會問自己:是什么導致了減速?是網絡相關的問題,磁盤 I/O 還是過多的 CPU 使用?

在本博客文章中,我們將展示如何為 Rust 應用程序進行測量,并生成跟蹤數據,從 DevOps 視角利用它們。我們的目標將是雙重的:

使用廣泛認可的 Jaeger UI 分析跟蹤,以獲取有關應用程序行為的見解。

從這些跟蹤數據中派生 RED(速率、錯誤和持續(xù)時間)指標,并在 Grafana 中監(jiān)視它們。如果您想進一步了解,我們建議參考以下資源:Weaveworks 的 RED 方法和 Google SRE 書籍中有關監(jiān)控分布式系統(tǒng)的部分。

現(xiàn)在,讓我們深入介紹步驟,其中我們將涵蓋以下關鍵方面:

為使用 Actix 構建的簡單 Web API 進行測量。

將您的跟蹤和指標數據推送到 Quickwit。

在 Jaeger UI 中檢測、診斷和解決問題。

在 Grafana 中監(jiān)視您的應用程序的 RED 指標(速率、錯誤、持續(xù)時間)。

在深入了解之前,請確保您的系統(tǒng)上已安裝并正確運行以下軟件:

Rust 1.68+

Docker

如果您仍在運行舊版本的 Docker,則需要安裝docker-compose。

構建并測量 Rust 應用

我們將使用 Actix Web 框架創(chuàng)建一個基本的 Rust 應用程序。這個應用程序是一個包含單個端點的 Web API。它將從受歡迎的 JSONPlaceholder 公共 Web API 獲取帖子及其評論,并將它們顯示為 JSON。為了更好地了解我們的應用程序生命周期并可能優(yōu)化它,我們將確保測量以下例程:

從 /posts 獲取帖子。

獲取每個帖子的評論 /posts/1/comments

創(chuàng)建一個名為rust-app-tracing的新目錄。在終端中切換到該目錄,并運行以下命令初始化一個新的 Rust 項目。

cargo new web-api

讓我們還要確保在web-api/Cargo.toml文件中擁有所需的依賴項。

actix-web:用于在 Rust 中構建 Web 應用程序的快速 Web 框架。

actix-web-opentelemetry:actix-web框架的 open-telemetry 擴展。

opentelemetry:Rust 的核心 open-telemetry SDK,包括跟蹤和指標。

opentelemetry-otlp:提供各種 open-telemetry 導出器的 crate。

reqwest:提供一個直觀的 API 來進行 HTTP 請求。

tokio:為我們的應用程序提供異步運行時。

Web API 應用程序代碼

首先,讓我們通過創(chuàng)建一個名為telemetry.rs的文件來配置應用程序跟蹤,我們將在其中處理所有跟蹤配置。


// telemetry.rs ... const SERVICE_NAME: &'static str = "quickwit-jaeger-demo"; pub fn init_telemetry(exporter_endpoint: &str) { // Create a gRPC exporter let exporter = opentelemetry_otlp::new_exporter() .tonic() .with_endpoint(exporter_endpoint); // Define a tracer let tracer = opentelemetry_otlp::new_pipeline() .tracing() .with_exporter(exporter) .with_trace_config( trace::new(vec![KeyValue::new( opentelemetry_semantic_conventions::SERVICE_NAME, SERVICE_NAME.to_string(), )])), ) .install_batch(opentelemetry::Tokio) .expect("Error: Failed to initialize the tracer."); // Define a subscriber. let subscriber = Registry::default(); // Level filter layer to filter traces based on level (trace, debug, info, warn, error). let level_filter_layer = EnvFilter::new("INFO")); // Layer for adding our configured tracer. let tracing_layer = tracing_opentelemetry::layer().with_tracer(tracer); // Layer for printing spans to stdout let formatting_layer = BunyanFormattingLayer::new( SERVICE_NAME.to_string(), std::stdout, ); global::new()); subscriber .with(level_filter_layer) .with(tracing_layer) .with(JsonStorageLayer) .with(formatting_layer) .init() }

Copy 接下來,我們將實現(xiàn)我們的 API 端點,并在處理程序函數中添加一些測量代碼。重要的是要注意,我們的重點不在于此應用程序的功能,而在于從應用程序生成有意義且可利用的跟蹤數據。

首先,我們有一些模型文件,允許我們對post和comment進行序列化和反序列化。

//models.rs ... #[derive(Debug, Serialize, Deserialize)] #[serde(rename_all = "camelCase")] pub struct Post { pub user_id: i64, pub id: i64, pub title: String, pub body: String, #[serde(default)] pub comments: Vec, } #[derive(Debug, Serialize, Deserialize)] #[serde(rename_all = "camelCase")] pub struct Comment { pub post_id: i64, pub id: i64, pub name: String, pub email: String, pub body: String, }


接下來,讓我們處理 API 端點處理程序。請注意,某些函數上裝飾有instrument屬性。這是我們如何在處理程序函數和它用于執(zhí)行任務的后續(xù)函數上啟用跟蹤的方法。


// lib.rs ... const BASE_API_URL: &'static str = "https://jsonplaceholder.typicode.com"; // The get_post handler #[instrument(level = "info", name = "get_posts", skip_all)] #[get("")] async fn get_posts() -> Result { // Randomly simulate errors in request handling let choices = [200, 400, 401, 200, 500, 501, 200, 500]; let mut rng = rand::thread_rng(); let choice = choices.choose(&mut rng) .unwrap() .clone(); match choice { 400..=401 => Ok(HttpResponse::from_u16(choice).unwrap())), 500..=501 => Ok(HttpResponse::from_u16(choice).unwrap())), _ => { let posts = fetch_posts(20) .await .map_err(actix_web::ErrorInternalServerError)?; Ok(HttpResponse::Ok().json(posts)) } } } // Fetching posts with a limit. #[instrument(level = "info", name = "fetch_posts")] async fn fetch_posts(limit: usize) -> anyhow::Result { let client = Client::new(); let url = format!("{}/posts", BASE_API_URL); let mut posts: Vec = request_url(&client, &url).await?; posts.truncate(limit); let post_idx_to_ids: Vec<(usize, i64)> = posts.iter().enumerate().map(|(idx, post)| (idx, post.id)).collect(); // fetch post comments one after another. for (index, post_id) in post_idx_to_ids { let comments = fetch_comments(&client, post_id).await?; posts[index].comments = comments } Ok(posts) } ...


在上面的片段中,我們僅發(fā)送跟蹤。也可以使用可靠的日志收集器來收集日志并將其發(fā)送到Quickwit或其他后端。

使用 Quickwit 收集跟蹤數據

現(xiàn)在我們已經構建了應用程序。讓我們與 Quickwit 一起運行,并確保生成的跟蹤被 Quickwit 索引。 與我們在之前的博客文章中所做的不同,我們將創(chuàng)建一個 docker-compose 文件來簡化 Quickwit、Jaeger 和 Grafana 之間的設置。以下 docker-compose 文件包含所有必要的配置。

QW_ENABLE_OTLP_ENDPOINT:允許 Quickwit 接受和攝取跟蹤和日志數據。

SPAN_STORAGE_TYPE、GRPC_STORAGE_SERVER、QW_ENABLE_JAEGER_ENDPOINT:允許 Jaeger 從 Quickwit 拉取跟蹤和日志以進行分析。

GF_PLUGINS_ALLOW_LOADING_UNSIGNED_PLUGINS:允許我們在 Grafana 中加載特定的插件。


# docker-compose.yaml version: '3' services: quickwit: image: quickwit/quickwit:latest command: run restart: always environment: QW_ENABLE_OTLP_ENDPOINT: true QW_ENABLE_JAEGER_ENDPOINT: true ports: - '7280:7280' - '7281:7281' volumes: - ./qwdata:/quickwit/qwdata jaeger: image: jaegertracing/jaeger-query:latest restart: always depends_on: - quickwit environment: SPAN_STORAGE_TYPE: 'grpc-plugin' GRPC_STORAGE_SERVER: 'quickwit:7281' ports: - '16686:16686' grafana: image: grafana/grafana-enterprise:latest restart: always user: root depends_on: - quickwit environment: GF_PLUGINS_ALLOW_LOADING_UNSIGNED_PLUGINS: 'quickwit-quickwit-datasource' ports: - '3000:3000' volumes: - ./grafana-storage:/var/lib/grafana



有了這個 docker-compose 文件,讓我們在項目目錄中創(chuàng)建所需的目錄以使服務正確運行。創(chuàng)建qwdata目錄以存儲 Quickwit 數據。 然后,下載并將 Quickwit Grafana 數據源插件放置在預期位置。

wget https://github.com/quickwit-oss/quickwit-datasource/releases/download/v0.2.0/quickwit-quickwit-datasource-0.2.0.zip && mkdir -p grafana-storage/plugins && unzip quickwit-quickwit-datasource-0.2.0.zip -d grafana-storage/plugins



現(xiàn)在讓我們通過運行以下命令啟動所有服務(Quickwit、Jaeger、Grafana): 如果沒有問題,現(xiàn)在我們可以運行 Web 應用程序并使用 cURL 幾次命中http://localhost:9000/post端點,以生成一些跟蹤。

curl -X GET http://localhost:9000/post 等待約 10 秒鐘,新的跟蹤將被索引并可供搜索。 您現(xiàn)在可以通過使用 cURL 搜索otel-traces-v0_6索引來檢查 Quickwit 是否已索引跟蹤數據。

curl -X POST http://localhost:7280/api/v1/otel-traces-v0_6/search -H 'Content-Type: application/json' -d '{ "query": "service_name:quickwit-jaeger-demo" }'


您也可以使用 Quickwit UIhttp://localhost:7280/ui/search查看數據。76e0d844-104a-11ee-962d-dac502259ad0.png

Jaeger 容器已經在運行中了,可以轉到http://localhost:16686查看我們的應用程序跟蹤。

77395e92-104a-11ee-962d-dac502259ad0.png

從上面的截圖可以看出,我們依次為每個帖子獲取評論。也就是說,我們一個接一個地進行了二十次請求。這使得整個請求處理時間更長(上面為 4.39s)。 但我們能不能更好地做? 在 Rust 開發(fā)人員擁有的所有優(yōu)秀工具中,答案是顯而易見的 "是的!"。讓我們利用 Tokio 和 Rustfutures crate的異步流特性,通過并行獲取評論。 讓我們更新我們的fetch_posts函數,以批量并行運行請求,每次同時進行十個請求。這應該可以進一步加速事情。

// Fetching posts with a limit. #[instrument(level = "info", name = "fetch_posts")] async fn fetch_posts(limit: usize) -> anyhow::Result { ... // fetch post comments concurrently. let tasks: Vec<_> = post_idx_to_ids .into_iter() .map(|(index, post_id)| { let moved_client = client.clone(); async move { let comments_fetch_result = fetch_comments(&moved_client, post_id).await; (index, comments_fetch_result) } }) .collect(); let mut stream = futures::iter(tasks) .buffer_unordered(10); // batch of 10 request at a time while let Some((index, comments_fetch_result)) = stream.next().await { let comments = comments_fetch_result?; posts[index].comments = comments; } ... }


通過這個改變,你會注意到我們現(xiàn)在處理請求的時間大約為2.46秒,同時你也可以直觀地看到我們的請求處理程序在運行期間最多同時運行了十個fetch_comments請求。

778f2cb4-104a-11ee-962d-dac502259ad0.png

Jaeger 適用于對單個跟蹤進行專注檢查。但如果我們想要監(jiān)視服務的延遲呢?如果我們想要計算具有給定跟蹤元數據的錯誤或請求的數量呢? 這就是 Grafana 儀表板的用處。我們想要從我們的跟蹤構建 RED 指標并在 Grafana 中可視化它們。 轉到http://localhost:3000/login,使用admin作為用戶名和密碼登錄。 登錄后,我們可以使用新發(fā)布的Quickwit 數據源插件連接到 Quickwit 并查詢我們的應用程序跟蹤。

77d02642-104a-11ee-962d-dac502259ad0.png

為了使 RED 指標監(jiān)控過程更加方便,我們準備了一個預配置的Grafana 儀表板供您下載并導入到您的 Grafana 實例中。 該儀表板作為一種強大的工具,用于可視化和理解性能。它包括三個面板:

第一個面板顯示每分鐘的請求數量。

第二個面板顯示每分鐘的錯誤數量。

第三個面板呈現(xiàn)每分鐘請求的持續(xù)時間百分位數。

為了觀察這些指標的運行情況,您可以使用 HTTP 基準測試工具,甚至可以使用本教程提供的此腳本發(fā)送多個并發(fā)請求到您的 Rust 應用程序。 現(xiàn)在讓我們來看一下 Grafana 儀表板的截圖,展示了運行腳本后的指標情況。7802395c-104a-11ee-962d-dac502259ad0.png

就是這樣!在這篇博客文章中,我們超越了基本的日志記錄,深入了解了分布式跟蹤以及如何使用它來監(jiān)視應用程序性能。 我們構建 Quickwit 的經驗告訴我們,分布式跟蹤對于了解由于調用 S3 或在本地磁盤上讀取數據而失去時間的位置非常重要。我們希望它對您也有所幫助 :) 愉快的編碼和觀察!

審核編輯:湯梓紅

聲明:本文內容及配圖由入駐作者撰寫或者入駐合作網站授權轉載。文章觀點僅代表作者本人,不代表電子發(fā)燒友網立場。文章及其配圖僅供工程師學習之用,如有內容侵權或者其他違規(guī)問題,請聯(lián)系本站處理。 舉報投訴
  • cpu
    cpu
    +關注

    關注

    68

    文章

    10876

    瀏覽量

    212124
  • Web
    Web
    +關注

    關注

    2

    文章

    1264

    瀏覽量

    69524
  • API
    API
    +關注

    關注

    2

    文章

    1504

    瀏覽量

    62157
  • 應用程序
    +關注

    關注

    37

    文章

    3283

    瀏覽量

    57747
  • Rust
    +關注

    關注

    1

    文章

    229

    瀏覽量

    6619

原文標題:【Rust日報】2023-06-20 使用Quickwit、Jaeger和Grafana監(jiān)控您的Rust應用程序

文章出處:【微信號:Rust語言中文社區(qū),微信公眾號:Rust語言中文社區(qū)】歡迎添加關注!文章轉載請注明出處。

收藏 人收藏

    評論

    相關推薦

    Rust GUI實踐之Tarui模塊

    Tauri是一個用于構建跨平臺本地應用程序的工具包,它使用Rust語言作為主要開發(fā)語言,可以在Windows,MacOS和Linux等平臺上運行。Tauri基于Web技術棧,可以使用HTML,CSS
    的頭像 發(fā)表于 09-19 16:42 ?6116次閱讀

    Rust GUI實踐之Rust-Qt模塊

    Rust-Qt 是 Rust 語言的一個 Qt 綁定庫,它允許 Rust 開發(fā)者使用 Qt 框架來創(chuàng)建跨平臺的圖形界面應用程序。Qt 是一個跨平臺的
    的頭像 發(fā)表于 09-30 16:43 ?1616次閱讀

    開放分布式追蹤(OpenTracing)入門與 Jaeger 實現(xiàn)

    。應用程序通過 API 寫入數據,client library 把 trace 信息按照應用程序指定的采樣策略傳遞給 jaeger-agent。Agent - 它是一個監(jiān)聽在 UDP 端口
    發(fā)表于 03-07 16:27

    阿里云容器Kubernetes監(jiān)控(二) - 使用Grafana展現(xiàn)Pod監(jiān)控數據

    摘要: 簡介 在kubernetes的監(jiān)控方案中,Heapster+Influxdb+Grafana的組合相比prometheus等開源方案而言更為簡單直接。而且Heapster在
    發(fā)表于 05-10 15:28

    只會用Python?教你在樹莓派上開始使用Rust

    。出現(xiàn)提示時,選擇默認安裝。安裝程序將在完成時通知,盡管安裝可能會花費一些時間,具體取決于的連接。/p》 安裝后安裝成功,但是您還不能開始使用它。如果嘗試按版本檢查Rust和Car
    發(fā)表于 05-20 08:00

    簡述linux-arm64 UOS安裝開源Grafana的步驟

    (linux-arm64)UOS安裝開源Grafana-7.2.0,和CentOS安裝步驟一樣Grafana是一款用Go語言開發(fā)的開源數據可視化工具,可以做數據監(jiān)控和數據統(tǒng)計,帶有告警功能,支持
    發(fā)表于 06-16 15:00

    多功能監(jiān)控系統(tǒng)應用程序免費下載

    本文檔的主要內容詳細介紹的是多功能監(jiān)控系統(tǒng)應用程序免費下載。
    發(fā)表于 03-29 08:00 ?2次下載
    多功能<b class='flag-5'>監(jiān)控</b>系統(tǒng)<b class='flag-5'>應用程序</b>免費下載

    加入移動應用程序世界的計劃是什么?

    如果剛開始,請確保雇用了著名的軟件開發(fā)公司來進行無錯誤的處理。如今競爭非常激烈,需要創(chuàng)建一個能夠很好地吸引客戶的高科技移動應用程序
    的頭像 發(fā)表于 07-10 16:05 ?1615次閱讀

    應用程序解決方案公告選擇正確的數字電位器

    應用程序解決方案公告選擇正確的數字電位器
    發(fā)表于 04-19 21:08 ?2次下載
    為<b class='flag-5'>您</b>的<b class='flag-5'>應用程序</b>解決方案公告選擇正確的數字電位器

    Jaeger Uber的分布式追蹤系統(tǒng)

    jaeger.zip
    發(fā)表于 04-26 10:11 ?0次下載
    <b class='flag-5'>Jaeger</b> Uber的分布式追蹤系統(tǒng)

    使用Thanos+Prometheus+Grafana構建監(jiān)控系統(tǒng)

    對于彈性伸縮和高可用的系統(tǒng)來說,一般有大量的指標數據需要收集和存儲,如何為這樣的系統(tǒng)打造一個監(jiān)控方案呢?本文介紹了如何使用 Thanos+Prometheus+Grafana 構建監(jiān)控系統(tǒng)。
    的頭像 發(fā)表于 05-05 21:14 ?2655次閱讀

    Grafana 9泰酷了吧

    Grafana 9.0 的主要重點是改善 Grafana 的用戶體驗,使可觀察性和數據可視化更易用也更容易獲得。無論是通過 Prometheus 和 Loki 可視化查詢生成器還是面板和儀表板搜索
    的頭像 發(fā)表于 05-30 11:30 ?576次閱讀
    <b class='flag-5'>Grafana</b> 9泰酷了吧

    虹科干貨 | 虹科教使用Grafana輕松實現(xiàn)OVL數據可視化

    虹科IIoT虹科教使用Grafana輕松實現(xiàn)OVL數據可視化前言本文以MySQL數據庫為例介紹如何通過Grafana來實現(xiàn)OVL數據的可視化,其中有兩種方法,一種是通過OVL的RESTAPI來實現(xiàn)
    的頭像 發(fā)表于 05-27 10:34 ?779次閱讀
    虹科干貨 | 虹科教<b class='flag-5'>您</b>使用<b class='flag-5'>Grafana</b>輕松實現(xiàn)OVL數據可視化

    Preemptive 全面的移動應用程序保護

    由于黑客越來越多地將目標對準消費者和企業(yè)移動應用程序,應用程序可能會給的組織帶來風險。例如,黑客可以使用反編譯器或反匯編器對的安卓或
    的頭像 發(fā)表于 07-06 10:41 ?688次閱讀
    Preemptive 全面的移動<b class='flag-5'>應用程序</b>保護

    基于Rust的Log日志庫介紹

    Rust是一門系統(tǒng)級編程語言,因其安全性、高性能和并發(fā)性而備受歡迎。在Rust應用程序中,日志記錄是一項非常重要的任務,因為它可以幫助開發(fā)人員了解應用程序的運行情況并解決問題。
    的頭像 發(fā)表于 09-19 14:49 ?3516次閱讀