動態

詳情 返回 返回

Jmeter壓測實戰:Jmeter二次開發之自定義函數 | 京東雲技術團隊 - 動態 詳情

1 前言

Jmeter是Apache基金會下的一款應用場景非常廣的壓力測試工具,具備輕量、高擴展性、分佈式等特性。Jmeter已支持實現隨機數、計數器、時間戳、大小寫轉換、屬性校驗等多種函數,方便使用人員使用。如果在使用過程中存在和業務強耦合的常用功能函數,在Jmeter不支持的情況下,那就需要單獨開發自定義函數實現特定功能。

本文介紹如何開發Jmeter自定義函數實現快速生成京東宙斯下單標準sign,同時深刻理解Jmeter的插件化機制及高擴展性特性。

2 開發準備

  1. Java基礎開發
  2. Maven基本使用
  3. 開發依賴版本
    JDK 1.8.0Maven 3.6.3Jmeter 5.4.3

3 自定義函數核心實現

3.1 新建項目

  • 新建maven項目,這裏項目名為:JSF_Sampler
  • 因為是基於Jmeter的擴展,需要依賴包Jmeter兩個核心包,分別是:
  • ApacheJMeter_core
  • ApacheJMeter_java
  • ApacehJMeter_functions

pom.xml文件核心配置如下

<groupId>com.jd.jmeter.jsf</groupId>
<artifactId>JSF_Sampler</artifactId>
<version>1.0-SNAPSHOT</version>
<properties>
        <maven.compiler.source>8</maven.compiler.source>
        <maven.compiler.target>8</maven.compiler.target>
        <project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
        <jmeter-version>5.4.3</jmeter-version>
</properties>
<dependencies>
        <dependency>
            <groupId>org.apache.jmeter</groupId>
            <artifactId>ApacheJMeter_core</artifactId>
            <version>${jmeter-version}</version>
        </dependency>
        <dependency>
            <groupId>org.apache.jmeter</groupId>
            <artifactId>ApacheJMeter_java</artifactId>
            <version>${jmeter-version}</version>
        </dependency>
        <dependency>
            <groupId>org.apache.jmeter</groupId>
            <artifactId>ApacheJMeter_functions</artifactId>
            <version>${jmeter-version}</version>
        </dependency>
    </dependencies>

3.2 繼承實現AbstractFunction類

實現類依次實現以下幾個步驟

1)新建實現類並繼承 AbstractFunction

  • 注意:實現類的包名必須包含xxx.functions.xxx,Jmeter使用命名規則實現實現類的加載。

2)重寫以下方法,每個方法的用途見下方代碼註釋

  • execute()
  • setParameters()
  • getReferenceKey()
  • getArgumentDesc()
   /**
     * 京東宙斯 下單標準字段常量
     */
    private static final String APP_KEY = "app_key";
    private static final String APP_SECRET = "app_secret";
    private static final String ACCESS_TOKEN = "access_token";
    private static final String TIMESTAMP = "timestamp";
    private static final String V = "v";
    private static final String METHOD = "method";
    private static final String BUY_PARAM_JSON = "360buy_param_json";
    /**
     * Jmeter中自定義的函數名,在Jmeter的函數助手中可以看到
     */
    private static final String FUNC_NAME = "__GenSignFunction";

    /**
     * 自定義函數的描述,入參,出參,方便使用人員參考使用
     */
    private static final List<String> desc = new ArrayList<>();

    static {
        desc.add("This function is used to generate the JD's JOS sign value");
    }
 /**
     * 此為自定義函數核心實現類,其中,入參SampleResult為上次運行的結果,Sampler為當前的採集器;
     * 返回值為該函數的返回值
     * @param sampleResult
     * @param sampler
     * @return
   * @throws InvalidVariableException
 */
 @Override
    public String execute(SampleResult sampleResult, Sampler sampler) throws InvalidVariableException {
        // 入參處理
        String param = String.valueOf((CompoundVariable)paramValues[0]);
        String signResult = paramHandler(param);

        return signResult;
    }

    /**
     * 按京東宙斯sign加密規則生成標準sign
     * @param param
     * @return
     */
    public String paramHandler(String param){
        Map<String,String> valueMap = new HashMap();
        // 按&符號分割
        String[] paramArray = param.split("&");
        for (int i = 0; i < paramArray.length-1; i++) {
            String key = paramArray[i].split("=")[0];
            String value = paramArray[i].split("=")[1];
            valueMap.put(key,value);
        };
        // 京東宙斯標準sign
        String josGign = EncryptUtil.getSignature(valueMap.get("app_secret")+BUY_PARAM_JSON+valueMap.get("360buy_param_json")
        +ACCESS_TOKEN+valueMap.get("access_token")
        +APP_KEY+valueMap.get("app_key")
        +METHOD+valueMap.get("method")
        +TIMESTAMP+valueMap.get("timestamp")
        +V+valueMap.get("v")
        +valueMap.get("app_secret"));
        return josGign;
    }

/**
* 配置入參,jmeter函數助手入參
*/
    @Override
    public void setParameters(Collection<CompoundVariable> collection) throws InvalidVariableException {
        paramValues = collection.toArray();
    }
/**
* 此方法返回自定義的函數名稱
*/
    @Override
    public String getReferenceKey() {
        return FUNC_NAME;
    }
/**
* 此方法返回函數描述信息
*/
    @Override
    public List<String> getArgumentDesc() {
        return desc;
    }

3.3 最終項目結構

4 Jmeter加載擴展包

以上開發完成,打包此項目,注意這裏的打包要包含依賴包。

4.1 maven構建配置

<build>
        <finalName>${project.artifactId}</finalName>
        <defaultGoal>install</defaultGoal>
        <plugins>
            <plugin>
                <groupId>org.apache.maven.plugins</groupId>
                <artifactId>maven-compiler-plugin</artifactId>
                <configuration>
                    <source>1.8</source>
                    <target>1.8</target>
                </configuration>
            </plugin>
            <plugin>
                <groupId>org.apache.maven.plugins</groupId>
                <artifactId>maven-assembly-plugin</artifactId>
                <configuration>
                    <descriptorRefs>
                        <descriptorRef>jar-with-dependencies</descriptorRef>
                    </descriptorRefs>
                </configuration>
                <executions>
                    <execution>
                        <id>assemble-all</id>
                        <phase>package</phase>
                        <goals>
                            <goal>single</goal>
                        </goals>
                    </execution>
                </executions>
            </plugin>
        </plugins>
    </build>

4.2 項目打包

打包指令如下
mvn package -Dmaven.test.skip=true

4.3 Jmeter加載擴展包

將打包後的擴展包放置到Jmeter的ext目錄:apache-jmeter-5.4.3/lib/ext/

啓動Jmeter後,Jmeter會自動加載ext目錄中的擴展包

打開Jmeter函數助手後,可以看到本次實現類中打印的相關日誌

5 自定義函數調用調試

5.1 打開Jmeter函數助手,選擇自定義函數

5.2 京東宙斯接口驗證

這裏使用京東快遞獲取預製運單號接口,輸入GET請求後,直接點擊運行函數【Generate & Copy to clipboard】,出參返回32位sign值。

GET請求入參
method=jingdong.etms.waybillcode.get&app_key=349559FAE87E66826499890862E40A44&access_token=c8c2bdc8d1684630bb771a503d5b5a7fkyzh×tamp=2022-01-28 15:10:00&360buy_param_json={"preNum":"1","customerCode":"10K43816","orderType":"0"}&v=2.0&sign=EBB52C6CEDA34703ADE72D4AA4D8F316&app_secret=29959e4cadc14ff4998d4fc26d1e5063

6 總結

本文通過自定義函數實現了京東宙斯下單標準sign的生成,希望通過本項目大家可以學習到:

  • 如何二次開發Jmeter,實現自己特有的自定義函數。
  • 理解為何官方介紹Jmeter是插件化的,高擴展性特性。
  • 更好的理解Jmeter內部處理機制。

作者:京東物流 苗浩衝

來源:京東雲開發者社區

user avatar u_16307147 頭像 jfren 頭像 tssc 頭像 lvweifu 頭像 codingtea 頭像 _58d8892485f34 頭像 qqxx6661 頭像 tonyyoung 頭像 litongjava 頭像 hulaxingxingxing 頭像 piano 頭像 tangge 頭像
點贊 13 用戶, 點贊了這篇動態!
點贊

Add a new 評論

Some HTML is okay.