po+selenium+unittest自動化測試項目實戰

一、項目工程目錄:

二、具體工程文件代碼:

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 ReadConf
 
 
class BasePage(object):
 
    # 讀取config.ini配置文件,傳入sections值
    url = ReadConf()
    # 傳入sections模塊
    standard_url = url.readConf("sections")
    # 這裡傳入sections模塊中的url
    base_url = standard_url['url']
 
    def __init__(self, driver, test_url=base_url):
        """
        構造函數
        :param driver
        :param 傳入url
        """
        self.driver = driver
        self.url = test_url
        # 設置全局元素隱式等待時間為10秒鐘
        self.driver.implicitly_wait(10)
 
    def open_url(self):
        """
        打開url
        :param:
        """
        url = self.url
        self.driver.get(url)
        title = self.driver.title
        print("項目名稱:", 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.loc
            loc = 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 True
        except:
            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 driver
    if 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 driver
 
 
if __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 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="my_stylesheet.css" rel="external nofollow"  type="text/css">'
    # run the test
    runner.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 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 "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 failed
Version 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 datetime
 
import sys
import unittest
from xml.sax import saxutils
 
PY3K = (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 = 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)
 
 
# ----------------------------------------------------------------------
# Template
 
class 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"/>
    %(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.nextElementSibling
    obj1.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=i
            changeImg(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" 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  {
    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)" rel="external nofollow" >詳情</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')" rel="external nofollow"  >
        %(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="#" rel="external nofollow"  rel="external nofollow"   onclick="show_img(this)">顯示截圖</a>
    <div align="center" class="screenshots"  style="display:none">
        <a class="close_shots"  href="#" rel="external nofollow"  rel="external nofollow"    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 value
        else:
            if isinstance(value, str):
                return value.decode("utf-8")
            else:
                return value
 
 
TestResult = unittest.TestResult
 
 
class _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 = 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.retry = retry
        self.trys = 0
        self.status = 0
        self.save_last_try = save_last_try
        self.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.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.
        if self.retry:
            if self.status == 1:
                self.trys += 1
                if self.trys <= self.retry:
                    if self.save_last_try:
                        t = self.result.pop(-1)
                        if t[0]==1:
                            self.failure_count-=1
                        else:
                            self.error_count -= 1
                    test=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 = desc
                    test(self)
                else:
                    self.status = 0
                    self.trys = 0
        self.complete_output()
 
    def addSuccess(self, test):
        self.success_count += 1
        self.status = 0
        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
        self.status = 1
        TestResult.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",""):
            pass
        else:
            try:
                driver = getattr(test, "driver")
                test.imgs.append(driver.get_screenshot_as_base64())
            except Exception:
                pass
        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
        self.status = 1
        TestResult.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",""):
            pass
        else:
            try:
                driver = getattr(test, "driver")
                test.imgs.append(driver.get_screenshot_as_base64())
            except Exception as e:
                pass
        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=2, title=None, description=None, retry=0,save_last_try=False):
        self.stream = stream
        self.retry = retry
        self.save_last_try=save_last_try
        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, 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 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(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_TMPL
 
    def _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 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 += 1
                elif n == 1:
                    nf += 1
                else:
                    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
            if 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 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]
        if self.verbosity > 1:
            doc = t._testMethodDoc or ''
        else:
            doc = ""
 
        desc = doc and ('%s: %s' % (name, doc)) or name
        if 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 = o
            else:
                uo = o.decode('utf-8', 'ignore')
        else:
            uo = o
        if isinstance(e, str):
            # ue = unicode(e.encode('string_escape'))
            if PY3K:
                ue = e
            elif 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 = e
 
        script = 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="data:image/jpg;base64,%s" style="display: block;" class="img"/>\n""" % img
                else:
                    tmp+=""" <img src="data:image/jpg;base64,%s" style="display: none;" class="img"/>\n""" % img
            imgs = 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:
            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)

(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 BasePage
 
 
class 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_browser
 
 
class 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關聯的物料id
    BOM_material_name_xpath = (By.XPATH, "//*[@id='material_sourceId']")  # BOM關聯的物料xpath
    BOM_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")  # 工序關聯的設備類型id
    procedure_device_type_xpath = (By.XPATH, "//*[@id='define_typeId']")  # 工序關聯的設備類型xpath
 
    workshop_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")  # 輸入車間IP
    workshop_management_text = (By.ID, "define_management")  # 輸入車間負責人
    workshop_name_id = (By.ID, "define_workshopId")  # 關聯的車間id
    workshop_name_xpath = (By.XPATH, "//*[@id='define_workshopId']")  # 關聯的車間xpath
 
    ipc_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")  # 輸入終端IP
    ipc_remarks_text = (By.ID, "define_remark")  # 輸入終端備註說明
    ipc_name_id = (By.ID, "define_ipcList")  # 關聯的工控機id
    ipc_name_xpath = (By.XPATH, "//*[@id='define_ipcList']")  # 關聯的工控機xpath
 
    factory_modeling_menu = (By.XPATH, '//*[text()="工廠建模"]')  # 點擊工廠建模菜單
    factory_id = (By.ID, "define_factoryId")  # 關聯的工廠id
    factory_xpath = (By.XPATH, "//*[@id='define_factoryId']")  # 關聯的工廠xpath
    factory_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")  # 關聯的部門id
    department_xpath = (By.XPATH, "//*[@id='define_deptId']")  # 關聯的部門xpath
    department_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")  # 關聯設備名稱id
    device_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")  # 設備關聯的設備類型id
    device_type_xpath = (By.XPATH, "//*[@id='define_deviceTypeId']")  # 設備關聯的設備類型xpath
    devices_type_id = (By.ID, "define_type")  # 設備組關聯的設備類型id
    devices_type_xpath = (By.XPATH, "//*[@id='define_type']")  # 設備組關聯的設備類型xpath
 
    device_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")  # 工藝路線關聯的物料id
    process_route_material_name_xpath = (By.XPATH, "//*[@id='define_materialId']")  # 工藝路線關聯的物料xpath
    process_route_procedure_name_id = (By.ID, "define_procedure")  # 工藝路線關聯的工序id
    process_route_procedure_name_xpath = (By.XPATH, "//*[@id='define_procedure']")  # 工藝路線關聯的工序xpath
    process_route_device_name_id = (By.ID, "define_deviceDoList0")  # 工藝路線關聯的設備id
    process_route_device_name_xpath = (By.XPATH, "//*[@id='define_deviceDoList0']")  # 工藝路線關聯的設備xpath
 
    standard_box_count_xpath = (By.XPATH, '//*[@id="define_standarNum0"]')  # 標準框數量xpath
 
    driver = None
    # 使用@classmethod裝飾器,setUpClass和 tearDownClass讓每類執行隻需要開啟一次瀏覽器即可
    # 使用@classmethod裝飾器時不要把要測試的網頁放置到setUpClass中那樣執行完第一個用例時不會再次打開瀏覽器
 
    @classmethod
    def setUpClass(cls):
        # 忽略告警信息
        warnings.simplefilter('ignore', ResourceWarning)
        cls.driver = open_browser()
        cls.driver.implicitly_wait(10)
        cls.driver.maximize_window()
 
    @classmethod
    def 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 os
 
 
class 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 items
 
 
if __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 os
 
 
def send_email():
    '''
    發送測試報告到郵箱
    :param:
    :return:
    '''
 
    # ----------1.跟發件相關的參數------
    # 發件服務器
    smtp_server = "smtp.qq.com"
    port = 25  # 端口
    # 發送者賬號
    sender = "[email protected]"
    # 發送者密碼
    password = "abc456789"
    # 接收人
    receiver = "[email protected]"
 
    # ----------2.編輯郵件的內容--------
    # 讀文件
    # os.getcwd()  # 獲取當前路徑
    # os.path.abspath('..')  # 獲取上級路徑,一個.表示上級目錄,兩個.表示上兩級目錄
    current_path = os.path.abspath('report')  # 讀取report文件夾裡的report.html
    file_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 LoginPage
 
 
class 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)
        # 輸入車間IP
        LoginPage(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)
        # 輸入終端ip
        LoginPage(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_email
 
current_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 discover
 
 
if __name__ == "__main__":
    # 測試報告為report.html
    result_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自動化測試項目實戰的文章就介紹到這瞭,更多相關po selenium unittest自動化內容請搜索WalkonNet以前的文章或繼續瀏覽下面的相關文章希望大傢以後多多支持WalkonNet!

推薦閱讀: