51Testing软件测试论坛

 找回密码
 (注-册)加入51Testing

QQ登录

只需一步,快速开始

微信登录,快人一步

手机号码,快捷登录

查看: 5020|回复: 6
打印 上一主题 下一主题

[转贴] 使用 Jacoco 实现 Android 端手工测试覆盖率统计

[复制链接]
  • TA的每日心情

    1720761397
  • 签到天数: 1 天

    连续签到: 1 天

    跳转到指定楼层
    1#
    发表于 2017-6-26 10:18:59 | 只看该作者 回帖奖励 |倒序浏览 |阅读模式
    背景

    前段时间在研究手工测试覆盖率问题,尝试将结果记录下来。有什么问题欢迎同学指正. : )
    由于现在单元测试在我们这小公司无法推行,且为了解决新功能测试以及回归测试在手工测试的情况下,即便用例再为详尽,也会存在遗漏的用例。通过统计手工测试覆盖率的数据,可以及时的完善用例。 经过了解准备使用Jacoco完成这个需求.Jacoco是Java Code Coverage的缩写,在统计完成Android代码覆盖率的时候使用的是Jacoco的离线插桩方式,在测试前先对文件进行插桩,在手工测试过程中会生成动态覆盖信息,最后统一对覆盖率进行处理,并生成报告;通过了解现在实现Android覆盖率的方法主要有两种方式,一是通过activity退出的时候添加覆盖率的统计,但是这种情况会修改app的源代码。另外一种是使用的是Android测试框架Instrumentation。这次需求的实现使用的是Instrumentation.。
    实现

    1. 将3个类文件放入项目test文件夹;


    具体各个类的代码如下:
    FinishListener:
    1. package 你的包名;
    2. public interface FinishListener {
    3.     void onActivityFinished();
    4.     void dumpIntermediateCoverage(String filePath);
    5. }
    复制代码
    InstrumentedActivity:
    1. package你的包名;
    2. import 你的启动的activity;
    3. import android.util.Log;

    4. public class InstrumentedActivity extends MainActivity {
    5.     public static String TAG = "InstrumentedActivity";

    6.     private你的包名.test.FinishListener mListener;

    7.     public void setFinishListener(FinishListener listener) {
    8.         mListener = listener;
    9.     }


    10.     @Override
    11.     public void onDestroy() {
    12.         Log.d(TAG + ".InstrumentedActivity", "onDestroy()");
    13.         super.finish();
    14.         if (mListener != null) {
    15.             mListener.onActivityFinished();
    16.         }
    17.     }

    18. }
    复制代码
    JacocoInstrumentation:
    1. package 包名.test;
    2. import java.io.File;
    3. import java.io.FileOutputStream;
    4. import java.io.IOException;
    5. import java.io.OutputStream;
    6. import android.app.Activity;
    7. import android.app.Instrumentation;
    8. import android.content.Intent;
    9. import android.os.Bundle;
    10. import android.os.Looper;
    11. import android.util.Log;

    12. public class JacocoInstrumentation extends Instrumentation implements
    13.         FinishListener {
    14.     public static String TAG = "JacocoInstrumentation:";
    15.     private static String DEFAULT_COVERAGE_FILE_PATH = "/mnt/sdcard/coverage.ec";

    16.     private final Bundle mResults = new Bundle();

    17.     private Intent mIntent;
    18.     private static final boolean LOGD = true;

    19.     private boolean mCoverage = true;

    20.     private String mCoverageFilePath;


    21.     /**
    22.      * Constructor
    23.      */
    24.     public JacocoInstrumentation() {

    25.     }

    26.     @Override
    27.     public void onCreate(Bundle arguments) {
    28.         Log.d(TAG, "onCreate(" + arguments + ")");
    29.         super.onCreate(arguments);
    30.         DEFAULT_COVERAGE_FILE_PATH = getContext().getFilesDir().getPath().toString() + "/coverage.ec";

    31.         File file = new File(DEFAULT_COVERAGE_FILE_PATH);
    32.         if (!file.exists()) {
    33.             try {
    34.                 file.createNewFile();
    35.             } catch (IOException e) {
    36.                 Log.d(TAG, "异常 : " + e);
    37.                 e.printStackTrace();
    38.             }
    39.         }
    40.         if (arguments != null) {
    41.             mCoverageFilePath = arguments.getString("coverageFile");
    42.         }

    43.         mIntent = new Intent(getTargetContext(), InstrumentedActivity.class);
    44.         mIntent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
    45.         start();
    46.     }

    47.     @Override
    48.     public void onStart() {
    49.         if (LOGD)
    50.             Log.d(TAG, "onStart()");
    51.         super.onStart();

    52.         Looper.prepare();
    53.         InstrumentedActivity activity = (InstrumentedActivity) startActivitySync(mIntent);
    54.         activity.setFinishListener(this);
    55.     }

    56.     private void generateCoverageReport() {
    57.         Log.d(TAG, "generateCoverageReport():" + getCoverageFilePath());
    58.         OutputStream out = null;
    59.         try {
    60.             out = new FileOutputStream(getCoverageFilePath(), false);
    61.             Object agent = Class.forName("org.jacoco.agent.rt.RT")
    62.                     .getMethod("getAgent")
    63.                     .invoke(null);

    64.             out.write((byte[]) agent.getClass().getMethod("getExecutionData", boolean.class)
    65.                     .invoke(agent, false));
    66.         } catch (Exception e) {
    67.             Log.d(TAG, e.toString(), e);
    68.         } finally {
    69.             if (out != null) {
    70.                 try {
    71.                     out.close();
    72.                 } catch (IOException e) {
    73.                     e.printStackTrace();
    74.                 }
    75.             }
    76.         }
    77.     }

    78.     private String getCoverageFilePath() {
    79.         if (mCoverageFilePath == null) {
    80.             return DEFAULT_COVERAGE_FILE_PATH;
    81.         } else {
    82.             return mCoverageFilePath;
    83.         }
    84.     }

    85.     private boolean setCoverageFilePath(String filePath){
    86.         if(filePath != null && filePath.length() > 0) {
    87.             mCoverageFilePath = filePath;
    88.             return true;
    89.         }
    90.         return false;
    91.     }


    92.     @Override
    93.     public void onActivityFinished() {
    94.         if (LOGD)
    95.             Log.d(TAG, "onActivityFinished()");
    96.         if (mCoverage) {
    97.             generateCoverageReport();
    98.         }
    99.         finish(Activity.RESULT_OK, mResults);
    100.     }

    101.     @Override
    102.     public void dumpIntermediateCoverage(String filePath){
    103.         // TODO Auto-generated method stub
    104.         if(LOGD){
    105.             Log.d(TAG,"Intermidate Dump Called with file name :"+ filePath);
    106.         }
    107.         if(mCoverage){
    108.             if(!setCoverageFilePath(filePath)){
    109.                 if(LOGD){
    110.                     Log.d(TAG,"Unable to set the given file path:"+filePath+" as dump target.");
    111.                 }
    112.             }
    113.             generateCoverageReport();
    114.             setCoverageFilePath(DEFAULT_COVERAGE_FILE_PATH);
    115.         }
    116.     }

    117. }
    复制代码
    2. 修改build.gradle文件
    增加Jacoco插件,打开覆盖率统计开关,生成日志报告.

    添加的代码内容:
    1. apply plugin: 'jacoco'

    2. jacoco {
    3.     toolVersion = "0.7.9"
    4. }
    5. android {
    6.     buildTypes {
    7.             debug { testCoverageEnabled = true
    8.     /**打开覆盖率统计开关/
    9.         }
    10. }

    11. def coverageSourceDirs = [
    12.         '../app/src/main/java'
    13. ]

    14. task jacocoTestReport(type: JacocoReport) {
    15.     group = "Reporting"
    16.     description = "Generate Jacoco coverage reports after running tests."
    17.     reports {
    18.         xml.enabled = true
    19.         html.enabled = true
    20.     }
    21.     classDirectories = fileTree(
    22.             dir: './build/intermediates/classes/debug',
    23.             excludes: ['**/R*.class',
    24.                        '**/*$InjectAdapter.class',
    25.                        '**/*$ModuleAdapter.class',
    26.                        '**/*$ViewInjector*.class'
    27.             ])
    28.     sourceDirectories = files(coverageSourceDirs)
    29.     executionData = files("$buildDir/outputs/code-coverage/connected/flavors/coverage.ec")

    30.     doFirst {
    31.         new File("$buildDir/intermediates/classes/").eachFileRecurse { file ->
    32.             if (file.name.contains('

    33. )) {
    34.                 file.renameTo(file.path.replace('

    35. , '

    36. ))
    37.             }
    38.         }
    39.     }
    40. }
    41. dependencies {
    42.         compile fileTree(dir: 'libs', include: ['*.jar'])
    43. }
    复制代码


    分享到:  QQ好友和群QQ好友和群 QQ空间QQ空间 腾讯微博腾讯微博 腾讯朋友腾讯朋友
    收藏收藏
    回复

    使用道具 举报

  • TA的每日心情

    1720761397
  • 签到天数: 1 天

    连续签到: 1 天

    2#
     楼主| 发表于 2017-6-26 10:21:07 | 只看该作者
    3. 修改AndroidManifest.xml文件
    添加以及修改部分:
    1. <uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"/>
    2. <activity android:label="InstrumentationActivity"    android:name="包名.test.InstrumentedActivity" />
    3. <instrumentation
    4.     android:handleProfiling="true"
    5.     android:label="CoverageInstrumentation"
    6.     android:name="包名.test.JacocoInstrumentation"
    7.     android:targetPackage="包名"/>
    复制代码

    4. 我们需要通过adb shell am instrument 包名/包名.test.JacocoInstrumentation 启动app;

    5. 进行app手工测试,测试完成后退出App,覆盖率文件会保存在手机/data/data/yourPackageName/files/coverage.ec目录

    6. 导出coverage.ec使用gradle jacocoTestReport分析覆盖率文件并生成html报告

    7. 查看覆盖率html报告


    app\build\reports\jacoco\jacocoTestReport\html目录下看到html报告

    打开index.html,就可以看到具体的覆盖率数据了
    回复 支持 反对

    使用道具 举报

  • TA的每日心情
    奋斗
    3 小时前
  • 签到天数: 2773 天

    连续签到: 4 天

    [LV.Master]测试大本营

    3#
    发表于 2017-6-26 13:18:01 | 只看该作者
    大赞一个!
    目前确实如果有这种方式,统计工作会得到很大的缓解
    回复 支持 反对

    使用道具 举报

  • TA的每日心情
    无聊
    前天 09:07
  • 签到天数: 11 天

    连续签到: 2 天

    [LV.3]测试连长

    4#
    发表于 2017-6-26 15:30:32 | 只看该作者
    感觉还是修改源码更方便,可以不用考虑启动方式
    回复 支持 反对

    使用道具 举报

    该用户从未签到

    5#
    发表于 2017-6-26 15:31:28 | 只看该作者
    赞,不过各个方法一点注释没有,小白表示完全不知道为啥这么写的
    回复 支持 反对

    使用道具 举报

  • TA的每日心情
    无聊
    2024-7-12 13:16
  • 签到天数: 1 天

    连续签到: 1 天

    [LV.1]测试小兵

    6#
    发表于 2017-6-26 15:32:16 | 只看该作者
    测试执行过程中会覆盖安装多次APK,请问覆盖率文件会被覆盖吗?
    回复 支持 反对

    使用道具 举报

  • TA的每日心情

    1720761397
  • 签到天数: 1 天

    连续签到: 1 天

    7#
     楼主| 发表于 2017-6-26 15:32:52 | 只看该作者
    乐哈哈yoyo 发表于 2017-6-26 15:32
    测试执行过程中会覆盖安装多次APK,请问覆盖率文件会被覆盖吗?

    如果怕被覆盖可以设置DEFAULT_COVERAGE_FILE_PATH 保存的路径。因为现在的文件是存到apkfile里面的。如果删除apk,文件也会被删除。
    回复 支持 反对

    使用道具 举报

    本版积分规则

    关闭

    站长推荐上一条 /1 下一条

    小黑屋|手机版|Archiver|51Testing软件测试网 ( 沪ICP备05003035号 关于我们

    GMT+8, 2024-9-21 10:32 , Processed in 0.073544 second(s), 22 queries .

    Powered by Discuz! X3.2

    © 2001-2024 Comsenz Inc.

    快速回复 返回顶部 返回列表