1
0
mirror of https://github.com/Unleash/unleash.git synced 2025-01-20 00:08:02 +01:00
unleash.unleash/src/lib/features/dependent-features/dependent-features-controller.ts

260 lines
8.5 KiB
TypeScript
Raw Normal View History

import { Response } from 'express';
import Controller from '../../routes/controller';
import { OpenApiService } from '../../services';
import {
IFlagResolver,
IUnleashConfig,
IUnleashServices,
NONE,
UPDATE_FEATURE_DEPENDENCY,
} from '../../types';
import { Logger } from '../../logger';
import {
CreateDependentFeatureSchema,
createRequestSchema,
createResponseSchema,
emptyResponse,
getStandardResponses,
ParentFeatureOptionsSchema,
} from '../../openapi';
import { IAuthRequest } from '../../routes/unleash-types';
import { InvalidOperationError } from '../../error';
import { DependentFeaturesService } from './dependent-features-service';
import { TransactionCreator, UnleashTransaction } from '../../db/transaction';
2023-09-29 14:02:15 +02:00
import { extractUsernameFromUser } from '../../util';
2023-09-29 14:02:15 +02:00
interface ProjectParams {
projectId: string;
}
interface FeatureParams extends ProjectParams {
2023-09-25 15:50:05 +02:00
child: string;
}
2023-09-29 14:02:15 +02:00
interface DeleteDependencyParams extends ProjectParams {
2023-09-25 15:50:05 +02:00
child: string;
parent: string;
}
const PATH = '/:projectId/features';
2023-09-25 15:50:05 +02:00
const PATH_FEATURE = `${PATH}/:child`;
const PATH_DEPENDENCIES = `${PATH_FEATURE}/dependencies`;
const PATH_PARENTS = `${PATH_FEATURE}/parents`;
2023-09-25 15:50:05 +02:00
const PATH_DEPENDENCY = `${PATH_FEATURE}/dependencies/:parent`;
type DependentFeaturesServices = Pick<
IUnleashServices,
2023-09-25 15:50:05 +02:00
| 'transactionalDependentFeaturesService'
| 'dependentFeaturesService'
| 'openApiService'
>;
export default class DependentFeaturesController extends Controller {
private transactionalDependentFeaturesService: (
db: UnleashTransaction,
) => DependentFeaturesService;
2023-09-25 15:50:05 +02:00
private dependentFeaturesService: DependentFeaturesService;
private readonly startTransaction: TransactionCreator<UnleashTransaction>;
private openApiService: OpenApiService;
private flagResolver: IFlagResolver;
private readonly logger: Logger;
constructor(
config: IUnleashConfig,
{
transactionalDependentFeaturesService,
2023-09-25 15:50:05 +02:00
dependentFeaturesService,
openApiService,
}: DependentFeaturesServices,
startTransaction: TransactionCreator<UnleashTransaction>,
) {
super(config);
this.transactionalDependentFeaturesService =
transactionalDependentFeaturesService;
2023-09-25 15:50:05 +02:00
this.dependentFeaturesService = dependentFeaturesService;
this.openApiService = openApiService;
this.flagResolver = config.flagResolver;
this.startTransaction = startTransaction;
this.logger = config.getLogger(
'/dependent-features/dependent-feature-service.ts',
);
this.route({
method: 'post',
path: PATH_DEPENDENCIES,
handler: this.addFeatureDependency,
permission: UPDATE_FEATURE_DEPENDENCY,
middleware: [
openApiService.validPath({
tags: ['Features'],
summary: 'Add a feature dependency.',
description:
'Add a dependency to a parent feature. Each environment will resolve corresponding dependency independently.',
operationId: 'addFeatureDependency',
requestBody: createRequestSchema(
'createDependentFeatureSchema',
),
responses: {
200: emptyResponse,
...getStandardResponses(401, 403, 404),
},
}),
],
});
2023-09-25 15:50:05 +02:00
this.route({
method: 'delete',
path: PATH_DEPENDENCY,
handler: this.deleteFeatureDependency,
permission: UPDATE_FEATURE_DEPENDENCY,
2023-09-25 15:50:05 +02:00
acceptAnyContentType: true,
middleware: [
openApiService.validPath({
tags: ['Features'],
summary: 'Deletes a feature dependency.',
description: 'Remove a dependency to a parent feature.',
operationId: 'deleteFeatureDependency',
responses: {
200: emptyResponse,
...getStandardResponses(401, 403, 404),
},
}),
],
});
this.route({
method: 'delete',
path: PATH_DEPENDENCIES,
handler: this.deleteFeatureDependencies,
permission: UPDATE_FEATURE_DEPENDENCY,
acceptAnyContentType: true,
middleware: [
openApiService.validPath({
tags: ['Features'],
summary: 'Deletes feature dependencies.',
description: 'Remove dependencies to all parent features.',
operationId: 'deleteFeatureDependencies',
responses: {
200: emptyResponse,
...getStandardResponses(401, 403, 404),
},
}),
],
});
this.route({
method: 'get',
path: PATH_PARENTS,
handler: this.getParentOptions,
permission: NONE,
middleware: [
openApiService.validPath({
tags: ['Features'],
summary: 'List parent options.',
description:
'List available parents who have no transitive dependencies.',
operationId: 'listParentOptions',
responses: {
200: createResponseSchema('parentFeatureOptionsSchema'),
...getStandardResponses(401, 403, 404),
},
}),
],
});
}
async addFeatureDependency(
req: IAuthRequest<FeatureParams, any, CreateDependentFeatureSchema>,
res: Response,
): Promise<void> {
2023-09-29 14:02:15 +02:00
const { child, projectId } = req.params;
const { variants, enabled, feature } = req.body;
if (this.config.flagResolver.isEnabled('dependentFeatures')) {
await this.startTransaction(async (tx) =>
this.transactionalDependentFeaturesService(
tx,
2023-09-29 14:02:15 +02:00
).upsertFeatureDependency(
{ child, projectId },
{
variants,
enabled,
feature,
},
extractUsernameFromUser(req.user),
),
);
res.status(200).end();
} else {
throw new InvalidOperationError(
'Dependent features are not enabled',
);
}
}
2023-09-25 15:50:05 +02:00
async deleteFeatureDependency(
req: IAuthRequest<DeleteDependencyParams, any, any>,
res: Response,
): Promise<void> {
2023-09-29 14:02:15 +02:00
const { child, parent, projectId } = req.params;
2023-09-25 15:50:05 +02:00
if (this.config.flagResolver.isEnabled('dependentFeatures')) {
2023-09-29 14:02:15 +02:00
await this.dependentFeaturesService.deleteFeatureDependency(
{
parent,
child,
},
projectId,
extractUsernameFromUser(req.user),
);
2023-09-25 15:50:05 +02:00
res.status(200).end();
} else {
throw new InvalidOperationError(
'Dependent features are not enabled',
);
}
}
async deleteFeatureDependencies(
req: IAuthRequest<FeatureParams, any, any>,
res: Response,
): Promise<void> {
2023-09-29 14:02:15 +02:00
const { child, projectId } = req.params;
if (this.config.flagResolver.isEnabled('dependentFeatures')) {
await this.dependentFeaturesService.deleteFeatureDependencies(
child,
2023-09-29 14:02:15 +02:00
projectId,
extractUsernameFromUser(req.user),
);
res.status(200).end();
} else {
throw new InvalidOperationError(
'Dependent features are not enabled',
);
}
}
async getParentOptions(
req: IAuthRequest<FeatureParams, any, any>,
res: Response<ParentFeatureOptionsSchema>,
): Promise<void> {
const { child } = req.params;
if (this.config.flagResolver.isEnabled('dependentFeatures')) {
const parentOptions =
await this.dependentFeaturesService.getParentOptions(child);
res.send(parentOptions);
} else {
throw new InvalidOperationError(
'Dependent features are not enabled',
);
}
}
}