From 5170769bf3f3b3ae7e69e68ebb198c70cb2dd16c Mon Sep 17 00:00:00 2001 From: Jonny Adshead Date: Thu, 16 Nov 2023 08:17:35 -0800 Subject: [PATCH] fix(loadModule): only register required fallbacks (#165) --- .../__tests__/loadModule.node.spec.js | 105 +++++++++++++----- packages/holocron/src/loadModule.node.js | 15 ++- 2 files changed, 87 insertions(+), 33 deletions(-) diff --git a/packages/holocron/__tests__/loadModule.node.spec.js b/packages/holocron/__tests__/loadModule.node.spec.js index 6d31ce7..956f836 100644 --- a/packages/holocron/__tests__/loadModule.node.spec.js +++ b/packages/holocron/__tests__/loadModule.node.spec.js @@ -839,26 +839,17 @@ describe('loadModule.node', () => { status: 200, statusText: 'OK', json: () => ({ - requiredExternals: [ - { - name: 'lodash', + requiredExternals: { + lodash: { version: '1.0.0', + semanticRange: '^1.0.0', browserIntegrity: '123-browser', nodeIntegrity: '123-node', }, - ], + }, }), ok: true, - }) - ); - - mockFetch.mockImplementationOnce(() => Promise.resolve({ - status: 200, - statusText: 'OK', - text: () => 'external fallback code', - ok: true, - }) - ); + })); mockFetch.mockImplementationOnce( makeFetchMock({ fetchText: moduleString }) @@ -874,7 +865,7 @@ describe('loadModule.node', () => { module: '1234', }, }, - enableUnlistedExternalFallbacks: true, + enableUnlistedExternalFallbacks: false, }, }), }); @@ -896,13 +887,7 @@ describe('loadModule.node', () => { module: { onModuleLoadConfig }, moduleName: 'awesome', }); - expect(externalRegistry.getRegisteredExternals()).toEqual({ - lodash: { - '1.0.0': { - str: 'external fallback code', - }, - }, - }); + expect(externalRegistry.getRegisteredExternals()).toEqual({}); }); it('throws an error when an external is required but the root module does not provide it', async () => { @@ -1187,7 +1172,7 @@ describe('loadModule.node', () => { }); }); - it('loads fallback external when enabled and semantic version mismatch', async () => { + it('only loads fallback for external when enabled and has semantic version mismatch', async () => { const mockFetch = jest.fn(); const onModuleLoadConfig = { environmentVariables: [{ name: 'COOL_API_URL', validate: jest.fn() }], @@ -1206,6 +1191,13 @@ describe('loadModule.node', () => { browserIntegrity: '1234-browser', nodeIntegrity: '1234-node', }, + mydash: { + semanticRange: '^1.0.0', + name: 'mydash', + version: '1.2.3', + browserIntegrity: '1234-browser', + nodeIntegrity: '1234-node', + }, }, }), }) @@ -1214,10 +1206,9 @@ describe('loadModule.node', () => { mockFetch.mockImplementationOnce(() => Promise.resolve({ status: 200, statusText: 'OK', - text: () => 'external fallback code', + text: () => 'lodash external fallback code', ok: true, - }) - ); + })); // mock fetch for module code mockFetch.mockImplementationOnce( @@ -1234,6 +1225,11 @@ describe('loadModule.node', () => { fallbackEnabled: true, module: '1234', }, + mydash: { + version: '1.2.3', + fallbackEnabled: true, + module: '1234', + }, }, enableUnlistedExternalFallbacks: false, }, @@ -1264,10 +1260,65 @@ describe('loadModule.node', () => { expect(externalRegistry.getRegisteredExternals()).toEqual({ lodash: { '1.2.3': { - str: 'external fallback code', + str: 'lodash external fallback code', + }, + }, + }); + }); + + it('does not load fallback when not required', async () => { + const mockFetch = jest.fn(); + const onModuleLoadConfig = { + environmentVariables: [{ name: 'COOL_API_URL', validate: jest.fn() }], + }; + const moduleString = { onModuleLoadConfig }; + + // mock fetching moduleConfig file + mockFetch.mockImplementationOnce( + makeFetchMock({ + fetchText: JSON.stringify({ + requiredExternals: { + mydash: { + semanticRange: '^1.0.0', + name: 'mydash', + version: '1.2.3', + browserIntegrity: '1234-browser', + nodeIntegrity: '1234-node', + }, + }, + }), + }) + ); + // mock fetch for module code + mockFetch.mockImplementationOnce( + makeFetchMock({ fetchText: moduleString }) + ); + + const loadModule = load({ + fetch: mockFetch, + getTenantRootModule: () => ({ + appConfig: { + providedExternals: { + mydash: { + version: '1.2.3', + fallbackEnabled: true, + module: '1234', + }, + }, + enableUnlistedExternalFallbacks: false, }, + }), + }); + + await loadModule('awesome', { + node: { + integrity: '123', + url: 'https://example.com/cdn/awesome/1.0.0/awesome.node.js', }, }); + + expect(externalRegistry.getRequiredExternalsRegistry()).toEqual({}); + expect(externalRegistry.getRegisteredExternals()).toEqual({}); }); it('does not load child module when root module does not set getTenantRootModule', async () => { diff --git a/packages/holocron/src/loadModule.node.js b/packages/holocron/src/loadModule.node.js index b0c59f4..25d4801 100644 --- a/packages/holocron/src/loadModule.node.js +++ b/packages/holocron/src/loadModule.node.js @@ -209,6 +209,7 @@ const validateRequiredExternals = ({ const moduleExternals = {}; Object.keys(requiredExternals).forEach((externalName) => { + let failedExternalMessage; const providedExternal = providedExternals[externalName]; const requiredExternal = requiredExternals[externalName]; const { @@ -219,7 +220,7 @@ const validateRequiredExternals = ({ if (!providedExternal) { // eslint-disable-next-line max-len -- long message - messages.push(`External '${externalName}' is required by ${moduleName}, but is not provided by the root module`); + failedExternalMessage = `External '${externalName}' is required by ${moduleName}, but is not provided by the root module`; if (!enableUnlistedExternalFallbacks) { moduleCanBeSafelyLoaded = false; } @@ -230,14 +231,14 @@ const validateRequiredExternals = ({ }) ) { // eslint-disable-next-line max-len -- long message - messages.push(`${externalName}@${semanticRange} is required by ${moduleName}, but the root module provides ${providedExternal.version}`); + failedExternalMessage = `${externalName}@${semanticRange} is required by ${moduleName}, but the root module provides ${providedExternal.version}`; if (fallbackBlockedByRootModule) { moduleCanBeSafelyLoaded = false; } } - if (fallbackExternalAvailable) { + if (fallbackExternalAvailable && failedExternalMessage) { moduleExternals[name] = { name, version, @@ -246,11 +247,13 @@ const validateRequiredExternals = ({ browserIntegrity, }; } + + if (failedExternalMessage) messages.push(failedExternalMessage); }); if (messages.length > 0) { if (moduleCanBeSafelyLoaded || process.env.ONE_DANGEROUSLY_ACCEPT_BREAKING_EXTERNALS === 'true') { - // eslint-disable-next-line no-console -- console necissary for error logging + // eslint-disable-next-line no-console -- console necessary for error logging console.warn(messages.join('\n')); } else { // eslint-disable-next-line unicorn/error-message -- not empty string @@ -272,8 +275,8 @@ const fetchModuleConfig = async (baseUrl) => { try { return await fetchAsset(moduleConfigUrl, true); - } catch (err) { - console.warn('Module Config failed to fetch and parse, external fallbacks will be ignored.', err); + } catch { + // swallow error } return null;