美文网首页编程技术类AndroidAndroid技术
uiautomator2.0+脱离PC运行(apk启动uiaut

uiautomator2.0+脱离PC运行(apk启动uiaut

作者: 一叶一声秋 | 来源:发表于2016-09-21 10:50 被阅读3696次

    效果:

    打开MyTest.apk,点击run uiautomator,就能直接运行你的脚本。

    方案概述:

    • 新建一个Android app工程MyTest,在Activity中添加Button,用于启动脚本

    • 给这个app添加系统签名

    • 在MyTest中新建一个module,命名为MyTestCase,用于编写脚本

    • 使用am instrument命令实现脚本的运行

    1. 新建一个Android应用,命名为MyTest

    新建一个Android应用

    选择Android的版本,并且选择empty Activity,一路next到Finish

    2. 修改activity_main.xml文件如下:

    <?xml version="1.0" encoding="utf-8"?>
    <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
        xmlns:tools="http://schemas.android.com/tools"
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        android:paddingBottom="@dimen/activity_vertical_margin"
        android:paddingLeft="@dimen/activity_horizontal_margin"
        android:paddingRight="@dimen/activity_horizontal_margin"
        android:paddingTop="@dimen/activity_vertical_margin"
        tools:context="cxq.com.mytest.MainActivity">
    
        <Button
            android:onClick="runMyUiautomator"
            android:id="@+id/runBtn"
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:text="run" />
    </RelativeLayout>
    

    3. 修改MainActivity文件如下:

    package cxq.com.mytest;
    
    import android.os.Bundle;
    import android.support.v7.app.AppCompatActivity;
    import android.util.Log;
    import android.view.View;
    import android.widget.Button;
    
    public class MainActivity extends AppCompatActivity {
        private static final String TAG = "MainActivity";
        Button runBtn;
    
        @Override
        protected void onCreate(Bundle savedInstanceState) {
            super.onCreate(savedInstanceState);
            setContentView(R.layout.activity_main);
            runBtn= (Button) findViewById(R.id.runBtn);
        }
    
        /**
         * 点击按钮对应的方法
         * @param v
         */
        public void runMyUiautomator(View v){
            Log.i(TAG, "runMyUiautomator: ");
            new UiautomatorThread().start();
        }
    
        /**
         * 运行uiautomator是个费时的操作,不应该放在主线程,因此另起一个线程运行
         */
        class UiautomatorThread extends Thread {
            @Override
            public void run() {
                super.run();
                String command=generateCommand("cxq.com.testcase", "TestDemo_1", "demo");
               CMDUtils.CMD_Result rs= CMDUtils.runCMD(command,true,true);
                Log.e(TAG, "run: " + rs.error + "-------" + rs.success);
            }
    
            /**
             * 生成命令
             * @param pkgName 包名
             * @param clsName 类名
             * @param mtdName 方法名
             * @return
             */
            public  String generateCommand(String pkgName, String clsName, String mtdName) {
                String command = "am instrument  --user 0 -w -r   -e debug false -e class "
                        + pkgName + "." + clsName + "#" + mtdName + " "
                        + pkgName + ".test/android.support.test.runner.AndroidJUnitRunner";
                Log.e("test1: ", command);
                return command;
            }
        }
    }
    

    其中CMDUtils.java内容如下:

    package cxq.com.mytest;
    
    import android.util.Log;
    
    import java.io.BufferedReader;
    import java.io.InputStreamReader;
    
    /**
     * 执行命令
     */
    public class CMDUtils {
    
        private static final String TAG = "CMDUtils";
    
        public static class CMD_Result {
            public int resultCode;
            public String error;
            public String success;
    
            public CMD_Result(int resultCode, String error, String success) {
                this.resultCode = resultCode;
                this.error = error;
                this.success = success;
            }
    
        }
    
        /**
         * 执行命令
         *
         * @param command         命令
         * @param isShowCommand   是否显示执行的命令
         * @param isNeedResultMsg 是否反馈执行的结果
         * @retrun CMD_Result
         */
        public static CMD_Result runCMD(String command, boolean isShowCommand,
                                        boolean isNeedResultMsg) {
            if (isShowCommand)
                Log.i(TAG, "runCMD:" + command);
            CMD_Result cmdRsult = null;
            int result;
            try {
                Process process = Runtime.getRuntime().exec(command);
                result = process.waitFor();
                if (isNeedResultMsg) {
                    StringBuilder successMsg = new StringBuilder();
                    StringBuilder errorMsg = new StringBuilder();
                    BufferedReader successResult = new BufferedReader(
                            new InputStreamReader(process.getInputStream()));
                    BufferedReader errorResult = new BufferedReader(
                            new InputStreamReader(process.getErrorStream()));
                    String s;
                    while ((s = successResult.readLine()) != null) {
                        successMsg.append(s);
                    }
                    while ((s = errorResult.readLine()) != null) {
                        errorMsg.append(s);
                    }
                    cmdRsult = new CMD_Result(result, errorMsg.toString(),
                            successMsg.toString());
                }
            } catch (Exception e) {
                Log.e(TAG, "run CMD:" + command + " failed");
                e.printStackTrace();
            }
            return cmdRsult;
        }
    
    }
    

    4. 新建一个Module,用于写你的测试脚本

    • 在Android Studio的左侧栏,右击“New -- Module -- Phone&Table Module”,并填写如下信息
    新建一个Module

    next之后,选择add no activity -- finish

    5. 修改mytestcase模块的gradle文件

    • 修改默认的runner,在defaultConfig中添加runner
     defaultConfig {
            applicationId "cxq.com.mytestcast"
            minSdkVersion 23
            targetSdkVersion 23
            versionCode 1
            versionName "1.0"
            testInstrumentationRunner "android.support.test.runner.AndroidJUnitRunner"
        }
    
    • 在dependencies中添加依赖
    dependencies {
        compile fileTree(dir: 'libs', include: ['*.jar'])
        compile 'com.android.support:appcompat-v7:23.3.0'
        compile 'junit:junit:4.12'
        compile 'com.android.support.test:runner:0.4.1'
        compile 'com.android.support.test.uiautomator:uiautomator-v18:2.1.2'
    }
    

    6. 新建你的测试用例TestOne

    新建你的测试用例

    代码如下:

    package cxq.com.mytestcast;
    
    import android.support.test.InstrumentationRegistry;
    import android.support.test.uiautomator.UiDevice;
    import android.support.test.uiautomator.UiObject;
    import android.support.test.uiautomator.UiObjectNotFoundException;
    import android.support.test.uiautomator.UiSelector;
    
    import org.junit.Test;
    import org.junit.runner.RunWith;
    
    @RunWith(AndroidJUnit4.class)
    public class TestOne {
        private UiDevice mDevice;
    
        @Test
        public void demo() throws UiObjectNotFoundException {
            mDevice = UiDevice.getInstance(InstrumentationRegistry.getInstrumentation());
            mDevice.pressHome();
            mDevice.pressHome();
            UiObject x=mDevice.findObject(new UiSelector().text("联系人"));
            x.click();
    
        }
    }
    

    Case说明:

    @RunWith注解,代表使用什么runner
    @Test注解,表示当前的方法为测试方法,同理还有其他的@Before等注解,具体case的写法本文不赘述

    运行一下你的case,测试是否有效,运行方法:点击带有Test注解的方法左侧绿色三角形

    AS运行用例

    此时,run窗口会有如下信息:

    run窗口信息

    如果你只是急于达到效果,而不注重原理,可以跳过原理分析的阅读
    原理分析
    可以看到,窗口中其实运行了3条命令

    • adb push D:\DEVl\MyTest\mytestcast\build\outputs\apk\mytestcast-debug.apk /data/local/tmp/cxq.com.mytestcast
      在我们点击运行之后,AS会自动将用例打包成apk文件,路径为
      $工程目录\build\outputs\apk\工程名-debug.apk
      这个apk文件中就包含着我们的测试用例

    • adb shell pm install -r "/data/local/tmp/cxq.com.mytestcast
      安装这个apk到手机

    • adb shell am instrument -w -r -e debug false -e class cxq.com.mytestcast.TestOne#demo cxq.com.mytestcast.test/android.support.test.runner.AndroidJUnitRunner
      使用AndroidJunitRunner启动你的用例

    通过分析这个过程,就知道AS是怎么把用例跑起来的,仿照这个原理就可以自己实现通过apk调用uiautomator用例,只要让app中的button的响应事件去执行am instrument命令即可,但是由于执行这个命令也是需要权限的,因此需要给app添加系统签名

    7. 给APP添加系统签名

    大致过程如下(具体的细节可以参照我之前写过的文章Android Studio自动生成带系统签名的apk):
    a. 修改App中的manifest.xml文件,添加android:sharedUserId="android.uid.system"

    <?xml version="1.0" encoding="utf-8"?>
    <manifest xmlns:android="http://schemas.android.com/apk/res/android"
        android:sharedUserId="android.uid.system"
        package="cxq.com.mytest">
    

    b. 生成js文件
    c. 使用keytool-importkeypair对jks文件引入系统签名
    d. 配置gradle(app)

    8. 通过app启动uiautomator

    运行MyTest下的app,界面如下:


    运行图

    点击RUN,则回到桌面,并且打开桌面上的联系人,至此,通过apk启动uiautomator教程完毕。

    相关文章

      网友评论

      • 流星雨_50cb:不签名能实现?
      • 彭業:我想问下群主,你是否成功了;
      • 4f69a0512ec2:既然有签名的问题就不能保证 testapp在随意的一个手机上运行,也就无法分享。

        为什么不用 startInstrumentation 呢?
      • King杨:感谢分享,能不能把源码贴出来
      • 残霜丶:不是很明白其中的原理,按照博主的方式无法运行,但是如果自己在androidTest里面写个测试方法后再用命了行方式运行命令时可以的,删除APK的.test里的bask.apk后测试失败。现在想研究下怎么打包一个类似uiautomator.jar的包,通过shell命令来启动。
        00abc348c748:请问一下,你解决了吗,我同样想将我的Uiautomator的测试用例打包成一个jar,然后放到云平台上去测试。在Android Studio下如何生成这个jar呢?
      • 御剑逍遥:那个app签系统签名platform.pk8,platform.x509.pem这两个文件要跟系统的版本对应吗?我重新签名了还是装不上,报Failure [INSTALL_FAILED_SHARED_USER_INCOMPATIBLE]
        一叶一声秋:@高焰 要对应的 不对应就会报兼容性错误
      • chenglansky:我想问下,uiautomator2.0生成apk模式和安卓项目一样么,可不可以用这种方法:http://jingyan.baidu.com/album/2f9b480d9f3c9841cb6cc237.html?picindex=5。我操作了,但是安装失败

      本文标题:uiautomator2.0+脱离PC运行(apk启动uiaut

      本文链接:https://www.haomeiwen.com/subject/nepoettx.html