使用React,Redux和Router进行真正的集成测试

by Marcelo Lotif

通过马塞洛·洛蒂夫(Marcelo Lotif)

使用React,Redux和Router进行真正的集成测试 (Real integration tests with React, Redux and Router)

After being bitten a couple of times by bad refactoring and a broken app — even with all my tests green — I started to research about integration tests in React. Possibly also with Redux and React Router.

在经历了糟糕的重构和坏掉的应用程序几次咬伤之后,即使我的所有测试都是绿色的,我也开始研究React中的集成测试。 可能还可以使用Redux和React Router。

To my absolute shock, I couldn’t find any good material out there. The ones I found either were doing incomplete integration tests or simply doing it the wrong way.

令我震惊的是,我找不到任何好的材料。 我发现这要么是在进行不完整的集成测试,要么就是以错误的方式进行。

So here we’re going to build an integration test that initializes a React component, fires a simulated user interaction and assert that our component changes the way we expect it to change.

因此,这里我们将构建一个集成测试,该测试将初始化一个React组件,触发一个模拟的用户交互,并断言我们的组件改变了我们期望它改变的方式。

What this is not about: unit testing. I’m not going to dive into this right now, but there is a very good reason we at Wave (we’re hiring, by the way!) are slowing down on our unit tests and switching to integration tests. Scroll to the bottom if you’re interested in that.

不是关于:单元测试。 我现在不打算深入探讨这个问题,但是有一个很好的理由说明我们Wave的速度 ( 我们正在招聘 ,顺便说一句!)正在减慢我们的单元测试并切换到集成测试。 如果对此感兴趣,请滚动到底部。

Disclosure: I wouldn’t have had those tests working as smoothly as they are now if it wasn’t for the great front end folks at Wave, especially the amazing Tommy Li who figured out how to connect the router, so thank you!

披露:如果不是Wave的优秀前端人员,尤其是那些想出如何连接路由器的惊人的Tommy Li ,那我将没有那些测试能够像现在这样平稳地工作,所以谢谢!

配置 (Setting up)

For this project, we are going to use React, Redux, React/Redux Router (optional) and Thunk (optional) to run the app, Jest and Enzyme for testing.

对于这个项目,我们将使用React , Redux , React / Redux Router (可选)和Thunk (可选)来运行应用程序, Jest和Enzyme进行测试。

I’ll skip the setup of all those since there are many great tutorials out there about that.

我将跳过所有这些设置,因为那里有很多很棒的教程。

To set up the basics of my integration test, I’m gonna cheat a little bit and create an util function with some boilerplate code:

为了建立集成测试的基础知识,我将作弊一点,并使用一些样板代码创建util函数:

测试中 (Testing)

In your test file, you will first need to import some dependencies, your reducer and your component:

在测试文件中,首先需要导入一些依赖项,reducer和组件:

Then, on the beforeEach function, set up your integration test variables using that util function:

然后,在beforeEach函数上,使用该util函数设置集成测试变量:

(If you don’t use React Router or Thunk, you can just remove their references here and on the util function and it’s going to work the same way.)

(如果您不使用React Router或Thunk,则可以在此处和util函数上删除它们的引用,并且它们将以相同的方式工作。)

Now you’re all set to mount your component and test it. Let’s imagine this component renders a div, which displays a text coming from the reducer. When clicking on it, the text should change to another string, let’s say ‘new text’. To test that interaction, you can simply do:

现在您已经准备好安装组件并对其进行测试。 让我们想象一下这个组件渲染了一个div ,它显示了来自reducer的文本。 单击它时,文本应更改为另一个字符串,例如“新文本”。 要测试这种交互,您可以简单地执行以下操作:

That’s it ☺ With this very simple code you’re testing the div calling an action producer on click, that dispatches an action to the reducer, that changes the data, triggering a re-render on the component, that is expected to change the way you want it to change. If any of those steps fail, your test goes red and you know that functionality of your app is not working.

就是这样☺通过这个非常简单的代码,您正在测试div并在单击时调用一个动作生成器,该动作生成器将动作分派给reducer,该动作将更改数据,触发组件的重新渲染,从而有望改变方式你想改变它。 如果这些步骤中的任何一个失败,则测试会变成红色,并且您知道应用程序的功能无法正常工作。

You can try to go deeper in this chain and assert some other things:

您可以尝试深入了解此链并断言其他一些事情:

测试API调用 (Testing API calls)

In the real world you’ll probably need to call some APIs to fetch data for your app, and that is the part you need to mock in order to test things effectively. We’ll use Jest here, which is not the best way to mock http requests, but I’ll do it for the convenience.

在现实世界中,您可能需要调用一些API来获取应用程序的数据,而这是您为了进行有效测试而需要模拟的部分。 我们将在这里使用Jest,这不是模拟HTTP请求的最佳方法,但是为了方便起见,我将使用它。

Assuming you use a hypothetical http client to call an endpoint through its get function when you click on the div, then set the return of this call into the reducer that gets displayed back in the div:

假设您在单击div时使用假设的http客户端通过其get函数调用端点,然后将此调用的返回值设置到在div中显示的reducer中:

In an even more real world application, that get function will return you a Promise object. Things become a little complicated from here because the simulated click function is unaware of that promise and there is no way of executing its then function. The reference to the object has been lost.

在更真实的应用程序中,该get函数将返回一个Promise对象。 从这里开始,事情变得有些复杂,因为模拟的click函数没有意识到那个承诺,并且无法执行then函数。 对对象的引用已丢失。

We will need to somehow wait for that promise to resolve before executing the assertions. We work around this by doing a little hack in an util function:

在执行断言之前,我们将需要以某种方式等待该承诺解决。 我们通过在util函数中进行一些修改来解决此问题:

And our test is now going to look like this:

现在,我们的测试将如下所示:

With the async … await statement , available since ES7, our test is going to wait until all promises have been resolved so it can make its assertions. Jest currently has no solution for this, but this hack works pretty well in real life.

自从ES7开始使用async…await语句,我们的测试将等待直到所有promise都已解决,以便可以进行声明。 Jest目前还没有解决方案,但是此hack在现实生活中效果很好。

If you have more complicated action producers with other promises being called in the resolve or reject of that first promise, I suggest you unit test those calls and also test the final results of all cases in integration tests.

如果您有更复杂的动作生产者,而在第一个承诺的解决拒绝中调用了其他承诺,则建议您对这些调用进行单元测试,并在集成测试中测试所有案例的最终结果。

更多测试 (More Testing)

In case you need to set an initial state to your component , you can dispatch actions manually until you reach the desired state:

如果需要为组件设置初始状态,则可以手动分派操作,直到达到所需状态为止:

store.dispatch({ payload: 'data', type: 'SOME_ACTION' });

You can also go crazy on those assertions and test every little thing, or keep it simple knowing the test coverage is going to be the same as if you have added unit tests on each of the layers of this app, but with a lot less code. In addition, you are also testing how those layers connect with each other and how your app responds to user input and data store changes.

您也可以为这些断言而疯狂,测试每件事,或者保持简单,因为知道测试覆盖范围与在此应用程序的每个层上都添加了单元测试一样,但是代码却少得多。 此外,您还将测试这些层如何相互连接以及您的应用如何响应用户输入和数据存储更改。

Please leave your opinion in the comments section, there is a lot of improvements to be made here and I’m happy to modify this according to your suggestions. Thanks!

请在评论部分中留下您的意见,这里有很多改进之处,我很乐意根据您的建议进行修改。 谢谢!

NO没有单元测试?!? (Y U NO UNIT TEST?!?)

We at Wave (did I mention we’re hiring?) have done a ton of front end unit tests before and, to be honest, the majority of them have been somewhat useless. Sure, they are at the core of TDD, but some reducers and action producers unit tests are just boilerplate code and don’t add much value to the code or the TDD process.

Wave之前 ( 我们是否提到过要聘用我们吗?)之前, 我们已经进行了大量的前端单元测试,老实说,大多数测试都没有用。 当然,它们是TDD的核心,但是某些化简工具和动作生产者单元测试只是样板代码,不会为代码或TDD流程增加太多价值。

You can actually do really good TDD with integration tests only, and they are going to be useful in the future to spot broken links between your app layers and ultimately to check if your app is behaving as expected, which is what automated tests are for.

实际上,您只能使用集成测试来做真正好的TDD,并且它们在将来将很有用,可以发现您的应用程序层之间断开的链接,并最终检查您的应用程序的行为是否符合预期,这就是自动化测试的目的。

Don’t get me wrong, we still unit test edge cases that are too complicated or annoying to reproduce on integration tests, but the majority of our unit tests became useless as soon as we added integration tests like the above. In the end, it means the time we now spend thinking about, developing and fixing tests is a lot lower than it was before and they are much more effective in spotting problems in the app. So, win win ☺

别误会,我们仍然对过于复杂或烦人的边缘测试进行单元测试,以至于无法在集成测试中重现,但是一旦添加了上述集成测试,我们的大多数单元测试就变得毫无用处。 最后,这意味着我们现在花在思考,开发和修复测试上的时间比以前少了很多,并且它们在发现应用程序中的问题上更加有效。 所以,双赢☺

One problem you might find is with deep mounting, instead of shallow rendering. You might think some component trees are too complicated to mount, but I’ll say another advantage of mounting the root component is to test if the child components are being instantiated correctly. If you have connected child components, you can test them separately if you prefer. I haven’t tried shallow rendering a connected component to see if this integration test setup still works, but you can try. If you don’t like to mount and don’t have connected child components, another possibility I haven’t explored is shallow render and then manually connecting them. The important thing here is to feel comfortable with the amount and the quality of the tests you’re writing, making sure they actually help in automatically doing some regression testing and discovering hidden issues for you.

您可能会发现的一个问题是深层安装而不是浅层渲染。 您可能会认为某些组件树太复杂而无法挂载,但是我会说挂载根组件的另一个好处是测试子组件是否被正确实例化。 如果已连接子组件,则可以根据需要单独测试它们。 我没有尝试浅化呈现连接的组件以查看此集成测试设置是否仍然有效,但是您可以尝试。 如果您不喜欢挂载并且没有连接子组件,那么我还没有探讨的另一种可能性是浅渲染,然后手动连接它们。 这里重要的是让您对正在编写的测试的数量和质量感到满意,确保它们实际上有助于自动执行一些回归测试并为您发现隐藏的问题。

翻译自: https://www.freecodecamp.org/news/real-integration-tests-with-react-redux-and-react-router-417125212638/

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/395406.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Go语言从入门到精通 - 数据类型转换

本节核心内容 介绍 Go语言数据类型转换的格式介绍 数据转换代码示例介绍 数据转换过程中的注意事项 本小节视频教程和代码:百度网盘 可先下载视频和源码到本地,边看视频边结合源码理解后续内容,边学边练。 Go语言数据类型转换 Go 语言使用类型…

JNI通过线程c回调java层的函数

1、参看博客:http://www.jianshu.com/p/e576c7e1c403 Android JNI 篇 - JNI回调的三种方法(精华篇) 2、参看博客: JNI层线程回调Java函数关键点及示例 http://blog.csdn.net/fu_shuwu/article/details/41121741 3 http://blog.cs…

signature=f7a4b29b93ef2b36608792fdef7f454a,Embedding of image authentication signatures

摘要:A method (), an apparatus, a computer readable medium and use of said method for authenticating an audio-visual signal (), such as a digital image or video, are disclosed. A signature is derived from all image regions, including areas with …

glob

主要是用来在匹配文件,相当shell中用通配符匹配. 用法: glob.glob(pathname) # 返回匹配的文件作为一个列表返回 glob.iglob(pathname) # 匹配到的文件名,返回一个迭代器 ps: pathname是路径, 可以是绝对和相对路径 匹配当前目录下有一个数字开头…

构建微服务:Spring boot 入门篇

Spring官方网站本身使用Spring框架开发,随着功能以及业务逻辑的日益复杂,应用伴随着大量的XML配置文件以及复杂的Bean依赖关系。随着Spring 3.0的发布,Spring IO团队逐渐开始摆脱XML配置文件,并且在开发过程中大量使用“约定优先配…

img 加载 svg占位符_如何使用SVG作为占位符以及其他图像加载技术

img 加载 svg占位符by Jos M. Prez由JosM.Prez 如何使用SVG作为占位符以及其他图像加载技术 (How to use SVG as a Placeholder, and Other Image Loading Techniques) I’m passionate about image performance optimisation and making images load fast on the web. One of…

hibernate 注解

参考链接地址:https://blog.csdn.net/wx5040257/article/details/78697119 主键生成策略:https://www.cnblogs.com/ph123/p/5692194.html 注解转载于:https://www.cnblogs.com/wangxuekui/p/10287647.html

iOS - UIScrollView

前言 NS_CLASS_AVAILABLE_IOS(2_0) interface UIScrollView : UIView <NSCoding>available(iOS 2.0, *) public class UIScrollView : UIView, NSCoding 移动设备的屏幕大小是极其有限的&#xff0c;因此直接展示在用户眼前的内容也相当有限。当展示的内容较多&…

机器学习的展望

现阶段越来越多的投入到机器学习的热潮中来&#xff0c;有的人很是兴奋&#xff0c;认为这是一场新和革命&#xff0c;一场终极人工智能来临的前夜。也有人表示悲观&#xff0c;认为不仅机器学习不代表终极人工智能&#xff0c; 也还非常不成熟。 大量的新生代投入到这个领域&a…

BZOJ3453 XLkxc(拉格朗日插值)

显然f(i)是一个k2项式&#xff0c;g(x)是f(i)的前缀和&#xff0c;则显然其是k3项式&#xff0c;插值即可。最后要求的东西大胆猜想是个k4项式继续插值就做完了。注意2p>maxint…… #include<iostream> #include<cstdio> #include<cmath> #include<cs…

电邮地址_利用这些简单的技巧来充分利用电子邮件的强大功能

电邮地址Let’s talk about some email features that are surprisingly under-used, and that can really benefit you — if you know how to use them. This article is suitable for both users and developers who want to become email Jedi.让我们讨论一些电子邮件功能&…

inputstream重新赋值之前需要close吗_变量提升真的搞懂了吗?打脸的一道题

变量提升真的搞懂了吗&#xff1f;打脸的一道题我们知道JS代码在执行之前&#xff0c;会做一系列的事情&#xff0c;其中就包括变量提升&#xff0c;原本以为把变量提升搞懂的我&#xff08;因为这两天一直在研究变量提升&#xff0c;自我感觉已经很良好了&#xff0c;哈哈哈&a…

html5语义化 兼容,HTML5语义化标签,兼容性问题

HTML5不仅仅作为HTML标记语言的一个最新版本&#xff0c;更重要的是它制定了web应用开发的一系列标准&#xff0c;成为第一个将web做为应用开发平台的HTML语言。HTML5定义了一系列的新元素&#xff0c;如新语义化标签&#xff0c;智能表单&#xff0c;多媒体标签等&#xff0c;…

Swift之 vm10虚拟机安装Mac OS X10.10教程

VM10装Mac OS X 10.9.3及更新到Mac OS X 10.10,让你的windows也能玩Swift 。 近期WWDC放出终极大招——新的编程语言Swift(雨燕),导致一大波程序猿的围观和跃跃欲试。当然了,工欲善其事,必先利其器,所以对于那些没有Mac又想要尝鲜的小伙伴肯定非常为难。可是&#xff0c;请放…

如何使用json开发web_如何通过使用JSON Web令牌简化应用程序的身份验证

如何使用json开发webby Sudheesh Shetty由Sudheesh Shetty 如何通过使用JSON Web令牌简化应用程序的身份验证 (How to simplify your app’s authentication by using JSON Web Token) Every application we come across today implements security measures so that the user…

c++ 实现录音并且指定到文件_通话自动录音,留下美好回忆,记录完整录音证据...

手机通话&#xff0c;如果自动录音多好&#xff0c;许多人与我一样抱有这个想法。记得华为Android版本5.0时代&#xff0c;手机没有自动录音功能&#xff0c;我一直到网上下载自动通话录音软件&#xff0c;有时甚至是下载ROOT版的带自动通话功能的EMUI版本进行刷机安装。那个时…

2639-Bone Collector II (01背包之第k优解)

题目链接&#xff1a; http://acm.hdu.edu.cn/showproblem.php?pid2639 求第k优解的关键代码&#xff1a; 用两个数组记录两种状态&#xff08;选择或不选择&#xff09;&#xff0c;并且只要记录前k次。在这两个数组中都是前k次可能的最优解。所以我们只要把这两个数组做比较…

html自动按键,VBS脚本和HTML DOM自动操作网页

本来是想通过JS实现对其他页面的控制&#xff0c;发现跨域无法获取页面DOM来操作。接着考虑bat&#xff0c;发现也实现不了&#xff0c;于是想到vbs。vbs还是很强大啊&#xff0c;病毒之类很多都是vbs脚本啊。vbs打开浏览器&#xff0c;然后通过dom来操作页面&#xff0c;可以实…

opencv在同一窗口打印多张图片

首先&#xff0c;由于cv2处理的图片是通过ndarray的格式操作的&#xff0c;也就是说通过array的拼接就可以实现图片的拼接&#xff0c;那么之后就可以通过简单的imshow将合并的图片打印从而达到在一个窗口中显示多张图片的目的。 import cv2 import numpy as npimg1 cv2.imrea…

dj打碟怎么学_学DJ打碟 - Rane声卡连接

上一篇内容中&#xff0c;老师讲过在学DJ打碟的时候&#xff0c;是离不开对软件方面的操作&#xff0c;其实每一个学习过程&#xff0c;当你学会之后&#xff0c;在“回头看”的时候&#xff0c;都会觉得&#xff1a;原来学DJ打碟这么简单啊&#xff0c;这就是已经学习过的人会…