iOS 测试 iOS-客户端代码覆盖测试 (Gcov)

李鹏 · 2019年06月03日 · 最后由 叶子 回复于 2023年02月15日 · 7919 次阅读
本帖已被设为精华帖!

👉 本文主要介绍如何对 iOS 客户端项目集成代码覆盖率检测,使用了 Gcov 工具来完成。

* 主要支持 Objective-C(目前貌似不支持 Swift)

* 适合大部分代码为 OC 语言编写的项目。

一、项目集成

1.项目设置

Generate Legacy Test Coverage Files

Instrument Program Flow

把这两项参数设置成 YES,找不到的话请看图(注意细节)

为了能够在真机上把文件取出来,需要配置 plist 文件打开文件共享

该设置允许 iTunes 或者 Xcode、电脑助手等软件看到该 App 的沙盒中的共享文件目录。

2.配置 Gcov

在 AppDelegate.m 的 didFinishLaunchingWithOptions 函数中,加入以下代码:

NSString *covFilePath = [NSHomeDirectory() stringByAppendingPathComponent:@"Documents/xperia_coverage_files"];
 setenv("GCOV_PREFIX", [covFilePath cStringUsingEncoding: NSUTF8StringEncoding], 1);
 setenv("GCOV_PREFIX_STRIP", "1", 1);

第一行代码是设置代码覆盖的结果文件放在 App 沙盒文件目录的位置(Documents 下面新建 xperia_coverage_files 文件夹)

3.在需要采集覆盖率的地方加入代码

ps: 也就是执行完测试之后要走的代码,目的是将代码执行情况刷入文件存储起来。

extern void __gcov_flush(void);
__gcov_flush();

比如在 viewDidload()函数或者 app 准备切到后台的时候来采集。

二、结果数据收集

4.采集 gcda 文件

连接 Xcode,打开 Window -> Devices and Simulators

找到对应设备的对应 App 的 container 内容,进行下载:

下载沙盒文件

下载完 container 后右键显示包内容

5.采集 gcno 文件

打开 Finder,按下 Shift+Command+H,然后

逐步进入:/Users/lipeng/Library/Developer/Xcode/DerivedData/CodeCov-fjkssrrlmpdspdgtdoyyhsuhrocu/Build/Intermediates.noindex/CodeCov.build/Debug-iphoneos/CodeCov.build/Objects-normal/arm64

项目名称可能不同,通过文件夹修改时间也可以确认是在哪个文件夹下。

6.将所有的 gcda 文件和 gcno 文件放到一个文件夹下。

Lcov 的使用

7.如果 Mac 上命令行输入 lcov 提示 command not found,也就是没有安装 lcov

lcov 官网 http://ltp.sourceforge.net/coverage/lcov.php

可以直接通过

brew install lcov

来安装 lcov。

要是连 homebrew 都没了解,那我不解释了。。。🙂

8.合成采集的代码覆盖信息到输出文件

lcov -c -d . -o myGcovResult.info

9.生成 html 文件

当前所在文件夹是放好 gcno 和 gcda 的 CodeCoverageFiles 文件夹。

genhtml -o html myGcovResult.info

结果如图所示:

命令行结果

文件目录截图
打开 html 文件夹中的 index.html

A.全局结果

B.单个文件覆盖率

(温馨提示:点击各个文件名可以进入到详情)

Congratulations !🎖🎖🎖

未完待续:

app 将 gcno 文件打包,gcda 文件打包,上传到服务器。

在服务器上生成对应的代码覆盖结果。


代码 Demo 区,展示几个关键函数

CodeCoverageTool.m 文件的部分内容

#import "CodeCoverageTool.h"

@interface LDCodeCoverage()

@end

@implementation CodeCoverageTool

- (instancetype)init {
    self = [super init];

    if (self) {
        self.mode = CodeCoverageEnabled;
    }

    return self;
}

+ (instancetype)sharedManager {
    static CodeCoverageTool *ct = nil;

    static dispatch_once_t onceToken;
    dispatch_once(&onceToken, ^{
        ct = [[[self class] alloc] init];
    });

    return ct;
}

// 主要生成代码覆盖数据的函数
- (void)flush {
    if (![self enabled]) {
        return ;
    }

  // 确认不是iPhone模拟器,然后设置代码覆盖结果文件的存储路径
#if !TARGET_IPHONE_SIMULATOR
    NSString *covFilePath = [NSHomeDirectory() stringByAppendingPathComponent:@"Documents/xperia_coverage_files"];
    setenv("GCOV_PREFIX", [covFilePath cStringUsingEncoding: NSUTF8StringEncoding], 1);
    setenv("GCOV_PREFIX_STRIP", "13", 1);
#endif

    [self flushCodeCoverage];
}

// 是否开启了代码覆盖开关
- (BOOL)enabled {
    return (self.config && (self.mode != CodeCoverageDisabled));
}

// 文中提到的写入覆盖数据的核心代码,封装成该函数
- (void)flushCodeCoverage {
    extern void __gcov_flush(void);
    __gcov_flush();
}

共收到 8 条回复 时间 点赞

期待楼主完善后面内容!

simple 回复

哈哈哈,产品质量还不达标,没有到代码覆盖提上日程的阶段,自己业余时间搞。

思寒_seveniruby 将本帖设为了精华贴 06月03日 17:41

在需要采集覆盖率的地方插入代码能做个简单的栗子吗😀

当然可以,稍等更新到正文中哈😃
ps: 该函数执行的时机可以是 app 关闭或者切到后台的时候,测试人员可以通过此类操作触发覆盖率生成(以及上传服务器等)。
了解下 iOS app 启动过程中调用的函数顺序即可,也可以咨询下客户端研发。

代码覆盖测试是测试什么?

wumeijun 回复

这问题百度就好了,哈哈哈。
辅助测试来发现测试用例遗漏的逻辑啊。发现开发多余分支等等。

simple [精彩盘点] TesterHome 社区 2019 年 度精华帖 中提及了此贴 12月24日 22:45

请问有遇到过这个问题么?生成 gcda 的时候,报错 profiling: xxxxx.gcda: cannot open: No such file or directory

请教一下,我生成的.info 文件里 SF:/Applications/Xcode.app/Contents/Developer/Platforms/iPhoneOS.platform/Developer/SDKs/iPhoneOS16.1.sdk,这个是通过什么成生的呢,在生成 html 报告的时候总说找不到这行,因为我没有装 OS16.1 的 sdk,我装了 15.0,在哪儿能修改这个设置呢

我自己有的路径
/Applications/Xcode.app/Contents/Developer/Platforms/iPhoneOS.platform/Developer/SDKs/iPhoneOS15.0.sdk

需要 登录 后方可回复, 如果你还没有账号请点击这里 注册