import { satisfies } from 'semver'; import { execSync } from 'child_process'; import { checkFilesDoNotExist, checkFilesExist, cleanupProject, detectPackageManager, newProject, packageManagerLockFile, readJson, runCLI, runCLIAsync, tmpProjPath, uniq, updateFile, updateJson, } from '../../utils'; describe('js e2e', () => { let scope: string; beforeEach(() => { scope = newProject(); }); afterEach(() => { cleanupProject(); }); it('should create libs with js executors (--compiler=swc)', async () => { const lib = uniq('lib'); runCLI( `generate @nrwl/js:lib ${lib} --buildable --compiler=swc --no-interactive` ); const libPackageJson = readJson(`libs/${lib}/package.json`); expect(libPackageJson.scripts).toBeUndefined(); expect(runCLI(`build ${lib}`)).toContain( 'Successfully compiled: 2 files with swc' ); checkFilesExist( `dist/libs/${lib}/package.json`, `dist/libs/${lib}/src/index.js`, `dist/libs/${lib}/src/lib/${lib}.js`, `dist/libs/${lib}/src/index.d.ts`, `dist/libs/${lib}/src/lib/${lib}.d.ts` ); checkFilesDoNotExist(`libs/${lib}/.babelrc`); const parentLib = uniq('parentlib'); runCLI( `generate @nrwl/js:lib ${parentLib} --buildable --compiler=swc --no-interactive` ); const parentLibPackageJson = readJson(`libs/${parentLib}/package.json`); expect(parentLibPackageJson.scripts).toBeUndefined(); expect((await runCLIAsync(`test ${parentLib}`)).combinedOutput).toContain( 'Ran all test suites' ); expect(runCLI(`build ${parentLib}`)).toContain( 'Successfully compiled: 2 files with swc' ); checkFilesExist( `dist/libs/${parentLib}/package.json`, `dist/libs/${parentLib}/src/index.js`, `dist/libs/${parentLib}/src/lib/${parentLib}.js`, `dist/libs/${parentLib}/src/index.d.ts`, `dist/libs/${parentLib}/src/lib/${parentLib}.d.ts` ); const tsconfig = readJson(`tsconfig.base.json`); expect(tsconfig.compilerOptions.paths).toEqual({ [`@${scope}/${lib}`]: [`libs/${lib}/src/index.ts`], [`@${scope}/${parentLib}`]: [`libs/${parentLib}/src/index.ts`], }); updateFile(`libs/${parentLib}/src/index.ts`, () => { return ` import { ${lib} } from '@${scope}/${lib}'; export * from './lib/${parentLib}'; `; }); const output = runCLI(`build ${parentLib}`); expect(output).toContain('1 task it depends on'); expect(output).toContain('Successfully compiled: 2 files with swc'); runCLI(`build ${parentLib}`); checkFilesExist(`dist/libs/${parentLib}/package.json`); updateJson(`libs/${lib}/.swcrc`, (json) => { json.jsc.externalHelpers = true; return json; }); runCLI(`build ${lib} --generateLockfile=true`); checkFilesExist( `dist/libs/${lib}/package.json`, `dist/libs/${lib}/${ packageManagerLockFile[detectPackageManager(tmpProjPath())] }` ); const swcHelpersFromRoot = readJson(`package.json`).dependencies['@swc/helpers']; const swcHelpersFromDist = readJson(`dist/libs/${lib}/package.json`) .peerDependencies['@swc/helpers']; expect(satisfies(swcHelpersFromDist, swcHelpersFromRoot)).toBeTruthy(); updateJson(`libs/${lib}/.swcrc`, (json) => { json.jsc.externalHelpers = false; return json; }); runCLI(`build ${lib}`); expect(readJson(`dist/libs/${lib}/package.json`)).not.toHaveProperty( 'peerDependencies.@swc/helpers' ); }, 240_000); it('should handle swcrc path mappings', async () => { const lib = uniq('lib'); runCLI(`generate @nrwl/js:lib ${lib} --compiler=swc --no-interactive`); // add a dummy x.ts file for path mappings updateFile( `libs/${lib}/src/x.ts`, ` export function x() { console.log('x'); } ` ); // update .swcrc to use path mappings updateJson(`libs/${lib}/.swcrc`, (json) => { json.jsc.paths = { 'src/*': ['src/*'], }; return json; }); // update lib.ts to use x updateFile(`libs/${lib}/src/lib/${lib}.ts`, () => { return ` // @ts-ignore import { x } from 'src/x'; export function myLib() { console.log(x()); } myLib(); `; }); // now run build runCLI(`build ${lib}`); // invoke the lib with node const result = execSync(`node dist/libs/${lib}/src/lib/${lib}.js`, { cwd: tmpProjPath(), }).toString(); expect(result).toContain('x'); }, 240_000); });