我是靠谱客的博主 糟糕麦片,最近开发中收集的这篇文章主要介绍HTMLTestRunner,觉得挺不错的,现在分享给大家,希望可以做个参考。

概述

   1 #-*- coding: utf-8 -*-
   2 """
   3 A TestRunner for use with the Python unit testing framework. It
   4 generates a HTML report to show the result at a glance.
   5 
   6 The simplest way to use this is to invoke its main method. E.g.
   7 
   8     import unittest
   9     import HTMLTestRunner
  10 
  11     ... define your tests ...
  12 
  13     if __name__ == '__main__':
  14         HTMLTestRunner.main()
  15 
  16 
  17 For more customization options, instantiates a HTMLTestRunner object.
  18 HTMLTestRunner is a counterpart to unittest's TextTestRunner. E.g.
  19 
  20     # output to a file
  21     fp = file('my_report.html', 'wb')
  22     runner = HTMLTestRunner.HTMLTestRunner(
  23                 stream=fp,
  24                 title='My unit test',
  25                 description='This demonstrates the report output by HTMLTestRunner.'
  26                 )
  27 
  28     # Use an external stylesheet.
  29     # See the Template_mixin class for more customizable options
  30     runner.STYLESHEET_TMPL = '<link rel="stylesheet" href="my_stylesheet.css" type="text/css">'
  31 
  32     # run the test
  33     runner.run(my_test_suite)
  34 
  35 
  36 ------------------------------------------------------------------------
  37 Copyright (c) 2004-2007, Wai Yip Tung
  38 All rights reserved.
  39 
  40 Redistribution and use in source and binary forms, with or without
  41 modification, are permitted provided that the following conditions are
  42 met:
  43 
  44 * Redistributions of source code must retain the above copyright notice,
  45   this list of conditions and the following disclaimer.
  46 * Redistributions in binary form must reproduce the above copyright
  47   notice, this list of conditions and the following disclaimer in the
  48   documentation and/or other materials provided with the distribution.
  49 * Neither the name Wai Yip Tung nor the names of its contributors may be
  50   used to endorse or promote products derived from this software without
  51   specific prior written permission.
  52 
  53 THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS
  54 IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
  55 TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A
  56 PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER
  57 OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
  58 EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
  59 PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
  60 PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
  61 LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
  62 NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
  63 SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
  64 """
  65 
  66 # URL: http://tungwaiyip.info/software/HTMLTestRunner.html
  67 
  68 __author__ = "Wai Yip Tung"
  69 __version__ = "0.8.3"
  70 
  71 
  72 """
  73 Change History
  74 Version 0.8.4 by GoverSky
  75 * Add sopport for 3.x
  76 * Add piechart for resultpiechart
  77 * Add Screenshot for selenium_case test
  78 * Add Retry on failed
  79 
  80 Version 0.8.3
  81 * Prevent crash on class or module-level exceptions (Darren Wurf).
  82 
  83 Version 0.8.2
  84 * Show output inline instead of popup window (Viorel Lupu).
  85 
  86 Version in 0.8.1
  87 * Validated XHTML (Wolfgang Borgert).
  88 * Added description of test classes and test cases.
  89 
  90 Version in 0.8.0
  91 * Define Template_mixin class for customization.
  92 * Workaround a IE 6 bug that it does not treat <script> block as CDATA.
  93 
  94 Version in 0.7.1
  95 * Back port to Python 2.3 (Frank Horowitz).
  96 * Fix missing scroll bars in detail log (Podi).
  97 """
  98 
  99 # TODO: color stderr
 100 # TODO: simplify javascript using ,ore than 1 class in the class attribute?
 101 import datetime
 102 
 103 import sys
 104 import unittest
 105 from xml.sax import saxutils
 106 
 107 PY3K = (sys.version_info[0] > 2)
 108 if PY3K:
 109     import io as StringIO
 110 else:
 111     import StringIO
 112 import copy
 113 
 114 # ------------------------------------------------------------------------
 115 # The redirectors below are used to capture output during testing. Output
 116 # sent to sys.stdout and sys.stderr are automatically captured. However
 117 # in some cases sys.stdout is already cached before HTMLTestRunner is
 118 # invoked (e.g. calling logging_demo.basicConfig). In order to capture those
 119 # output, use the redirectors for the cached stream.
 120 #
 121 # e.g.
 122 #   >>> logging_demo.basicConfig(stream=HTMLTestRunner.stdout_redirector)
 123 #   >>>
 124 
 125 class OutputRedirector(object):
 126     """ Wrapper to redirect stdout or stderr """
 127 
 128     def __init__(self, fp):
 129         self.fp = fp
 130 
 131     def write(self, s):
 132         self.fp.write(s)
 133 
 134     def writelines(self, lines):
 135         self.fp.writelines(lines)
 136 
 137     def flush(self):
 138         self.fp.flush()
 139 
 140 
 141 stdout_redirector = OutputRedirector(sys.stdout)
 142 stderr_redirector = OutputRedirector(sys.stderr)
 143 
 144 
 145 # ----------------------------------------------------------------------
 146 # Template
 147 
 148 class Template_mixin(object):
 149     """
 150     Define a HTML template for report customerization and generation.
 151 
 152     Overall structure of an HTML report
 153 
 154     HTML
 155     +------------------------+
 156     |<html>                  |
 157     |  <head>                |
 158     |                        |
 159     |   STYLESHEET           |
 160     |   +----------------+   |
 161     |   |                |   |
 162     |   +----------------+   |
 163     |                        |
 164     |  </head>               |
 165     |                        |
 166     |  <body>                |
 167     |                        |
 168     |   HEADING              |
 169     |   +----------------+   |
 170     |   |                |   |
 171     |   +----------------+   |
 172     |                        |
 173     |   REPORT               |
 174     |   +----------------+   |
 175     |   |                |   |
 176     |   +----------------+   |
 177     |                        |
 178     |   ENDING               |
 179     |   +----------------+   |
 180     |   |                |   |
 181     |   +----------------+   |
 182     |                        |
 183     |  </body>               |
 184     |</html>                 |
 185     +------------------------+
 186     """
 187 
 188     STATUS = {
 189         0: u'通过',
 190         1: u'失败',
 191         2: u'错误',
 192     }
 193 
 194     DEFAULT_TITLE = 'Unit Test Report'
 195     DEFAULT_DESCRIPTION = ''
 196 
 197     # ------------------------------------------------------------------------
 198     # HTML Template
 199 
 200     HTML_TMPL = r"""<?xml version="1.0" encoding="UTF-8"?>
 201 <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Strict//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-strict.dtd">
 202 <html xmlns="http://www.w3.org/1999/xhtml">
 203 <head>
 204     <title>%(title)s</title>
 205     <meta name="generator" content="%(generator)s"/>
 206     <meta http-equiv="Content-Type" content="text/html; charset=UTF-8"/>
 207     %(stylesheet)s
 208 </head>
 209 <body>
 210 <script language="javascript" type="text/javascript">
 211 output_list = Array();
 212 
 213 /* level - 0:Summary; 1:Failed; 2:All */
 214 function showCase(level) {
 215     trs = document.getElementsByTagName("tr");
 216     for (var i = 0; i < trs.length; i++) {
 217         tr = trs[i];
 218         id = tr.id;
 219         if (id.substr(0,2) == 'ft') {
 220             if (level < 1) {
 221                 tr.className = 'hiddenRow';
 222             }
 223             else {
 224                 tr.className = '';
 225             }
 226         }
 227         if (id.substr(0,2) == 'pt') {
 228             if (level > 1) {
 229                 tr.className = '';
 230             }
 231             else {
 232                 tr.className = 'hiddenRow';
 233             }
 234         }
 235     }
 236 }
 237 
 238 
 239 function showClassDetail(cid, count) {
 240     var id_list = Array(count);
 241     var toHide = 1;
 242     for (var i = 0; i < count; i++) {
 243         tid0 = 't' + cid.substr(1) + '.' + (i+1);
 244         tid = 'f' + tid0;
 245         tr = document.getElementById(tid);
 246         if (!tr) {
 247             tid = 'p' + tid0;
 248             tr = document.getElementById(tid);
 249         }
 250         id_list[i] = tid;
 251         if (tr.className) {
 252             toHide = 0;
 253         }
 254     }
 255     for (var i = 0; i < count; i++) {
 256         tid = id_list[i];
 257         if (toHide) {
 258             document.getElementById(tid).className = 'hiddenRow';
 259         }
 260         else {
 261             document.getElementById(tid).className = '';
 262         }
 263     }
 264 }
 265 
 266 
 267 function showTestDetail(div_id){
 268     var details_div = document.getElementById(div_id)
 269     var displayState = details_div.style.display
 270     // alert(displayState)
 271     if (displayState != 'block' ) {
 272         displayState = 'block'
 273         details_div.style.display = 'block'
 274     }
 275     else {
 276         details_div.style.display = 'none'
 277     }
 278 }
 279 
 280 
 281 function html_escape(s) {
 282     s = s.replace(/&/g,'&amp;');
 283     s = s.replace(/</g,'&lt;');
 284     s = s.replace(/>/g,'&gt;');
 285     return s;
 286 }
 287 
 288 function drawCircle(pass, fail, error){ 
 289     var color = ["#6c6","#c60","#c00"];  
 290     var data = [pass,fail,error]; 
 291     var text_arr = ["pass", "fail", "error"];
 292 
 293     var canvas = document.getElementById("circle");  
 294     var ctx = canvas.getContext("2d");  
 295     var startPoint=0;
 296     var width = 25, height = 12;
 297     var posX = 112 * 2 + 20, posY = 30;
 298     var textX = posX + width + 5, textY = posY + 10;
 299     for(var i=0;i<data.length;i++){  
 300         ctx.fillStyle = color[i];  
 301         ctx.beginPath();  
 302         ctx.moveTo(112,84);   
 303         ctx.arc(112,84,84,startPoint,startPoint+Math.PI*2*(data[i]/(data[0]+data[1]+data[2])),false);  
 304         ctx.fill();  
 305         startPoint += Math.PI*2*(data[i]/(data[0]+data[1]+data[2]));  
 306         ctx.fillStyle = color[i];  
 307         ctx.fillRect(posX, posY + 20 * i, width, height);  
 308         ctx.moveTo(posX, posY + 20 * i);  
 309         ctx.font = 'bold 14px';
 310         ctx.fillStyle = color[i];
 311         var percent = text_arr[i] + ":"+data[i];  
 312         ctx.fillText(percent, textX, textY + 20 * i);  
 313 
 314     }
 315 }
 316 
 317 
 318 function show_img(obj) {
 319     var obj1 = obj.nextElementSibling
 320     obj1.style.display='block'
 321     var index = 0;//每张图片的下标,
 322     var len = obj1.getElementsByTagName('img').length;
 323     var imgyuan = obj1.getElementsByClassName('imgyuan')[0]
 324     //var start=setInterval(autoPlay,500);
 325     obj1.οnmοuseοver=function(){//当鼠标光标停在图片上,则停止轮播
 326         clearInterval(start);
 327     }
 328     obj1.οnmοuseοut=function(){//当鼠标光标停在图片上,则开始轮播
 329         start=setInterval(autoPlay,1000);
 330     }    
 331     for (var i = 0; i < len; i++) {
 332         var font = document.createElement('font')
 333         imgyuan.appendChild(font)
 334     }
 335     var lis = obj1.getElementsByTagName('font');//得到所有圆圈
 336     changeImg(0)
 337     var funny = function (i) {
 338         lis[i].onmouseover = function () {
 339             index=i
 340             changeImg(i)
 341         }
 342     }
 343     for (var i = 0; i < lis.length; i++) {
 344         funny(i);
 345     }
 346     
 347     function autoPlay(){
 348         if(index>len-1){
 349             index=0;
 350             clearInterval(start); //运行一轮后停止
 351         }
 352         changeImg(index++);
 353     }
 354     imgyuan.style.width= 25*len +"px";
 355     //对应圆圈和图片同步
 356     function changeImg(index) {
 357         var list = obj1.getElementsByTagName('img');
 358         var list1 = obj1.getElementsByTagName('font');
 359         for (i = 0; i < list.length; i++) {
 360             list[i].style.display = 'none';
 361             list1[i].style.backgroundColor = 'white';
 362         }
 363         list[index].style.display = 'block';
 364         list1[index].style.backgroundColor = 'blue';
 365     }
 366 
 367 }
 368 function hide_img(obj){
 369     obj.parentElement.style.display = "none";
 370     obj.parentElement.getElementsByClassName('imgyuan')[0].innerHTML = "";
 371 }
 372 </script>
 373 <div class="piechart">
 374     <div>
 375         <canvas id="circle" width="350" height="168" </canvas>
 376     </div>
 377 </div>
 378 %(heading)s
 379 %(report)s
 380 %(ending)s
 381 
 382 </body>
 383 </html>
 384 """
 385     # variables: (title, generator, stylesheet, heading, report, ending)
 386 
 387 
 388     # ------------------------------------------------------------------------
 389     # Stylesheet
 390     #
 391     # alternatively use a <link> for external style sheet, e.g.
 392     #   <link rel="stylesheet" href="$url" type="text/css">
 393 
 394     STYLESHEET_TMPL = """
 395 <style type="text/css" media="screen">
 396 body        { font-family: verdana, arial, helvetica, sans-serif; font-size: 80%; }
 397 table       { font-size: 100%; }
 398 pre  { 
 399     white-space: pre-wrap;
 400     word-wrap: break-word;
 401 }
 402 
 403 /* -- heading ---------------------------------------------------------------------- */
 404 h1 {
 405     font-size: 16pt;
 406     color: gray;
 407 }
 408 .heading {
 409     margin-top: 0ex;
 410     margin-bottom: 1ex;
 411 }
 412 
 413 .heading .attribute {
 414     margin-top: 1ex;
 415     margin-bottom: 0;
 416 }
 417 
 418 .heading .description {
 419     margin-top: 4ex;
 420     margin-bottom: 6ex;
 421 }
 422 
 423 /* -- css div popup ------------------------------------------------------------------------ */
 424 a.popup_link {
 425 }
 426 
 427 a.popup_link:hover {
 428     color: red;
 429 }
 430 .img{
 431     height: 100%;
 432     border-collapse: collapse;
 433     border: 2px solid #777;
 434 }
 435 
 436 .screenshots {
 437     z-index: 100;
 438     position:absolute;
 439     height: 80%;
 440     left: 50%;
 441     top: 50%;
 442     transform: translate(-50%,-50%);
 443     display: none;
 444 }
 445 
 446 .imgyuan{
 447     height: 20px;
 448     border-radius: 12px;
 449     background-color: red;
 450     padding-left: 13px;
 451     margin: 0 auto;
 452     position: relative;
 453     top: -40px;
 454     background-color: rgba(1, 150, 0, 0.3);
 455 }
 456 .imgyuan font{
 457     border:1px solid white;
 458     width:11px; 
 459     height:11px;
 460     border-radius:50%;
 461     margin-right: 9px;
 462     margin-top: 4px;
 463     display: block;
 464     float: left;
 465     background-color: white;
 466 }
 467 .close_shots {
 468     background-image: url();
 469     background-size: 22px 22px;
 470     -moz-background-size: 22px 22px;
 471     background-repeat: no-repeat;
 472     position: absolute;
 473     top: 5px;
 474     right: 5px;
 475     height: 22px;
 476     z-index: 99;
 477     width: 22px;
 478 }
 479 .popup_window {
 480     display: none;
 481     position: relative;
 482     left: 0px;
 483     top: 0px;
 484     padding: 10px;
 485     background-color: #E6E6D6;
 486     font-family: "Lucida Console", "Courier New", Courier, monospace;
 487     text-align: left;
 488     font-size: 8pt;
 489 }
 490 
 491 }
 492 /* -- report ------------------------------------------------------------------------ */
 493 #show_detail_line {
 494     margin-top: 3ex;
 495     margin-bottom: 1ex;
 496 }
 497 
 498 #result_table {
 499     margin: 1em 0;
 500     width: 100%;
 501     overflow: hidden;
 502     background:  #FFF;
 503     color:  #024457;
 504     border-radius:   10px;
 505     border: 1px solid #167F92;
 506 }
 507 #result_table th {
 508       border: 1px solid #FFFFFF;
 509       background-color: #167F92;
 510       color: #FFF;
 511       padding: 0.5em;
 512       &:first-child {
 513         display: table-cell;
 514         text-align: center;
 515       }
 516       &:nth-child(2) {
 517         display: table-cell;
 518         span {display:none;}
 519         &:after {content:attr(data-th);}
 520       }
 521       @media (min-width: 480px) {
 522         &:nth-child(2) {
 523           span {display: block;}
 524           &:after {display: none;}
 525         }
 526       }
 527     }
 528 #result_table td {
 529        word-wrap: break-word;
 530       max-width: 7em;
 531       padding: 0.3em;
 532       &:first-child {
 533         display: table-cell;
 534         text-align: center;
 535       }
 536       @media (min-width: 400px) {
 537         border: 1px solid #D9E4E6;
 538       }
 539     }
 540 
 541 #result_table  th, td {
 542       margin: .5em 1em;
 543       @media (min-width: 400px) {
 544         display: table-cell;
 545         padding: 1em;
 546       }
 547     }
 548 
 549 #total_row  { font-weight: bold; }
 550 .passClass  { background-color: #6c6;  !important ;}
 551 .failClass  { background-color: #c60;  !important ;}
 552 .errorClass { background-color: #c00; !important ; }
 553 .passCase   { color: #6c6; }
 554 .failCase   { color: #c60; font-weight: bold; }
 555 .errorCase  { color: #c00; font-weight: bold; }
 556 tr[id^=pt]  td { background-color: rgba(73,204,144,.3) !important ; }
 557 tr[id^=ft]  td { background-color: rgba(252,161,48,.3) !important; }
 558 tr[id^=et]  td { background-color: rgba(249,62,62,.3) !important ; }
 559 .hiddenRow  { display: none; }
 560 .testcase   { margin-left: 2em; }
 561 
 562 /* -- ending ---------------------------------------------------------------------- */
 563 #ending {
 564 }
 565 
 566 
 567 .piechart{  
 568     position:absolute;  ;
 569     top:20px;  
 570     left:300px; 
 571     width: 200px;
 572     float: left;
 573     display:  inline;
 574 }
 575 
 576 
 577 </style>
 578 """
 579 
 580     # ------------------------------------------------------------------------
 581     # Heading
 582     #
 583 
 584     HEADING_TMPL = """<div class='heading'>
 585 <h1>%(title)s</h1>
 586 %(parameters)s
 587 <p class='description'>%(description)s</p>
 588 </div>
 589 
 590 """  # variables: (title, parameters, description)
 591 
 592     HEADING_ATTRIBUTE_TMPL = """<p class='attribute'><strong>%(name)s:</strong> %(value)s</p>
 593 """  # variables: (name, value)
 594 
 595     # ------------------------------------------------------------------------
 596     # Report
 597     #
 598 
 599     REPORT_TMPL = """
 600 <p id='show_detail_line'>显示
 601 <a href='javascript:showCase(0)'>概要</a>
 602 <a href='javascript:showCase(1)'>失败</a>
 603 <a href='javascript:showCase(2)'>所有</a>
 604 </p>
 605 
 606 <table id='result_table'>
 607 <colgroup>
 608 <col align='left' />
 609 <col align='right' />
 610 <col align='right' />
 611 <col align='right' />
 612 <col align='right' />
 613 <col align='right' />
 614 <col align='right' />
 615 </colgroup>
 616 <tr id='header_row'>
 617     <th>测试组/测试用例</th>
 618     <th>总数</th>
 619     <th>通过</th>
 620     <th>失败</th>
 621     <th>错误</th>
 622     <th>视图</th>
 623     <th>错误截图</th>
 624 </tr>
 625 %(test_list)s
 626 <tr id='total_row'>
 627     <th>统计</th>
 628     <th>%(count)s</th>
 629     <th>%(Pass)s</th>
 630     <th>%(fail)s</th>
 631     <th>%(error)s</th>
 632     <th>&nbsp;</th>
 633     <th>&nbsp;</th>
 634 </tr>
 635 </table>
 636 <script>
 637     showCase(1);
 638     drawCircle(%(Pass)s, %(fail)s, %(error)s);
 639 </script>
 640 """
 641     # variables: (test_list, count, Pass, fail, error)
 642 
 643     REPORT_CLASS_TMPL = r"""
 644 <tr class='%(style)s'>
 645     <td>%(desc)s</td>
 646     <td>%(count)s</td>
 647     <td>%(Pass)s</td>
 648     <td>%(fail)s</td>
 649     <td>%(error)s</td>
 650     <td><a href="javascript:showClassDetail('%(cid)s',%(count)s)">详情</a></td>
 651     <td>&nbsp;</td>
 652 </tr>
 653 """  # variables: (style, desc, count, Pass, fail, error, cid)
 654 
 655     REPORT_TEST_WITH_OUTPUT_TMPL = r"""
 656 <tr id='%(tid)s' class='%(Class)s'>
 657     <td ><div class='testcase'>%(desc)s</div></td>
 658     <td colspan='5' align='center'>
 659 
 660     <!--css div popup start-->
 661     <span class='status %(style)s'>
 662     <a class="popup_link" οnfοcus='this.blur();' href="javascript:showTestDetail('div_%(tid)s')" >
 663         %(status)s</a></span>
 664 
 665     <div id='div_%(tid)s' class="popup_window">
 666         <div style='text-align: right; color:red;cursor:pointer'>
 667         <a οnfοcus='this.blur();' οnclick="document.getElementById('div_%(tid)s').style.display = 'none' " >
 668            [x]</a>
 669         </div>
 670         <pre>
 671         %(script)s
 672         </pre>
 673     </div>
 674     <!--css div popup end-->
 675 
 676     </td>
 677     <td>%(img)s</td>
 678 </tr>
 679 """  # variables: (tid, Class, style, desc, status,img)
 680 
 681     REPORT_TEST_NO_OUTPUT_TMPL = r"""
 682 <tr id='%(tid)s' class='%(Class)s'>
 683     <td><div class='testcase'>%(desc)s</div></td>
 684     <td colspan='5' align='center'><span class='status %(style)s'>%(status)s</span></td>
 685     <td>%(img)s</td>
 686 </tr>
 687 """  # variables: (tid, Class, style, desc, status,img)
 688 
 689     REPORT_TEST_OUTPUT_TMPL = r"""
 690 %(id)s: %(output)s
 691 """  # variables: (id, output)
 692 
 693 
 694     IMG_TMPL = r"""
 695         <a href="#"  οnclick="show_img(this)">显示截图</a>
 696     <div align="center" class="screenshots"  style="display:none">
 697         <a class="close_shots"  href="#"   οnclick="hide_img(this)"></a>
 698         %(imgs)s
 699         <div class="imgyuan"></div>
 700     </div>
 701     """
 702     # ------------------------------------------------------------------------
 703     # ENDING
 704     #
 705 
 706     ENDING_TMPL = """<div id='ending'>&nbsp;</div>"""
 707 
 708     # -------------------- The end of the Template class -------------------
 709 
 710     def __getattribute__(self, item):
 711         value = object.__getattribute__(self, item)
 712         if PY3K:
 713             return value
 714         else:
 715             if isinstance(value, str):
 716                 return value.decode("utf-8")
 717             else:
 718                 return value
 719 
 720 
 721 TestResult = unittest.TestResult
 722 
 723 
 724 class _TestResult(TestResult):
 725     # note: _TestResult is a pure representation of results.
 726     # It lacks the output and reporting ability compares to unittest._TextTestResult.
 727 
 728     def __init__(self, verbosity=1, retry=0,save_last_try=True):
 729         TestResult.__init__(self)
 730         self.stdout0 = None
 731         self.stderr0 = None
 732         self.success_count = 0
 733         self.failure_count = 0
 734         self.error_count = 0
 735         self.verbosity = verbosity
 736 
 737         # result is a list of result in 4 tuple
 738         # (
 739         #   result code (0: success; 1: fail; 2: error),
 740         #   TestCase object,
 741         #   Test output (byte string),
 742         #   stack trace,
 743         # )
 744         self.result = []
 745         self.retry = retry
 746         self.trys = 0
 747         self.status = 0
 748         self.save_last_try = save_last_try
 749         self.outputBuffer = StringIO.StringIO()
 750 
 751     def startTest(self, test):
 752         test.imgs = []
 753         # test.imgs = getattr(test, "imgs", [])
 754         TestResult.startTest(self, test)
 755         self.outputBuffer.seek(0)
 756         self.outputBuffer.truncate()
 757         stdout_redirector.fp = self.outputBuffer
 758         stderr_redirector.fp = self.outputBuffer
 759         self.stdout0 = sys.stdout
 760         self.stderr0 = sys.stderr
 761         sys.stdout = stdout_redirector
 762         sys.stderr = stderr_redirector
 763 
 764     def complete_output(self):
 765         """
 766         Disconnect output redirection and return buffer.
 767         Safe to call multiple times.
 768         """
 769         if self.stdout0:
 770             sys.stdout = self.stdout0
 771             sys.stderr = self.stderr0
 772             self.stdout0 = None
 773             self.stderr0 = None
 774         return self.outputBuffer.getvalue()
 775 
 776     def stopTest(self, test):
 777         # Usually one of addSuccess, addError or addFailure would have been called.
 778         # But there are some path in unittest that would bypass this.
 779         # We must disconnect stdout in stopTest(), which is guaranteed to be called.
 780         if self.retry:
 781             if self.status == 1:
 782                 self.trys += 1
 783                 if self.trys <= self.retry:
 784                     if self.save_last_try:
 785                         t = self.result.pop(-1)
 786                         if t[0]==1:
 787                             self.failure_count-=1
 788                         else:
 789                             self.error_count -= 1
 790                     test=copy.copy(test)
 791                     sys.stderr.write("Retesting... ")
 792                     sys.stderr.write(str(test))
 793                     sys.stderr.write('..%d n' % self.trys)
 794                     doc = test._testMethodDoc or ''
 795                     if doc.find('_retry')!=-1:
 796                         doc = doc[:doc.find('_retry')]
 797                     desc ="%s_retry:%d" %(doc, self.trys)
 798                     if not PY3K:
 799                         if isinstance(desc, str):
 800                             desc = desc.decode("utf-8")
 801                     test._testMethodDoc = desc
 802                     test(self)
 803                 else:
 804                     self.status = 0
 805                     self.trys = 0
 806         self.complete_output()
 807 
 808     def addSuccess(self, test):
 809         self.success_count += 1
 810         self.status = 0
 811         TestResult.addSuccess(self, test)
 812         output = self.complete_output()
 813         self.result.append((0, test, output, ''))
 814         if self.verbosity > 1:
 815             sys.stderr.write('ok ')
 816             sys.stderr.write(str(test))
 817             sys.stderr.write('n')
 818         else:
 819             sys.stderr.write('.')
 820 
 821     def addError(self, test, err):
 822         self.error_count += 1
 823         self.status = 1
 824         TestResult.addError(self, test, err)
 825         _, _exc_str = self.errors[-1]
 826         output = self.complete_output()
 827         self.result.append((2, test, output, _exc_str))
 828         if not getattr(test, "driver",""):
 829             pass
 830         else:
 831             try:
 832                 driver = getattr(test, "driver")
 833                 test.imgs.append(driver.get_screenshot_as_base64())
 834             except Exception:
 835                 pass
 836         if self.verbosity > 1:
 837             sys.stderr.write('E  ')
 838             sys.stderr.write(str(test))
 839             sys.stderr.write('n')
 840         else:
 841             sys.stderr.write('E')
 842 
 843     def addFailure(self, test, err):
 844         self.failure_count += 1
 845         self.status = 1
 846         TestResult.addFailure(self, test, err)
 847         _, _exc_str = self.failures[-1]
 848         output = self.complete_output()
 849         self.result.append((1, test, output, _exc_str))
 850         if not getattr(test, "driver",""):
 851             pass
 852         else:
 853             try:
 854                 driver = getattr(test, "driver")
 855                 test.imgs.append(driver.get_screenshot_as_base64())
 856             except Exception as e:
 857                 pass
 858         if self.verbosity > 1:
 859             sys.stderr.write('F  ')
 860             sys.stderr.write(str(test))
 861             sys.stderr.write('n')
 862         else:
 863             sys.stderr.write('F')
 864 
 865 
 866 class HTMLTestRunner(Template_mixin):
 867     def __init__(self, stream=sys.stdout, verbosity=2, title=None, description=None, retry=0,save_last_try=False):
 868         self.stream = stream
 869         self.retry = retry
 870         self.save_last_try=save_last_try
 871         self.verbosity = verbosity
 872         if title is None:
 873             self.title = self.DEFAULT_TITLE
 874         else:
 875             self.title = title
 876         if description is None:
 877             self.description = self.DEFAULT_DESCRIPTION
 878         else:
 879             self.description = description
 880 
 881         self.startTime = datetime.datetime.now()
 882 
 883     def run(self, test):
 884         "Run the given test case or test suite."
 885         result = _TestResult(self.verbosity, self.retry, self.save_last_try)
 886         test(result)
 887         self.stopTime = datetime.datetime.now()
 888         self.generateReport(test, result)
 889         if PY3K:
 890             # for python3
 891             # print('nTime Elapsed: %s' % (self.stopTime - self.startTime),file=sys.stderr)
 892             output = 'nTime Elapsed: %s' % (self.stopTime - self.startTime)
 893             sys.stderr.write(output)
 894         else:
 895             print >> sys.stderr, 'nTime Elapsed: %s' % (self.stopTime - self.startTime)
 896         return result
 897 
 898     def sortResult(self, result_list):
 899         # unittest does not seems to run in any particular order.
 900         # Here at least we want to group them together by class.
 901         rmap = {}
 902         classes = []
 903         for n, t, o, e in result_list:
 904             cls = t.__class__
 905             if not cls in rmap:
 906                 rmap[cls] = []
 907                 classes.append(cls)
 908             rmap[cls].append((n, t, o, e))
 909         r = [(cls, rmap[cls]) for cls in classes]
 910         return r
 911 
 912     def getReportAttributes(self, result):
 913         """
 914         Return report attributes as a list of (name, value).
 915         Override this to add custom attributes.
 916         """
 917         startTime = str(self.startTime)[:19]
 918         duration = str(self.stopTime - self.startTime)
 919         status = []
 920         if result.success_count:
 921             status.append(u'<span class="tj passCase">Pass&nbsp</span>%s'   % result.success_count)
 922         if result.failure_count:
 923             status.append(u'<span class="tj failCase">Failure&nbsp</span>%s'   % result.failure_count)
 924         if result.error_count:
 925             status.append(u'<span class="tj errorCase">Error&nbsp</span>%s'   % result.error_count)
 926         if status:
 927             status = ' '.join(status)
 928         else:
 929             status = 'none'
 930         return [
 931             (u'开始时间', startTime),
 932             (u'耗时', duration),
 933             (u'状态', status),
 934         ]
 935 
 936     def generateReport(self, test, result):
 937         report_attrs = self.getReportAttributes(result)
 938         generator = 'HTMLTestRunner %s' % __version__
 939         stylesheet = self._generate_stylesheet()
 940         heading = self._generate_heading(report_attrs)
 941         report = self._generate_report(result)
 942         ending = self._generate_ending()
 943         output = self.HTML_TMPL % dict(
 944             title=saxutils.escape(self.title),
 945             generator=generator,
 946             stylesheet=stylesheet,
 947             heading=heading,
 948             report=report,
 949             ending=ending,
 950         )
 951         if PY3K:
 952             self.stream.write(output.encode())
 953         else:
 954             self.stream.write(output.encode('utf8'))
 955 
 956     def _generate_stylesheet(self):
 957         return self.STYLESHEET_TMPL
 958 
 959     def _generate_heading(self, report_attrs):
 960         a_lines = []
 961         for name, value in report_attrs:
 962             line = self.HEADING_ATTRIBUTE_TMPL % dict(
 963                 name=name,
 964                 value=value,
 965             )
 966             a_lines.append(line)
 967         heading = self.HEADING_TMPL % dict(
 968             title=saxutils.escape(self.title),
 969             parameters=''.join(a_lines),
 970             description=saxutils.escape(self.description),
 971         )
 972         return heading
 973 
 974     def _generate_report(self, result):
 975         rows = []
 976         sortedResult = self.sortResult(result.result)
 977         for cid, (cls, cls_results) in enumerate(sortedResult):
 978             # subtotal for a class
 979             np = nf = ne = 0
 980             for n, t, o, e in cls_results:
 981                 if n == 0:
 982                     np += 1
 983                 elif n == 1:
 984                     nf += 1
 985                 else:
 986                     ne += 1
 987 
 988             # format class description
 989             if cls.__module__ == "__main__":
 990                 name = cls.__name__
 991             else:
 992                 name = "%s.%s" % (cls.__module__, cls.__name__)
 993             doc = cls.__doc__ and cls.__doc__.split("n")[0] or ""
 994             desc = doc and '%s: %s' % (name, doc) or name
 995             if not PY3K:
 996                 if isinstance(desc, str):
 997                     desc = desc.decode("utf-8")
 998 
 999             row = self.REPORT_CLASS_TMPL % dict(
1000                 style=ne > 0 and 'errorClass' or nf > 0 and 'failClass' or 'passClass',
1001                 desc=desc,
1002                 count=np + nf + ne,
1003                 Pass=np,
1004                 fail=nf,
1005                 error=ne,
1006                 cid='c%s' % (cid + 1),
1007             )
1008             rows.append(row)
1009 
1010             for tid, (n, t, o, e) in enumerate(cls_results):
1011                 self._generate_report_test(rows, cid, tid, n, t, o, e)
1012 
1013         report = self.REPORT_TMPL % dict(
1014             test_list=u''.join(rows),
1015             count=str(result.success_count + result.failure_count + result.error_count),
1016             Pass=str(result.success_count),
1017             fail=str(result.failure_count),
1018             error=str(result.error_count),
1019         )
1020         return report
1021 
1022     def _generate_report_test(self, rows, cid, tid, n, t, o, e):
1023         # e.g. 'pt1.1', 'ft1.1', etc
1024         has_output = bool(o or e)
1025         tid = (n == 0 and 'p' or 'f') + 't%s.%s' % (cid + 1, tid + 1)
1026         name = t.id().split('.')[-1]
1027         if self.verbosity > 1:
1028             doc = t._testMethodDoc or ''
1029         else:
1030             doc = ""
1031 
1032         desc = doc and ('%s: %s' % (name, doc)) or name
1033         if not PY3K:
1034             if isinstance(desc, str):
1035                 desc = desc.decode("utf-8")
1036         tmpl = has_output and self.REPORT_TEST_WITH_OUTPUT_TMPL or self.REPORT_TEST_NO_OUTPUT_TMPL
1037 
1038         # o and e should be byte string because they are collected from stdout and stderr?
1039         if isinstance(o, str):
1040             # uo = unicode(o.encode('string_escape'))
1041             if PY3K:
1042                 uo = o
1043             else:
1044                 uo = o.decode('utf-8', 'ignore')
1045         else:
1046             uo = o
1047         if isinstance(e, str):
1048             # ue = unicode(e.encode('string_escape'))
1049             if PY3K:
1050                 ue = e
1051             elif e.find("Error") != -1 or e.find("Exception") != -1:
1052                 es = e.decode('utf-8', 'ignore').split('n')
1053                 es[-2] = es[-2].decode('unicode_escape')
1054                 ue = u"n".join(es)
1055             else:
1056                 ue = e.decode('utf-8', 'ignore')
1057         else:
1058             ue = e
1059 
1060         script = self.REPORT_TEST_OUTPUT_TMPL % dict(
1061             id=tid,
1062             output=saxutils.escape(uo + ue),
1063         )
1064         if getattr(t,'imgs',[]):
1065             # 判断截图列表,如果有则追加
1066             tmp = u""
1067             for i, img in enumerate(t.imgs):
1068                 if i==0:
1069                     tmp+=""" <img src="https://img-blog.csdnimg.cn/2022010619551766918.jpg" style="display: block;" class="img"/>n""" % img
1070                 else:
1071                     tmp+=""" <img src="https://img-blog.csdnimg.cn/2022010619551766918.jpg" style="display: none;" class="img"/>n""" % img
1072             imgs = self.IMG_TMPL % dict(imgs=tmp)
1073         else:
1074             imgs = u"""无截图"""
1075 
1076         row = tmpl % dict(
1077             tid=tid,
1078             Class=(n == 0 and 'hiddenRow' or 'none'),
1079             style=n == 2 and 'errorCase' or (n == 1 and 'failCase' or 'passCase'),
1080             desc=desc,
1081             script=script,
1082             status=self.STATUS[n],
1083             img=imgs,
1084         )
1085         rows.append(row)
1086         if not has_output:
1087             return
1088 
1089     def _generate_ending(self):
1090         return self.ENDING_TMPL
1091 
1092 
1093 ##############################################################################
1094 # Facilities for running tests from the command line
1095 ##############################################################################
1096 
1097 # Note: Reuse unittest.TestProgram to launch test. In the future we may
1098 # build our own launcher to support more specific command line
1099 # parameters like test title, CSS, etc.
1100 class TestProgram(unittest.TestProgram):
1101     """
1102     A variation of the unittest.TestProgram. Please refer to the base
1103     class for command line parameters.
1104     """
1105 
1106     def runTests(self):
1107         # Pick HTMLTestRunner as the default test runner.
1108         # base class's testRunner parameter is not useful because it means
1109         # we have to instantiate HTMLTestRunner before we know self.verbosity.
1110         if self.testRunner is None:
1111             self.testRunner = HTMLTestRunner(verbosity=self.verbosity)
1112         unittest.TestProgram.runTests(self)
1113 
1114 
1115 main = TestProgram
1116 
1117 ##############################################################################
1118 # Executing this module from the command line
1119 ##############################################################################
1120 
1121 if __name__ == "__main__":
1122     main(module=None)

 

转载于:https://www.cnblogs.com/jayson-0425/p/10037090.html

最后

以上就是糟糕麦片为你收集整理的HTMLTestRunner的全部内容,希望文章能够帮你解决HTMLTestRunner所遇到的程序开发问题。

如果觉得靠谱客网站的内容还不错,欢迎将靠谱客网站推荐给程序员好友。

本图文内容来源于网友提供,作为学习参考使用,或来自网络收集整理,版权属于原作者所有。
点赞(70)

评论列表共有 0 条评论

立即
投稿
返回
顶部