程序员在旅途

用这生命中的每一秒,给自己一个不后悔的未来!

0%

基于Selenium工具对Web应用进行功能测试

一、背景。

  最新一年(2019)的软件测试大赛即将拉开帷幕,今天与往年相比,增加了自主可控的分赛项,进一步的丰富了比赛的内容,各项评分依据也更加明确,因此能够根据分数更有针对性的找出自己的代码问题了,从而获得更好地分数,提高脚本编写能力。评测工具完善速度之快,超出个人想象,可以看得出来主办方费心了,预祝软件测试大赛发展的越来越好。
  自主可控赛项的比赛内容有三部分构成,分别为:功能测试、性能测试、众包测试,其中功能测试的内容就是利用Selenium测试工具对Web应用进行功能测试。之所以称作为自主可控,是因为待测试Web应用是部署在国产的应用服务器上面的,采用的国产CPU、操作系统等,具备完全自主知识产权,编写的脚本也必须启动国产浏览器(360浏览器、QQ浏览器)来测试Web应用才可以得分。
  有关Selenium的介绍与环境安装,在之前的一篇博客里面也写过,链接为Selenium介绍及环境搭建。经过又一段时间的学习,对这个有了进一步的认识,加上此次比赛需要使用国产浏览器的环境,此前并不了解这方面的东西,因此打算在这里再次记录一下学习过程与心得体会。
  对软件测试感兴趣的,想参加的比赛的,官网地址在这里全国大学生软件测试大赛

二、Selenium介绍以及环境配置。

  2.1、Selenium介绍:
  我们知道,软件测试的目的在于在规定的条件下对程序进行操作,以发现程序错误,衡量软件质量,并对其是否能满足设计要求进行评估。由于在某些特定的场景下,例如:重复单一的数据录入或是击键等测试操作,使用手动测试会进行大量重复性的操作,浪费大量时间和人力,因此,这种情况下有必要引入自动化测试来对手动测试进行补充,从而提高测试效率。
  根据对测试模块规模的不同,敏捷大师Mike Cohn提出了测试金字塔的概念,从而为测试分层打下基础,以此来提高自动化测试的效率。测试金字塔如下图所示:
测试金字塔
  随着测试规模的不断增大,测试成本与测试难度也在不断增加。图中的三层,代表的含义如下:①Unit测试即单元测试,是对程序的函数方法进行测试,属于白盒测试的一种类型,通常由开发人员编写相关的测试脚本进行测试。在Java中我们常会用Junit框架进行测试。②Service服务测试即对相关的API接口进行测试,目的是为了测试服务提供的数据等输出结果是否能够正常获取,以此来验证相关业务逻辑代码是否书写正确,常见的工具有Postman等。③UI测试是对整个系统进行的完整模块的测试,主要来验证系统的界面元素是否符合规范、提供的功能是否满足预期,常见的测试工具有Selenium、Appium等。
  Selenium是进行UI测试的一种框架,主要是对Web应用进行功能测试,从2004年诞生至今,已经发展到3.0版本,是一个非常优秀的开源测试框架。Selenium的框架底层使用JavaScript模拟真实用户对浏览器进行操作,测试脚本执行时,浏览器根据脚本代码做出点击,输入,打开,验证等操作,从普通终端用户的角度测试应用程序,以期来发现程序的Bugs,促进程序功能的完善。如果需要进一步了解此框架,可以去github了解源码,或者官网了解他的近期发展情况。
  2.2 、环境配置:
  这里介绍的是Windows系统下使用Java语言进行测试脚本的编写,环境配置的整体流程为:1,下载JDK,配置好Java运行环境;2,下载Selenium的Java开发包,编写脚本需要依赖这个包;3,下载与浏览器版本对应的driver驱动,并添加到环境变量path下。由于是在mooctest平台做题目,因此还需要下载平台提供的最新eclipse评分插件(版本3.0.6)。
  1)JDK安装配置,在CMD命令行输入命令: java -version 检查是否安装配置成功。
java-version
  2)下载eclipse,并从mooctest平台下载最新的eclipse插件,将插件安装到eclipse。安装之后,eclipse的工具栏应该出现了mooctest的按钮。也可以下载mooctest提供的含插件的Eclipse,但是需要更新插件至最新版本。
mooctest插件
  3)Selenium 工具包下载配置。可以从官网或者mooctest下载比赛要求的selenium-standalone-server-3.9.1.jar工具包(此包的名称需要修改成项目依赖的包名selenium-standalone.jar),下载后,放置于 C盘 mooctest文件夹下面,如果是第一次使用这个平台,则需要自己新建一个mooctest文件夹,然后把包放进去。
  4) 下载浏览器驱动。自主可控赛项,举办方要求使用国产浏览器,在这里采用了360浏览器。由于360极速模式基于chromium开源项目,因此这里下载Chromedriver即可驱动360浏览器。查看浏览器版本对应的driver版本下载即可,然后配置好环境变量。
Chromedriver-path
  5)在系统环境变量中指定webdriver.chrome.bin的值为360浏览器的路径值,在启动浏览器的时候选择极速模式。
webdriver-chrome-bin
  到这里,环境就是配置完毕了,其中要特别注意Chromedriver的版本,浏览器路径的配置问题。完成后,打开Eclipse下载代码,测试运行,检验环境是否正确。
  使用mooctest插件的run and submit 提交代码,当eclipse控制台输出driver和浏览器的路径,浏览器能正常打开,说明环境配置正确,这时候就可以按照规则文档进行相关的测试了。
run-submit测试图

三、Selenium 使用方法。

  3.1、代码示例:
  在配置好环境之后,之后就可以使用selenium进行元素的查找与点击等操作从而对WEB应用进行全面的功能测试。一个简单的示例如下:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
public class Example {
public static void test(WebDriver driver) {
try {
//10S的元素查找时间
driver.manage().timeouts().implicitlyWait(10, TimeUnit.SECONDS);
//请求打开一个网页
driver.get("https://www.baidu.com");
//查找元素,并向元素输入内容
WebElement kwElement = driver.findElement(By.id("kw"));
kwElement.clear();
kwElement.sendKeys("安徽工业大学");
driver.findElement(By.id("su")).click();
driver.findElement(By.linkText("欢迎访问安徽工业大学门户网站!")).click();
} catch (Exception e) {
}

}
public static void main(String[] args) {
//创建一个ChromeDriver对象,后续与网页元素的交互都得要依赖这个对象。
WebDriver driver = new ChromeDriver();
try {
test(driver);
} catch (Exception e) {
e.printStackTrace();
} finally {
driver.quit();
}
}

}

  在上面的代码中,最重要的两个类是 WebDriver、WebElement,driver通过各种不同的方式查找到Web元素,然后使用该对象进行各种操作(如点击、输入元素、清除元素)与元素交互,从而实现了Web应用的测试。
  3.2 、WebDriver API 的使用:
  3.2.1 控制浏览器:
  对浏览器的控制是Selenium的一个重要应用,可以对浏览器最大化,设置浏览器的高度和宽度以及对浏览器进行导航操作等。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
// 浏览器打开网址。
driver.get("https://www.baidu.com");
// 浏览器最大化。
driver.manage().window().maximize();
// 根据driver获取浏览器窗口对象,并设置浏览器的窗口大小。对浏览器的窗口管理,都是通过该对象进行的
Window window = driver.manage().window();
Dimension arg0 = new Dimension(600, 900);
window.setSize(arg0);
// 根据driver获取浏览器导航对象,根据此对象,可以对浏览器进行刷新,前进后退操作。
Navigation navigate = driver.navigate();
navigate.back();
navigate.forward();
navigate.refresh();
// 浏览器关闭
driver.close();

  3.2.2 元素定位方法:
  使用Selenium对Web应用进行功能测试,首先做的就是定位到目标元素,然后才能测试其是否满足功能需求。我们可以通过浏览器的自带的调试工具,来获取网页元素的属性。
元素定位方法
  1)id,通过元素的ID属性来定位一个或者多个元素。

1
2
3
WebElement kwElement = driver.findElement(By.id("kw"));
kwElement.clear();
kwElement.sendKeys("安徽工业大学");

  2)name,通过元素的name属性来定位一个或者多个元素。

1
WebElement kwElement = driver.findElement(By.name("***"));

  3)className,tag,css selector。

1
2
3
driver.findElement(By.className("***"));
driver.findElement(By.tagName("***"));
driver.findElement(By.cssSelector("***"));

  4)linkText,通过链接文字去查找元素;partialLinkText,通过部分链接文字去查找元素。

1
2
3
<a href="www.baidu.com">
<span class="label">百度一下,你就知道</span>
</a>

  以上面的例子为例,这个在页面上展示的就是一个超链接,我们可以通过超链接的全部或者部分文字来定位元素。

1
2
driver.findElement(By.linkText("百度一下,你就知道"));
driver.findElement(By.partialLinkText("百度一下"));

  5)XPath定位方式,XPath 是一门在 XML 文档中查找信息的语言。XPath 可用来在 XML 文档中对元素和属性进行遍历。web网页内容是根据HTML文档结构进行编排的,XPath 是根据元素节点在web网页中的节点位置来进行定位。如下图所示:

1
driver.findElement(By.xpath("(.//*[normalize-space(text()) and normalize-space(.)='学籍证明'])[1]/following::div[1]"));

  在一个完整的Web页面中,拥有相同属性的网页元素通常不止一个,因此,上面所说的driver.findElement拥有复数形式,即:driver.findElements(By.***(“***“));具体是用什么方法,这个要看页面的整体布局。
  在有些时候,一个页面会嵌套多个frame,这时候如果想要定位某一个frame中的元素,需要切换到frame中去,操作完成之后,返回默认文档。如下:

1
2
3
4
//根据id找到frame元素(上面几种定位方式都可以),然后切换过去。
WebElement frameElement = driver.findElement(By.id("frame_id"));
driver.switchTo().frame(frameElement);
driver.switchTo().defaultContent();

  3.2.3 鼠标、键盘事件操作:
  在Web测试中,常见的鼠标操作有:单击、双击、右击、拖拽、悬浮等操作,这些方法封装在了Actions类中。方法有:① contextClick(),右击; ② doubleClick(),双击; ③ dragAndDrop(source, target),拖拽; ④  moveToElement() 悬浮在某一元素上。

1
2
3
4
5
6
7
8
//构造Actions类,使用该类进行鼠标的各种操作
Actions actions = new Actions(driver);
//鼠标右击
actions.contextClick(kwElement).perform();
//鼠标双击
actions.doubleClick().perform();
//鼠标悬浮在某一元素上
actions.moveToElement(kwElement).perform();

  键盘的一些操作主要就是输入键盘上的有特殊功能的字符,比如 回车键,删除键等以达到相应地目的。键符封装在了Keys类中,主要有:删除、空格、制表、回车等。使用组合的Ctrl+a、c等还可以实现选择和复制等功能。

1
2
3
4
5
6
7
8
9
10
//删除
kwElement.sendKeys(Keys.BACK_SPACE);
//空格
kwElement.sendKeys(Keys.SPACE);
//回车
kwElement.sendKeys(Keys.ENTER);
//选择
kwElement.sendKeys(Keys.CONTROL,"a");
//复制
kwElement.sendKeys(Keys.CONTROL,"c");

  3.3 、 断言的使用 :
  断言常用来验证应用程序的状态是否和预期的一致。常见的断言包括:验证页面内容,如标题是否为X或当前位置是否正确,或是验证该复选框是否被勾选。
  Selenium 提供了三种模式的断言:assert 、verify、waitfor。三种模式代表的含义分别为:
   ①  assert 失败时,该测试将终止。
   ②  verify 失败时,该测试将继续执行,并将错误记录下来。
   ③  waitfor 用于等待某些条件变为真。如果该条件为真,继续执行。如果该条件为假,则将失败并暂停测试。

1
2
//判断当前页面title与预期结果是否一致         
Assert.assertEquals(driver.getTitle(), "百度一下,你就知道");

四 、航天中认功能测试案例。

  4.1、运行结果:
航天中认测试结果
  4.2 、代码示例:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
  public class Example {

public static void test(WebDriver driver) {
try {

driver.manage().timeouts().implicitlyWait(10, TimeUnit.SECONDS);
driver.manage().window().maximize();
driver.get("http://114.116.106.156/show-how/common/login.jsp");
//登陆
driver.findElement(By.id("username")).sendKeys("test006");
driver.findElement(By.id("password")).sendKeys("1");
driver.findElement(By.xpath("//*[@id=\"userForm\"]/div[1]/div[4]/button")).click();
Thread.sleep(1000);
//修改个人信息
driver.findElement(By.xpath("//*[@id=\"main-menu\"]/li[2]/a")).click();
Thread.sleep(1000);
driver.findElement(By.xpath("//*[@id=\"main-menu\"]/li[2]/ul/li[1]/a")).click();
Thread.sleep(1000);
driver.findElement(By.xpath("//*[@id=\"main-menu\"]/li[2]/ul/li[1]/ul/li[1]/a")).click();
Thread.sleep(1000);
//修改维护信息
driver.findElement(By.xpath("//*[@id=\"pimInfo_email\"]")).clear();
Thread.sleep(1000);
driver.findElement(By.xpath("//*[@id=\"pimInfo_email\"]")).sendKeys("123456@163.com");
Thread.sleep(1000);
driver.findElement(By.xpath("//*[@id=\"pimInfo_cellphone\"]")).clear();
Thread.sleep(1000);
driver.findElement(By.xpath("//*[@id=\"pimInfo_cellphone\"]")).sendKeys("12345678901");
Thread.sleep(1000);
driver.findElement(By.xpath("//*[@id=\"submitButton\"]")).click();
Thread.sleep(5000);
//退出浏览器
driver.close();
} catch (Exception e) {
// TODO: handle exception
}
}

public static void main(String[] args) {
WebDriver driver = new ChromeDriver();
try {
test(driver);
} catch (Exception e) {
e.printStackTrace();
} finally {
driver.quit();
}
}
}