python3.6+selenium+HtmlTestRunner全自动测试框架

本文提供python3.6+selenium+HtmlTestRunner全自动测试框架源码。该框架特色是可添加定时任务、执行用例、生成含截图的测试报告。思路是从用例集调用用例,参数和xpath存于excel。还介绍了配置文件、封装浏览器、日志处理等内容及测试用例写法。

python3.6+selenium+HtmlTestRunner全自动测试框架源码

概述

特色:可添加定时任务,执行测试用例,生测试报告,测试报告中添加截图功能
思路:先从用例集(testcase)调用每条用例(src>page)目前是从excel中读取xpath(PageElements),之后传入参数(data>.xlsx) 目前所有的入参和xpath都是存储在excel中的,正确的是应该存放在数据库中! 框架使用时,需要维护两个excel,testcase和page用例流程。

  1. 配置文件

    config
    
    -------------------config.ini-------配置文件
    
    data
    
    -------------------test_data.xlsx------入参,譬如登录名和密码
    
    logs
    
    -------------------日志存放地方
    
    mains
    
    -------------------应用启动位置
    
    -------------------__init__.py
    
    -------------------main.py------执行excute()
    
    PageElements
    
    -------------------login.xlsx---------用于存放每个用例上的元素路径信息,每个用例占用一个sheet页
    
    report
    
    -------------------村饭测试报告
    
    screenshots
    
    -------------------存放截图的位置
    
    src
    
    -------------------框架的核心代码
    
    -------------------page
    
    --------------------------------------__init__.py
    
    --------------------------------------login.py-----每个用例的执行流程封装
    
    -------------------public
    
    --------------------------------------__init__.py
    
    --------------------------------------basepage.py-----基类方法封装
    
    --------------------------------------browser_engine.py-----浏览器创建driver方法封装
    
    --------------------------------------logger.py-----记录日日志方法封装
    
    --------------------------------------readexcel.py-----读取excel方法封装
    
    --------------------------------------sendemail.py-----发送邮件方法封装
    
    testcase
    
    -------------------test_CIMsysytem.py----------测试用例集封装,需以test_开头命名
    
    tools
    
    -------------------chromwdriver.exe
    
    -------------------HtmelTestRunner_cn.py---------第三方包存放
    
  2. config文件的内容如下

    [browserType]
    #browserName = Firefox
    browserName = Chrome
    #browserName = IE
    #browserName = CSframe
    [CSframe]
    CSframePath = C:\Users\ZWK\Desktop\tencent.exe
    
    [serverIP]
    server_ip = 127.0.0.1
    
    [port]
    port = 8080
    
    [dataBase]
    dbhost = 199.31.196.68
    dpport = 8080
    dbname = oracle
    username =tencent.com
    password = tencent@
    
    
    [email]
    emailservice = 199.31.153.201
    from_user = zhangwenke@qq.com
    from_user_pwd = aabbccdd!
    to_user = Tencent.com@qq.com
    
    
    [timePoint]
    Point = 18_49
    
    [logLevel]
    #printLevel = NOSET
    #printLevel = DEBUG
    printLevel = INFO
    #printLevel = WARNING
    #printLevel = ERROR
    #printLevel = CRITICAL
    

    注:
    (1) CSframePath这个适用于基于chrome内核开发的客户端,搞懂这个,可是花费了好长时间,走了不少弯路
    (2) 有人获取config.ini配置文件时出错,是因为配置的路径错误,注意base_dir = os.path.dirname(os.path.dirname(os.path.dirname(os.path.abspath(file))))的用法,不会的朋友可以自行百度

  3. 封装浏览器

from selenium.webdriver.chrome.options import Options
import configparser
import sys,os


from src.public import logger
logger=logger.Logger("Browser")


class Browser(object):

   def create_driver(self):
       config = configparser.ConfigParser()
       base_dir = os.path.dirname(os.path.dirname(os.path.dirname(os.path.abspath(__file__))))
       config.read(base_dir + "\config\config.ini")

       browser = config.get("browserType","browserName")
       logger.info("Select %s as your browser!" % browser)

       webdriver_path = os.path.join(base_dir + r"\tools\chromedriver.exe")
       logger.info("There path of wedriver.exe is %s" % webdriver_path)
       CSframePath = config.get("CSframe","CSframePath")

       if browser == "Firefox":
           self.driver = webdriver.Firefox()
           self.driver.maximize_window()
       elif browser == "Chrome":
           chrome_options = Options()
           self.driver = webdriver.Chrome(executable_path=webdriver_path,options=chrome_options)
           self.driver.maximize_window()
       elif browser == "IE":
           self.driver = webdriver.Ie()
       elif browser == "CSframe":
           chrome_options = Options()
           chrome_options.add_argument('--headless')
           chrome_options.add_argument('--no-sandbox')
           chrome_options.add_argument('--disable-dev-shm-usage')
           chrome_options.add_argument('--disable-gpu')
           chrome_options.binary_location = CSframePath
           self.driver = webdriver.Chrome(executable_path=webdriver_path, options=chrome_options)

       else:
           logger.info("There is no such enumeration: %s" % browser)
           raise NameError("Please enter a valid type of targeting elements:%s in config file!" % browser)

       return self.driver
  1. 通用基类封装
from selenium.common.exceptions import NoSuchElementException
from selenium.webdriver.common.action_chains import ActionChains
import os
from src.public import logger
from selenium.webdriver.support.select import Select
from selenium.webdriver.common.keys import Keys
from random import choice

logger = logger.Logger("BasePage")
base_dir = os.path.dirname(os.path.dirname(os.path.dirname(os.path.abspath(__file__))))

class BasePage(object):
   "定义一个页面基类,让所有页面都继承这个类,封装一些常用的页面操作方法到这个类"

   def __init__(self, driver):
       self.driver = driver

   # 截图,保存在根目录下的screenshots
   def take_screenshot(self):
       screen_dir = os.path.join(base_dir + '\screenshots\\')
       if os.path.exists(screen_dir) and os.path.isdir(screen_dir):
           pass
       else:
           os.mkdir(screen_dir)
       rq = time.strftime('%Y%m%d%H%M%S', time.localtime(time.time()))
       screen_name = screen_dir + rq + '.png'
       logger.info(screen_name)
       try:
           self.driver.get_screenshot_as_file(screen_name)
           logger.info("Had take screenshot and saved!")
       except Exception as e:
           logger.error("Failed to take screenshot!", format(e))

   # 查找元素
   def find_element(self, selector, i=None):
       element = ''
       '''
           element={'username':'xpath=>//html/div/1','password':'id=>pwd'}
           selector = element[key]
       '''
       if '=>' not in selector:
           return self.driver.find_element_by_id(selector)
       selector_by = selector.split('=>')[0]
       selector_value = selector.split('=>')[1]
       
       '''
           适用于输入关键字,之后点击匹配的值的方法,如通用类中的input_and_click(),场景就是:系统中,输入债券,点击匹配的值
           element={'username':'xpath=>//html/div/1','password':'xpath=>//html/div/1||//html/div/2'}
           selector = element[key]
       '''

       if "||" in selector_value:
           if selector_by == "x" or selector_by == "xpath":
               selector_value_list = selector_value.split("||")
               num = len(selector_value_list)
               if i <= num-1:#此处注意定义i应用
                   selector_value = selector_value_list[i]
                   element = self.driver.find_element_by_xpath(selector_value)
               else:
                   logger.info("Your input value %s out of range of list length %s" % (i,num))
                   raise NameError("所输入的值超出List的长度!")
           else:
               logger.info("Only support xpath when many tags!")
               raise NameError("多个元素路径在一起时,仅支持通过xpath方式!")

       if selector_by == 'id':
           try:
               element = self.driver.find_element_by_id(selector_value)
               logger.info("Had find the element \' %s \' successful "
                           "by %s via value: %s " % (element.text, selector_by, selector_value))
           except NoSuchElementException as e:
               logger.error("NoSuchElementException: %s" % e)
               self.take_screenshot()
       elif selector_by == "n" or selector_by == 'name':
           element = self.driver.find_element_by_name(selector_value)
       elif selector_by == 'css_selector':
           element = self.driver.find_element_by_css_selector(selector_value)
       elif selector_by == 'classname':
           element = self.driver.find_element_by_class_name(selector_value)
       elif selector_by == "l" or selector_by == 'link_text':
           element = self.driver.find_element_by_link_text(selector_value)
       elif selector_by == "p" or selector_by == 'partial_link_text':
           element = self.driver.find_element_by_partial_link_text(selector_value)
       elif selector_by == "t" or selector_by == 'tag_name':
           element = self.driver.find_element_by_tag_name(selector_value)
       elif selector_by == "x" or selector_by == 'xpath':
           try:
               element = self.driver.find_element_by_xpath(selector_value)
               logger.info("Had find the element \' %s \' successfully "
                           "by %s via value: %s " % (element.text, selector_by, selector_value))
           except NoSuchElementException as e:
               logger.error("NoSuchElementException: %s" % e)
               self.take_screenshot()
       elif selector_by == "s" or selector_by == 'selector_selector':
           element = self.driver.find_element_by_css_selector(selector_value)
       else:
           raise NameError("Please enter a valid type of targeted elements.")

       return element

   # 输入
   def input(self, selector, text):
       el = self.find_element(selector)
       try:
           el.clear()
           el.send_keys(text)
           logger.info("Had type \' %s \' in inputBox" % text)
       except NameError as e:
           logger.error("Failed to type in input box with %s" % e)
           self.take_screenshot()

   def clear(self, selector):
       el = self.find_element(selector)
       try:
           el.clear()
           logger.info("清除输入框文本信息OK")
       except NameError as e:
           logger.error("清除输入框内容失败:抛出异常: %s" % e)

   @staticmethod
   def sleep(seconds):
       time.sleep(seconds)
       logger.info("Sleep for %d seconds" % seconds) #静态方法:不强制要求传递参数,类可以不用实例化就能调用该方法


   # 点击
   def click(self, selector):
       el = self.find_element(selector)
       try:
           el.click()
       except NameError as e:
           logger.error("Failed to click the element with %s" % e)
           self.take_screenshot()

   #切换到最新句柄
   def switch_to_current_handle(self):
       self.driver.switch_to_window(self.driver.window_handles[-1])
       logger.info("The current handle is %s" % self.driver.window_handles[-1])

   # 切到iframe
   def switch_frame(self):
       iframe = self.find_element('classname=>embed-responsive-item')
       try:
           self.driver.switch_to_frame(iframe)
       except NameError as e:
           logger.error("Failed to click the element with %s" % e)
           self.take_screenshot()

   # 处理标准下拉选择框,随机选择
   def select(self, id):
       select1 = self.find_element(id)
       try:
           options_list = select1.find_elements_by_tag_name('option')
           del options_list[0]
           s1 = choice(options_list)
           Select(select1).select_by_visible_text(s1.text)
           logger.info("随机选的是:%s" % s1.text)
       except NameError as e:
           logger.error("Failed to click the element with %s" % e)
           self.take_screenshot()

   #鼠标悬停
   def float(self, selector):
       move = self.find_element(selector)
       ActionChains(self.driver).move_to_element(move).perform()

   # 模拟回车键
   def enter(self, selector):
       e1 = self.find_element(selector)
       e1.send_keys(Keys.ENTER)

   # 模拟TAB键
   def tab(self, selector):
       e1 = self.find_element(selector)
       e1.send_keys(Keys.TAB)

   # 模拟鼠标左击
   def leftclick(self, element):
       ActionChains(self.driver).click(element).perform()

   # 关闭当前窗口
   def close_window(self):
       try:
           self.driver.close()
           logger.info("关闭当前窗口.")
       except NameError as e:
           logger.error("关闭当前窗口出错,抛出错误提示:%s." % e)
           self.take_screenshot()

   def get_page_title(self):
       logger.info("当前打开的url地址标题为:%s" % self.driver.title)
       return self.driver.title

   # 获取警示框,并得到提示框信息和关闭提示框
   def get_alert(self):
       el = self.driver.switch_to.alert  # 获取窗口弹窗的方法
       try:
           assert '用户名或者密码错误' in el.text  # el.text方法获取提示框内容
           logger.info("弹窗提示正确")
           el.accept()  # 点击弹窗确认按钮
       except Exception as e:
           logger.info('弹窗提示错误', format(e))
           self.take_screenshot()


   def isElementExist(self, selector):
       flag = True
       try:
           self.find_element(selector)
           return flag
       except:
           flag = False
           logger.info("Not find element!")
           self.take_screenshot()
           return flag


#-------------------------自定义通用类方法-------------------------------
   # 在输入框中输入值,之后在匹配的结果中点击选择,适用于输入关键字后,选择匹配的选项
   def input_and_click(self,selector,text):#优化的地方就是,直接定义i=0,而不是None
       el1 = self.find_element(selector,0)
       el1.clear()
       time.sleep(0.5)
       try:
           el1.send_keys(text)
           time.sleep(1)
           el2 = self.find_element(selector,1)
           el2.click()
       except Exception as e:
           logger.info('模糊匹配错误!', format(e))
           self.take_screenshot()
  1. logger封装
import logging.handlers
import os
import time


import configparser
config = configparser.ConfigParser()
base_dir = os.path.dirname(os.path.dirname(os.path.dirname(os.path.abspath(__file__))))
config.read(base_dir + "\config\config.ini")
printLevel = config.get("logLevel", "printLevel")


class Logger(object):
   def __init__(self,name):
       self.name = name#定义name后,则不会出现日志重复打印的问题
       self.logger = logging.getLogger(self.name)

       # 设置输出的等级
       LEVELS = {'NOSET': logging.NOTSET,
                 'DEBUG': logging.DEBUG,
                 'INFO': logging.INFO,
                 'WARNING': logging.WARNING,
                 'ERROR': logging.ERROR,
                 'CRITICAL': logging.CRITICAL}
       # 创建文件目录
       base_dir = os.path.dirname(os.path.dirname(os.path.dirname(os.path.abspath(__file__))))
       logs_dir = os.path.join(base_dir+"\logs")
       if os.path.exists(logs_dir) and os.path.isdir(logs_dir):
           pass
       else:
           os.mkdir(logs_dir)
       # 修改log保存位置
       timestamp = time.strftime("%Y-%m-%d", time.localtime())
       logfilename = '%s.txt' % timestamp
       logfilepath = os.path.join(logs_dir, logfilename)
       rotatingFileHandler = logging.handlers.RotatingFileHandler(filename=logfilepath,
                                                                  maxBytes=1024 * 1024 * 50,
                                                                  backupCount=5)#设置日志大小不能超过50M

       # 设置输出格式
       formatter = logging.Formatter('[%(asctime)s] [%(levelname)s] %(message)s', '%Y-%m-%d %H:%M:%S')
       rotatingFileHandler.setFormatter(formatter)
       # 控制台句柄
       console = logging.StreamHandler()
       console.setLevel(logging.NOTSET)
       console.setFormatter(formatter)
       # 添加内容到日志句柄中
       self.logger.addHandler(rotatingFileHandler)
       self.logger.addHandler(console)
       # 设置日志的打印级别
       self.logger.setLevel(LEVELS[printLevel])


   def info(self, message):
       self.logger.info(message)

   def debug(self, message):
       self.logger.debug(message)

   def warning(self, message):
       self.logger.warning(message)

   def error(self, message):
       self.logger.error(message)

注:解决python日志重复打印的问题!
6. 一条测试用例的写法

import time
from src.public.basepage import BasePage
from src.public.readexcel import ReadExcle

from src.public import logger
logger=logger.Logger("LoginPage")

class LoginPage(BasePage, ReadExcle):  # 一定要继承这几个类

  '''加载页面元素位置'''
  global element, base_dir
  base_dir = os.path.dirname(os.path.dirname(os.path.dirname(os.path.abspath(__file__))))
  readelements = ReadExcle(base_dir + "\PageElements\login.xlsx")
  element, listjs = readelements.get('登陆') 

  '''
      element={'username':'xpath=>//html/div/1','password':'id=>pwd'}
  '''
  logger.info("element is :%s " % element)
  logger.info("listjs is :%s " % listjs)

  '''页面操作,调用了basepage里边的方法'''

  def login(username,password)
      self.input(element['username'], username)  
      self.input(element['password'], password)
      self.click(element['login'])
      check_point = self.isElementExist(element["checkTag"])
  	  return check_point
  1. 测试用例集写法
from src.page import login_page
import unittest,os
from src.public.readexcel import ReadExcle
from src.public import logger
logger=logger.Logger("MyTestCase")

base_dir = os.path.dirname(os.path.dirname(os.path.abspath(__file__)))
readcell = ReadExcle(base_dir + "\data\login_data.xlsx")

class MyTestCase(unittest.TestCase):
  '''
  被注释的代码是有问题的,因为有三个cases,而每次调用self.driver的时候,都会重新创建一个driver,也就是说重新打开一个浏览器
  ,所以执行三条用例,就会分别打开三个浏览器,在仅执行一次的setUp,tearDown中,加入classmethod,使用setUpClass

  def setUp(self):
      browser = Browser()
      self.driver = browser.create_driver()
      browser.open_url("http://127.0.0.1:8000/login/")

  def tearDown(self):
      self.driver.quit()


  def test_login(self):
      login_obj = login_page.LoginPage(self.driver)
      asser login_obj.login("zhangwenke@qq.com","123456")
'''


  @classmethod
  def setUpClass(cls):
      browser = Browser()
      cls.driver = browser.create_driver()
      browser.wait(5)
      browser.open_url("http://127.0.0.1:8000/login/")

  @classmethod
  def tearDownClass(cls):
      cls.driver.quit()
      logger.info("End test.........")

  def test_login(self):
      login = login_page.LoginPage(self.driver)
      username = str(readcell.read_cell("Sheet1",1,1))
      password = str(readcell.read_cell("Sheet1",1,2))
      logger.info("[%s,%s]" % (username,password))
      login.all_operate(username,password)
      self.assertEqual(True, login.check_success(), "登录成功")
      self.imgs.append(self.driver.get_screenshot_as_base64())
      '''
          如果想使用HtmlTestRunner_cn生成截图的话,就在每个用例下面加一个self.imgs.append(self.driver.get_screenshot_as_base64())
          即可,其他和原代码HtmlTestRunner无异
      '''

  def test_view_report_1(self):
      pass
  def test_view_report_2(self):
      pass
  def test_view_report_3(self):
      pass
  def test_view_report_4(self):
      pass
  def test_view_report_5(self):
      raise NameError
      self.imgs.append(self.driver.get_screenshot_as_base64())


if __name__ == "__main__":
  unittest.main()#在这个页面中可以调试代码,如果在main.py文件调试的话,获取不到细致的错误信息


  1. 执行文件main的写法
from tools import HTMLTestRunner_cn
import sys,os,time
import configparser


from src.public import logger
logger=logger.Logger("Excute Testcase")

base_dir = os.path.dirname(os.path.dirname(os.path.abspath(__file__)))

def discover():
  suite = unittest.defaultTestLoader.discover(
      start_dir=os.path.join(
          os.path.dirname(__name__),
          'testcase'),  # 这个是待执行用例的目录
      pattern='test_*.py',  # 这个是匹配脚本名称的规则,以test开头的用例
      top_level_dir=None  # 这个是顶层目录的名称,一般默认等于None就行了
  )
  return suite


def excute():
  now = time.strftime('%Y-%m-%d %H-%M-%S')
  report_dir = os.path.join(base_dir+r"\report")
  if os.path.exists(report_dir) and os.path.isdir(report_dir):
      pass
  else:
      os.mkdir(report_dir)
  filePath = report_dir + (r"\pyResult %s.html" % now)
  fp = open(filePath, 'wb')
  runner = HTMLTestRunner_cn.HTMLTestRunner(stream=fp,title='Python Test Report',description='Python Report')

  config = configparser.ConfigParser()
  config.read(base_dir + "\config\config.ini")
  time_point = config.get("timePoint", "Point")
  logger.info('Scheduled time point is %s' % time_point)

  k = 1
  while k < 2:
      timing = time.strftime('%H_%M', time.localtime(time.time()))
      if timing == time_point:# 17_35指17:35,这个可以根据需要设定时间
          logger.info('start to run scripts at %s' % timing)
          runner.run(discover())
          logger.info('Finish runing scripts at %s' % timing)
          break
      else:
          time.sleep(10)#每隔10s访问一次
          print("Wating for scheuled time point >>>[%s]<<<,Now is :>>>>>%s" % (time_point,timing))
      fp.close()
     


if __name__ == "__main__":
  logger.info("Start scheduled testing.........")
  excute()

  1. 邮箱发送封装
import os.path
import configparser
from email.mime.text import MIMEText
from email.header import Header
import smtplib

from src.public import logger
logger=logger.Logger('Email')

class Email(object):

    def send_mail(self,file_new):
        config = configparser.ConfigParser()
        base_dir = os.path.dirname(os.path.dirname(os.path.dirname(os.path.abspath(__file__))))
        config.read(base_dir + "\config\config.ini", encoding='UTF-8')

        emailserver = config.get("emailserver", "emailservice")
        from_user = config.get("emailserver", "from_user")
        from_passwd = config.get("emailserver", "from_user_pwd")
        to_user = config.get("emailserver", "to_user")
        logger.info("E-mail infomation:%s||%s||%s||%s" % (emailserver,from_user,from_passwd,to_user))

        f = open(file_new, 'rb')
        mail_boy = f.read()
        f.close()
        msg = MIMEText(mail_boy, 'html', 'utf-8')  # 定义邮件正文
        msg['Subject'] = Header('V2200自动化测试报告', 'utf-8')  # 定义邮件标题
        smtp = smtplib.SMTP()
        smtp.connect(emailserver)  # 连接邮箱服务器
        logger.info("连接邮箱服务器!")
        smtp.login(from_user, from_passwd)  # 邮件发送方登陆
        smtp.sendmail(from_user, to_user, msg.as_string())  # 邮件发送者和接收者
        smtp.quit()
        logger.info("邮件已经发送,请注意查收!")

    def new_report(self,report_path):
        lists = os.listdir(report_path)  # 得到项目目录下所有的文件和文件夹
        lists.sort(key=lambda fn: os.path.getmtime(report_path + '\\' + fn))  # 将得到的文件和文件夹按创建时间排序
        file_new = os.path.join(report_path, lists[-1])  # 获取最新创建的文件
        logger.info(file_new)
        return file_new

if __name__ == "__main__":
    sample = Email()
    base_dir = os.path.dirname(os.path.dirname(os.path.dirname(os.path.abspath(__file__))))
    report_path = os.path.join(base_dir,"/report")#只需要找到report文件夹的位置即可
    report = sample.new_report(report_path)
    sample.send_mail(report)

  1. excel封装
import xlrd

class ReadExcle(object):
    def __init__(self, file, tag='True'):
        self.file = file
        self.tag = tag
        
    def read(self, sheetname, n=1, num=1000):  # i,sheet索引
        ExcelFile = xlrd.open_workbook(self.file)
        table = ExcelFile.sheet_by_name(sheetname)
        nrows = table.nrows  # 行数
        ncols = table.ncols  # 列数
        j = 0  # 循环次数
        for row in range(1, nrows):
            j += 1
            line = []
            if self.tag == 'True':
                for col in range(0, ncols):
                    line.append(table.cell(row, col).value)
                yield line
            elif self.tag == 'False':
                if j >= n and j < n + num:
                    for col in range(0, ncols):
                        line.append(table.cell(row, col).value)
                    yield line

    def get(self, sheetname):
        ExcelFile = xlrd.open_workbook(self.file)
        sheet = ExcelFile.sheet_by_name(sheetname)  # 'Sheet1'
        nrows = sheet.nrows  # 总行数
        list0 = []  # 元素名称列表
        list1 = []  # 元素路径列表
        list2 = []  # js列表
        for i in range(1, nrows):  # i为行数
            if sheet.row(i)[2].value != 'null':
                r1 = sheet.row(i)[2].value
                r2 = sheet.row(i)[3].value
                list0.append(sheet.row(i)[0].value)
                list1.append(r1 + '=>' + r2)#'=>'与basepage中保持一致
                dict1 = dict(zip(list0, list1))
            else:
                list2.append(sheet.row(i)[3].value)
                """
                dict1={'username':'xpath=>//html/div/1','password':'id=>pwd'}
                """
        return dict1, list2

    def read_cell(self, sheetname, i, j):
        ExcelFile = xlrd.open_workbook(self.file)
        table = ExcelFile.sheet_by_name(sheetname)
        return table.cell(i, j).value

    def read_ncols(self, sheetname, ncols, n=1, num=1000):  # i,sheet索引
        ExcelFile = xlrd.open_workbook(self.file)
        table = ExcelFile.sheet_by_name(sheetname)
        nrows = table.nrows  # 行数
        ncols = table.ncols  # 列数
        j = 0  # 循环次数
        for row in range(1, nrows):
            j += 1
            line = []
            if self.tag == 'True':
                for col in range(0, ncols):
                    line.append(table.cell(row, col).value)
                yield line
            elif self.tag == 'False':
                if j >= n and j < n + num:
                    for col in range(0, ncols):
                        line.append(table.cell(row, col).value)
                    yield line


  1. 生成测试报告
    在这里插入图片描述

  2. HtmlTestRunner.py

# coding=utf-8
# URL: http://tungwaiyip.info/software/HTMLTestRunner.html

__author__ = "Wai Yip Tung,  Findyou"
__version__ = "0.8.2.2"


# TODO: color stderr
# TODO: simplify javascript using ,ore than 1 class in the class attribute?

import datetime
import io
import sys
import time
import unittest
from xml.sax import saxutils
import sys

PY3K = (sys.version_info[0] > 2)
if PY3K:
    import io as StringIO
else:
    import StringIO
import copy



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):

    STATUS = {
        0: '通过',
        1: '失败',
        2: '错误',
    }

    DEFAULT_TITLE = '自动化测试报告'
    DEFAULT_DESCRIPTION = ''
    DEFAULT_TESTER = 'Zhangwenke'

    # ------------------------------------------------------------------------
    # HTML Template

    HTML_TMPL = r"""<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Strict//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-strict.dtd">
<html xmlns="http://www.w3.org/1999/xhtml">
<head>
    <title>%(title)s</title>
    <meta name="generator" content="%(generator)s"/>
    <meta http-equiv="Content-Type" content="text/html; charset=UTF-8"/>
    <link href="http://libs.baidu.com/bootstrap/3.0.3/css/bootstrap.min.css" rel="stylesheet">
    <script src="http://libs.baidu.com/jquery/2.0.0/jquery.min.js"></script>
    <script src="http://libs.baidu.com/bootstrap/3.0.3/js/bootstrap.min.js"></script>
    %(stylesheet)s
</head>
<body >
<script language="javascript" type="text/javascript">
output_list = Array();

/*level 调整增加只显示通过用例的分类 --Findyou
0:Summary //all hiddenRow
1:Failed  //pt hiddenRow, ft none
2:Pass    //pt none, ft hiddenRow
3:All     //pt none, ft none
*/
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 == 2 || level == 0 ) {
                tr.className = 'hiddenRow';
            }
            else {
                tr.className = '';
            }
        }
        if (id.substr(0,2) == 'pt') {
            if (level < 2) {
                tr.className = 'hiddenRow';
            }
            else {
                tr.className = '';
            }
        }
    }

    //加入【详细】切换文字变化 --Findyou
    detail_class=document.getElementsByClassName('detail');
	//console.log(detail_class.length)
	if (level == 3) {
		for (var i = 0; i < detail_class.length; i++){
			detail_class[i].innerHTML="收起"
		}
	}
	else{
			for (var i = 0; i < detail_class.length; i++){
			detail_class[i].innerHTML="详细"
		}
	}
}

function showClassDetail(cid, count) {
    var id_list = Array(count);
    var toHide = 1;
    for (var i = 0; i < count; i++) {
        //ID修改 点 为 下划线 -Findyou
        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];
        //修改点击无法收起的BUG,加入【详细】切换文字变化 --Findyou
        if (toHide) {
            document.getElementById(tid).className = 'hiddenRow';
            document.getElementById(cid).innerText = "详细"
        }
        else {
            document.getElementById(tid).className = '';
            document.getElementById(cid).innerText = "收起"
        }
    }
}


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 = ["#009100","#FF0000","#6C3365"];  
    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" type="text/css">

    STYLESHEET_TMPL = """
<style type="text/css" media="screen">
body        { font-family: Microsoft YaHei,Tahoma,arial,helvetica,sans-serif;padding: 20px; font-size: 80%; }
table       { font-size: 100%; }

/* -- heading ---------------------------------------------------------------------- */
.heading {
    margin-top: 0ex;
    margin-bottom: 1ex;
}

.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: #FFFCEC;
    font-family: "Lucida Console", "Courier New", Courier, monospace;
    text-align: left;
    font-size: 8pt;
}

}
/* -- report ---------------------用例的名字--------------------------------------------------- */
#total_row  { font-weight: bold; }
.passCase   { color: #5cb85c; }
.failCase   { color: #FF0000; font-weight: bold; }
.errorCase  { color: #6C3365; font-weight: bold; }
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:40px;  
    left:500px; 
    width: 180px;
    float: left;
    display:  inline;
}

.

</style>
"""

    # ------------------------------------------------------------------------
    # Heading
    #

    HEADING_TMPL = """<div class='heading'>
<h1 style="font-family: Microsoft YaHei">%(title)s</h1>
%(parameters)s
<h3 class='description'>%(description)s</h3>
</div>

"""  # variables: (title, parameters, description)

    HEADING_ATTRIBUTE_TMPL = """<p class='attribute'><strong>%(name)s : </strong> %(value)s</p>
"""  # variables: (name, value)

    # ------------------------------------------------------------------------
    # Report
    #
    # 汉化,加美化效果 --Findyou
    REPORT_TMPL = """
<p id='show_detail_line'>
<a class="btn btn-primary" href='javascript:showCase(0)'>概要{ %(passrate)s }</a>
<a class="btn btn-danger" href='javascript:showCase(1)'>失败{ %(fail)s }</a>
<a class="btn btn-success" href='javascript:showCase(2)'>通过{ %(Pass)s }</a>
<a class="btn btn-warning" href='javascript:showCase(3)'>所有{ %(count)s }</a>
</p>
<table id='result_table' class="table table-condensed ">
<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' class="text-center" style="font-weight: bold;font-size: 18px;background-color:#336666;color:white;">
    <td style="width:300px">TestSuit</td>
    <td>Total</td>
    <td>Pass</td>
    <td>Failure</td>
    <td>Error</td>
    <td>Detail</td>
    <td style="width:200px">ScreenShots</td>
</tr>
%(test_list)s
<tr id='total_row' class="text-center active" >
    <td>总计</td>
    <td>%(count)s</td>
    <td>%(Pass)s</td>
    <td>%(fail)s</td>
    <td>%(error)s</td>
    <td>&nbsp;</td>
    <td>通过率:%(passrate)s</td>
</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 warning'> 
    <td>%(desc)s</td>
    <td class="text-center">%(count)s</td>
    <td class="text-center">%(Pass)s</td>
    <td class="text-center">%(fail)s</td>
    <td class="text-center">%(error)s</td>
    <td class="text-center"><a href="javascript:showClassDetail('%(cid)s',%(count)s)" class="detail" id='%(cid)s'>详细</a></td>
        <td>&nbsp;</td>
</tr>
"""  # variables: (style, desc, count, Pass, fail, error, cid)

    # 失败 的样式,去掉原来JS效果,美化展示效果  -Findyou
    REPORT_TEST_WITH_OUTPUT_TMPL = r"""
<tr id='%(tid)s' class='%(Class)s '>
    <td class='%(style)s'><div class='testcase'>%(desc)s</div></td>
    <td colspan='5' align='center'>
    <!--默认收起错误信息 -Findyou  -->
    <button id='btn_%(tid)s' type="button"  class="btn btn-danger btn-xs collapsed" data-toggle="collapse" data-target='#div_%(tid)s'>%(status)s</button>
    <div id='div_%(tid)s' class="collapse">

    <!-- 默认展开错误信息 -Findyou
    <button id='btn_%(tid)s' type="button"  class="btn btn-danger btn-xs" data-toggle="collapse" data-target='#div_%(tid)s'>%(status)s</button>
    <div id='div_%(tid)s' class="collapse in"> -->
    <pre>
    %(script)s
    </pre>
    </div>
    </td>
    <td>%(img)s</td>
</tr>
""" # variables: (tid, Class, style, desc, status,img)

    # 通过 的样式,加标签效果  -Findyou
    REPORT_TEST_NO_OUTPUT_TMPL = r"""
<tr id='%(tid)s' class='%(Class)s '>
    <td class='%(style)s'><div class='testcase'>%(desc)s</div></td>
    <td colspan='5' align='center'><span class="label label-success success">%(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
    #
    # 增加返回顶部按钮  --Findyou
    ENDING_TMPL = """<div id='ending'>&nbsp;</div>
    <div style=" position:fixed;right:50px; bottom:30px; width:20px; height:20px;cursor:pointer">
    <a href="#"><span class="glyphicon glyphicon-eject" style = "font-size:30px;" aria-hidden="true">
    </span></a></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 = []
        # 增加一个测试通过率 --Findyou


        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=1, title=None, description=None, tester=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
        if tester is None:
            self.tester = self.DEFAULT_TESTER
        else:
            self.tester = tester

        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 cls not in rmap:
                rmap[cls] = []
                classes.append(cls)
            rmap[cls].append((n, t, o, e))
        r = [(cls, rmap[cls]) for cls in classes]
        return r

    # 替换测试结果status为通过率 --Findyou
    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 = []
        status.append('共 %s' % (result.success_count + result.failure_count + result.error_count))
        if result.success_count: status.append('通过 %s' % result.success_count)
        if result.failure_count: status.append('失败 %s' % result.failure_count)
        if result.error_count:   status.append('错误 %s' % result.error_count)
        if status:
            status = ','.join(status)
            self.passrate = str("%.2f%%" % (float(result.success_count) / float(
                result.success_count + result.failure_count + result.error_count) * 100))
        else:
            status = 'none'
        return [
            ('测试人员', self.tester),
            ('开始时间', startTime),
            ('合计耗时', duration),
            ('测试结果', status + ",通过率= " + self.passrate),
        ]

    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

    # 增加Tester显示 -Findyou
    def _generate_heading(self, report_attrs):
        a_lines = []
        for name, value in report_attrs:
            line = self.HEADING_ATTRIBUTE_TMPL % dict(
                name=saxutils.escape(name),
                value=saxutils.escape(value),
            )
            a_lines.append(line)
        heading = self.HEADING_TMPL % dict(
            title=saxutils.escape(self.title),
            parameters=''.join(a_lines),
            description=saxutils.escape(self.description),
            tester=saxutils.escape(self.tester),
        )
        return heading

    # 生成报告  --Findyou添加注释
    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=''.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),
            passrate=self.passrate,
        )
        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)
        # ID修改点为下划线,支持Bootstrap折叠展开特效 - Findyou
        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.shortDescription() 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
        tmpl = has_output and (n==0 and self.REPORT_TEST_NO_OUTPUT_TMPL or self.REPORT_TEST_WITH_OUTPUT_TMPL) or self.REPORT_TEST_NO_OUTPUT_TMPL

        # utf-8 支持中文 - Findyou
        # 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="https://img-blog.csdnimg.cn/2022010618100644798.jpg" style="display: block;" class="img"/>\n""" % img
                else:
                    tmp += """ <img src="https://img-blog.csdnimg.cn/2022010618100644798.jpg" 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



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



if __name__ == "__main__":
    main(module=None)

1.HTMLTestRunner.py 放置在 C:\Python36\Lib 下 2.涉及到创建目录和时间,需要在脚本开头 import os import time 3.执行脚本中删除语句 unittest.main() ,一般在脚本最后,然后添加如下语句: #导入HTMLTestRunner库,这句也可以放在脚本开头 from HTMLTestRunner import HTMLTestRunner #定义脚本标题,加u为了防止中文乱码 report_title = u&#39;登陆模块测试报告&#39; #定义脚本内容,加u为了防止中文乱码 desc = u&#39;手机JPG登陆模块测试报告详情:&#39; #定义date为日期,time为时间 date=time.strftime("%Y%m%d") time=time.strftime("%Y%m%d%H%M%S") #定义path为文件路径,目录级别,可根据实际情况自定义修改 path= &#39;D:/Python_test/&#39;+ date +"/login/"+time+"/" #定义报告文件路径和名字,路径为前面定义的path,名字为report(可自定义),格式为.html report_path = path+"report.html" #判断是否定义的路径目录存在,不能存在则创建 if not os.path.exists(path): os.makedirs(path) else: pass #定义一个测试容器 testsuite = unittest.TestSuite() #将测试用例添加到容器 testsuite.addTest(测试类名("测试方法名1")) testsuite.addTest(测试类名("测试方法名2")) #将运行结果保存到report,名字为定义的路径和文件名,运行脚本 with open(report_path, &#39;wb&#39;) as report: runner = HTMLTestRunner(stream=report, title=report_title, description=desc) runner.run(testsuite) #关闭report,脚本结束 report.close()
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值