Created
September 9, 2024 04:51
-
-
Save isaackogan/17aa2a1340284fd88d8981d675768665 to your computer and use it in GitHub Desktop.
Remove private routes from an openapi spec. This will ONLY add refs that are *used* by the whitelisted routes, and hide protected routes / components
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
import {OpenAPISpec} from "../../@types/express/openapi"; | |
function addAllowedPathsToPrivateSpec( | |
publicSpec: Partial<OpenAPISpec>, | |
privateSpec: OpenAPISpec, | |
allowedPaths: string[] | |
): void { | |
publicSpec.paths ||= {}; | |
for (const path in privateSpec.paths) { | |
if (allowedPaths.includes(path)) { | |
publicSpec.paths[path] = privateSpec.paths[path]; | |
} | |
} | |
} | |
function recursivelyExtractRefsFromObject( | |
obj: Record<string, any>, | |
refs: Set<string> = new Set<string>() | |
): Set<string> { | |
for (const key in obj) { | |
if (key === "$ref") { | |
refs.add(obj[key]); | |
} else if (typeof obj[key] === "object") { | |
recursivelyExtractRefsFromObject(obj[key], refs); | |
} | |
} | |
return refs; | |
} | |
function cleanRefName(ref: string): string { | |
return ref.split("/").pop() as string; | |
} | |
class SetWithPop<T> extends Set<T> { | |
pop(): T { | |
const value = this.values().next().value; | |
this.delete(value); | |
return value; | |
} | |
} | |
function addRefsToSpec( | |
publicSpec: Partial<OpenAPISpec>, | |
privateSpec: OpenAPISpec, | |
baseRefs: Set<string> | |
) { | |
let addedRefs = new Set<string>(); | |
let refs = new SetWithPop<string>(baseRefs); | |
publicSpec.components = {schemas: {}}; | |
while (refs.size > 0) { | |
const ref = refs.pop(); | |
const refName = cleanRefName(ref); | |
publicSpec.components.schemas[refName] = privateSpec.components.schemas[refName]; | |
const newRefs = recursivelyExtractRefsFromObject(publicSpec.components.schemas); | |
newRefs.forEach(newRef => { | |
if (!addedRefs.has(newRef)) { | |
refs.add(newRef); | |
addedRefs.add(newRef); | |
} | |
}) | |
} | |
} | |
export function createSpecWithAllowedRoutes( | |
privateSpec: OpenAPISpec, | |
allowedRoutes: string[], | |
): Partial<OpenAPISpec> { | |
const publicSpec: Partial<OpenAPISpec> = {}; | |
publicSpec.components = {schemas: {}}; | |
addAllowedPathsToPrivateSpec(publicSpec, privateSpec, allowedRoutes); | |
const baseRefs = recursivelyExtractRefsFromObject(publicSpec); | |
addRefsToSpec(publicSpec, privateSpec, baseRefs); | |
return publicSpec; | |
} |
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment
Types: