测试对于一个应用系统来说是非常关键的,但纯单元测方式(XUnit)的方式在B/S结构的信息管理系统中仍然有一些不足,比如要测试界面元素,网页导行方面,需要其它的一些测试架框作为辅助,这两天开始查找并学习了一些网站测试方面的框架。把学习心得发出来与大家工享。

  一、几种常用框架的介绍。

  1、NUnitAsp官方网址:http://nunitasp.sourceforge.net/

  优点:可以在测试代码中直接使用使用调用网页中的服务器控件。并获取该控件的属性。

  缺点:不够灵活。正如他的官方网上所说的:

  NUnitAsp主要是对后台代码进行单元测试,即它主要是用于测试服务端的逻辑,JavaScript和其它的客户端代码是被忽略的。

  2、WatiN官方网址:http://watin.sourceforge.net/

  优点:非常灵活,可以模拟用户在客户端浏览器中的大部份操作,API也比较简单。

  缺点:文档较少,官方网上的例子也很少。


  因为我的目的是测试网站里网页的导行,主要是要模拟用户的操作,因此选择了使用WatiN。下面的内容也主要是讲述它的使用。

  二、简单的例子

  下面先看一下官司方网站上给出的一个简单的例子。

  [Test]
  publicvoidSearchForWatiNOnGoogle()
  ...{
  /**//*这一句代码会打开一个IE浏览器,打开www.google.com网址。
  *在创建军了IE对象以后也可以用ie.GoTo(url);来转到其它网址。
  */
  using(IEie=newIE("http://www.google.com"))
  ...{
  /**//*在浏览器中查找name属性为"q"的输入控件,然后调用它的TypeText方法
  *模拟用户用键盘输入"WatiN"
  *注意TypeText方法是真的模拟用户一个一个字母地输入进这个输入框的。
  *如果只是想设置该控件的内容可以把这句改为:
  *ie.TextField(Find.ByName("q")).Value="WatiN";
  */
  ie.TextField(Find.ByName("q")).TypeText("WatiN");
  ie.Button(Find.ByName("btnG")).Click();
  Assert.IsTrue(ie.ContainsText("WatiN"));
  }
  }
 

  这段代码的作用是模仿用户打开IE浏览器到GOOGLE网站查询WatiN的内容,然后确定有没有打到这些信息。

  三、WatiN里常用的类

  WatiN.Core.Find

  普遍使用的类是Find类的,它是一个工厂类,主要使用它的静态方法来实现一些查询条件。比如像上面的例子中的这个语句“ie.Button(Find.ByName(”btnG“)).Click();”调用了Find的静态方法ByName来查询一个name属性为指定值的HTML元素,然后再调用IE对象的Button方法把这个元素转换为按钮对象。

  WatiN.Core.IE

  这应该是关键的类了。他常用的是属性和方法:

  1.属性

  HtmlDialogs返回当前对象用JavaScript打开的模式窗口(需然帮助文档中说非模式窗口也包括在内,但在试用中发现用window.open打开的窗口没有被)

  Frames返回当前的象里的所有Frames

  2.方法

  静态方法AttachToIE与一个已经找开的IE关连。

  Button,TextField,Image,Div等一系列方法。与Find对象共同使用用于返回IE中特定的按钮,输入框,等HTML元素。

  四、应用中的一些技巧

  1、如何去捕获一个新弹出的窗口。

  背景:有些链接是从一个新弹出的窗口中打开的,我如果关连上这类型的窗口。

  方案:下面以一个具体例子为例。系统中有一个登陆界面:login.aspx登陆成功能后会调用JAVASCRIPT的.open方法打开系统的主窗口,然后关闭当前这个窗login.aspx。 $new_page$

  刚开始时,我的代码是这样写的

  publicvoidLogin(stringuid,stringpasswd)
  ...{
  stringurl=Host+"login.aspx";
  ie=newIE();
  ie.GoTo(url);
  ie.WaitForComplete();
  ie.TextField(Find.ById("txtUserName")).Value=uid;
  ie.TextField(Find.ById("txtPwd")).Value=passwd;
  ie.Button(Find.ById("btnLogin")).Click;//注意这里,我每次执行到这里都出错。下面会说明原因。
  ie=IE.AttachToIE(Find.ByTitle(newWatiN.Core.Comparers.RegexComparer(newSystem.Text.RegularExpressions.Regex(".*综合信息系统.*"))));
  ie.WaitForComplete();
  MainFrame=ie.Frame(Find.ByName("MainFrame_00001"));
  MenuFrame=ie.Frame(Find.ById("leftMenu"));
  }

  看到上面的注译没有?那么执行完后WatiN都会出现一个等待完成超时的异常,这是因为点击了登陆按钮后,如果登陆成功的话,系统已经把这个窗口关闭了,因为它根本不可能完成。呵呵~~```所以要修改一下。


  publicvoidLogin(stringuid,stringpasswd)
  ...{
  stringurl=Host+"frmlogon.aspx";
  ie=newIE();
  ie.GoTo(url);
  ie.WaitForComplete();
  ie.TextField(Find.ById("txtUserName")).Value=uid;
  ie.TextField(Find.ById("txtPwd")).Value=passwd;
  ie.Button(Find.ById("btnLogin")).ClickNoWait();//这个方法改成这样,那点击后不会等代码IE完成了。
  //下面这句是处理登陆的技巧所在,系统打开了另一个窗口。
  //下面这句是用正则表达式捕获这个窗口。
  ie=IE.AttachToIE(Find.ByTitle(newWatiN.Core.Comparers.RegexComparer(newSystem.Text.RegularExpressions.Regex(".*综合信息系统.*"))));
  ie.WaitForComplete();
  MainFrame=ie.Frame(Find.ByName("MainFrame_00001"));
  MenuFrame=ie.Frame(Find.ById("leftMenu"));//把系统里的Frame先保存下来。
  }
 


  2、对于alert、confirm等javascript弹出的窗口的捕获。

  背景:一个系统经常会使用以上这些javascript来弹出一些提示信息,如果捕获这些窗口,并模拟用户点击这些窗口上的OK或Cancel按钮?

  方案:其实WatiN在默认情况下,都会自动地去点击这些弹出式窗口上的Cancel按钮的,但如果用户要明确点击哪些事件的话可以对IE对象增加一个“查看器”()


  //错误事例
  protectedvoidf()
  ...{
  WatiN.Core.DialogHandlers.ConfirmDialogHandlerdh=newWatiN.Core.DialogHandlers.ConfirmDialogHandler();
  ie.AddDialogHandler(dh);//这句增加了一个控制器
  /**//*
  假设yzp_dic_btn_cls这个按钮的onclick事件会用JS弹出一个Confirm窗口。
  错误,测会在这里弹出一个窗口,然后等待用户进行操作(随便点击一个按钮才继续执行)
  然后因为用户已经点击了按钮关闭弹出窗口。下一句的dh.WaitUntilExists(3);将会超时,抛出异常
  */
  doc.Button(Find.ByName("yzp_dic_btn_cls")).Click();
  dh.WaitUntilExists(3);//等待弹出窗口的出来。多等三秒。
  dh.OKButton.Click();//点击这个窗口的OK按钮
  ie.RemoveDialogHandler(dh);
  doc.Button(Find.ByName("btn_close")).Click();
 


  正确的示例应该是:

  protectedvoidf()
  ...{
  WatiN.Core.DialogHandlers.ConfirmDialogHandlerdh=newWatiN.Core.DialogHandlers.ConfirmDialogHandler();
  ie.AddDialogHandler(dh);
  doc.Button(Find.ByName("yzp_dic_btn_cls")).ClickNoWait();//把原来的Click改成这个方法。
  dh.WaitUntilExists(3);//等待弹出窗口的出来。多等三秒。
  dh.OKButton.Click();//点击这个窗口的OK按钮
  ie.RemoveDialogHandler(dh);
  doc.Button(Find.ByName("btn_close")).Click();
  }
 

  3、关于用JS弹出的Modal窗口(模式窗口)的处理。

  背景:有些地方需要弹出模式窗口来处理数据。

  方解:当点击了弹出模式窗口的按钮或连接后马上用IE对像的HtmlDialogs属生来获取模式窗口。

  protectedvoidf()
  ...{
  //假设下面这行代码会弹出一个模式窗口把原来的Click改成这个方法。
  //记住这里要用ClickNoWait而不能用Click,否则在模式窗口关闭之前代码不会继续执行。
  ie.Button(Find.ByName("yzp_dic_btn_cls")).ClickNoWait();
  ie.HtmlDialogs[ie.HtmlDialogs.Length-1].TextField(Find.ByName("Q")).Value="Hello";
  ie.HtmlDialogs[ie.HtmlDialogs.Length-1].Button(Find.ByName("btn_query")).Click();
  }
 

  总结

  这个框架还是比较简单,主要要主意的事情是HTML元素的Click与ClickNoWait方法的区别。还有对于输入框,是直接设置它的Value还是调用TypeText或AppendText方法模拟用户手工输入内容。前者不会激发该控件的任何JS事件,而后者会激KeyDown、KeyUP等事件。