精华内容
下载资源
问答
  • 自动化脚本编写实例-python

    万次阅读 多人点赞 2017-07-03 13:32:15
    自动化脚本编写实例

    自动化脚本编写实例

    完整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 = arg
        def 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 = 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"]')

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

    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]
        #newPwdBlankErr
        if data['pwdNew'] == "":
            return errcode[6]
        #charErr
        strTmp = data['pwdNew']
        for x in xrange(0,len(data['pwdNew'])):
            if ord(strTmp[x]) < 33 or ord(strTmp[x]) > 127:#ASCII表示范围:32-127
                return errcode[2]
        #lenErr
        if len(data['pwdNew']) > 63 or len(data['pwdNew']) < 5:
            return errcode[1]
        #oldPwdErr
        if pwd != loginData.login_data['login_pwd']:
            return errcode[0]
        #pwdSameErr
        if data['pwdNew'] == pwd:
            return errcode[4]
        #no error
        return None

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

    def checkResponse(error):
        if error == None:
            return
    
        webText = 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 测试报告样例

    demo

    12.2 完整的测试代码

    # -*- coding: UTF-8 -*-
    #!/usr/bin/env python
    
    from selenium import webdriver
    
    import 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 TimeoutException
    
    import 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 = arg
    
        def 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 = loginPwd
        pwd='admin'
        #'oldPwdBlankErr'
        if data['pwdOld'] == "":
            return errcode[5]
        #newPwdBlankErr
        if data['pwdNew'] == "":
            return errcode[6]
        #charErr
        strTmp = data['pwdNew']
        for x in xrange(0,len(data['pwdNew'])):
            if ord(strTmp[x]) < 33 or ord(strTmp[x]) > 127:#ASCII表示范围:32-127
                return errcode[2]
        #lenErr
        if len(data['pwdNew']) > 63 or len(data['pwdNew']) < 5:
            return errcode[1]
        #oldPwdErr
        if pwd != data['pwdOld']:
            return errcode[0]
        #pwdSameErr
        if data['pwdNew'] == data['pwdOld']:
            return errcode[4]
        #no error
        return None
    
    # 6. 获取输入错误数据之后的页面提示语
    def checkResponse(error):
        if error == None:
            return
        # webText = driver.find_element_by_xpath('//*[@id="PwdTip"]').text
        webText = 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 False
    
    def getText(xpath):
        time.sleep(1)
        return driver.find_element_by_xpath(xpath).text
    
    if __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()
    
    展开全文
  • 比较完整的常用adb命令,mokey命令大全,和monkey自动化脚本编写
  • 主要介绍了详解基于Android的Appium+Python自动化脚本编写,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
  • 自动化脚本编写方法

    2021-01-31 17:26:05
  • 自动化脚本编写实例

    万次阅读 2018-02-07 14:39:56
    自动化脚本编写实例 打开浏览器访问pto登陆修改管理员密码单元测试数据检查输入的数据合法性获取输入错误数据之后的页面提示语编写测试用例编写单元测试类 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()
     
    • 1
    • 2
    • 3
    • 4
    • 5

    2. 登陆

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

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

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

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

    class loginClass(object):
        """docstring for login"""
        def __init__(self, arg):
            self.login_pwd = arg
        def login(self):
            waitandSendkeys('//*[@id="Pwd"]', self.login_pwd)
            waitandClick('//*[@id="Save"]')
     
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7

    3. 修改管理员密码

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

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

    class changePwdClass(object):
        """docstring for changePwdClass"""
        def __init__(self, pwdNew, pwdOld):
            self.pwdNew = pwdNew        
            self.pwdOld = 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"]')
     
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14

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

    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' : '请输入新密码'
    }
     
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14

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

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

    def checkData(data, loginPwd):#检查顺序跟页面顺序相同
        pwd = loginPwd
        #'oldPwdBlankErr'
        if data['pwdOld'] == "":
            return errcode[5]
        #newPwdBlankErr
        if data['pwdNew'] == "":
            return errcode[6]
        #charErr
        strTmp = data['pwdNew']
        for x in xrange(0,len(data['pwdNew'])):
            if ord(strTmp[x]) < 33 or ord(strTmp[x]) > 127:#ASCII表示范围:32-127
                return errcode[2]
        #lenErr
        if len(data['pwdNew']) > 63 or len(data['pwdNew']) < 5:
            return errcode[1]
        #oldPwdErr
        if pwd != loginData.login_data['login_pwd']:
            return errcode[0]
        #pwdSameErr
        if data['pwdNew'] == pwd:
            return errcode[4]
        #no error
        return None
     
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24

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

    def checkResponse(error):
        if error == None:
            return
    
        webText = 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
     
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12

    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
        ]
     
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8

    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])
     
    • 1
    • 2
    • 3
    • 4

    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])
     
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24
    • 25
    • 26
    • 27

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

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

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

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

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

    10. 关闭浏览器

    def closeDriver():
        time.sleep(3)
        driver.quit()
        os.system('killall chromedriver')
        os.system('killall geckodriver')
     
    • 1
    • 2
    • 3
    • 4
    • 5

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

    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()
     
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7

    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)
     
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8

    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)
     
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11

    12. 测试报告和代码示例

    12.1 测试报告样例

    demo

    12.2 完整的测试代码

    # -*- coding: UTF-8 -*-
    #!/usr/bin/env python
    
    from selenium import webdriver
    
    import 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 TimeoutException
    
    import 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 = arg
    
        def 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 = loginPwd
        pwd='admin'
        #'oldPwdBlankErr'
        if data['pwdOld'] == "":
            return errcode[5]
        #newPwdBlankErr
        if data['pwdNew'] == "":
            return errcode[6]
        #charErr
        strTmp = data['pwdNew']
        for x in xrange(0,len(data['pwdNew'])):
            if ord(strTmp[x]) < 33 or ord(strTmp[x]) > 127:#ASCII表示范围:32-127
                return errcode[2]
        #lenErr
        if len(data['pwdNew']) > 63 or len(data['pwdNew']) < 5:
            return errcode[1]
        #oldPwdErr
        if pwd != data['pwdOld']:
            return errcode[0]
        #pwdSameErr
        if data['pwdNew'] == data['pwdOld']:
            return errcode[4]
        #no error
        return None
    
    # 6. 获取输入错误数据之后的页面提示语
    def checkResponse(error):
        if error == None:
            return
        # webText = driver.find_element_by_xpath('//*[@id="PwdTip"]').text
        webText = 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 False
    
    def getText(xpath):
        time.sleep(1)
        return driver.find_element_by_xpath(xpath).text
    
    if __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()
    展开全文
  • 自动化脚本编写规范

    2013-11-05 17:07:33
    能够把大家的设计和代码组装在一起,因此有必要对自动化测试脚本编写进行统一的规范化,下面就先来介绍我们的项目组整理编写的自动化脚本编写的规范。 1.自动化脚本编写的规范 1)基本信息 在每个脚本模块...
    自动化测试脚本编写规范(1)
    

    为了使所有的测试工程师在进行自动化设计和测试时能够使编写的脚本风格一致、步骤一致,能够把大家的设计和代码组装在一起,因此有必要对自动化测试脚本编写进行统一的规范化,下面就先来介绍我们的项目组整理编写的自动化脚本编写的规范。

    1.自动化脚本编写的规范

    1)基本信息

    在每个脚本模块的最上面,必须写上脚本运行的软件和硬件环境(如IE版本、QTP版本、数据库版本等)、外包项目名称、脚本编写人(使用英文名或中文拼音缩写)、脚本创建时间、脚本修改时间、修改说明、输入参数、输出参数、脚本描述等。

    2)常量命名规范

    常量的命名应该全部用大写,使用"_"作为单词间的分隔符,单词尽量使用全名称,如,Public Const MSG_EMPTY_ROW As String = "有空行存在"。

    使用Public而不是早期版本的global来声明变量。

    另外,对常量的声明必须带上类型,如前面的As String。

    3)变量命名规范

    变量命名应该简单,应尽量使用缩写。如果是一般的值类型(如integer string),则直接使用变量用途命名。尽量使用全名,例如,Dim name As String;如果是一般的临时性变量定义,应该尽可能地简单,例如,Dim i As Integer;如果名称由多个单词组成,则取每个单词的首字母,如EntityManager缩写为em,ProcedureManager缩写为pm;如果名称由一个单词组成,则对单词进行分段取首字母,如Entity缩写为et。缩写应该控制在3个字母以内,且尽量清晰。

    4)参数命名规范

    参数命名的原则是全部用小写,如果参数包括两个或两个以上的单词时,首单词字母小写,其他单词首字母大写,如stepName、stepDescription。

    5)函数命名规范

    此处函数包括sub和function,函数表示的是一个动作,所以它的结构应该是动词+名词,动词必须小写,后面的名称首字母大写,如getMaterialCode。函数命名尽量不要使用缩写,而且它的名称应该使人一目了然,能够从名称就知道这个函数的功能,不要使用无意义的函数名称。当函数名称不足以表达其功能时,应使用在函数头部加上让调用者足够明白的注释。

    6)代码注释规范

    注释务必做到准确简洁,能够充分表达代码实现的功能。

    7)空行

    空行是区分代码块与块的间隔,在函数之间必须加上空行;而在函数内部,变量声明块和实现块(实现块指除变量声明外的其他代码)要使用空行来间隔,实现块的内部,通过空行来标识一个功能段。

    8)缩进

    必须严格执行缩进,变量声明块不缩进,实现块必须保证全部缩进(不可能有实现块是行首对齐的);对于基本的控制结构来说,必须要有缩进,如IF、DO、WITH、FOR、WHILE块。

    9)续行

    对于过长的语句来说,必须使用续行,续行位置要有明显意义,例如,sql ="SELECT [code],[name] FROM [Person]"_&"WHERE [code] LIKE'001%'"。

    另外,还要通过管理对象库来提高代码的可读性,通过修改命名来达到更加易读的效果。对于使用比较频繁的代码块来说,最好将其写成函数,并尽量将功能复杂的大函数拆分成小函数。

    注意:在任何地方,不要写ElseIf语句,最好转换成If…Else…Endif结构。

    2.业务组件测试

    BPT为Bussiness Process Testing的缩写,译为业务组件测试。

    1)业务组件的简介

    业务组件是组成流程测试的基本单元,组合不同的业务组件可以实现不同的业务流程测试。如将黄金交易系统的登录作为一个组件,将交割申报作为一个组件等,然后可以将这些组件按照一定的业务流程组合在一起,以满足不同业务流的测试。这里业务组件可以重复使用,从而在一定程度上提高自动化开发的效率。

    2)业务组件测试的优点

    业务组件测试有以下几个优点:

    相关业务人员可以在没有脚本的环境下组合业务组件,实现业务流程。

    对业务人员的编程能力没有要求,业务人员只需了解系统的业务流程,不用关心具体的脚本实现。这一点也实现了业务层和脚本层的分离。

    一旦某个组件开发完毕,即可在不同的流程中使用该组件,实现高可复用性,从而加快业务流程测试的速度。

    明确角色分工,业务人员负责流程的开发、组织;QTP工程师负责脚本的开发、维护,以及相应函数库的开发、维护。

    因为实现了脚本的复用,提高了自动化开发的效率,在无形中降低了测试过程中维护的时间和成本。

    3)业务组件测试的简易流程

    业务组件测试的简易流程如图11-19所示。


    (点击查看大图)图11-19 业务组件测试的简易流程

    从图11-19可以看出整个过程分为两条线:第一条是由业务人员划分组件并组合不同的组件实现的不同流程测试;第二条是QTP专家负责组件的脚本的具体实现并负责调试成功,上传到QC供业务人员测试时调用。

    注意:

    此过程需要QC有Bussiness Process Testing组件许可的支持,也就是需要单独向HP购买。

    3.整个流程的开发过程

    下面我们还是以黄金外包项目为例,简单地演示一下整个流程的开发过程。

    1)划分组件

    个人网银交割申报业务划分为以下几部分:

    登录。

    递延交割申报。

    递延交割当日委托查询。

    递延交割当日成交查询。

    递延交割历史委托查询。

    历史递延交割成交查询。

    注销。

    2)组织业务测试流程

    组织业务测试流程为:登录-递延交割申报-递延交割当日委托查询-递延交割当日成交查询-递延交割历史委托查询-历史递延交割成交查询-注销。
    展开全文
  • java+selenium自动化脚本编写

    千次阅读 2019-06-04 19:13:00
    实训项目:创盟后台管理,页面自动化脚本编写 使用工具:java+selenium 1)java+selenium环境搭建文档 2)创盟项目后台管理系统链接 java+selenium环境搭建 一、Selenium介绍 Selenium 主要用于Web 应用程序的...
    实训项目:创盟后台管理,页面自动化脚本编写
    使用工具:java+selenium
    1)java+selenium环境搭建文档
    2)创盟项目后台管理系统链接
    一、Selenium介绍
    Selenium 主要用于Web 应用程序的自动化测试,但并不局限于此,它还支持所有基于Web 的管理任务自动化。
    Selenium 的特点如下。
    •  开源,免费;
    • 多浏览器支持:FireFox、Chrome、IE、Opera、Edge;
    • 多平台支持:Linux 、Windows、MAC;
    • 多语言支持:Java、Python、Ruby、C#、JavaScript、C++;
    • 对Web 页面有良好的支持;
    •  简单(API 简单)、灵活(用开发语言驱动);
    二、Win7下环境搭建
    1、所需资源
    • JAVA安装包:jdk-8u121-windows-x64
    • eclipse安装包:eclipse-inst-win64
    • selenium jar包:selenium-java-2.53.0.zip
    2、安装java+eclipse
    java+eclipse安装较为简单,可参照 java安装教程图解eclipse安装图解
    3、安装selenium
    1).将selenium-java-2.53.0.zip解压得到如下目录:

     

    2).打开eclipse在创建的项目上右键Build Path--→Configure Build Path...
     

     

    3).点击Add Library...
     

     

    4).选择User Library 选项,点击“Next”
     

     

    5).点击User Libraries...
     

     

    6).创建一个Library 的目录,名字随便取
     

     

    7).选择解压的selenium-java 目录,添加其下面的所有jar 文件。(可以通过Ctrl+a 一次选择一个目录下
    的所有文件进行添加。)包括libs 目录下的所有包也要添加。
     

     

    8).导入jar 包完成,如下图:
     

     

    4.安装浏览器驱动
    以Chome为例,安装Chrome 浏览器驱动,得到chromedriver.exe 文件放到系统环境变量Path 下面。
    或在脚本中写入如下代码:
    。。。
    System.setProperty("webdriver.chrome.driver", "D:\\chromedriver2.28\\chromedriver.exe");
    WebDriver driver = new ChromeDriver();
    。。。
    注:"D:\\chromedriver2.28\\chromedriver.exe"是chromedriver.exe存放的路径(最好是纯英文路径)
     
    至此,selenium+java环境搭建成功,开启页面自动化之旅。
     
     
     
     
     

    转载于:https://www.cnblogs.com/TomBombadil/p/10975554.html

    展开全文
  • postman接口自动化脚本编写实例 postman我想大家都很熟悉,有些小伙伴说请求不通,我想可能是设置里面ssl certificate verification的问题 1.下面可以看我们的实例了 我们先请求一个login的登录接口,用户名...
  • 基于Android的Appium+Python自动化脚本编写 1.Appium Appium是一个开源测试自动化框架,可用于原生,混合和移动Web应用程序测试, 它使用WebDriver协议驱动iOS,Android和Windows应用程序。 通过Appium,我们可以...
  • python-自动化脚本编写实例(进价)

    千次阅读 2018-12-02 17:11:12
    在网上看了一遍博主写的自动化脚本编写用例,里面写的非常详细,先封装好点击和输入的方法,且在方法中加入了异常处理机制,在使用的过程中直接调用这些方法就行,还加入的显示等待,以及等待加载的方法,避免页面还...
  • jmeter自动化脚本编写

    万次阅读 2018-11-06 14:33:55
    1.录制脚本   2.用户自定义变量添加变量 http代理服务器过滤规则设定 例如:ip 地址过滤,我们想访问的网址为www.test.com,在包含模式里面 添加 .+(test\.com).+ 即可。 例如...
  • UI自动化脚本编写思路

    千次阅读 2011-08-29 10:47:51
    项目页面自动化分享(二) ——脚本编写思路 在项目的开发和第一轮测试阶段,我花了部分时间来编写脚本。当我开始执...

空空如也

空空如也

1 2 3 4 5 ... 20
收藏数 231,897
精华内容 92,758
关键字:

自动化脚本编写