概述
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,'&'); 283 s = s.replace(/</g,'<'); 284 s = s.replace(/>/g,'>'); 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> </th> 633 <th> </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> </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'> </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 </span>%s' % result.success_count) 922 if result.failure_count: 923 status.append(u'<span class="tj failCase">Failure </span>%s' % result.failure_count) 924 if result.error_count: 925 status.append(u'<span class="tj errorCase">Error </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所遇到的程序开发问题。
如果觉得靠谱客网站的内容还不错,欢迎将靠谱客网站推荐给程序员好友。
本图文内容来源于网友提供,作为学习参考使用,或来自网络收集整理,版权属于原作者所有。
发表评论 取消回复