-
-
Notifications
You must be signed in to change notification settings - Fork 183
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
MSW loading in react native is not happening #389
Comments
Hi, @turjoy-real. Not sure I fully understand the issue, could you describe it a bit more for me? The reason If we call |
Yes it starts rendering the app with default APIs. After MSW loads, it then uses MSW mocks. MSW is supposed to load before AppRegistry. But react native doesn't wait for the promise to resolve loading MSW and throws error saying there's no AppRegistry. |
Yeah, this is the culprit. RN expects you to call Do you know if we can defer the promise await to the App.registerComponent(async () => {
await enableMocking()
return App
}) I suppose not but I have zero clue about how RN behaves. |
No this is not working too. React Native doesn't expect a async callback function. This is the new error: Error: Element type is invalid: expected a string (for built-in components) or a class/function (for composite components) but got: object. This is happening even if I remove the await step. It's taking the unresolved promise object. Used this: AppRegistry.registerComponent(appName, async () => { Instead of: AppRegistry.registerComponent(appName, () => App); |
This error suggests the second argument is expected to be a React component. This means we can create a wrapper component and delegate the promise resolution to the state. App.registerComponent(appName, () => {
const [isMockingEnabled, setMockingEnabled] = React.useState(false)
React.useEffect(() => {
enableMocking().then(() => setMockingEnabled(true))
}, [])
if (!isMockingEnabled) return null
return <App />
}) Can you please give this a try, @turjoy-real? Thanks. |
Hi @kettanaito, Thanks for the idea. Successfully made MSW APIs the default for all E2E testing API calls. The changes had to be made in the App.tsx(or App.js) to be able to use hooks. function App() {
const [loading, setLoading] = React.useState(true);
React.useEffect(() => {
enableMocking().then(() => setLoading(false));
}, []);
return (
!loading && <Component/>
);
} And for the dynamic imports of msw.polyfills and mock server in typescript, updated tsconfig.json to use 'ES2022': {
"extends": "@react-native/typescript-config/tsconfig.json",
"compilerOptions": {
"typeRoots": ["./types"],
"module": "ES2022"
}
}
Leaving index.js as it was: import {AppRegistry} from 'react-native';
import App from './App';
import {name as appName} from './app.json';
AppRegistry.registerComponent(appName, () => App); Here is the App commit with the changes. Thank you for your support. It was exciting to see it resolved finally with such a simple solution. Let me know if I can contribute with you in any other issues. |
That's awesome! 🎉 We need to update the React Native integration docs with this so it's clear to everyone. |
Yes, that'll be helpful for everyone! Let me know if and how I can help. Should I prepare a draft and post in this thread? |
here’s a real example showing how to conditionally enable mocking in a React Native app. import React, { useEffect, useState } from 'react';
import Config from 'react-native-config';
import 'react-native-gesture-handler';
import { AppProvider } from '@/providers';
import { RouterApp } from './src/routes';
const isMockEnabled = !!Number(Config.LOAD_MOCK);
async function enableMocking() {
try {
if (!isMockEnabled) return;
await import('./msw.polyfills');
const { serverApp } = await import('./src/tests/server/config/serverApp');
serverApp.listen();
} catch (error) {
console.error('Error enabling mocking:', error);
throw error;
}
}
function useMocking() {
const [mockReady, setMockReady] = useState(false);
useEffect(() => {
const initializeMocking = async () => {
if (isMockEnabled) {
try {
await enableMocking();
setMockReady(true);
} catch (error) {
console.error('Failed to enable mocking:', error);
}
}
};
// eslint-disable-next-line @typescript-eslint/no-floating-promises
initializeMocking();
}, []);
return mockReady;
}
function App() {
const mockReady = useMocking();
if (isMockEnabled && !mockReady) return null;
return (
<AppProvider>
<RouterApp />
</AppProvider>
);
}
let AppEntryPoint = App;
if (Config.LOAD_STORYBOOK === 'true') {
// eslint-disable-next-line @typescript-eslint/no-unsafe-assignment, @typescript-eslint/no-var-requires, @typescript-eslint/no-unsafe-member-access
AppEntryPoint = require('./.storybook').default;
}
export default AppEntryPoint; |
Preloading MSW fails to detect if App is registered.
Extra line needs to be added which loads data without MSW first, then recognises MSW.
The text was updated successfully, but these errors were encountered: