2023-09-19 11:23:21 +02:00
|
|
|
import { Response } from 'express';
|
|
|
|
import Controller from '../../routes/controller';
|
|
|
|
import { OpenApiService } from '../../services';
|
|
|
|
import {
|
|
|
|
IFlagResolver,
|
|
|
|
IUnleashConfig,
|
|
|
|
IUnleashServices,
|
2023-09-26 14:03:24 +02:00
|
|
|
NONE,
|
2023-10-04 09:27:14 +02:00
|
|
|
UPDATE_FEATURE_DEPENDENCY,
|
2023-09-19 11:23:21 +02:00
|
|
|
} from '../../types';
|
|
|
|
import { Logger } from '../../logger';
|
|
|
|
import {
|
|
|
|
CreateDependentFeatureSchema,
|
|
|
|
createRequestSchema,
|
2023-09-26 14:03:24 +02:00
|
|
|
createResponseSchema,
|
2023-09-19 11:23:21 +02:00
|
|
|
emptyResponse,
|
|
|
|
getStandardResponses,
|
2023-09-26 14:03:24 +02:00
|
|
|
ParentFeatureOptionsSchema,
|
2023-09-19 11:23:21 +02:00
|
|
|
} from '../../openapi';
|
|
|
|
import { IAuthRequest } from '../../routes/unleash-types';
|
|
|
|
import { InvalidOperationError } from '../../error';
|
|
|
|
import { DependentFeaturesService } from './dependent-features-service';
|
2023-09-25 10:12:32 +02:00
|
|
|
import { TransactionCreator, UnleashTransaction } from '../../db/transaction';
|
2023-09-29 14:02:15 +02:00
|
|
|
import { extractUsernameFromUser } from '../../util';
|
2023-09-19 11:23:21 +02:00
|
|
|
|
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;
|
2023-09-19 11:23:21 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
const PATH = '/:projectId/features';
|
2023-09-25 15:50:05 +02:00
|
|
|
const PATH_FEATURE = `${PATH}/:child`;
|
2023-09-19 11:23:21 +02:00
|
|
|
const PATH_DEPENDENCIES = `${PATH_FEATURE}/dependencies`;
|
2023-09-26 14:03:24 +02:00
|
|
|
const PATH_PARENTS = `${PATH_FEATURE}/parents`;
|
2023-09-25 15:50:05 +02:00
|
|
|
const PATH_DEPENDENCY = `${PATH_FEATURE}/dependencies/:parent`;
|
2023-09-19 11:23:21 +02:00
|
|
|
|
|
|
|
type DependentFeaturesServices = Pick<
|
|
|
|
IUnleashServices,
|
2023-09-25 15:50:05 +02:00
|
|
|
| 'transactionalDependentFeaturesService'
|
|
|
|
| 'dependentFeaturesService'
|
|
|
|
| 'openApiService'
|
2023-09-19 11:23:21 +02:00
|
|
|
>;
|
|
|
|
|
|
|
|
export default class DependentFeaturesController extends Controller {
|
2023-09-25 10:12:32 +02:00
|
|
|
private transactionalDependentFeaturesService: (
|
|
|
|
db: UnleashTransaction,
|
|
|
|
) => DependentFeaturesService;
|
|
|
|
|
2023-09-25 15:50:05 +02:00
|
|
|
private dependentFeaturesService: DependentFeaturesService;
|
|
|
|
|
2023-09-25 10:12:32 +02:00
|
|
|
private readonly startTransaction: TransactionCreator<UnleashTransaction>;
|
2023-09-19 11:23:21 +02:00
|
|
|
|
|
|
|
private openApiService: OpenApiService;
|
|
|
|
|
|
|
|
private flagResolver: IFlagResolver;
|
|
|
|
|
|
|
|
private readonly logger: Logger;
|
|
|
|
|
|
|
|
constructor(
|
|
|
|
config: IUnleashConfig,
|
2023-09-25 10:12:32 +02:00
|
|
|
{
|
|
|
|
transactionalDependentFeaturesService,
|
2023-09-25 15:50:05 +02:00
|
|
|
dependentFeaturesService,
|
2023-09-25 10:12:32 +02:00
|
|
|
openApiService,
|
|
|
|
}: DependentFeaturesServices,
|
|
|
|
startTransaction: TransactionCreator<UnleashTransaction>,
|
2023-09-19 11:23:21 +02:00
|
|
|
) {
|
|
|
|
super(config);
|
2023-09-25 10:12:32 +02:00
|
|
|
this.transactionalDependentFeaturesService =
|
|
|
|
transactionalDependentFeaturesService;
|
2023-09-25 15:50:05 +02:00
|
|
|
this.dependentFeaturesService = dependentFeaturesService;
|
2023-09-19 11:23:21 +02:00
|
|
|
this.openApiService = openApiService;
|
|
|
|
this.flagResolver = config.flagResolver;
|
2023-09-25 10:12:32 +02:00
|
|
|
this.startTransaction = startTransaction;
|
2023-09-19 11:23:21 +02:00
|
|
|
this.logger = config.getLogger(
|
|
|
|
'/dependent-features/dependent-feature-service.ts',
|
|
|
|
);
|
|
|
|
|
|
|
|
this.route({
|
|
|
|
method: 'post',
|
|
|
|
path: PATH_DEPENDENCIES,
|
|
|
|
handler: this.addFeatureDependency,
|
2023-10-04 09:27:14 +02:00
|
|
|
permission: UPDATE_FEATURE_DEPENDENCY,
|
2023-09-19 11:23:21 +02:00
|
|
|
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,
|
2023-10-04 09:27:14 +02:00
|
|
|
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),
|
|
|
|
},
|
|
|
|
}),
|
|
|
|
],
|
|
|
|
});
|
2023-09-26 09:38:34 +02:00
|
|
|
|
|
|
|
this.route({
|
|
|
|
method: 'delete',
|
|
|
|
path: PATH_DEPENDENCIES,
|
|
|
|
handler: this.deleteFeatureDependencies,
|
2023-10-04 09:27:14 +02:00
|
|
|
permission: UPDATE_FEATURE_DEPENDENCY,
|
2023-09-26 09:38:34 +02:00
|
|
|
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),
|
|
|
|
},
|
|
|
|
}),
|
|
|
|
],
|
|
|
|
});
|
2023-09-26 14:03:24 +02:00
|
|
|
|
|
|
|
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),
|
|
|
|
},
|
|
|
|
}),
|
|
|
|
],
|
|
|
|
});
|
2023-09-19 11:23:21 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
async addFeatureDependency(
|
|
|
|
req: IAuthRequest<FeatureParams, any, CreateDependentFeatureSchema>,
|
|
|
|
res: Response,
|
|
|
|
): Promise<void> {
|
2023-09-29 14:02:15 +02:00
|
|
|
const { child, projectId } = req.params;
|
2023-09-19 11:23:21 +02:00
|
|
|
const { variants, enabled, feature } = req.body;
|
|
|
|
|
|
|
|
if (this.config.flagResolver.isEnabled('dependentFeatures')) {
|
2023-09-25 10:12:32 +02:00
|
|
|
await this.startTransaction(async (tx) =>
|
|
|
|
this.transactionalDependentFeaturesService(
|
|
|
|
tx,
|
2023-09-29 14:02:15 +02:00
|
|
|
).upsertFeatureDependency(
|
|
|
|
{ child, projectId },
|
|
|
|
{
|
|
|
|
variants,
|
|
|
|
enabled,
|
|
|
|
feature,
|
|
|
|
},
|
2023-10-10 09:25:03 +02:00
|
|
|
req.user,
|
2023-09-29 14:02:15 +02:00
|
|
|
),
|
2023-09-19 11:23:21 +02:00
|
|
|
);
|
|
|
|
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,
|
2023-10-10 09:25:03 +02:00
|
|
|
req.user,
|
2023-09-29 14:02:15 +02:00
|
|
|
);
|
2023-09-25 15:50:05 +02:00
|
|
|
res.status(200).end();
|
|
|
|
} else {
|
|
|
|
throw new InvalidOperationError(
|
|
|
|
'Dependent features are not enabled',
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
2023-09-26 09:38:34 +02:00
|
|
|
|
|
|
|
async deleteFeatureDependencies(
|
|
|
|
req: IAuthRequest<FeatureParams, any, any>,
|
|
|
|
res: Response,
|
|
|
|
): Promise<void> {
|
2023-09-29 14:02:15 +02:00
|
|
|
const { child, projectId } = req.params;
|
2023-09-26 09:38:34 +02:00
|
|
|
|
|
|
|
if (this.config.flagResolver.isEnabled('dependentFeatures')) {
|
|
|
|
await this.dependentFeaturesService.deleteFeatureDependencies(
|
|
|
|
child,
|
2023-09-29 14:02:15 +02:00
|
|
|
projectId,
|
2023-10-10 09:25:03 +02:00
|
|
|
req.user,
|
2023-09-26 09:38:34 +02:00
|
|
|
);
|
|
|
|
res.status(200).end();
|
|
|
|
} else {
|
|
|
|
throw new InvalidOperationError(
|
|
|
|
'Dependent features are not enabled',
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
2023-09-26 14:03:24 +02:00
|
|
|
|
|
|
|
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',
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
2023-09-19 11:23:21 +02:00
|
|
|
}
|