frontend-app-api: apply package detection filters at runtime

Signed-off-by: Patrik Oldsberg <poldsberg@gmail.com>
This commit is contained in:
Patrik Oldsberg
2023-10-16 17:35:15 +02:00
parent 7187f2953e
commit f78ac58f88
5 changed files with 84 additions and 19 deletions
+5
View File
@@ -0,0 +1,5 @@
---
'@backstage/frontend-app-api': patch
---
Filters for discovered packages are now also applied at runtime. This makes it possible to disable packages through the `app.experimental.packages` config at runtime.
+8
View File
@@ -16,6 +16,14 @@
export interface Config {
app?: {
experimental?: {
/**
* @visibility frontend
* @deepVisibility frontend
*/
packages?: 'all' | { include?: string[]; exclude?: string[] };
};
/**
* @deepVisibility frontend
*/
@@ -116,7 +116,7 @@ export interface ExtensionTree {
export function createExtensionTree(options: {
config: Config;
}): ExtensionTree {
const features = getAvailableFeatures();
const features = getAvailableFeatures(options.config);
const { instances } = createInstances({
features,
config: options.config,
@@ -286,7 +286,7 @@ export function createApp(options: {
overrideBaseUrlConfigs(defaultConfigLoaderSync()),
);
const discoveredFeatures = getAvailableFeatures();
const discoveredFeatures = getAvailableFeatures(config);
const loadedFeatures = (await options.featureLoader?.({ config })) ?? [];
const allFeatures = Array.from(
new Set([
@@ -16,24 +16,29 @@
import { createPlugin } from '@backstage/frontend-plugin-api';
import { getAvailableFeatures } from './discovery';
import { ConfigReader } from '@backstage/config';
const globalSpy = jest.fn();
Object.defineProperty(global, '__@backstage/discovered__', {
get: globalSpy,
});
const config = new ConfigReader({
app: { experimental: { packages: 'all' } },
});
describe('getAvailableFeatures', () => {
afterEach(jest.resetAllMocks);
it('should discover nothing with undefined global', () => {
expect(getAvailableFeatures()).toEqual([]);
expect(getAvailableFeatures(config)).toEqual([]);
});
it('should discover nothing with empty global', () => {
globalSpy.mockReturnValue({
modules: [],
});
expect(getAvailableFeatures()).toEqual([]);
expect(getAvailableFeatures(config)).toEqual([]);
});
it('should discover a plugin', () => {
@@ -41,24 +46,24 @@ describe('getAvailableFeatures', () => {
globalSpy.mockReturnValue({
modules: [{ default: testPlugin }],
});
expect(getAvailableFeatures()).toEqual([testPlugin]);
expect(getAvailableFeatures(config)).toEqual([testPlugin]);
});
it('should ignore garbage', () => {
globalSpy.mockReturnValueOnce({ modules: [{ default: null }] });
expect(getAvailableFeatures()).toEqual([]);
expect(getAvailableFeatures(config)).toEqual([]);
globalSpy.mockReturnValueOnce({ modules: [{ default: undefined }] });
expect(getAvailableFeatures()).toEqual([]);
expect(getAvailableFeatures(config)).toEqual([]);
globalSpy.mockReturnValueOnce({ modules: [{ default: Symbol() }] });
expect(getAvailableFeatures()).toEqual([]);
expect(getAvailableFeatures(config)).toEqual([]);
globalSpy.mockReturnValueOnce({ modules: [{ default: () => {} }] });
expect(getAvailableFeatures()).toEqual([]);
expect(getAvailableFeatures(config)).toEqual([]);
globalSpy.mockReturnValueOnce({ modules: [{ default: 0 }] });
expect(getAvailableFeatures()).toEqual([]);
expect(getAvailableFeatures(config)).toEqual([]);
globalSpy.mockReturnValueOnce({ modules: [{ default: false }] });
expect(getAvailableFeatures()).toEqual([]);
expect(getAvailableFeatures(config)).toEqual([]);
globalSpy.mockReturnValueOnce({ modules: [{ default: true }] });
expect(getAvailableFeatures()).toEqual([]);
expect(getAvailableFeatures(config)).toEqual([]);
});
it('should discover multiple plugins', () => {
@@ -72,7 +77,7 @@ describe('getAvailableFeatures', () => {
{ default: test3Plugin },
],
});
expect(getAvailableFeatures()).toEqual([
expect(getAvailableFeatures(config)).toEqual([
test1Plugin,
test2Plugin,
test3Plugin,
@@ -14,28 +14,75 @@
* limitations under the License.
*/
import { Config, ConfigReader } from '@backstage/config';
import {
BackstagePlugin,
ExtensionOverrides,
} from '@backstage/frontend-plugin-api';
interface DiscoveryGlobal {
modules: Array<{ name: string; default: unknown }>;
modules: Array<{ name: string; export?: string; default: unknown }>;
}
function readPackageDetectionConfig(config: Config) {
const packages = config.getOptional('app.experimental.packages');
if (packages === undefined || packages === null) {
return undefined;
}
if (typeof packages === 'string') {
if (packages !== 'all') {
throw new Error(
`Invalid app.experimental.packages mode, got '${packages}', expected 'all'`,
);
}
return {};
}
if (typeof packages !== 'object' || Array.isArray(packages)) {
throw new Error(
"Invalid config at 'app.experimental.packages', expected object",
);
}
const packagesConfig = new ConfigReader(
packages,
'app.experimental.packages',
);
return {
include: packagesConfig.getOptionalStringArray('include'),
exclude: packagesConfig.getOptionalStringArray('exclude'),
};
}
/**
* @public
*/
export function getAvailableFeatures(): (
| BackstagePlugin
| ExtensionOverrides
)[] {
export function getAvailableFeatures(
config: Config,
): (BackstagePlugin | ExtensionOverrides)[] {
const discovered = (
window as { '__@backstage/discovered__'?: DiscoveryGlobal }
)['__@backstage/discovered__'];
const detection = readPackageDetectionConfig(config);
if (!detection) {
return [];
}
return (
discovered?.modules.map(m => m.default).filter(isBackstageFeature) ?? []
discovered?.modules
.filter(({ name }) => {
if (detection.exclude?.includes(name)) {
return false;
}
if (detection.include && !detection.include.includes(name)) {
return false;
}
return true;
})
.map(m => m.default)
.filter(isBackstageFeature) ?? []
);
}