pytest加載測試用例的一個問題:A session is either terminated or not started


一. 引出問題

為了測試一種情況,我在目錄TestCases下新建了一個test_login.py和test_welcome.py,然后發現在模塊里面各自運行時,都是正常的,在命令行運行時,后者可以正常,前者總是出錯:

 

 

二. 測試代碼分析

先看看兩個模塊的代碼是怎么設計的

test_login.py

 

from appium import webdriver
from selenium.webdriver.support.wait import WebDriverWait
from selenium.webdriver.support import expected_conditions as EC
from appium.webdriver.common.mobileby import MobileBy
import time, pytest
import logging
from Common.my_logging import *


#操作對象的設備信息
desired_caps = {}
desired_caps["platformName"] = "Android"
desired_caps["platformVersion"] = "5.1.1"
desired_caps["deviceName"] = "Android Emulator"
desired_caps["appPackage"] = "com.xxzb.fenwoo"
desired_caps["appActivity"] = "com.xxzb.fenwoo.activity.addition.WelcomeActivity"

#連接appium
driver = webdriver.Remote('http://127.0.0.1:4723/wd/hub', desired_caps)
time.sleep(3)
logging.info("加載test_login")
logging.info("test_login_driver是: " + str(driver))



@pytest.mark.login
def test_login_success():
    logging.info("test_login開始了!")
    logging.info("test_login的driver是: " + str(driver))
    size = driver.get_window_size()
    for i in range(3):
        driver.swipe(size["width"] * 0.9, size["height"] * 0.5, size["width"] * 0.1, size["height"] * 0.5, 200)
    WebDriverWait(driver, 10, 1).until(EC.visibility_of_element_located((MobileBy.ID, "com.xxzb.fenwoo:id/btn_start")))
    driver.find_element_by_id("com.xxzb.fenwoo:id/btn_start").click()
    WebDriverWait(driver, 10, 1).until(EC.visibility_of_element_located((MobileBy.ID, "com.xxzb.fenwoo:id/btn_login")))
    driver.find_element_by_id("com.xxzb.fenwoo:id/btn_login").click()
    WebDriverWait(driver, 10, 1).until(EC.visibility_of_element_located((MobileBy.ID, "com.xxzb.fenwoo:id/et_phone")))
    driver.find_element_by_id("com.xxzb.fenwoo:id/et_phone").send_keys("18684720553")
    driver.find_element_by_id("com.xxzb.fenwoo:id/btn_next_step").click()
    WebDriverWait(driver, 10, 1).until(EC.visibility_of_element_located((MobileBy.ID, "com.xxzb.fenwoo:id/et_pwd")))
    driver.find_element_by_id("com.xxzb.fenwoo:id/et_pwd").send_keys("python")
    driver.find_element_by_id("com.xxzb.fenwoo:id/btn_next_step").click()
    assert True == True
View Code

 

test_welcome.py

from appium import webdriver
from selenium.webdriver.support.wait import WebDriverWait
from selenium.webdriver.support import expected_conditions as EC
from appium.webdriver.common.mobileby import MobileBy
import time
import pytest
import logging
from Common.my_logging import *



#操作對象的設備信息
desired_caps = {}
desired_caps["platformName"] = "Android"
desired_caps["platformVersion"] = "5.1.1"
desired_caps["deviceName"] = "Android Emulator"
desired_caps["appPackage"] = "com.xxzb.fenwoo"
desired_caps["appActivity"] = "com.xxzb.fenwoo.activity.addition.WelcomeActivity"

#連接appium
driver = webdriver.Remote('http://127.0.0.1:4723/wd/hub', desired_caps)
time.sleep(3)
logging.info("加載test_welcome")
logging.info("test_welcome_driver是: " + str(driver))



#滑屏操作
@pytest.mark.welcome
def test_welcome():
    logging.info("test_welcome開始了!")
    logging.info("test_welcome的driver是: " + string(driver))
    size = driver.get_window_size()
    for i in range(3):
        driver.swipe(size["width"] * 0.9, size["height"] * 0.5, size["width"] * 0.1, size["height"] * 0.5, 200)
    WebDriverWait(driver, 10, 1).until(EC.visibility_of_element_located((MobileBy.ID, "com.xxzb.fenwoo:id/btn_start")))
    driver.find_element_by_id("com.xxzb.fenwoo:id/btn_start").click()
    driver.close_app()
    driver.quit()
    assert True == True
View Code

可以發現,兩個模塊中都定義了driver = webdriver.Remote('http://127.0.0.1:4723/wd/hub', desired_caps),而在pytest收集測試用例的時候,有一個順序:

1. 先收集以test_*.py或*_test.py形式的模塊,pytest在收集的時候是按照字母a-z,數字0-9的順序

2. 然后收集類之外的以test開頭的測試方法

3. 最后收集Test開頭的類里面的test開頭的測試方法

三. 分析

簡單的畫了一個圖,大概說明下pytest收集用例的順序,先收集test_login.py時會創建一個session為00b7be的session對象,然后收集test_welcome.py時會創建一個session為187161的session對象,由於是同一個appium server,端口都為4723,自然后面的對象建立后,前面的會斷開。所以測試方法test_login_success運行時,會提示"A session is either terminated or not started"

從日志里也可以看到這點

Tue, 14 Apr 2020 16:21:04  test_login.py  <module> [line:21] INFO 加載test_login
Tue, 14 Apr 2020 16:21:04  test_login.py  <module> [line:22] INFO test_login_driver是: <appium.webdriver.webdriver.WebDriver (session="00b7bec7-8d8c-4579-8fdb-322abcf4c83c")>
Tue, 14 Apr 2020 16:21:15  test_welcome.py  <module> [line:23] INFO 加載test_welcome
Tue, 14 Apr 2020 16:21:15  test_welcome.py  <module> [line:24] INFO test_welcome_driver是: <appium.webdriver.webdriver.WebDriver (session="1871614e-6cdf-443c-b6e0-fe9b3175839d")>
Tue, 14 Apr 2020 16:21:15  test_login.py  test_login_success [line:28] INFO test_login開始了!
Tue, 14 Apr 2020 16:21:15  test_login.py  test_login_success [line:29] INFO test_login的driver是: <appium.webdriver.webdriver.WebDriver (session="00b7bec7-8d8c-4579-8fdb-322abcf4c83c")>

一種修改的辦法是,在兩個模塊的方法中,初始化driver對象,這樣執行測試方法時,能保證自己的driver是一個在線的狀態。但這種方法不太合理,因為一個方法實例化一個driver對象,那么N個方法就要初始化N個driver對象,顯然是不合適的。所以有必要將公共的driver提取出來,保證一個driver能貫穿整個測試流程

 


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM