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

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

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

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

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

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

[DB]host = 127.0.0.1username = rootpassword = rootport = 3306database = cai_test

[HTTP]# 接口的urlbaseurl = http://xx.xxxx.xxport = 8080timeout = 1.0readconf.py文件主要用于读取conf.ini中的数据信息# *_*coding:utf-8 *_*__author__ = "Test Lu"import os,codecsimport 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 value

     def 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 webdriverimport time,osimport common.config# from common.logs import MyLogproject_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‘:                break        else:            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 photo    def __del__(self):        time.sleep(2)        self.driver.close()        self.driver.quit()下面介绍下,config文件主要用于读取文件中的信息:
import os,xlrdfrom common.logs import MyLogfrom xml.etree import ElementTree as ElementTreemylogger = 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.text            table[table_name] = sql        database[name] = table        mylogger.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 sql    else:        mylogger.info("查下的信息为空,传递的参数有误!数据库名称:【%s】,表信息【%s】,查询的id【%s】"                      %(database_name,table_name,sql_id))

接着介绍最简单的日志logs.py模块:
# logging模块支持我们自定义封装一个新日志类import logging,timeimport os.pathclass 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":                break        else:            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添加handler        self.logger.addHandler(fh)        self.logger.addHandler(ch)    def getlog(self):        return self.logger

common模块最后一个是test_runner.py这个方法主要是用来执行全部的测试用例
import time,HTMLTestRunnerimport unittestfrom 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 = cases        self.title = title        self.des = description    def run(self):        for filename in os.listdir(project_dir):            if filename == "report":                break        else:            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 pymysqlimport readconfimport common.config as confreadconf_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 = None        self.cursor = None    def 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.cursor        except 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文件

以上就是小编写的UI自动化框架,也是小编第一次写这种博文,该博为原创博文,转载请标明出处,谢谢。喜欢的朋友也可以给小编我点个赞吧,我会继续努力学习,与大家共同成长哒!

 



 



原文地址:https://www.cnblogs.com/tepy/p/10786430.html

时间: 2024-10-11 22:24:24

selenium+python做web端自动化测试框架与实例详解教程的相关文章

python+requests接口自动化测试框架实例详解教程

转自https://my.oschina.net/u/3041656/blog/820023 摘要: python + requests实现的接口自动化框架详细教程 前段时间由于公司测试方向的转型,由原来的web页面功能测试转变成接口测试,之前大多都是手工进行,利用postman和jmeter进行的接口测试,后来,组内有人讲原先web自动化的测试框架移驾成接口的自动化框架,使用的是java语言,但对于一个学java,却在学python的我来说,觉得python比起java更简单些,所以,我决定自

【python3+request】python3+requests接口自动化测试框架实例详解教程

转自:https://my.oschina.net/u/3041656/blog/820023 前段时间由于公司测试方向的转型,由原来的web页面功能测试转变成接口测试,之前大多都是手工进行,利用postman和jmeter进行的接口测试,后来,组内有人讲原先web自动化的测试框架移驾成接口的自动化框架,使用的是java语言,但对于一个学java,却在学python的我来说,觉得python比起java更简单些,所以,我决定自己写python的接口自动化测试框架,由于本人也是刚学习python,

新手入门:史上最全Web端即时通讯技术原理详解

前言 有关IM(InstantMessaging)聊天应用(如:微信,QQ).消息推送技术(如:现今移动端APP标配的消息推送模块)等即时通讯应用场景下,大多数都是桌面应用程序或者native应用较为流行,而网上关于原生IM(相关文章请参见:<IM架构篇>.<IM综合资料>.<IM/推送的通信格式.协议篇>.<IM心跳保活篇>.<IM安全篇>.<实时音视频开发>).消息推送应用(参见:<推送技术好文>)的通信原理介绍也较多

Web端即时通讯技术原理详解

前言 有关IM(InstantMessaging)聊天应用(如:微信,QQ).消息推送技术(如:现今移动端APP标配的消息推送模块)等即时通讯应用场景下,大多数都是桌面应用程序或者native应用较为流行,而网上关于原生IM(相关文章请参见:<IM架构篇>.<IM综合资料>.<IM/推送的通信格式.协议篇>.<IM心跳保活篇>.<IM安全篇>.<实时音视频开发>).消息推送应用(参见:<推送技术好文>)的通信原理介绍也较多

Android网络请求框架AsyncHttpClient实例详解(配合JSON解析调用接口)

最近做项目要求使用到网络,想来想去选择了AsyncHttpClient框架开进行APP开发.在这里把我工作期间遇到的问题以及对AsyncHttpClient的使用经验做出相应总结,希望能对您的学习有所帮助. 首先按照惯例先来简单了解一些AsyncHttpClient网络框架的一些知识. 1.简介 Android中网络请求一般使用Apache HTTP Client或者采用HttpURLConnect,但是直接使用这两个类库需要写大量的代码才能完成网络post和get请求,而使用android-a

selenium+python自动化94-行为事件(ActionChains)源码详解

ActionChains简介 actionchains是selenium里面专门处理鼠标相关的操作如:鼠标移动,鼠标按钮操作,按键和上下文菜单(鼠标右键)交互.这对于做更复杂的动作非常有用,比如悬停和拖放. actionchains也可以和快捷键结合起来使用,如ctrl,shif,alt结合鼠标一起使用 当你使用actionchains对象方法,行为事件是存储在actionchains对象队列.当你使用perform(),事件按顺序执行. 方法一:可以写一长串 menu = driver.fin

Java Executor 框架的实例详解

大多数并发都是通过任务执行的方式来实现的. 一般有两种方式执行任务:串行和并行. class SingleThreadWebServer {  public static void main(String[] args) throws Exception {   ServerSocket socket = new ServerSocket(80);   while(true) {    Socket conn = socket.accept();    handleRequest(conn);

简单Web UI 自动化测试框架 pyse

WebUI automation testing framework based on Selenium and unittest. 基于 selenium 和 unittest 的 Web UI自动化测试框架. 特点 默认使用CSS定位,同时支持多种定位方法(id\name\class\link_text\xpath\css). 基于Selenium二次封装,使用更简单. 提供脚手架,快速生成自动化测试项目. 自动生成/reports/目录,以及HTML测试报告生成. 自带断言方法,断言tit

webservciescxf框架之客户端与服务端实例详解

webservciescxf框架之客户端与服务端实例详解 可以关注我之前发的文章,那是采用jdk发布服务并且使用wsimpor来生成客户端的. 但本文采用的是soap1.2协议,而wsimport仅对soap1.1协议有效,所以,本文采用的是 cxf框架提供的wsdl2java 来生成客户端,如下: wsdl2java -d . http://127.0.0.1/framework?wsdl 另外,需要强调的是wsdl2java工具(axis好像也提供了)既支持soap1.1协议,也支持soap