首先你得知道什麼是Selenium?
Selenium是一個基於瀏覽器的自動化測試工具,它提供了一種跨平臺、跨瀏覽器的端到端的web自動化解決方案。Selenium主要包括三部分:Selenium IDE、Selenium WebDriver 和Selenium Grid。html
-
Selenium IDE:Firefox的一個擴展,它能夠進行錄製回放,並把錄製的操做以多種語言(例如java、python等)的形式導出成測試用例。java
-
Selenium WebDriver:提供Web自動化所需的API,主要用做瀏覽器控制、頁面元素選擇和調試。不一樣的瀏覽器須要不一樣的WebDriver。python
-
Selenium Grid:提供了在不一樣機器的不一樣瀏覽器上運行selenium測試的能力。web
下面我會使用思惟導圖目錄結構介紹基礎測試框架,編寫測試用例進行功能測試用例,但願對您的學習有所幫助。面試
設計思路chrome
框架採用python3 + selenium3 + PO + yaml + ddt + unittest等技術編寫成基礎測試框架,能適應平常測試工做須要。windows
-
使用Page Object模式將頁面定位和業務操做分開,分離測試對象(元素對象)和測試腳本(用例腳本),一個頁面建一個對象類,提升用例的可維護性;瀏覽器
-
使用yaml管理頁面控件元素數據和測試用例數據。例如元素ID等發生變化時,不須要去修改測試代碼,只須要在對應的頁面元素yaml文件中修改便可;ruby
-
分模塊管理,互不影響,隨時組裝,即拿即用。bash
測試框架分層設計
把常見的操做和查找封裝成基礎類,無論是什麼產品,可直接拿來複用
-
業務層主要是封裝對象頁面類,一個頁面建一個類,業務層頁面繼承基礎層
-
用例層針對產品頁面功能進行構造模擬執行測試
-
框架層提供基礎組件,支撐整個流程執行及功能擴展,給用例層提供各頁面的元素數據、用例測試數據,測試報告輸出等
測試框架目錄結構
以下思惟導圖目錄結構介紹:
編寫用例方法
若是對軟件測試、接口測試、自動化測試、面試經驗交流。感興趣能夠關注咱們愛碼小士,公衆號內會有不按期的發放免費的資料連接,這些資料都是從各個技術網站蒐集、整理出來的,若是你有好的學習資料能夠私聊發我,我會註明出處以後分享給你們。
testinfo: - id: test_login001 title: 登陸測試 info: 打開抽屜首頁 testcase: - element_info: login-link-a find_type: ID operate_type: click info: 打開登陸對話框 - element_info: mobile find_type: ID operate_type: send_keys info: 輸入手機號 - element_info: mbpwd find_type: ID operate_type: send_keys info: 輸入密碼 - element_info: //input[@class='keeplogin'] find_type: XPATH operate_type: click info: 單擊取消自動登陸單選框 - element_info: //span[text()='登陸'] find_type: XPATH operate_type: click info: 單擊登陸按鈕 - element_info: userProNick find_type: ID operate_type: perform info: 鼠標懸停帳戶菜單 - element_info: //a[@class='logout'] find_type: XPATH operate_type: click info: 選擇退出 check: - element_info: //div[@class='box-mobilelogin'] /div[1]/span find_type: XPATH info: 檢查輸入手機號或密碼,登陸異常提示 - element_info: userProNick find_type: ID info: 成功登陸 - element_info: reg-link-a find_type: ID info: 檢查退出登陸是否成功 login.yaml
例如,咱們要新增登陸功能測試用例:
首先,只需在testyaml目錄下新增一個頁面對象yaml文件,參考login.yaml格式編寫便可。這些文件是提供給封裝頁面對象類調用並執行定位識別操做。
- id: test_login001.1 detail : 手機號和密碼爲空登陸 screenshot : phone_pawd_empty data: phone: "" password: "" check : - 手機號不能爲空 - id: test_login001.2 detail : 手機號爲空登陸 screenshot : phone_empty data : phone: "" password : aa check : - 手機號不能爲空 - id: test_login001.3 detail : 密碼爲空登陸 screenshot : pawd_empty data : phone : 13511112222 password: "" check : - 密碼不能爲空 - id: test_login001.4 detail : 非法手機號登陸 screenshot : phone_error data : phone : abc password: aa check : - 手機號格式不對 - id: test_login001.5 detail : 手機號或密碼不匹配 screenshot : pawd_error data : phone : 13511112222 password: aa check : - 帳號密碼錯誤 - id: test_login001.6 detail : 手機號和密碼正確 screenshot : phone_pawd_success data : phone : 13865439800 password: ******** check : - yingoja login_data.yaml login_data.yaml
其次,在testdata目錄下新增一個login_data.yaml文件提供給登陸接口傳參的測試數據,編寫格式參考login_data.yaml文件。
#!/usr/bin/env python # _*_ coding:utf-8 _*_ __author__ = 'YinJia' import os,sys sys.path.append(os.path.dirname(os.path.dirname (os.path.dirname(__file__)))) from config import setting from selenium.webdriver.support.select import Select from selenium.webdriver.common.action_chains import ActionChains from selenium.webdriver.common.by import By from public.page_obj.base import Page from time import sleep from public.models.GetYaml import getyaml testData = getyaml(setting.TEST_Element_YAML + '/' + 'login.yaml') class login(Page): """ 用戶登陸頁面 """ url = '/' dig_login_button_loc = (By.ID, testData. get_elementinfo(0)) def dig_login(self): """ 首頁登陸 :return: """ self.find_element(*self.dig_login_button_loc) .click() sleep(1) # 定位器,經過元素屬性定位元素對象 # 手機號輸入框 login_phone_loc = (By.ID,testData. get_elementinfo(1)) # 密碼輸入框 login_password_loc = (By.ID,testData. get_elementinfo(2)) # 取消自動登陸 keeplogin_button_loc = (By.XPATH,testData. get_elementinfo(3)) # 單擊登陸 login_user_loc = (By.XPATH,testData. get_elementinfo(4)) # 退出登陸 login_exit_loc = (By.ID, testData. get_elementinfo(5)) # 選擇退出 login_exit_button_loc = (By.XPATH,testData. get_elementinfo(6))def login_phone(self,phone): """ 登陸手機號 :param username: :return: """ self.find_element(*self.login_phone_loc). send_keys(phone)def login_password(self,password): """ 登陸密碼 :param password: :return: """ self.find_element(*self.login_password_loc). send_keys(password) def keeplogin(self): """ 取消單選自動登陸 :return: """ self.find_element(*self.keeplogin_button_loc). click()def login_button(self): """ 登陸按鈕 :return: """ self.find_element(*self.login_user_loc).click() def login_exit(self): """ 退出系統 :return: """ above = self.find_element(*self.login_exit_loc) ActionChains(self.driver).move_to_element(above). perform() sleep(2) self.find_element(*self.login_exit_button_loc) .click()def user_login(self,phone,password): """ 登陸入口 :param username: 用戶名 :param password: 密碼 :return: """ self.open() self.dig_login() self.login_phone(phone) self.login_password(password) sleep(1) self.keeplogin() sleep(1) self.login_button() sleep(1) phone_pawd_error_hint_loc = (By.XPATH,testData. get_CheckElementinfo(0)) user_login_success_loc = (By.ID,testData. get_CheckElementinfo(1)) exit_login_success_loc = (By.ID,testData. get_CheckElementinfo(2)) # 手機號或密碼錯誤提示 def phone_pawd_error_hint(self): return self.find_element(*self.phone_pawd_error_ hint_loc).text# 登陸成功用戶名 def user_login_success_hint(self): return self.find_element(*self.user_login_ success_loc).text # 退出登陸 def exit_login_success_hint(self): return self.find_element(*self.exit_login_ success_loc).textloginPage.py
而後,在page_obj目錄下新增一個loginPage.py文件,是用來封裝登陸頁面對象類,執行登陸測試流程操做。
#!/usr/bin/env python # _*_ coding:utf-8 _*_ __author__ = 'YinJia' import os,sys sys.path.append(os.path.dirname(os.path. dirname(__file__))) import unittest,ddt,yaml from config import setting from public.models import myunit,screenshot from public.page_obj.loginPage import login from public.models.log import Log try: f =open(setting.TEST_DATA_YAML + '/' + 'login_data.yaml',encoding='utf-8') testData = yaml.load(f) except FileNotFoundError as file: log = Log() log.error("文件不存在:{0}".format(file)) @ddt.ddt class Demo_UI(myunit.MyTest): """抽屜新熱榜登陸測試""" def user_login_verify(self,phone,password): """ 用戶登陸 :param phone: 手機號 :param password: 密碼 :return: """ login(self.driver).user_login(phone,password) def exit_login_check(self): """ 退出登陸 :return: """ login(self.driver).login_exit() @ddt.data(*testData) def test_login(self,datayaml): """ 登陸測試 :param datayaml: 加載login_data登陸測試數據 :return: """ log = Log() log.info("當前執行測試用例ID-> {0} ; 測試點-> {1}".format(datayaml['id'],datayaml['detail'])) # 調用登陸方法 self.user_login_verify(datayaml['data']['phone'], datayaml['data']['password']) po = login(self.driver) if datayaml['screenshot'] == 'phone_pawd_success': log.info("檢查點-> {0}".format (po.user_login_success_hint())) self.assertEqual(po.user_login_success_hint(), datayaml['check'][0], "成功登陸,返回實際結果是->: {0}".format(po.user_login_success_hint())) log.info("成功登陸,返回實際結果是->: {0}".format(po.user_login_success_hint())) screenshot.insert_img(self.driver, datayaml ['screenshot'] + '.jpg') log.info("-----> 開始執行退出流程操做") self.exit_login_check() po_exit = login(self.driver) log.info("檢查點-> 找到{0}元素,表示退出成功!".format(po_exit.exit_login_success_hint())) self.assertEqual(po_exit.exit_login_success_hint(), '註冊',"退出登陸,返回實際結果是->: {0}".format(po_exit.exit_login_success_hint())) log.info("退出登陸,返回實際結果是->: {0}".format(po_exit.exit_login_success_hint())) else: log.info("檢查點-> {0}".format(po.phone _pawd_error_hint())) self.assertEqual(po.phone_pawd_error_hint(), datayaml['check'][0] , "異常登陸,返回實際結果是->: {0}".format(po.phone_pawd_error_hint())) log.info("異常登陸,返回實際結果是->: {0}".format(po.phone_pawd_error_hint())) screenshot.insert_img(self.driver,datayaml ['screenshot'] + '.jpg') if __name__=='__main__': unittest.main() login_sta.py
最後,在testcase目錄下建立測試用例文件login_sta.py,採用ddt數據驅動讀取yaml測試數據文件
綜上所述,編寫用例方法只須要按以上四個步驟建立->編寫便可。
執行以下主程序,可看輸出的實際結果。
#!/usr/bin/env python # _*_ coding:utf-8 _*_ __author__ = 'YinJia' import os,sys sys.path.append(os.path.dirname(__file__)) from config import setting import unittest,time from package.HTMLTestRunner import HTMLTestRunner from public.models.newReport import new_report from public.models.sendmail import send_mail # 測試報告存放文件夾,如不存在,則自動建立 一個report目錄 if not os.path.exists(setting.TEST_REPORT):os.makedirs (setting.TEST_REPORT + '/' + "screenshot") def add_case(test_path=setting.TEST_DIR): """加載全部的測試用例""" discover = unittest.defaultTestLoader.discover (test_path, pattern='*_sta.py') return discover def run_case(all_case,result_path=setting.TEST_REPORT): """執行全部的測試用例""" now = time.strftime("%Y-%m-%d %H_%M_%S") filename = result_path + '/' + now + 'result.html' fp = open(filename,'wb') runner = HTMLTestRunner(stream=fp,title=' 抽屜新熱榜UI自動化測試報告', description='環境:windows 7 瀏覽器:chrome', tester='Jason') runner.run(all_case) fp.close() report = new_report(setting.TEST_REPORT) #調用模塊生成最新的報告 send_mail(report) #調用發送郵件模塊 if __name__ =="__main__": cases = add_case() run_case(cases)
測試結果展現
HTML報告日誌
HTML報告點擊截圖,彈出截圖
測試報告經過的日誌
自動截圖存放指定的目錄
郵件測試報告