自动化脚本编写实例

  • 打开浏览器访问pto
  • 登陆
  • 修改管理员密码
  • 单元测试数据
  • 检查输入的数据合法性
  • 获取输入错误数据之后的页面提示语
  • 编写测试用例
  • 编写单元测试类
    • 1 单元测试中的通用操作
    • 2 测试类
  • 进行单元测试并生成测试报告
  • 关闭浏览器
  • 异常处理
    • 1 点击函数
    • 2 填写表单
    • 3 元素加载
  • 测试报告和代码示例
    • 1 测试报告样例
    • 2 完整的测试代码

完整demo请参看:https://github.com/niununu/k2p_web_test

本文以修改用户名密码单元为案例,编写测试脚本。完成修改用户名密码模块单元测试。

(ps.这个demo中登陆密码为“admin”)

1. 打开浏览器,访问p.to

# 1. 打开浏览器,访问p.to
driver = webdriver.Chrome()
def openDriver():driver.get("http://p.to")driver.maximize_window()

2. 登陆

登陆这动作传入的参数只有一个“用户密码”

需要执行的操作有两个:1. 向输入框输入密码 2. 点击确定

需要注意的是在登陆的时候可能出现页面还没有加载出来,我们的程序就开始填写表单的情况。

为了防止异常出现,编写了函数waitandSendkeys和waitandClick来处理异常。(后面将会介绍异常处理函数)

class loginClass(object):"""docstring for login"""def __init__(self, arg):self.login_pwd = argdef login(self):waitandSendkeys('//*[@id="Pwd"]', self.login_pwd)waitandClick('//*[@id="Save"]')

3. 修改管理员密码

需要传入的参数有两个:1.旧密码 2. 新密码

要注意的是由于修改管理员密码是一个弹窗,所以要判断等弹窗弹出之后再进行操作

class changePwdClass(object):"""docstring for changePwdClass"""def __init__(self, pwdNew, pwdOld):self.pwdNew = pwdNew        self.pwdOld = pwdOlddef changeUserPwd(self):waitandClick('//*[@id="Con"]/div[1]/ul[2]/li[1]')waitandClick('//*[@id="Con"]/div[1]/ul[2]/li[1]/ul/li[3]')waitforDisplay('//*[@id="_Widget"]')waitandSendkeys('//*[@id="PwdOld"]', self.pwdOld)waitandSendkeys('//*[@id="PwdNew"]', self.pwdNew)waitandSendkeys('//*[@id="PwdCfm"]', self.pwdNew)waitandClick('//*[@id="SavePwd"]')

到这里,我们可以完成修改用户名密码这一动作。后面将进行单元测试。

4. 单元测试数据

修改用户名密码这个功能的防呆规则如下:

输入项 允许输入 可为空 格式规范 合法性 依赖项
原管理员密码 字符串 长度限制:5-63; 字符集:英文字符集; 需要与管理员密码相同
新管理员密码 字符串 长度限制:5-63; 字符集:英文字符集;
确认管理员密码 字符串 需要与新管理员密码相同

根据防呆规则可以列出:1.可能出现的错误 2.出现错误时页面应有的提示语

#可能出现的错误
errcode = ['oldPwdErr', 'lenErr', 'charErr', 'matchErr', 'pwdSameErr',\'oldPwdBlankErr', 'newPwdBlankErr']#出现错误时页面应有的提示语
errTips = {'oldPwdErr' :'原密码错误','lenErr' : '新密码长度应为5~63位','charErr' : "新密码包含非法字符",'matchErr' : '两次密码输入不一致','pwdSameErr' : '新密码与原密码相同,请重新输入','oldPwdBlankErr' : '请输入原密码','newPwdBlankErr' : '请输入新密码'
}

5. 检查输入的数据合法性

需要输入的数据为要检查的data和登陆密码

def checkData(data, loginPwd):#检查顺序跟页面顺序相同pwd = loginPwd#'oldPwdBlankErr'if data['pwdOld'] == "":return errcode[5]#newPwdBlankErrif data['pwdNew'] == "":return errcode[6]#charErrstrTmp = data['pwdNew']for x in xrange(0,len(data['pwdNew'])):if ord(strTmp[x]) < 33 or ord(strTmp[x]) > 127:#ASCII表示范围:32-127return errcode[2]#lenErrif len(data['pwdNew']) > 63 or len(data['pwdNew']) < 5:return errcode[1]#oldPwdErrif pwd != loginData.login_data['login_pwd']:return errcode[0]#pwdSameErrif data['pwdNew'] == pwd:return errcode[4]#no errorreturn None

6. 获取输入错误数据之后的页面提示语

def checkResponse(error):if error == None:returnwebText = getText('//*[@id="PwdTip"]')if webText == False:#没有提示print('###Error: no tips on web!')else:webText = webText.decode('UTF-8')waitandClick('//*[@id="ModifyPwd"]/i')time.sleep(1)return webText

7. 编写测试用例

    data = [{"pwdNew" : "12345678","pwdOld" : '8dadla'},#"oldPwdErr"{"pwdNew" : "admi","pwdOld" : 'admin'},#lenErr{'pwdNew' : '1  2  3','pwdOld' : 'admin'},#charErr{'pwdNew' : 'admin','pwdOld' : 'admin'},#pwdSameErr{'pwdNew' : "",'pwdOld' : ""},#oldPwdBlank{'pwdNew' : "",'pwdOld' : "admin"}#newPwdBlank]

8.编写单元测试类

8.1 单元测试中的通用操作

单元测试中,不同的部分应该是数据,所以可以定义一个通用的操作。

其中self.assertEqual(checkResponse(error), errTips[error])是判定测试是否通过的条件:页面提示语是否正确。

def commonAction(self, arg):error = checkData(arg)changeUserPwd.main(arg)self.assertEqual(checkResponse(error), errTips[error])

8.2 测试类

测试类中主要包括了测试用例6个,和对应的以“test”开头的测试函数。

这里继承了python的unittest。

关于unittest的语法请参考:http://www.jb51.net/article/65856.htm

class TestCase(unittest.TestCase):data = [{"pwdNew" : "12345678","pwdOld" : '8dadla'},#"oldPwdErr"{"pwdNew" : "admi","pwdOld" : '*'},#lenErr{'pwdNew' : '1  2  3','pwdOld' : '*'},#charErr{'pwdNew' : 'admin','pwdOld' : '*'},#pwdSameErr{'pwdNew' : "",'pwdOld' : ""},#oldPwdBlank{'pwdNew' : "",'pwdOld' : "*"}#newPwdBlank]def commonAction(self, arg):error = checkData(arg)changeUserPwd.main(arg)self.assertEqual(checkResponse(error), errTips[error])def test_oldPwdErr(self):self.commonAction(self.data[0])def test_lenErr(self):self.commonAction(self.data[1])def test_charErr(self):self.commonAction(self.data[2])def test_pwdSameErr(self):self.commonAction(self.data[3])def test_oldPwdBlank(self):self.commonAction(self.data[4])def test_newPwdBlank(self):self.commonAction(self.data[5])

9. 进行单元测试并生成测试报告

这里利用了HTMLTestRunner来生成测试报告。

HTMLTestRunner语法请参看:https://testerhome.com/topics/7576

生成的测试报告将会存放在reports/test_report文件夹下,按照时间命名。测试报告的title叫做“修改管理员密码试报告”

unittest.main(testRunner=HtmlTestRunner.HTMLTestRunner(output='test_report',report_title='修改管理员密码试报告'))

10. 关闭浏览器

def closeDriver():time.sleep(3)driver.quit()os.system('killall chromedriver')os.system('killall geckodriver')

到这里,我们可以完成修改用户名密码模块的单元测试了,为了增加代码的健壮性,下面介绍异常处理。

11. 异常处理

11.1 点击函数

点击按钮的时候可能出现的异常情况是:可能页面元素还没有加载出来的时候,点击的动作就发生了。这样就会引发找不到元素异常。

解决的方法是通过显示等待,每10ms检查一次页面元素是否加载完成,完成后就点击,否则就等到超时时间之后结束动作。

def waitandClick(xpath):try:WebDriverWait(driver, 10).until(EC.element_to_be_clickable((By.XPATH, xpath)))except TimeoutException as e:print('Error:waitandClick, TimeoutException, xpath = %s\n' % xpath)else:driver.find_element_by_xpath(xpath).click()

11.2 填写表单

在填写表单时,除了页面元素还没有加载完成的异常外,还可能原有表单中有文本,而我们的输入则是以追加模式填写的。这就会导致填写的文本不准确。

def waitandSendkeys(xpath, keys):try:WebDriverWait(driver, 10).until(EC.presence_of_element_located((By.XPATH, xpath)))except TimeoutException as e:print('Error:waitandSendkeys, TimeoutException, xpath = %s\n' % xpath)else:driver.find_element_by_xpath(xpath).clear()driver.find_element_by_xpath(xpath).send_keys(keys)

11.3 元素加载

在元素加载中可能出现: 1. 在超时时间内元素没有加载完成 2. 查询的元素根本不存在

针对这两种情况进行异常处理:

def waitforDisplay(xpath):try:WebDriverWait(driver, 10).until(EC.presence_of_element_located((By.XPATH, xpath)))except TimeoutException as e:print('Error:waitforDisplay, TimeoutException, xpath = %s\n' % xpath)else:try:process = driver.find_element_by_xpath(xpath)WebDriverWait(driver, 10).until(lambda driver: process.is_displayed())except NoSuchElementException as e:print('Error:waitforDisplay, NoSuchElementException, xpath = %s\n' % xpath)

12. 测试报告和代码示例

12.1 测试报告样例

12.2 完整的测试代码

# -*- coding: UTF-8 -*-
#!/usr/bin/env pythonfrom selenium import webdriverimport time, os
from selenium.webdriver.support.ui import WebDriverWait
from selenium.webdriver.support import expected_conditions as EC
from selenium.webdriver.common.by import By
from selenium.common.exceptions import NoSuchElementException
from selenium.common.exceptions import TimeoutExceptionimport unittest
import HtmlTestRunner
import sys
reload(sys)
sys.setdefaultencoding('utf-8') # 1. 打开浏览器,访问p.to
driver = webdriver.Chrome()
def openDriver():driver.get("http://p.to")driver.maximize_window()# 2. 登陆
class loginClass(object):"""docstring for login"""def __init__(self, arg):self.login_pwd = argdef login(self):waitandSendkeys('//*[@id="Pwd"]', self.login_pwd)waitandClick('//*[@id="Save"]')def login(data):openDriver()test1 = loginClass(data)test1.login()# 3.修改管理员密码
class changePwdClass(object):"""docstring for changePwdClass"""def __init__(self, arg):self.pwdNew = arg.get('pwdNew', '')self.pwdOld = arg.get('pwdOld', '')def changeUserPwd(self):waitandClick('//*[@id="Con"]/div[1]/ul[2]/li[1]')waitandClick('//*[@id="Con"]/div[1]/ul[2]/li[1]/ul/li[3]')waitforDisplay('//*[@id="_Widget"]')waitandSendkeys('//*[@id="PwdOld"]', self.pwdOld)waitandSendkeys('//*[@id="PwdNew"]', self.pwdNew)waitandSendkeys('//*[@id="PwdCfm"]', self.pwdNew)waitandClick('//*[@id="SavePwd"]')def changeUserPwd_main(data):changePwdObj = changePwdClass(data)changePwdObj.changeUserPwd()# 4. 单元测试数据
errcode = ['oldPwdErr', 'lenErr', 'charErr', 'matchErr', 'pwdSameErr',\'oldPwdBlankErr', 'newPwdBlankErr']
errTips = {'oldPwdErr' :'原密码错误','lenErr' : '新密码长度应为5~63位','charErr' : "新密码包含非法字符",'matchErr' : '两次密码输入不一致','pwdSameErr' : '新密码与原密码相同,请重新输入','oldPwdBlankErr' : '请输入原密码','newPwdBlankErr' : '请输入新密码'
}# 5. 检查输入的数据合法性
def checkData(data):#检查顺序跟页面顺序相同#pwd = loginPwdpwd='admin'#'oldPwdBlankErr'if data['pwdOld'] == "":return errcode[5]#newPwdBlankErrif data['pwdNew'] == "":return errcode[6]#charErrstrTmp = data['pwdNew']for x in xrange(0,len(data['pwdNew'])):if ord(strTmp[x]) < 33 or ord(strTmp[x]) > 127:#ASCII表示范围:32-127return errcode[2]#lenErrif len(data['pwdNew']) > 63 or len(data['pwdNew']) < 5:return errcode[1]#oldPwdErrif pwd != data['pwdOld']:return errcode[0]#pwdSameErrif data['pwdNew'] == data['pwdOld']:return errcode[4]#no errorreturn None# 6. 获取输入错误数据之后的页面提示语
def checkResponse(error):if error == None:return# webText = driver.find_element_by_xpath('//*[@id="PwdTip"]').textwebText = getText('//*[@id="PwdTip"]')if webText == False:#没有提示print('###Error: no tips on web!')else:webText = webText.decode('UTF-8')waitandClick('//*[@id="ModifyPwd"]/i')return webText# 8.单元测试类
class TestCase(unittest.TestCase):# 7. 编写测试用例data = [{"pwdNew" : "12345678","pwdOld" : '8dadla'},#"oldPwdErr"{"pwdNew" : "admi","pwdOld" : 'admin'},#lenErr{'pwdNew' : '1  2  3','pwdOld' : 'admin'},#charErr{'pwdNew' : 'admin','pwdOld' : 'admin'},#pwdSameErr{'pwdNew' : "",'pwdOld' : ""},#oldPwdBlank{'pwdNew' : "",'pwdOld' : "admin"}#newPwdBlank]def commonAction(self, arg):error = checkData(arg)changeUserPwd_main(arg)self.assertEqual(checkResponse(error), errTips[error])time.sleep(1)def test_oldPwdErr(self):self.commonAction(self.data[0])def test_lenErr(self):self.commonAction(self.data[1])def test_charErr(self):self.commonAction(self.data[2])def test_pwdSameErr(self):self.commonAction(self.data[3])def test_oldPwdBlank(self):self.commonAction(self.data[4])def test_newPwdBlank(self):self.commonAction(self.data[5])# 10. 关闭浏览器
def closeDriver():time.sleep(3)driver.quit()os.system('killall chromedriver')os.system('killall geckodriver')# 11. 异常处理
## 11.1 点击函数
def waitandClick(xpath):try:WebDriverWait(driver, 10).until(EC.element_to_be_clickable((By.XPATH, xpath)))except TimeoutException as e:print('Error:waitandClick, TimeoutException, xpath = %s\n' % xpath)else:driver.find_element_by_xpath(xpath).click()## 11.2 填写表单
def waitandSendkeys(xpath, keys):try:WebDriverWait(driver, 10).until(EC.presence_of_element_located((By.XPATH, xpath)))except TimeoutException as e:print('Error:waitandSendkeys, TimeoutException, xpath = %s\n' % xpath)else:driver.find_element_by_xpath(xpath).clear()driver.find_element_by_xpath(xpath).send_keys(keys)## 11.3 元素加载
def waitforDisplay(xpath):try:WebDriverWait(driver, 10).until(EC.presence_of_element_located((By.XPATH, xpath)))except TimeoutException as e:print('Error:waitforDisplay, TimeoutException, xpath = %s\n' % xpath)else:try:process = driver.find_element_by_xpath(xpath)WebDriverWait(driver, 10).until(lambda driver: process.is_displayed())except NoSuchElementException as e:print('Error:waitforDisplay, NoSuchElementException, xpath = %s\n' % xpath)def elementIsDisplayed(xpath):try:driver.find_element_by_xpath(xpath)except NoSuchElementException as e:return Falsedef getText(xpath):time.sleep(1)return driver.find_element_by_xpath(xpath).textif __name__ == '__main__':openDriver()login('admin')#data = {'pwdNew'='admin', 'pwdOld'='12345678'}#changeUserPwd_main(data)#9. 进行单元测试并生成测试报告unittest.main(testRunner=HtmlTestRunner.HTMLTestRunner(output='test_report',report_title='修改管理员密码试报告'))closeDriver()

自动化脚本编写实例-python相关推荐

  1. 自动化脚本编写-python

    自动化脚本编写实例 打开浏览器访问pto 登陆 修改管理员密码 单元测试数据 检查输入的数据合法性 获取输入错误数据之后的页面提示语 编写测试用例 编写单元测试类 1 单元测试中的通用操作 2 测试类 ...

  2. QPython+uiautomator2安卓手机自动化脚本编写

    QPython+uiautomator2安卓手机自动化脚本编写 开始 手机端运行 不需要尝试的 换个思路 使用图像匹配 常见问题 uiautomator2使用说明:https://github.com ...

  3. python自动化脚本编写教程_开发工具pycharm写第一个Python自动化程序案例|python基础教程|python入门|python教程...

    https://www.xin3721.com/eschool/pythonxin3721/ 版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明. ...

  4. 子域名挖掘,子域名爆破,Python脚本编写(Python安全攻防)。

    1.什么是域名. 域名(Domain Name),又称网域,是由一串用点分隔的名字组成的Internet上某一台计算机或计算机组的名称,用于在数据传输时对计算机的定位标识(有时也指地理位置). 由于I ...

  5. GDB自动化脚本编写笔记一

    Author:ZERO-A-ONE Date:2019-12-24 作为UNIX/Linux下使用广泛的调试器,gdb不仅提供了丰富的命令,还引入了对脚本的支持:一种是对已存在的脚本语言支持,比如py ...

  6. python自动化脚本-一 个 Python 自动化脚本引发的惨案!把公司搞死了?

    一个法国程序员 Nicolas Beauvais 在推特上发了一系列帖子,控诉 一个云服务商搞死他们公司了,从而引发了一场骚动. 他是一家仅有 2 人的微型 AI 创业公司 Raisup 的 CTO, ...

  7. android课堂小测试,[小薇班]APPIUM自动化测试 优酷APP测试脚本编写实例

    本课程主要包含: Appium入门基础 以优酷APP为例,介绍如何编写自动化脚本. Appium概述 appium 是一个自动化测试开源工具,支持 iOS, Android, Windows, Fir ...

  8. datetime报错 sql脚本_Linux中Mysql数据库备份shell脚本编写实例

    学了段时间的Linux,也学习了shell脚本编写的基本命令与语法,现做一个综合案例来详细讲解. 要求:1).每天凌晨备份数据库shaoxiao到/data/backup/db中 2).备份开始和备份 ...

  9. python自动化脚本开发是什么意思_python自动化脚本编写电脑回事

    ·通过添加表.字段或者索引来修改数据库的结构在以前的脚本中,我们总是先定义一个常量,告诉脚本我们要处理HKEY_CURRENT_USER或HKEY_LOCAL_MACHINE注册表配置单元然后重启一下 ...

最新文章

  1. 聊一聊 软件系统中的“热力学第二定律”
  2. 第二十三讲 解一阶微分方程组
  3. Apache Kafka-初体验Kafka(02)-Centos7下搭建单节点kafka_配置参数详解_基本命令实操
  4. JavaMail发送简单邮件
  5. 【Android】Looper消息分发(msg.target.dispatchMessage), Handler消息处理(消息回调/外部回调/自身回调)
  6. 深度学习之卷积神经网络(13)DenseNet
  7. [渝粤教育] 西南科技大学 电子技术基础 在线考试复习资料(1)
  8. WPF 不支持从调度程序线程以外的线程对其 SourceCollection 进行的更改
  9. ThinkPHP的介绍和安装
  10. 用 label 控制 Pod 的位置 - 每天5分钟玩转 Docker 容器技术(128)
  11. @Autowired与@Resource
  12. 拓端tecdat|R语言在BRFSS数据中可视化分析探索糖尿病的影响因素
  13. 基于React实现的【绿色版电子书阅读器】,支持离线下载
  14. 游戏设计情境探秘之声音
  15. CSDN日报20170612 ——《程序员,感觉技术停滞了怎么办?》
  16. 使用Word脚注、尾注给论文添加参考文献并将编号修改成[1]的方法
  17. signature=735f4378ec01919f23285d0d2557be19,OPENSSL编程 第二十章 椭圆曲线
  18. 2019第十届蓝桥杯大赛软件类省赛C++ B组真题题解
  19. 可能是最好的正则表达式的教程笔记了吧...
  20. 文章搬运工具,自媒体文章搬运工具,公众号文章搬运工具免费

热门文章

  1. 超级干货 :如何创造性地应用深度学习视觉模型于非视觉任务(附代码)
  2. 解决mac下无法剪贴、复制、粘贴问题
  3. 联想微型计算机白屏怎么办,联想液晶显示器白屏怎么办
  4. js 如何获取td标签的值
  5. java jacob下载_jacob-1.18_src jacob开发包网站下载的 - 下载 - 搜珍网
  6. 阿里云弹性公网EIP收费价格表
  7. Learning Disentangled Representations of Negation and Uncertainty
  8. HTML入门必看-含源码案例(表格-列表-表单-图片-视频-音频)
  9. rpmbuild工作目录
  10. oracle中常用的字符串查找、替换以及截取函数的用法!