最近受到万点暴击,由于公司业务出现问题,工作任务没那么繁重,有时间摸索selenium+python自动化测试,结合网上查到的资料自己编写出适合web自动化测试的框架,由于本人也是刚刚开始学习python,这套自动化框架目前已经基本完成了所以总结下编写的得失,便于以后回顾温习,有许多不足的的地方,也遇到了各种奇葩问题,希望大神们多多指教。
首先我们要了解什么是自动化测试,简单的说编写代码、脚本,让软件自动运行,发现缺陷,代替部分的手工测试。了解了自动化测试后,我们要清楚一个框架需要分那些模块:
上图的框架适合大多数的自动化测试,比如web UI 、接口自动化测试都可以采用,如大佬有好的方法请多多指教,简单说明下每个模块:
common:存放一些共通的方法
data:存放一些文件信息
logs:存放程序中写入的日志信息
picture:存放程序中截图文件信息
report:存放测试报告
test_case:存放编写具体的测试用例
conf.ini、readconf.py:存放编写的配置信息
下面就具体介绍每个模块的内容:conf.ini主要存放一些不会轻易改变的信息,编写的代码如下:
现在就从上往下结束吧!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': 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,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.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,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": 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,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 = 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 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 = 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语句
下面介绍下其他模块的内容:test_data.xlsx文件主要是存放一些用户信息,以及url信息,这样修改用户信息与url信息就不要修改代码方便以后操作!logs是在代码运行时候产生的日志信息,picture是存放图片信息,report存放输入的报告信息,
test_case是编写用户的模块需要所有的用例名称都要以test开头来命名哦,这是因为unittest在进行测试时会自动匹配test_case文件夹下面所有test开头的.py文件
总结:
感谢每一个认真阅读我文章的人!!!
作为一位过来人也是希望大家少走一些弯路,如果你不想再体验一次学习时找不到资料,没人解答问题,坚持几天便放弃的感受的话,在这里我给大家分享一些自动化测试的学习资源,希望能给你前进的路上带来帮助。
视频文档获取方式:
这份文档和视频资料,对于想从事【软件测试】的朋友来说应该是最全面最完整的备战仓库,这个仓库也陪伴我走过了最艰难的路程,希望也能帮助到你!以上均可以分享,点下方进群即可自行领取。