@redocly/openapi-core
Version:
See https://github.com/Redocly/redocly-cli
90 lines (85 loc) • 2.89 kB
text/typescript
import type { Oas3Rule, Oas2Rule } from '../../visitors';
import type { Location } from '../../ref-utils';
import type { UserContext } from '../../walk';
import type {
Oas2Definition,
Oas2Operation,
Oas2PathItem,
Oas2SecurityScheme,
} from '../../typings/swagger';
import type {
Oas3Definition,
Oas3_1Definition,
Oas3Operation,
Oas3PathItem,
Oas3SecurityScheme,
} from '../../typings/openapi';
export const SecurityDefined: Oas3Rule | Oas2Rule = (opts: {
exceptions?: { path: string; methods?: string[] }[];
}) => {
const referencedSchemes = new Map<
string,
{
defined?: boolean;
from: Location[];
}
>();
const operationsWithoutSecurity: Location[] = [];
let eachOperationHasSecurity: boolean = true;
let path: string | undefined;
return {
Root: {
leave(root: Oas2Definition | Oas3Definition | Oas3_1Definition, { report }: UserContext) {
for (const [name, scheme] of referencedSchemes.entries()) {
if (scheme.defined) continue;
for (const reportedFromLocation of scheme.from) {
report({
message: `There is no \`${name}\` security scheme defined.`,
location: reportedFromLocation.key(),
});
}
}
if (root.security || eachOperationHasSecurity) {
return;
} else {
for (const operationLocation of operationsWithoutSecurity) {
report({
message: `Every operation should have security defined on it or on the root level.`,
location: operationLocation.key(),
});
}
}
},
},
SecurityScheme(_securityScheme: Oas2SecurityScheme | Oas3SecurityScheme, { key }: UserContext) {
referencedSchemes.set(key.toString(), { defined: true, from: [] });
},
SecurityRequirement(requirements, { location }) {
for (const requirement of Object.keys(requirements)) {
const authScheme = referencedSchemes.get(requirement);
const requirementLocation = location.child([requirement]);
if (!authScheme) {
referencedSchemes.set(requirement, { from: [requirementLocation] });
} else {
authScheme.from.push(requirementLocation);
}
}
},
PathItem: {
enter(pathItem: Oas2PathItem | Oas3PathItem, { key }: UserContext) {
path = key as string;
},
Operation(operation: Oas2Operation | Oas3Operation, { location, key }: UserContext) {
const isException = opts.exceptions?.some(
(item) =>
item.path === path &&
(!item.methods || item.methods?.some((method) => method.toLowerCase() === key))
);
if (!operation?.security && !isException) {
eachOperationHasSecurity = false;
operationsWithoutSecurity.push(location);
}
},
},
};
};