软件测试项目实战【不爱听书】测试全套教程以及源码

前言

软件测试流程:需求分析—>测试计划—>测试设计—>测试执行—>测试报告

一、需求分析

“不爱听书”是一个为用户提供创作音乐和收听音频的平台。对于该项目的需求分析,提炼出相关测试点。

基本功能需求

用户可以进行注册、登录与退出账户——成功注册与登录与退出

注册登录之后,用户可以进行音频的录制、上传、以及查看音频列表——音频能成功上传、录制、展示

用户可以创建属于自己的专辑、关联自己喜欢的音频以及展示专辑列表——成功创建专辑,成功绑定音频,展示音频列表

易用性需求:符合常见标准与规范,用户操作更方便舒适

界面需求:界面控件正常使用,布局、排版合理

性能需求:多个用户同时登陆时运行速度正常、内存正常、系统稳定;

安全需求:用户密码是否加密显示,保障用户的私人信息不被窃取

兼容性需求:各种浏览器都能正常访问;用户数据在后台存储时互不影响

二、测试计划

测试目的 测试系统功能实现是否正常,是否符合用户需求和软件需求
测试前提 系统正常运行
测试范围 整个系统
测试方式 手工测试+自动化测试(编写自动化脚本 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 是通过测试用例,此用户名和密码在数据库中

如果你觉得自己学习效率低,缺乏正确的指导,可以加入资源丰富、学习氛围浓厚的技术社群 点我进测试行业圈,里面有我们收集的配套教程和相关技术文档提供和测试媛/猿一起做一个有温度的软件测试工程师,不让测试人孤寂的成长!

posted @ 2022-09-26 16:47  乐却思蜀  阅读(788)  评论(0编辑  收藏  举报