1. 程式人生 > >基於APPIUM測試微信公眾號的UI自動化測試框架(結合Allure2測試報告框架)

基於APPIUM測試微信公眾號的UI自動化測試框架(結合Allure2測試報告框架)

clas ava sed rom pos enabled via 代碼管理 ons

框架初衷

前兩周組內的小夥伴跟我說她現在測試的微信公眾號項目(保險)每次上新產品時測試起來很費時,存在大量的重復操作(點點點),手工測試每個產品可能需要半天到一天的時間,復雜的產品需要兩天。
由於保險下單的過程中字段比較多,輸入費勁的同時測試用例也很多(不同年齡段、工種、有無社保等),且!每個產品的頁面都有部分差異!
問我能否基於UI自動化提高她測試新產品的測試速度,同時用於上線時生產的驗證。
因為我寫過微信公眾號頁面的UI監控腳本,也嘗試過基於appium的多機並發測試,於是我就想,能否搭建一個框架,讓小夥伴每次測試新產品的時候只要輸入測試數據+修改產品差異部分代碼,然後框架分發給不同的手機去執行,最後展示測試報告?

最終效果

一個case大約3-5分鐘,三臺手機執行測話三個新產品半天就能測完。
下面是放到jenkins上運行demo的測試報告。


技術分享圖片

下面是用例運行失敗時的界面,提供截圖、重試、case日誌以及appium的日誌。


技術分享圖片

框架介紹

1.主要工具

JAVA 版本1.8
appium-server 版本1.6.3
appium java-client 版本5.0.0-BETA8
testNG 用例組織
Allure2 測試報告
Jenkins 持續集成
Git 代碼管理

2.工程目錄及主要代碼

技術分享圖片

pages:沒有采用PO模式,頁面以接口的形式定義,頁面元素即為變量。
pageoptions:頁面功能封裝在pageoptions包中,封裝成靜態方法。
testcase:繼承BaseDriver,driver初始化後即可執行測試。
util:appiumserver啟動工具類、失敗自動截圖等

下面是每個package內的代碼。

2.1 POM.XML

<?xml version="1.0" encoding="UTF-8"?>

<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
    xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
    <modelVersion>4.0.0</modelVersion>

    <groupId>com.cpeoc</groupId>
    <artifactId>jyx</artifactId>
    <version>0.0.1-SNAPSHOT</version>

    <name>jyx</name>

    <properties>
        <java.version>1.7</java.version>
        <aspectj.version>1.8.10</aspectj.version>
        <project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
    </properties>

    <dependencies>
        <dependency>
            <groupId>org.testng</groupId>
            <artifactId>testng</artifactId>
            <version>6.11</version>
        </dependency>
        <dependency>
            <groupId>io.appium</groupId>
            <artifactId>java-client</artifactId>
            <version>5.0.0-BETA8</version>
        </dependency>

        <dependency>
            <groupId>io.qameta.allure</groupId>
            <artifactId>allure-testng</artifactId>
            <version>2.0-BETA14</version>
            <scope>test</scope>
        </dependency>
        <dependency>
            <groupId>org.hamcrest</groupId>
            <artifactId>hamcrest-all</artifactId>
            <version>1.3</version>
            <scope>test</scope>
        </dependency>
        <dependency>
            <groupId>org.slf4j</groupId>
            <artifactId>slf4j-simple</artifactId>
            <version>1.7.21</version>
            <scope>test</scope>
        </dependency>
        <dependency>
            <groupId>dom4j</groupId>
            <artifactId>dom4j</artifactId>
            <version>1.6.1</version>
        </dependency>
        <dependency>
            <groupId>com.belerweb</groupId>
            <artifactId>pinyin4j</artifactId>
            <version>2.5.1</version>
        </dependency>
    </dependencies>

    <build>
        <plugins>
            <plugin>
                <groupId>org.apache.maven.plugins</groupId>
                <artifactId>maven-compiler-plugin</artifactId>
                <version>3.3</version>
                <configuration>
                    <source>1.7</source>
                    <target>1.7</target>
                </configuration>
            </plugin>
            <plugin>
                <groupId>org.apache.maven.plugins</groupId>
                <artifactId>maven-surefire-plugin</artifactId>
                <version>2.18.1</version>
                <configuration>
                    <testFailureIgnore>true</testFailureIgnore>
                    <argLine>
                        -javaagent:"${settings.localRepository}/org/aspectj/aspectjweaver/${aspectj.version}/aspectjweaver-${aspectj.version}.jar"
                    </argLine>
                    <suiteXmlFiles>
                        <suiteXmlFile>testng-OnePhone.xml</suiteXmlFile>
                    </suiteXmlFiles>
                    <!-- <workingDirectory>target\</workingDirectory> -->
                </configuration>
                <dependencies>
                    <dependency>
                        <groupId>org.aspectj</groupId>
                        <artifactId>aspectjweaver</artifactId>
                        <version>${aspectj.version}</version>
                    </dependency>
                </dependencies>
            </plugin>
        </plugins>
    </build>

</project>

2.2pageoptions包

package com.cpeoc.jyx.pageoptions;

import org.openqa.selenium.WebElement;
import org.openqa.selenium.support.ui.WebDriverWait;
import org.testng.Assert;

import com.cpeoc.jyx.pages.Index;
import com.cpeoc.jyx.util.PinYinUtil;
import com.cpeoc.jyx.util.WaitUtil;

import io.appium.java_client.AppiumDriver;

/**
 * 首頁通用功能實現類
 * @author ken
 * @date 2018年6月15日
 */
public class IndexOp {
    /**
     * 切換到指定城市
     * @param driver
     * @param wait
     * @param cityName 城市名
     */
    public static void switchCity(AppiumDriver<WebElement> driver,WebDriverWait wait,String cityName){
        
        String firstWordtOfCity = PinYinUtil.getFirstWordFromChinese(cityName);
        String cityNameXpath = "//*[contains(text(),‘"+cityName+"‘)]";
        String cityPinYinXpath = "//*[contains(text(),‘"+firstWordtOfCity+"‘)]";
        WaitUtil.waitElementByXpath(wait, Index.LOCATE).click();
        WaitUtil.waitElementByXpath(wait, cityPinYinXpath).click();
        WaitUtil.waitElementByXpath(wait, cityNameXpath).click();
        WebElement locate = WaitUtil.waitElementByXpath(wait, Index.LOCATE);
        Assert.assertEquals(locate.getText(), cityName, "城市切換有bug!");
        
    }
    
}

2.3pages包

package com.cpeoc.jyx.pages;
/**
 * 簡易險首頁
 * @author ken
 * @date 2018年6月8日
 * @see 
 *  首頁用到的元素全部定義在這裏,用xpath保存
 */
public interface Index {
    
    /**
     * 全國站切換確認窗口
     */
    String ALERT_BTN = "//*[@class=‘alert-btn‘]";
    
    /**
     * 定位城市
     */
    String LOCATE = "//*[contains(@class,‘locate‘)]";
    
    /**
     * 意外險
     */
    String  ACCIDENT ="//a[contains(@href,‘categoryCode=1000‘)]";
    
    /**
     * 健康險
     */
    String  HEALTH ="//a[contains(@href,‘categoryCode=1300‘)]";
    
    /**
     * 財產險
     */
    String  PROPERTY ="//a[contains(@href,‘categoryCode=1200‘)]";
    
    /**
     * 旅遊險
     */
    String  TOURISM ="//a[contains(@href,‘categoryCode=1100‘)]";
    
    /**
     * 航空險
     */
    String  AVIATION ="//a[contains(@href,‘categoryCode=1103‘)]";

}

2.4testcase包

一個簡單case。

package com.cpeoc.jyx.testcases.index;

import io.qameta.allure.Description;
import io.qameta.allure.Epic;

import org.testng.annotations.Test;

import com.cpeoc.jyx.pageoptions.IndexOp;
import com.cpeoc.jyx.util.BaseDriver;

/**
 * 城市定位 -- 切換
 * 
 * @author ken
 * @date 2018年6月12日
 * @see 測試微信號:王九 東莞切換到重慶 分組:wangjiu
 */
@Epic("城市定位")
public class TestCitySelect extends BaseDriver {

    @Test(groups = { "wangjiu" })
    @Description("測試城市切換")
    public void testCitySelect() {
        System.out.println("運行測試用例------------------TestCitySelect");
        // 1.點擊城市進入城市選擇頁。點擊C-重慶市
        IndexOp.switchCity(driver, wait, "重慶市");

    }

}

產品下單case。

package com.cpeoc.jyx.testcases.insbuy;


import io.qameta.allure.Description;
import io.qameta.allure.Epic;
import io.qameta.allure.Story;

import org.openqa.selenium.WebElement;
import org.testng.annotations.AfterMethod;
import org.testng.annotations.DataProvider;
import org.testng.annotations.Test;

import com.cpeoc.jyx.pageoptions.CommonPageOp;
import com.cpeoc.jyx.pageoptions.IndexOp;
import com.cpeoc.jyx.pages.BaiWanAnXinLiao;
import com.cpeoc.jyx.pages.Common;
import com.cpeoc.jyx.util.BaseDriver;
import com.cpeoc.jyx.util.WaitUtil;

/**
 * 百萬安心療-重慶
 * @author ken
 * @date 2018年6月14日
 */
@Epic("產品後買")
@Story("百萬安心療-重慶")
public class TestBaiWanAnXinLiao extends BaseDriver{
    
    @Test(dataProvider = "data",groups={"wangjiu"})
    @Description("百萬安心療-重慶-購買流程")
    public void testBaiWanAnXinLiao(String cityName,String disctName,
            String insStartDate,String apltName,String apltCretNo,String apltCellPhone,
            String email,String isrdRelation,String hasSocialSecurity,String province,String expect) {
        
        
        System.out.println("運行測試用例------------------TestBaiWanAnXinLiao");
        
        // 1.點擊城市進入城市選擇頁。點擊C-重慶市
        IndexOp.switchCity(driver, wait,cityName);
        
        // 2.點擊百萬安心醫療,選擇基礎版,點擊購買
        //WaitUtil.waitElementByXpath(wait, BaiWanAnXinLiao.name).click();
        WebElement bwaxl = WaitUtil.waitElementByXpath(wait, BaiWanAnXinLiao.NAME);
        while(true){
            if(bwaxl.isEnabled()){
                break;
            }   
        }
        bwaxl.click();
        
        WaitUtil.waitElementByXpath(wait, Common.BUG).click();
        
        // 3.選擇日期控件直接輸入起保日期
        WaitUtil.waitElementByXpath(wait, Common.INS_START_DATE).click();
        CommonPageOp.datePicker(driver, wait, insStartDate);
        
        // 4.選擇投保地區
        WaitUtil.waitElementByXpath(wait, Common.AREA_FULL_NAME).click();
        String disctXpath = "//*[contains(text(),‘"+disctName+"‘)]";
        WaitUtil.waitElementByXpath(wait, disctXpath).click();
        
        // 5.輸入投保人信息
        WebElement elApltName = WaitUtil.waitElementByXpath(wait, Common.APLT_NAME);
        elApltName.click();
        elApltName.sendKeys(apltName);
        WebElement elApltCretNo = WaitUtil.waitElementByXpath(wait, Common.APLT_CRETNO);
        elApltCretNo.click();
        elApltCretNo.sendKeys(apltCretNo);
        WebElement elApltCellPhone = WaitUtil.waitElementByXpath(wait, Common.APLT_CELLPHONE);
        elApltCellPhone.click();
        elApltCellPhone.sendKeys(apltCellPhone);
        //上滑
        CommonPageOp.swipeUpOnWebview(driver, wait);
        
        WebElement elEmail = WaitUtil.waitElementByXpath(wait, Common.APLT_EMAIL);
        elEmail.click();
        elEmail.sendKeys(email);
        WaitUtil.waitElementByXpath(wait, Common.IS_RELATION).click();
        CommonPageOp.selectRelation(driver, wait, isrdRelation);
        //被保人信息     
        String socialSecurityXpath = "//*[text()=‘"+hasSocialSecurity+"‘]";
        WaitUtil.waitElementByXpath(wait, socialSecurityXpath).click();
        WaitUtil.waitElementByXpath(wait, Common.PROVINCE).click();
        CommonPageOp.chooseProvince(driver, wait, province);
    
        //點擊確定
        WaitUtil.waitElementByXpath(wait, Common.CONFIRM).click();
        
        System.out.println("離開測試用例------------------TestBaiWanAnXinLiao");
    }

    @AfterMethod
    public void backToIndex(){
        System.out.println("AfterMethod-----------------------------------");
        //點擊左上角關閉按鈕,點擊公眾號菜單,進入簡易險
        CommonPageOp.backToIndex(driver, wait);
        
    }
    
    @DataProvider(name = "data")
    public Object[][] data() {
        return new Object[][] {
                { "重慶市" /**城市*/,"彭水苗族土家族自治縣"/**城區*/,"2018-7-10"/**起保日期*/,"陸福鏗"/**投保人姓名*/,
                "450121199010******"/**投保人身份證號*/,"1589990***2"/**投保人手機號*/,"1054057***@qq.com"/**郵箱*/,
                "本人"/**與被保人關系*/,"無"/**有無社保"*/,"重慶市-萬盛區"/**省市*/,"期望結果"},

                };
    }
}

2.5util包

失敗自動截圖。

package com.cpeoc.jyx.util;

import io.appium.java_client.AppiumDriver;
import io.qameta.allure.Attachment;

import org.openqa.selenium.OutputType;
import org.openqa.selenium.WebElement;
import org.testng.IHookCallBack;
import org.testng.IHookable;
import org.testng.ITestResult;
/**
 * 失敗自動截圖監聽類
 * @author ken
 * @date 2018年6月21日
 * @see
 *  繼承testng的接口
 *  使用Allure附件註解
 */
public class AllureReporterListener implements IHookable {

    @Override
    public void run(IHookCallBack callBack, ITestResult testResult) {
        callBack.runTestMethod(testResult);
        if (testResult.getThrowable() != null) {
            try {
                //takeScreenShot(testResult.getMethod().getMethodName());
                takeScreenShotA(testResult);
            } catch (Exception e) {
                e.printStackTrace();
            }
        }
    }

    @Attachment(value = "失敗截圖", type = "image/png")
    private byte[] takeScreenShotA( ITestResult testResult) throws Exception {
        System.out.println("-----------監聽到用例運行失敗,截圖------------");
        BaseDriver b = (BaseDriver) testResult.getInstance();
        AppiumDriver<WebElement> driver = b.getDriver();
        driver.context(Config.NATIVE_CONTEXT);
        System.out.println("-----------切換context到NATIVE_APP---------");
        return driver.getScreenshotAs(OutputType.BYTES);
    }

}

2.6testng.xml

<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE suite SYSTEM "http://testng.org/testng-1.0.dtd">

<suite name="簡易險測試用例集" parallel="tests">
    <listeners>
        <listener class-name="com.cpeoc.jyx.util.AllureReporterListener"/>
    </listeners>
    
    <test name="wangjiu">
        <!-- 這裏是手機信息 -->
        <parameter name="deviceName" value="28a38e6f" />
        <parameter name="platformVersion" value="5.1" />
        <!-- 這裏是要調試的測試類  -->
        <classes>
            <class name="com.cpeoc.jyx.testcases.index.TestCitySelect" />
            <class name="com.cpeoc.jyx.testcases.index.TestCityDG" />
            <class name="com.cpeoc.jyx.testcases.insBuy.TestBaiWanAnXinLiao"/>
        </classes>  
    </test>
    <test name="miaomiao">
        <!-- 這裏是手機信息 -->
        <parameter name="deviceName" value="fe3123d2" />
        <parameter name="platformVersion" value="7.0" />
        <!-- 這裏是要調試的測試類 --> 
        <classes>
            <class name="com.cpeoc.jyx.testcases.index.TestCityQG" />
            <class name="com.cpeoc.jyx.testcases.insBuy.TestBaiWanAnXinLiao"/>      
        </classes>  
    </test>
</suite>

最後

當然,裏邊還有很多具體的業務和代碼實現沒有介紹,只是給有需要的同學一點借鑒,歡迎交流。



作者:dancingking
鏈接:https://www.jianshu.com/p/87f0b13dbfd1
來源:簡書
簡書著作權歸作者所有,任何形式的轉載都請聯系作者獲得授權並註明出處。

基於APPIUM測試微信公眾號的UI自動化測試框架(結合Allure2測試報告框架)