增量代码的覆盖率

我们每次修改代码大多数都是增量的,而通过lcov生成的代码覆盖率报告都是全量的,不方便查看,所以写了一个脚本,增量了增量部分的数据。脚本如下:

#!/usr/bin/python2
######################################################################
# Purpose:  根据git提交记录,修改代码覆盖率的html,增加增量的数据
# Author:   rodinluo
######################################################################
__doc__ = '''
USAGE:
python ./incre_lconv.py <since>..<until> <check_dir> <lcov_html_dir><project_name> 
example:
python incre_lcov.py 8b2628dfe839b6086e74721c93469b0cf18720e7..HEAD '["test1dir","test2dir"]'  /data/ projectname
since:git 开始的版本
util:git 结束的版本
check_dir:关注的代码目录
lcov_html_dir:已经通过lcov生成的html根目录
project_name:项目名
本脚本应该在project_name目录执行,project_name目录包含test1dir\test2dir,/data下面是project_name/*
'''
import sys, os, re
import json
import subprocess as commands
from pprint import *
from bs4 import *

DEBUG = 0

class UTCover(object) :
    def __init__(self, since_until, check_dirs, lcov_dir, root_dir) :
        self.since, self.until = since_until.split('..')
        self.check_dirs = json.loads(check_dirs)
        self.lcov_dir = lcov_dir
        self.root_dir = root_dir

    def get_src(self):
        satus, output = commands.getstatusoutput("git diff --name-only %s %s" %(self.since, self.until))
        if DEBUG: print(output)
        src_files = [f for f in output.split('\n')
                        for m in self.check_dirs if m in f
                        if os.path.splitext(f)[1][1:] in ['c', 'cpp']]
        if DEBUG: pprint(src_files)
        return src_files

    def get_change(self, src_files):
        changes = {}
        for f in src_files:
            satus, output = commands.getstatusoutput("git log --oneline %s..%s %s | awk '{print $1}'" %(self.since, self.until, f))
            commits = output.split('\n')
            cmd = "git blame %s | grep -E '(%s)' | awk  -F' *|)' '{print $6}'" %(f, '|'.join(commits))
            satus, lines = commands.getstatusoutput(cmd)
            changes[f] = [ int(i) for i in lines.split('\n') if i.isdigit() ]

        if DEBUG: pprint(changes)
        return changes

    def update_html(self, f, lines):
        hf = os.path.join(self.lcov_dir, self.root_dir, f + '.gcov.html')
        print(hf)
        if not os.path.exists(hf):
            return None

        try:
            html = open(hf, "r", encoding="utf-8")
            handle = html.read()
        except UnicodeDecodeError:
            print("decode error:" + hf)
            return None
        soup = BeautifulSoup(handle,'lxml')
        a = soup.findAll("a", {"name":True})
        lineNoCov = 0
        lineCov = 0
        for i in range(len(a)):
            if int(a[i]["name"]) in lines:
                a[i].span["class"][0] = ("updateLineNum")
                span = a[i].span.findNextSibling()
                if span:
                    if span["class"][0] == "lineNoCov":
                        lineNoCov = lineNoCov + 1
                    else:
                        if span["class"][0] == "lineCov":
                            lineCov = lineCov + 1

        headerCovTableEntry = soup.findAll("td", { "class" : "headerCovTableEntry"}, limit = 2)
        if not headerCovTableEntry:
            print("css.class headerCovTableEntry not found")
            return None
        contents0 = str(lineCov) + "  -  " + headerCovTableEntry[0].contents[0]
        headerCovTableEntry[0].contents[0].replaceWith(contents0)

        contents1 = str(lineNoCov + lineCov) + "  -  " + headerCovTableEntry[1].contents[0]
        headerCovTableEntry[1].contents[0].replaceWith(contents1)

        headerCovTableEntryX = soup.findAll("td", { "class" : ["headerCovTableEntryLo", "headerCovTableEntryMed", "headerCovTableEntryHi"]}, limit = 1)
        if not headerCovTableEntryX :
            print("css.class headerCovTableEntry* not found")
            return None
        
        totalLine = lineNoCov + lineCov
        if totalLine == 0:
            totalLine = 1
        contents2 = str(round(lineCov / totalLine, 3)  * 100) + "%  -  " + headerCovTableEntryX[0].contents[0]
        headerCovTableEntryX[0].contents[0].replaceWith(contents2)

        with open(hf, 'w',encoding="utf-8") as fp:
            fp.write(soup.prettify())

    def check(self):
        with open(self.lcov_dir + "/gcov.css", 'a+') as f:
            f.write('''
/* Source code view: line update number format */
span.updateLineNum    
{    
    background-color: #33CC00;    
}
                    ''')
        src_files = self.get_src()
        changes = self.get_change(src_files)
        for f, lines in changes.items():
            self.update_html(f, lines)

if len(sys.argv) == 1:
    print (__doc__)
    sys.exit(0)

ret = UTCover(*sys.argv[1:]).check()
sys.exit(ret)

效果如下,其中最左边的行号绿底部分表示的是本次修改的代码行,最上面的Lines中,Hit的左边是本次修改代码的覆盖行数,右边是总的覆盖行数,Total的左边是本次修改代码的行数,右边是总的行数,Coverge的左边是本次修改代码的覆盖率,右边是总的覆盖率:


screenshot-20220401-100604.png
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 220,884评论 6 513
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 94,212评论 3 395
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 167,351评论 0 360
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 59,412评论 1 294
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 68,438评论 6 397
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 52,127评论 1 308
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,714评论 3 420
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 39,636评论 0 276
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 46,173评论 1 319
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 38,264评论 3 339
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 40,402评论 1 352
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 36,073评论 5 347
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,763评论 3 332
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 32,253评论 0 23
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 33,382评论 1 271
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 48,749评论 3 375
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 45,403评论 2 358

推荐阅读更多精彩内容