1
0
mirror of https://github.com/Unleash/unleash.git synced 2025-01-06 00:07:44 +01:00
unleash.unleash/src/lib/routes/edge-api/index.ts
Christopher Kolstad 6cc51b7ba0
docs: Added docs for edge endpoints (#3501)
Adding documentation for the edge endpoints. Also separating request and
response schema for our validate endpoint to make clear that we expect a
list of strings as input, but yield tokens as output.

---------

Co-authored-by: Gastón Fournier <gaston@getunleash.io>
Co-authored-by: Thomas Heartman <thomas@getunleash.ai>
2023-04-14 15:39:39 +02:00

140 lines
5.2 KiB
TypeScript

import { Response } from 'express';
import Controller from '../controller';
import { IUnleashConfig, IUnleashServices } from '../../types';
import { Logger } from '../../logger';
import { NONE } from '../../types/permissions';
import { createResponseSchema } from '../../openapi/util/create-response-schema';
import { IAuthRequest, RequestBody } from '../unleash-types';
import { createRequestSchema } from '../../openapi/util/create-request-schema';
import {
validatedEdgeTokensSchema,
ValidatedEdgeTokensSchema,
} from '../../openapi/spec/validated-edge-tokens-schema';
import ClientInstanceService from '../../services/client-metrics/instance-service';
import EdgeService from '../../services/edge-service';
import { OpenApiService } from '../../services/openapi-service';
import {
emptyResponse,
getStandardResponses,
} from '../../openapi/util/standard-responses';
import { BulkMetricsSchema } from '../../openapi/spec/bulk-metrics-schema';
import ClientMetricsServiceV2 from '../../services/client-metrics/metrics-service-v2';
import { clientMetricsEnvBulkSchema } from '../../services/client-metrics/schema';
import { TokenStringListSchema } from '../../openapi';
export default class EdgeController extends Controller {
private readonly logger: Logger;
private edgeService: EdgeService;
private openApiService: OpenApiService;
private metricsV2: ClientMetricsServiceV2;
private clientInstanceService: ClientInstanceService;
constructor(
config: IUnleashConfig,
{
edgeService,
openApiService,
clientMetricsServiceV2,
clientInstanceService,
}: Pick<
IUnleashServices,
| 'edgeService'
| 'openApiService'
| 'clientMetricsServiceV2'
| 'clientInstanceService'
>,
) {
super(config);
this.logger = config.getLogger('edge-api/index.ts');
this.edgeService = edgeService;
this.openApiService = openApiService;
this.metricsV2 = clientMetricsServiceV2;
this.clientInstanceService = clientInstanceService;
this.route({
method: 'post',
path: '/validate',
handler: this.getValidTokens,
permission: NONE,
middleware: [
this.openApiService.validPath({
tags: ['Edge'],
summary: 'Check which tokens are valid',
description:
'This operation accepts a list of tokens to validate. Unleash will validate each token you provide. For each valid token you provide, Unleash will return the token along with its type and which projects it has access to.',
operationId: 'getValidTokens',
requestBody: createRequestSchema('tokenStringListSchema'),
responses: {
200: createResponseSchema('validatedEdgeTokensSchema'),
...getStandardResponses(400, 413, 415),
},
}),
],
});
this.route({
method: 'post',
path: '/metrics',
handler: this.bulkMetrics,
permission: NONE, // should have a permission but not bound to any environment
middleware: [
this.openApiService.validPath({
tags: ['Edge'],
summary: 'Send metrics from Edge',
description: `This operation accepts batched metrics from Edge. Metrics will be inserted into Unleash's metrics storage`,
operationId: 'bulkMetrics',
requestBody: createRequestSchema('bulkMetricsSchema'),
responses: {
202: emptyResponse,
...getStandardResponses(400, 413, 415),
},
}),
],
});
}
async getValidTokens(
req: RequestBody<TokenStringListSchema>,
res: Response<ValidatedEdgeTokensSchema>,
): Promise<void> {
const tokens = await this.edgeService.getValidTokens(req.body.tokens);
this.openApiService.respondWithValidation<ValidatedEdgeTokensSchema>(
200,
res,
validatedEdgeTokensSchema.$id,
tokens,
);
}
async bulkMetrics(
req: IAuthRequest<void, void, BulkMetricsSchema>,
res: Response<void>,
): Promise<void> {
const { body, ip: clientIp } = req;
const { metrics, applications } = body;
try {
let promises: Promise<void>[] = [];
for (const app of applications) {
promises.push(
this.clientInstanceService.registerClient(app, clientIp),
);
}
if (metrics && metrics.length > 0) {
const data = await clientMetricsEnvBulkSchema.validateAsync(
metrics,
);
promises.push(this.metricsV2.registerBulkMetrics(data));
}
await Promise.all(promises);
res.status(202).end();
} catch (e) {
res.status(400).end();
}
}
}