import { OpenAPIV3 } from 'openapi-types'; import { addonCreateUpdateSchema, addonParameterSchema, addonSchema, addonsSchema, addonTypeSchema, adminCountSchema, adminFeaturesQuerySchema, apiTokenSchema, apiTokensSchema, applicationSchema, applicationsSchema, batchFeaturesSchema, bulkToggleFeaturesSchema, changePasswordSchema, clientApplicationSchema, clientFeatureSchema, clientFeaturesQuerySchema, clientFeaturesSchema, clientMetricsSchema, cloneFeatureSchema, constraintSchema, contextFieldSchema, contextFieldsSchema, createApiTokenSchema, createFeatureSchema, createFeatureStrategySchema, createInvitedUserSchema, createUserSchema, dateSchema, edgeTokenSchema, emailSchema, environmentProjectSchema, environmentSchema, environmentsProjectSchema, environmentsSchema, eventSchema, eventsSchema, exportQuerySchema, exportResultSchema, featureEnvironmentMetricsSchema, featureEnvironmentSchema, featureEventsSchema, featureMetricsSchema, featureSchema, featuresSchema, featureStrategySchema, featureStrategySegmentSchema, featureTagSchema, featureTypeSchema, featureTypesSchema, featureUsageSchema, featureVariantsSchema, feedbackSchema, groupSchema, groupsSchema, groupUserModelSchema, healthCheckSchema, healthOverviewSchema, healthReportSchema, idSchema, importTogglesSchema, importTogglesValidateItemSchema, importTogglesValidateSchema, instanceAdminStatsSchema, legalValueSchema, loginSchema, meSchema, nameSchema, overrideSchema, parametersSchema, passwordSchema, patchesSchema, patchSchema, patSchema, patsSchema, permissionSchema, playgroundConstraintSchema, playgroundFeatureSchema, playgroundRequestSchema, playgroundResponseSchema, playgroundSegmentSchema, playgroundStrategySchema, profileSchema, projectEnvironmentSchema, projectOverviewSchema, projectSchema, projectsSchema, projectStatsSchema, proxyClientSchema, proxyFeatureSchema, proxyFeaturesSchema, publicSignupTokenCreateSchema, publicSignupTokenSchema, publicSignupTokensSchema, publicSignupTokenUpdateSchema, pushVariantsSchema, requestsPerSecondSchema, requestsPerSecondSegmentedSchema, resetPasswordSchema, roleSchema, sdkContextSchema, searchEventsSchema, segmentSchema, setStrategySortOrderSchema, setUiConfigSchema, sortOrderSchema, splashSchema, stateSchema, strategiesSchema, strategySchema, tagsBulkAddSchema, tagSchema, tagsSchema, tagTypeSchema, tagTypesSchema, tagWithVersionSchema, tokenStringListSchema, tokenUserSchema, uiConfigSchema, updateApiTokenSchema, updateFeatureSchema, updateFeatureStrategySchema, updateTagTypeSchema, updateUserSchema, upsertContextFieldSchema, upsertSegmentSchema, upsertStrategySchema, userSchema, usersGroupsBaseSchema, usersSchema, usersSearchSchema, validatedEdgeTokensSchema, validatePasswordSchema, validateTagTypeSchema, variantSchema, variantFlagSchema, variantsSchema, versionSchema, } from './spec'; import { IServerOption } from '../types'; import { mapValues, omitKeys } from '../util'; import { openApiTags } from './util'; import { URL } from 'url'; import apiVersion from '../util/version'; import { maintenanceSchema } from './spec/maintenance-schema'; import { toggleMaintenanceSchema } from './spec/toggle-maintenance-schema'; import { bulkRegistrationSchema } from './spec/bulk-registration-schema'; import { bulkMetricsSchema } from './spec/bulk-metrics-schema'; import { clientMetricsEnvSchema } from './spec/client-metrics-env-schema'; import { updateTagsSchema } from './spec/update-tags-schema'; import { batchStaleSchema } from './spec/batch-stale-schema'; import { createApplicationSchema } from './spec/create-application-schema'; // Schemas must have an $id property on the form "#/components/schemas/mySchema". export type SchemaId = typeof schemas[keyof typeof schemas]['$id']; // Schemas must list all their $refs in `components`, including nested schemas. export type SchemaRef = typeof schemas[keyof typeof schemas]['components']; // JSON schema properties that should not be included in the OpenAPI spec. export interface JsonSchemaProps { $id: string; components: object; } type SchemaWithMandatoryFields = Partial< Omit< OpenAPIV3.ReferenceObject | OpenAPIV3.SchemaObject, '$id' | 'components' > > & JsonSchemaProps; interface UnleashSchemas { [name: string]: SchemaWithMandatoryFields; } interface OpenAPIV3DocumentWithServers extends OpenAPIV3.Document { servers: OpenAPIV3.ServerObject[]; } // All schemas in `openapi/spec` should be listed here. export const schemas: UnleashSchemas = { adminCountSchema, adminFeaturesQuerySchema, addonParameterSchema, addonSchema, addonCreateUpdateSchema, addonsSchema, addonTypeSchema, apiTokenSchema, apiTokensSchema, applicationSchema, applicationsSchema, batchFeaturesSchema, batchStaleSchema, bulkRegistrationSchema, bulkMetricsSchema, bulkToggleFeaturesSchema, changePasswordSchema, clientApplicationSchema, clientFeatureSchema, clientFeaturesQuerySchema, clientFeaturesSchema, clientMetricsSchema, clientMetricsEnvSchema, cloneFeatureSchema, constraintSchema, contextFieldSchema, contextFieldsSchema, createApiTokenSchema, createApplicationSchema, createFeatureSchema, createFeatureStrategySchema, createInvitedUserSchema, createUserSchema, dateSchema, edgeTokenSchema, emailSchema, environmentSchema, environmentProjectSchema, environmentsSchema, environmentsProjectSchema, eventSchema, eventsSchema, exportResultSchema, exportQuerySchema, featureEnvironmentMetricsSchema, featureEnvironmentSchema, featureEventsSchema, featureMetricsSchema, featureSchema, featuresSchema, featureStrategySchema, featureStrategySegmentSchema, featureTagSchema, featureTypeSchema, featureTypesSchema, featureUsageSchema, featureVariantsSchema, feedbackSchema, groupSchema, groupsSchema, groupUserModelSchema, healthCheckSchema, healthOverviewSchema, healthReportSchema, idSchema, instanceAdminStatsSchema, legalValueSchema, loginSchema, maintenanceSchema, toggleMaintenanceSchema, meSchema, nameSchema, overrideSchema, parametersSchema, passwordSchema, patchesSchema, patchSchema, patSchema, patsSchema, permissionSchema, playgroundConstraintSchema, playgroundFeatureSchema, playgroundRequestSchema, playgroundResponseSchema, playgroundSegmentSchema, playgroundStrategySchema, profileSchema, projectEnvironmentSchema, projectSchema, projectsSchema, proxyClientSchema, proxyFeatureSchema, proxyFeaturesSchema, publicSignupTokenCreateSchema, publicSignupTokenSchema, publicSignupTokensSchema, publicSignupTokenUpdateSchema, pushVariantsSchema, projectStatsSchema, resetPasswordSchema, requestsPerSecondSchema, requestsPerSecondSegmentedSchema, roleSchema, sdkContextSchema, searchEventsSchema, segmentSchema, setStrategySortOrderSchema, setUiConfigSchema, sortOrderSchema, splashSchema, stateSchema, strategiesSchema, strategySchema, tagsBulkAddSchema, tagSchema, tagsSchema, tagTypeSchema, tagTypesSchema, tagWithVersionSchema, tokenUserSchema, tokenStringListSchema, uiConfigSchema, updateApiTokenSchema, updateFeatureSchema, updateFeatureStrategySchema, updateTagTypeSchema, updateUserSchema, updateTagsSchema, upsertContextFieldSchema, upsertSegmentSchema, upsertStrategySchema, userSchema, usersGroupsBaseSchema, usersSchema, usersSearchSchema, validatedEdgeTokensSchema, validatePasswordSchema, validateTagTypeSchema, variantSchema, variantFlagSchema, variantsSchema, versionSchema, projectOverviewSchema, importTogglesSchema, importTogglesValidateSchema, importTogglesValidateItemSchema, }; // Remove JSONSchema keys that would result in an invalid OpenAPI spec. export const removeJsonSchemaProps = ( schema: T, ): OpenAPIV3.SchemaObject => { return omitKeys(schema, '$id', 'components'); }; const findRootUrl: (unleashUrl: string, baseUriPath: string) => string = ( unleashUrl: string, baseUriPath?: string, ) => { if (!baseUriPath) { return unleashUrl; } const baseUrl = new URL(unleashUrl); const url = baseUrl.pathname.indexOf(baseUriPath) >= 0 ? `${baseUrl.protocol}//${baseUrl.host}` : baseUrl.toString(); return baseUriPath.startsWith('/') ? new URL(baseUriPath, url).toString() : url; }; export const createOpenApiSchema = ({ unleashUrl, baseUriPath, }: Pick): Omit< OpenAPIV3DocumentWithServers, 'paths' > => { const url = findRootUrl(unleashUrl, baseUriPath); return { openapi: '3.0.3', servers: baseUriPath ? [{ url }] : [], info: { title: 'Unleash API', version: apiVersion, }, security: [{ apiKey: [] }], components: { securitySchemes: { apiKey: { type: 'apiKey', in: 'header', name: 'Authorization', }, }, schemas: mapValues(schemas, removeJsonSchemaProps), }, tags: openApiTags, }; }; export * from './util'; export * from './spec';