461 lines
14 KiB
TypeScript

import { E2eTestRunner } from '../../utils/test-runners';
import {
getProjects,
readJson,
readNxJson,
readProjectConfiguration,
updateJson,
} from '@nx/devkit';
import { createTreeWithEmptyWorkspace } from '@nx/devkit/testing';
import {
generateTestHostApplication,
generateTestRemoteApplication,
} from '../utils/testing';
import { getRootTsConfigPathInTree } from '@nx/js';
describe('MF Remote App Generator', () => {
it('should generate a remote mf app with no host', async () => {
// ARRANGE
const tree = createTreeWithEmptyWorkspace({ layout: 'apps-libs' });
// ACT
await generateTestRemoteApplication(tree, {
name: 'test',
port: 4201,
typescriptConfiguration: false,
standalone: false,
skipFormat: true,
});
// ASSERT
expect(tree.read('test/webpack.config.js', 'utf-8')).toMatchSnapshot();
const tsconfigJson = readJson(tree, getRootTsConfigPathInTree(tree));
expect(tsconfigJson.compilerOptions.paths['test/Module']).toEqual([
'test/src/app/remote-entry/entry.module.ts',
]);
});
it('should generate a remote mf app with no host when --typescriptConfiguration=true', async () => {
// ARRANGE
const tree = createTreeWithEmptyWorkspace({ layout: 'apps-libs' });
// ACT
await generateTestRemoteApplication(tree, {
name: 'test',
port: 4201,
typescriptConfiguration: true,
standalone: false,
skipFormat: true,
});
// ASSERT
expect(tree.read('test/webpack.config.ts', 'utf-8')).toMatchSnapshot();
});
it('should generate a remote mf app with a host', async () => {
// ARRANGE
const tree = createTreeWithEmptyWorkspace({ layout: 'apps-libs' });
await generateTestHostApplication(tree, {
name: 'host',
typescriptConfiguration: false,
standalone: false,
skipFormat: true,
});
// ACT
await generateTestRemoteApplication(tree, {
name: 'test',
host: 'host',
typescriptConfiguration: false,
standalone: false,
skipFormat: true,
});
// ASSERT
expect(tree.read('host/webpack.config.js', 'utf-8')).toMatchSnapshot();
expect(tree.read('test/webpack.config.js', 'utf-8')).toMatchSnapshot();
});
it('should generate a remote mf app with a host when --typescriptConfiguration=true', async () => {
// ARRANGE
const tree = createTreeWithEmptyWorkspace({ layout: 'apps-libs' });
await generateTestHostApplication(tree, {
name: 'host',
typescriptConfiguration: true,
standalone: false,
skipFormat: true,
});
// ACT
await generateTestRemoteApplication(tree, {
name: 'test',
host: 'host',
typescriptConfiguration: true,
standalone: false,
skipFormat: true,
});
// ASSERT
expect(tree.read('host/webpack.config.ts', 'utf-8')).toMatchSnapshot();
expect(tree.read('test/webpack.config.ts', 'utf-8')).toMatchSnapshot();
});
it('should error when a remote app is attempted to be generated with an incorrect host', async () => {
// ARRANGE
const tree = createTreeWithEmptyWorkspace({ layout: 'apps-libs' });
// ACT
try {
await generateTestRemoteApplication(tree, {
name: 'test',
host: 'host',
standalone: false,
skipFormat: true,
});
} catch (error) {
// ASSERT
expect(error.message).toEqual(
'The name of the application to be used as the host app does not exist. (host)'
);
}
});
it('should generate a remote mf app and automatically find the next port available', async () => {
// ARRANGE
const tree = createTreeWithEmptyWorkspace({ layout: 'apps-libs' });
await generateTestRemoteApplication(tree, {
name: 'existing',
port: 4201,
standalone: false,
skipFormat: true,
});
// ACT
await generateTestRemoteApplication(tree, {
name: 'test',
standalone: false,
skipFormat: true,
});
// ASSERT
const project = readProjectConfiguration(tree, 'test');
expect(project.targets.serve.options.port).toEqual(4202);
});
it('should generate a remote mf app and automatically find the next port available even when there are no other targets', async () => {
// ARRANGE
const tree = createTreeWithEmptyWorkspace({ layout: 'apps-libs' });
// ACT
await generateTestRemoteApplication(tree, {
name: 'test',
standalone: false,
skipFormat: true,
});
// ASSERT
const project = readProjectConfiguration(tree, 'test');
expect(project.targets.serve.options.port).toEqual(4201);
});
it('should not set the remote as the default project', async () => {
// ARRANGE
const tree = createTreeWithEmptyWorkspace({ layout: 'apps-libs' });
// ACT
await generateTestRemoteApplication(tree, {
name: 'test',
port: 4201,
standalone: false,
skipFormat: true,
});
// ASSERT
const { defaultProject } = readNxJson(tree);
expect(defaultProject).toBeUndefined();
});
it('should generate the a remote setup for standalone components', async () => {
// ARRANGE
const tree = createTreeWithEmptyWorkspace({ layout: 'apps-libs' });
// ACT
await generateTestRemoteApplication(tree, {
name: 'test',
typescriptConfiguration: false,
});
// ASSERT
expect(tree.exists(`test/src/app/app.module.ts`)).toBeFalsy();
expect(tree.exists(`test/src/app/app.component.ts`)).toBeFalsy();
expect(
tree.exists(`test/src/app/remote-entry/entry.module.ts`)
).toBeFalsy();
expect(tree.read(`test/src/bootstrap.ts`, 'utf-8')).toMatchSnapshot();
expect(
tree.read(`test/module-federation.config.js`, 'utf-8')
).toMatchSnapshot();
expect(
tree.read(`test/src/app/remote-entry/entry.component.ts`, 'utf-8')
).toMatchSnapshot();
expect(tree.read(`test/src/app/app.routes.ts`, 'utf-8')).toMatchSnapshot();
expect(
tree.read(`test/src/app/remote-entry/entry.routes.ts`, 'utf-8')
).toMatchSnapshot();
const tsconfigJson = readJson(tree, getRootTsConfigPathInTree(tree));
expect(tsconfigJson.compilerOptions.paths['test/Routes']).toEqual([
'test/src/app/remote-entry/entry.routes.ts',
]);
});
it('should generate the a remote setup for standalone components when --typescriptConfiguration=true', async () => {
// ARRANGE
const tree = createTreeWithEmptyWorkspace({ layout: 'apps-libs' });
// ACT
await generateTestRemoteApplication(tree, {
name: 'test',
typescriptConfiguration: true,
skipFormat: true,
});
// ASSERT
expect(tree.exists(`test/src/app/app.module.ts`)).toBeFalsy();
expect(tree.exists(`test/src/app/app.component.ts`)).toBeFalsy();
expect(
tree.exists(`test/src/app/remote-entry/entry.module.ts`)
).toBeFalsy();
expect(tree.read(`test/src/bootstrap.ts`, 'utf-8')).toMatchSnapshot();
expect(
tree.read(`test/module-federation.config.ts`, 'utf-8')
).toMatchSnapshot();
expect(
tree.read(`test/src/app/remote-entry/entry.component.ts`, 'utf-8')
).toMatchSnapshot();
expect(tree.read(`test/src/app/app.routes.ts`, 'utf-8')).toMatchSnapshot();
expect(
tree.read(`test/src/app/remote-entry/entry.routes.ts`, 'utf-8')
).toMatchSnapshot();
});
it('should not generate an e2e project when e2eTestRunner is none', async () => {
// ARRANGE
const tree = createTreeWithEmptyWorkspace({ layout: 'apps-libs' });
// ACT
await generateTestRemoteApplication(tree, {
name: 'remote1',
e2eTestRunner: E2eTestRunner.None,
standalone: false,
skipFormat: true,
});
// ASSERT
const projects = getProjects(tree);
expect(projects.has('remote1-e2e')).toBeFalsy();
});
it('should generate a correct app component when inline template is used', async () => {
// ARRANGE
const tree = createTreeWithEmptyWorkspace({ layout: 'apps-libs' });
// ACT
await generateTestRemoteApplication(tree, {
name: 'test',
inlineTemplate: true,
standalone: false,
skipFormat: true,
});
// ASSERT
expect(tree.read('test/src/app/app.component.ts', 'utf-8'))
.toMatchInlineSnapshot(`
"import { Component } from '@angular/core';
@Component({
selector: 'proj-root',
template: '<router-outlet></router-outlet>'
})
export class AppComponent {}"
`);
});
it('should update the index.html to use the remote entry component selector for root when standalone', async () => {
// ARRANGE
const tree = createTreeWithEmptyWorkspace({ layout: 'apps-libs' });
// ACT
await generateTestRemoteApplication(tree, {
name: 'test',
skipFormat: true,
});
// ASSERT
expect(tree.read('test/src/index.html', 'utf-8')).not.toContain(
'proj-root'
);
expect(tree.read('test/src/index.html', 'utf-8')).toContain(
'proj-test-entry'
);
});
describe('--ssr', () => {
it('should generate the correct files', async () => {
// ARRANGE
const tree = createTreeWithEmptyWorkspace({ layout: 'apps-libs' });
// ACT
await generateTestRemoteApplication(tree, {
name: 'test',
ssr: true,
typescriptConfiguration: false,
standalone: false,
});
// ASSERT
const project = readProjectConfiguration(tree, 'test');
expect(
tree.exists(`test/src/app/remote-entry/entry.module.ts`)
).toBeTruthy();
expect(
tree.read(`test/src/app/app.module.ts`, 'utf-8')
).toMatchSnapshot();
expect(tree.read(`test/src/bootstrap.ts`, 'utf-8')).toMatchSnapshot();
expect(
tree.read(`test/src/bootstrap.server.ts`, 'utf-8')
).toMatchSnapshot();
expect(tree.read(`test/src/main.server.ts`, 'utf-8')).toMatchSnapshot();
expect(tree.read(`test/server.ts`, 'utf-8')).toMatchSnapshot();
expect(
tree.read(`test/module-federation.config.js`, 'utf-8')
).toMatchSnapshot();
expect(
tree.read(`test/webpack.server.config.js`, 'utf-8')
).toMatchSnapshot();
expect(
tree.read(`test/src/app/remote-entry/entry.component.ts`, 'utf-8')
).toMatchSnapshot();
expect(
tree.read(`test/src/app/app.routes.ts`, 'utf-8')
).toMatchSnapshot();
expect(
tree.read(`test/src/app/remote-entry/entry.routes.ts`, 'utf-8')
).toMatchSnapshot();
expect(project.targets.server).toMatchSnapshot();
expect(
tree.read(`test/src/app/remote-entry/entry.routes.ts`, 'utf-8')
).toMatchSnapshot();
expect(project.targets['static-server']).toMatchSnapshot();
});
it('should generate the correct files when --typescriptConfiguration=true', async () => {
// ARRANGE
const tree = createTreeWithEmptyWorkspace({ layout: 'apps-libs' });
// ACT
await generateTestRemoteApplication(tree, {
name: 'test',
ssr: true,
typescriptConfiguration: true,
standalone: false,
skipFormat: true,
});
// ASSERT
const project = readProjectConfiguration(tree, 'test');
expect(
tree.exists(`test/src/app/remote-entry/entry.module.ts`)
).toBeTruthy();
expect(
tree.read(`test/src/app/app.module.ts`, 'utf-8')
).toMatchSnapshot();
expect(tree.read(`test/src/bootstrap.ts`, 'utf-8')).toMatchSnapshot();
expect(
tree.read(`test/src/bootstrap.server.ts`, 'utf-8')
).toMatchSnapshot();
expect(tree.read(`test/src/main.server.ts`, 'utf-8')).toMatchSnapshot();
expect(tree.read(`test/server.ts`, 'utf-8')).toMatchSnapshot();
expect(
tree.read(`test/module-federation.config.ts`, 'utf-8')
).toMatchSnapshot();
expect(
tree.read(`test/webpack.server.config.ts`, 'utf-8')
).toMatchSnapshot();
expect(
tree.read(`test/src/app/remote-entry/entry.component.ts`, 'utf-8')
).toMatchSnapshot();
expect(
tree.read(`test/src/app/app.routes.ts`, 'utf-8')
).toMatchSnapshot();
expect(
tree.read(`test/src/app/remote-entry/entry.routes.ts`, 'utf-8')
).toMatchSnapshot();
expect(project.targets.server).toMatchSnapshot();
expect(
tree.read(`test/src/app/remote-entry/entry.routes.ts`, 'utf-8')
).toMatchSnapshot();
expect(project.targets['static-server']).toMatchSnapshot();
});
describe('compat', () => {
it('should generate the correct main.server.ts', async () => {
const tree = createTreeWithEmptyWorkspace({ layout: 'apps-libs' });
updateJson(tree, 'package.json', (json) => ({
...json,
dependencies: {
'@angular/core': '15.2.0',
},
}));
await generateTestRemoteApplication(tree, {
name: 'test',
ssr: true,
skipFormat: true,
});
expect(tree.read(`test/src/main.server.ts`, 'utf-8')).toMatchSnapshot();
});
});
});
describe('--project-name-and-root-format=derived', () => {
it('should generate remote', async () => {
const tree = createTreeWithEmptyWorkspace({ layout: 'apps-libs' });
await generateTestRemoteApplication(tree, {
name: 'test',
port: 4201,
projectNameAndRootFormat: 'derived',
typescriptConfiguration: false,
standalone: false,
skipFormat: true,
});
expect(tree.exists('apps/test/webpack.config.js')).toBe(true);
expect(readProjectConfiguration(tree, 'test').root).toBe('apps/test');
});
it('should generate remote in a directory', async () => {
const tree = createTreeWithEmptyWorkspace({ layout: 'apps-libs' });
await generateTestRemoteApplication(tree, {
name: 'test',
port: 4201,
directory: 'shared',
projectNameAndRootFormat: 'derived',
typescriptConfiguration: false,
standalone: false,
skipFormat: true,
});
expect(tree.exists('apps/shared/test/webpack.config.js')).toBe(true);
expect(readProjectConfiguration(tree, 'shared-test').root).toBe(
'apps/shared/test'
);
});
});
});