日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区

您的位置:首頁技術文章
文章詳情頁

Python unittest如何生成HTMLTestRunner模塊

瀏覽:25日期:2022-07-12 08:24:50

生成 HTMLTestRunner 模塊

unittest 里面是不能生成 html 格式報告的,需要導入一個第三方的模塊:HTMLTestRunner

方法1.這個模塊下載不能通過 pip 安裝了,只能下載后手動導入,下載地址:

http://tungwaiyip.info/software/HTMLTestRunner.html

方法2.在 python 安裝文件的 Lib 目錄下新增文件 HTMLTestRunner.py

兩種模板如下,建議使用第一種(第一種模板更加美觀)

文件內容如下:

(1)第一種模板

# -*- coding: utf-8 -*-'''A TestRunner for use with the Python unit testing framework. Itgenerates a HTML report to show the result at a glance.The simplest way to use this is to invoke its main method. E.g. import unittest import HTMLTestRunner ... define your tests ... if __name__ == ’__main__’: HTMLTestRunner.main()For more customization options, instantiates a HTMLTestRunner object.HTMLTestRunner is a counterpart to unittest’s TextTestRunner. E.g. # output to a file fp = file(’my_report.html’, ’wb’) runner = HTMLTestRunner.HTMLTestRunner(stream=fp,title=’My unit test’,description=’This demonstrates the report output by HTMLTestRunner.’) # Use an external stylesheet. # See the Template_mixin class for more customizable options runner.STYLESHEET_TMPL = ’<link rel='stylesheet' href='http://www.b3g6.com/bcjs/my_stylesheet.css' rel='external nofollow' rel='external nofollow' type='text/css'>’ # run the test runner.run(my_test_suite)------------------------------------------------------------------------Copyright (c) 2004-2007, Wai Yip TungAll rights reserved.Redistribution and use in source and binary forms, with or withoutmodification, are permitted provided that the following conditions aremet:* Redistributions of source code must retain the above copyright notice, this list of conditions and the following disclaimer.* Redistributions in binary form must reproduce the above copyright notice, this list of conditions and the following disclaimer in the documentation and/or other materials provided with the distribution.* Neither the name Wai Yip Tung nor the names of its contributors may be used to endorse or promote products derived from this software without specific prior written permission.THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS 'ASIS' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITEDTO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR APARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNEROR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, ORPROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OFLIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDINGNEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THISSOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.'''# URL: http://tungwaiyip.info/software/HTMLTestRunner.html__author__ = 'Wai Yip Tung'__version__ = '0.9.1''''Change HistoryVersion 0.9.1* 用Echarts添加執行情況統計圖 (灰藍)Version 0.9.0* 改成Python 3.x (灰藍)Version 0.8.3* 使用 Bootstrap稍加美化 (灰藍)* 改為中文 (灰藍)Version 0.8.2* Show output inline instead of popup window (Viorel Lupu).Version in 0.8.1* Validated XHTML (Wolfgang Borgert).* Added description of test classes and test cases.Version in 0.8.0* Define Template_mixin class for customization.* Workaround a IE 6 bug that it does not treat <script> block as CDATA.Version in 0.7.1* Back port to Python 2.3 (Frank Horowitz).* Fix missing scroll bars in detail log (Podi).'''# TODO: color stderr# TODO: simplify javascript using ,ore than 1 class in the class attribute?import datetimeimport sysimport ioimport timeimport unittestfrom xml.sax import saxutilsimport getpass# ------------------------------------------------------------------------# The redirectors below are used to capture output during testing. Output# sent to sys.stdout and sys.stderr are automatically captured. However# in some cases sys.stdout is already cached before HTMLTestRunner is# invoked (e.g. calling logging.basicConfig). In order to capture those# output, use the redirectors for the cached stream.## e.g.# >>> logging.basicConfig(stream=HTMLTestRunner.stdout_redirector)# >>>class OutputRedirector(object): ''' Wrapper to redirect stdout or stderr ''' def __init__(self, fp): self.fp = fp def write(self, s): self.fp.write(s) def writelines(self, lines): self.fp.writelines(lines) def flush(self): self.fp.flush()stdout_redirector = OutputRedirector(sys.stdout)stderr_redirector = OutputRedirector(sys.stderr)# ----------------------------------------------------------------------# Templateclass Template_mixin(object): ''' Define a HTML template for report customerization and generation. Overall structure of an HTML report HTML +------------------------+ |<html> | | <head>| | | | STYLESHEET | | +----------------+ | | || | | +----------------+ | | | | </head>| | | | <body>| | | | HEADING | | +----------------+ | | || | | +----------------+ | | | | REPORT| | +----------------+ | | || | | +----------------+ | | | | ENDING| | +----------------+ | | || | | +----------------+ | | | | </body>| |</html> | +------------------------+ ''' STATUS = { 0: u’通過’, 1: u’失敗’, 2: u’錯誤’, } DEFAULT_TITLE = ’Unit Test Report’ DEFAULT_DESCRIPTION = ’’ # ------------------------------------------------------------------------ # HTML Template HTML_TMPL = r'''<?xml version='1.0' encoding='UTF-8'?><!DOCTYPE html PUBLIC '-//W3C//DTD XHTML 1.0 Strict//EN' 'http://www.w3.org/TR/xhtml1/DTD/xhtml1-strict.dtd'><html xmlns='http://www.w3.org/1999/xhtml'><head> <title>%(title)s</title> <meta name='generator' content='%(generator)s'/> <meta http-equiv='Content-Type' content='text/html; charset=UTF-8'/> <link rel='external nofollow' rel='stylesheet'> <script src='https://cdn.bootcss.com/echarts/3.8.5/echarts.common.min.js'></script> <!-- <script type='text/javascript' src='http://www.b3g6.com/bcjs/js/echarts.common.min.js'></script> --> %(stylesheet)s</head><body> <script language='javascript' type='text/javascript'><!-- output_list = Array(); /* level - 0:Summary; 1:Failed; 2:All */ function showCase(level) { trs = document.getElementsByTagName('tr'); for (var i = 0; i < trs.length; i++) { tr = trs[i]; id = tr.id; if (id.substr(0,2) == ’ft’) {if (level < 1) { tr.className = ’hiddenRow’;}else { tr.className = ’’;} } if (id.substr(0,2) == ’pt’) {if (level > 1) { tr.className = ’’;}else { tr.className = ’hiddenRow’;} } } } function showClassDetail(cid, count) { var id_list = Array(count); var toHide = 1; for (var i = 0; i < count; i++) { tid0 = ’t’ + cid.substr(1) + ’.’ + (i+1); tid = ’f’ + tid0; tr = document.getElementById(tid); if (!tr) {tid = ’p’ + tid0;tr = document.getElementById(tid); } id_list[i] = tid; if (tr.className) {toHide = 0; } } for (var i = 0; i < count; i++) { tid = id_list[i]; if (toHide) {document.getElementById(’div_’+tid).style.display = ’none’document.getElementById(tid).className = ’hiddenRow’; } else {document.getElementById(tid).className = ’’; } } } function showTestDetail(div_id){ var details_div = document.getElementById(div_id) var displayState = details_div.style.display // alert(displayState) if (displayState != ’block’ ) { displayState = ’block’ details_div.style.display = ’block’ } else { details_div.style.display = ’none’ } } function html_escape(s) { s = s.replace(/&/g,’&’); s = s.replace(/</g,’<’); s = s.replace(/>/g,’>’); return s; } /* obsoleted by detail in <div> function showOutput(id, name) { var w = window.open('', //url name, 'resizable,scrollbars,status,width=800,height=450'); d = w.document; d.write('<pre>'); d.write(html_escape(output_list[id])); d.write('n'); d.write('<a href=’javascript:window.close()’>close</a>n'); d.write('</pre>n'); d.close(); } */ --></script> <div id='div_base'> %(heading)s %(report)s %(ending)s %(chart_script)s </div></body></html>''' # variables: (title, generator, stylesheet, heading, report, ending, chart_script) ECHARTS_SCRIPT = ''' <script type='text/javascript'> // 基于準備好的dom,初始化echarts實例 var myChart = echarts.init(document.getElementById(’chart’)); // 指定圖表的配置項和數據 var option = { title : {text: ’測試執行情況’,x:’center’ }, tooltip : {trigger: ’item’,formatter: '{a} <br/>{b} : {c} (2rprchx%%)' }, color: [’#95b75d’, ’grey’, ’#b64645’], legend: {orient: ’vertical’,left: ’left’,data: [’通過’,’失敗’,’錯誤’] }, series : [{ name: ’測試執行情況’, type: ’pie’, radius : ’60%%’, center: [’50%%’, ’60%%’], data:[ {value:%(Pass)s, name:’通過’}, {value:%(fail)s, name:’失敗’}, {value:%(error)s, name:’錯誤’} ], itemStyle: { emphasis: { shadowBlur: 10, shadowOffsetX: 0, shadowColor: ’rgba(0, 0, 0, 0.5)’ } }} ] }; // 使用剛指定的配置項和數據顯示圖表。 myChart.setOption(option); </script> ''' # variables: (Pass, fail, error) # ------------------------------------------------------------------------ # Stylesheet # # alternatively use a <link> for external style sheet, e.g. # <link rel='stylesheet' href='http://www.b3g6.com/bcjs/$url' rel='external nofollow' rel='external nofollow' type='text/css'> STYLESHEET_TMPL = '''<style type='text/css' media='screen'> body { font-family: Microsoft YaHei,Consolas,arial,sans-serif; font-size: 80%; } table { font-size: 100%; } pre { white-space: pre-wrap;word-wrap: break-word; } /* -- heading ---------------------------------------------------------------------- */ h1 { font-size: 16pt; color: gray; } .heading { margin-top: 0ex; margin-bottom: 1ex; } .heading .attribute { margin-top: 1ex; margin-bottom: 0; } .heading .description { margin-top: 2ex; margin-bottom: 3ex; } /* -- css div popup ------------------------------------------------------------------------ */ a.popup_link { } a.popup_link:hover { color: red; } .popup_window { display: none; position: relative; left: 0px; top: 0px; /*border: solid #627173 1px; */ padding: 10px; /*background-color: #E6E6D6; */ font-family: 'Lucida Console', 'Courier New', Courier, monospace; text-align: left; font-size: 8pt; /* width: 500px;*/ } } /* -- report ------------------------------------------------------------------------ */ #show_detail_line { margin-top: 3ex; margin-bottom: 1ex; } #result_table { width: 99%; } #header_row { font-weight: bold; color: #303641; background-color: #ebebeb; } #total_row { font-weight: bold; } .passClass { background-color: #bdedbc; } .failClass { background-color: #ffefa4; } .errorClass { background-color: #ffc9c9; } .passCase { color: #6c6; } .failCase { color: #FF6600; font-weight: bold; } .errorCase { color: #c00; font-weight: bold; } .hiddenRow { display: none; } .testcase { margin-left: 2em; } /* -- ending ---------------------------------------------------------------------- */ #ending { } #div_base {position:absolute;top:0%;left:5%;right:5%;width: auto;height: auto;margin: -15px 0 0 0; }</style>''' # ------------------------------------------------------------------------ # Heading # HEADING_TMPL = ''' <div class=’page-header’> <h1>%(title)s</h1> %(parameters)s </div> <div style='float: left;width:50%%;'><p class=’description’>%(description)s</p></div> <div style='width:50%%;height:400px;float:left;'></div>''' # variables: (title, parameters, description) HEADING_ATTRIBUTE_TMPL = '''<p class=’attribute’><strong>%(name)s:</strong> %(value)s</p>''' # variables: (name, value) # ------------------------------------------------------------------------ # Report # REPORT_TMPL = u''' <div class='btn-group btn-group-sm'> <button onclick=’javascript:showCase(0)’>總結</button> <button onclick=’javascript:showCase(1)’>失敗</button> <button onclick=’javascript:showCase(2)’>全部</button> </div> <p></p> <table id=’result_table’ class='table table-bordered'> <colgroup> <col align=’left’ /> <col align=’right’ /> <col align=’right’ /> <col align=’right’ /> <col align=’right’ /> <col align=’right’ /> </colgroup> <tr id=’header_row’> <td>測試套件/測試用例</td> <td>總數</td> <td>通過</td> <td>失敗</td> <td>錯誤</td> <td>查看</td> </tr> %(test_list)s <tr id=’total_row’> <td>總計</td> <td>%(count)s</td> <td>%(Pass)s</td> <td>%(fail)s</td> <td>%(error)s</td> <td> </td> </tr> </table>''' # variables: (test_list, count, Pass, fail, error) REPORT_CLASS_TMPL = u''' <tr class=’%(style)s’> <td>%(desc)s</td> <td>%(count)s</td> <td>%(Pass)s</td> <td>%(fail)s</td> <td>%(error)s</td> <td><a href='javascript:showClassDetail(’%(cid)s’,%(count)s)' rel='external nofollow' rel='external nofollow' >詳情</a></td> </tr>''' # variables: (style, desc, count, Pass, fail, error, cid) REPORT_TEST_WITH_OUTPUT_TMPL = r'''<tr id=’%(tid)s’ class=’%(Class)s’> <td class=’%(style)s’><div class=’testcase’>%(desc)s</div></td> <td colspan=’5’ align=’center’> <!--css div popup start--> <a onfocus=’this.blur();’ href='javascript:showTestDetail(’div_%(tid)s’)' rel='external nofollow' rel='external nofollow' > %(status)s</a> <div id=’div_%(tid)s’ class='popup_window'> <pre>%(script)s</pre> </div> <!--css div popup end--> </td></tr>''' # variables: (tid, Class, style, desc, status) REPORT_TEST_NO_OUTPUT_TMPL = r'''<tr id=’%(tid)s’ class=’%(Class)s’> <td class=’%(style)s’><div class=’testcase’>%(desc)s</div></td> <td colspan=’5’ align=’center’>%(status)s</td></tr>''' # variables: (tid, Class, style, desc, status) REPORT_TEST_OUTPUT_TMPL = r'''%(id)s: %(output)s''' # variables: (id, output) # ------------------------------------------------------------------------ # ENDING # ENDING_TMPL = '''<div id=’ending’> </div>'''# -------------------- The end of the Template class -------------------TestResult = unittest.TestResultclass _TestResult(TestResult): # note: _TestResult is a pure representation of results. # It lacks the output and reporting ability compares to unittest._TextTestResult. def __init__(self, verbosity=1): TestResult.__init__(self) self.stdout0 = None self.stderr0 = None self.success_count = 0 self.failure_count = 0 self.error_count = 0 self.verbosity = verbosity # result is a list of result in 4 tuple # ( # result code (0: success; 1: fail; 2: error), # TestCase object, # Test output (byte string), # stack trace, # ) self.result = [] self.subtestlist = [] def startTest(self, test): TestResult.startTest(self, test) # just one buffer for both stdout and stderr self.outputBuffer = io.StringIO() stdout_redirector.fp = self.outputBuffer stderr_redirector.fp = self.outputBuffer self.stdout0 = sys.stdout self.stderr0 = sys.stderr sys.stdout = stdout_redirector sys.stderr = stderr_redirector def complete_output(self): ''' Disconnect output redirection and return buffer. Safe to call multiple times. ''' if self.stdout0: sys.stdout = self.stdout0 sys.stderr = self.stderr0 self.stdout0 = None self.stderr0 = None return self.outputBuffer.getvalue() def stopTest(self, test): # Usually one of addSuccess, addError or addFailure would have been called. # But there are some path in unittest that would bypass this. # We must disconnect stdout in stopTest(), which is guaranteed to be called. self.complete_output() def addSuccess(self, test): if test not in self.subtestlist: self.success_count += 1 TestResult.addSuccess(self, test) output = self.complete_output() self.result.append((0, test, output, ’’)) if self.verbosity > 1:sys.stderr.write(’ok ’)sys.stderr.write(str(test))sys.stderr.write(’n’) else:sys.stderr.write(’.’) def addError(self, test, err): self.error_count += 1 TestResult.addError(self, test, err) _, _exc_str = self.errors[-1] output = self.complete_output() self.result.append((2, test, output, _exc_str)) if self.verbosity > 1: sys.stderr.write(’E ’) sys.stderr.write(str(test)) sys.stderr.write(’n’) else: sys.stderr.write(’E’) def addFailure(self, test, err): self.failure_count += 1 TestResult.addFailure(self, test, err) _, _exc_str = self.failures[-1] output = self.complete_output() self.result.append((1, test, output, _exc_str)) if self.verbosity > 1: sys.stderr.write(’F ’) sys.stderr.write(str(test)) sys.stderr.write(’n’) else: sys.stderr.write(’F’) def addSubTest(self, test, subtest, err): if err is not None: if getattr(self, ’failfast’, False):self.stop() if issubclass(err[0], test.failureException):self.failure_count += 1errors = self.failureserrors.append((subtest, self._exc_info_to_string(err, subtest)))output = self.complete_output()self.result.append((1, test, output + ’nSubTestCase Failed:n’ + str(subtest), self._exc_info_to_string(err, subtest)))if self.verbosity > 1: sys.stderr.write(’F ’) sys.stderr.write(str(subtest)) sys.stderr.write(’n’)else: sys.stderr.write(’F’) else:self.error_count += 1errors = self.errorserrors.append((subtest, self._exc_info_to_string(err, subtest)))output = self.complete_output()self.result.append( (2, test, output + ’nSubTestCase Error:n’ + str(subtest), self._exc_info_to_string(err, subtest)))if self.verbosity > 1: sys.stderr.write(’E ’) sys.stderr.write(str(subtest)) sys.stderr.write(’n’)else: sys.stderr.write(’E’) self._mirrorOutput = True else: self.subtestlist.append(subtest) self.subtestlist.append(test) self.success_count += 1 output = self.complete_output() self.result.append((0, test, output + ’nSubTestCase Pass:n’ + str(subtest), ’’)) if self.verbosity > 1:sys.stderr.write(’ok ’)sys.stderr.write(str(subtest))sys.stderr.write(’n’) else:sys.stderr.write(’.’)class HTMLTestRunner(Template_mixin): def __init__(self, stream=sys.stdout, verbosity=1,, tester=getpass.getuser(), description='測試詳情如下:'): self.stream = stream self.verbosity = verbosity self.tester = tester ''' verbosity: =1的時候 默認值為1,不限制完整結果,即單個用例成功輸出’.’,失敗輸出’F’,錯誤輸出’E’ =0的時候。不輸出信息 =2的時候,需要打印詳細的返回信息 stream:測試報告寫入文件的存儲區域 title:測試報告的主題 tester:默認獲取本機用戶名 description:測試報告的描述 ''' if title is None: self.title = self.DEFAULT_TITLE else: self.title = title if description is None: self.description = self.DEFAULT_DESCRIPTION else: self.description = description self.startTime = datetime.datetime.now() def run(self, test): 'Run the given test case or test suite.' result = _TestResult(self.verbosity) test(result) self.stopTime = datetime.datetime.now() self.generateReport(test, result) print(’nTime Elapsed: %s’ % (self.stopTime - self.startTime), file=sys.stderr) return result def sortResult(self, result_list): # unittest does not seems to run in any particular order. # Here at least we want to group them together by class. rmap = {} classes = [] for n, t, o, e in result_list: cls = t.__class__ if cls not in rmap:rmap[cls] = []classes.append(cls) rmap[cls].append((n, t, o, e)) r = [(cls, rmap[cls]) for cls in classes] return r def getReportAttributes(self, result): ''' Return report attributes as a list of (name, value). Override this to add custom attributes. ''' startTime = str(self.startTime)[:19] duration = str(self.stopTime - self.startTime) status = [] if result.success_count: status.append(u’通過 %s’ % result.success_count) if result.failure_count: status.append(u’失敗 %s’ % result.failure_count) if result.error_count: status.append(u’錯誤 %s’ % result.error_count) if status: status = ’ ’.join(status) else: status = ’none’ return [ (u’測試人員’, self.tester), (u’開始時間’, startTime), (u’運行時長’, duration), (u’狀態’, status) ] def generateReport(self, test, result): report_attrs = self.getReportAttributes(result) generator = ’HTMLTestRunner %s’ % __version__ stylesheet = self._generate_stylesheet() heading = self._generate_heading(report_attrs) report = self._generate_report(result) ending = self._generate_ending() chart = self._generate_chart(result) output = self.HTML_TMPL % dict( title=saxutils.escape(self.title), generator=generator, stylesheet=stylesheet, heading=heading, report=report, ending=ending, chart_script=chart ) self.stream.write(output.encode(’utf8’)) def _generate_stylesheet(self): return self.STYLESHEET_TMPL def _generate_heading(self, report_attrs): a_lines = [] for name, value in report_attrs: line = self.HEADING_ATTRIBUTE_TMPL % dict(name=saxutils.escape(name),value=saxutils.escape(value), ) a_lines.append(line) heading = self.HEADING_TMPL % dict( title=saxutils.escape(self.title), parameters=’’.join(a_lines), description=saxutils.escape(self.description), ) return heading def _generate_report(self, result): rows = [] sortedResult = self.sortResult(result.result) for cid, (cls, cls_results) in enumerate(sortedResult): # subtotal for a class np = nf = ne = 0 for n, t, o, e in cls_results:if n == 0: np += 1elif n == 1: nf += 1else: ne += 1 # format class description if cls.__module__ == '__main__':name = cls.__name__ else:name = '%s.%s' % (cls.__module__, cls.__name__) doc = cls.__doc__ and cls.__doc__.split('n')[0] or '' desc = doc and ’%s: %s’ % (name, doc) or name row = self.REPORT_CLASS_TMPL % dict(style=ne > 0 and ’errorClass’ or nf > 0 and ’failClass’ or ’passClass’,desc=desc,count=np + nf + ne,Pass=np,fail=nf,error=ne,cid=’c%s’ % (cid + 1), ) rows.append(row) for tid, (n, t, o, e) in enumerate(cls_results):self._generate_report_test(rows, cid, tid, n, t, o, e) report = self.REPORT_TMPL % dict( test_list=’’.join(rows), count=str(result.success_count + result.failure_count + result.error_count), Pass=str(result.success_count), fail=str(result.failure_count), error=str(result.error_count), ) return report def _generate_chart(self, result): chart = self.ECHARTS_SCRIPT % dict( Pass=str(result.success_count), fail=str(result.failure_count), error=str(result.error_count), ) return chart def _generate_report_test(self, rows, cid, tid, n, t, o, e): # e.g. ’pt1.1’, ’ft1.1’, etc has_output = bool(o or e) tid = (n == 0 and ’p’ or ’f’) + ’t%s.%s’ % (cid + 1, tid + 1) name = t.id().split(’.’)[-1] doc = t.shortDescription() or '' desc = doc and (’%s: %s’ % (name, doc)) or name tmpl = has_output and self.REPORT_TEST_WITH_OUTPUT_TMPL or self.REPORT_TEST_NO_OUTPUT_TMPL script = self.REPORT_TEST_OUTPUT_TMPL % dict( id=tid, output=saxutils.escape(o + e), ) row = tmpl % dict( tid=tid, Class=(n == 0 and ’hiddenRow’ or ’none’), style=(n == 2 and ’errorCase’ or (n == 1 and ’failCase’ or ’none’)), desc=desc, script=script, status=self.STATUS[n], ) rows.append(row) if not has_output: return def _generate_ending(self): return self.ENDING_TMPL############################################################################### Facilities for running tests from the command line############################################################################### Note: Reuse unittest.TestProgram to launch test. In the future we may# build our own launcher to support more specific command line# parameters like test title, CSS, etc.class TestProgram(unittest.TestProgram): ''' A variation of the unittest.TestProgram. Please refer to the base class for command line parameters. ''' def runTests(self): # Pick HTMLTestRunner as the default test runner. # base class’s testRunner parameter is not useful because it means # we have to instantiate HTMLTestRunner before we know self.verbosity. if self.testRunner is None: self.testRunner = HTMLTestRunner(verbosity=self.verbosity) unittest.TestProgram.runTests(self)main = TestProgram############################################################################### Executing this module from the command line##############################################################################if __name__ == '__main__': main(module=None)

(2)第二種模板

'''A TestRunner for use with the Python unit testing framework. Itgenerates a HTML report to show the result at a glance.The simplest way to use this is to invoke its main method. E.g. import unittest import HTMLTestRunner ... define your tests ... if __name__ == ’__main__’: HTMLTestRunner.main()For more customization options, instantiates a HTMLTestRunner object.HTMLTestRunner is a counterpart to unittest’s TextTestRunner. E.g. # output to a file fp = file(’my_report.html’, ’wb’) runner = HTMLTestRunner.HTMLTestRunner(stream=fp,title=’My unit test’,description=’This demonstrates the report output by HTMLTestRunner.’) # Use an external stylesheet. # See the Template_mixin class for more customizable options runner.STYLESHEET_TMPL = ’<link rel='stylesheet' href='http://www.b3g6.com/bcjs/my_stylesheet.css' rel='external nofollow' rel='external nofollow' type='text/css'>’ # run the test runner.run(my_test_suite)------------------------------------------------------------------------Copyright (c) 2004-2007, Wai Yip TungAll rights reserved.Redistribution and use in source and binary forms, with or withoutmodification, are permitted provided that the following conditions aremet:* Redistributions of source code must retain the above copyright notice, this list of conditions and the following disclaimer.* Redistributions in binary form must reproduce the above copyright notice, this list of conditions and the following disclaimer in the documentation and/or other materials provided with the distribution.* Neither the name Wai Yip Tung nor the names of its contributors may be used to endorse or promote products derived from this software without specific prior written permission.THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS 'ASIS' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITEDTO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR APARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNEROR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, ORPROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OFLIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDINGNEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THISSOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.'''# URL: http://tungwaiyip.info/software/HTMLTestRunner.html__author__ = 'Wai Yip Tung'__version__ = '0.8.2''''Change HistoryVersion 0.8.2* Show output inline instead of popup window (Viorel Lupu).Version in 0.8.1* Validated XHTML (Wolfgang Borgert).* Added description of test classes and test cases.Version in 0.8.0* Define Template_mixin class for customization.* Workaround a IE 6 bug that it does not treat <script> block as CDATA.Version in 0.7.1* Back port to Python 2.3 (Frank Horowitz).* Fix missing scroll bars in detail log (Podi).'''# TODO: color stderr# TODO: simplify javascript using ,ore than 1 class in the class attribute?import datetimeimport ioimport sysimport timeimport unittestfrom xml.sax import saxutils# ------------------------------------------------------------------------# The redirectors below are used to capture output during testing. Output# sent to sys.stdout and sys.stderr are automatically captured. However# in some cases sys.stdout is already cached before HTMLTestRunner is# invoked (e.g. calling logging.basicConfig). In order to capture those# output, use the redirectors for the cached stream.## e.g.# >>> logging.basicConfig(stream=HTMLTestRunner.stdout_redirector)# >>>class OutputRedirector(object): ''' Wrapper to redirect stdout or stderr ''' def __init__(self, fp): self.fp = fp def write(self, s): self.fp.write(s) def writelines(self, lines): self.fp.writelines(lines) def flush(self): self.fp.flush()stdout_redirector = OutputRedirector(sys.stdout)stderr_redirector = OutputRedirector(sys.stderr)# ----------------------------------------------------------------------# Templateclass Template_mixin(object): ''' Define a HTML template for report customerization and generation. Overall structure of an HTML report HTML +------------------------+ |<html> | | <head>| | | | STYLESHEET | | +----------------+ | | || | | +----------------+ | | | | </head>| | | | <body>| | | | HEADING | | +----------------+ | | || | | +----------------+ | | | | REPORT| | +----------------+ | | || | | +----------------+ | | | | ENDING| | +----------------+ | | || | | +----------------+ | | | | </body>| |</html> | +------------------------+ ''' STATUS = { 0: ’pass’, 1: ’fail’, 2: ’error’, } DEFAULT_TITLE = ’Unit Test Report’ DEFAULT_DESCRIPTION = ’’ # ------------------------------------------------------------------------ # HTML Template HTML_TMPL = r'''<?xml version='1.0' encoding='UTF-8'?><!DOCTYPE html PUBLIC '-//W3C//DTD XHTML 1.0 Strict//EN' 'http://www.w3.org/TR/xhtml1/DTD/xhtml1-strict.dtd'><html xmlns='http://www.w3.org/1999/xhtml'><head> <title>%(title)s</title> <meta name='generator' content='%(generator)s'/> <meta http-equiv='Content-Type' content='text/html; charset=UTF-8'/> %(stylesheet)s</head><body><script language='javascript' type='text/javascript'><!--output_list = Array();/* level - 0:Summary; 1:Failed; 2:All */function showCase(level) { trs = document.getElementsByTagName('tr'); for (var i = 0; i < trs.length; i++) { tr = trs[i]; id = tr.id; if (id.substr(0,2) == ’ft’) { if (level < 1) {tr.className = ’hiddenRow’; } else {tr.className = ’’; } } if (id.substr(0,2) == ’pt’) { if (level > 1) {tr.className = ’’; } else {tr.className = ’hiddenRow’; } } }}function showClassDetail(cid, count) { var id_list = Array(count); var toHide = 1; for (var i = 0; i < count; i++) { tid0 = ’t’ + cid.substr(1) + ’.’ + (i+1); tid = ’f’ + tid0; tr = document.getElementById(tid); if (!tr) { tid = ’p’ + tid0; tr = document.getElementById(tid); } id_list[i] = tid; if (tr.className) { toHide = 0; } } for (var i = 0; i < count; i++) { tid = id_list[i]; if (toHide) { document.getElementById(’div_’+tid).style.display = ’none’ document.getElementById(tid).className = ’hiddenRow’; } else { document.getElementById(tid).className = ’’; } }}function showTestDetail(div_id){ var details_div = document.getElementById(div_id) var displayState = details_div.style.display // alert(displayState) if (displayState != ’block’ ) { displayState = ’block’ details_div.style.display = ’block’ } else { details_div.style.display = ’none’ }}function html_escape(s) { s = s.replace(/&/g,’&’); s = s.replace(/</g,’<’); s = s.replace(/>/g,’>’); return s;}/* obsoleted by detail in <div>function showOutput(id, name) { var w = window.open('', //url name, 'resizable,scrollbars,status,width=800,height=450'); d = w.document; d.write('<pre>'); d.write(html_escape(output_list[id])); d.write('n'); d.write('<a href=’javascript:window.close()’>close</a>n'); d.write('</pre>n'); d.close();}*/--></script>%(heading)s%(report)s%(ending)s</body></html>''' # variables: (title, generator, stylesheet, heading, report, ending) # ------------------------------------------------------------------------ # Stylesheet # # alternatively use a <link> for external style sheet, e.g. # <link rel='stylesheet' href='http://www.b3g6.com/bcjs/$url' rel='external nofollow' rel='external nofollow' type='text/css'> STYLESHEET_TMPL = '''<style type='text/css' media='screen'>body { font-family: verdana, arial, helvetica, sans-serif; font-size: 80%; }table { font-size: 100%; }pre { }/* -- heading ---------------------------------------------------------------------- */h1 { font-size: 16pt; color: gray;}.heading { margin-top: 0ex; margin-bottom: 1ex;}.heading .attribute { margin-top: 1ex; margin-bottom: 0;}.heading .description { margin-top: 4ex; margin-bottom: 6ex;}/* -- css div popup ------------------------------------------------------------------------ */a.popup_link {}a.popup_link:hover { color: red;}.popup_window { display: none; position: relative; left: 0px; top: 0px; /*border: solid #627173 1px; */ padding: 10px; background-color: #E6E6D6; font-family: 'Lucida Console', 'Courier New', Courier, monospace; text-align: left; font-size: 8pt; width: 500px;}}/* -- report ------------------------------------------------------------------------ */#show_detail_line { margin-top: 3ex; margin-bottom: 1ex;}#result_table { width: 80%; border-collapse: collapse; border: 1px solid #777;}#header_row { font-weight: bold; color: white; background-color: #777;}#result_table td { border: 1px solid #777; padding: 2px;}#total_row { font-weight: bold; }.passClass { background-color: #6c6; }.failClass { background-color: #c60; }.errorClass { background-color: #c00; }.passCase { color: #6c6; }.failCase { color: #c60; font-weight: bold; }.errorCase { color: #c00; font-weight: bold; }.hiddenRow { display: none; }.testcase { margin-left: 2em; }/* -- ending ---------------------------------------------------------------------- */#ending {}</style>''' # ------------------------------------------------------------------------ # Heading # HEADING_TMPL = '''<div class=’heading’><h1>%(title)s</h1>%(parameters)s<p class=’description’>%(description)s</p></div>''' # variables: (title, parameters, description) HEADING_ATTRIBUTE_TMPL = '''<p class=’attribute’><strong>%(name)s:</strong> %(value)s</p>''' # variables: (name, value) # ------------------------------------------------------------------------ # Report # REPORT_TMPL = '''<p id=’show_detail_line’>Show<a href=’javascript:showCase(0)’>Summary</a><a href=’javascript:showCase(1)’>Failed</a><a href=’javascript:showCase(2)’>All</a></p><table id=’result_table’><colgroup><col align=’left’ /><col align=’right’ /><col align=’right’ /><col align=’right’ /><col align=’right’ /><col align=’right’ /></colgroup><tr id=’header_row’> <td>Test Group/Test case</td> <td>Count</td> <td>Pass</td> <td>Fail</td> <td>Error</td> <td>View</td></tr>%(test_list)s<tr id=’total_row’> <td>Total</td> <td>%(count)s</td> <td>%(Pass)s</td> <td>%(fail)s</td> <td>%(error)s</td> <td> </td></tr></table>''' # variables: (test_list, count, Pass, fail, error) REPORT_CLASS_TMPL = r'''<tr class=’%(style)s’> <td>%(desc)s</td> <td>%(count)s</td> <td>%(Pass)s</td> <td>%(fail)s</td> <td>%(error)s</td> <td><a href='javascript:showClassDetail(’%(cid)s’,%(count)s)' rel='external nofollow' rel='external nofollow' >Detail</a></td></tr>''' # variables: (style, desc, count, Pass, fail, error, cid) REPORT_TEST_WITH_OUTPUT_TMPL = r'''<tr id=’%(tid)s’ class=’%(Class)s’> <td class=’%(style)s’><div class=’testcase’>%(desc)s</div></td> <td colspan=’5’ align=’center’> <!--css div popup start--> <a onfocus=’this.blur();’ href='javascript:showTestDetail(’div_%(tid)s’)' rel='external nofollow' rel='external nofollow' > %(status)s</a> <div id=’div_%(tid)s’ class='popup_window'> <div style=’text-align: right; color:red;cursor:pointer’> <a onfocus=’this.blur();’ onclick='document.getElementById(’div_%(tid)s’).style.display = ’none’ ' > [x]</a> </div> <pre> %(script)s </pre> </div> <!--css div popup end--> </td></tr>''' # variables: (tid, Class, style, desc, status) REPORT_TEST_NO_OUTPUT_TMPL = r'''<tr id=’%(tid)s’ class=’%(Class)s’> <td class=’%(style)s’><div class=’testcase’>%(desc)s</div></td> <td colspan=’5’ align=’center’>%(status)s</td></tr>''' # variables: (tid, Class, style, desc, status) REPORT_TEST_OUTPUT_TMPL = r'''%(id)s: %(output)s''' # variables: (id, output) # ------------------------------------------------------------------------ # ENDING # ENDING_TMPL = '''<div id=’ending’> </div>'''# -------------------- The end of the Template class -------------------TestResult = unittest.TestResultclass _TestResult(TestResult): # note: _TestResult is a pure representation of results. # It lacks the output and reporting ability compares to unittest._TextTestResult. def __init__(self, verbosity=1): TestResult.__init__(self) self.stdout0 = None self.stderr0 = None self.success_count = 0 self.failure_count = 0 self.error_count = 0 self.verbosity = verbosity # result is a list of result in 4 tuple # ( # result code (0: success; 1: fail; 2: error), # TestCase object, # Test output (byte string), # stack trace, # ) self.result = [] def startTest(self, test): TestResult.startTest(self, test) # just one buffer for both stdout and stderr self.outputBuffer= io.StringIO() stdout_redirector.fp = self.outputBuffer stderr_redirector.fp = self.outputBuffer self.stdout0 = sys.stdout self.stderr0 = sys.stderr sys.stdout = stdout_redirector sys.stderr = stderr_redirector def complete_output(self): ''' Disconnect output redirection and return buffer. Safe to call multiple times. ''' if self.stdout0: sys.stdout = self.stdout0 sys.stderr = self.stderr0 self.stdout0 = None self.stderr0 = None return self.outputBuffer.getvalue() def stopTest(self, test): # Usually one of addSuccess, addError or addFailure would have been called. # But there are some path in unittest that would bypass this. # We must disconnect stdout in stopTest(), which is guaranteed to be called. self.complete_output() def addSuccess(self, test): self.success_count += 1 TestResult.addSuccess(self, test) output = self.complete_output() self.result.append((0, test, output, ’’)) if self.verbosity > 1: sys.stderr.write(’ok ’) sys.stderr.write(str(test)) sys.stderr.write(’n’) else: sys.stderr.write(’.’) def addError(self, test, err): self.error_count += 1 TestResult.addError(self, test, err) _, _exc_str = self.errors[-1] output = self.complete_output() self.result.append((2, test, output, _exc_str)) if self.verbosity > 1: sys.stderr.write(’E ’) sys.stderr.write(str(test)) sys.stderr.write(’n’) else: sys.stderr.write(’E’) def addFailure(self, test, err): self.failure_count += 1 TestResult.addFailure(self, test, err) _, _exc_str = self.failures[-1] output = self.complete_output() self.result.append((1, test, output, _exc_str)) if self.verbosity > 1: sys.stderr.write(’F ’) sys.stderr.write(str(test)) sys.stderr.write(’n’) else: sys.stderr.write(’F’)class HTMLTestRunner(Template_mixin): ''' ''' def __init__(self, stream=sys.stdout, verbosity=1, title=None, description=None): self.stream = stream self.verbosity = verbosity if title is None: self.title = self.DEFAULT_TITLE else: self.title = title if description is None: self.description = self.DEFAULT_DESCRIPTION else: self.description = description self.startTime = datetime.datetime.now() def run(self, test): 'Run the given test case or test suite.' result = _TestResult(self.verbosity) test(result) self.stopTime = datetime.datetime.now() self.generateReport(test, result) #print(sys.stderr, 'nTimeElapsed: %s' % (self.stopTime-self.startTime)) sys.stderr.write(’nTime Elapsed: %sn’ % (self.stopTime - self.startTime)) return result def sortResult(self, result_list): # unittest does not seems to run in any particular order. # Here at least we want to group them together by class. rmap = {} classes = [] for n,t,o,e in result_list: cls = t.__class__ if not cls in rmap:rmap[cls] = []classes.append(cls) rmap[cls].append((n,t,o,e)) r = [(cls, rmap[cls]) for cls in classes] return r def getReportAttributes(self, result): ''' Return report attributes as a list of (name, value). Override this to add custom attributes. ''' startTime = str(self.startTime)[:19] duration = str(self.stopTime - self.startTime) status = [] if result.success_count: status.append(’Pass %s’ % result.success_count) if result.failure_count: status.append(’Failure %s’ % result.failure_count) if result.error_count: status.append(’Error %s’ % result.error_count ) if status: status = ’ ’.join(status) else: status = ’none’ return [ (’Start Time’, startTime), (’Duration’, duration), (’Status’, status), ] def generateReport(self, test, result): report_attrs = self.getReportAttributes(result) generator = ’HTMLTestRunner %s’ % __version__ stylesheet = self._generate_stylesheet() heading = self._generate_heading(report_attrs) report = self._generate_report(result) ending = self._generate_ending() output = self.HTML_TMPL % dict( title = saxutils.escape(self.title), generator = generator, stylesheet = stylesheet, heading = heading, report = report, ending = ending, ) self.stream.write(output.encode(’utf8’)) def _generate_stylesheet(self): return self.STYLESHEET_TMPL def _generate_heading(self, report_attrs): a_lines = [] for name, value in report_attrs: line = self.HEADING_ATTRIBUTE_TMPL % dict( name = saxutils.escape(name), value = saxutils.escape(value),) a_lines.append(line) heading = self.HEADING_TMPL % dict( title = saxutils.escape(self.title), parameters = ’’.join(a_lines), description = saxutils.escape(self.description), ) return heading def _generate_report(self, result): rows = [] sortedResult = self.sortResult(result.result) for cid, (cls, cls_results) in enumerate(sortedResult): # subtotal for a class np = nf = ne = 0 for n,t,o,e in cls_results:if n == 0: np += 1elif n == 1: nf += 1else: ne += 1 # format class description if cls.__module__ == '__main__':name = cls.__name__ else:name = '%s.%s' % (cls.__module__, cls.__name__) doc = cls.__doc__ and cls.__doc__.split('n')[0] or '' desc = doc and ’%s: %s’ % (name, doc) or name row = self.REPORT_CLASS_TMPL % dict(style = ne > 0 and ’errorClass’ or nf > 0 and ’failClass’ or ’passClass’,desc = desc,count = np+nf+ne,Pass = np,fail = nf,error = ne,cid = ’c%s’ % (cid+1), ) rows.append(row) for tid, (n,t,o,e) in enumerate(cls_results):self._generate_report_test(rows, cid, tid, n, t, o, e) report = self.REPORT_TMPL % dict( test_list = ’’.join(rows), count = str(result.success_count+result.failure_count+result.error_count), Pass = str(result.success_count), fail = str(result.failure_count), error = str(result.error_count), ) return report def _generate_report_test(self, rows, cid, tid, n, t, o, e): # e.g. ’pt1.1’, ’ft1.1’, etc has_output = bool(o or e) tid = (n == 0 and ’p’ or ’f’) + ’t%s.%s’ % (cid+1,tid+1) name = t.id().split(’.’)[-1] doc = t.shortDescription() or '' desc = doc and (’%s: %s’ % (name, doc)) or name tmpl = has_output and self.REPORT_TEST_WITH_OUTPUT_TMPL or self.REPORT_TEST_NO_OUTPUT_TMPL # o and e should be byte string because they are collected from stdout and stderr? if isinstance(o,str): # TODO: some problem with ’string_escape’: it escape n and mess up formating # uo = unicode(o.encode(’string_escape’)) uo = o else: uo = o if isinstance(e,str): # TODO: some problem with ’string_escape’: it escape n and mess up formating # ue = unicode(e.encode(’string_escape’)) ue = e else: ue = e script = self.REPORT_TEST_OUTPUT_TMPL % dict( id = tid, output = saxutils.escape(uo+ue), ) row = tmpl % dict( tid = tid, Class = (n == 0 and ’hiddenRow’ or ’none’), style = n == 2 and ’errorCase’ or (n == 1 and ’failCase’ or ’none’), desc = desc, script = script, status = self.STATUS[n], ) rows.append(row) if not has_output: return def _generate_ending(self): return self.ENDING_TMPL############################################################################### Facilities for running tests from the command line############################################################################### Note: Reuse unittest.TestProgram to launch test. In the future we may# build our own launcher to support more specific command line# parameters like test title, CSS, etc.class TestProgram(unittest.TestProgram): ''' A variation of the unittest.TestProgram. Please refer to the base class for command line parameters. ''' def runTests(self): # Pick HTMLTestRunner as the default test runner. # base class’s testRunner parameter is not useful because it means # we have to instantiate HTMLTestRunner before we know self.verbosity. if self.testRunner is None: self.testRunner = HTMLTestRunner(verbosity=self.verbosity) unittest.TestProgram.runTests(self)main = TestProgram############################################################################### Executing this module from the command line##############################################################################if __name__ == '__main__': main(module=None)

以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支持好吧啦網。

標簽: Python 編程
相關文章:
日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区
亚洲另类av| 国产精品香蕉| av中文资源在线资源免费观看| 国产极品久久久久久久久波多结野 | 日韩久久视频| 久久久男人天堂| 久久精品电影| 黄色成人精品网站| 亚洲人成网站在线在线观看| 日韩一区精品| 欧美精品影院| 久久精品三级| 神马午夜久久| 快she精品国产999| 婷婷久久免费视频| 国产精品成人国产| sm久久捆绑调教精品一区| 日韩在线观看| 欧美高清一区| 蜜桃tv一区二区三区| 日韩精品一级中文字幕精品视频免费观看 | 亚洲国内欧美| 亚洲欧美网站| 国产亚洲精aa在线看| 精品女同一区二区三区在线观看| 秋霞国产精品| 亚洲精品在线二区| 精品国产一级| 欧美日韩国产一区二区三区不卡 | 999国产精品999久久久久久| 伊人久久亚洲美女图片| 日韩精品久久久久久久软件91| 欧美国产三级| 久久国产电影| 日韩国产高清在线| 精品久久不卡| 亚洲激精日韩激精欧美精品| 亚洲精品自拍| 日产精品一区二区| 免费成人av在线播放| 国产精品久久久久久模特| 欧美不卡高清一区二区三区| 免费日韩av| 久久精品国产免费| 国产亚洲精品v| 国产精品99精品一区二区三区∴| 欧美一区二区三区高清视频 | 久久99精品久久久野外观看| 91精品综合| 日本少妇精品亚洲第一区| 9999国产精品| 亚洲精品成人一区| 精品一区二区三区亚洲| 99精品美女| 欧美一区久久| 99精品综合| 日本va欧美va欧美va精品| 韩国三级一区| 日韩1区2区日韩1区2区| 久久精品青草| 国产高清亚洲| 蜜桃视频一区二区三区在线观看 | 精品午夜视频| 久热re这里精品视频在线6| 成人污污视频| 日韩激情精品| 99久久夜色精品国产亚洲狼| 日韩国产在线观看| 欧美.日韩.国产.一区.二区| 嫩草伊人久久精品少妇av杨幂| 宅男噜噜噜66国产日韩在线观看| 精品一区二区三区亚洲| 亚洲精品高潮| 在线日韩av| 成人国产精选| 久久激情五月婷婷| 视频在线观看一区二区三区| 色婷婷狠狠五月综合天色拍| 国产精品手机在线播放| 免费在线欧美视频| 欧美sss在线视频| 国产日产高清欧美一区二区三区 | 欧美日韩一二| 国内揄拍国内精品久久| 日韩国产欧美在线播放| 亚洲国产一区二区在线观看 | www.com.cn成人| 国产麻豆一区二区三区| 丝袜a∨在线一区二区三区不卡| 日韩欧美午夜| 国产精品sss在线观看av| 免费在线观看一区二区三区| 91精品国产成人观看| 日韩av专区| 国产精品美女在线观看直播| 一区二区国产在线| 日韩一区二区久久| 99精品在线观看| 日韩一区二区在线免费| 国产一区二区三区久久| 国产精品美女久久久久久不卡| 亚洲美女91| 视频一区中文字幕| 91久久中文| 好看不卡的中文字幕| 欧美一区三区| 日韩大片在线播放| 国产传媒在线观看| 精品视频自拍| 精品视频国内| 国际精品欧美精品| 国产一区二区三区久久久久久久久| 国产精品日韩精品在线播放| 欧美日韩亚洲国产精品| 日韩1区2区日韩1区2区| 日韩毛片一区| 日本不卡一区二区三区| 日韩有吗在线观看| 午夜精品影视国产一区在线麻豆| 久久99伊人| 亚洲香蕉久久| 蜜桃免费网站一区二区三区| 香蕉视频成人在线观看| 欧美专区一区二区三区| 亚洲欧美成人综合| 日本h片久久| 久久国产精品免费一区二区三区| 日本亚洲视频| 日韩午夜视频在线| 日韩视频在线一区二区三区| 亚洲综合不卡| 免费在线欧美视频| 日韩国产在线一| 国产欧美69| 国产日韩一区| 精品在线网站观看| 国产高清亚洲| 91青青国产在线观看精品| 97精品97| 欧美福利在线| 噜噜噜躁狠狠躁狠狠精品视频 | 99riav国产精品| 久热精品在线| 欧美日韩亚洲三区| 久久中文精品| 日韩免费福利视频| 99国产精品久久久久久久| 蜜臀av国产精品久久久久| 国产色噜噜噜91在线精品| 精品国产不卡| 国产综合亚洲精品一区二| 亚洲精品精选| 国产精品久久国产愉拍| 六月婷婷综合| 中文日韩在线| 欧美亚洲综合视频| 91综合网人人| 亚洲综合日韩| 国产精品网址| 亚洲风情在线资源| 9色精品在线| 欧美一区影院| 9999国产精品| 久久国产88| 国产精品激情| 今天的高清视频免费播放成人| 在线观看一区| 精品久久久久中文字幕小说| 美女少妇全过程你懂的久久| 亚洲精品国产精品粉嫩| 精品久久久久久久| 欧美在线亚洲| 国产精品一区二区三区av麻 | 美女久久久久久| 激情综合自拍| 久久狠狠久久| 婷婷国产精品| 欧美在线看片| 亚洲91精品| 国产亚洲精aa在线看| 久久精品国产www456c0m| 亚洲香蕉久久| 精品欠久久久中文字幕加勒比| 欧美gv在线| 日本成人中文字幕在线视频| 最新中文字幕在线播放| 日韩影院在线观看| 久久久久久一区二区| 日韩精品一二三四| 国产videos久久| 蜜臀av性久久久久蜜臀aⅴ四虎| 精品午夜视频| 日本欧美在线看| 日韩在线短视频| 欧美天堂一区| 国产精品呻吟| 成人在线网站| 久久不卡国产精品一区二区| 视频一区视频二区中文字幕| 午夜久久中文|