在軟件生命周期各個階段都逐漸開始重視交付速度的情況下,全球化測試同樣也面臨了敏捷的挑戰(zhàn)。因此自動化測試也開始在全球化測試領(lǐng)域變得流行起來,但是由于受限于翻譯驗證性測試中"一次編寫,多個語言環(huán)境下運行"的特點,在利用 Selenium 進行 Web 自動化測試時對元素的一致定位成了編寫自動化腳本的瓶頸。本文將針對現(xiàn)階段遇到的問題結(jié)合實例詳細闡述在利用 Selenium 對 Web 應用進行自動化測試時定位元素的方法,并針對全球化測試中遇到的難點給予重點解答。
翻譯驗證性測試中定位 Web 元素遇到的挑戰(zhàn)
翻譯驗證性測試是全球化測試中的一部分,旨在驗證軟件用戶界面中需要翻譯的字符串是否已經(jīng)正確翻譯。為了提高效率,目前一般采用先由熟悉產(chǎn)品的測試支持工程師書寫測試用例,再由熟悉各國語言的測試人員執(zhí)行用例或者先由測試支持工程師將復雜用例的執(zhí)行結(jié)果在各個語言環(huán)境下截圖,然后再發(fā)送給測試人員進行驗證兩種模式。目前隨著持續(xù)交付模式的出現(xiàn),無論是測試人員多語言環(huán)境下重復執(zhí)行用例還是測試支持工程師多語言環(huán)境下手動截圖都難以滿足持續(xù)交付對速度的要求,因此測試支持工程師編寫自動化腳本完成多語言環(huán)境下的自動截圖變成了提高效率的必然選擇。
在傳統(tǒng)的 Web 自動化測試應用場景中,由于程序編碼的不規(guī)范,元素通常很難直接通過靜態(tài) ID 或 name 定位,幸好有 CSS 選擇器和 XPath 定位為我們提供了新的可能。然而由于 DHTML 和 Ajax 的盛行,新的挑戰(zhàn)又擺在了測試人員面前,由 Dojo 等客戶端腳本生成的 DOM 結(jié)構(gòu)異常復雜:層次繁多且一個 DOM 樹上可能存在多個頁面的 DOM 子節(jié)點,而且其中元素的 ID 是動態(tài)的(該 ID 一般由一個前綴和一個動態(tài)的數(shù)字組成,前綴表明了元素對應的部件類型,數(shù)字則由運行時該小部件在 DOM 樹上的位置確定)。這導致了有些元素甚至需要依靠 CSS 偽類再運用組合選擇器才能精確定位的到。而對于在翻譯驗證性測試中的自動化,這種問題尤其突出,由于翻譯驗證性測試中的自動化腳本“一次編寫,多個語言環(huán)境下”運行的要求,元素定位的后“救命稻草”-CSS 偽類中的內(nèi)文本匹配也無法開箱即用了。
下面筆者將通過實例詳細闡述元素定位的方法,以期讀者能靈活運用以應對上述的各種挑戰(zhàn)。
Selenium IDE 與 WebDriver 定位元素概覽
Selenium 是一款強大的瀏覽器自動化工具,利用它可以實現(xiàn) Web 界面測試的自動化。根據(jù)實現(xiàn)原理和使用方式的不同,其又分為 Selenium IDE 和 WebDriver,前者以瀏覽器插件的方式賦予用戶簡單快速地創(chuàng)建自動化腳本的能力,后者則通過 API 庫的形式向用戶提供更加靈活和健壯的 Web 自動化能力。
Selenium IDE 在其命令中采用 locatorType=location 的格式進行定位,該方法將定位到條件匹配的第一個元素,一般情況下 locatorType 可以省略;WebDriver 利用 findElement(By.locatorType("value") 或 findElements(By.locatorType("value") 函數(shù)查找元素,前者將返回一個的元素,后者將返回一個元素列表。具體的定位由 By.locatorType("value") 實現(xiàn)。
除去定位使用的方法而言,二者本質(zhì)上都是通過 locatorType 進行定位的,常用的 locatorType 包括 id,name,link text,dom/JavaScript,xpath,css 等,示例見清單 2(行末括號中的數(shù)字顯示當前行定位器將定位到清單 1 HTML 文檔中的第幾行元素):
清單 1. HTML 示例 1
(01)<html>
(02) <body>
(03) <form id="loginForm" name="loginFrom">
(04) <input class="required" name="username" type="text" />
(05) <input class="required passfield" name="password" type="password" />
(06) <input name="continue" type="submit" value="Login" />
(07) <input name="continue" type="button" value="Clear" />
(08) </form>
(09) <p>Are you sure you want to do this?</p>
(10) <a href="continue.html">Continue</a>
(11) <a href="cancel.html">Cancel</a>
(12)</body>
(13)<html>
清單 2. Selenium 和 WebDriver 中通用的定位方式
//by ID
id=loginForm(03)//in IDE
WebElement element = driver.findElement(By.id("loginForm"));(03)//in WebDriver
//by Name
name=continue type=button(07)//in IDE
WebElement element = driver.findElement(By.name("loginForm"));(03)//in WebDriver
//by link text
link=Continue(10)//in IDE
WebElement element = driver.findElement(By.linkText("Continue"));(10)//in WebDriver
//by DOM or JavaScript
dom=document.forms[0].elements['username'](04)//use DOM in IDE
WebElement element = (WebElement) ((JavascriptExecutor)driver).
executeScript("return $('.required')[0]"); (04)//use JQuery in WebDriver
//by XPath
//form[@id='loginForm']/input[1](04)//by IDE
WebElement element = driver.findElement(By.xpath("//input[@name='username']"));(04)//by WebDriver
//by CSS
css=#loginForm input[type="password"](05)//by IDE
WebElement e = driver.findElement(By.cssSelector("input.passfield")(05)//by WebDriver
通過 ID 和 name 定位是高效也是的定位方式,不過由于 name 不一定,在定位時匹配條件的元素可能有多個,因此這種情況下只會定位到匹配條件的第一個元素。針對多個元素具有相同 name(或鏈接文本)屬性的情況還需額外增加其他的過濾器才能進行精確定位,如清單 2 中 IDE 利用 name 進行定位的示例。
DOM 代表了整個 HTML 文檔的結(jié)構(gòu),使用 JavaScript 可以訪問 DOM 中的節(jié)點。Selenium IDE 基于 DOM 結(jié)構(gòu)可以使用 JavaScript 的點操作符進行層次定位以簡化定位操作,由于只有 DOM 定位器以“document”開頭,因此“dom=”也可以省略;此外在 WebDriver 中甚至可以通過執(zhí)行任意的返回值為 DOM 對象的 JavaScript 語句來查找元素,如清單 2 中 WebDriver 使用 JQuery(一種 JavaScript 框架)的元素查找函數(shù)$進行定位。
XPath 初是用來在 XML 文檔中定位 DOM 節(jié)點的語言,由于 HTML 也可以算作 XML 的一種實現(xiàn),所以 Selenium 也可以利用 XPath 這一強大的語言來定位 Web 元素。XPath 在傳統(tǒng)屬性定位之外擴展了諸如“定位第三個多選框”等定位能力,以便應對沒有 ID 或 name 屬性的情況。利用 Xpath 可以通過路徑,或者相對于一個可精確定位的元素的相對路徑來定位。為了保證定位的健壯性,推薦使用相對路徑和基于位置關(guān)系的定位。同樣由于只有 XPath 定位器以“//”開頭,所以“xpath=”也可以省略。
CSS (Cascading Style Sheets) 是一種用于渲染 HTML 或者 XML 文檔的語言,CSS 利用其選擇器可以將樣式屬性綁定到文檔中的指定元素,即前端開發(fā)人員可以利用 CSS 設(shè)定頁面上每一個元素的樣式。所以理論上說無論一個元素定位有多復雜,既然開發(fā)人員能夠定位到并設(shè)置樣式,那么測試人員同樣應該也能定位繼而操作該元素。這也正是 Selenium 官方極力推薦使用 CSS 定位,而不是 XPath 定位的主要原因。CSS 定位被推崇的另一個原因是不同的瀏覽器 XPath 引擎不同甚至沒有自己的 Xpath 引擎,這導致了 XPath 定位速度較慢,而采用 CSS 定位往往能用更簡潔的語法快速定位到復雜的元素。因此后文將詳細介紹 CSS 定位器的使用方法。
除了以上通用的定位方式之外,在 Selenium IDE 中還可通過 identifier 進行定位,這是通用也是默認的定位方式,其定位時首先將 identifier 的值視為 ID 繼而查找匹配元素,沒有匹配項之后再把其看作 name 進行查找,直到找到第一個匹配的元素為止。由于其是默認的定位方式,因此“identifier=”通常也會省略。對于 WebDriver 而言,特有的定位方式更多。首先它可以通過鏈接文本進行部分匹配,如清單 3 所示;然后它還可以通過標簽名或者 CSS 類名的方式進行定位,利用 findElement 的話這兩種方式均是定位到第一個匹配條件的元素,利用 findElements 則可獲得一個所有滿足條件的元素列表,如清單 3 所示。