質(zhì)量系列 - 基于Jacoco的增量覆蓋率實(shí)現(xiàn)與落地

前言

測試團(tuán)隊(duì)在執(zhí)行自動化或者黑盒測試時筛严,希望同時獲取代碼的覆蓋率牺堰,測研團(tuán)隊(duì)由此開發(fā)了第一代自動化覆蓋率平臺雏胃。隨著業(yè)務(wù)迭代掂林,存量代碼越來越多,使用過程中遇到了很多新的問題鼓蜒,例如:

  1. 無法統(tǒng)計(jì)增量代碼覆蓋率痹换,以便量化測試完整度
  2. 不支持合并覆蓋率報(bào)告,多人多環(huán)境協(xié)作測試時無法獲得完整統(tǒng)計(jì)數(shù)據(jù)
  3. 報(bào)告手動生成都弹,以及生成報(bào)告的必要信息也需要人肉收集娇豫,系統(tǒng)間自動化程度低,用戶使用效率低

針對上述的問題畅厢,測試研發(fā)團(tuán)隊(duì)開發(fā)了覆蓋率平臺2.0版本冯痢,實(shí)現(xiàn)了增量代碼覆蓋率,包括定時采樣框杜,自動合并報(bào)告等功能浦楣,以賦能團(tuán)隊(duì)精準(zhǔn)測試能力。

方案設(shè)計(jì)

增量代碼覆蓋率基于Jacoco實(shí)現(xiàn)咪辱,Jacoco是基于JVM虛擬機(jī)的使用最廣的第三方代碼覆蓋率開源工具振劳。我們的設(shè)計(jì)主要針對JacocoCore模塊,Analyze模塊進(jìn)行功能擴(kuò)展油狂,在數(shù)據(jù)分析中加入增量行計(jì)數(shù)邏輯澎迎,以實(shí)現(xiàn)增量覆蓋率統(tǒng)計(jì)庐杨。出于體系建設(shè)考慮,我們集成增量覆蓋率功能到DevOps發(fā)布流程夹供,完善了質(zhì)量量化和風(fēng)險(xiǎn)約束能力,設(shè)計(jì)方案如下:


image

增量覆蓋率實(shí)現(xiàn)方案

Rubik自動化平臺會根據(jù)發(fā)布系統(tǒng)推送的發(fā)布事件自動觸發(fā)定時采樣仁堪,數(shù)據(jù)合并哮洽,另外項(xiàng)目管理系統(tǒng)按一定規(guī)則讀取統(tǒng)計(jì)數(shù)據(jù),并且會對覆蓋率未達(dá)標(biāo)的發(fā)布流程進(jìn)行卡點(diǎn)約束弦聂。

主要功能說明

CodeDiff數(shù)據(jù)解析

增量行數(shù)據(jù)是計(jì)算增量覆蓋率的前提鸟辅,Rubik平臺通過發(fā)布系統(tǒng)獲得被測站點(diǎn)的包版本與生產(chǎn)包版本,調(diào)用GitLabApi獲取差異代碼數(shù)據(jù)莺葫,差異代碼數(shù)據(jù)為純字符串格式匪凉,解析轉(zhuǎn)換為差異行數(shù)據(jù),轉(zhuǎn)換邏輯如下:
/// /* 解析GitDiff數(shù)據(jù) /* @param diff 代碼差異生數(shù)據(jù) /* @return 增量行數(shù)組 /*/ public static int[] parseIncrLines(String diff) { GitDiffHelper helper = new GitDiffHelper(diff); helper.parse(); return helper.newLines; } private void parse(){ if (diff == null || diff.length() == 0) { return; } // 跳過文件信息 nextLineIfMinusFile(); nextLineIfPlusFile(); while (!eof()) { // 解析差異行數(shù)據(jù)塊 parseBlock(); } }

采樣數(shù)據(jù)分析

Jacoco通過各個維度的計(jì)數(shù)器逐層累加實(shí)現(xiàn)捺檬,分別為:

  • 指令計(jì)數(shù)器(CounterImpl)
  • 行計(jì)數(shù)器(LineImpl)
  • 方法計(jì)算節(jié)點(diǎn)(MethodCoverageImpl)
  • 類計(jì)算節(jié)點(diǎn)(ClassCoverageImpl)
  • Package計(jì)算節(jié)點(diǎn)(PackageCoverageImpl)
  • Module計(jì)算節(jié)點(diǎn)(BundleCoverageImpl)
  • 站點(diǎn)計(jì)算節(jié)點(diǎn)(Jacoco未提供再层,可自行實(shí)現(xiàn))

通過從底層指令計(jì)數(shù)器開始逐層累加,最終得到站點(diǎn)級統(tǒng)計(jì)信息堡纬。為實(shí)現(xiàn)增量行統(tǒng)計(jì)聂受,我們將增量行與全量行分開,在計(jì)算節(jié)點(diǎn)父類中(CoverageNodeImpl)中增加了增量行計(jì)數(shù)器烤镐。
public class CoverageNodeImpl implements ICoverageNode{ ... /// /* 全量行計(jì)數(shù)器 // protected CounterImpl lineCounter; ///* /* 增量行計(jì)數(shù)器 /*/ protected CounterImpl diffLineCounter; ...

在原計(jì)數(shù)邏輯中加入增量行計(jì)數(shù)邏輯蛋济,如下:

public class SourceNodeImpl extends CoverageNodeImpl implements ISourceNode{ private LineImpl[] lines; // 由GitDiff計(jì)算得到的差異行數(shù)據(jù) private int[] diffLines; ... // 由行內(nèi)指令計(jì)數(shù)器累加行計(jì)數(shù)器 private void incrementLine(final ICounter instructions, final ICounter branches, final int line){ ensureCapacity(line, line); final LineImpl l = getLine(line); final int oldTotal = l.getInstructionCounter().getTotalCount(); final int oldCovered = l.getInstructionCounter().getCoveredCount(); boolean isDiffLine; if (l == LineImpl.EMPTY) { // 確定是否為增量行 isDiffLine = diffLines != null && Arrays.binarySearch(diffLines, line) >= 0; } else { isDiffLine = l.isDiffLine(); } lines[line - offset] = l.increment(instructions, branches, isDiffLine); // Increment line counter: if (instructions.getTotalCount() > 0) { if (instructions.getCoveredCount() == 0) { if (oldTotal == 0) { lineCounter = lineCounter.increment(CounterImpl.COUNTER_1_0); // 增量行處理邏輯:處理已覆蓋行 if (isDiffLine) { diffLineCounter = diffLineCounter.increment(CounterImpl.COUNTER_1_0); } } } else { if (oldTotal == 0) { lineCounter = lineCounter.increment(CounterImpl.COUNTER_0_1); // 增量行處理邏輯:處理未覆蓋行 if (isDiffLine) { diffLineCounter = diffLineCounter.increment(CounterImpl.COUNTER_0_1); } } else { if (oldCovered == 0) { lineCounter = lineCounter.increment(-1, +1); // 增量行處理邏輯:處理部分覆蓋行 if (isDiffLine) { diffLineCounter = diffLineCounter.increment(-1, +1); } } } } } }

另外行計(jì)數(shù)器中Jacoco通過四維數(shù)組單例,用固定數(shù)量的對象表示8^4(4096)種計(jì)數(shù)情況炮叶,實(shí)現(xiàn)了計(jì)數(shù)緩存碗旅,以提高內(nèi)存使用率,這里增加了增量行標(biāo)志位以區(qū)別全量行計(jì)數(shù)器镜悉,但是Jacoco自身的緩存計(jì)數(shù)器(Fix類)無法適配增量的情況祟辟,所以這里也同樣增加了增量行緩存計(jì)數(shù)器,即DiffFix類积瞒,這里會帶來固定的4096個DiffFix對象的額外開銷川尖,但是對整體性能影響幾乎可以忽略。

public abstract class LineImpl implements ILine{ ... private final boolean isDiffLine; private static final LineImpl[][][][] SINGLETONS = new LineImpl[SINGLETON_INS_LIMIT + 1][][][]; private static final LineImpl[][][][] DIFF_SINGLETONS = new LineImpl[SINGLETON_INS_LIMIT + 1][][][]; static { // 全量行計(jì)數(shù)緩存 for (int i = 0; i <= SINGLETON_INS_LIMIT; i++) { SINGLETONS[i] = new LineImpl[SINGLETON_INS_LIMIT + 1][][]; for (int j = 0; j <= SINGLETON_INS_LIMIT; j++) { SINGLETONS[i][j] = new LineImpl[SINGLETON_BRA_LIMIT + 1][]; for (int k = 0; k <= SINGLETON_BRA_LIMIT; k++) { SINGLETONS[i][j][k] = new LineImpl[SINGLETON_BRA_LIMIT + 1]; for (int l = 0; l <= SINGLETON_BRA_LIMIT; l++) { SINGLETONS[i][j][k][l] = new Fix(i, j, k, l); } } } } // 增量行計(jì)數(shù)緩存 for (int i = 0; i <= SINGLETON_INS_LIMIT; i++) { DIFF_SINGLETONS[i] = new LineImpl[SINGLETON_INS_LIMIT + 1][][]; for (int j = 0; j <= SINGLETON_INS_LIMIT; j++) { DIFF_SINGLETONS[i][j] = new LineImpl[SINGLETON_BRA_LIMIT + 1][]; for (int k = 0; k <= SINGLETON_BRA_LIMIT; k++) { DIFF_SINGLETONS[i][j][k] = new LineImpl[SINGLETON_BRA_LIMIT + 1]; for (int l = 0; l <= SINGLETON_BRA_LIMIT; l++) { DIFF_SINGLETONS[i][j][k][l] = new DiffFix(i, j, k, l); } } } } }

覆蓋率報(bào)告

出于對可讀性的要求茫孔,我們沒有采用Jacoco原生的Html報(bào)告叮喳,而是獨(dú)立開發(fā)了相對更為簡潔的增量/全量報(bào)告,如下:


image

全環(huán)境覆蓋率報(bào)告如下:

image

數(shù)據(jù)合并

測試過程往往經(jīng)過多次發(fā)布缰贝,可能因?yàn)榉峙釡y馍悟,也可能因?yàn)樾迯?fù)缺陷,每次JVM啟動后需要將之前的采樣數(shù)據(jù)合并到下一次采樣數(shù)據(jù)中繼續(xù)累加剩晴,Rubik平臺接收發(fā)布事件并按以下規(guī)則自動合并:

  • 站點(diǎn)發(fā)布新版本前進(jìn)行最后一次采樣
  • 站點(diǎn)發(fā)布新版本中锣咒,健康檢查通過后立即進(jìn)行一次采樣侵状,并且同時開啟定時采樣
  • 任意一次采樣都將進(jìn)行向前自動合并數(shù)據(jù),向前查找規(guī)則為:同一站點(diǎn)毅整,同一環(huán)境趣兄,同一代碼分支的最近一次采樣數(shù)據(jù)

雖然可以在PAones項(xiàng)目管理平臺的發(fā)布工單中查看站點(diǎn)覆蓋率報(bào)告,但想實(shí)時查看站點(diǎn)的增量覆蓋情況悼嫉,用戶可登錄Rubik平臺艇潭,指定自己的測試環(huán)境,就可以方便的看到被測環(huán)境內(nèi)所有站點(diǎn)的增量覆蓋率(按每小時定時采樣戏蔑,也可手動觸發(fā)實(shí)時采樣)蹋凝,從而相對精準(zhǔn)的控制測試進(jìn)度,減少漏測問題总棵。效果如下圖鳍寂。


image

項(xiàng)目實(shí)施中遇到的問題

數(shù)據(jù)合并問題

現(xiàn)象

覆蓋率平臺平均每天需要對400+個站點(diǎn)提供分析服務(wù),另外算上每小時定時采樣情龄,一天完成超過8000次采樣分析以及報(bào)告生成迄汛,在上線運(yùn)行一段時間后發(fā)現(xiàn),偶爾會出現(xiàn)服務(wù)響應(yīng)慢或卡頓刃唤,甚至不可用現(xiàn)象隔心。

分析

針對異常時內(nèi)存分析發(fā)現(xiàn),主要堆積的對象是SessionInfoStore尚胞。


image

SessionInfoStore是Jacoco用來進(jìn)行代碼分析展示的底層類硬霍,包含所有的執(zhí)行類信息,在自動合并過程中Jacoco默認(rèn)對SessionInfo進(jìn)行了累加而非合并笼裳,導(dǎo)致每進(jìn)行一次合并采樣文件數(shù)據(jù)量都會增加30%到50%唯卖,隨著不斷對采樣數(shù)據(jù)合并,加載文件所消耗內(nèi)存急劇增加躬柬,直到并發(fā)加載幾個文件就導(dǎo)致內(nèi)存耗盡拜轨,頻繁觸發(fā)GC,通過復(fù)盤發(fā)現(xiàn)允青,一份采樣文件由最初的10K到問題出現(xiàn)時可以擴(kuò)大到800M到1.5G橄碾。

優(yōu)化方案

通過調(diào)查發(fā)現(xiàn)SessionInfo只在原生Html報(bào)告中需要使用,去掉后不影響自研報(bào)告展示颠锉,也不會破壞Jacoco分析數(shù)據(jù)流程法牲,于是優(yōu)(cu)雅(bao)的將合并數(shù)據(jù)中的對應(yīng)邏輯直接去除,最終解決了這個問題琼掠,修改代碼如下:
/// /* Deserialization of execution data from binary streams. /*/ public class ExecutionDataReader{ ... // Rubik報(bào)告無需合并SessionInfo private void readSessionInfo() throws IOException{ // if (sessionInfoVisitor == null) { // throw new IOException("No session info visitor."); // } // final String id = in.readUTF(); // final long start = in.readLong(); // final long dump = in.readLong(); // sessionInfoVisitor.visitSessionInfo(new SessionInfo(id, start, dump)); } // 合并采樣數(shù)據(jù) private void readExecutionData() throws IOException{ if (executionDataVisitor == null) { throw new IOException("No execution data visitor."); } final long id = in.readLong(); final String name = in.readUTF(); final boolean[] probes = in.readBooleanArray(); executionDataVisitor.visitClassExecution(new ExecutionData(id, name, probes)); }

后續(xù)規(guī)劃

增量覆蓋率為測試結(jié)果量化提供了能力支撐拒垃,一定程度上解決了測試結(jié)果的信任問題,也為測試團(tuán)隊(duì)質(zhì)量分提供了基礎(chǔ)能力瓷蛙,幫助信也研發(fā)中心在Devops體系化建設(shè)上又推進(jìn)了一步悼瓮。接下來效能研發(fā)團(tuán)隊(duì)還將在精準(zhǔn)測試方向上進(jìn)行更多嘗試戈毒,包括自動回歸范圍分析,代碼調(diào)用鏈路等横堡,歡迎大家繼續(xù)關(guān)注埋市。

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市翅萤,隨后出現(xiàn)的幾起案子恐疲,更是在濱河造成了極大的恐慌,老刑警劉巖套么,帶你破解...
    沈念sama閱讀 218,284評論 6 506
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異碳蛋,居然都是意外死亡胚泌,警方通過查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,115評論 3 395
  • 文/潘曉璐 我一進(jìn)店門肃弟,熙熙樓的掌柜王于貴愁眉苦臉地迎上來玷室,“玉大人,你說我怎么就攤上這事笤受∏铉停” “怎么了?”我有些...
    開封第一講書人閱讀 164,614評論 0 354
  • 文/不壞的土叔 我叫張陵箩兽,是天一觀的道長津肛。 經(jīng)常有香客問我,道長汗贫,這世上最難降的妖魔是什么身坐? 我笑而不...
    開封第一講書人閱讀 58,671評論 1 293
  • 正文 為了忘掉前任,我火速辦了婚禮落包,結(jié)果婚禮上部蛇,老公的妹妹穿的比我還像新娘。我一直安慰自己咐蝇,他們只是感情好涯鲁,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,699評論 6 392
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著有序,像睡著了一般抹腿。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上笔呀,一...
    開封第一講書人閱讀 51,562評論 1 305
  • 那天幢踏,我揣著相機(jī)與錄音,去河邊找鬼许师。 笑死房蝉,一個胖子當(dāng)著我的面吹牛僚匆,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播搭幻,決...
    沈念sama閱讀 40,309評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼咧擂,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了檀蹋?” 一聲冷哼從身側(cè)響起松申,我...
    開封第一講書人閱讀 39,223評論 0 276
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎俯逾,沒想到半個月后贸桶,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 45,668評論 1 314
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡桌肴,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,859評論 3 336
  • 正文 我和宋清朗相戀三年皇筛,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片坠七。...
    茶點(diǎn)故事閱讀 39,981評論 1 348
  • 序言:一個原本活蹦亂跳的男人離奇死亡水醋,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出彪置,到底是詐尸還是另有隱情拄踪,我是刑警寧澤,帶...
    沈念sama閱讀 35,705評論 5 347
  • 正文 年R本政府宣布拳魁,位于F島的核電站惶桐,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏的猛。R本人自食惡果不足惜耀盗,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,310評論 3 330
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望卦尊。 院中可真熱鬧叛拷,春花似錦、人聲如沸岂却。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,904評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽躏哩。三九已至署浩,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間扫尺,已是汗流浹背筋栋。 一陣腳步聲響...
    開封第一講書人閱讀 33,023評論 1 270
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留正驻,地道東北人弊攘。 一個月前我還...
    沈念sama閱讀 48,146評論 3 370
  • 正文 我出身青樓抢腐,卻偏偏與公主長得像,于是被迫代替她去往敵國和親襟交。 傳聞我的和親對象是個殘疾皇子迈倍,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,933評論 2 355

推薦閱讀更多精彩內(nèi)容