328 lines
10 KiB
TypeScript
328 lines
10 KiB
TypeScript
import {
|
|
checkFilesExist,
|
|
cleanupProject,
|
|
expectTestsPass,
|
|
getPackageManagerCommand,
|
|
isOSX,
|
|
killProcessAndPorts,
|
|
newProject,
|
|
readJson,
|
|
runCLI,
|
|
runCLIAsync,
|
|
runCommand,
|
|
runCommandUntil,
|
|
runE2ETests,
|
|
uniq,
|
|
updateFile,
|
|
updateJson,
|
|
} from '@nx/e2e/utils';
|
|
import { ChildProcess } from 'child_process';
|
|
import { join } from 'path';
|
|
|
|
describe('@nx/react-native (legacy)', () => {
|
|
let proj: string;
|
|
let appName = uniq('my-app');
|
|
let libName = uniq('lib');
|
|
|
|
beforeAll(() => {
|
|
proj = newProject();
|
|
// we create empty preset above which skips creation of `production` named input
|
|
updateJson('nx.json', (nxJson) => {
|
|
nxJson.namedInputs = {
|
|
default: ['{projectRoot}/**/*', 'sharedGlobals'],
|
|
production: ['default'],
|
|
sharedGlobals: [],
|
|
};
|
|
return nxJson;
|
|
});
|
|
runCLI(
|
|
`generate @nx/react-native:application ${appName} --bunlder=webpack --e2eTestRunner=cypress --install=false --no-interactive`,
|
|
{ env: { NX_ADD_PLUGINS: 'false' } }
|
|
);
|
|
runCLI(
|
|
`generate @nx/react-native:library ${libName} --buildable --publishable --importPath=${proj}/${libName} --no-interactive`
|
|
);
|
|
});
|
|
afterAll(() => cleanupProject());
|
|
|
|
it('should build for web', async () => {
|
|
const results = runCLI(`build ${appName}`);
|
|
expect(results).toContain('Successfully ran target build');
|
|
});
|
|
|
|
it('should test and lint', async () => {
|
|
const componentName = uniq('Component');
|
|
runCLI(
|
|
`generate @nx/react-native:component ${componentName} --project=${libName} --export --no-interactive`
|
|
);
|
|
|
|
updateFile(`apps/${appName}/src/app/App.tsx`, (content) => {
|
|
let updated = `// eslint-disable-next-line @typescript-eslint/no-unused-vars\nimport {${componentName}} from '${proj}/${libName}';\n${content}`;
|
|
return updated;
|
|
});
|
|
|
|
expectTestsPass(await runCLIAsync(`test ${appName}`));
|
|
expectTestsPass(await runCLIAsync(`test ${libName}`));
|
|
|
|
const appLintResults = await runCLIAsync(`lint ${appName}`);
|
|
expect(appLintResults.combinedOutput).toContain(
|
|
'Successfully ran target lint'
|
|
);
|
|
|
|
const libLintResults = await runCLIAsync(`lint ${libName}`);
|
|
expect(libLintResults.combinedOutput).toContain(
|
|
'Successfully ran target lint'
|
|
);
|
|
});
|
|
|
|
it('should run e2e for cypress', async () => {
|
|
if (runE2ETests()) {
|
|
let results = runCLI(`e2e ${appName}-e2e`);
|
|
expect(results).toContain('Successfully ran target e2e');
|
|
|
|
results = runCLI(`e2e ${appName}-e2e --configuration=ci`);
|
|
expect(results).toContain('Successfully ran target e2e');
|
|
}
|
|
});
|
|
|
|
it('should bundle-ios', async () => {
|
|
const iosBundleResult = await runCLIAsync(
|
|
`bundle-ios ${appName} --sourcemapOutput=../../dist/apps/${appName}/ios/main.map`
|
|
);
|
|
expect(iosBundleResult.combinedOutput).toContain(
|
|
'Done writing bundle output'
|
|
);
|
|
expect(() => {
|
|
checkFilesExist(`dist/apps/${appName}/ios/main.jsbundle`);
|
|
checkFilesExist(`dist/apps/${appName}/ios/main.map`);
|
|
}).not.toThrow();
|
|
});
|
|
|
|
it('should bundle-android', async () => {
|
|
const androidBundleResult = await runCLIAsync(
|
|
`bundle-android ${appName} --sourcemapOutput=../../dist/apps/${appName}/android/main.map`
|
|
);
|
|
expect(androidBundleResult.combinedOutput).toContain(
|
|
'Done writing bundle output'
|
|
);
|
|
expect(() => {
|
|
checkFilesExist(`dist/apps/${appName}/android/main.jsbundle`);
|
|
checkFilesExist(`dist/apps/${appName}/android/main.map`);
|
|
}).not.toThrow();
|
|
});
|
|
|
|
it('should start', async () => {
|
|
let process: ChildProcess;
|
|
const port = 8081;
|
|
|
|
try {
|
|
process = await runCommandUntil(
|
|
`start ${appName} --interactive=false --port=${port}`,
|
|
(output) => {
|
|
return (
|
|
output.includes(`http://localhost:${port}`) ||
|
|
output.includes('Starting JS server...') ||
|
|
output.includes('Welcome to Metro')
|
|
);
|
|
}
|
|
);
|
|
} catch (err) {
|
|
console.error(err);
|
|
}
|
|
|
|
// port and process cleanup
|
|
try {
|
|
if (process && process.pid) {
|
|
await killProcessAndPorts(process.pid, port);
|
|
}
|
|
} catch (err) {
|
|
expect(err).toBeFalsy();
|
|
}
|
|
});
|
|
|
|
it('should serve', async () => {
|
|
let process: ChildProcess;
|
|
const port = 8081;
|
|
|
|
try {
|
|
process = await runCommandUntil(
|
|
`serve ${appName} --interactive=false --port=${port}`,
|
|
(output) => {
|
|
return output.includes(`http://localhost:${port}`);
|
|
}
|
|
);
|
|
} catch (err) {
|
|
console.error(err);
|
|
}
|
|
|
|
// port and process cleanup
|
|
try {
|
|
if (process && process.pid) {
|
|
await killProcessAndPorts(process.pid, port);
|
|
}
|
|
} catch (err) {
|
|
expect(err).toBeFalsy();
|
|
}
|
|
});
|
|
|
|
if (isOSX()) {
|
|
// TODO(@meeroslav): this test is causing git-hasher to overflow with arguments. Enable when it's fixed.
|
|
xit('should pod install', async () => {
|
|
expect(async () => {
|
|
await runCLIAsync(`pod-install ${appName}`);
|
|
checkFilesExist(`apps/${appName}/ios/Podfile.lock`);
|
|
}).not.toThrow();
|
|
});
|
|
}
|
|
|
|
it('should create storybook with application', async () => {
|
|
runCLI(
|
|
`generate @nx/react-native:storybook-configuration ${appName} --generateStories --no-interactive`
|
|
);
|
|
checkFilesExist(
|
|
`apps/${appName}/.storybook/main.ts`,
|
|
`apps/${appName}/src/app/App.stories.tsx`
|
|
);
|
|
});
|
|
|
|
it('should upgrade native for application', async () => {
|
|
expect(() => runCLI(`upgrade ${appName}`)).not.toThrow();
|
|
});
|
|
|
|
it('should build publishable library', async () => {
|
|
const componentName = uniq('Component');
|
|
|
|
runCLI(
|
|
`generate @nx/react-native:component ${componentName} --project=${libName} --export`
|
|
);
|
|
expect(() => {
|
|
runCLI(`build ${libName}`);
|
|
checkFilesExist(`dist/libs/${libName}/index.esm.js`);
|
|
checkFilesExist(`dist/libs/${libName}/src/index.d.ts`);
|
|
}).not.toThrow();
|
|
});
|
|
|
|
it('sync npm dependencies for autolink', async () => {
|
|
// Add npm package with native modules
|
|
runCommand(
|
|
`${
|
|
getPackageManagerCommand().addDev
|
|
} react-native-image-picker @react-native-async-storage/async-storage`
|
|
);
|
|
|
|
// Add import for Nx to pick up
|
|
updateFile(join('apps', appName, 'src/app/App.tsx'), (content) => {
|
|
return `import AsyncStorage from '@react-native-async-storage/async-storage';${content}`;
|
|
});
|
|
|
|
await runCLIAsync(`sync-deps ${appName}`);
|
|
let result = readJson(join('apps', appName, 'package.json'));
|
|
expect(result).toMatchObject({
|
|
dependencies: {
|
|
'@react-native-async-storage/async-storage': '*',
|
|
},
|
|
});
|
|
|
|
await runCLIAsync(
|
|
`sync-deps ${appName} --include=react-native-image-picker`
|
|
);
|
|
result = readJson(join('apps', appName, 'package.json'));
|
|
expect(result).toMatchObject({
|
|
dependencies: {
|
|
'@react-native-async-storage/async-storage': '*',
|
|
'react-native-image-picker': '*',
|
|
},
|
|
});
|
|
|
|
await runCLIAsync(`sync-deps ${appName} --all`);
|
|
result = readJson(join('apps', appName, 'package.json'));
|
|
expect(result).toMatchObject({
|
|
dependencies: {
|
|
'@react-native-async-storage/async-storage': '*',
|
|
'react-native-image-picker': '*',
|
|
},
|
|
devDependencies: {
|
|
'@nx/react-native': '*',
|
|
},
|
|
});
|
|
});
|
|
|
|
it('should tsc app', async () => {
|
|
expect(() => {
|
|
const pmc = getPackageManagerCommand();
|
|
runCommand(
|
|
`${pmc.runUninstalledPackage} tsc -p apps/${appName}/tsconfig.app.json`
|
|
);
|
|
checkFilesExist(
|
|
`dist/out-tsc/apps/${appName}/src/main.js`,
|
|
`dist/out-tsc/apps/${appName}/src/main.d.ts`,
|
|
`dist/out-tsc/apps/${appName}/src/app/App.js`,
|
|
`dist/out-tsc/apps/${appName}/src/app/App.d.ts`,
|
|
`dist/out-tsc/libs/${libName}/src/index.js`,
|
|
`dist/out-tsc/libs/${libName}/src/index.d.ts`
|
|
);
|
|
}).not.toThrow();
|
|
});
|
|
|
|
it('should support generating projects with the new name and root format', () => {
|
|
const appName = uniq('app1');
|
|
const libName = uniq('@my-org/lib1');
|
|
|
|
runCLI(
|
|
`generate @nx/react-native:application ${appName} --project-name-and-root-format=as-provided --install=false --no-interactive`,
|
|
{ env: { NX_ADD_PLUGINS: 'false' } }
|
|
);
|
|
|
|
// check files are generated without the layout directory ("apps/") and
|
|
// using the project name as the directory when no directory is provided
|
|
checkFilesExist(`${appName}/src/app/App.tsx`);
|
|
// check tests pass
|
|
const appTestResult = runCLI(`test ${appName}`);
|
|
expect(appTestResult).toContain(
|
|
`Successfully ran target test for project ${appName}`
|
|
);
|
|
|
|
// assert scoped project names are not supported when --project-name-and-root-format=derived
|
|
expect(() =>
|
|
runCLI(
|
|
`generate @nx/react-native:library ${libName} --buildable --project-name-and-root-format=derived`
|
|
)
|
|
).toThrow();
|
|
|
|
runCLI(
|
|
`generate @nx/react-native:library ${libName} --buildable --project-name-and-root-format=as-provided`
|
|
);
|
|
|
|
// check files are generated without the layout directory ("libs/") and
|
|
// using the project name as the directory when no directory is provided
|
|
checkFilesExist(`${libName}/src/index.ts`);
|
|
// check tests pass
|
|
const libTestResult = runCLI(`test ${libName}`);
|
|
expect(libTestResult).toContain(
|
|
`Successfully ran target test for project ${libName}`
|
|
);
|
|
});
|
|
|
|
it('should run build with vite bundler and e2e with playwright', async () => {
|
|
const appName2 = uniq('my-app');
|
|
runCLI(
|
|
`generate @nx/react-native:application ${appName2} --bundler=vite --e2eTestRunner=playwright --install=false --no-interactive`,
|
|
{ env: { NX_ADD_PLUGINS: 'false' } }
|
|
);
|
|
const buildResults = runCLI(`build ${appName2}`);
|
|
expect(buildResults).toContain('Successfully ran target build');
|
|
if (runE2ETests()) {
|
|
const e2eResults = runCLI(`e2e ${appName2}-e2e`);
|
|
expect(e2eResults).toContain('Successfully ran target e2e');
|
|
}
|
|
|
|
runCLI(
|
|
`generate @nx/react-native:storybook-configuration ${appName2} --generateStories --no-interactive`
|
|
);
|
|
checkFilesExist(
|
|
`apps/${appName2}/.storybook/main.ts`,
|
|
`apps/${appName2}/src/app/App.stories.tsx`
|
|
);
|
|
});
|
|
});
|