Here's my lazy ponent:
const LazyBones = React.lazy(() => import('@graveyard/Bones')
.then(module => ({default: module.BonesComponent}))
export default LazyBones
I'm importing it like this:
import Bones from './LazyBones'
export default () => (
<Suspense fallback={<p>Loading bones</p>}>
<Bones />
</Suspense>
)
And in my test I have this kind of thing:
import * as LazyBones from './LazyBones';
describe('<BoneYard />', function() {
let Bones;
let wrapper;
beforeEach(function() {
Bones = sinon.stub(LazyBones, 'default');
Bones.returns(() => (<div />));
wrapper = shallow(<BoneYard />);
});
afterEach(function() {
Bones.restore();
});
it('renders bones', function() {
console.log(wrapper)
expect(wrapper.exists(Bones)).to.equal(true);
})
})
What I expect is for the test to pass, and the console.log to print out:
<Suspense fallback={{...}}>
<Bones />
</Suspense>
But instead of <Bones />
I get <lazy />
and it fails the test.
How can I mock out the imported Lazy React ponent, so that my simplistic test passes?
Here's my lazy ponent:
const LazyBones = React.lazy(() => import('@graveyard/Bones')
.then(module => ({default: module.BonesComponent}))
export default LazyBones
I'm importing it like this:
import Bones from './LazyBones'
export default () => (
<Suspense fallback={<p>Loading bones</p>}>
<Bones />
</Suspense>
)
And in my test I have this kind of thing:
import * as LazyBones from './LazyBones';
describe('<BoneYard />', function() {
let Bones;
let wrapper;
beforeEach(function() {
Bones = sinon.stub(LazyBones, 'default');
Bones.returns(() => (<div />));
wrapper = shallow(<BoneYard />);
});
afterEach(function() {
Bones.restore();
});
it('renders bones', function() {
console.log(wrapper)
expect(wrapper.exists(Bones)).to.equal(true);
})
})
What I expect is for the test to pass, and the console.log to print out:
<Suspense fallback={{...}}>
<Bones />
</Suspense>
But instead of <Bones />
I get <lazy />
and it fails the test.
How can I mock out the imported Lazy React ponent, so that my simplistic test passes?
Share Improve this question edited Sep 11, 2019 at 14:01 AncientSwordRage asked Sep 11, 2019 at 13:24 AncientSwordRageAncientSwordRage 7,67421 gold badges100 silver badges193 bronze badges 04 Answers
Reset to default 7 +100I'm not sure this is the answer you're looking for, but it sounds like part of the problem is shallow
. According to this thread, shallow
won't work with React.lazy
.
However, mount
also doesn't work when trying to stub a lazy ponent - if you debug the DOM output (with console.log(wrapper.debug())
) you can see that Bones
is in the DOM, but it's the real (non-stubbed-out) version.
The good news: if you're only trying to check that Bones
exists, you don't have to mock out the ponent at all! This test passes:
import { Bones } from "./Bones";
import BoneYard from "./app";
describe("<BoneYard />", function() {
it("renders bones", function() {
const wrapper = mount(<BoneYard />);
console.log(wrapper.debug());
expect(wrapper.exists(Bones)).to.equal(true);
wrapper.unmount();
});
});
If you do need to mock the ponent for a different reason, jest
will let you do that, but it sounds like you're trying to avoid jest
. This thread discusses some other options in the context of jest
(e.g.
mocking Suspense
and lazy
) which may also work with sinon
.
You don't need to resolve lazy()
function by using .then(x => x.default)
React already does that for you.
React.lazy takes a function that must call a dynamic import(). This must return a Promise which resolves to a module with a default export containing a React ponent. React code splitting
Syntax should look something like:
const LazyBones = React.lazy(() => import("./LazyBones"))
Example:
// LazyComponent.js
import React from 'react'
export default () => (
<div>
<h1>I'm Lazy</h1>
<p>This ponent is Lazy</p>
</div>
)
// App.js
import React, { lazy, Suspense } from 'react'
// This will import && resolve LazyComponent.js that located in same path
const LazyComponent = lazy(() => import('./LazyComponent'))
// The lazy ponent should be rendered inside a Suspense ponent
function App() {
return (
<div className="App">
<Suspense fallback={<p>Loading...</p>}>
<LazyComponent />
</Suspense>
</div>
)
}
As for Testing, you can follow the React testing example that shipped by default within create-react-app
and change it a little bit.
Create a new file called LazyComponent.test.js
and add:
// LazyComponent.test.js
import React, { lazy, Suspense } from 'react'
import { render, screen } from '@testing-library/react'
const LazyComponent = lazy(() => import('./LazyComponent'))
test('renders lazy ponent', async () => {
// Will render the lazy ponent
render(
<Suspense fallback={<p>Loading...</p>}>
<LazyComponent />
</Suspense>
)
// Match text inside it
const textToMatch = await screen.findByText(/I'm Lazy/i)
expect(textToMatch).toBeInTheDocument()
})
Live Example: Click on the Tests Tab just next to Browser tab. if it doesn't work, just reload the page.
You can find more react-testing-library plex examples at their Docs website.
I needed to test my lazy ponent using Enzyme. Following approach worked for me to test on ponent loading pletion:
const myComponent = React.lazy(() =>
import('@material-ui/icons')
.then(module => ({
default: module.KeyboardArrowRight
})
)
);
Test Code ->
//mock actual ponent inside suspense
jest.mock("@material-ui/icons", () => {
return {
KeyboardArrowRight: () => "KeyboardArrowRight",
}
});
const lazyComponent = mount(<Suspense fallback={<div>Loading...</div>}>
{<myComponent>}
</Suspense>);
const ponentToTestLoaded = await ponentToTest.type._result; // to get actual ponent in suspense
expect(ponentToTestLoaded.text())`.toEqual("KeyboardArrowRight");
This is hacky but working well for Enzyme library.
To mock you lazy ponent first think is to transform the test to asynchronous and wait till ponent exist like:
import CustomComponent, { Bones } from './Components';
it('renders bones', async () => {
const wrapper = mount(<Suspense fallback={<p>Loading...</p>}>
<CustomComponent />
</Suspense>
await Bones;
expect(wrapper.exists(Bones)).toBeTruthy();
}