一、项目工程目录:

二、具体工程文件代码:

1、新建一个包名:common(用于存放基本函数封装)

(1)在common包下新建一个base.py文件,作用:页面操作封装。base.py文件代码如下:

# coding=utf-8"""
------------------------------------
@Time : 2020/01/15
@Auth : Anker
@File : base.py
@Description:页面操作封装
@IDE  : PyCharm
@Motto: Believe in yourself and persistence can make success!
------------------------------------
"""from selenium.webdriver.support.wait import WebDriverWait
from config.read_config import ReadConfclass BasePage(object):# 读取config.ini配置文件,传入sections值url = ReadConf()# 传入sections模块standard_url = url.readConf("sections")# 这里传入sections模块中的urlbase_url = standard_url['url']def __init__(self, driver, test_url=base_url):"""构造函数:param driver:param 传入url"""self.driver = driverself.url = test_url# 设置全局元素隐式等待时间为10秒钟self.driver.implicitly_wait(10)def open_url(self):"""打开url:param:"""url = self.urlself.driver.get(url)title = self.driver.titleprint("项目名称:", title + "2.6")print("项目地址:", self.driver.current_url)def back(self):"""浏览器后退按钮:param:"""self.driver.back()def forward(self):"""浏览器前进按钮:param:"""self.driver.forward()def close(self):"""关闭并停止浏览器服务:param:"""self.driver.quit()def find_element(self, *loc):"""判断定位方式(常见的有8种获取元素的方法):param * loc"""try:WebDriverWait(self.driver, 20).until(lambda driver: driver.find_element(*loc).is_displayed())return self.driver.find_element(*loc)except:print("元素在页面中未找到!", *loc)def find_elements(self, *loc):return self.driver.find_elements(*loc)def input_content(self, loc, content):"""文本框内容输入:param loc:param content"""self.find_element(*loc).send_keys(content)def send_keys(self, loc, value, clear_first=True, click_first=True):try:# getattr相当于self.locloc = getattr(self, "_%s" % loc)if click_first:self.mouse_click(loc)  # 调用鼠标点击事件方法if clear_first:self.mouse_clear(loc)  # 调用鼠标清理事件方法self.find_element(*loc).send_keys(value)except ArithmeticError:print(u"%s 页面中未能找到 %s 元素" % (self, loc))def mouse_clear(self, loc):"""鼠标清理事件:param loc"""return self.find_element(*loc).clear()def mouse_click(self, loc):"""鼠标点击事件:param loc"""return self.find_element(*loc).click()def script(self, src):return self.driver.execute_script(src)def switch_frame(self, loc):return self.driver.switch_to_frame(loc)def isElementPresent(self, element_xpath):"""封装一个函数,用来判断页面某个值是否存在:param element_xpath"""try:self.driver.find_element_by_xpath(element_xpath)return Trueexcept:return False

(2)在common包下新建一个driver.py文件,作用:浏览器选择,默认为谷歌浏览器。driver.py文件代码如下:

# coding=utf-8"""
------------------------------------
@Time : 2020/01/15
@Auth : Anker
@File : driver.py
@Description:浏览器选择,默认为谷歌浏览器
@IDE  : PyCharm
@Motto: Believe in yourself and persistence can make success!
------------------------------------
"""from selenium import webdriver
browser_type = "Chrome"def open_browser():"""浏览器选择(Selenium支持Chrome、Firefox、IE浏览器):param:"""global driverif browser_type == 'Firefox':driver = webdriver.Firefox()elif browser_type == 'Chrome':driver = webdriver.Chrome()elif browser_type == 'IE':driver = webdriver.Ie()elif browser_type == '':driver = webdriver.Chrome()return driverif __name__ == '__main__':driver = open_browser()

(3)在common包下新建一个HTMLTestRunner.py文件,作用:用于生成html报告文件。HTMLTestRunner.py文件代码如下:

#-*- coding: utf-8 -*-
"""
A TestRunner for use with the Python unit testing framework. It
generates 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 unittestimport 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 filefp = 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 optionsrunner.STYLESHEET_TMPL = '<link rel="stylesheet" href="my_stylesheet.css" type="text/css">'# run the testrunner.run(my_test_suite)------------------------------------------------------------------------
Copyright (c) 2004-2007, Wai Yip Tung
All rights reserved.Redistribution and use in source and binary forms, with or without
modification, are permitted provided that the following conditions are
met:* 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 copyrightnotice, this list of conditions and the following disclaimer in thedocumentation and/or other materials provided with the distribution.
* Neither the name Wai Yip Tung nor the names of its contributors may beused to endorse or promote products derived from this software withoutspecific prior written permission.THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS
IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A
PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER
OR 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, OR
PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
"""# URL: http://tungwaiyip.info/software/HTMLTestRunner.html__author__ = "Wai Yip Tung"
__version__ = "0.8.3""""
Change History
Version 0.8.4 by GoverSky
* Add sopport for 3.x
* Add piechart for resultpiechart
* Add Screenshot for selenium_case test
* Add Retry on failedVersion 0.8.3
* Prevent crash on class or module-level exceptions (Darren Wurf).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 sys
import unittest
from xml.sax import saxutilsPY3K = (sys.version_info[0] > 2)
if PY3K:import io as StringIO
else:import StringIO
import copy# ------------------------------------------------------------------------
# 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_demo.basicConfig). In order to capture those
# output, use the redirectors for the cached stream.
#
# e.g.
#   >>> logging_demo.basicConfig(stream=HTMLTestRunner.stdout_redirector)
#   >>>class OutputRedirector(object):""" Wrapper to redirect stdout or stderr """def __init__(self, fp):self.fp = fpdef 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 reportHTML+------------------------+|<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 TemplateHTML_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(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,'&amp;');s = s.replace(/</g,'&lt;');s = s.replace(/>/g,'&gt;');return s;
}function drawCircle(pass, fail, error){var color = ["#6c6","#c60","#c00"];var data = [pass,fail,error];var text_arr = ["pass", "fail", "error"];var canvas = document.getElementById("circle");var ctx = canvas.getContext("2d");var startPoint=0;var width = 20, height = 10;var posX = 112 * 2 + 20, posY = 30;var textX = posX + width + 5, textY = posY + 10;for(var i=0;i<data.length;i++){ctx.fillStyle = color[i];ctx.beginPath();ctx.moveTo(112,84);ctx.arc(112,84,84,startPoint,startPoint+Math.PI*2*(data[i]/(data[0]+data[1]+data[2])),false);ctx.fill();startPoint += Math.PI*2*(data[i]/(data[0]+data[1]+data[2]));ctx.fillStyle = color[i];ctx.fillRect(posX, posY + 20 * i, width, height);ctx.moveTo(posX, posY + 20 * i);ctx.font = 'bold 14px';ctx.fillStyle = color[i];var percent = text_arr[i] + ":"+data[i];ctx.fillText(percent, textX, textY + 20 * i);}
}function show_img(obj) {var obj1 = obj.nextElementSiblingobj1.style.display='block'var index = 0;//每张图片的下标,var len = obj1.getElementsByTagName('img').length;var imgyuan = obj1.getElementsByClassName('imgyuan')[0]//var start=setInterval(autoPlay,500);obj1.onmouseover=function(){//当鼠标光标停在图片上,则停止轮播clearInterval(start);}obj1.onmouseout=function(){//当鼠标光标停在图片上,则开始轮播start=setInterval(autoPlay,1000);}for (var i = 0; i < len; i++) {var font = document.createElement('font')imgyuan.appendChild(font)}var lis = obj1.getElementsByTagName('font');//得到所有圆圈changeImg(0)var funny = function (i) {lis[i].onmouseover = function () {index=ichangeImg(i)}}for (var i = 0; i < lis.length; i++) {funny(i);}function autoPlay(){if(index>len-1){index=0;clearInterval(start); //运行一轮后停止}changeImg(index++);}imgyuan.style.width= 25*len +"px";//对应圆圈和图片同步function changeImg(index) {var list = obj1.getElementsByTagName('img');var list1 = obj1.getElementsByTagName('font');for (i = 0; i < list.length; i++) {list[i].style.display = 'none';list1[i].style.backgroundColor = 'white';}list[index].style.display = 'block';list1[index].style.backgroundColor = 'blue';}}
function hide_img(obj){obj.parentElement.style.display = "none";obj.parentElement.getElementsByClassName('imgyuan')[0].innerHTML = "";
}
</script>
<div class="piechart"><div><canvas id="circle" width="350" height="168" </canvas></div>
</div>
%(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="$url" 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  {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: 4ex;margin-bottom: 6ex;
}/* -- css div popup ------------------------------------------------------------------------ */
a.popup_link {
}a.popup_link:hover {color: red;
}
.img{height: 100%;border-collapse: collapse;border: 2px solid #777;
}.screenshots {z-index: 100;position:absolute;height: 80%;left: 50%;top: 50%;transform: translate(-50%,-50%);display: none;
}.imgyuan{height: 20px;border-radius: 12px;background-color: red;padding-left: 13px;margin: 0 auto;position: relative;top: -40px;background-color: rgba(1, 150, 0, 0.3);
}
.imgyuan font{border:1px solid white;width:11px;height:11px;border-radius:50%;margin-right: 9px;margin-top: 4px;display: block;float: left;background-color: white;
}
.close_shots {background-image: url();background-size: 22px 22px;-moz-background-size: 22px 22px;background-repeat: no-repeat;position: absolute;top: 5px;right: 5px;height: 22px;z-index: 99;width: 22px;
}
.popup_window {display: none;position: relative;left: 0px;top: 0px;padding: 10px;background-color: #E6E6D6;font-family: "Lucida Console", "Courier New", Courier, monospace;text-align: left;font-size: 8pt;
}}
/* -- report ------------------------------------------------------------------------ */
#show_detail_line {margin-top: 3ex;margin-bottom: 1ex;
}#result_table {margin: 1em 0;width: 100%;overflow: hidden;background:  #FFF;color:  #024457;border-radius:   10px;border: 1px solid #167F92;
}
#result_table th {border: 1px solid #FFFFFF;background-color: #167F92;color: #FFF;padding: 0.5em;&:first-child {display: table-cell;text-align: center;}&:nth-child(2) {display: table-cell;span {display:none;}&:after {content:attr(data-th);}}@media (min-width: 480px) {&:nth-child(2) {span {display: block;}&:after {display: none;}}}}
#result_table td {word-wrap: break-word;max-width: 7em;padding: 0.3em;&:first-child {display: table-cell;text-align: center;}@media (min-width: 400px) {border: 1px solid #D9E4E6;}}#result_table  th, td {margin: .5em 1em;@media (min-width: 400px) {display: table-cell;padding: 1em;}}#total_row  { font-weight: bold; }
.passClass  { background-color: #6c6;  !important ;}
.failClass  { background-color: #c60;  !important ;}
.errorClass { background-color: #c00; !important ; }
.passCase   { color: #6c6; }
.failCase   { color: #c60; font-weight: bold; }
.errorCase  { color: #c00; font-weight: bold; }
tr[id^=pt]  td { background-color: rgba(73,204,144,.3) !important ; }
tr[id^=ft]  td { background-color: rgba(252,161,48,.3) !important; }
tr[id^=et]  td { background-color: rgba(249,62,62,.3) !important ; }
.hiddenRow  { display: none; }
.testcase   { margin-left: 2em; }/* -- ending ---------------------------------------------------------------------- */
#ending {
}.piechart{position:absolute;  ;top:20px;left:300px;width: 200px;float: left;display:  inline;
}</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'>显示
<a href='javascript:showCase(0)'>概要</a>
<a href='javascript:showCase(1)'>失败</a>
<a href='javascript:showCase(2)'>所有</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' />
<col align='right' />
</colgroup>
<tr id='header_row'><th>测试组/测试用例</th><th>总数</th><th>通过</th><th>失败</th><th>错误</th><th>视图</th><th>错误截图</th>
</tr>
%(test_list)s
<tr id='total_row'><th>统计</th><th>%(count)s</th><th>%(Pass)s</th><th>%(fail)s</th><th>%(error)s</th><th>&nbsp;</th><th>&nbsp;</th>
</tr>
</table>
<script>showCase(1);drawCircle(%(Pass)s, %(fail)s, %(error)s);
</script>
"""# 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)">详情</a></td><td>&nbsp;</td>
</tr>
"""  # variables: (style, desc, count, Pass, fail, error, cid)REPORT_TEST_WITH_OUTPUT_TMPL = r"""
<tr id='%(tid)s' class='%(Class)s'><td ><div class='testcase'>%(desc)s</div></td><td colspan='5' align='center'><!--css div popup start--><span class='status %(style)s'><a class="popup_link" onfocus='this.blur();' href="javascript:showTestDetail('div_%(tid)s')" >%(status)s</a></span><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><td>%(img)s</td>
</tr>
"""  # variables: (tid, Class, style, desc, status,img)REPORT_TEST_NO_OUTPUT_TMPL = r"""
<tr id='%(tid)s' class='%(Class)s'><td><div class='testcase'>%(desc)s</div></td><td colspan='5' align='center'><span class='status %(style)s'>%(status)s</span></td><td>%(img)s</td>
</tr>
"""  # variables: (tid, Class, style, desc, status,img)REPORT_TEST_OUTPUT_TMPL = r"""
%(id)s: %(output)s
"""  # variables: (id, output)IMG_TMPL = r"""<a href="#"  onclick="show_img(this)">显示截图</a><div align="center" class="screenshots"  style="display:none"><a class="close_shots"  href="#"   onclick="hide_img(this)"></a>%(imgs)s<div class="imgyuan"></div></div>"""# ------------------------------------------------------------------------# ENDING#ENDING_TMPL = """<div id='ending'>&nbsp;</div>"""# -------------------- The end of the Template class -------------------def __getattribute__(self, item):value = object.__getattribute__(self, item)if PY3K:return valueelse:if isinstance(value, str):return value.decode("utf-8")else:return valueTestResult = 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, retry=0,save_last_try=True):TestResult.__init__(self)self.stdout0 = Noneself.stderr0 = Noneself.success_count = 0self.failure_count = 0self.error_count = 0self.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.retry = retryself.trys = 0self.status = 0self.save_last_try = save_last_tryself.outputBuffer = StringIO.StringIO()def startTest(self, test):test.imgs = []# test.imgs = getattr(test, "imgs", [])TestResult.startTest(self, test)self.outputBuffer.seek(0)self.outputBuffer.truncate()stdout_redirector.fp = self.outputBufferstderr_redirector.fp = self.outputBufferself.stdout0 = sys.stdoutself.stderr0 = sys.stderrsys.stdout = stdout_redirectorsys.stderr = stderr_redirectordef complete_output(self):"""Disconnect output redirection and return buffer.Safe to call multiple times."""if self.stdout0:sys.stdout = self.stdout0sys.stderr = self.stderr0self.stdout0 = Noneself.stderr0 = Nonereturn 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.if self.retry:if self.status == 1:self.trys += 1if self.trys <= self.retry:if self.save_last_try:t = self.result.pop(-1)if t[0]==1:self.failure_count-=1else:self.error_count -= 1test=copy.copy(test)sys.stderr.write("Retesting... ")sys.stderr.write(str(test))sys.stderr.write('..%d \n' % self.trys)doc = test._testMethodDoc or ''if doc.find('_retry')!=-1:doc = doc[:doc.find('_retry')]desc ="%s_retry:%d" %(doc, self.trys)if not PY3K:if isinstance(desc, str):desc = desc.decode("utf-8")test._testMethodDoc = desctest(self)else:self.status = 0self.trys = 0self.complete_output()def addSuccess(self, test):self.success_count += 1self.status = 0TestResult.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 += 1self.status = 1TestResult.addError(self, test, err)_, _exc_str = self.errors[-1]output = self.complete_output()self.result.append((2, test, output, _exc_str))if not getattr(test, "driver",""):passelse:try:driver = getattr(test, "driver")test.imgs.append(driver.get_screenshot_as_base64())except Exception:passif 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 += 1self.status = 1TestResult.addFailure(self, test, err)_, _exc_str = self.failures[-1]output = self.complete_output()self.result.append((1, test, output, _exc_str))if not getattr(test, "driver",""):passelse:try:driver = getattr(test, "driver")test.imgs.append(driver.get_screenshot_as_base64())except Exception as e:passif 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=2, title=None, description=None, retry=0,save_last_try=False):self.stream = streamself.retry = retryself.save_last_try=save_last_tryself.verbosity = verbosityif title is None:self.title = self.DEFAULT_TITLEelse:self.title = titleif description is None:self.description = self.DEFAULT_DESCRIPTIONelse:self.description = descriptionself.startTime = datetime.datetime.now()def run(self, test):"Run the given test case or test suite."result = _TestResult(self.verbosity, self.retry, self.save_last_try)test(result)self.stopTime = datetime.datetime.now()self.generateReport(test, result)if PY3K:# for python3# print('\nTime Elapsed: %s' % (self.stopTime - self.startTime),file=sys.stderr)output = '\nTime Elapsed: %s' % (self.stopTime - self.startTime)sys.stderr.write(output)else:print >> sys.stderr, '\nTime Elapsed: %s' % (self.stopTime - self.startTime)return resultdef 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 rdef 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'<span class="tj passCase">Pass</span>%s' % result.success_count)if result.failure_count:status.append(u'<span class="tj failCase">Failure</span>%s' % result.failure_count)if result.error_count:status.append(u'<span class="tj errorCase">Error</span>%s' % result.error_count)if status:status = ' '.join(status)else:status = 'none'return [(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()output = self.HTML_TMPL % dict(title=saxutils.escape(self.title),generator=generator,stylesheet=stylesheet,heading=heading,report=report,ending=ending,)if PY3K:self.stream.write(output.encode())else:self.stream.write(output.encode('utf8'))def _generate_stylesheet(self):return self.STYLESHEET_TMPLdef _generate_heading(self, report_attrs):a_lines = []for name, value in report_attrs:line = self.HEADING_ATTRIBUTE_TMPL % dict(name=name,value=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 headingdef _generate_report(self, result):rows = []sortedResult = self.sortResult(result.result)for cid, (cls, cls_results) in enumerate(sortedResult):# subtotal for a classnp = nf = ne = 0for n, t, o, e in cls_results:if n == 0:np += 1elif n == 1:nf += 1else:ne += 1# format class descriptionif 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 nameif not PY3K:if isinstance(desc, str):desc = desc.decode("utf-8")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=u''.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 reportdef _generate_report_test(self, rows, cid, tid, n, t, o, e):# e.g. 'pt1.1', 'ft1.1', etchas_output = bool(o or e)tid = (n == 0 and 'p' or 'f') + 't%s.%s' % (cid + 1, tid + 1)name = t.id().split('.')[-1]if self.verbosity > 1:doc = t._testMethodDoc or ''else:doc = ""desc = doc and ('%s: %s' % (name, doc)) or nameif not PY3K:if isinstance(desc, str):desc = desc.decode("utf-8")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):# uo = unicode(o.encode('string_escape'))if PY3K:uo = oelse:uo = o.decode('utf-8', 'ignore')else:uo = oif isinstance(e, str):# ue = unicode(e.encode('string_escape'))if PY3K:ue = eelif e.find("Error") != -1 or e.find("Exception") != -1:es = e.decode('utf-8', 'ignore').split('\n')es[-2] = es[-2].decode('unicode_escape')ue = u"\n".join(es)else:ue = e.decode('utf-8', 'ignore')else:ue = escript = self.REPORT_TEST_OUTPUT_TMPL % dict(id=tid,output=saxutils.escape(uo + ue),)if getattr(t,'imgs',[]):# 判断截图列表,如果有则追加tmp = u""for i, img in enumerate(t.imgs):if i==0:tmp+=""" <img src="https://img-blog.csdnimg.cn/2022010613392051095.jpg" style="display: block;" class="img"/>\n""" % imgelse:tmp+=""" <img src="https://img-blog.csdnimg.cn/2022010613392051095.jpg" style="display: none;" class="img"/>\n""" % imgimgs = self.IMG_TMPL % dict(imgs=tmp)else:imgs = u"""无截图"""row = tmpl % dict(tid=tid,Class=(n == 0 and 'hiddenRow' or 'none'),style=n == 2 and 'errorCase' or (n == 1 and 'failCase' or 'passCase'),desc=desc,script=script,status=self.STATUS[n],img=imgs,)rows.append(row)if not has_output:returndef _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 baseclass 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)

(4)在common包下新建一个login.py文件,作用:登录操作。login.py文件代码如下:

# coding=utf-8"""
------------------------------------
@Time : 2020/01/15
@Auth : Anker
@File : login.py
@purpose:登录操作
@IDE  : PyCharm
@Motto: Believe in yourself and persistence can make success!
------------------------------------
"""from selenium.webdriver.common.by import By
from common.base import BasePageclass LoginPage(BasePage):# 通过id的方式去定位用户名username_loc = (By.ID, "login_username")# 通过id的方式去定位用户密码password_loc = (By.ID, "login_password")# 通过xpath的方式去定位登录按钮login_button_loc = (By.XPATH, "//*[text()='登 录']/..")def input_username(self, username):        # 输入用户名self.find_element(*self.username_loc).send_keys(username)def input_password(self, password):        # 输入密码self.find_element(*self.password_loc).send_keys(password)def click_login_button(self):              # 点击登录按钮self.find_element(*self.login_button_loc).click()def login(self, username="18288888888", password="123456"):self.open_url()self.input_username(username)self.input_password(password)self.click_login_button()return self.driver

(5)在common包下新建一个unit.py文件,作用:封装浏览器的启动和关闭的操作。unit.py文件代码如下:

# coding=utf-8"""
------------------------------------
@Time : 2020/01/15
@Auth : Anker
@File : unit.py
@Description:封装浏览器的启动和关闭的操作
@IDE  : PyCharm
@Motto: Believe in yourself and persistence can make success!
------------------------------------
"""import unittest
import warnings  # 导入警告包,避免日志打印中出现报红的无关紧要的警告
from selenium.webdriver.common.by import By
from common.driver import open_browserclass Unit(unittest.TestCase):knowledge_graph_menu = (By.XPATH, '//*/span[text()="知识图谱"]')  # 点击知识图谱菜单system_management_menu = (By.XPATH, '//*/span[text()="系统管理"]')  # 点击系统管理菜单material_definition_menu = (By.XPATH, '//*[text()="物料定义"]')  # 点击物料定义菜单material_name_text = (By.ID, "material_name")  # 输入物料名称material_code_text = (By.ID, "material_code")  # 输入物料编码material_unit_text = (By.ID, "material_unit")  # 输入物料单位BOM_material_name_id = (By.ID, "material_sourceId")  # BOM关联的物料idBOM_material_name_xpath = (By.XPATH, "//*[@id='material_sourceId']")  # BOM关联的物料xpathBOM_definition_menu = (By.XPATH, '//*[text()="BOM定义"]')  # 点击BOM定义菜单procedure_definition_menu = (By.XPATH, '//*[text()="工序定义"]')  # 点击工序定义菜单procedure_name_text = (By.ID, "define_name")  # 输入工序名称procedure_device_type_id = (By.ID, "define_typeId")  # 工序关联的设备类型idprocedure_device_type_xpath = (By.XPATH, "//*[@id='define_typeId']")  # 工序关联的设备类型xpathworkshop_definition_menu = (By.XPATH, '//*[text()="车间定义"]')  # 点击车间定义菜单workshop_code_text = (By.ID, "define_code")  # 输入车间编码workshop_name_text = (By.ID, "define_name")  # 输入车间名称workshop_ip_text = (By.ID, "define_ip")  # 输入车间IPworkshop_management_text = (By.ID, "define_management")  # 输入车间负责人workshop_name_id = (By.ID, "define_workshopId")  # 关联的车间idworkshop_name_xpath = (By.XPATH, "//*[@id='define_workshopId']")  # 关联的车间xpathipc_definition_menu = (By.XPATH, '//*[text()="终端定义"]')  # 点击终端定义菜单ipc_name_text = (By.ID, "define_name")  # 输入终端名称ipc_address_text = (By.ID, "define_address")  # 输入终端地址ipc_ip_text = (By.ID, "define_ipcIp")  # 输入终端IPipc_remarks_text = (By.ID, "define_remark")  # 输入终端备注说明ipc_name_id = (By.ID, "define_ipcList")  # 关联的工控机idipc_name_xpath = (By.XPATH, "//*[@id='define_ipcList']")  # 关联的工控机xpathfactory_modeling_menu = (By.XPATH, '//*[text()="工厂建模"]')  # 点击工厂建模菜单factory_id = (By.ID, "define_factoryId")  # 关联的工厂idfactory_xpath = (By.XPATH, "//*[@id='define_factoryId']")  # 关联的工厂xpathfactory_name_text = (By.ID, "define_name")  # 输入工厂名称factory_address_text = (By.ID, "define_address")  # 输入工厂地址department_management_menu = (By.XPATH, '//*[text()="部门管理"]')  # 点击部门管理菜单department_id = (By.ID, "define_deptId")  # 关联的部门iddepartment_xpath = (By.XPATH, "//*[@id='define_deptId']")  # 关联的部门xpathdepartment_name_text = (By.ID, "define_name")  # 输入部门名称add_button = (By.XPATH, '//*[text()="新 增"]/..')  # 点击新增按钮enter_xpath = (By.XPATH, "//*[@class='ant-select-search__field']")  # 按回车键confirm_button = (By.XPATH, "//*[text()='确 定']/..")  # 点击确定按钮# 新增设备时选择是否采集yes_no_xpath = (By.XPATH, '//*[@id="define_needCollect"]/label[1]/span[text()="是"]')# 新增工序时是否使用模具tools_yes_no_xpath = (By.XPATH, '//*[@id="define_moudle"]/label[2]/span[text()="否"]')# 物料选择自制或外购self_control_xpath = (By.XPATH, '//*/span[text()="自制"]')# 选择是否主工艺路线process_route_yes_no_xpath = (By.XPATH, '//*[@id="define_techType"]/label[1]/span[text()="是"]')device_definition_menu = (By.XPATH, '//*[text()="设备定义"]')  # 点击设备定义菜单device_type_text = (By.ID, "define_name")  # 输入设备类型device_list_menu = (By.XPATH, '//*[text()="设备列表"]')  # 点击设备列表菜单device_code_text = (By.ID, "define_code")  # 输入设备编码device_name_text = (By.ID, "define_name")  # 输入设备名称device_name_id = (By.ID, "define_deviceIds")  # 关联设备名称iddevice_name_xpath = (By.XPATH, "//*[@id='define_deviceIds']")  # 关联设备名称xpath# 点击新增设备组页面的启用按钮enable_xpath_text = (By.XPATH, '//*[@id="define_status"]/label[1]/span[text()="启用"]')device_type_id = (By.ID, "define_deviceTypeId")  # 设备关联的设备类型iddevice_type_xpath = (By.XPATH, "//*[@id='define_deviceTypeId']")  # 设备关联的设备类型xpathdevices_type_id = (By.ID, "define_type")  # 设备组关联的设备类型iddevices_type_xpath = (By.XPATH, "//*[@id='define_type']")  # 设备组关联的设备类型xpathdevice_supplier_text = (By.ID, "define_brandModel")  # 输入设备供应商devices_list_menu = (By.XPATH, '//*[text()="设备组列表"]')  # 点击设备组列表菜单devices_list_code_text = (By.ID, "define_groupCode")  # 设备组编码devices_list_name_text = (By.ID, "define_groupName")  # 设备组名称process_route_menu = (By.XPATH, '//*[text()="工艺路线"]')  # 点击工艺路线菜单process_route_material_name_id = (By.ID, "define_materialId")  # 工艺路线关联的物料idprocess_route_material_name_xpath = (By.XPATH, "//*[@id='define_materialId']")  # 工艺路线关联的物料xpathprocess_route_procedure_name_id = (By.ID, "define_procedure")  # 工艺路线关联的工序idprocess_route_procedure_name_xpath = (By.XPATH, "//*[@id='define_procedure']")  # 工艺路线关联的工序xpathprocess_route_device_name_id = (By.ID, "define_deviceDoList0")  # 工艺路线关联的设备idprocess_route_device_name_xpath = (By.XPATH, "//*[@id='define_deviceDoList0']")  # 工艺路线关联的设备xpathstandard_box_count_xpath = (By.XPATH, '//*[@id="define_standarNum0"]')  # 标准框数量xpathdriver = None# 使用@classmethod装饰器,setUpClass和 tearDownClass让每类执行只需要开启一次浏览器即可# 使用@classmethod装饰器时不要把要测试的网页放置到setUpClass中那样执行完第一个用例时不会再次打开浏览器@classmethoddef setUpClass(cls):# 忽略告警信息warnings.simplefilter('ignore', ResourceWarning)cls.driver = open_browser()cls.driver.implicitly_wait(10)cls.driver.maximize_window()@classmethoddef tearDownClass(cls):cls.driver.quit()if __name__ == '__main__':# 调用unittest的TestSuite(),理解为管理case的一个容器(测试套件)# unittest框架的自动化测试用例是以0-9、A-Z来排序执行测试用例suite = unittest.TestSuite()runner = unittest.TextTestRunner()

2、新建一个包名:config(用于存放配置文件、读取配置文件函数、业务变量)

(1)在config包下新建一个config.ini配置文件,作用:通过读取配置文件来获取变量。config.py文件代码如下:

# 通过读取配置文件来获取变量[sections]
url = https://www.baidu.com/

(2)在config包下新建一个read_config.py文件,作用:读取配置文件信息。read_config.py文件代码如下:

# coding=utf-8"""
------------------------------------
@Time : 2020/01/15
@Auth : Anker
@File : read_config.py
@purpose:读取配置文件信息
@IDE  : PyCharm
@Motto: Believe in yourself and persistence can make success!
------------------------------------
"""import configparser
import osclass ReadConf:def __init__(self):current_path = os.path.dirname(os.path.relpath(__file__))# 获取配置文件路径config_path = os.path.join(current_path, "config.ini")# 创建管理对象self.conf = configparser.ConfigParser()# 读ini文件self.conf.read(config_path, encoding="utf-8")def readConf(self, param):# 获取所有的section# sections = self.conf.sections()# print(sections)# 获取某个sections中的所有值,将其转化为字典items = dict(self.conf.items(param))return itemsif __name__ == '__main__':test = ReadConf()url = test.readConf("sections")   # 传入sections的值print('sections值为: ', url)

(3)在config包下新建一个variable.py文件,作用:业务关联的变量。variable.py文件代码如下:

# coding=utf-8"""
------------------------------------
@Time : 2020/01/15
@Auth : Anker
@File : variable.py
@purpose:业务关联的变量
@IDE  : PyCharm
@Motto: Believe in yourself and persistence can make success!
------------------------------------
"""import time
import random                                     # random模块用于随机数生成# 获取系统当前时间,并在1~1000中随机生成一个动态变量数值
# date_prefix = time.strftime("%Y-%m-%d %H:%M:%S") + "_编号" + str(random.randint(1, 1000))
date_prefix = time.strftime("%Y%m%d") + "_编号" + str(random.randint(1, 1000))factory_name = "中国" + date_prefix
factory_address = "浙江" + date_prefix
department_name = "测试部" + date_prefix
workshop_code = "chejian" + date_prefix
workshop_name = "车间" + date_prefix
workshop_ip = "192.168.1.1"
workshop_management = "管理员"
ipc_name = "终端" + date_prefix
ipc_address = "浙江" + date_prefix
ipc_ip = "1"
device_type = "设备类型" + date_prefix
device_code = "shebei" + date_prefix
device_name = "设备" + date_prefix
device_group_list_code = "设备组编码" + date_prefix
device_group_list = "设备组" + date_prefix
material_name = "测试物料" + date_prefix
material_code = "wuliao" + date_prefix
material_code_path = '//*[text()="' + material_code + '"]'
procedure_name = "测试工序" + date_prefix
process_route = "主工艺路线"
orderNo = "order" + date_prefix
first_quality_contests = "首检" + date_prefix
first_quality_types = "文本输入型"
patrol_quality_contests = "巡检" + date_prefix
patrol_quality_types = "文本输入型"
first_check_programme = "首检方案" + date_prefix
patrol_check_programme = "巡检方案" + date_prefix
mould_warehouse = "模具仓库" + date_prefix
mould_warehouse_address = "中国科技" + date_prefix
tools_library = "工装仓库" + date_prefix
tools_library_address = "中国科技" + date_prefix
mould_storehouse = "模具库位" + date_prefix
mould_storehouse_address = "中国科技" + date_prefix
tool_storehouse = "工装库位" + date_prefix
tool_storehouse_address = "中国科技" + date_prefix
mould_code = "mould" + date_prefix
mould_name = "模具" + date_prefix
device_supplier = "供应商" + date_prefix

3、新建一个包名:report(用于存放测试报告和发送邮件函数)

(1)在report包下新建一个send_email.py文件,作用:发送邮件。send_email.py文件代码如下:

# coding=utf-8"""
------------------------------------
@Time : 2020/01/15
@Auth : Anker
@File : send_email.py
@purpose:发送邮件
@IDE  : PyCharm
@Motto: Believe in yourself and persistence can make success!
------------------------------------
"""import smtplib
from email.mime.multipart import MIMEMultipart
from email.mime.text import MIMEText
import osdef send_email():'''发送测试报告到邮箱:param::return:'''# ----------1.跟发件相关的参数------# 发件服务器smtp_server = "smtp.qq.com"port = 25  # 端口# 发送者账号sender = "456789@qq.com"# 发送者密码password = "abc456789"# 接收人receiver = "456789@qq.com"# ----------2.编辑邮件的内容--------# 读文件# os.getcwd()  # 获取当前路径# os.path.abspath('..')  # 获取上级路径,一个.表示上级目录,两个.表示上两级目录current_path = os.path.abspath('report')  # 读取report文件夹里的report.htmlfile_path = os.path.join(current_path, "report.html")with open(file_path, "rb") as fp:mail_body = fp.read()message = MIMEMultipart()message["from"] = sender                    # 发件人message["to"] = receiver                    # 收件人message["subject"] = "UI自动化测试"   # 主题# 正文# 通过decode()方法转换字符body = MIMEText('<h1 style="color:red;font-size:100px">大家好:<h1 \n h1>UI自动化测试执行结果如下,''详情请见附件</h1><img src="cid:small">' + mail_body.decode(), "html", "utf-8")message.attach(body)# 附件enclosure = MIMEText(mail_body, "base64", "utf-8")enclosure["Content-Type"] = "application/octet-stream"enclosure["Content-Disposition"] = 'attachment; filename="report.html"'message.attach(enclosure)# ----------3.发送邮件--------------try:smtp = smtplib.SMTP()smtp.connect(smtp_server)  # 连接服务器smtp.login(sender, password)except:smtp = smtplib.SMTP_SSL(smtp_server, port)smtp.login(sender, password)  # 登录smtp.sendmail(sender, receiver, message.as_string())  # 发送smtp.quit()

4、新建一个包名:test_case(用于存放测试用例)

(1)在test_case包下新建一个test_login.py文件,作用:登录测试用例。test_login.py文件代码如下:

# coding=utf-8"""
------------------------------------
@Time : 2020/01/15
@Auth : Anker
@File : test_login.py
@purpose:测试用户登录
@IDE  : PyCharm
@Motto: Believe in yourself and persistence can make success!
------------------------------------
"""import unittest
from common import unit
from common.login import LoginPageclass LoginTest(unit.Unit):def user_login_verify(self, username="", password=""):LoginPage(self.driver).login(username, password)def test_login1(self):'''用户名、密码正确'''self.user_login_verify(username="1828888888", password="000000")def test_login2(self):'''用户名正确,密码不正确'''self.user_login_verify(username="1828888888", password="111")def test_login3(self):'''用户名不正确,密码正确'''self.user_login_verify(username="aaa", password="000000")def test_login4(self):'''用户名、密码都不正确'''self.user_login_verify(username="aaa", password="111111")def test_login5(self):'''用户名、密码为空'''self.user_login_verify(username=" ", password=" ")def test_login6(self):'''用户名为空,密码不为空'''self.user_login_verify(username=" ", password="090909")def test_login7(self):'''用户名不为空,密码为空'''self.user_login_verify(username="123", password=" ")if __name__ == '__main__':unittest.main()

(2)在test_case包下新建一个test_PC.py文件,作用:业务测试用例(冒烟测试)。test_PC.py文件代码如下:

# coding=utf-8"""
------------------------------------
@Time : 2020/01/15
@Auth : Anker
@File : test_PC.py
@purpose:冒烟测试用例
@IDE  : PyCharm
@Motto: Believe in yourself and persistence can make success!
------------------------------------
"""import time
from selenium.webdriver.common.keys import Keys   # 获取键盘事件
from common import unit
from common.login import LoginPage
from config import variable                       # 导入配置文件里的变量名class TestCase(unit.Unit):def test_case01_add_factory(self):"""新增工厂"""# 调用登录方法LoginPage(self.driver).login()time.sleep(2)# 展开知识图谱LoginPage(self.driver).find_element(*self.knowledge_graph_menu).click()time.sleep(1)# 进入工厂建模页面LoginPage(self.driver).find_element(*self.factory_modeling_menu).click()time.sleep(1)# 点击"新增"按钮LoginPage(self.driver).find_element(*self.add_button).click()time.sleep(1)# 输入工厂名称LoginPage(self.driver).find_element(*self.factory_name_text).send_keys(variable.factory_name)# 输入工厂地址LoginPage(self.driver).find_element(*self.factory_address_text).send_keys(variable.factory_address)# 点击"确定"按钮LoginPage(self.driver).find_element(*self.confirm_button).click()print("新增的工厂:", variable.factory_name)def test_case02_add_department(self):"""新增部门"""time.sleep(2)# 展开系统管理LoginPage(self.driver).find_element(*self.system_management_menu).click()time.sleep(1)# 进入部门管理页面LoginPage(self.driver).find_element(*self.department_management_menu).click()time.sleep(1)# 点击"新增"按钮LoginPage(self.driver).find_element(*self.add_button).click()time.sleep(1)# 输入部门名称LoginPage(self.driver).find_element(*self.department_name_text).send_keys(variable.department_name)time.sleep(1)# 关联的工厂名称LoginPage(self.driver).find_element(*self.factory_id).click()time.sleep(1)LoginPage(self.driver).find_elements(*self.factory_xpath)[1].send_keys(variable.factory_name)time.sleep(1)LoginPage(self.driver).find_element(*self.enter_xpath).send_keys(Keys.ENTER)time.sleep(1)# 点击“确定”按钮LoginPage(self.driver).find_element(*self.confirm_button).click()print("新增的部门:", variable.department_name)def test_case03_add_workshop(self):"""新增车间"""time.sleep(2)# 展开知识图谱LoginPage(self.driver).find_element(*self.knowledge_graph_menu).click()time.sleep(1)# 进入工厂建模页面LoginPage(self.driver).find_element(*self.factory_modeling_menu).click()time.sleep(1)# 进入车间定义页面LoginPage(self.driver).find_element(*self.workshop_definition_menu).click()time.sleep(1)# 点击"新增"按钮LoginPage(self.driver).find_element(*self.add_button).click()time.sleep(1)# 输入车间编码LoginPage(self.driver).find_element(*self.workshop_code_text).send_keys(variable.workshop_code)# 输入车间名称LoginPage(self.driver).find_element(*self.workshop_name_text).send_keys(variable.workshop_name)time.sleep(1)# 关联的部门名称LoginPage(self.driver).find_element(*self.department_id).click()time.sleep(1)LoginPage(self.driver).find_elements(*self.department_xpath)[1].send_keys(variable.department_name)time.sleep(1)LoginPage(self.driver).find_element(*self.enter_xpath).send_keys(Keys.ENTER)time.sleep(1)# 输入车间IPLoginPage(self.driver).find_element(*self.workshop_ip_text).send_keys(variable.workshop_ip)# 输入车间负责人LoginPage(self.driver).find_element(*self.workshop_management_text).send_keys(variable.workshop_management)# 点击"确定"按钮LoginPage(self.driver).find_element(*self.confirm_button).click()print("新增的车间:", variable.workshop_name)def test_case04_add_ipc(self):"""新增终端"""time.sleep(2)# 进入终端定义页面LoginPage(self.driver).find_element(*self.ipc_definition_menu).click()time.sleep(1)# 点击"新增"按钮LoginPage(self.driver).find_element(*self.add_button).click()time.sleep(1)# 输入终端名称LoginPage(self.driver).find_element(*self.ipc_name_text).send_keys(variable.ipc_name)# 输入终端地址LoginPage(self.driver).find_element(*self.ipc_address_text).send_keys(variable.ipc_address)# 输入终端ipLoginPage(self.driver).find_element(*self.ipc_ip_text).send_keys(variable.ipc_ip)# 输入终端备注信息LoginPage(self.driver).find_element(*self.ipc_remarks_text).send_keys("测试环境")time.sleep(1)# 点击"确定"按钮LoginPage(self.driver).find_element(*self.confirm_button).click()print("新增的终端:", variable.ipc_name)print("终端ip:", variable.ipc_ip)def test_case05_add_device_type(self):"""新增设备类型"""time.sleep(2)# 进入设备定义页面LoginPage(self.driver).find_element(*self.device_definition_menu).click()time.sleep(1)# 点击"新增"按钮LoginPage(self.driver).find_element(*self.add_button).click()time.sleep(1)# 输入设备类型LoginPage(self.driver).find_element(*self.device_type_text).send_keys(variable.device_type)# 点击"确定"按钮LoginPage(self.driver).find_element(*self.confirm_button).click()print("新增的设备类型:", variable.device_type)def test_case06_add_device(self):"""新增设备"""time.sleep(2)# 进入设备列表页面LoginPage(self.driver).find_element(*self.device_list_menu).click()time.sleep(1)# 点击"新增"按钮LoginPage(self.driver).find_element(*self.add_button).click()time.sleep(1)# 输入设备编码LoginPage(self.driver).find_element(*self.device_code_text).send_keys(variable.device_code)# 输入设备名称LoginPage(self.driver).find_element(*self.device_name_text).send_keys(variable.device_name)time.sleep(1)# 关联的设备类型LoginPage(self.driver).find_element(*self.device_type_id).click()time.sleep(1)LoginPage(self.driver).find_elements(*self.device_type_xpath)[1].send_keys(variable.device_type)time.sleep(1)LoginPage(self.driver).find_elements(*self.enter_xpath)[1].send_keys(Keys.ENTER)time.sleep(1)# 绑定的工控机LoginPage(self.driver).find_element(*self.ipc_name_id).click()time.sleep(1)LoginPage(self.driver).find_elements(*self.ipc_name_xpath)[1].send_keys(variable.ipc_name)time.sleep(1)LoginPage(self.driver).find_elements(*self.enter_xpath)[2].send_keys(Keys.ENTER)time.sleep(1)# 是否采集LoginPage(self.driver).find_element(*self.yes_no_xpath).click()time.sleep(1)# 所属车间LoginPage(self.driver).find_element(*self.workshop_name_id).click()time.sleep(1)LoginPage(self.driver).find_elements(*self.workshop_name_xpath)[1].send_keys(variable.workshop_name)time.sleep(1)LoginPage(self.driver).find_elements(*self.enter_xpath)[3].send_keys(Keys.ENTER)time.sleep(1)# 输入设备供应商LoginPage(self.driver).find_element(*self.device_supplier_text).send_keys(variable.device_supplier)time.sleep(1)# 点击"确定"按钮LoginPage(self.driver).find_element(*self.confirm_button).click()print("新增的设备:", variable.device_name)print("设备供应商:", variable.device_supplier)def test_case07_add_device_group_list(self):"""新增设备组"""time.sleep(2)# 进入设备组列表页面LoginPage(self.driver).find_element(*self.devices_list_menu).click()time.sleep(1)# 点击"新增"按钮LoginPage(self.driver).find_element(*self.add_button).click()time.sleep(1)# 输入设备组编码LoginPage(self.driver).find_element(*self.devices_list_code_text).send_keys(variable.devices_list_code)# 输入设备组名称LoginPage(self.driver).find_element(*self.devices_list_name_text).send_keys(variable.devices_list_name)time.sleep(1)# 关联的设备LoginPage(self.driver).find_element(*self.device_name_id).click()time.sleep(1)LoginPage(self.driver).find_elements(*self.device_name_xpath)[1].send_keys(variable.device_name)time.sleep(1)LoginPage(self.driver).find_elements(*self.enter_xpath)[1].send_keys(Keys.ENTER)time.sleep(1)# 点击"启用"按钮LoginPage(self.driver).find_element(*self.enable_xpath_text).click()time.sleep(1)# 关联的设备类型LoginPage(self.driver).find_element(*self.devices_type_id).click()time.sleep(1)LoginPage(self.driver).find_elements(*self.devices_type_xpath)[1].send_keys(variable.device_type)time.sleep(1)LoginPage(self.driver).find_elements(*self.enter_xpath)[2].send_keys(Keys.ENTER)time.sleep(1)# 绑定的工控机LoginPage(self.driver).find_element(*self.ipc_name_id).click()time.sleep(1)LoginPage(self.driver).find_elements(*self.ipc_name_xpath)[1].send_keys(variable.ipc_name)time.sleep(1)LoginPage(self.driver).find_elements(*self.enter_xpath)[3].send_keys(Keys.ENTER)time.sleep(1)# 点击"启用"按钮(再次点击启用按钮的原因是:当选择绑定的工控机,按回车键后,需点击其它地方,才能将光标跳转到下个车间选择框)LoginPage(self.driver).find_element(*self.enable_xpath_text).click()time.sleep(1)# 所属车间LoginPage(self.driver).find_element(*self.workshop_name_id).click()time.sleep(1)LoginPage(self.driver).find_elements(*self.workshop_name_xpath)[1].send_keys(variable.workshop_name)time.sleep(1)LoginPage(self.driver).find_elements(*self.enter_xpath)[4].send_keys(Keys.ENTER)time.sleep(1)# 点击"确定"按钮LoginPage(self.driver).find_element(*self.confirm_button).click()print("新增的设备组:", variable.devices_list_name)def test_case08_add_material(self):"""新增物料"""time.sleep(2)# 进入物料定义页面LoginPage(self.driver).find_element(*self.material_definition_menu).click()time.sleep(1)# 点击"新增"按钮LoginPage(self.driver).find_element(*self.add_button).click()time.sleep(1)# 输入物料名称LoginPage(self.driver).find_element(*self.material_name_text).send_keys(variable.material_name)# 输入物料编码LoginPage(self.driver).find_element(*self.material_code_text).send_keys(variable.material_code)# 点击选择"自制"LoginPage(self.driver).find_element(*self.self_control_xpath).click()# 输入计量单位LoginPage(self.driver).find_element(*self.material_unit_text).send_keys("kg")# 点击"确定"按钮LoginPage(self.driver).find_element(*self.confirm_button).click()print("新增的物料:", variable.material_name)def test_case09_add_BOM(self):"""新增BOM"""time.sleep(2)# 进入BOM定义页面LoginPage(self.driver).find_element(*self.BOM_definition_menu).click()time.sleep(1)# 点击"新增"按钮LoginPage(self.driver).find_element(*self.add_button).click()time.sleep(1)# 关联的物料LoginPage(self.driver).find_element(*self.BOM_material_name_id).click()time.sleep(1)LoginPage(self.driver).find_elements(*self.BOM_material_name_xpath)[1].send_keys(variable.material_name)time.sleep(1)LoginPage(self.driver).find_elements(*self.enter_xpath)[1].send_keys(Keys.ENTER)time.sleep(1)# 点击"确定"按钮LoginPage(self.driver).find_element(*self.confirm_button).click()def test_case10_add_procedure(self):"""新增工序"""time.sleep(2)# 进入工序定义页面LoginPage(self.driver).find_element(*self.procedure_definition_menu).click()time.sleep(1)# 点击"新增"按钮LoginPage(self.driver).find_element(*self.add_button).click()time.sleep(1)# 输入工序名称LoginPage(self.driver).find_element(*self.procedure_name_text).send_keys(variable.procedure_name)time.sleep(1)# 关联设备类型LoginPage(self.driver).find_element(*self.procedure_device_type_id).click()time.sleep(1)LoginPage(self.driver).find_elements(*self.procedure_device_type_xpath)[1].send_keys(variable.device_type)time.sleep(1)LoginPage(self.driver).find_element(*self.enter_xpath).send_keys(Keys.ENTER)time.sleep(1)# 是否使用模具LoginPage(self.driver).find_element(*self.tools_yes_no_xpath).click()time.sleep(1)# 点击"确定"按钮LoginPage(self.driver).find_element(*self.confirm_button).click()print("新增的工序:", variable.procedure_name)def test_case11_add_process_route(self):"""新增工艺路线"""standard_box_count = str(variable.random.randint(1, 1000))  # 标准框数量随机生成time.sleep(2)# 进入工艺路线页面LoginPage(self.driver).find_element(*self.process_route_menu).click()time.sleep(1)# 判断新增的物料是否配置过工艺路线res = LoginPage(self.driver).isElementPresent(variable.material_code_path)if res:print("该物料已配置工艺路线")time.sleep(1)# 跳出工艺路线页面,进入知识图谱菜单LoginPage(self.driver).find_element(*self.knowledge_graph_menu).click()else:# 点击"新增"按钮LoginPage(self.driver).find_element(*self.add_button).click()time.sleep(1)# 关联物料名称LoginPage(self.driver).find_element(*self.process_route_material_name_id).click()time.sleep(1)LoginPage(self.driver).find_elements(*self.process_route_material_name_xpath)[1].\send_keys(variable.material_name)time.sleep(1)LoginPage(self.driver).find_elements(*self.enter_xpath)[1].send_keys(Keys.ENTER)time.sleep(1)# 选择是否主工艺路线LoginPage(self.driver).find_element(*self.process_route_yes_no_xpath).click()time.sleep(1)# 关联的工序名称LoginPage(self.driver).find_element(*self.process_route_procedure_name_id).click()time.sleep(1)LoginPage(self.driver).find_elements(*self.process_route_procedure_name_xpath)[1].\send_keys(variable.procedure_name)time.sleep(1)LoginPage(self.driver).find_elements(*self.enter_xpath)[2].send_keys(Keys.ENTER)time.sleep(1)# 关联设备LoginPage(self.driver).find_element(*self.process_route_device_name_id).click()time.sleep(1)LoginPage(self.driver).find_elements(*self.process_route_device_name_xpath)[1].\send_keys(variable.device_name)time.sleep(1)LoginPage(self.driver).find_elements(*self.enter_xpath)[2].send_keys(Keys.ENTER)time.sleep(1)# 默认标准框数量LoginPage(self.driver).find_element(*self.standard_box_count_xpath).click()time.sleep(1)LoginPage(self.driver).find_element(*self.standard_box_count_xpath).send_keys(standard_box_count)time.sleep(1)# 点击"确定"按钮LoginPage(self.driver).find_element(*self.confirm_button).click()

5、新建一个run_all_test_case.py文件,作用:业务测试用例(新增工厂)。run_all_test_case.py文件代码如下:

# coding=utf-8"""
------------------------------------
@Time : 2020/01/10
@Auth : Anker
@File : run_all_test_case.py
@purpose:执行所有测试用例
@IDE  : PyCharm
@Motto: Believe in yourself and persistence can make success!
------------------------------------
"""import unittest
from common import HTMLTestRunner
import os
from report.send_email import send_emailcurrent_path = os.getcwd()  # 当前文件路径
case_path = os.path.join(current_path, "test_case")  # 用例路径# 存放报告路径
report_path = os.path.join(current_path, "report")# discover找出以test开头的用例
def all_case():discover = unittest.defaultTestLoader.discover(case_path, "test*.py")return discoverif __name__ == "__main__":# 测试报告为report.htmlresult_path = os.path.join(report_path, "report.html")# 打开文件,把结果写进文件中,w,有内容的话,清空了再写进去fp = open(result_path, "wb")runner = HTMLTestRunner.HTMLTestRunner(stream=fp,title="UI自动化测试报告",description="用例执行情况")# 调用all_case函数返回值runner.run(all_case())# 关闭刚才打开的文件fp.close()# 调用发送邮件方法send_email()

po+selenium+unittest自动化测试项目实战相关推荐

  1. pytest测试实战 电子书_电子书丨Selenium 3+Python 3自动化测试项目实战:从菜鸟到高手...

    ▊<Selenium 3+Python 3自动化测试项目实战:从菜鸟到高手> 田春成 著 电子书售价:39.5元 2019年9月出版 Selenium是目前非常流行的一种自动化测试工具.本 ...

  2. python3自动化测试书籍推荐_免费送书 | 《Selenium 3+Python 3自动化测试项目实战:从菜鸟到高手》...

    点击上方蓝色字体,关注我们 免费送书 2019年就剩「2个月」了,你的读书计划进行得如何? 莫踌躇,光荣之路马上送你本书! 读完它,请为你的2019年画上个美丽的圈圈! <Selenium 3+ ...

  3. Selenium Web 自动化 - 项目实战(三)

    Selenium Web 自动化 - 项目实战(三) 2016-08-10 目录 1 关键字驱动概述 2 框架更改总览 3 框架更改详解   3.1 解析新增页面目录   3.2 解析新增测试用例目录 ...

  4. WindowsGUI自动化测试项目实战+辛酸过程+经验分享

    WindowsGUI自动化测试项目实战+辛酸过程+经验分享 一.前言 ⚜ 起因 ⚜ 项目要求 ⚜ 预研过程 ⚜⚜ 框架选型 ⚜⚜ 关于UIaotumation框架 ⚜ 预研成果 二.项目介绍

  5. Python+Selenium自动化测试项目实战

    第 1 章 自动化测试 1.1.自动化测试介绍 自动化测试就是通过自动化测试工具帮我们打开浏览器,输入网址,输入账号密码登录,及登录后的操作,总的说来自动化测试就是通过自动化测试脚本来帮我们从繁琐重复 ...

  6. 一文8个步骤从0到1实现Python+Selenium自动化测试项目实战【建议收藏】

    目录 第 1 章.自动化测试 第 2 章.Python基础 第 3 章.元素定位方式 第 4 章.元素 | 浏览器操作方法 第 5 章.元素等待 第 6 章.鼠标和键盘操作 第 7 章.UnitTes ...

  7. Python+Selenium自动化测试项目实战【建议收藏】

    第 1 章 自动化测试 1.1.自动化测试介绍 自动化测试就是通过自动化测试工具帮我们打开浏览器,输入网址,输入账号密码登录,及登录后的操作,总的说来自动化测试就是通过自动化测试脚本来帮我们从繁琐重复 ...

  8. 精品向丨软件测试企业级Web自动化测试项目实战(附完整项目)

    今天给大家分享一个简单易操作的实战项目(已开源) 项目名称 ET开源商场系统 项目描述 ETshop是一个电子商务B2C电商平台系统,功能强大,安全便捷.适合企业及个人快速构建个性化网上商城. 包含P ...

  9. Python + Selenium + UnitTest自动化测试面试题目整理

    1.如何使用Selenium进行自动化测试 ? 结合Python + Selenium + UnitTest 2.自动化测试框架 分层+PO模式: 包含基础封装层BasePage.PO页面对象层.Te ...

  10. Python核心场景自动化测试项目实战(二)

    往期关联文章回顾: Pytest+Allure+Jenkins接口自动化项目实战(一) [Python篇]核心场景接口自动化方案(一) 以前我们用python+unittest+HTMLTestRun ...

最新文章

  1. Win7如何快速打开命令提示符
  2. 在C/C++语言中使用正则表达式
  3. JAVA实现创建Excel表并导出(转发)
  4. 为啥非要用Python?Excel不香吗?
  5. linux-tar命令详解
  6. SCOI2019凉凉记
  7. iPhone开发视频教程 Objective-C部分 (51课时)
  8. android 上下收缩动画,Android 带有弹出收缩动画的扇形菜单实例
  9. 内置方法 __new__ __del__
  10. ASP.NET Core使用Session
  11. 听某个老师的ElasticSearch记的笔记了
  12. 是时候搞清楚煎饼大妈的真实收入了!来看看专业的研究流程
  13. Paddle 印刷电路板(PCB)瑕疵检测
  14. javascript 动态设置样式style
  15. 学期计算机教学工作反思,小学信息技术老师一月工作反思总结教育教学笔记
  16. View 事件分发机制
  17. 对话系统 | (1) 任务导向型对话系统 -- 对话管理模型研究最新进展
  18. 多种卫星遥感数据反演光合有效辐射数据服务
  19. 【云和恩墨】嵌入云端:12c Policy-Managed Cluster为Oracle DBaaS助力
  20. 深户集体户口借出须知

热门文章

  1. brep文件在线预览
  2. 简易电影售票系统(附部分总结)
  3. 18、HX1838红外遥控模块控制led
  4. java打印sscil码_SSCI期刊投稿指南库
  5. matlab errorbar 例子,科学网—【MATLAB】如何画水平errorbar - 叶瑞杰的博文
  6. Python学习之——np.dot()与np.multiply()与*之间的区别
  7. Java反编译xml实现_如何反编译apk文件得到源码和XML文件
  8. 用yum下载安装gcc
  9. c++ 获取硬盘序列号serialnumber
  10. tracepro中文pojie版-tracepro附安装教程