前言
软件测试流程:需求分析—>测试计划—>测试设计—>测试执行—>测试报告
一、需求分析
“不爱听书”是一个为用户提供创作音乐和收听音频的平台。对于该项目的需求分析,提炼出相关测试点。
基本功能需求
用户可以进行注册、登录与退出账户——成功注册与登录与退出
注册登录之后,用户可以进行音频的录制、上传、以及查看音频列表——音频能成功上传、录制、展示
用户可以创建属于自己的专辑、关联自己喜欢的音频以及展示专辑列表——成功创建专辑,成功绑定音频,展示音频列表
易用性需求:符合常见标准与规范,用户操作更方便舒适
界面需求:界面控件正常使用,布局、排版合理
性能需求:多个用户同时登陆时运行速度正常、内存正常、系统稳定;
安全需求:用户密码是否加密显示,保障用户的私人信息不被窃取
兼容性需求:各种浏览器都能正常访问;用户数据在后台存储时互不影响
二、测试计划
测试目的 | 测试系统功能实现是否正常,是否符合用户需求和软件需求 |
测试前提 | 系统正常运行 |
测试范围 | 整个系统 |
测试方式 | 手工测试+自动化测试(编写自动化脚本 selenium) |
测试环境 | Windows10系统、chrome浏览器、Firefox浏览器 |
测试模块 | 用户模块(注册、登录、退出) |
音频模块(上传、音频列表、在线录制) | |
专辑模块(创建专辑;专辑列表) |
三、测试用例的设计
四、测试执行
以专辑列表页为例
1.手工测试
2.单元测试(Junit框架)
单元测试就是针对最小的功能单元编写测试代码,Java 程序最小的功能单元是方法,因此,单元测试就是针对Java方法的测试,进而检查方法的正确性,这里运用Junit框架进行测试,导入依赖后进行如下测试:
1)针对UserRepo类对用户注册的插入元素、用户登录有关的取出元素方法进行了测试
public class TestUserRepo {
public String username;
public String password;
UserRepo userRepo = new UserRepo();
@Test
public void testInsert(){
UserDO userNow = new UserDO(username, password);
userNow.setUsername("hhh");
userNow.setPassword("123");
userRepo.insert(userNow);
}
@Test
public void testselectOneByUsername() {
System.out.println(userRepo.selectOneByUsername("hhh"));
}
}
注册:插入一个新用户,由于“hhh”这个用户在数据库中已经存在,运行失败;
登录:查询用户信息(uid,uesername,password),运行通过。
2)针对专辑相关操作如展示专辑列表、创建专辑进行测试
public class TestAlbumRepo {
AlbumRepo albumRepo=new AlbumRepo();
@Test
public void testInsert() {
albumRepo.insert(10, "文章标题", "封面图", 2);
}
@Test
public void testSelectListByUid(){
System.out.println(albumRepo.selectListByUid(10));
}
}
结果:专辑插入与展示功能正常,运行通过
关于Junit框架
1)Junit特点
a.JUnit可以灵活的选择执行哪些测试方法,可以一键执行全部测试方法
b.JUnit可以生成全部方法的测试报告
c.单元测试中的某个方法测试失败了,不会影响其他测试方法的测试
d.在测试方法上使用@Test注解:标注该方法是一个测试方法
e.测试方法必须是public void,即公共、无返回数据
f.选中测试方法,选中 “JUnit 运行”,如果 测试良好则是绿色;如果测试失败,则是红色
2)Junit常用注解(Junit4.xxx版本)
@Test:在Junit3中,所有的测试类必须继承Junit的测试基类。在Junit4中,定义一个测试方法只需要在方法前加上@Test。
@Ignore: @Ignore修饰的方法会被忽略不执行同时不计入用例数。但要注意此标注的时候不能与其它标注一起使用,否则无效。不建议使用Ignore标注,因为容易忘记更新测试方法,造成用例遗漏等。
@BeforeClass:测试类里所有用例运行之前,运行一次这个标注修饰的方法(只会运行一次)。
@AfterClass:跟@BeforeClass对应,在测试类里所有用例运行之后运行一次该标注修饰的方法,用于处理一些测试后续工作,例如清理数据,恢复现场。
@Before:每个用例运行之前都运行一次该标注修饰的方法,适用于独立的用例间,运行次数取决于用例数。
@After:每个用例运行之后都运行一次该标注修饰的方法,适用于独立的用例间,运行次数取决于用例数。
3)自动化测试
1)注册的测试(test_1register)
2)登录是正常的登陆时的情况及专辑列表页面的展示的测试(test_loginBynormalAndList)
3)登录是异常登录时的情况及专辑列表页面的显示的测试(test_loginByAbnormalAndList)
from selenium import webdriver
import unittest
from selenium.webdriver.common.by import By
from ddt import ddt, unpack, data
import time # 导入需要的工具包
@ddt
class User(unittest.TestCase):
def setUp(self):
print("----setUp-----")
self.driver = webdriver.Chrome()
self.url = "http://127.0.0.1:8080/studio/"
self.driver.maximize_window()
time.sleep(6)
# 注册
@data(['山山', '123'], ['', ''])
# @unittest.skip("skipping")
@unpack
def test_1register(self, username, password):
driver = self.driver
url = self.url
driver.get(url)
time.sleep(1)
driver.find_element(By.LINK_TEXT, '注册').click()
time.sleep(1)
driver.find_element(By.ID, 'username').send_keys(username)
time.sleep(1)
driver.find_element(By.ID, 'password').send_keys(password)
time.sleep(1)
driver.find_element(By.ID, 'submit').click()
time.sleep(2)
driver.quit()
# 登录+展示专辑列表(成功登录时的测试)
@data(['山山', '123'], ['hhh', '123'])
# @unittest.skip("skipping")
@unpack
def test_loginBynormalAndList(self, username, password):
driver = self.driver
url = self.url
driver.get(url)
driver.implicitly_wait(10) # 智能等待
driver.find_element(By.LINK_TEXT, '登录').click()
time.sleep(6)
driver.find_element(By.ID, 'username').send_keys(username)
driver.find_element(By.ID, 'password').send_keys(password)
time.sleep(4)
driver.find_element(By.ID, "submit").click()
time.sleep(5)
driver.find_element(By.LINK_TEXT, '专辑列表').click()
time.sleep(6)
driver.quit()
# 登录+展示列表(异常登录时的测试)
@data(['山山', '13'], ['hhh', '1234'])
@unpack
def test_loginByAbnormalAndList(self, user, password):
driver = self.driver
url = self.url
driver.get(url)
driver.implicitly_wait(10) # 智能等待
driver.find_element(By.LINK_TEXT, '登录').click()
time.sleep(6)
driver.find_element(By.ID, 'username').send_keys(user)
driver.find_element(By.ID, 'password').send_keys(password)
time.sleep(4)
driver.find_element(By.ID, "submit").click()
time.sleep(5)
url = self.url
driver.get(url)
time.sleep(3)
driver.find_element(By.LINK_TEXT, '专辑列表').click()
time.sleep(6)
alert = driver.switch_to.alert # 定位弹出框操作句柄
time.sleep(3)
alert.accept() # 关闭弹出框
time.sleep(5)
driver.quit()
def tearDown(self):
print("----tearDown----")
self.driver.quit()
if __name__ == "__main__":
unittest.main()
五、测试报告
1.手工测试报告
见上方🔝手工测试框图
2.自动化测试用例报告(生成HTML测试报告)
import HTMLTestRunner
import os
import sys
import time
import unittest
def createsuite():
discovers = unittest.defaultTestLoader.discover("../testProjectListening", pattern="listeningTest.py", top_level_dir=None) # 一个类就是一个测试套
print(discovers)
return discovers
if __name__ == "__main__":
# 1.创建一个文件夹
curpath = sys.path[0] # 当前工具类的一个集合/路径集合数组,0指的是当前文件所在路径
print(sys.path)
print(sys.path[0])
# 2.当前路径下resultReport文件夹不存在时,创建
if not os.path.exists(curpath + '/resultreport'):
os.mkdir(curpath + '/resultreport')
# 解决重复命名问题
# 时间 时分秒 ——》名称绝对不会重复
now = time.strftime("%Y-%m-%d %H %M %S", time.localtime(time.time()))
print(now)
print(time.time())
print(time.localtime(time.time()))
# 3.准备HTML报告输出的文件
# 文件名是路径+文件名称+时间
filename = curpath + '/resultreport/' + now + 'resultreport.html'
# 打开HTML文件,wb以写的方式输入运行的那个结果
with open(filename, 'wb') as fp:
# 括号中的参数是HTML报告中的参数
runner = HTMLTestRunner.HTMLTestRunner(stream=fp, title=u"测试报告", description=u"用例执行情况",
verbosity=2) # verbosity:结果的详细程度
suite = createsuite()
runner.run(suite)
测试报告生成:
分析:error 是因为用户名与密码不对应,在数据库中不存在;pass 是通过测试用例,此用户名和密码在数据库中
最后感谢每一个认真阅读我文章的人,礼尚往来总是要有的,虽然不是什么很值钱的东西,如果你用得到的话可以直接拿走:
这些资料,对于【软件测试】的朋友来说应该是最全面最完整的备战仓库,这个仓库也陪伴上万个测试工程师们走过最艰难的路程,希望也能帮助到你!有需要的小伙伴可以点击下方小卡片领取