如何测试 React 路由 ?

sxkk20082年前知识分享209

前言

本文承接上文 如何测试 React 异步组件?,这次我将继续使用 @testing-library/react 来测试我们的 React 应用,并简要简要说明如何测试路由系统。

基本示例

以下代码使用 react-router V6 版本, V5 使用 Switch 包裹组件

通常我们的程序会写下如下代码:

首先我们有 2 个页面

  • src/routes/home.jsx 主页
export default function Home() {
  return (
    <main style={{ padding: '1rem 0' }}>
      <h2>这是主页h2>
    main>
  )
}
  • src/routes/about.jsx 关于页面
export default function Home() {
  return (
    <main style={{ padding: '1rem 0' }}>
      <h2>这是关于页h2>
    main>
  )
}

然后使用 HashRouter 或者 BrowserRouter 包裹,形成我们的程序的主入口index.jsx

  • src/index.jsx 程序入口
import { HashRouter, Routes, Route, Link } from 'react-router-dom'
import Home from './routes/home'
import About from './routes/about'

const NoMatch = () => <div>No Founddiv>

function App() {
  return (
    <HashRouter>
      <div>
        <Link to="/">主页Link>
        <Link to="/about">关于Link>
      div>
      <Routes>
        <Route path="/" element={<Home />} />
        <Route path="/about" element={<About />} />
        <Route path="*" element={<NoMatch />} />
      Routes>
    HashRouter>
  )
}

export default App

因为我们的页面足够简单,所以页面不会报错,那如果当页面变量的复杂,页面下的其中一个组件报错,就会引起白屏

例如 现在在 about 页面下添加一个错误组件

import React from 'react'

function AboutContent() {
  throw new Error('抛出一个测试错误')
}

export default function About() {
  return (
    <main style={{ padding: '1rem 0' }}>
      <h2>这是关于页h2>
      <AboutContent />
    main>
  )
}

此时页面就会报错,但如果我们没有点击 about 页面,我们的程序任然正常运行,所以我们需要对路由进行测试。

测试方法

我们知道 @testing-library/react 是运行在 node 环境中的,但浏览器中并没有 HashRouter 或者 BrowserRouter ,所以我们需要一个用到 MemoryRouter

此时我们需要将原先的 index.jsx 拆分到 app.jsx

  • src/index.jsx 入口
import React from 'react'
import ReactDOM from 'react-dom'
import App from './App'
import { HashRouter } from 'react-router-dom'

ReactDOM.render(
  <React.StrictMode>
    <HashRouter>
      <App />
    HashRouter>
  React.StrictMode>,
  document.getElementById('root')
)
  • src/app.jsx
import { HashRouter, Routes, Route, Link } from 'react-router-dom'
import Home from './routes/home'
import About from './routes/about'

const NoMatch = () => <div>No Founddiv>

function App() {
  return (
    <HashRouter>
      <div>
        <Link to="/">主页Link>
        <Link to="/about">关于Link>
      div>
      <Routes>
        <Route path="/" element={<Home />} />
        <Route path="/about" element={<About />} />
        <Route path="*" element={<NoMatch />} />
      Routes>
    HashRouter>
  )
}

export default App

此时我们可以添加单元测试

import { render, screen } from '@testing-library/react'
import userEvent from '@testing-library/user-event'
import { createMemoryHistory } from 'history'
import React from 'react'
import { Router } from 'react-router-dom'
import App from './App'

test('测试整个路由系统', () => {
  render(
    <MemoryRouter>
      <App />
    MemoryRouter>
  )
  expect(screen.getByText(/这是主页/i)).toBeInTheDocument()

  userEvent.click(screen.getByText(/关于/i))
  expect(screen.getByText(/这是关于页/i)).toBeInTheDocument()
})

MemoryRouter 有 2 个参数

  • 第一个参数 initialEntries={["/users/mjackson"]} 配置初始化路由
  • 第二个参数 initialIndex 默认是 initialEntries 中的最后一个值

测试 404 页面

test('测试路由未匹配', () => {
  render(
    <MemoryRouter initialEntries={['/some/bad/route']}>
      <App />
    MemoryRouter>
  )

  expect(screen.getByText(/Not Found/i)).toBeInTheDocument()
})

通用测试

当页面有多个的时候,我们可以添加一个通用组件来确保每个页面都没有错误

import { useLocation } from 'react-router-dom'

export const LocationDisplay = ({ children }) => {
  const location = useLocation()

  return (
    <>
      <div data-testid="location-display">{location.pathname}div>
      {children}
    >
  )
}

将 url 显示在页面上, 通过遍历确保每个页面可以正确渲染。

let routes = ['/', '/about']

routes.forEach((route) => {
  test(`确保${route} 的 url 可以正确显示`, () => {
    render(
      <MemoryRouter initialEntries={[route]}>
        <LocationDisplay>
          <App />
        LocationDisplay>
      MemoryRouter>
    )

    expect(screen.getByText(new RegExp(route))).toBeInTheDocument()
  })
})

小结

以下是路由测试的步骤:

  1. 将程序和使用什么路由分开;
  2. 使用 MemoryRouter 来测试;
  3. 通过 userEvent.click 点击确保页面可以正确渲染;
  4. 提供一个公共包裹组件,通过遍历来测试每个页面,确保渲染

以上就是本文的全部内容,那么如何测试 react hooks ?请关注我,我会尽快出 React test 系列的下文。

希望这篇文章对大家有所帮助,也可以参考我往期的文章或者在评论区交流你的想法和心得,欢迎一起探索前端。

相关文章

在 2021 年 star 排行榜单中,tauri 一年增长了 18k 排名第五,我们就很好奇,Tauri 有什么优势呢?

然后我分别用 tauri 和 Electron.JS 打包测试一个 hello world 程序,一起来看下它们的大小。

大小对比

  • Electron.JS 62.5mb
  • Tauri 4.32mb

Tauri 构建的桌面程序太小了,远不是 Electron.JS 可以相比的,因为它放弃了体积巨大的  Chromium 内核   和  nodejs,前端使用操作系统的  webview,后端集成了  Rust。 Tauri 提供了初始化程序的模板,比如原生 js, react, svelte.js, vue.js 等等。

image.png

初步尝试 tauri,并且与 electron.js 对比

什么是 Tauri?Tauri 是一个为所有主流桌面平台构建小型、快速二进制文件的框架。开发人员可以集成任何编译成 HTML、 JS 和 CSS 的前端框架来构建他们的用户界面。应用程序的后端是一个...

AI绘画秀:人工智能开启艺术的未来

AI绘画秀:人工智能开启艺术的未来

  随着人工智能技术的不断发展,其应用领域也越来越广泛。在艺术领域中,人工智能也逐渐开始发挥出其独特的作用。AI绘画秀就是其中很引人注目的一种应用,它将人工智能与美术相结合,...

将 React 应用迁移至 Vite

将 React 应用迁移至 Vite

前言当下,在项目开发的过程中,对于大多数人来说,会使用 create react app 来创建 react 应用,它开箱即用,零配置,但随着项目中代码量的增加,你的项目构建时间也会随之增加,开发服务...

如何升级到 React 18发布候选版

目录安装客户端渲染 API 的更新服务端渲染 API 的更新自动批处理 (Automatic Batching)用于第三方库的 API更新严格模式 (Strict Mode)配置你的测试环境不再支持...

AI合成人脸:数字时代的美丽幻象

AI合成人脸:数字时代的美丽幻象

  随着人工智能技术的发展,AI合成人脸成为当今数字时代的一大热门话题。这项技术通过深度学习算法和大规模数据集训练,能够生成逼真的人脸图像,引发了广泛的讨论与争议。  1....

创建 React 应用的 7 种方式,你用过几种?

一:Create-React-App首先,我们第一个是想到的是 react 官方团队提供的脚手架工具 Create React App(cra)npx create-react-app my-app...

发表评论    

◎欢迎参与讨论,请在这里发表您的看法、交流您的观点。