frontend-*-api: add initial support for feature flags

Co-authored-by: Camila Belo <camilaibs@gmail.com>
Co-authored-by: Fredrik Adelöw <freben@gmail.com>
Co-authored-by: Vincenzo Scamporlino <vincenzos@spotify.com>
Co-authored-by: Philipp Hugenroth <philipph@spotify.com>
Signed-off-by: Patrik Oldsberg <poldsberg@gmail.com>
This commit is contained in:
Patrik Oldsberg
2023-11-17 13:04:26 +01:00
parent e539735435
commit 59709286b3
11 changed files with 131 additions and 22 deletions
+5
View File
@@ -0,0 +1,5 @@
---
'@backstage/frontend-plugin-api': patch
---
Add feature flags to plugins and extension overrides.
+5
View File
@@ -0,0 +1,5 @@
---
'@backstage/frontend-app-api': patch
---
Collect and register feature flags from plugins and extension overrides.
@@ -22,6 +22,8 @@ import React from 'react';
import { Route } from 'react-router-dom';
import { collectLegacyRoutes } from './collectLegacyRoutes';
// eslint-disable-next-line @backstage/no-relative-monorepo-imports
import { toInternalBackstagePlugin } from '../../frontend-plugin-api/src/wiring/createPlugin';
describe('collectLegacyRoutes', () => {
it('should collect legacy routes', () => {
@@ -37,7 +39,7 @@ describe('collectLegacyRoutes', () => {
expect(
collected.map(p => ({
id: p.id,
extensions: p.extensions.map(e => ({
extensions: toInternalBackstagePlugin(p).extensions.map(e => ({
id: e.id,
attachTo: e.attachTo,
disabled: e.disabled,
@@ -23,6 +23,8 @@ import {
import { toInternalExtensionOverrides } from '../../../frontend-plugin-api/src/wiring/createExtensionOverrides';
import { ExtensionParameters } from './readAppExtensionsConfig';
import { AppNodeSpec } from '@backstage/frontend-plugin-api';
// eslint-disable-next-line @backstage/no-relative-monorepo-imports
import { toInternalBackstagePlugin } from '../../../frontend-plugin-api/src/wiring/createPlugin';
/** @internal */
export function resolveAppNodeSpecs(options: {
@@ -42,7 +44,10 @@ export function resolveAppNodeSpecs(options: {
);
const pluginExtensions = plugins.flatMap(source => {
return source.extensions.map(extension => ({ ...extension, source }));
return toInternalBackstagePlugin(source).extensions.map(extension => ({
...extension,
source,
}));
});
const overrideExtensions = overrides.flatMap(
override => toInternalExtensionOverrides(override).extensions,
@@ -93,6 +93,10 @@ import { AppNode } from '@backstage/frontend-plugin-api';
import { toLegacyPlugin } from '../routing/toLegacyPlugin';
import { InternalAppContext } from './InternalAppContext';
import { CoreRouter } from '../extensions/CoreRouter';
// eslint-disable-next-line @backstage/no-relative-monorepo-imports
import { toInternalBackstagePlugin } from '../../../frontend-plugin-api/src/wiring/createPlugin';
// eslint-disable-next-line @backstage/no-relative-monorepo-imports
import { toInternalExtensionOverrides } from '../../../frontend-plugin-api/src/wiring/createExtensionOverrides';
const builtinExtensions = [
Core,
@@ -304,6 +308,26 @@ export function createSpecializedApp(options?: {
const appIdentityProxy = new AppIdentityProxy();
const apiHolder = createApiHolder(tree, config, appIdentityProxy);
const featureFlagApi = apiHolder.get(featureFlagsApiRef);
if (featureFlagApi) {
for (const feature of features) {
if (feature.$$type === '@backstage/BackstagePlugin') {
toInternalBackstagePlugin(feature).featureFlags.forEach(flag =>
featureFlagApi.registerFlag({
name: flag.name,
pluginId: feature.id,
}),
);
}
if (feature.$$type === '@backstage/ExtensionOverrides') {
toInternalExtensionOverrides(feature).featureFlags.forEach(flag =>
featureFlagApi.registerFlag({ name: flag.name, pluginId: '' }),
);
}
}
}
const routeInfo = extractRouteInfoFromAppNode(tree.root);
const routeBindings = resolveRouteBindings(
options?.bindRoutes,
+14 -7
View File
@@ -218,15 +218,13 @@ export interface BackstagePlugin<
ExternalRoutes extends AnyExternalRoutes = AnyExternalRoutes,
> {
// (undocumented)
$$type: '@backstage/BackstagePlugin';
readonly $$type: '@backstage/BackstagePlugin';
// (undocumented)
extensions: Extension<unknown>[];
readonly externalRoutes: ExternalRoutes;
// (undocumented)
externalRoutes: ExternalRoutes;
readonly id: string;
// (undocumented)
id: string;
// (undocumented)
routes: Routes;
readonly routes: Routes;
}
export { BackstageUserIdentity };
@@ -607,13 +605,15 @@ export type ExtensionInputValues<
// @public (undocumented)
export interface ExtensionOverrides {
// (undocumented)
$$type: '@backstage/ExtensionOverrides';
readonly $$type: '@backstage/ExtensionOverrides';
}
// @public (undocumented)
export interface ExtensionOverridesOptions {
// (undocumented)
extensions: Extension<unknown>[];
// (undocumented)
featureFlags?: FeatureFlagConfig[];
}
// @public
@@ -631,6 +631,11 @@ export interface ExternalRouteRef<
export { FeatureFlag };
// @public
export type FeatureFlagConfig = {
name: string;
};
export { FeatureFlagsApi };
export { featureFlagsApiRef };
@@ -702,6 +707,8 @@ export interface PluginOptions<
// (undocumented)
externalRoutes?: ExternalRoutes;
// (undocumented)
featureFlags?: FeatureFlagConfig[];
// (undocumented)
id: string;
// (undocumented)
routes?: Routes;
@@ -26,6 +26,7 @@ describe('createExtensionOverrides', () => {
{
"$$type": "@backstage/ExtensionOverrides",
"extensions": [],
"featureFlags": [],
"version": "v1",
}
`);
@@ -60,6 +61,7 @@ describe('createExtensionOverrides', () => {
"output": {},
},
],
"featureFlags": [],
"version": "v1",
}
`);
@@ -15,21 +15,24 @@
*/
import { Extension } from './createExtension';
import { FeatureFlagConfig } from './types';
/** @public */
export interface ExtensionOverridesOptions {
extensions: Extension<unknown>[];
featureFlags?: FeatureFlagConfig[];
}
/** @public */
export interface ExtensionOverrides {
$$type: '@backstage/ExtensionOverrides';
readonly $$type: '@backstage/ExtensionOverrides';
}
/** @internal */
export interface InternalExtensionOverrides extends ExtensionOverrides {
version: string;
extensions: Extension<unknown>[];
readonly version: 'v1';
readonly extensions: Extension<unknown>[];
readonly featureFlags: FeatureFlagConfig[];
}
/** @public */
@@ -40,6 +43,7 @@ export function createExtensionOverrides(
$$type: '@backstage/ExtensionOverrides',
version: 'v1',
extensions: options.extensions,
featureFlags: options.featureFlags ?? [],
} as InternalExtensionOverrides;
}
@@ -50,12 +54,12 @@ export function toInternalExtensionOverrides(
const internal = overrides as InternalExtensionOverrides;
if (internal.$$type !== '@backstage/ExtensionOverrides') {
throw new Error(
`Invalid translation resource, bad type '${internal.$$type}'`,
`Invalid extension overrides instance, bad type '${internal.$$type}'`,
);
}
if (internal.version !== 'v1') {
throw new Error(
`Invalid translation resource, bad version '${internal.version}'`,
`Invalid extension overrides instance, bad version '${internal.version}'`,
);
}
return internal;
@@ -16,6 +16,7 @@
import { Extension } from './createExtension';
import { ExternalRouteRef, RouteRef } from '../routing';
import { FeatureFlagConfig } from './types';
/** @public */
export type AnyRoutes = { [name in string]: RouteRef };
@@ -32,6 +33,7 @@ export interface PluginOptions<
routes?: Routes;
externalRoutes?: ExternalRoutes;
extensions?: Extension<unknown>[];
featureFlags?: FeatureFlagConfig[];
}
/** @public */
@@ -39,11 +41,20 @@ export interface BackstagePlugin<
Routes extends AnyRoutes = AnyRoutes,
ExternalRoutes extends AnyExternalRoutes = AnyExternalRoutes,
> {
$$type: '@backstage/BackstagePlugin';
id: string;
extensions: Extension<unknown>[];
routes: Routes;
externalRoutes: ExternalRoutes;
readonly $$type: '@backstage/BackstagePlugin';
readonly id: string;
readonly routes: Routes;
readonly externalRoutes: ExternalRoutes;
}
/** @public */
export interface InternalBackstagePlugin<
Routes extends AnyRoutes = AnyRoutes,
ExternalRoutes extends AnyExternalRoutes = AnyExternalRoutes,
> extends BackstagePlugin<Routes, ExternalRoutes> {
readonly version: 'v1';
readonly extensions: Extension<unknown>[];
readonly featureFlags: FeatureFlagConfig[];
}
/** @public */
@@ -54,10 +65,28 @@ export function createPlugin<
options: PluginOptions<Routes, ExternalRoutes>,
): BackstagePlugin<Routes, ExternalRoutes> {
return {
...options,
$$type: '@backstage/BackstagePlugin',
version: 'v1',
id: options.id,
routes: options.routes ?? ({} as Routes),
externalRoutes: options.externalRoutes ?? ({} as ExternalRoutes),
extensions: options.extensions ?? [],
$$type: '@backstage/BackstagePlugin',
};
featureFlags: options.featureFlags ?? [],
} as InternalBackstagePlugin<Routes, ExternalRoutes>;
}
/** @internal */
export function toInternalBackstagePlugin(
plugin: BackstagePlugin,
): InternalBackstagePlugin {
const internal = plugin as InternalBackstagePlugin;
if (internal.$$type !== '@backstage/BackstagePlugin') {
throw new Error(`Invalid plugin instance, bad type '${internal.$$type}'`);
}
if (internal.version !== 'v1') {
throw new Error(
`Invalid plugin instance, bad version '${internal.version}'`,
);
}
return internal;
}
@@ -45,3 +45,4 @@ export {
type ExtensionOverrides,
type ExtensionOverridesOptions,
} from './createExtensionOverrides';
export type { FeatureFlagConfig } from './types';
@@ -0,0 +1,25 @@
/*
* Copyright 2023 The Backstage Authors
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
/**
* Feature flag configuration.
*
* @public
*/
export type FeatureFlagConfig = {
/** Feature flag name */
name: string;
};