剛剛以SCRUM的方式結(jié)束了一個(gè)的ASP.NET網(wǎng)站的測(cè)試的第一個(gè)Spring,因?yàn)閳F(tuán)隊(duì)從無(wú)到有實(shí)現(xiàn)自動(dòng)化測(cè)試系統(tǒng),有必要把這次的經(jīng)驗(yàn)和教訓(xùn)總結(jié)一下,以便后續(xù)的Spring可以獲取一些有意義的借鑒。
創(chuàng)新互聯(lián)服務(wù)項(xiàng)目包括東勝網(wǎng)站建設(shè)、東勝網(wǎng)站制作、東勝網(wǎng)頁(yè)制作以及東勝網(wǎng)絡(luò)營(yíng)銷(xiāo)策劃等。多年來(lái),我們專(zhuān)注于互聯(lián)網(wǎng)行業(yè),利用自身積累的技術(shù)優(yōu)勢(shì)、行業(yè)經(jīng)驗(yàn)、深度合作伙伴關(guān)系等,向廣大中小型企業(yè)、政府機(jī)構(gòu)等提供互聯(lián)網(wǎng)行業(yè)的解決方案,東勝網(wǎng)站推廣取得了明顯的社會(huì)效益與經(jīng)濟(jì)效益。目前,我們服務(wù)的客戶(hù)以成都為中心已經(jīng)輻射到東勝省份的部分城市,未來(lái)相信會(huì)繼續(xù)擴(kuò)大服務(wù)區(qū)域并繼續(xù)獲得客戶(hù)的支持與信任!
因?yàn)槭且粋€(gè)技術(shù)博客,所以使用SCRUM管理這個(gè)測(cè)試項(xiàng)目的經(jīng)驗(yàn)放在別的地方分享,這系列的文章分享一下使用VSTT和Selenium結(jié)合實(shí)現(xiàn)自動(dòng)化測(cè)試系統(tǒng)的經(jīng)驗(yàn)。
Selenium簡(jiǎn)介
Selenium主要是一個(gè)錄制并回放的自動(dòng)化測(cè)試用例編制工具,由一個(gè)錄制工具Selenium IDE(一個(gè)Firefox插件,當(dāng)然這個(gè)工具也可以回放啦),一個(gè)回放工具Selenium Remote Control在其他機(jī)器和其他操作系統(tǒng)上進(jìn)行回放。Selenium的一個(gè)好處就是你可以使用它測(cè)試所有操作系統(tǒng)下的所有主流瀏覽器,至于Linux下面的konqueror和gnome下面自帶的瀏覽器,沒(méi)有試過(guò)Selenium是否支持,當(dāng)然那個(gè)控制臺(tái)界面下的瀏覽器就更沒(méi)有試過(guò)啦。Selenium還有一個(gè)Selenium Grid,據(jù)說(shuō)很強(qiáng)大,因?yàn)轫?xiàng)目比較緊,就沒(méi)有花時(shí)間去看它。
至于Selenium各個(gè)工具的用法,它的官網(wǎng)上有詳細(xì)的文檔,如果文檔也沒(méi)說(shuō)清楚的話,那就直接讀源代碼吧。
Selenium和VSTT的整合
Selenium可以根據(jù)錄制的步驟生成直接在NUnit中使用的C#代碼,這些代碼基本上都可以在VSTT中直接使用,就是一些屬性需要更改。例如[TestFixture]改成[TestClass],[Test]改成[TestMethod]之類(lèi)的,改好以后,啟動(dòng)Selenium-RC,就可以直接在VSTT里面當(dāng)作普通的單元測(cè)試用例執(zhí)行了。
Selenium代碼優(yōu)化
既然要做自動(dòng)化測(cè)試,那么有一點(diǎn)是必須要時(shí)刻考慮的,就是在產(chǎn)品開(kāi)發(fā)過(guò)程中,程序界面甚至是內(nèi)部的類(lèi)庫(kù)接口也是時(shí)刻改變的。而Selenium只能記錄當(dāng)時(shí)錄制測(cè)試用例的界面情況,因此需需要將它生成的代碼分解一下,以面向?qū)ο蟮姆绞絹?lái)重寫(xiě)。例如下面這段代碼的目的是測(cè)試用戶(hù)可以查看自己的博客:
[TestMethod]
public void TheTestTest()
{
selenium.Open("/");
selenium.Click("link=登錄");
selenium.WaitForPageToLoad("30000");
selenium.Type("tbUserName", "donjuan");
selenium.Type("tbPassword", "");
selenium.Click("btnLogin");
selenium.WaitForPageToLoad("30000");
selenium.Click("link=donjuan");
selenium.WaitForPageToLoad("30000");
selenium.Click("link=博客");
selenium.WaitForPageToLoad("30000");
}
但是網(wǎng)頁(yè)頁(yè)面布局,或者Html控件的Id、文本等內(nèi)容隨時(shí)都會(huì)被程序員修改,修改的原因有多種,例如修復(fù)新的錯(cuò)誤(Bug),或者僅僅就是代碼重構(gòu)。因此作為測(cè)試團(tuán)隊(duì),不能總是認(rèn)為網(wǎng)頁(yè)的內(nèi)容一成不變的。而象登錄這種操作,大部分測(cè)試用例都會(huì)用到,所以最好只要為登錄動(dòng)作創(chuàng)建唯一的代碼 。有多個(gè)方案:
1. 為登錄創(chuàng)建一個(gè)獨(dú)立的測(cè)試用例,本來(lái)登錄這個(gè)功能就是要測(cè)試的嘛,在編輯自動(dòng)化測(cè)試用例列表的時(shí)候,把登錄用例放在最前面。
2. 為登錄動(dòng)作創(chuàng)建一個(gè)單獨(dú)的函數(shù),例如LogOn(),然后在其他測(cè)試用例當(dāng)中(包括登錄的測(cè)試用例)調(diào)用這個(gè)函數(shù),另外,因?yàn)榭赡軙?huì)需要用到不同的 用戶(hù),所以最好把用戶(hù)名和密碼等變量提取出來(lái),變成LogOn(string username, string password)之類(lèi)的函數(shù)。
兩個(gè)方案,顯然是第二個(gè)方案的彈性大,但是對(duì)于第一個(gè)方案,如果測(cè)試人員都是新手,且對(duì)代碼不熟悉的話,建議可以考慮。
于是我們的代碼就變成類(lèi)似下面的代碼:
using System;
//
//這個(gè)異常是故意創(chuàng)建出來(lái),用來(lái)封裝所有在測(cè)試代碼中發(fā)生的錯(cuò)誤
//
public class CaseErrorException : Exception
{
public CaseErrorException(string message)
: base()
{
}
public CaseErrorException(Exception inner)
: this(null, inner)
{
}
public CaseErrorException(string message, Exception inner)
: base(message == null ? "測(cè)試代碼錯(cuò)誤,請(qǐng)修復(fù)測(cè)試代碼,查看InnerException屬性!" :
string.Format("測(cè)試代碼錯(cuò)誤,請(qǐng)修復(fù)測(cè)試代碼,詳細(xì)錯(cuò)誤信息:{0};或者查看InnerException屬性!", message),
inner)
{
}
}
public class UserOperationsHelper
{
public void LogOn(string username, string password)
{
// string.Empty留出來(lái)為測(cè)試目的服務(wù)
if (username == null)
throw new CaseErrorException(new ArgumentNullException("username"));
if (password == null)
throw new CaseErrorException(new ArgumentNullException("password"));
selenium.Open("/");
selenium.Click("link=登錄");
selenium.WaitForPageToLoad(Consts.TimeToWaitForPageLoad);
selenium.Type("tbUserName", username);
selenium.Type("tbPassword", password);
selenium.Click("btnLogin");
selenium.WaitForPageToLoad(Consts.TimeToWaitForPageLoad);
}
}
public static class Consts
{
//將等待的時(shí)間提取成一個(gè)公開(kāi)的函數(shù),因?yàn)樵诮窈蟠笠?guī)模的測(cè)試
//過(guò)程中,很多自動(dòng)化測(cè)試用例不簡(jiǎn)單地執(zhí)行,會(huì)導(dǎo)致網(wǎng)站響應(yīng)速度
//變慢,所以
public const string TimeToWaitForPageLoad = "30000";
}
public class TestLibrary
{
public UserOperationsHelper UserHelper { get; private set; }
}
public class TestClass
{
[TestMethod]
public void LogOnTest()
{
var username = "donjuan";
var password = "它是個(gè)秘密";
TestLibrary.UserHelper.LogOn(username, password);
//在測(cè)試過(guò)程中,我們發(fā)現(xiàn)這個(gè)鏈接是
//根據(jù)用戶(hù)名而變的,為了擴(kuò)展性,動(dòng)態(tài)生成其標(biāo)識(shí)文本
selenium.Click(string.Format("link={0}", username));
selenium.WaitForPageToLoad(Consts.TimeToWaitForPageLoad);
selenium.Click("link=博客");
selenium.WaitForPageToLoad(Consts.TimeToWaitForPageLoad);
//執(zhí)行一些必要的測(cè)試驗(yàn)證過(guò)程
Assert.IsTrue(selenium.IsTextPresented(...));
}
}
這里稍微解釋一下,創(chuàng)建自動(dòng)化測(cè)試代碼,就是為了節(jié)省手工重復(fù)測(cè)試的工作量以及測(cè)試失誤的風(fēng)險(xiǎn)。但只要是代碼,都會(huì)有可能出錯(cuò),因此自動(dòng)化測(cè)試框架里面創(chuàng)建了一個(gè)CaseErrorException,這樣在每次分析測(cè)試用例失敗的時(shí)候,可以一眼區(qū)分開(kāi)測(cè)試代碼的錯(cuò)誤和產(chǎn)品代碼中的錯(cuò)誤。例如在UserOperationHelper.LogOn函數(shù)中的參數(shù)檢查,當(dāng)然啦,在測(cè)試過(guò)程當(dāng)中,有可能需要測(cè)試不輸入用戶(hù)名或者密碼的情況下,驗(yàn)證登錄界面是否正常工作的情況。因此在驗(yàn)證參數(shù)的時(shí)候,特意為這種情況留下了String.Empty的入口,而對(duì)于null值,則基本上可以判斷是因?yàn)闇y(cè)試人員在編寫(xiě)代碼上的失誤(具體原因會(huì)在數(shù)據(jù)驅(qū)動(dòng)測(cè)試?yán)锩嬷v到)。
至于TestLibrary的初始化,完全可以放到每一個(gè)測(cè)試類(lèi)型的TestInitializer里面,如下表所示:
[TestClass] public class AddBlogTest { private TestContext testContextInstance; public TestContext TestContext { get { return testContextInstance; } set { testContextInstance = value; } }
private TestLibrary TestLibrary; private ISelenium selenium;
[TestInitialize] public void SetupTest() { TestLibrary = TestLibrary.SetupTest(TestContext); selenium = TestLibrary.Selenium; }
[TestCleanup] public void TeardownTest() { TestLibrary.Shutdown(); } } |
咋看起來(lái),把LogOn測(cè)試用例分解成那么多的類(lèi)型,有點(diǎn)畫(huà)蛇添足,實(shí)際上這些函數(shù)庫(kù)正是為了更方便地創(chuàng)建后續(xù)的測(cè)試用例耗費(fèi)的磨刀的功夫。例如下面的代碼是基于一些創(chuàng)建好了的函數(shù)編寫(xiě)的測(cè)試用例:
[TestMethod] public void CreateBlog() { TestLibrary.UserHelper.LogOnAsAdmin(); var blog = TestLibrary.BlogHelper.CreateBlog("博客的標(biāo)題", "博客的鏈接");
selenium.Click("link=管理博客"); selenium.WaitForPageToLoad(Consts.TimeToWaitForPageLoad);
Assert.IsTrue(selenium.IsElementPresent(string.Format("link={0}", blog.Title))); } |
下一篇文章網(wǎng)站測(cè)試自動(dòng)化系統(tǒng)—數(shù)據(jù)驅(qū)動(dòng)測(cè)試講解這個(gè)框架和數(shù)據(jù)驅(qū)動(dòng)測(cè)試的結(jié)合。