​selenium+python做web端自动化测试框架与实例详解教程

news/2024/4/18 19:49:58

最近受到万点暴击,由于公司业务出现问题,工作任务没那么繁重,有时间摸索selenium+python自动化测试,结合网上查到的资料自己编写出适合web自动化测试的框架,由于本人也是刚刚开始学习python,这套自动化框架目前已经基本完成了所以总结下编写的得失,便于以后回顾温习,有许多不足的的地方,也遇到了各种奇葩问题,希望大神们多多指教。

首先我们要了解什么是自动化测试,简单的说编写代码、脚本,让软件自动运行,发现缺陷,代替部分的手工测试。了解了自动化测试后,我们要清楚一个框架需要分那些模块:
 

 

上图的框架适合大多数的自动化测试,比如web UI  、接口自动化测试都可以采用,如大佬有好的方法请多多指教,简单说明下每个模块:

  • common:存放一些共通的方法
  • data:存放一些文件信息
  • logs:存放程序中写入的日志信息
  • picture:存放程序中截图文件信息
  • report:存放测试报告
  • test_case:存放编写具体的测试用例
  • conf.ini、readconf.py:存放编写的配置信息
     

下面就具体介绍每个模块的内容:conf.ini主要存放一些不会轻易改变的信息,编写的代码如下:

[DATABASE]
host = 127.0.0.1
username = root
password = root
port = 3306
database = cai_test[HTTP]
# 接口的url
baseurl = http://xx.xxxx.xx
port = 8080
timeout = 1.0
readconf.py文件主要用于读取conf.ini中的数据信息
# *_*coding:utf-8 *_*
__author__ = "Test Lu"
import os,codecs
import configparser
prodir = os.path.dirname(os.path.abspath(__file__))
conf_prodir = os.path.join(prodir,'conf.ini')
class Read_conf():def __init__(self):with open(conf_prodir) as fd:data = fd.read()#清空文件信息if data[:3] ==codecs.BOM_UTF8:data = data[3:]file = codecs.open(conf_prodir,'w')file.write(data)file.close()self.cf = configparser.ConfigParser()self.cf.read(conf_prodir)def  get_http(self,name):value = self.cf.get("HTTP",name)return valuedef get_db(self,name):return self.cf.get("DATABASE",name)

 这里需要注意,python3.0以上版本与python2.7版本import configparser的方法有一些区别读取一些配置文集就介绍完了,下面就说说common包下的公共文件

 

现在就从上往下结束吧!common主要是封装的一些定位元素的方法:

# *_*coding:utf-8 *_*
__author__ = "Test Lu"
from selenium import webdriver
import time,os
import common.config
# from common.logs import MyLog
project_dir = os.path.dirname(os.path.dirname(os.path.abspath(__file__)))
class Comm(object):def __init__(self,driver):self.driver = driver# self.driver = webdriver.Firefox()self.driver = webdriver.Chrome()self.driver.maximize_window()def open_url(self,url):self.driver.get(url)self.driver.implicitly_wait(30)# selenium 定位方法def locate_element(self,loatetype,value):if (loatetype == 'id'):el = self.driver.find_element_by_id(value)if (loatetype == 'name'):el = self.driver.find_element_by_name(value)if (loatetype == 'class_name'):el = self.driver.find_element_by_class_name(value)if (loatetype == 'tag_name'):el = self.driver.find_elements_by_tag_name(value)if (loatetype == 'link'):el = self.driver.find_element_by_link_text(value)if (loatetype == 'css'):el = self.driver.find_element_by_css_selector(value)if (loatetype == 'partial_link'):el = self.driver.find_element_by_partial_link_text(value)if (loatetype == 'xpath'):el = self.driver.find_element_by_xpath(value)return el if el else None# selenium 点击def click(self,loatetype,value):self.locate_element(loatetype,value).click()#selenium 输入def input_data(self,loatetype,value,data):self.locate_element(loatetype,value).send_keys(data)#获取定位到的指定元素def get_text(self, loatetype, value):return self.locate_element(loatetype, value).text# 获取标签属性def get_attr(self, loatetype, value, attr):return self.locate_element(loatetype, value).get_attribute(attr)# 页面截图def sc_shot(self,id):for filename in os.listdir(os.path.dirname(os.getcwd())) :if filename == 'picture':breakelse:os.mkdir(os.path.dirname(os.getcwd()) + '/picture/')photo = self.driver.get_screenshot_as_file(project_dir +  '/picture/'+ str(id) + str('_') + time.strftime("%Y-%m-%d-%H-%M-%S") + '.png')return photodef __del__(self):time.sleep(2)self.driver.close()self.driver.quit()

下面介绍下,config文件主要用于读取文件中的信息:

import os,xlrd
from common.logs import MyLog
from xml.etree import ElementTree as ElementTree
mylogger = MyLog.get_log()
project_dir = os.path.dirname(os.getcwd())def user_Add():'''excel文件中读取用户登录信息'''with xlrd.open_workbook(project_dir+'/data/test_data.xlsx') as files:table_user = files.sheet_by_name('userdata')try:username = str(int(table_user.cell(1,0).value))except:username = str(table_user.cell(1,0).value)try:passwd = str(int(table_user.cell(1,1).value))except:passwd = str(table_user.cell(1,1).value)try:check = str(int(table_user.cell(1, 2).value))except Exception:check = str(table_user.cell(1, 2).value)table_url = files.sheet_by_name('base_url')base_url = str(table_url.cell(1,0).value)return (username,passwd,base_url,check)#从xml文件中读取信息,定义全局一个字典来存取xml读出的信息
database={}
def set_read_xml():sql_path = os.path.join(project_dir,'data','SQL.xml')data =ElementTree.parse(sql_path)for db in data.findall('database'):name = db.get('name')table = {}for tb in db.getchildren():table_name = tb.get("name")sql = {}for data in tb.getchildren():sql_id = data.get("id")sql[sql_id] = data.texttable[table_name] = sqldatabase[name] = tablemylogger.info("读取的xml文件的信息%s" %database)
def get_sql_sen(database_name,table_name,sql_id):set_read_xml()db = database.get(database_name).get(table_name)if db.get(sql_id):sql = db.get(sql_id).strip()mylogger.info("返回sql语句信息%s" % sql)return sqlelse:mylogger.info("查下的信息为空,传递的参数有误!数据库名称:【%s】,表信息【%s】,查询的id【%s】"%(database_name,table_name,sql_id))

接着介绍最简单的日志logs.py模块:

# logging模块支持我们自定义封装一个新日志类
import logging,time
import os.path
class Logger(object):def __init__(self, logger,cases="./"):       self.logger = logging.getLogger(logger)self.logger.setLevel(logging.DEBUG)self.cases = cases# 创建一个handler,用于写入日志文件for filename in os.listdir(os.path.dirname(os.getcwd())):if filename == "logs":breakelse:os.mkdir(os.path.dirname(os.getcwd())+'/logs')rq = time.strftime('%Y%m%d%H%M', time.localtime(time.time()))log_path = os.path.dirname(os.getcwd()) + '/logs/'  log_name = log_path + rq + '.log'  # 文件名# 将日志写入磁盘fh = logging.FileHandler(log_name)fh.setLevel(logging.INFO)# 创建一个handler,用于输出到控制台ch = logging.StreamHandler()ch.setLevel(logging.INFO)# 定义handler的输出格式formatter = logging.Formatter('%(asctime)s - %(name)s - %(levelname)s - %(message)s')fh.setFormatter(formatter)ch.setFormatter(formatter)# 给logger添加handlerself.logger.addHandler(fh)self.logger.addHandler(ch)def getlog(self):return self.logger

common模块最后一个是test_runner.py这个方法主要是用来执行全部的测试用例

import time,HTMLTestRunner
import unittest
from common.config import *
project_dir = os.path.abspath(os.path.join(os.path.dirname(__file__),os.pardir))
class TestRunner(object):''' 执行测试用例 '''def __init__(self, cases="../",title="Auto Test Report",description="Test case execution"):self.cases = casesself.title = titleself.des = descriptiondef run(self):for filename in os.listdir(project_dir):if filename == "report":breakelse:os.mkdir(project_dir+'/report')# fp = open(project_dir+"/report/" + "report.html", 'wb')now = time.strftime("%Y-%m-%d_%H_%M_%S")# fp = open(project_dir+"/report/"+"result.html", 'wb')fp = open(project_dir+"/report/"+ now +"result.html", 'wb')tests =  unittest.defaultTestLoader.discover(self.cases,pattern='test*.py',top_level_dir=None)runner = HTMLTestRunner.HTMLTestRunner(stream=fp, title=self.title, description=self.des)runner.run(tests)fp.close()

以上就是common公共模块所有的模块,简单说下在写这些公共模块时,出现了各种问题,特别是读取xml文件的,唉!对于一个python的小白真是心酸啊!接着说下db模块的内容,db模块主要是读取sql语句以及返回对应的值!

import pymysql
import readconf
import common.config as conf
readconf_conf = readconf.Read_conf()host = readconf_conf.get_db("host")
username = readconf_conf.get_db("username")
password = readconf_conf.get_db("password")
port = readconf_conf.get_db("port")
database = readconf_conf.get_db("database")
config_db = {'host': str(host),'user': username,'password': password,'port': int(port),'db': database
}
class Mysql_DB():def __init__(self):'''初始化数据库'''self.db = Noneself.cursor = Nonedef connect_db(self):'''创建连接数据库'''try:self.db = pymysql.connect(**config_db)#创建游标位置self.cursor = self.db.cursor()# print("链接数据库成功")conf.mylogger.info("链接IP为%s的%s数据库成功" %(host,database))except ConnectionError as ex:conf.mylogger.error(ex)def get_sql_result(self,sql,params,state):self.connect_db()try:self.cursor.execute(sql, params)self.db.commit()# return self.cursorexcept ConnectionError as ex:self.db.rollback()if state==0:return self.cursor.fetchone()else:return self.cursor.fetchall()def close_db(self):print("关闭数据库")conf.mylogger.info("关闭数据库")self.db.close()

 刚开始写db模块是一直对字典模块的信息怎样传递到数据链接的模块,进过网上查询好些资料才彻底解决,对自己来说也是一种进步,哈哈,下面说下自己踩的坑,帮助自己以后学习**config_db把字典变成关键字参数传递,下面举例说明下:如果kwargs={'a':1,'b':2,'c':3}那么**kwargs这个等价为test(a=1,b=2,c=3)是不是很简单!哈哈 以上就是框架的主要模块,其他的模块每个项目与每个系统都不一样,在这里就是列举出来了,因为就算写出来大家也不能复用,下面就给大家看看小白还有哪些模块

 

看下了下data模块下的xml模块大家可能用的到,就给大家贴出来吧!因为ui测试主要就用到select与delete语句,所以也没有写多么复杂的sql语句

<?xml version="1.0" encoding="utf-8" ?>
<data><database name="database_member"><table name="table_member"><sql id="select_member">select * from user where real_name=%s</sql><sql id="select_member_one">select mobile from user where mobile=%s</sql><sql id="delete_member">delete from user where mobile=%s</sql><sql id="insert_member">insert into user(id) value(%s)</sql><sql id="update_member">uodate user set real_name = %s where uuid=%s</sql></table></database>
</data>

 下面介绍下其他模块的内容:test_data.xlsx文件主要是存放一些用户信息,以及url信息,这样修改用户信息与url信息就不要修改代码方便以后操作!logs是在代码运行时候产生的日志信息,picture是存放图片信息,report存放输入的报告信息, test_case是编写用户的模块需要所有的用例名称都要以test开头来命名哦,这是因为unittest在进行测试时会自动匹配test_case文件夹下面所有test开头的.py文件


http://www.ppmy.cn/news/365127.html

相关文章

CarEye 视频平台解决方案

前言 随着4G&#xff0c;5G技术和互联网技术的普及&#xff0c;流媒体应用越来越广发应用到生活中。本文档主要介绍CarEye视频服务器的主要软件硬件构成&#xff0c;功能实现。 功能介绍 CarEye 媒体服务器主要实现以下基本功能&#xff1a; 遵照GB28181/GT1078协议实设备到…

Ubuntu下安装和管理不同版本CMake,根据软件安装的需要选用不同版本的CMake

0. Foreword 在安装VTK&#xff08;Visualization Toolkit&#xff09;时发现需要用3.12版本以上的CMake才行&#xff0c;但是系统带的是3.10版本的&#xff0c;这时可以有两个方法&#xff1a; 最简单粗暴就是把系统原本的CMake覆盖了&#xff0c;重新装新版其次是麻烦一点&…

银行流水里的借和贷分别是什么意思?

在银行流水中&#xff0c;借和贷分别代表了银行账户中的资金流入和流出情况。 借表示银行账户中的资金流出或支出&#xff0c;也称为“扣减”、“支出”、“划出”等&#xff0c;如用户进行了一笔取款&#xff0c;或者银行向客户收取了利息或者手续费&#xff0c;都会在流水中…

【基于容器的部署、扩展和管理】3.8 容器镜像仓库和镜像管理

第一章&#xff1a;【云原生概念和技术】 第二章&#xff1a;【容器化应用程序设计和开发】 第三章&#xff1a;【3.1 容器编排系统和Kubernetes集群的构建】 第三章&#xff1a;【3.2 基于容器的应用程序部署和升级】 第三章&#xff1a;【3.3 自动化扩展和负载均衡】 第…

美洲新世界多开器 NewWorld多开器

下载地址1&#xff1a;https://www.lanzoui.com/nwdkq 下载地址2&#xff1a;https://www.lanzoux.com/nwdkq

奥丁神叛多开器

下载地址1&#xff1a;https://www.lanzoui.com/oddkq 下载地址2&#xff1a;https://www.lanzoux.com/oddkq

无尽的拉格朗日多开器

下载地址1&#xff1a;https://www.lanzoui.com/wjdkq 下载地址2&#xff1a;https://www.lanzoux.com/wjdkq

游戏多开器制作教程

这里讲解怎么制作多开器&#xff0c;简单易懂的讲解&#xff0c;希望能够记录学习过程。 用易语言来制作自己的小工具&#xff0c;还是挺有成就感的。每个人都可以。 1.常见的几种游戏多开限制-简单说明 2.什么是互斥体 3.互斥体的类型与查找 4.编写多开代码实现游戏多开

诺亚传说多开器

同时支持怀旧版与正式版 下载地址1&#xff1a;www.lanzoui.com/nydkq 下载地址2&#xff1a;www.lanzoux.com/nydkq

苹果多开方法

说到多开软件&#xff0c;安卓的多开软件就十分简单&#xff0c;直接利用多开分身创造就行&#xff0c;甚至手机就自带了分身了。 但是ios是如何实现分身的呢&#xff1f; 一般来说&#xff0c;苹果用户下载软件都是在AppStore里面直接下载&#xff0c;就目前来说&#xff0…

[cookies]pc多开独立的chrome[谷歌浏览器]

多开独立浏览器的方法复制快捷方式,生成多个快捷方式文件新建多个文件夹&#xff0c;用于存放独立浏览器的缓存之类的文件右键打开快捷方式的属性在图中蓝色标注那列&#xff0c; 新增命令 # 注意等号后面跟的是新建文件存放缓存的路径[E:\chrome_1]&#xff0c;回归到第三点# …

奇怪的IE浏览器不能多开窗口现象!

导读&#xff1a; 今天在帮一位长辈解决电脑问题的时候遇到了一个奇怪的现象。 现象特征&#xff1a; 使用IE浏览器浏览网页的时候&#xff0c;最多只能打开4/5个页面&#xff0c;再打开页面窗口的时候&#xff0c;状态栏没有任何反映&#xff0c;窗口不再刷新。 分析&#xff…

LINE电脑PC无限多开助手,LINE电脑多开器

LINE电脑PC无限多开助手LINE电脑多开器 现在很多人都在用电脑版的LINE&#xff0c;不过正常情况下&#xff0c;电脑版LINE只能打开一个&#xff0c;但是对于一些有特殊需求的用户来说&#xff0c;可能要同时登陆两个或者更多的LINE账号&#xff0c;因为这款多开器可以无线在电脑…

ie浏览器多开-----同时登陆多个账号

1、在电脑桌面右键 找到 新建快捷方式 在上图输入框中输入 "C:\Program Files\Internet Explorer\iexplore.exe" -noframemerging -private 后 点击下一步 为快捷方式命名&#xff0c;点击完成 注&#xff1a;"C:\Program Files\Internet Explorer\iexplore.…

虚拟机可以多开一台服务器吗,游戏多开必备:虚拟机vmware安装

很多朋友都不会安装虚拟机机,今天找来了一个非常详细的教程。虚拟机主要有那些用处呢? 第一:起到防封作用,指部分游戏。 第二:在一起回合制游戏或者没有多开软件的游戏项目上可以充分利用虚拟机来实现游戏多开。第三:防病毒,喜欢看大片的朋友们装一个吧!工作室中病毒可不是一…

批量多开独立Google浏览器

批量多开独立Google浏览器 背景 需求&#xff1a;每个浏览器上收藏的书签、增加的拓展程序都可以实现独立性。 实现思路 使用命令行参数--user-data-dirD:\myChrome\9557。 –user-data-dir 意思就是用户的数据chrome数据放到指定目录上。我们每个谷歌浏览器使用不用的 --…

cpu多开测试软件,游戏多开CPU优化工具

游戏多开后电脑受不了,有没有降低CPU使用率的软件? 降低CPU使用率的操作: 系统优化设置 ,删除Windows强加的附件 用记NOTEPAD修改\winnt\inf\sysoc.inf(按住shift从该文件上点右键,选择打开方式为记事本,或直接打开记事本,选择文件,文件类型用全选,找到该文件,打开即…

火狐浏览器启动参数_火狐浏览器启动参数介绍

作为吴小松最喜欢的浏览器之一&#xff1a;功能强大的火狐(Firefox)可以在启动时带一些特殊的参数&#xff0c;添加参数的方法就是在火狐的快捷方式的属性里面的目标那一行&#xff0c;最后添加参数&#xff0c;注意前面要留一个空格。当然了&#xff0c;Linux终端和Windows 的…

麒麟服务器开启多个终端,厉害:麒麟多开同步器

麒麟鼠标键盘同步器是一种工具,可让您控制一台计算机上的多个计算机键盘和鼠标。使用其他计算机打开多台计算机时,将自动配置每台计算机的鼠标和键盘是,也就是说,如果要控制另一台计算机的屏幕,则需要运行该计算机的键盘来控制它。现在,使用此Kirin鼠标和键盘同步器,您可…

python操作模拟器多开操作_窗口多开模拟器同步操作器▲按键精灵脚本▲

[Visual Basic] 纯文本查看 复制代码Dim WindowsCount//定义一个变量,存入要多开的窗口数量 WindowsCount = 0//变量赋值为0 Dim Windows(999)//定义一个数组,用来存入游戏句柄,可存入1000个游戏窗口,我想应该也许大概可能没人能开1000个游戏窗口吧,要是开1000个记事本窗口…