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

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

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

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

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

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

  1. [DATABASE]

  2. host = 127.0.0.1

  3. username = root

  4. password = root

  5. port = 3306

  6. database = cai_test

  7. [HTTP]

  8. # 接口的url

  9. baseurl = http://xx.xxxx.xx

  10. port = 8080

  11. timeout = 1.0

  12. readconf.py文件主要用于读取conf.ini中的数据信息

  13. # *_*coding:utf-8 *_*

  14. __author__ = "Test Lu"

  15. import os,codecs

  16. import configparser

  17. prodir = os.path.dirname(os.path.abspath(__file__))

  18. conf_prodir = os.path.join(prodir,'conf.ini')

  19. class Read_conf():

  20. def __init__(self):

  21. with open(conf_prodir) as fd:

  22. data = fd.read()

  23. #清空文件信息

  24. if data[:3] ==codecs.BOM_UTF8:

  25. data = data[3:]

  26. file = codecs.open(conf_prodir,'w')

  27. file.write(data)

  28. file.close()

  29. self.cf = configparser.ConfigParser()

  30. self.cf.read(conf_prodir)

  31. def get_http(self,name):

  32. value = self.cf.get("HTTP",name)

  33. return value

  34. def get_db(self,name):

  35. return self.cf.get("DATABASE",name)

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

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

  1. # *_*coding:utf-8 *_*

  2. __author__ = "Test Lu"

  3. from selenium import webdriver

  4. import time,os

  5. import common.config

  6. # from common.logs import MyLog

  7. project_dir = os.path.dirname(os.path.dirname(os.path.abspath(__file__)))

  8. class Comm(object):

  9. def __init__(self,driver):

  10. self.driver = driver

  11. # self.driver = webdriver.Firefox()

  12. self.driver = webdriver.Chrome()

  13. self.driver.maximize_window()

  14. def open_url(self,url):

  15. self.driver.get(url)

  16. self.driver.implicitly_wait(30)

  17. # selenium 定位方法

  18. def locate_element(self,loatetype,value):

  19. if (loatetype == 'id'):

  20. el = self.driver.find_element_by_id(value)

  21. if (loatetype == 'name'):

  22. el = self.driver.find_element_by_name(value)

  23. if (loatetype == 'class_name'):

  24. el = self.driver.find_element_by_class_name(value)

  25. if (loatetype == 'tag_name'):

  26. el = self.driver.find_elements_by_tag_name(value)

  27. if (loatetype == 'link'):

  28. el = self.driver.find_element_by_link_text(value)

  29. if (loatetype == 'css'):

  30. el = self.driver.find_element_by_css_selector(value)

  31. if (loatetype == 'partial_link'):

  32. el = self.driver.find_element_by_partial_link_text(value)

  33. if (loatetype == 'xpath'):

  34. el = self.driver.find_element_by_xpath(value)

  35. return el if el else None

  36. # selenium 点击

  37. def click(self,loatetype,value):

  38. self.locate_element(loatetype,value).click()

  39. #selenium 输入

  40. def input_data(self,loatetype,value,data):

  41. self.locate_element(loatetype,value).send_keys(data)

  42. #获取定位到的指定元素

  43. def get_text(self, loatetype, value):

  44. return self.locate_element(loatetype, value).text

  45. # 获取标签属性

  46. def get_attr(self, loatetype, value, attr):

  47. return self.locate_element(loatetype, value).get_attribute(attr)

  48. # 页面截图

  49. def sc_shot(self,id):

  50. for filename in os.listdir(os.path.dirname(os.getcwd())) :

  51. if filename == 'picture':

  52. break

  53. else:

  54. os.mkdir(os.path.dirname(os.getcwd()) + '/picture/')

  55. photo = self.driver.get_screenshot_as_file(project_dir + '/picture/'

  56. + str(id) + str('_') + time.strftime("%Y-%m-%d-%H-%M-%S") + '.png')

  57. return photo

  58. def __del__(self):

  59. time.sleep(2)

  60. self.driver.close()

  61. self.driver.quit()

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

  1. import os,xlrd

  2. from common.logs import MyLog

  3. from xml.etree import ElementTree as ElementTree

  4. mylogger = MyLog.get_log()

  5. project_dir = os.path.dirname(os.getcwd())

  6. def user_Add():

  7. '''excel文件中读取用户登录信息'''

  8. with xlrd.open_workbook(project_dir+'/data/test_data.xlsx') as files:

  9. table_user = files.sheet_by_name('userdata')

  10. try:

  11. username = str(int(table_user.cell(1,0).value))

  12. except:

  13. username = str(table_user.cell(1,0).value)

  14. try:

  15. passwd = str(int(table_user.cell(1,1).value))

  16. except:

  17. passwd = str(table_user.cell(1,1).value)

  18. try:

  19. check = str(int(table_user.cell(1, 2).value))

  20. except Exception:

  21. check = str(table_user.cell(1, 2).value)

  22. table_url = files.sheet_by_name('base_url')

  23. base_url = str(table_url.cell(1,0).value)

  24. return (username,passwd,base_url,check)

  25. #从xml文件中读取信息,定义全局一个字典来存取xml读出的信息

  26. database={}

  27. def set_read_xml():

  28. sql_path = os.path.join(project_dir,'data','SQL.xml')

  29. data =ElementTree.parse(sql_path)

  30. for db in data.findall('database'):

  31. name = db.get('name')

  32. table = {}

  33. for tb in db.getchildren():

  34. table_name = tb.get("name")

  35. sql = {}

  36. for data in tb.getchildren():

  37. sql_id = data.get("id")

  38. sql[sql_id] = data.text

  39. table[table_name] = sql

  40. database[name] = table

  41. mylogger.info("读取的xml文件的信息%s" %database)

  42. def get_sql_sen(database_name,table_name,sql_id):

  43. set_read_xml()

  44. db = database.get(database_name).get(table_name)

  45. if db.get(sql_id):

  46. sql = db.get(sql_id).strip()

  47. mylogger.info("返回sql语句信息%s" % sql)

  48. return sql

  49. else:

  50. mylogger.info("查下的信息为空,传递的参数有误!数据库名称:【%s】,表信息【%s】,查询的id【%s】"

  51. %(database_name,table_name,sql_id))

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

  1. # logging模块支持我们自定义封装一个新日志类

  2. import logging,time

  3. import os.path

  4. class Logger(object):

  5. def __init__(self, logger,cases="./"):

  6. self.logger = logging.getLogger(logger)

  7. self.logger.setLevel(logging.DEBUG)

  8. self.cases = cases

  9. # 创建一个handler,用于写入日志文件

  10. for filename in os.listdir(os.path.dirname(os.getcwd())):

  11. if filename == "logs":

  12. break

  13. else:

  14. os.mkdir(os.path.dirname(os.getcwd())+'/logs')

  15. rq = time.strftime('%Y%m%d%H%M', time.localtime(time.time()))

  16. log_path = os.path.dirname(os.getcwd()) + '/logs/'

  17. log_name = log_path + rq + '.log' # 文件名

  18. # 将日志写入磁盘

  19. fh = logging.FileHandler(log_name)

  20. fh.setLevel(logging.INFO)

  21. # 创建一个handler,用于输出到控制台

  22. ch = logging.StreamHandler()

  23. ch.setLevel(logging.INFO)

  24. # 定义handler的输出格式

  25. formatter = logging.Formatter('%(asctime)s - %(name)s - %(levelname)s - %(message)s')

  26. fh.setFormatter(formatter)

  27. ch.setFormatter(formatter)

  28. # 给logger添加handler

  29. self.logger.addHandler(fh)

  30. self.logger.addHandler(ch)

  31. def getlog(self):

  32. return self.logger

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

 
  1. import time,HTMLTestRunner

  2. import unittest

  3. from common.config import *

  4. project_dir = os.path.abspath(os.path.join(os.path.dirname(__file__),os.pardir))

  5. class TestRunner(object):

  6. ''' 执行测试用例 '''

  7. def __init__(self, cases="../",title="Auto Test Report",description="Test case execution"):

  8. self.cases = cases

  9. self.title = title

  10. self.des = description

  11. def run(self):

  12. for filename in os.listdir(project_dir):

  13. if filename == "report":

  14. break

  15. else:

  16. os.mkdir(project_dir+'/report')

  17. # fp = open(project_dir+"/report/" + "report.html", 'wb')

  18. now = time.strftime("%Y-%m-%d_%H_%M_%S")

  19. # fp = open(project_dir+"/report/"+"result.html", 'wb')

  20. fp = open(project_dir+"/report/"+ now +"result.html", 'wb')

  21. tests = unittest.defaultTestLoader.discover(self.cases,pattern='test*.py',top_level_dir=None)

  22. runner = HTMLTestRunner.HTMLTestRunner(stream=fp, title=self.title, description=self.des)

  23. runner.run(tests)

  24. fp.close()

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

  1. import pymysql

  2. import readconf

  3. import common.config as conf

  4. readconf_conf = readconf.Read_conf()

  5. host = readconf_conf.get_db("host")

  6. username = readconf_conf.get_db("username")

  7. password = readconf_conf.get_db("password")

  8. port = readconf_conf.get_db("port")

  9. database = readconf_conf.get_db("database")

  10. config_db = {

  11. 'host': str(host),

  12. 'user': username,

  13. 'password': password,

  14. 'port': int(port),

  15. 'db': database

  16. }

  17. class Mysql_DB():

  18. def __init__(self):

  19. '''初始化数据库'''

  20. self.db = None

  21. self.cursor = None

  22. def connect_db(self):

  23. '''创建连接数据库'''

  24. try:

  25. self.db = pymysql.connect(**config_db)

  26. #创建游标位置

  27. self.cursor = self.db.cursor()

  28. # print("链接数据库成功")

  29. conf.mylogger.info("链接IP为%s的%s数据库成功" %(host,database))

  30. except ConnectionError as ex:

  31. conf.mylogger.error(ex)

  32. def get_sql_result(self,sql,params,state):

  33. self.connect_db()

  34. try:

  35. self.cursor.execute(sql, params)

  36. self.db.commit()

  37. # return self.cursor

  38. except ConnectionError as ex:

  39. self.db.rollback()

  40. if state==0:

  41. return self.cursor.fetchone()

  42. else:

  43. return self.cursor.fetchall()

  44. def close_db(self):

  45. print("关闭数据库")

  46. conf.mylogger.info("关闭数据库")

  47. 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语句

 
  1. <?xml version="1.0" encoding="utf-8" ?>

  2. <data>

  3. <database name="database_member">

  4. <table name="table_member">

  5. <sql id="select_member">

  6. select * from user where real_name=%s

  7. </sql>

  8. <sql id="select_member_one">

  9. select mobile from user where mobile=%s

  10. </sql>

  11. <sql id="delete_member">

  12. delete from user where mobile=%s

  13. </sql>

  14. <sql id="insert_member">

  15. insert into user(id) value(%s)

  16. </sql>

  17. <sql id="update_member">

  18. uodate user set real_name = %s where uuid=%s

  19. </sql>

  20. </table>

  21. </database>

  22. </data>

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

 

总结:

感谢每一个认真阅读我文章的人!!!

作为一位过来人也是希望大家少走一些弯路,如果你不想再体验一次学习时找不到资料,没人解答问题,坚持几天便放弃的感受的话,在这里我给大家分享一些自动化测试的学习资源,希望能给你前进的路上带来帮助。

软件测试面试文档

我们学习必然是为了找到高薪的工作,下面这些面试题是来自阿里、腾讯、字节等一线互联网大厂最新的面试资料,并且有字节大佬给出了权威的解答,刷完这一套面试资料相信大家都能找到满意的工作。

 

          视频文档获取方式:
这份文档和视频资料,对于想从事【软件测试】的朋友来说应该是最全面最完整的备战仓库,这个仓库也陪伴我走过了最艰难的路程,希望也能帮助到你!以上均可以分享,点下方小卡片即可自行领取。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/758614.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

【NLP笔记】RNN总结

文章目录 经典RNN单向RNN双向RNNDeep RNNRNN特性总结 变体RNNLSTMGRU 参考及转载内容&#xff1a; 循环神经网络&#xff08;RNN&#xff09;深度学习05-RNN循环神经网络完全理解RNN&#xff08;循环神经网络&#xff09; 传统的CNN&#xff08;Covolutional Neural Network&am…

Redis6.0多线程的疑惑解答

1.Redis6.0之前的版本真的是单线程吗&#xff1f; Redis在处理客户端的请求是&#xff0c;包括获取(socket读)、解析、执行、内容返回(socket 写)等都有一个 顺序串行的主线程处理&#xff0c;这就是所谓的"单线程"。但如果严格来讲并不是单线程&#xff0c;除了主线…

浏览量这么低,还要不要继续坚持?

哈喽&#xff0c;你好啊&#xff0c;我是雷工&#xff01; 曾经在一个群里聊天&#xff0c;有群友看到我两位数的浏览量&#xff0c;说到&#xff1a;浏览量这么低还坚持什么&#xff1f; 浏览量低是事实&#xff0c;大多数是十几二十的&#xff0c;上百的都是少数&#xff0c…

python社区垃圾分类管理平台的设计与实现flask-django-php-nodejs

近些年来&#xff0c;随着科技的飞速发展&#xff0c;互联网的普及逐渐延伸到各行各业中&#xff0c;给人们生活带来了十分的便利&#xff0c;社区垃圾分类管理平台利用计算机网络实现信息化管理&#xff0c;使整个社区垃圾分类管理的发展和服务水平有显著提升。 语言&#xf…

Swift 从获取所有 NSObject 对象聊起:ObjC、汇编语言以及底层方法调用链(一)

概览 Swift 语言给我们的印象是&#xff1a;简洁、现代化和可以“心安神泰”的完全信赖。不过&#xff0c;在一些特殊情况下我们唯有进入 Swift 底层的动态世界方能真正地“随遇而安”。 保安局“刘局长”曾语重心长的教导过我们&#xff1a;“非常时期&#xff0c;用非常方法…

有了std::thread,为什么还需要引入std::jthread?

C进阶专栏&#xff1a;http://t.csdnimg.cn/HGkeZ 目录 1.前言 2.std::is_invocable_v 3.std::jthread 3.1.构造函数 3.2.std::jthread无需join/detach使用实例 3.3.std::jthread处理外部请求中断实 3.4.处理中断请求示例代码 4.特性 5.总结 1.前言 C11以来提供了C原…

Redis实现高可用方案

文章目录 前言一、主从模式1.1 复制流程1.2 优缺点 二、哨兵模式2.1 介绍2.2 哨兵的工作模式 三、集群模式3.1 Cluster集群节点的通讯3.2 Hash Slot插槽算法3.3 Redis Cluster集群3.4 故障转移 前言 如果单机部署Redis服务的话&#xff0c;一旦Reids宕机&#xff0c;那么整个服…

英伟达 V100、A100/800、H100/800 GPU 对比

近期&#xff0c;不论是国外的 ChatGPT&#xff0c;还是国内诸多的大模型&#xff0c;让 AIGC 的市场一片爆火。而在 AIGC 的种种智能表现背后&#xff0c;均来自于堪称天文数字的算力支持。以 ChatGPT 为例&#xff0c;据微软高管透露&#xff0c;为 ChatGPT 提供算力支持的 A…

centos 环境部署

一、安装redis 1. 升级 GCC 最直接的解决方式是升级你的 GCC 编译器到支持 C11 标准的版本。CentOS 7 默认的 GCC 版本较旧&#xff0c;可能不支持 _Atomic。你可以通过以下步骤升级 GCC&#xff1a; 启用 CentOS 的 Software Collections (SCL) 仓库&#xff0c;该仓库提供了…

王老吉药业开拓数字经济“新蓝海”,成立数字经济研究所,科技赋能新品压片糖

3月12日&#xff0c;广州王老吉药业股份有限公司&#xff08;以下简称“王老吉药业”&#xff09;召开第十一届312感恩活动新闻发布会&#xff0c;宣告王老吉数字经济研究所成立&#xff0c;并发布王老吉压片糖新品。一系列重要重要举措&#xff0c;无一不标志着王老吉药业正以…

Java SE入门及基础(44)

目录 I / O流(上) 1. 什么是I / O流 过程分析 I / O的来源 Java 中的 I / O流 2. 字节流 OutputStream 常用方法 文件输出流 FileOutputStream 构造方法 示例 InputStream 常用方法 文件输入流 FileInputStream 构造方法 示例 综合练习 字节流应用场景 Java SE文…

自动化测试报告生成(Allure)

&#x1f345; 视频学习&#xff1a;文末有免费的配套视频可观看 &#x1f345; 关注公众号【互联网杂货铺】&#xff0c;回复 1 &#xff0c;免费获取软件测试全套资料&#xff0c;资料在手&#xff0c;涨薪更快 之前尝试使用过testNG自带的测试报告、优化过reportNG的测试报告…

算法·动态规划Dynamic Programming

很多人听到动态规划或者什么dp数组了&#xff0c;或者是做到一道关于动态规划的题目时&#xff0c;就会有一种他很难且不好解决的恐惧心理&#xff0c;但是如果我们从基础的题目开始深入挖掘动规思想&#xff0c;在后边遇到动态规划的难题时就迎难而解了。  其实不然&#xff…

linux:线程互斥

个人主页 &#xff1a; 个人主页 个人专栏 &#xff1a; 《数据结构》 《C语言》《C》《Linux》 文章目录 前言一、线程互斥问题解释互斥量的接口 二、加锁的原理三、 死锁死锁四个必要条件避免死锁 总结 前言 本文是对于线程互斥的知识总结 一、线程互斥 问题 我们先看下面…

财报解读:出海“窗口期”再现,汇量科技保驾护航的底气源于什么

大数据时代&#xff0c;每个人的喜好都被精准捕捉。购物APP、购物网站们&#xff0c;都仿佛一位贴心的时尚顾问。而这源于个性化广告经过深度学习和智能算法得来的结果。 随着广告市场的竞争愈演愈烈&#xff0c;广告主们需要更为精准、高效的个性化投放。近日&#xff0c;深耕…

基于SSM的宿舍管理系统的设计与实现(JSP,MySQL)

摘 要 随着社会发展、信息技术的普及&#xff0c;人们日常管理工作也发生了巨大的变化。信息化技术之渗透各行业的方方面面。学生宿舍管理作为校园管理工作的重要一环&#xff0c;不仅关系到学生自身的确切利益&#xff0c;同时也是对校园管理工作重大考验。近来年由于在校学生…

leetcode代码记录(移除链表元素

目录 1. 题目&#xff1a;2. 我的代码&#xff1a;小结&#xff1a; 1. 题目&#xff1a; 给你一个链表的头节点 head 和一个整数 val &#xff0c;请你删除链表中所有满足 Node.val val 的节点&#xff0c;并返回 新的头节点 。 示例 1&#xff1a; 输入&#xff1a;head […

Flutter开发进阶之瞧瞧Widget

Flutter开发进阶之瞧瞧Widget 在Flutter开发中,WIdget是构建界面的基本单元;Widget是不可变的,意味着一旦创建如果需要改变UI就需要重新创建一个新的Widget;在实际开发中,Widget通常由一个个Widget组合而成,从而形成嵌套的树形结构,复杂的UI就是由这一个个Widget构建而…

【C语言】—— 指针三 : 参透数组传参的本质

【C语言】—— 指针三 &#xff1a; 参透数组传参的本质 一、数组名的理解二、使用指针访问数组2.1、指针访问数组2.2、[ ] 的深入理解2.3、数组与指针的区别 三、一维数组的传参本质四、数组指针变量4.1、数组指针变量是什么4.2、 数组指针的初始化 五、二维数组传参的本质 一…

简单了解多线程

并发和并行 并发&#xff1a; 在同一时刻&#xff0c;多个指令在单一CPU上交替指向 并行&#xff1a;在同一时刻&#xff0c;多个指令在多个CPU上同时执行 2核4线程&#xff0c;4核8线程&#xff0c;8核16线程&#xff0c;16核32线程 基础实现线程的方式 Thread :继承类 &…