自动化(Automation)是指机器设备、系统或过程(生产、管理过程)在没有人或较少人的直接参与下,按照人的要求,经过自动检测、信息处理、分析判断、操纵控制,实现预期的目标的过程。
这是教科书里面的自动化的定义,回归到自动化测试其实自动化测试就是什么呢?
指的是测试的过程在没有人或者较少的人为的干预的情况下进行的测试,再简单点说就是用程序或者脚本来测试程序,那么在web自动化测试中主要用来把测试人员从繁琐的内容中解放出来,主要做一些比如需要多次输入,多次运行的,比如我们用边界值,等价类设计的很多测试数据需要执行,比如业务流程需要执行很多遍的时候我们就可以使用,web自动化测试
PS:这里有一套2022最新版的软件测试全套自学教程,包含了以下内容,记得一定要下载:
☑ 215集-零基础到精通全套视频课程
☑ [PPT+代码]-完整配套的教学课件
☑ 18套-测试实战项目源码
☑ 37套-测试工具软件包
☑ 268道-测试猿毕业学员真实面试题
☑ 500个-面试简历模板(信息完整)
>> 下载300G软测学习资料包【视频教程+PPT+课件+项目源码】
现在主流的web测试工具我们常用的就是selenium的那一套工具包括


Web自动化测试一般使用设计测试用例的方法跟功能测试相同,使用等价类划分,边界值,因果图,场景法等等就好了
Web自动化测试实施的使用我们一般会采用po模式设计
PO是page object的简称,
核心思想是通过对界面元素的封装减少冗余代码,同时在后期维护中,若元素定位发生变化, 只需要调整页面元素封装的代码,提高测试用例的可维护性、可读性。
PO模式可以把一个页面分为三层,对象库层、操作层、业务层。
对象库层:封装定位元素的方法。
操作层:封装对元素的操作。
业务层:将一个或多个操作组合起来完成一个业务功能。
比如登录:需要输入帐号、密码、点击登录三个操作。
测试脚本只需要调用业务层代码就可以完成
当出现页面需要的时候只需要测试代码可以完全不用修改只需要修改操作层就好了
web自动化测试流程和功能测试基本一致:
下面我们已登录需求为例
(1)需求分析


(2)设计测试用例与测试数据
| ID | 测试模块 | 标题 | 前置条件 | 测试输入 |
|---|---|---|---|---|
| login_001 | 登录 | 错误的手机号登录 | 网络正常,功能正常 | 手机号错误其他输入项正常 |
【下方为测试数据】
[
{
"username": "21888888888",
"pwd": "123456",
"code": "8888",
"ast_msg": "账号格式不匹配",
"desc": "用户名错误"
},
{
"username": "12888888888",
"pwd": "123456",
"code": "8888",
"ast_msg": "账号格式不匹配",
"desc": "用户名错误"
},
{
"username": "1088888888",
"pwd": "123456",
"code": "8888",
"ast_msg": "账号格式不匹配",
"desc": "用户名错误"
},
{
"username": "138888888889",
"pwd": "123456",
"code": "8888",
"ast_msg": "账号格式不匹配",
"desc": "用户名错误"
},
{
"username": "32888888888",
"pwd": "123456",
"code": "8888",
"ast_msg": "账号格式不匹配",
"desc": "用户名错误"
},
{
"username": " ",
"pwd": "123456",
"code": "8888",
"ast_msg": "用户名不能为空",
"desc": "用户名错误"
}
]
(3)搭建web自动化测试环境
(4)设计web自动化测试框架

(5)编写代码
1. from selenium.webdriver.common.by import By
2.
3. from utils import UtilsDriver
4. from base.page_base import BasePage
5.
6.
7. # 界面对象层
8.
9. class PageLogin(BasePage):
10.
11. # 账号元素
12. def find_username(self):
13. return self.driver.find_element_by_id("username")
14. # return self.driver.find_element(*self.username)
15. # return self.get_element(self.username)
16.
17. # 密码元素
18. def find_pwd(self):
19. return self.driver.find_element(By.ID,"password")
20.
21. # 验证码元素
22. def find_vcode(self):
23. return self.driver.find_element_by_id("verify_code")
24.
25. # 按钮开始登录元素
26. def find_login_btn(self):
27. # return self.driver.find_element_by_name("sbtbutton")
28. return self.driver.find_element(By.NAME,"sbtbutton")
29.
30. # 操作层
31. class HandleLogin(object):
32. def __init__(self):
33. self.page_login=PageLogin()
34.
35. def input_username(self,username):
36. self.page_login.find_username().send_keys(username)
37.
38. def input_pwd(self,pwd):
39. self.page_login.find_pwd().send_keys(pwd)
40.
41. def input_vcode(self,code):
42. self.page_login.find_vcode().send_keys(code)
43.
44. def click_login_btn(self):
45. self.page_login.find_login_btn().click()
46.
47. # 业务层
48. # 输入用户名密码验证码 点击登录
49. class LoginProxy(object):
50. def __init__(self):
51. self.handle_login = HandleLogin()
52.
53. def login(self,username,pwd,code):
54. self.handle_login.input_username(username)
55. self.handle_login.input_pwd(pwd)
56. self.handle_login.input_vcode(code)
57. self.handle_login.click_login_btn()
脚本执行代码 创建test_login
1. # from selenium import webdriver
2. from po.page_home import HomeProxy
3. from po.page_login import LoginProxy
4. from utils import UtilsDriver,get_data
5. import time
6. import pytest
7. import allure
8.
9. @allure.feature("登录功能")
10. class Test_login:
11. def setup_class(self):
12. self.login_p=LoginProxy()
13. self.home_p=HomeProxy()
14. def setup(self):
15.
16. # 进入首页
17. UtilsDriver.get_driver().get("http://127.0.0.1/")
18.
19. # 进入到login界面
20. self.home_p.go_login_page()
21. def teardown_class(self):
22. time.sleep(2)
23. UtilsDriver.quit_driver()
24.
25. @pytest.mark.parametrize(["username","pwd","code","asrt_msg"],get_data())
26. @allure.story("登录用户名错误")
27. def test_login_username_error(self,username,pwd,code,asrt_msg):
28. self.login_p.login(username,pwd,code)
29. time.sleep(1)
30. # 登录是否成功的预期结果
31. res = UtilsDriver.get_msg()
32. assert asrt_msg in res
(6)执行测试用例
[pytest]
addopts = -s --alluredir report
testpaths = ./script
python_files = test_*.py *test.py
python_classes = Test_*
python_functions = test_*
(7)生成测试报告
可以使用allure生成测试报告

很好奇,就使用rubyonrails自动化单元测试而言,你们正在做什么?您是否创建了一个脚本来在cron中运行rake作业并将结果邮寄给您?git中的预提交Hook?只是手动调用?我完全理解测试,但想知道在错误发生之前捕获错误的最佳实践是什么。让我们理所当然地认为测试本身是完美无缺的,并且可以正常工作。下一步是什么以确保他们在正确的时间将可能有害的结果传达给您? 最佳答案 不确定您到底想听什么,但是有几个级别的自动代码库控制:在处理某项功能时,您可以使用类似autotest的内容获得关于哪些有效,哪些无效的即时反馈。要确保您的提
我正在编写一个包含C扩展的gem。通常当我写一个gem时,我会遵循TDD的过程,我会写一个失败的规范,然后处理代码直到它通过,等等......在“ext/mygem/mygem.c”中我的C扩展和在gemspec的“扩展”中配置的有效extconf.rb,如何运行我的规范并仍然加载我的C扩展?当我更改C代码时,我需要采取哪些步骤来重新编译代码?这可能是个愚蠢的问题,但是从我的gem的开发源代码树中输入“bundleinstall”不会构建任何native扩展。当我手动运行rubyext/mygem/extconf.rb时,我确实得到了一个Makefile(在整个项目的根目录中),然后当
我有一个围绕一些对象的包装类,我想将这些对象用作散列中的键。包装对象和解包装对象应映射到相同的键。一个简单的例子是这样的:classAattr_reader:xdefinitialize(inner)@inner=innerenddefx;@inner.x;enddef==(other)@inner.x==other.xendenda=A.new(o)#oisjustanyobjectthatallowso.xb=A.new(o)h={a=>5}ph[a]#5ph[b]#nil,shouldbe5ph[o]#nil,shouldbe5我试过==、===、eq?并散列所有无济于事。
我有一些Ruby代码,如下所示:Something.createdo|x|x.foo=barend我想编写一个测试,它使用double代替block参数x,这样我就可以调用:x_double.should_receive(:foo).with("whatever").这可能吗? 最佳答案 specify'something'dox=doublex.should_receive(:foo=).with("whatever")Something.should_receive(:create).and_yield(x)#callthere
Sinatra新手;我正在运行一些rspec测试,但在日志中收到了一堆不需要的噪音。如何消除日志中过多的噪音?我仔细检查了环境是否设置为:test,这意味着记录器级别应设置为WARN而不是DEBUG。spec_helper:require"./app"require"sinatra"require"rspec"require"rack/test"require"database_cleaner"require"factory_girl"set:environment,:testFactoryGirl.definition_file_paths=%w{./factories./test/
我遵循MichaelHartl的“RubyonRails教程:学习Web开发”,并创建了检查用户名和电子邮件长度有效性的测试(名称最多50个字符,电子邮件最多255个字符)。test/helpers/application_helper_test.rb的内容是:require'test_helper'classApplicationHelperTest在运行bundleexecraketest时,所有测试都通过了,但我看到以下消息在最后被标记为错误:ERROR["test_full_title_helper",ApplicationHelperTest,1.820016791]test
我收到这个错误:RuntimeError(自动加载常量Apps时检测到循环依赖当我使用多线程时。下面是我的代码。为什么会这样?我尝试多线程的原因是因为我正在编写一个HTML抓取应用程序。对Nokogiri::HTML(open())的调用是一个同步阻塞调用,需要1秒才能返回,我有100,000多个页面要访问,所以我试图运行多个线程来解决这个问题。有更好的方法吗?classToolsController0)app.website=array.join(',')putsapp.websiteelseapp.website="NONE"endapp.saveapps=Apps.order("
我已经构建了一些serverspec代码来在多个主机上运行一组测试。问题是当任何测试失败时,测试会在当前主机停止。即使测试失败,我也希望它继续在所有主机上运行。Rakefile:namespace:specdotask:all=>hosts.map{|h|'spec:'+h.split('.')[0]}hosts.eachdo|host|begindesc"Runserverspecto#{host}"RSpec::Core::RakeTask.new(host)do|t|ENV['TARGET_HOST']=hostt.pattern="spec/cfengine3/*_spec.r
我在app/helpers/sessions_helper.rb中有一个帮助程序文件,其中包含一个方法my_preference,它返回当前登录用户的首选项。我想在集成测试中访问该方法。例如,这样我就可以在测试中使用getuser_path(my_preference)。在其他帖子中,我读到这可以通过在测试文件中包含requiresessions_helper来实现,但我仍然收到错误NameError:undefinedlocalvariableormethod'my_preference'.我做错了什么?require'test_helper'require'sessions_hel
所以我开始关注ruby,很多东西看起来不错,但我对隐式return语句很反感。我理解默认情况下让所有内容返回self或nil但不是语句的最后一个值。对我来说,它看起来非常脆弱(尤其是)如果你正在使用一个不打算返回某些东西的方法(尤其是一个改变状态/破坏性方法的函数!),其他人可能最终依赖于一个返回对方法的目的并不重要,并且有很大的改变机会。隐式返回有什么意义?有没有办法让事情变得更简单?总是有返回以防止隐含返回被认为是好的做法吗?我是不是太担心这个了?附言当人们想要从方法中返回特定的东西时,他们是否经常使用隐式返回,这不是让你组中的其他人更容易破坏彼此的代码吗?当然,记录一切并给出