Unittest - HTMLTestRunner.py(新版)

报告样式效果:

 

 

报告源码:HTMLTestRunner.py

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

 

posted @ 2021-01-19 20:25  Juno3550  阅读(2252)  评论(0编辑  收藏  举报