skywalking自定義插件開發

来源:https://www.cnblogs.com/wind-wound/archive/2023/04/24/17349722.html
-Advertisement-
Play Games

skywalking是使用位元組碼操作技術和AOP概念攔截Java類方法的方式來追蹤鏈路的,由於skywalking已經打包了位元組碼操作技術和鏈路追蹤的上下文傳播,因此只需定義攔截點即可。 這裡以skywalking-8.7.0版本為例。 關於插件攔截的原理,可以看我的另一篇文章:skywalking ...


skywalking是使用位元組碼操作技術和AOP概念攔截Java類方法的方式來追蹤鏈路的,由於skywalking已經打包了位元組碼操作技術和鏈路追蹤的上下文傳播,因此只需定義攔截點即可。

這裡以skywalking-8.7.0版本為例。
關於插件攔截的原理,可以看我的另一篇文章:skywalking插件工作原理剖析

1. 創建插件模塊

apm-sniffer/apm-sdk-plugin 目錄下創建一個插件maven子模塊。

2. 插件開發

(1)思路

  1. 定義攔截點,通常是類的方法
  2. 定義攔截器,支持在攔截方法執行前後進行日誌採集
  3. 定義配置文件,啟用攔截點
  4. 編譯打包,將生成的jar放到探針的plugins目錄下

(2)定義攔截點

① 官方提供的攔截點擴展入口

skywalking提供了2種供擴展的攔截點:

  • ClassInstanceMethodsEnhancePluginDefine:支持定義構造方法和實例方法的攔截點。
  • ClassStaticMethodsEnhancePluginDefine:支持定義靜態方法的攔截點。

當然還可以直接擴展 ClassEnhancePluginDefine,這個類是上面兩個類的父類。這種方式較為麻煩,一般不推薦使用。

這裡以攔截實例方法為例,繼承 ClassInstanceMethodsEnhancePluginDefine 類。

② 類攔截規則

skywalking提供了4種類攔截的規則:

  • byName:類名匹配(包名+類名)
  • byClassAnnotationMatch:類註解匹配
  • byMethodAnnotationMatch:方法註解匹配
  • byHierarchyMatch:父類或介面匹配

註意:

  1. 這裡的匹配規則要用字元串,不要用類引用的方式(byName(ThirdPartyClass.class.getName())),否則可能會導致探針異常。
  2. 註解匹配的方式,不支持繼承的註解
  3. 父類或介面匹配的方法,儘量避免使用,否則可能會出現一些難以預料的問題

③ 設置要攔截的類名

實現 enhanceClass() 方法,定義要攔截的類名,必須是全路徑的名稱,即包名+類名。

④ 設置攔截的實例方法和攔截器的類名

實現 getInstanceMethodsInterceptPoints() 方法,定義要攔截的實例方法,以及對應攔截器的類名。攔截器類名也是包名+類名。

這裡支持定義多個實例方法,每個實例方法可以使用不同的攔截器。還支持攔截私有方法(private)。

⑤ 代碼示例

下麵的代碼實現的功能是:使用攔截器 MingBaoServiceInterceptor 攔截 MingBaoService 類的 service 方法。

public class MingBaoServiceInstrumentation extends ClassInstanceMethodsEnhancePluginDefine {
    // 要攔截的類
    private static final String ENHANCE_CLASS = "com.mingbao.service.MingBaoService";
    // 攔截器的類名
    private static final String INTERCEPT_CLASS = "org.apache.skywalking.apm.plugin.mingbao.service.MingBaoServiceInterceptor";

    /**
     * 定義要攔截的類名
     */
    @Override
    protected ClassMatch enhanceClass() {
        return NameMatch.byName(ENHANCE_CLASS);
    }
    /**
     * 定義要攔截類的方法,以及對應的攔截器
     */
    @Override
    public InstanceMethodsInterceptPoint[] getInstanceMethodsInterceptPoints() {
        return new InstanceMethodsInterceptPoint[] {
                new InstanceMethodsInterceptPoint() {
                    @Override
                    public ElementMatcher<MethodDescription> getMethodsMatcher() {
                        // 這裡是要攔截的方法
                        return named("service");
                    }
                    @Override
                    public String getMethodsInterceptor() {
                        // 定義攔截器的類名
                        return INTERCEPT_CLASS;
                    }
                    @Override
                    public boolean isOverrideArgs() {
                        // 如果有要改方法參數的需求,這裡可以設置成true
                        return false;
                    }
                }
        };
    }
    /**
     * 這裡是攔截構造方法,忽略
     */
    @Override
    public ConstructorInterceptPoint[] getConstructorsInterceptPoints() { return null; }
}

(3)定義攔截器

① 3種常見的攔截器介面

  • InstanceMethodsAroundInterceptor:實例方法攔截器
  • StaticMethodsAroundInterceptor:靜態方法攔截器
  • InstanceConstructorInterceptor:構造方法攔截器

要攔截對應的方法,必須要實現對應的介面。這裡以實現 InstanceMethodsAroundInterceptor 介面,攔截實例方法為例。

② 在方法執行前攔截

實現 beforeMethod(EnhancedInstance, Method, Object[], Class<?>[], MethodInterceptResult) 方法,此方法會在被攔截的方法執行前執行。此方法中一般是定義日誌鏈路節點span對象,一個span對象對應著日誌鏈路中的一個節點。

  • 攔截方法參數說明:
1.EnhancedInstance objInst:被增強的實例,一般用不上
2.Method method:被攔截的方法
3.Object[] allArguments:被攔截方法的入參
4.Class<?>[] argumentsTypes:被攔截方法的入參的類型
5.MethodInterceptResult result:此參數可以作為被攔截方法的返回參數,如果給此參數賦值了,會阻斷被攔截方法的執行,直接返回此參數。
	可以通過defineReturnValue()方法來定義要返回的數據。
  • 鏈路節點對象span的類型:

節點對象都實現了 AbstractSpan 介面,可以藉助ContextManager類來創建和獲取節點對象。

創建一個span對象後,就會生成一個鏈路日誌的節點。

1.EntrySpan:入口層span,它會作為一條鏈路的起點。如接收Http請求的介面層、Dubbo服務的提供方以及MQ消費者。
2.LocalSpan:中間層span,它會出現在鏈路的中間節點上。如一個業務方法被調用。
3.ExitSpan:出口層span,它會作為一條鏈路的終點。如發送Http請求的工具、Dubbo服務的調用方以及MQ生產者。
  • 鏈路節點對象span常用的設置項
1.component:組件類型,比如說Tomcat、Dubbo、SpringMVC...可以從ComponentsDefine類中定義好的一些官方組件類型中選,自定義的組件類型是無法在UI中顯示出來的。也可以不設置值,預設會顯示Unknown。(可選的類型就那麼多,一般自定義時根本找不到合適的)。
2.layer:日誌層級,可以從SpanLayer類中選擇,一共就5個:DB、RPC_FRAMEWORK、HTTP、MQ和CACHE。可選的也不多,不合適可以不設置,預設會顯示Unknown。
3.tag:日誌標簽,支持自定義日誌欄位,可以通過 span.tag(new StringTag("msg"), msg) 的方式來設置。結合後端配置項 core.default.searchableTracesTags可以達到自定義欄位搜索的目的。

③ 在方法執行後攔截

實現 afterMethod(EnhancedInstance, Method, Object[], Class<?>[], Object) 方法,此方法會在被攔截的方法執行前執行。此方法中一般是將方法的返回數據記錄到鏈路節點對象中。

  • 攔截方法參數說明:
前4個參數和beforeMethod()方法中一樣,介紹下最後那個參數
Object ret:方法的返回數據

④ 代碼示例

public class MingBaoServiceInterceptor implements InstanceMethodsAroundInterceptor {
    private static final Gson GSON = new Gson();
    /**
     * 在攔截方法前執行
     */
    @Override
    public void beforeMethod(EnhancedInstance objInst, Method method, Object[] allArguments, Class<?>[] argumentsTypes, MethodInterceptResult result) throws Throwable {
        // 解析方法入參
        String message = (String) allArguments[0];

        // 初始化span,這裡創建了一個EntrySpan
        ContextCarrier contextCarrier = new ContextCarrier();
        AbstractSpan span = ContextManager.createEntrySpan(MethodUtil.generateOperationName(method), contextCarrier);
        // 自定義標簽,記錄方法入參
        span.tag(new StringTag("req"), req);

        // 下麵的參數如果不合適可以不設置
        span.setLayer(SpanLayer.MQ);
        span.setComponent(new OfficialComponent(999, "mbService"));
    }
    /**
     * 在攔截方法後執行
     */
    @Override
    public Object afterMethod(EnhancedInstance objInst, Method method, Object[] allArguments, Class<?>[] argumentsTypes, Object ret) throws Throwable {
        // 獲取上下文中的span對象
        AbstractSpan span = ContextManager.activeSpan();
        // 自定義標簽,記錄方法出參
        span.tag(new StringTag("resp"), GSON.toJson(ret));

        // 停止日誌記錄,移除上下文
        ContextManager.stopSpan();
        // 返回方法出參
        return ret;
    }
    /**
     * 記錄方法異常
     */
    @Override
    public void handleMethodException(EnhancedInstance objInst, Method method, Object[] allArguments, Class<?>[] argumentsTypes, Throwable t) {
        ContextManager.activeSpan().log(t);
    }
}

(4)定義配置文件

在自定義插件模塊的resources目錄下定義 skywalking-plugin.def 配置文件,該文件用於幫助探針啟動時載入插件時,尋找插件攔截點。

註意:自定義插件時,不管是類還是配置文件,都要把apache的許可證註釋帶上,可以參考其他插件類文件中最上面被註釋的那一段。

mingbao-service=org.apache.skywalking.apm.plugin.mingbao.service.MingBaoServiceInstrumentation

3. 使用插件

(1)插件打包

對自定義的插件子模塊執行mvn package操作,構建完成後會生成一個名稱類似 mingbao-service-plugin-8.7.0.jar 的jar包,將jar包拷貝到 skywalking-agent/plugins 目錄下。

(2)使用自定義插件

自定義插件的使用和自帶插件使用方式相同,將 skywalking-agent 打包到項目鏡像中,使用javaagent探針啟動即可。

這裡有個建議:如果使用docker來部署項目,可以將 skywalking-agent 目錄放到項目同級目錄下,併在項目同級目錄下構建docker鏡像。因為docker build命令無法操作命令執行目錄的父級目錄所包含的其他文件。因此要保證 skywalking-agent 要在執行docker build命令的目錄下。

4. 檢查插件生效

啟動項目後,調用被攔截的類方法,然後看UI上是否生成了對應的日誌。一般情況UI上會延遲幾秒鐘才會生成日誌。

5. 可能會遇到的問題

(1)插件不生效

在探針的logs目錄下(docker鏡像部署的項目要先進入鏡像才能看到),會生成 skywalking-api.log 目錄,日誌預設級別為 INFO。插件不生效時,一般情況下,日誌文件中一定有錯誤日誌。

  • SecurityException

如果遇到報錯:java.lang.SecurityException: Invalid signature file digest for Manifest main attributes,那麼一般是因為自定義插件中依賴了第三方依賴包,在打包時生成了 *.SF*.RSA 文件,把上述文件刪掉即可。可以使用下麵的命令:

zip -d mingbao-service-plugin-8.7.0.jar 'META-INF/*SF' 'META-INF/*RSA'

您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 1,局部變數與全局變數 1,定義 局部變數:就是在函數體內的變數,在python中冒號“:”後面的變數都是局部變數,當然局部與全局也是一個相對的概念。比如出現函數嵌套的情況。 全局變數:就是在模塊中所有函數都可以調用的變數,一般在函數體外被定義。 2,使用過程 函數內的局部變數,在函數體外是不可以使 ...
  • 〇、Maven 0.1 什麼是Maven? Apache Maven is a software project management and comprehension tool. Based on the concept of a project object model (POM), Mave ...
  • 教程簡介 GWT Highcharts是一個基於 Java 的開源庫,可在 GWT 應用程式中提供優雅且功能豐富的 Highcharts 可視化,並可與 GWT 小部件庫一起使用。 GWT Highcharts入門教程 - 從基本概念到高級概念,從簡單的步驟瞭解GWT Highcharts,其中包括 ...
  • 教程簡介 Perl入門教程 - 從基本到高級概念的簡單簡單步驟瞭解Perl,包括簡介,環境,語法,數據類型,變數,標量,數組,哈希,IF ... ELSE,迴圈,運算符,日期和時間,子程式,引用,格式,文件I / O,目錄,錯誤處理,特殊變數,編碼標準,正則表達式,發送電子郵件,套接字編程,面向對象 ...
  • 泛型 問題解決 一個計算sum的函數 func sum(slice []int) int { var res int for _, value := range slice { res += value } return res } ​ 如果需要提供對int,float64,string三種數據類型 ...
  • 簡要 pyinstaller模塊主要用於python代碼打包成exe程式直接使用,這樣在其它電腦上即使沒有python環境也是可以運行的。 用法 一.安裝 pyinstaller屬於第三方庫,因此在使用的時候需提前安裝 pip install pyinstaller 二.配置spec文件 1.配置生 ...
  • 來源:https://developer.aliyun.com/article/705239 1 什麼是KeyDB? KeyDB是Redis的高性能分支,專註於多線程,記憶體效率和高吞吐量。除了多線程之外,KeyDB還具有僅在Redis Enterprise中可用的功能,例如Active Replic ...
  • Rust + Bevy 實現的 Flappy Bird 游戲 簡介 一個使用 bevy 引擎復刻的 Flappy Bird 經典小游戲。 通過該項目我們可以學到:bevy 的自定義組件,自定義插件,自定義資源,sprite 的旋轉,sprite 的移動,sprite sheet 動畫的定義使用,狀態 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...