Jmeter(二十九)Jmeter-Question之“Ant集成报告模板优化”
也是在和朋友探讨的时候,发现一个问题,Jmeter在与Ant集成的时候,通常选用的模板是jmeter自带的两个样式表
该自带的样式,节省了大家搭建框架的时间,不需要自己重新写样式,当然也相对简洁;
做接口测试时,我们通常跑的接口有很多,其日志的也是相对比较大的,因此对于一些报错原因、响应报文想查看,便形成了一种障碍;自带的模板不带有查看响应报文的样式,因此需要一种能够直观查看一些类似成功率、失败率以及响应有误能够直接查看的样式模板。
找到一份模板,是copy这位大师的模板。http://www.cnblogs.com/puresoul/p/4808416.html。
样式源码Copy一份。
1 <?xml version="1.0" encoding="GB2312"?> 2 <xsl:stylesheet xmlns:xsl="http://www.w3.org/1999/XSL/Transform" version="1.0"> 3 <xsl:output method="html" indent="no" encoding="UTF-8" doctype-public="-//W3C//DTD HTML 4.01 Transitional//EN" doctype-system="http://www.w3.org/TR/html4/loose.dtd"/> 4 <xsl:strip-space elements="*"/> 5 6 7 <xsl:template name="detail"> 8 <xsl:variable name="allFailureCount" select="count(/testResults/*[attribute::s='false'])" /> 9 10 <xsl:if test="$allFailureCount > 0"> 11 <h2>Failure Detail</h2> 12 13 <xsl:for-each select="/testResults/*[not(@lb = preceding::*/@lb)]"> 14 15 <xsl:variable name="failureCount" select="count(../*[@lb = current()/@lb][attribute::s='false'])" /> 16 17 <xsl:if test="$failureCount > 0"> 18 <h3><xsl:value-of select="@lb" /></h3> 19 20 <table align="center" class="details" border="0" cellpadding="5" cellspacing="2" width="95%"> 21 <tr valign="top"> 22 <th>Response</th> 23 <th>Failure Message</th> 24 </tr> 25 26 <xsl:for-each select="/testResults/*[@lb = current()/@lb][attribute::s='false']"> 27 <tr> 28 <td><xsl:value-of select="@rc | @rs" /> - <xsl:value-of select="@rm" /></td> 29 <td><xsl:value-of select="assertionResult/failureMessage" /></td> 30 </tr> 31 </xsl:for-each> 32 33 </table> 34 </xsl:if> 35 36 </xsl:for-each> 37 </xsl:if> 38 </xsl:template> 39 40 <xsl:template match="/testResults"> 41 <html lang="en"> 42 <head> 43 <meta name="Author" content="shanhe.me"/> 44 <title>JMeter Test Results</title> 45 <style type="text/css"><![CDATA[ 46 47 * { margin: 0; padding: 0 } 48 49 table.details tr th{ 50 color: #ffffff; 51 font-weight: bold; 52 text-align:center; 53 background:#2674a6; 54 line-height:2em; 55 } 56 57 table.details tr:nth-child(odd){background:#FFFFFF;border:1px solid #CCC;line-height:2em;} 58 table.details tr:nth-child(even){background:#EDF3FE;border:1px solid #CCC;line-height:2em;} 59 table.details td{border:1px solid black;} 60 .Failure { 61 font-weight:bold; color:red; 62 } 63 html{ width: 100%; height: 100%; background: #b4b4b4; font-size: 12px } 64 body { width: 95%; height: 95%; margin: 0 auto; } 65 table { border: none; border-collapse: collapse; table-layout: fixed;word-wrap:break-word;word-break:break-all; } 66 #panel-wrap {position:relative;width: 100%;height: 100%;} 67 td { vertical-align: baseline; font-size: 12px } 68 69 #left-panel { position: absolute; left: 0; top: 0; bottom: 0; width: 30%; overflow: auto; background: #dee4ea } 70 #left-panel li.navigation { font-weight: bold; cursor: default; color: #9da8b2; line-height: 18px; background-position: 12px 5px; background-repeat: no-repeat; padding: 0 0 0 25px; background-image: url() } 71 #left-panel li.success { color: #565b60 } 72 #left-panel li.failure { color: red } 73 #left-panel li { list-style: none; color: black; cursor: pointer } 74 #left-panel li.selected { background-repeat: repeat-x; color: white; background: url() } 75 #left-panel div { line-height: 20px; background-position: 25px 3px; background-repeat: no-repeat; padding: 0 0 0 45px } 76 #left-panel div.success { background-image: url() } 77 #left-panel div.failure { background-image: url() } 78 #left-panel div.detail { display: none } 79 #right-panel { position: absolute; right: 0; top: 0; bottom: 0; left: 30%; overflow: auto; background: white } 80 #right-panel .group { font-size: 12px; font-weight: bold; line-height: 16px; padding: 0 0 0 18px; counter-reset: assertion; background-repeat: repeat-x; background-image: url() } 81 #right-panel .zebra { background-repeat: repeat; padding: 0 0 0 18px; background-image: url() } 82 83 #right-panel .data { line-height: 19px; } 84 #right-panel pre.data { white-space: pre } 85 #right-panel tbody.failure { color: red } 86 #right-panel td.key { min-width: 108px } 87 #right-panel td.delimiter { min-width: 18px } 88 #right-panel td.assertion:before { counter-increment: assertion; content: counter(assertion) ". " } 89 #right-panel td.assertion { color: black } 90 #right-panel .trail { border-top: 1px solid #b4b4b4 } 91 92 ]]></style> 93 <script type="text/javascript"><![CDATA[ 94 95 var onclick_li = (function() { 96 var last_selected = null; 97 return function(li) { 98 if( last_selected == li ) 99 return; 100 if( last_selected ) 101 last_selected.className = ""; 102 last_selected = li; 103 last_selected.className = "selected"; 104 document.getElementById("right-panel").innerHTML = last_selected.firstChild.nextSibling.innerHTML; 105 return false; 106 }; 107 })(); 108 109 var patch_timestamp = function() { 110 var spans = document.getElementsByTagName("span"); 111 var len = spans.length; 112 for( var i = 0; i < len; ++i ) { 113 var span = spans[i]; 114 if( "patch_timestamp" == span.className ) 115 span.innerHTML = new Date( parseInt( span.innerHTML ) ); 116 } 117 }; 118 119 var patch_navigation_class = (function() { 120 121 var set_class = function(el, flag) { 122 if(el) { 123 el.className += flag ? " success" : " failure"; 124 } 125 }; 126 127 var traverse = function(el, group_el, flag) { 128 while(1) { 129 if(el) { 130 if(el.className == 'navigation') { 131 set_class(group_el, flag); 132 group_el = el; 133 flag = true; 134 } else { 135 var o = el.firstChild; 136 o = o ? o.className : null; 137 flag = flag ? (o == 'success') : false; 138 } 139 el = el.nextSibling; 140 } else { 141 set_class(group_el, flag); 142 break; 143 } 144 } 145 }; 146 147 return function() { 148 var o = document.getElementById("result-list"); 149 o = o ? o.firstChild : null; 150 if(o) 151 traverse(o, null, true); 152 }; 153 })(); 154 155 window.onload = function() { 156 patch_timestamp(); 157 patch_navigation_class(); 158 var o = document.getElementById("result-list"); 159 o = o ? o.firstChild : null; 160 o = o ? o.nextSibling : null; 161 if(o) 162 onclick_li(o); 163 }; 164 function checkfailure() { 165 if (document.getElementById("bt").innerHTML == "查看失败") { 166 document.getElementById("bt").innerHTML = "查看全部"; 167 var trs = document.getElementsByTagName("table")[1].getElementsByTagName('tr'); 168 for( var i = 1; i < trs.length; i++ ) { 169 var tr = trs[i]; 170 if( "Failure" != tr.className ) 171 tr.style.display = 'none'; 172 } 173 }else if(document.getElementById("bt").innerHTML == "查看全部") { 174 document.getElementById("bt").innerHTML = "查看失败"; 175 var trs = document.getElementsByTagName("table")[1].getElementsByTagName('tr'); 176 for( var i = 1; i < trs.length; i++ ) { 177 var tr = trs[i]; 178 if( "Failure" != tr.className ) 179 tr.style.display = ''; 180 } 181 } 182 }; 183 184 ]]></script> 185 </head> 186 <body> 187 <h2>Summary</h2> 188 <table align="center" class="details" cellpadding="5" cellspacing="2" width="100%" > 189 <tr valign="top"> 190 <th>执行总数</th> 191 <th>成功数</th> 192 <th>失败数</th> 193 <th>成功率</th> 194 <th>Average Time</th> 195 <th>Min Time</th> 196 <th>Max Time</th> 197 </tr> 198 <tr valign="top"> 199 <xsl:variable name="allCount" select="count(/testResults/*)" /> 200 <xsl:variable name="allFailureCount" select="count(/testResults/*[attribute::s='false'])" /> 201 <xsl:variable name="allSuccessCount" select="count(/testResults/*[attribute::s='true'])" /> 202 <xsl:variable name="allSuccessPercent" select="$allSuccessCount div $allCount" /> 203 <xsl:variable name="allTotalTime" select="sum(/testResults/*/@t)" /> 204 <xsl:variable name="allAverageTime" select="$allTotalTime div $allCount" /> 205 <xsl:variable name="allMinTime"> 206 <xsl:call-template name="min"> 207 <xsl:with-param name="nodes" select="/testResults/*/@t" /> 208 </xsl:call-template> 209 </xsl:variable> 210 <xsl:variable name="allMaxTime"> 211 <xsl:call-template name="max"> 212 <xsl:with-param name="nodes" select="/testResults/*/@t" /> 213 </xsl:call-template> 214 </xsl:variable> 215 <xsl:attribute name="class"> 216 <xsl:choose> 217 <xsl:when test="$allFailureCount > 0">Failure</xsl:when> 218 </xsl:choose> 219 </xsl:attribute> 220 <td align="center"> 221 <xsl:value-of select="$allCount" /> 222 </td> 223 <td align="center"> 224 <xsl:value-of select="$allSuccessCount" /> 225 </td> 226 <td align="center"> 227 <xsl:value-of select="$allFailureCount" /> 228 </td> 229 <td align="center"> 230 <xsl:call-template name="display-percent"> 231 <xsl:with-param name="value" select="$allSuccessPercent" /> 232 </xsl:call-template> 233 </td> 234 <td align="center"> 235 <xsl:call-template name="display-time"> 236 <xsl:with-param name="value" select="$allAverageTime" /> 237 </xsl:call-template> 238 </td> 239 <td align="center"> 240 <xsl:call-template name="display-time"> 241 <xsl:with-param name="value" select="$allMinTime" /> 242 </xsl:call-template> 243 </td> 244 <td align="center"> 245 <xsl:call-template name="display-time"> 246 <xsl:with-param name="value" select="$allMaxTime" /> 247 </xsl:call-template> 248 </td> 249 </tr> 250 </table> 251 <button class="button" id="bt" onclick="checkfailure()" style="float:right">查看失败</button> 252 <h2>Pages</h2> 253 <table align="center" class="details" cellpadding="5" cellspacing="2" width="100%"> 254 <tr valign="top"> 255 <th width="30%">URL</th> 256 <th>执行总数</th> 257 <th>失败</th> 258 <th>成功率</th> 259 <th>Average Time</th> 260 <th>Min Time</th> 261 <th>Max Time</th> 262 </tr> 263 <xsl:for-each select="/testResults/*[not(@lb = preceding::*/@lb)]"> 264 <xsl:variable name="label" select="@lb" /> 265 <xsl:variable name="count" select="count(../*[@lb = current()/@lb])" /> 266 <xsl:variable name="failureCount" select="count(../*[@lb = current()/@lb][attribute::s='false'])" /> 267 <xsl:variable name="successCount" select="count(../*[@lb = current()/@lb][attribute::s='true'])" /> 268 <xsl:variable name="successPercent" select="$successCount div $count" /> 269 <xsl:variable name="totalTime" select="sum(../*[@lb = current()/@lb]/@t)" /> 270 <xsl:variable name="averageTime" select="$totalTime div $count" /> 271 <xsl:variable name="minTime"> 272 <xsl:call-template name="min"> 273 <xsl:with-param name="nodes" select="../*[@lb = current()/@lb]/@t" /> 274 </xsl:call-template> 275 </xsl:variable> 276 <xsl:variable name="maxTime"> 277 <xsl:call-template name="max"> 278 <xsl:with-param name="nodes" select="../*[@lb = current()/@lb]/@t" /> 279 </xsl:call-template> 280 </xsl:variable> 281 <tr valign="top"> 282 <xsl:attribute name="class"> 283 <xsl:choose> 284 <xsl:when test="$failureCount > 0">Failure</xsl:when> 285 </xsl:choose> 286 </xsl:attribute> 287 <td align="left"> 288 <xsl:value-of select="$label" /> 289 </td> 290 <td align="center"> 291 <xsl:value-of select="$count" /> 292 </td> 293 <td align="center"> 294 <xsl:value-of select="$failureCount" /> 295 </td> 296 <td align="center"> 297 <xsl:call-template name="display-percent"> 298 <xsl:with-param name="value" select="$successPercent" /> 299 </xsl:call-template> 300 </td> 301 <td align="center"> 302 <xsl:call-template name="display-time"> 303 <xsl:with-param name="value" select="$averageTime" /> 304 </xsl:call-template> 305 </td> 306 <td align="center"> 307 <xsl:call-template name="display-time"> 308 <xsl:with-param name="value" select="$minTime" /> 309 </xsl:call-template> 310 </td> 311 <td align="center"> 312 <xsl:call-template name="display-time"> 313 <xsl:with-param name="value" select="$maxTime" /> 314 </xsl:call-template> 315 </td> 316 </tr> 317 </xsl:for-each> 318 </table> 319 320 <h2>ErrorDetail</h2> 321 <div id="panel-wrap"> 322 <div id="left-panel"> 323 <ol id="result-list"> 324 <!-- 只把失败的生成html --> 325 <xsl:for-each select="*[attribute::s='false']"> 326 <!-- group with the previous sibling --> 327 <xsl:if test="position() = 1 or @tn != preceding-sibling::*[1]/@tn"> 328 <li class="navigation">Thread: <xsl:value-of select="@tn"/></li> 329 </xsl:if> 330 <li onclick="return onclick_li(this);"> 331 <div> 332 <xsl:attribute name="class"> 333 <xsl:choose> 334 <xsl:when test="@s = 'true'">success</xsl:when> 335 <xsl:otherwise>failure</xsl:otherwise> 336 </xsl:choose> 337 </xsl:attribute> 338 <xsl:value-of select="@lb"/> 339 </div><div class="detail"> 340 <div class="group">Sampler</div> 341 <div class="zebra"> 342 <table> 343 <tr><td class="data key">Timestamp</td><td class="data delimiter">:</td><td class="data"><span class="patch_timestamp"><xsl:value-of select="@ts"/></span></td></tr> 344 <tr><td class="data key">Time</td><td class="data delimiter">:</td><td class="data"><xsl:value-of select="@t"/> ms</td></tr> 345 <tr><td class="data key">Latency</td><td class="data delimiter">:</td><td class="data"><xsl:value-of select="@lt"/> ms</td></tr> 346 <tr><td class="data key">Sample Count</td><td class="data delimiter">:</td><td class="data"><xsl:value-of select="@sc"/></td></tr> 347 <tr><td class="data key">Error Count</td><td class="data delimiter">:</td><td class="data"><xsl:value-of select="@ec"/></td></tr> 348 <tr><td class="data key">Response Code</td><td class="data delimiter">:</td><td class="data"><xsl:value-of select="@rc"/></td></tr> 349 <tr><td class="data key">Response Message</td><td class="data delimiter">:</td><td class="data"><xsl:value-of select="@rm"/></td></tr> 350 </table> 351 </div> 352 <div class="trail"></div> 353 <xsl:if test="count(assertionResult) > 0"> 354 <div class="group">Assertion</div> 355 <div class="zebra"> 356 <table> 357 <xsl:for-each select="assertionResult"> 358 <tbody> 359 <xsl:attribute name="class"> 360 <xsl:choose> 361 <xsl:when test="failure = 'true'">failure</xsl:when> 362 <xsl:when test="error = 'true'">failure</xsl:when> 363 </xsl:choose> 364 </xsl:attribute> 365 <tr><td class="data assertion" colspan="3"><xsl:value-of select="name"/></td></tr> 366 <tr><td class="data key">Failure</td><td class="data delimiter">:</td><td class="data"><xsl:value-of select="failure"/></td></tr> 367 <tr><td class="data key">Error</td><td class="data delimiter">:</td><td class="data"><xsl:value-of select="error"/></td></tr> 368 <tr><td class="data key">Failure Message</td><td class="data delimiter">:</td><td class="data"><xsl:value-of select="failureMessage"/></td></tr> 369 </tbody> 370 </xsl:for-each> 371 </table> 372 </div> 373 <div class="trail"></div> 374 </xsl:if> 375 <div class="group">Request</div> 376 <div class="zebra"> 377 <table> 378 <tr><td class="data key">Method/Url</td><td class="data delimiter">:</td><td class="data"><pre class="data"><xsl:value-of select="method"/><xsl:text> </xsl:text><xsl:value-of select="java.net.URL"/></pre></td></tr> 379 <tr><td class="data key">Query String</td><td class="data delimiter">:</td><td class="data"><pre class="data"><xsl:value-of select="queryString"/></pre></td></tr> 380 </table> 381 </div> 382 <div class="trail"></div> 383 <div class="group">Response</div> 384 <div class="zebra"> 385 <table> 386 <tr><td class="data key">Response Data</td><td class="data delimiter">:</td><td class="data"><pre class="data"><xsl:value-of select="responseData"/></pre></td></tr> 387 </table> 388 </div> 389 <div class="trail"></div> 390 </div> 391 </li> 392 </xsl:for-each> 393 </ol> 394 </div> 395 <div id="right-panel"></div> 396 </div> 397 </body> 398 </html> 399 </xsl:template> 400 401 402 <xsl:template name="min"> 403 <xsl:param name="nodes" select="/.." /> 404 <xsl:choose> 405 <xsl:when test="not($nodes)">NaN</xsl:when> 406 <xsl:otherwise> 407 <xsl:for-each select="$nodes"> 408 <xsl:sort data-type="number" /> 409 <xsl:if test="position() = 1"> 410 <xsl:value-of select="number(.)" /> 411 </xsl:if> 412 </xsl:for-each> 413 </xsl:otherwise> 414 </xsl:choose> 415 </xsl:template> 416 417 <xsl:template name="max"> 418 <xsl:param name="nodes" select="/.." /> 419 <xsl:choose> 420 <xsl:when test="not($nodes)">NaN</xsl:when> 421 <xsl:otherwise> 422 <xsl:for-each select="$nodes"> 423 <xsl:sort data-type="number" order="descending" /> 424 <xsl:if test="position() = 1"> 425 <xsl:value-of select="number(.)" /> 426 </xsl:if> 427 </xsl:for-each> 428 </xsl:otherwise> 429 </xsl:choose> 430 </xsl:template> 431 432 <xsl:template name="display-percent"> 433 <xsl:param name="value" /> 434 <xsl:value-of select="format-number($value,'0.00%')" /> 435 </xsl:template> 436 437 <xsl:template name="display-time"> 438 <xsl:param name="value" /> 439 <xsl:value-of select="format-number($value,'0 ms')" /> 440 </xsl:template> 441 442 </xsl:stylesheet>
可以将该段代码写入一个.xsl文件中,在build.xml文件中指向该样式。对了,还有一个需要注意的地方,就是,务必将Build.xml文件的编码格式和该样式的编码格式统一,不然,构建之后会出现中文乱码的情况。
看看生成的报告样式。
该模板完全适用于接口自动化框架中。