moonflower 1.4.8 → 1.4.9

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -1,2 +1,2 @@
1
- "use strict";Object.defineProperty(exports,Symbol.toStringTag,{value:"Module"});const q=require("path"),h=require("ts-morph"),u=require("../../utils/logger.cjs"),S=require("../discoveryModule/discoverImports/discoverImports.cjs"),k=require("../discoveryModule/discoverRouterFiles/discoverRouterFiles.cjs"),K=require("../discoveryModule/discoverRouters/discoverRouters.cjs"),H=require("../manager/OpenApiManager.cjs"),I=require("./getSourceFileTimestamp.cjs"),M=require("./nodeParsers.cjs"),U=require("./parseEndpoint.cjs"),E=require("./parseExposedModels.cjs"),N=require("./sourceFileCache.cjs");function W(e){const r=Object.create(null,{[Symbol.toStringTag]:{value:"Module"}});if(e){for(const a in e)if(a!=="default"){const c=Object.getOwnPropertyDescriptor(e,a);Object.defineProperty(r,a,c.get?c:{enumerable:!0,get:()=>e[a]})}}return r.default=e,Object.freeze(r)}const x=W(q),_=({logLevel:e,tsconfigPath:r,sourceFilePaths:a,sourceFileDiscovery:c,incremental:s})=>{const t=H.OpenApiManager.getInstance();if(t.isReady())return;e&&u.Logger.setLevel(e),u.Logger.info("Preparing OpenAPI spec");const o=new h.Project({tsConfigFilePath:x.resolve(r),skipFileDependencyResolution:!0}),{explicitRouters:n,discoveredRouterFiles:p,allSourceFiles:f}=(()=>{const l=(a??[]).map(d=>x.resolve(d)).map(d=>o.getSourceFileOrThrow(d)),z=l.flatMap(d=>({fileName:d.getFilePath(),sourceFile:d,routers:K.discoverRouters(d)})),{discoveredRouterFiles:L,discoveredSourceFiles:j}=(()=>{if(c===!1)return{discoveredRouterFiles:[],discoveredSourceFiles:[]};const d=performance.now(),g=k.discoverRouterFiles({targetPath:typeof c=="object"?c.rootPath:".",tsConfigPath:r});return u.Logger.info(`File discovery took ${Math.round(performance.now()-d)}ms`),g})(),$=l.reduce((d,g)=>d.some(w=>w.getFilePath()===g.getFilePath())?d:d.concat(g),j);return{explicitRouters:z,discoveredRouterFiles:L,allSourceFiles:$}})(),T=n.reduce((i,m)=>i.some(l=>l.fileName===m.fileName)?i:i.concat(m),p),F=f.flatMap(i=>O(i)).filter(i=>!!i);F.length>0&&F[0]&&t.setHeader(F[0]);const v=f.flatMap(i=>R(i));t.setExposedModels(v);const b=typeof s=="object"&&s.cachePath?s.cachePath:x.resolve(process.cwd(),"node_modules",".cache","moonflower"),y=C(T,{cachePath:b,timestampCache:{}});t.setStats({discoveredRouterFiles:p.map(i=>({path:i.fileName,routers:i.routers.named.map(m=>({name:m,endpoints:y.filter(l=>l.sourceFilePath===i.fileName).map(l=>`${l.method.toUpperCase()} ${l.path}`)}))})),explicitRouterFiles:n.map(i=>({path:i.fileName,routers:i.routers.named.map(m=>({name:m,endpoints:y.filter(l=>l.sourceFilePath===i.fileName).map(l=>`${l.method.toUpperCase()} ${l.path}`)}))}))}),t.setEndpoints(y),t.markAsReady()},C=(e,r,a)=>{const c=performance.now(),s=e.map(t=>P(t,r));return u.Logger.info(`Router analysis took ${Math.round(performance.now()-c)}ms`),s.map((t,o)=>({fileName:e[o].fileName,timeTaken:t.timing})).sort((t,o)=>o.timeTaken-t.timeTaken).filter(t=>t.timeTaken>500).forEach(t=>{u.Logger.info(`- [${t.fileName}] Took ${Math.round(t.timeTaken)}ms to analyze`)}),s.flatMap(t=>t.endpoints)},P=(e,r,a)=>{const c=I.getSourceFileTimestamp(e.sourceFile,r.timestampCache),s=N.SourceFileCache.getCachedResults(e.sourceFile,c,r.cachePath);if(s)return u.Logger.debug(`[${e.fileName}] Found cached results`),{endpoints:s.endpoints,timing:0};u.Logger.debug(`[${e.fileName}] Analyzing...`);const t=performance.now(),o=A(e),n=performance.now();return u.Logger.debug(`[${e.fileName}] Analyzed in ${n-t}ms`),N.SourceFileCache.cacheResults(e.sourceFile,c,r.cachePath,o),{endpoints:o,timing:n-t}},A=(e,r)=>{const a=[],s=["get","post","put","delete","del","patch"].join("|");return e.routers.named.forEach(t=>{const o=new RegExp(`${t}\\.(?:${s})`);e.sourceFile.forEachChild(n=>{const p=n.getText();o.test(p)&&(M.resolveEndpointPath(n),a.push(U.parseEndpoint(n,e.fileName)))})}),a},O=e=>{const r=S.discoverImportedName({sourceFile:e,originalName:"useApiHeader"});if(!r)return null;const a=e.forEachChildAsArray().filter(o=>o.isKind(h.SyntaxKind.ExpressionStatement)).find(o=>r&&o.getText().startsWith(r));if(!a)return null;const c=a.getFirstDescendantByKindOrThrow(h.SyntaxKind.ObjectLiteralExpression),s=M.getValuesOfObjectLiteral(c),t=o=>typeof o=="string"||Array.isArray(o)&&o.every(n=>typeof n=="string")?o:o.reduce((n,p)=>typeof p=="string"?n:{...n,[p.identifier]:t(p.value)},{});return t(s)},R=e=>{const r=[],a=S.discoverImportedName({sourceFile:e,originalName:"useExposeApiModel"}),c=S.discoverImportedName({sourceFile:e,originalName:"useExposeNamedApiModels"});return e.forEachChildAsArray().filter(s=>s.isKind(h.SyntaxKind.ExpressionStatement)).map(s=>{if(a&&s.getText().startsWith(a)){const n=(s.getFirstChild()?.getChildrenOfKind(h.SyntaxKind.SyntaxList)||[])[0].getFirstChild();if(!n)return;r.push(E.parseExposedModel(n));return}if(c&&s.getText().startsWith(c)){const n=(s.getFirstChild()?.getChildrenOfKind(h.SyntaxKind.SyntaxList)||[])[0].getFirstChild();if(!n)return;E.parseNamedExposedModels(n).forEach(f=>r.push(f))}}),r};exports.analyzeMultipleSourceFiles=C;exports.analyzeSourceFileApiHeader=O;exports.analyzeSourceFileEndpoints=A;exports.analyzeSourceFileExposedModels=R;exports.analyzeSourceFileWithCache=P;exports.prepareOpenApiSpec=_;
1
+ "use strict";Object.defineProperty(exports,Symbol.toStringTag,{value:"Module"});const q=require("path"),f=require("ts-morph"),u=require("../../utils/logger.cjs"),T=require("../discoveryModule/discoverImports/discoverImports.cjs"),K=require("../discoveryModule/discoverRouterFiles/discoverRouterFiles.cjs"),H=require("../discoveryModule/discoverRouters/discoverRouters.cjs"),I=require("../manager/OpenApiManager.cjs"),U=require("./getSourceFileTimestamp.cjs"),C=require("./nodeParsers.cjs"),W=require("./parseEndpoint.cjs"),N=require("./parseExposedModels.cjs"),M=require("./sourceFileCache.cjs");function _(t){const r=Object.create(null,{[Symbol.toStringTag]:{value:"Module"}});if(t){for(const c in t)if(c!=="default"){const i=Object.getOwnPropertyDescriptor(t,c);Object.defineProperty(r,c,i.get?i:{enumerable:!0,get:()=>t[c]})}}return r.default=t,Object.freeze(r)}const E=_(q),D=({logLevel:t,tsconfigPath:r,sourceFilePaths:c,sourceFileDiscovery:i,incremental:n,profiling:l="stats"})=>{const e=I.OpenApiManager.getInstance();if(e.isReady())return;t&&u.Logger.setLevel(t),u.Logger.info("Preparing OpenAPI spec");const o=new f.Project({tsConfigFilePath:E.resolve(r),skipFileDependencyResolution:!0}),{explicitRouters:s,discoveredRouterFiles:m,allSourceFiles:y}=(()=>{const d=(c??[]).map(p=>E.resolve(p)).map(p=>o.getSourceFileOrThrow(p)),b=d.flatMap(p=>({fileName:p.getFilePath(),sourceFile:p,routers:H.discoverRouters(p)})),{discoveredRouterFiles:z,discoveredSourceFiles:k}=(()=>{if(i===!1)return{discoveredRouterFiles:[],discoveredSourceFiles:[]};const p=performance.now(),x=K.discoverRouterFiles({targetPath:typeof i=="object"?i.rootPath:".",tsConfigPath:r});return l!=="off"&&u.Logger.info(`File discovery took ${Math.round(performance.now()-p)}ms`),x})(),w=d.reduce((p,x)=>p.some(j=>j.getFilePath()===x.getFilePath())?p:p.concat(x),k);return{explicitRouters:b,discoveredRouterFiles:z,allSourceFiles:w}})(),g=s.reduce((a,h)=>a.some(d=>d.fileName===h.fileName)?a:a.concat(h),m),F=y.flatMap(a=>R(a)).filter(a=>!!a);F.length>0&&F[0]&&e.setHeader(F[0]);const $=y.flatMap(a=>v(a));e.setExposedModels($);const L=typeof n=="object"&&n.cachePath?n.cachePath:E.resolve(process.cwd(),"node_modules",".cache","moonflower"),S=P(g,{cachePath:L,timestampCache:{},profiling:l});e.setStats({discoveredRouterFiles:m.map(a=>({path:a.fileName,routers:a.routers.named.map(h=>({name:h,endpoints:S.filter(d=>d.sourceFilePath===a.fileName).map(d=>`${d.method.toUpperCase()} ${d.path}`)}))})),explicitRouterFiles:s.map(a=>({path:a.fileName,routers:a.routers.named.map(h=>({name:h,endpoints:S.filter(d=>d.sourceFilePath===a.fileName).map(d=>`${d.method.toUpperCase()} ${d.path}`)}))}))}),e.setEndpoints(S),e.markAsReady()},P=(t,r,c)=>{const i=r.profiling??"stats",n=performance.now(),l=t.map(e=>A(e,r));return i!=="off"&&u.Logger.info(`Router analysis took ${Math.round(performance.now()-n)}ms`),i==="stats"?l.map((e,o)=>({fileName:t[o].fileName,timeTaken:e.timing})).sort((e,o)=>o.timeTaken-e.timeTaken).filter(e=>e.timeTaken>500).forEach(e=>{u.Logger.info(`- [${e.fileName}] Took ${Math.round(e.timeTaken)}ms to analyze`)}):i==="debug"&&l.map((e,o)=>({fileName:t[o].fileName,timeTaken:e.timing,endpointTimings:e.endpointTimings})).sort((e,o)=>o.timeTaken-e.timeTaken).forEach(e=>{u.Logger.info(`- [${e.fileName}] Took ${Math.round(e.timeTaken)}ms to analyze`),e.endpointTimings.sort((o,s)=>s.timing-o.timing).forEach(o=>{u.Logger.info(` - ${o.method} ${o.path} (${Math.round(o.timing)}ms)`),o.sectionTimings.filter(s=>s.timing>=1).sort((s,m)=>m.timing-s.timing).forEach(s=>{u.Logger.info(` - ${s.section}: ${Math.round(s.timing)}ms`)})})}),l.flatMap(e=>e.endpoints)},A=(t,r,c)=>{const i=U.getSourceFileTimestamp(t.sourceFile,r.timestampCache),n=M.SourceFileCache.getCachedResults(t.sourceFile,i,r.cachePath);if(n)return u.Logger.debug(`[${t.fileName}] Found cached results`),{endpoints:n.endpoints,timing:0,endpointTimings:[]};u.Logger.debug(`[${t.fileName}] Analyzing...`);const l=performance.now(),{endpoints:e,endpointTimings:o}=O(t),s=performance.now();return u.Logger.debug(`[${t.fileName}] Analyzed in ${s-l}ms`),M.SourceFileCache.cacheResults(t.sourceFile,i,r.cachePath,e),{endpoints:e,timing:s-l,endpointTimings:o}},O=(t,r)=>{const c=[],i=[],l=["get","post","put","delete","del","patch"].join("|");return t.routers.named.forEach(e=>{const o=new RegExp(`${e}\\.(?:${l})`);t.sourceFile.forEachChild(s=>{const m=s.getText();if(o.test(m)){C.resolveEndpointPath(s);const y=performance.now(),{endpoint:g,sectionTimings:F}=W.parseEndpoint(s,t.fileName);i.push({method:g.method,path:g.path,timing:performance.now()-y,sectionTimings:F}),c.push(g)}})}),{endpoints:c,endpointTimings:i}},R=t=>{const r=T.discoverImportedName({sourceFile:t,originalName:"useApiHeader"});if(!r)return null;const c=t.forEachChildAsArray().filter(e=>e.isKind(f.SyntaxKind.ExpressionStatement)).find(e=>r&&e.getText().startsWith(r));if(!c)return null;const i=c.getFirstDescendantByKindOrThrow(f.SyntaxKind.ObjectLiteralExpression),n=C.getValuesOfObjectLiteral(i),l=e=>typeof e=="string"||Array.isArray(e)&&e.every(o=>typeof o=="string")?e:e.reduce((o,s)=>typeof s=="string"?o:{...o,[s.identifier]:l(s.value)},{});return l(n)},v=t=>{const r=[],c=T.discoverImportedName({sourceFile:t,originalName:"useExposeApiModel"}),i=T.discoverImportedName({sourceFile:t,originalName:"useExposeNamedApiModels"});return t.forEachChildAsArray().filter(n=>n.isKind(f.SyntaxKind.ExpressionStatement)).map(n=>{if(c&&n.getText().startsWith(c)){const o=(n.getFirstChild()?.getChildrenOfKind(f.SyntaxKind.SyntaxList)||[])[0].getFirstChild();if(!o)return;r.push(N.parseExposedModel(o));return}if(i&&n.getText().startsWith(i)){const o=(n.getFirstChild()?.getChildrenOfKind(f.SyntaxKind.SyntaxList)||[])[0].getFirstChild();if(!o)return;N.parseNamedExposedModels(o).forEach(m=>r.push(m))}}),r};exports.analyzeMultipleSourceFiles=P;exports.analyzeSourceFileApiHeader=R;exports.analyzeSourceFileEndpoints=O;exports.analyzeSourceFileExposedModels=v;exports.analyzeSourceFileWithCache=A;exports.prepareOpenApiSpec=D;
2
2
  //# sourceMappingURL=analyzerModule.cjs.map
@@ -1 +1 @@
1
- {"version":3,"file":"analyzerModule.cjs","sources":["../../../src/openapi/analyzerModule/analyzerModule.ts"],"sourcesContent":["import * as path from 'path'\nimport { SourceFile, SyntaxKind } from 'ts-morph'\nimport { Project } from 'ts-morph'\n\nimport { Logger } from '../../utils/logger'\nimport { discoverImportedName } from '../discoveryModule/discoverImports/discoverImports'\nimport {\n\tDiscoveredSourceFile,\n\tdiscoverRouterFiles,\n} from '../discoveryModule/discoverRouterFiles/discoverRouterFiles'\nimport { discoverRouters } from '../discoveryModule/discoverRouters/discoverRouters'\nimport { ApiDocsHeader, OpenApiManager } from '../manager/OpenApiManager'\nimport { EndpointData, ExposedModelData } from '../types'\nimport { getSourceFileTimestamp, TimestampCache } from './getSourceFileTimestamp'\nimport { getValuesOfObjectLiteral, resolveEndpointPath } from './nodeParsers'\nimport { parseEndpoint } from './parseEndpoint'\nimport { parseExposedModel, parseNamedExposedModels } from './parseExposedModels'\nimport { SourceFileCache } from './sourceFileCache'\n\ntype Props = {\n\tlogLevel?: Parameters<(typeof Logger)['setLevel']>[0]\n\ttsconfigPath: string\n\tsourceFilePaths?: string[]\n\tsourceFileDiscovery?: boolean | FileDiscoveryConfig\n\tincremental?:\n\t\t| boolean\n\t\t| {\n\t\t\t\tcachePath: string\n\t\t }\n}\n\ntype FileDiscoveryConfig = {\n\trootPath: string\n}\n\n/**\n * @param tsconfigPath Path to tsconfig file relative to project root\n * @param sourceFilePaths Array of router source files relative to project root\n */\nexport const prepareOpenApiSpec = ({\n\tlogLevel,\n\ttsconfigPath,\n\tsourceFilePaths,\n\tsourceFileDiscovery,\n\tincremental,\n}: Props) => {\n\tconst openApiManager = OpenApiManager.getInstance()\n\n\tif (openApiManager.isReady()) {\n\t\treturn\n\t}\n\n\tif (logLevel) {\n\t\tLogger.setLevel(logLevel)\n\t}\n\n\tLogger.info('Preparing OpenAPI spec')\n\n\tconst project = new Project({\n\t\ttsConfigFilePath: path.resolve(tsconfigPath),\n\t\tskipFileDependencyResolution: true,\n\t})\n\n\tconst { explicitRouters, discoveredRouterFiles, allSourceFiles } = (() => {\n\t\tconst sourceFilesToAdd = sourceFilePaths ?? []\n\t\tconst resolvedSourceFilePaths = sourceFilesToAdd.map((filepath) => path.resolve(filepath))\n\t\tconst sourceFiles = resolvedSourceFilePaths.map((filePath) => project.getSourceFileOrThrow(filePath))\n\t\tconst explicitRouters = sourceFiles.flatMap((file) => ({\n\t\t\tfileName: file.getFilePath(),\n\t\t\tsourceFile: file,\n\t\t\trouters: discoverRouters(file),\n\t\t}))\n\n\t\tconst { discoveredRouterFiles, discoveredSourceFiles } = (() => {\n\t\t\tif (sourceFileDiscovery === false) {\n\t\t\t\treturn { discoveredRouterFiles: [], discoveredSourceFiles: [] }\n\t\t\t}\n\n\t\t\tconst startTime = performance.now()\n\t\t\tconst files = discoverRouterFiles({\n\t\t\t\ttargetPath: typeof sourceFileDiscovery === 'object' ? sourceFileDiscovery.rootPath : '.',\n\t\t\t\ttsConfigPath: tsconfigPath,\n\t\t\t})\n\t\t\tLogger.info(`File discovery took ${Math.round(performance.now() - startTime)}ms`)\n\t\t\treturn files\n\t\t})()\n\n\t\tconst allSourceFiles = sourceFiles.reduce(\n\t\t\t(acc, current) =>\n\t\t\t\tacc.some((r) => r.getFilePath() === current.getFilePath()) ? acc : acc.concat(current),\n\t\t\tdiscoveredSourceFiles,\n\t\t)\n\n\t\treturn { explicitRouters, discoveredRouterFiles, allSourceFiles }\n\t})()\n\n\tconst filesToAnalyze = explicitRouters.reduce(\n\t\t(acc, current) => (acc.some((r) => r.fileName === current.fileName) ? acc : acc.concat(current)),\n\t\tdiscoveredRouterFiles,\n\t)\n\n\tconst apiHeaders = allSourceFiles\n\t\t.flatMap((file) => analyzeSourceFileApiHeader(file))\n\t\t.filter((headers) => !!headers)\n\tif (apiHeaders.length > 0 && apiHeaders[0]) {\n\t\topenApiManager.setHeader(apiHeaders[0])\n\t}\n\n\tconst exposedModels = allSourceFiles.flatMap((file) => analyzeSourceFileExposedModels(file))\n\n\topenApiManager.setExposedModels(exposedModels)\n\n\tconst cachePath = (() => {\n\t\tif (typeof incremental === 'object' && incremental.cachePath) {\n\t\t\treturn incremental.cachePath\n\t\t}\n\t\treturn path.resolve(process.cwd(), 'node_modules', '.cache', 'moonflower')\n\t})()\n\tconst endpoints = analyzeMultipleSourceFiles(filesToAnalyze, {\n\t\tincremental: incremental !== false,\n\t\tcachePath,\n\t\ttimestampCache: {},\n\t})\n\n\topenApiManager.setStats({\n\t\tdiscoveredRouterFiles: discoveredRouterFiles.map((file) => ({\n\t\t\tpath: file.fileName,\n\t\t\trouters: file.routers.named.map((r) => ({\n\t\t\t\tname: r,\n\t\t\t\tendpoints: endpoints\n\t\t\t\t\t.filter((e) => e.sourceFilePath === file.fileName)\n\t\t\t\t\t.map((e) => `${e.method.toUpperCase()} ${e.path}`),\n\t\t\t})),\n\t\t})),\n\t\texplicitRouterFiles: explicitRouters.map((file) => ({\n\t\t\tpath: file.fileName,\n\t\t\trouters: file.routers.named.map((r) => ({\n\t\t\t\tname: r,\n\t\t\t\tendpoints: endpoints\n\t\t\t\t\t.filter((e) => e.sourceFilePath === file.fileName)\n\t\t\t\t\t.map((e) => `${e.method.toUpperCase()} ${e.path}`),\n\t\t\t})),\n\t\t})),\n\t})\n\n\topenApiManager.setEndpoints(endpoints)\n\topenApiManager.markAsReady()\n}\n\nexport const analyzeMultipleSourceFiles = (\n\tfiles: DiscoveredSourceFile[],\n\tconfig: {\n\t\tincremental: boolean\n\t\tcachePath: string\n\t\ttimestampCache: TimestampCache\n\t},\n\tfilterEndpointPaths?: string[],\n): EndpointData[] => {\n\tconst startTime = performance.now()\n\tconst analyzedFiles = files.map((file) => analyzeSourceFileWithCache(file, config, filterEndpointPaths))\n\tLogger.info(`Router analysis took ${Math.round(performance.now() - startTime)}ms`)\n\n\tanalyzedFiles\n\t\t.map((f, index) => ({\n\t\t\tfileName: files[index].fileName,\n\t\t\ttimeTaken: f.timing,\n\t\t}))\n\t\t.sort((a, b) => b.timeTaken - a.timeTaken)\n\t\t.filter((t) => t.timeTaken > 500)\n\t\t.forEach((t) => {\n\t\t\tLogger.info(`- [${t.fileName}] Took ${Math.round(t.timeTaken)}ms to analyze`)\n\t\t})\n\n\treturn analyzedFiles.flatMap((f) => f.endpoints)\n}\n\nexport const analyzeSourceFileWithCache = (\n\tfile: DiscoveredSourceFile,\n\tconfig: {\n\t\tincremental: boolean\n\t\tcachePath: string\n\t\ttimestampCache: TimestampCache\n\t},\n\tfilterEndpointPaths?: string[],\n): { endpoints: EndpointData[]; timing: number } => {\n\tconst timestamp = getSourceFileTimestamp(file.sourceFile, config.timestampCache)\n\tconst cachedResults = SourceFileCache.getCachedResults(file.sourceFile, timestamp, config.cachePath)\n\n\tif (cachedResults) {\n\t\tLogger.debug(`[${file.fileName}] Found cached results`)\n\t\treturn { endpoints: cachedResults.endpoints, timing: 0 }\n\t}\n\tLogger.debug(`[${file.fileName}] Analyzing...`)\n\n\tconst t1 = performance.now()\n\tconst endpoints = analyzeSourceFileEndpoints(file, filterEndpointPaths)\n\tconst t2 = performance.now()\n\tLogger.debug(`[${file.fileName}] Analyzed in ${t2 - t1}ms`)\n\tSourceFileCache.cacheResults(file.sourceFile, timestamp, config.cachePath, endpoints)\n\treturn { endpoints, timing: t2 - t1 }\n}\n\nexport const analyzeSourceFileEndpoints = (\n\tfile: DiscoveredSourceFile,\n\tfilterEndpointPaths?: string[],\n): EndpointData[] => {\n\tconst endpoints: EndpointData[] = []\n\tconst operations = ['get', 'post', 'put', 'delete', 'del', 'patch']\n\tconst joinedOperations = operations.join('|')\n\n\tfile.routers.named.forEach((routerName) => {\n\t\tconst routerPattern = new RegExp(`${routerName}\\\\.(?:${joinedOperations})`)\n\t\tfile.sourceFile.forEachChild((node) => {\n\t\t\tconst nodeText = node.getText()\n\n\t\t\tif (routerPattern.test(nodeText)) {\n\t\t\t\tconst endpointPath = resolveEndpointPath(node) ?? ''\n\n\t\t\t\tif (filterEndpointPaths && !filterEndpointPaths.some((path) => endpointPath.includes(path))) {\n\t\t\t\t\treturn\n\t\t\t\t}\n\n\t\t\t\tendpoints.push(parseEndpoint(node, file.fileName))\n\t\t\t}\n\t\t})\n\t})\n\n\treturn endpoints\n}\n\nexport const analyzeSourceFileApiHeader = (sourceFile: SourceFile): ApiDocsHeader | null => {\n\tconst nameOfUseApiHeader = discoverImportedName({\n\t\tsourceFile,\n\t\toriginalName: 'useApiHeader',\n\t})\n\n\tif (!nameOfUseApiHeader) {\n\t\treturn null\n\t}\n\n\tconst node = sourceFile\n\t\t.forEachChildAsArray()\n\t\t.filter((node) => node.isKind(SyntaxKind.ExpressionStatement))\n\t\t.find((node) => nameOfUseApiHeader && node.getText().startsWith(nameOfUseApiHeader))\n\n\tif (!node) {\n\t\treturn null\n\t}\n\n\tconst targetNode = node.getFirstDescendantByKindOrThrow(SyntaxKind.ObjectLiteralExpression)\n\tconst values = getValuesOfObjectLiteral(targetNode)\n\n\tconst collapseObject = (v: string | string[] | typeof values): any => {\n\t\tif (typeof v === 'string') {\n\t\t\treturn v\n\t\t}\n\t\tif (Array.isArray(v) && v.every((value) => typeof value === 'string')) {\n\t\t\treturn v\n\t\t}\n\n\t\treturn v.reduce((acc, current) => {\n\t\t\tif (typeof current === 'string') {\n\t\t\t\treturn acc\n\t\t\t}\n\t\t\treturn {\n\t\t\t\t...acc,\n\t\t\t\t[current.identifier]: collapseObject(current.value as string[]),\n\t\t\t}\n\t\t}, {})\n\t}\n\treturn collapseObject(values)\n}\n\nexport const analyzeSourceFileExposedModels = (sourceFile: SourceFile): ExposedModelData[] => {\n\tconst models: ExposedModelData[] = []\n\n\tconst nameOfUseExposeApiModel = discoverImportedName({\n\t\tsourceFile,\n\t\toriginalName: 'useExposeApiModel',\n\t})\n\n\tconst nameOfUseExposeNamedApiModels = discoverImportedName({\n\t\tsourceFile,\n\t\toriginalName: 'useExposeNamedApiModels',\n\t})\n\n\tsourceFile\n\t\t.forEachChildAsArray()\n\t\t.filter((node) => node.isKind(SyntaxKind.ExpressionStatement))\n\t\t.map((node) => {\n\t\t\tif (nameOfUseExposeApiModel && node.getText().startsWith(nameOfUseExposeApiModel)) {\n\t\t\t\tconst callExpressionNode = node.getFirstChild()\n\t\t\t\tconst syntaxListChildren = callExpressionNode?.getChildrenOfKind(SyntaxKind.SyntaxList) || []\n\n\t\t\t\tconst firstChild = syntaxListChildren[0].getFirstChild()\n\t\t\t\tif (!firstChild) {\n\t\t\t\t\treturn\n\t\t\t\t}\n\n\t\t\t\tmodels.push(parseExposedModel(firstChild))\n\t\t\t\treturn\n\t\t\t}\n\n\t\t\tif (nameOfUseExposeNamedApiModels && node.getText().startsWith(nameOfUseExposeNamedApiModels)) {\n\t\t\t\tconst callExpressionNode = node.getFirstChild()\n\t\t\t\tconst syntaxListChildren = callExpressionNode?.getChildrenOfKind(SyntaxKind.SyntaxList) || []\n\n\t\t\t\tconst firstChild = syntaxListChildren[0].getFirstChild()\n\t\t\t\tif (!firstChild) {\n\t\t\t\t\treturn\n\t\t\t\t}\n\n\t\t\t\tconst parsedModels = parseNamedExposedModels(firstChild)\n\t\t\t\tparsedModels.forEach((model) => models.push(model))\n\t\t\t}\n\t\t})\n\treturn models\n}\n"],"names":["prepareOpenApiSpec","logLevel","tsconfigPath","sourceFilePaths","sourceFileDiscovery","incremental","openApiManager","OpenApiManager","Logger","project","Project","path","explicitRouters","discoveredRouterFiles","allSourceFiles","sourceFiles","filepath","filePath","file","discoverRouters","discoveredSourceFiles","startTime","files","discoverRouterFiles","acc","current","r","filesToAnalyze","apiHeaders","analyzeSourceFileApiHeader","headers","exposedModels","analyzeSourceFileExposedModels","cachePath","endpoints","analyzeMultipleSourceFiles","e","config","filterEndpointPaths","analyzedFiles","analyzeSourceFileWithCache","f","index","a","b","timestamp","getSourceFileTimestamp","cachedResults","SourceFileCache","t1","analyzeSourceFileEndpoints","t2","joinedOperations","routerName","routerPattern","node","nodeText","resolveEndpointPath","parseEndpoint","sourceFile","nameOfUseApiHeader","discoverImportedName","SyntaxKind","targetNode","values","getValuesOfObjectLiteral","collapseObject","v","value","models","nameOfUseExposeApiModel","nameOfUseExposeNamedApiModels","firstChild","parseExposedModel","parseNamedExposedModels","model"],"mappings":"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"}
1
+ {"version":3,"file":"analyzerModule.cjs","sources":["../../../src/openapi/analyzerModule/analyzerModule.ts"],"sourcesContent":["import * as path from 'path'\nimport { SourceFile, SyntaxKind } from 'ts-morph'\nimport { Project } from 'ts-morph'\n\nimport { Logger } from '../../utils/logger'\nimport { discoverImportedName } from '../discoveryModule/discoverImports/discoverImports'\nimport {\n\tDiscoveredSourceFile,\n\tdiscoverRouterFiles,\n} from '../discoveryModule/discoverRouterFiles/discoverRouterFiles'\nimport { discoverRouters } from '../discoveryModule/discoverRouters/discoverRouters'\nimport { ApiDocsHeader, OpenApiManager } from '../manager/OpenApiManager'\nimport { EndpointData, ExposedModelData } from '../types'\nimport { getSourceFileTimestamp, TimestampCache } from './getSourceFileTimestamp'\nimport { getValuesOfObjectLiteral, resolveEndpointPath } from './nodeParsers'\nimport { parseEndpoint, SectionTiming } from './parseEndpoint'\nimport { parseExposedModel, parseNamedExposedModels } from './parseExposedModels'\nimport { SourceFileCache } from './sourceFileCache'\n\ntype Props = {\n\tlogLevel?: Parameters<(typeof Logger)['setLevel']>[0]\n\ttsconfigPath: string\n\tsourceFilePaths?: string[]\n\tsourceFileDiscovery?: boolean | FileDiscoveryConfig\n\tincremental?:\n\t\t| boolean\n\t\t| {\n\t\t\t\tcachePath: string\n\t\t }\n\tprofiling?: 'stats' | 'off' | 'debug'\n}\n\ntype FileDiscoveryConfig = {\n\trootPath: string\n}\n\ntype EndpointTiming = { method: string; path: string; timing: number; sectionTimings: SectionTiming[] }\n\n/**\n * @param tsconfigPath Path to tsconfig file relative to project root\n * @param sourceFilePaths Array of router source files relative to project root\n */\nexport const prepareOpenApiSpec = ({\n\tlogLevel,\n\ttsconfigPath,\n\tsourceFilePaths,\n\tsourceFileDiscovery,\n\tincremental,\n\tprofiling = 'stats',\n}: Props) => {\n\tconst openApiManager = OpenApiManager.getInstance()\n\n\tif (openApiManager.isReady()) {\n\t\treturn\n\t}\n\n\tif (logLevel) {\n\t\tLogger.setLevel(logLevel)\n\t}\n\n\tLogger.info('Preparing OpenAPI spec')\n\n\tconst project = new Project({\n\t\ttsConfigFilePath: path.resolve(tsconfigPath),\n\t\tskipFileDependencyResolution: true,\n\t})\n\n\tconst { explicitRouters, discoveredRouterFiles, allSourceFiles } = (() => {\n\t\tconst sourceFilesToAdd = sourceFilePaths ?? []\n\t\tconst resolvedSourceFilePaths = sourceFilesToAdd.map((filepath) => path.resolve(filepath))\n\t\tconst sourceFiles = resolvedSourceFilePaths.map((filePath) => project.getSourceFileOrThrow(filePath))\n\t\tconst explicitRouters = sourceFiles.flatMap((file) => ({\n\t\t\tfileName: file.getFilePath(),\n\t\t\tsourceFile: file,\n\t\t\trouters: discoverRouters(file),\n\t\t}))\n\n\t\tconst { discoveredRouterFiles, discoveredSourceFiles } = (() => {\n\t\t\tif (sourceFileDiscovery === false) {\n\t\t\t\treturn { discoveredRouterFiles: [], discoveredSourceFiles: [] }\n\t\t\t}\n\n\t\t\tconst startTime = performance.now()\n\t\t\tconst files = discoverRouterFiles({\n\t\t\t\ttargetPath: typeof sourceFileDiscovery === 'object' ? sourceFileDiscovery.rootPath : '.',\n\t\t\t\ttsConfigPath: tsconfigPath,\n\t\t\t})\n\t\t\tif (profiling !== 'off') {\n\t\t\t\tLogger.info(`File discovery took ${Math.round(performance.now() - startTime)}ms`)\n\t\t\t}\n\t\t\treturn files\n\t\t})()\n\n\t\tconst allSourceFiles = sourceFiles.reduce(\n\t\t\t(acc, current) =>\n\t\t\t\tacc.some((r) => r.getFilePath() === current.getFilePath()) ? acc : acc.concat(current),\n\t\t\tdiscoveredSourceFiles,\n\t\t)\n\n\t\treturn { explicitRouters, discoveredRouterFiles, allSourceFiles }\n\t})()\n\n\tconst filesToAnalyze = explicitRouters.reduce(\n\t\t(acc, current) => (acc.some((r) => r.fileName === current.fileName) ? acc : acc.concat(current)),\n\t\tdiscoveredRouterFiles,\n\t)\n\n\tconst apiHeaders = allSourceFiles\n\t\t.flatMap((file) => analyzeSourceFileApiHeader(file))\n\t\t.filter((headers) => !!headers)\n\tif (apiHeaders.length > 0 && apiHeaders[0]) {\n\t\topenApiManager.setHeader(apiHeaders[0])\n\t}\n\n\tconst exposedModels = allSourceFiles.flatMap((file) => analyzeSourceFileExposedModels(file))\n\n\topenApiManager.setExposedModels(exposedModels)\n\n\tconst cachePath = (() => {\n\t\tif (typeof incremental === 'object' && incremental.cachePath) {\n\t\t\treturn incremental.cachePath\n\t\t}\n\t\treturn path.resolve(process.cwd(), 'node_modules', '.cache', 'moonflower')\n\t})()\n\tconst endpoints = analyzeMultipleSourceFiles(filesToAnalyze, {\n\t\tincremental: incremental !== false,\n\t\tcachePath,\n\t\ttimestampCache: {},\n\t\tprofiling,\n\t})\n\n\topenApiManager.setStats({\n\t\tdiscoveredRouterFiles: discoveredRouterFiles.map((file) => ({\n\t\t\tpath: file.fileName,\n\t\t\trouters: file.routers.named.map((r) => ({\n\t\t\t\tname: r,\n\t\t\t\tendpoints: endpoints\n\t\t\t\t\t.filter((e) => e.sourceFilePath === file.fileName)\n\t\t\t\t\t.map((e) => `${e.method.toUpperCase()} ${e.path}`),\n\t\t\t})),\n\t\t})),\n\t\texplicitRouterFiles: explicitRouters.map((file) => ({\n\t\t\tpath: file.fileName,\n\t\t\trouters: file.routers.named.map((r) => ({\n\t\t\t\tname: r,\n\t\t\t\tendpoints: endpoints\n\t\t\t\t\t.filter((e) => e.sourceFilePath === file.fileName)\n\t\t\t\t\t.map((e) => `${e.method.toUpperCase()} ${e.path}`),\n\t\t\t})),\n\t\t})),\n\t})\n\n\topenApiManager.setEndpoints(endpoints)\n\topenApiManager.markAsReady()\n}\n\nexport const analyzeMultipleSourceFiles = (\n\tfiles: DiscoveredSourceFile[],\n\tconfig: {\n\t\tincremental: boolean\n\t\tcachePath: string\n\t\ttimestampCache: TimestampCache\n\t\tprofiling?: 'stats' | 'off' | 'debug'\n\t},\n\tfilterEndpointPaths?: string[],\n): EndpointData[] => {\n\tconst profiling = config.profiling ?? 'stats'\n\tconst startTime = performance.now()\n\tconst analyzedFiles = files.map((file) => analyzeSourceFileWithCache(file, config, filterEndpointPaths))\n\n\tif (profiling !== 'off') {\n\t\tLogger.info(`Router analysis took ${Math.round(performance.now() - startTime)}ms`)\n\t}\n\n\tif (profiling === 'stats') {\n\t\tanalyzedFiles\n\t\t\t.map((f, index) => ({ fileName: files[index].fileName, timeTaken: f.timing }))\n\t\t\t.sort((a, b) => b.timeTaken - a.timeTaken)\n\t\t\t.filter((t) => t.timeTaken > 500)\n\t\t\t.forEach((t) => {\n\t\t\t\tLogger.info(`- [${t.fileName}] Took ${Math.round(t.timeTaken)}ms to analyze`)\n\t\t\t})\n\t} else if (profiling === 'debug') {\n\t\tanalyzedFiles\n\t\t\t.map((f, index) => ({\n\t\t\t\tfileName: files[index].fileName,\n\t\t\t\ttimeTaken: f.timing,\n\t\t\t\tendpointTimings: f.endpointTimings,\n\t\t\t}))\n\t\t\t.sort((a, b) => b.timeTaken - a.timeTaken)\n\t\t\t.forEach((t) => {\n\t\t\t\tLogger.info(`- [${t.fileName}] Took ${Math.round(t.timeTaken)}ms to analyze`)\n\t\t\t\tt.endpointTimings\n\t\t\t\t\t.sort((a, b) => b.timing - a.timing)\n\t\t\t\t\t.forEach((ep) => {\n\t\t\t\t\t\tLogger.info(` - ${ep.method} ${ep.path} (${Math.round(ep.timing)}ms)`)\n\t\t\t\t\t\tep.sectionTimings\n\t\t\t\t\t\t\t.filter((s) => s.timing >= 1)\n\t\t\t\t\t\t\t.sort((a, b) => b.timing - a.timing)\n\t\t\t\t\t\t\t.forEach((s) => {\n\t\t\t\t\t\t\t\tLogger.info(` - ${s.section}: ${Math.round(s.timing)}ms`)\n\t\t\t\t\t\t\t})\n\t\t\t\t\t})\n\t\t\t})\n\t}\n\n\treturn analyzedFiles.flatMap((f) => f.endpoints)\n}\n\nexport const analyzeSourceFileWithCache = (\n\tfile: DiscoveredSourceFile,\n\tconfig: {\n\t\tincremental: boolean\n\t\tcachePath: string\n\t\ttimestampCache: TimestampCache\n\t\tprofiling?: 'stats' | 'off' | 'debug'\n\t},\n\tfilterEndpointPaths?: string[],\n): { endpoints: EndpointData[]; timing: number; endpointTimings: EndpointTiming[] } => {\n\tconst timestamp = getSourceFileTimestamp(file.sourceFile, config.timestampCache)\n\tconst cachedResults = SourceFileCache.getCachedResults(file.sourceFile, timestamp, config.cachePath)\n\n\tif (cachedResults) {\n\t\tLogger.debug(`[${file.fileName}] Found cached results`)\n\t\treturn { endpoints: cachedResults.endpoints, timing: 0, endpointTimings: [] }\n\t}\n\tLogger.debug(`[${file.fileName}] Analyzing...`)\n\n\tconst t1 = performance.now()\n\tconst { endpoints, endpointTimings } = analyzeSourceFileEndpoints(file, filterEndpointPaths)\n\tconst t2 = performance.now()\n\tLogger.debug(`[${file.fileName}] Analyzed in ${t2 - t1}ms`)\n\tSourceFileCache.cacheResults(file.sourceFile, timestamp, config.cachePath, endpoints)\n\treturn { endpoints, timing: t2 - t1, endpointTimings }\n}\n\nexport const analyzeSourceFileEndpoints = (\n\tfile: DiscoveredSourceFile,\n\tfilterEndpointPaths?: string[],\n): { endpoints: EndpointData[]; endpointTimings: EndpointTiming[] } => {\n\tconst endpoints: EndpointData[] = []\n\tconst endpointTimings: EndpointTiming[] = []\n\tconst operations = ['get', 'post', 'put', 'delete', 'del', 'patch']\n\tconst joinedOperations = operations.join('|')\n\n\tfile.routers.named.forEach((routerName) => {\n\t\tconst routerPattern = new RegExp(`${routerName}\\\\.(?:${joinedOperations})`)\n\t\tfile.sourceFile.forEachChild((node) => {\n\t\t\tconst nodeText = node.getText()\n\n\t\t\tif (routerPattern.test(nodeText)) {\n\t\t\t\tconst endpointPath = resolveEndpointPath(node) ?? ''\n\n\t\t\t\tif (filterEndpointPaths && !filterEndpointPaths.some((path) => endpointPath.includes(path))) {\n\t\t\t\t\treturn\n\t\t\t\t}\n\n\t\t\t\tconst t1 = performance.now()\n\t\t\t\tconst { endpoint, sectionTimings } = parseEndpoint(node, file.fileName)\n\t\t\t\tendpointTimings.push({\n\t\t\t\t\tmethod: endpoint.method,\n\t\t\t\t\tpath: endpoint.path,\n\t\t\t\t\ttiming: performance.now() - t1,\n\t\t\t\t\tsectionTimings,\n\t\t\t\t})\n\t\t\t\tendpoints.push(endpoint)\n\t\t\t}\n\t\t})\n\t})\n\n\treturn { endpoints, endpointTimings }\n}\n\nexport const analyzeSourceFileApiHeader = (sourceFile: SourceFile): ApiDocsHeader | null => {\n\tconst nameOfUseApiHeader = discoverImportedName({\n\t\tsourceFile,\n\t\toriginalName: 'useApiHeader',\n\t})\n\n\tif (!nameOfUseApiHeader) {\n\t\treturn null\n\t}\n\n\tconst node = sourceFile\n\t\t.forEachChildAsArray()\n\t\t.filter((node) => node.isKind(SyntaxKind.ExpressionStatement))\n\t\t.find((node) => nameOfUseApiHeader && node.getText().startsWith(nameOfUseApiHeader))\n\n\tif (!node) {\n\t\treturn null\n\t}\n\n\tconst targetNode = node.getFirstDescendantByKindOrThrow(SyntaxKind.ObjectLiteralExpression)\n\tconst values = getValuesOfObjectLiteral(targetNode)\n\n\tconst collapseObject = (v: string | string[] | typeof values): any => {\n\t\tif (typeof v === 'string') {\n\t\t\treturn v\n\t\t}\n\t\tif (Array.isArray(v) && v.every((value) => typeof value === 'string')) {\n\t\t\treturn v\n\t\t}\n\n\t\treturn v.reduce((acc, current) => {\n\t\t\tif (typeof current === 'string') {\n\t\t\t\treturn acc\n\t\t\t}\n\t\t\treturn {\n\t\t\t\t...acc,\n\t\t\t\t[current.identifier]: collapseObject(current.value as string[]),\n\t\t\t}\n\t\t}, {})\n\t}\n\treturn collapseObject(values)\n}\n\nexport const analyzeSourceFileExposedModels = (sourceFile: SourceFile): ExposedModelData[] => {\n\tconst models: ExposedModelData[] = []\n\n\tconst nameOfUseExposeApiModel = discoverImportedName({\n\t\tsourceFile,\n\t\toriginalName: 'useExposeApiModel',\n\t})\n\n\tconst nameOfUseExposeNamedApiModels = discoverImportedName({\n\t\tsourceFile,\n\t\toriginalName: 'useExposeNamedApiModels',\n\t})\n\n\tsourceFile\n\t\t.forEachChildAsArray()\n\t\t.filter((node) => node.isKind(SyntaxKind.ExpressionStatement))\n\t\t.map((node) => {\n\t\t\tif (nameOfUseExposeApiModel && node.getText().startsWith(nameOfUseExposeApiModel)) {\n\t\t\t\tconst callExpressionNode = node.getFirstChild()\n\t\t\t\tconst syntaxListChildren = callExpressionNode?.getChildrenOfKind(SyntaxKind.SyntaxList) || []\n\n\t\t\t\tconst firstChild = syntaxListChildren[0].getFirstChild()\n\t\t\t\tif (!firstChild) {\n\t\t\t\t\treturn\n\t\t\t\t}\n\n\t\t\t\tmodels.push(parseExposedModel(firstChild))\n\t\t\t\treturn\n\t\t\t}\n\n\t\t\tif (nameOfUseExposeNamedApiModels && node.getText().startsWith(nameOfUseExposeNamedApiModels)) {\n\t\t\t\tconst callExpressionNode = node.getFirstChild()\n\t\t\t\tconst syntaxListChildren = callExpressionNode?.getChildrenOfKind(SyntaxKind.SyntaxList) || []\n\n\t\t\t\tconst firstChild = syntaxListChildren[0].getFirstChild()\n\t\t\t\tif (!firstChild) {\n\t\t\t\t\treturn\n\t\t\t\t}\n\n\t\t\t\tconst parsedModels = parseNamedExposedModels(firstChild)\n\t\t\t\tparsedModels.forEach((model) => models.push(model))\n\t\t\t}\n\t\t})\n\treturn models\n}\n"],"names":["prepareOpenApiSpec","logLevel","tsconfigPath","sourceFilePaths","sourceFileDiscovery","incremental","profiling","openApiManager","OpenApiManager","Logger","project","Project","path","explicitRouters","discoveredRouterFiles","allSourceFiles","sourceFiles","filepath","filePath","file","discoverRouters","discoveredSourceFiles","startTime","files","discoverRouterFiles","acc","current","r","filesToAnalyze","apiHeaders","analyzeSourceFileApiHeader","headers","exposedModels","analyzeSourceFileExposedModels","cachePath","endpoints","analyzeMultipleSourceFiles","e","config","filterEndpointPaths","analyzedFiles","analyzeSourceFileWithCache","f","index","a","b","t","ep","timestamp","getSourceFileTimestamp","cachedResults","SourceFileCache","t1","endpointTimings","analyzeSourceFileEndpoints","t2","joinedOperations","routerName","routerPattern","node","nodeText","resolveEndpointPath","endpoint","sectionTimings","parseEndpoint","sourceFile","nameOfUseApiHeader","discoverImportedName","SyntaxKind","targetNode","values","getValuesOfObjectLiteral","collapseObject","v","value","models","nameOfUseExposeApiModel","nameOfUseExposeNamedApiModels","firstChild","parseExposedModel","parseNamedExposedModels","model"],"mappings":"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"}
@@ -4,6 +4,7 @@ import { DiscoveredSourceFile } from '../discoveryModule/discoverRouterFiles/dis
4
4
  import { ApiDocsHeader } from '../manager/OpenApiManager';
5
5
  import { EndpointData, ExposedModelData } from '../types';
6
6
  import { TimestampCache } from './getSourceFileTimestamp';
7
+ import { SectionTiming } from './parseEndpoint';
7
8
 
8
9
  type Props = {
9
10
  logLevel?: Parameters<(typeof Logger)['setLevel']>[0];
@@ -13,25 +14,38 @@ type Props = {
13
14
  incremental?: boolean | {
14
15
  cachePath: string;
15
16
  };
17
+ profiling?: 'stats' | 'off' | 'debug';
16
18
  };
17
19
  type FileDiscoveryConfig = {
18
20
  rootPath: string;
19
21
  };
20
- export declare const prepareOpenApiSpec: ({ logLevel, tsconfigPath, sourceFilePaths, sourceFileDiscovery, incremental, }: Props) => void;
22
+ type EndpointTiming = {
23
+ method: string;
24
+ path: string;
25
+ timing: number;
26
+ sectionTimings: SectionTiming[];
27
+ };
28
+ export declare const prepareOpenApiSpec: ({ logLevel, tsconfigPath, sourceFilePaths, sourceFileDiscovery, incremental, profiling, }: Props) => void;
21
29
  export declare const analyzeMultipleSourceFiles: (files: DiscoveredSourceFile[], config: {
22
30
  incremental: boolean;
23
31
  cachePath: string;
24
32
  timestampCache: TimestampCache;
33
+ profiling?: "stats" | "off" | "debug";
25
34
  }, filterEndpointPaths?: string[]) => EndpointData[];
26
35
  export declare const analyzeSourceFileWithCache: (file: DiscoveredSourceFile, config: {
27
36
  incremental: boolean;
28
37
  cachePath: string;
29
38
  timestampCache: TimestampCache;
39
+ profiling?: "stats" | "off" | "debug";
30
40
  }, filterEndpointPaths?: string[]) => {
31
41
  endpoints: EndpointData[];
32
42
  timing: number;
43
+ endpointTimings: EndpointTiming[];
44
+ };
45
+ export declare const analyzeSourceFileEndpoints: (file: DiscoveredSourceFile, filterEndpointPaths?: string[]) => {
46
+ endpoints: EndpointData[];
47
+ endpointTimings: EndpointTiming[];
33
48
  };
34
- export declare const analyzeSourceFileEndpoints: (file: DiscoveredSourceFile, filterEndpointPaths?: string[]) => EndpointData[];
35
49
  export declare const analyzeSourceFileApiHeader: (sourceFile: SourceFile) => ApiDocsHeader | null;
36
50
  export declare const analyzeSourceFileExposedModels: (sourceFile: SourceFile) => ExposedModelData[];
37
51
  export {};
@@ -1 +1 @@
1
- {"version":3,"file":"analyzerModule.d.ts","sourceRoot":"","sources":["../../../src/openapi/analyzerModule/analyzerModule.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,UAAU,EAAc,MAAM,UAAU,CAAA;AAGjD,OAAO,EAAE,MAAM,EAAE,MAAM,oBAAoB,CAAA;AAE3C,OAAO,EACN,oBAAoB,EAEpB,MAAM,4DAA4D,CAAA;AAEnE,OAAO,EAAE,aAAa,EAAkB,MAAM,2BAA2B,CAAA;AACzE,OAAO,EAAE,YAAY,EAAE,gBAAgB,EAAE,MAAM,UAAU,CAAA;AACzD,OAAO,EAA0B,cAAc,EAAE,MAAM,0BAA0B,CAAA;AAMjF,KAAK,KAAK,GAAG;IACZ,QAAQ,CAAC,EAAE,UAAU,CAAC,CAAC,OAAO,MAAM,CAAC,CAAC,UAAU,CAAC,CAAC,CAAC,CAAC,CAAC,CAAA;IACrD,YAAY,EAAE,MAAM,CAAA;IACpB,eAAe,CAAC,EAAE,MAAM,EAAE,CAAA;IAC1B,mBAAmB,CAAC,EAAE,OAAO,GAAG,mBAAmB,CAAA;IACnD,WAAW,CAAC,EACT,OAAO,GACP;QACA,SAAS,EAAE,MAAM,CAAA;KAChB,CAAA;CACJ,CAAA;AAED,KAAK,mBAAmB,GAAG;IAC1B,QAAQ,EAAE,MAAM,CAAA;CAChB,CAAA;AAMD,eAAO,MAAM,kBAAkB,mFAM5B,KAAK,SAsGP,CAAA;AAED,eAAO,MAAM,0BAA0B,UAC/B,oBAAoB,EAAE,UACrB;IACP,WAAW,EAAE,OAAO,CAAA;IACpB,SAAS,EAAE,MAAM,CAAA;IACjB,cAAc,EAAE,cAAc,CAAA;CAC9B,wBACqB,MAAM,EAAE,KAC5B,YAAY,EAiBd,CAAA;AAED,eAAO,MAAM,0BAA0B,SAChC,oBAAoB,UAClB;IACP,WAAW,EAAE,OAAO,CAAA;IACpB,SAAS,EAAE,MAAM,CAAA;IACjB,cAAc,EAAE,cAAc,CAAA;CAC9B,wBACqB,MAAM,EAAE,KAC5B;IAAE,SAAS,EAAE,YAAY,EAAE,CAAC;IAAC,MAAM,EAAE,MAAM,CAAA;CAgB7C,CAAA;AAED,eAAO,MAAM,0BAA0B,SAChC,oBAAoB,wBACJ,MAAM,EAAE,KAC5B,YAAY,EAuBd,CAAA;AAED,eAAO,MAAM,0BAA0B,eAAgB,UAAU,KAAG,aAAa,GAAG,IAyCnF,CAAA;AAED,eAAO,MAAM,8BAA8B,eAAgB,UAAU,KAAG,gBAAgB,EA4CvF,CAAA"}
1
+ {"version":3,"file":"analyzerModule.d.ts","sourceRoot":"","sources":["../../../src/openapi/analyzerModule/analyzerModule.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,UAAU,EAAc,MAAM,UAAU,CAAA;AAGjD,OAAO,EAAE,MAAM,EAAE,MAAM,oBAAoB,CAAA;AAE3C,OAAO,EACN,oBAAoB,EAEpB,MAAM,4DAA4D,CAAA;AAEnE,OAAO,EAAE,aAAa,EAAkB,MAAM,2BAA2B,CAAA;AACzE,OAAO,EAAE,YAAY,EAAE,gBAAgB,EAAE,MAAM,UAAU,CAAA;AACzD,OAAO,EAA0B,cAAc,EAAE,MAAM,0BAA0B,CAAA;AAEjF,OAAO,EAAiB,aAAa,EAAE,MAAM,iBAAiB,CAAA;AAI9D,KAAK,KAAK,GAAG;IACZ,QAAQ,CAAC,EAAE,UAAU,CAAC,CAAC,OAAO,MAAM,CAAC,CAAC,UAAU,CAAC,CAAC,CAAC,CAAC,CAAC,CAAA;IACrD,YAAY,EAAE,MAAM,CAAA;IACpB,eAAe,CAAC,EAAE,MAAM,EAAE,CAAA;IAC1B,mBAAmB,CAAC,EAAE,OAAO,GAAG,mBAAmB,CAAA;IACnD,WAAW,CAAC,EACT,OAAO,GACP;QACA,SAAS,EAAE,MAAM,CAAA;KAChB,CAAA;IACJ,SAAS,CAAC,EAAE,OAAO,GAAG,KAAK,GAAG,OAAO,CAAA;CACrC,CAAA;AAED,KAAK,mBAAmB,GAAG;IAC1B,QAAQ,EAAE,MAAM,CAAA;CAChB,CAAA;AAED,KAAK,cAAc,GAAG;IAAE,MAAM,EAAE,MAAM,CAAC;IAAC,IAAI,EAAE,MAAM,CAAC;IAAC,MAAM,EAAE,MAAM,CAAC;IAAC,cAAc,EAAE,aAAa,EAAE,CAAA;CAAE,CAAA;AAMvG,eAAO,MAAM,kBAAkB,8FAO5B,KAAK,SAyGP,CAAA;AAED,eAAO,MAAM,0BAA0B,UAC/B,oBAAoB,EAAE,UACrB;IACP,WAAW,EAAE,OAAO,CAAA;IACpB,SAAS,EAAE,MAAM,CAAA;IACjB,cAAc,EAAE,cAAc,CAAA;IAC9B,SAAS,CAAC,EAAE,OAAO,GAAG,KAAK,GAAG,OAAO,CAAA;CACrC,wBACqB,MAAM,EAAE,KAC5B,YAAY,EA0Cd,CAAA;AAED,eAAO,MAAM,0BAA0B,SAChC,oBAAoB,UAClB;IACP,WAAW,EAAE,OAAO,CAAA;IACpB,SAAS,EAAE,MAAM,CAAA;IACjB,cAAc,EAAE,cAAc,CAAA;IAC9B,SAAS,CAAC,EAAE,OAAO,GAAG,KAAK,GAAG,OAAO,CAAA;CACrC,wBACqB,MAAM,EAAE,KAC5B;IAAE,SAAS,EAAE,YAAY,EAAE,CAAC;IAAC,MAAM,EAAE,MAAM,CAAC;IAAC,eAAe,EAAE,cAAc,EAAE,CAAA;CAgBhF,CAAA;AAED,eAAO,MAAM,0BAA0B,SAChC,oBAAoB,wBACJ,MAAM,EAAE,KAC5B;IAAE,SAAS,EAAE,YAAY,EAAE,CAAC;IAAC,eAAe,EAAE,cAAc,EAAE,CAAA;CAgChE,CAAA;AAED,eAAO,MAAM,0BAA0B,eAAgB,UAAU,KAAG,aAAa,GAAG,IAyCnF,CAAA;AAED,eAAO,MAAM,8BAA8B,eAAgB,UAAU,KAAG,gBAAgB,EA4CvF,CAAA"}
@@ -1,143 +1,161 @@
1
- import * as y from "path";
2
- import { Project as v, SyntaxKind as h } from "ts-morph";
1
+ import * as T from "path";
2
+ import { Project as O, SyntaxKind as F } from "ts-morph";
3
3
  import { Logger as m } from "../../utils/logger.mjs";
4
- import { discoverImportedName as E } from "../discoveryModule/discoverImports/discoverImports.mjs";
5
- import { discoverRouterFiles as $ } from "../discoveryModule/discoverRouterFiles/discoverRouterFiles.mjs";
4
+ import { discoverImportedName as N } from "../discoveryModule/discoverImports/discoverImports.mjs";
5
+ import { discoverRouterFiles as v } from "../discoveryModule/discoverRouterFiles/discoverRouterFiles.mjs";
6
6
  import { discoverRouters as w } from "../discoveryModule/discoverRouters/discoverRouters.mjs";
7
- import { OpenApiManager as k } from "../manager/OpenApiManager.mjs";
8
- import { getSourceFileTimestamp as z } from "./getSourceFileTimestamp.mjs";
7
+ import { OpenApiManager as z } from "../manager/OpenApiManager.mjs";
8
+ import { getSourceFileTimestamp as b } from "./getSourceFileTimestamp.mjs";
9
9
  import { getValuesOfObjectLiteral as j, resolveEndpointPath as L } from "./nodeParsers.mjs";
10
- import { parseEndpoint as b } from "./parseEndpoint.mjs";
11
- import { parseExposedModel as K, parseNamedExposedModels as H } from "./parseExposedModels.mjs";
12
- import { SourceFileCache as N } from "./sourceFileCache.mjs";
13
- const oe = ({
14
- logLevel: t,
15
- tsconfigPath: n,
10
+ import { parseEndpoint as K } from "./parseEndpoint.mjs";
11
+ import { parseExposedModel as H, parseNamedExposedModels as U } from "./parseExposedModels.mjs";
12
+ import { SourceFileCache as C } from "./sourceFileCache.mjs";
13
+ const se = ({
14
+ logLevel: o,
15
+ tsconfigPath: i,
16
16
  sourceFilePaths: c,
17
- sourceFileDiscovery: a,
18
- incremental: s
17
+ sourceFileDiscovery: n,
18
+ incremental: r,
19
+ profiling: l = "stats"
19
20
  }) => {
20
- const e = k.getInstance();
21
+ const e = z.getInstance();
21
22
  if (e.isReady())
22
23
  return;
23
- t && m.setLevel(t), m.info("Preparing OpenAPI spec");
24
- const o = new v({
25
- tsConfigFilePath: y.resolve(n),
24
+ o && m.setLevel(o), m.info("Preparing OpenAPI spec");
25
+ const t = new O({
26
+ tsConfigFilePath: T.resolve(i),
26
27
  skipFileDependencyResolution: !0
27
- }), { explicitRouters: r, discoveredRouterFiles: d, allSourceFiles: f } = (() => {
28
- const l = (c ?? []).map((p) => y.resolve(p)).map((p) => o.getSourceFileOrThrow(p)), S = l.flatMap((p) => ({
29
- fileName: p.getFilePath(),
30
- sourceFile: p,
31
- routers: w(p)
32
- })), { discoveredRouterFiles: M, discoveredSourceFiles: R } = (() => {
33
- if (a === !1)
28
+ }), { explicitRouters: s, discoveredRouterFiles: u, allSourceFiles: x } = (() => {
29
+ const p = (c ?? []).map((d) => T.resolve(d)).map((d) => t.getSourceFileOrThrow(d)), P = p.flatMap((d) => ({
30
+ fileName: d.getFilePath(),
31
+ sourceFile: d,
32
+ routers: w(d)
33
+ })), { discoveredRouterFiles: S, discoveredSourceFiles: $ } = (() => {
34
+ if (n === !1)
34
35
  return { discoveredRouterFiles: [], discoveredSourceFiles: [] };
35
- const p = performance.now(), F = $({
36
- targetPath: typeof a == "object" ? a.rootPath : ".",
37
- tsConfigPath: n
36
+ const d = performance.now(), y = v({
37
+ targetPath: typeof n == "object" ? n.rootPath : ".",
38
+ tsConfigPath: i
38
39
  });
39
- return m.info(`File discovery took ${Math.round(performance.now() - p)}ms`), F;
40
- })(), T = l.reduce(
41
- (p, F) => p.some((O) => O.getFilePath() === F.getFilePath()) ? p : p.concat(F),
42
- R
40
+ return l !== "off" && m.info(`File discovery took ${Math.round(performance.now() - d)}ms`), y;
41
+ })(), R = p.reduce(
42
+ (d, y) => d.some((k) => k.getFilePath() === y.getFilePath()) ? d : d.concat(y),
43
+ $
43
44
  );
44
- return { explicitRouters: S, discoveredRouterFiles: M, allSourceFiles: T };
45
- })(), C = r.reduce(
46
- (i, u) => i.some((l) => l.fileName === u.fileName) ? i : i.concat(u),
47
- d
48
- ), g = f.flatMap((i) => B(i)).filter((i) => !!i);
45
+ return { explicitRouters: P, discoveredRouterFiles: S, allSourceFiles: R };
46
+ })(), h = s.reduce(
47
+ (a, f) => a.some((p) => p.fileName === f.fileName) ? a : a.concat(f),
48
+ u
49
+ ), g = x.flatMap((a) => V(a)).filter((a) => !!a);
49
50
  g.length > 0 && g[0] && e.setHeader(g[0]);
50
- const A = f.flatMap((i) => V(i));
51
- e.setExposedModels(A);
52
- const P = typeof s == "object" && s.cachePath ? s.cachePath : y.resolve(process.cwd(), "node_modules", ".cache", "moonflower"), x = U(C, {
53
- cachePath: P,
54
- timestampCache: {}
51
+ const M = x.flatMap((a) => _(a));
52
+ e.setExposedModels(M);
53
+ const A = typeof r == "object" && r.cachePath ? r.cachePath : T.resolve(process.cwd(), "node_modules", ".cache", "moonflower"), E = W(h, {
54
+ cachePath: A,
55
+ timestampCache: {},
56
+ profiling: l
55
57
  });
56
58
  e.setStats({
57
- discoveredRouterFiles: d.map((i) => ({
58
- path: i.fileName,
59
- routers: i.routers.named.map((u) => ({
60
- name: u,
61
- endpoints: x.filter((l) => l.sourceFilePath === i.fileName).map((l) => `${l.method.toUpperCase()} ${l.path}`)
59
+ discoveredRouterFiles: u.map((a) => ({
60
+ path: a.fileName,
61
+ routers: a.routers.named.map((f) => ({
62
+ name: f,
63
+ endpoints: E.filter((p) => p.sourceFilePath === a.fileName).map((p) => `${p.method.toUpperCase()} ${p.path}`)
62
64
  }))
63
65
  })),
64
- explicitRouterFiles: r.map((i) => ({
65
- path: i.fileName,
66
- routers: i.routers.named.map((u) => ({
67
- name: u,
68
- endpoints: x.filter((l) => l.sourceFilePath === i.fileName).map((l) => `${l.method.toUpperCase()} ${l.path}`)
66
+ explicitRouterFiles: s.map((a) => ({
67
+ path: a.fileName,
68
+ routers: a.routers.named.map((f) => ({
69
+ name: f,
70
+ endpoints: E.filter((p) => p.sourceFilePath === a.fileName).map((p) => `${p.method.toUpperCase()} ${p.path}`)
69
71
  }))
70
72
  }))
71
- }), e.setEndpoints(x), e.markAsReady();
72
- }, U = (t, n, c) => {
73
- const a = performance.now(), s = t.map((e) => W(e, n));
74
- return m.info(`Router analysis took ${Math.round(performance.now() - a)}ms`), s.map((e, o) => ({
75
- fileName: t[o].fileName,
76
- timeTaken: e.timing
77
- })).sort((e, o) => o.timeTaken - e.timeTaken).filter((e) => e.timeTaken > 500).forEach((e) => {
73
+ }), e.setEndpoints(E), e.markAsReady();
74
+ }, W = (o, i, c) => {
75
+ const n = i.profiling ?? "stats", r = performance.now(), l = o.map((e) => I(e, i));
76
+ return n !== "off" && m.info(`Router analysis took ${Math.round(performance.now() - r)}ms`), n === "stats" ? l.map((e, t) => ({ fileName: o[t].fileName, timeTaken: e.timing })).sort((e, t) => t.timeTaken - e.timeTaken).filter((e) => e.timeTaken > 500).forEach((e) => {
78
77
  m.info(`- [${e.fileName}] Took ${Math.round(e.timeTaken)}ms to analyze`);
79
- }), s.flatMap((e) => e.endpoints);
80
- }, W = (t, n, c) => {
81
- const a = z(t.sourceFile, n.timestampCache), s = N.getCachedResults(t.sourceFile, a, n.cachePath);
82
- if (s)
83
- return m.debug(`[${t.fileName}] Found cached results`), { endpoints: s.endpoints, timing: 0 };
84
- m.debug(`[${t.fileName}] Analyzing...`);
85
- const e = performance.now(), o = I(t), r = performance.now();
86
- return m.debug(`[${t.fileName}] Analyzed in ${r - e}ms`), N.cacheResults(t.sourceFile, a, n.cachePath, o), { endpoints: o, timing: r - e };
87
- }, I = (t, n) => {
88
- const c = [], s = ["get", "post", "put", "delete", "del", "patch"].join("|");
89
- return t.routers.named.forEach((e) => {
90
- const o = new RegExp(`${e}\\.(?:${s})`);
91
- t.sourceFile.forEachChild((r) => {
92
- const d = r.getText();
93
- o.test(d) && (L(r), c.push(b(r, t.fileName)));
78
+ }) : n === "debug" && l.map((e, t) => ({
79
+ fileName: o[t].fileName,
80
+ timeTaken: e.timing,
81
+ endpointTimings: e.endpointTimings
82
+ })).sort((e, t) => t.timeTaken - e.timeTaken).forEach((e) => {
83
+ m.info(`- [${e.fileName}] Took ${Math.round(e.timeTaken)}ms to analyze`), e.endpointTimings.sort((t, s) => s.timing - t.timing).forEach((t) => {
84
+ m.info(` - ${t.method} ${t.path} (${Math.round(t.timing)}ms)`), t.sectionTimings.filter((s) => s.timing >= 1).sort((s, u) => u.timing - s.timing).forEach((s) => {
85
+ m.info(` - ${s.section}: ${Math.round(s.timing)}ms`);
86
+ });
87
+ });
88
+ }), l.flatMap((e) => e.endpoints);
89
+ }, I = (o, i, c) => {
90
+ const n = b(o.sourceFile, i.timestampCache), r = C.getCachedResults(o.sourceFile, n, i.cachePath);
91
+ if (r)
92
+ return m.debug(`[${o.fileName}] Found cached results`), { endpoints: r.endpoints, timing: 0, endpointTimings: [] };
93
+ m.debug(`[${o.fileName}] Analyzing...`);
94
+ const l = performance.now(), { endpoints: e, endpointTimings: t } = B(o), s = performance.now();
95
+ return m.debug(`[${o.fileName}] Analyzed in ${s - l}ms`), C.cacheResults(o.sourceFile, n, i.cachePath, e), { endpoints: e, timing: s - l, endpointTimings: t };
96
+ }, B = (o, i) => {
97
+ const c = [], n = [], l = ["get", "post", "put", "delete", "del", "patch"].join("|");
98
+ return o.routers.named.forEach((e) => {
99
+ const t = new RegExp(`${e}\\.(?:${l})`);
100
+ o.sourceFile.forEachChild((s) => {
101
+ const u = s.getText();
102
+ if (t.test(u)) {
103
+ L(s);
104
+ const x = performance.now(), { endpoint: h, sectionTimings: g } = K(s, o.fileName);
105
+ n.push({
106
+ method: h.method,
107
+ path: h.path,
108
+ timing: performance.now() - x,
109
+ sectionTimings: g
110
+ }), c.push(h);
111
+ }
94
112
  });
95
- }), c;
96
- }, B = (t) => {
97
- const n = E({
98
- sourceFile: t,
113
+ }), { endpoints: c, endpointTimings: n };
114
+ }, V = (o) => {
115
+ const i = N({
116
+ sourceFile: o,
99
117
  originalName: "useApiHeader"
100
118
  });
101
- if (!n)
119
+ if (!i)
102
120
  return null;
103
- const c = t.forEachChildAsArray().filter((o) => o.isKind(h.ExpressionStatement)).find((o) => n && o.getText().startsWith(n));
121
+ const c = o.forEachChildAsArray().filter((e) => e.isKind(F.ExpressionStatement)).find((e) => i && e.getText().startsWith(i));
104
122
  if (!c)
105
123
  return null;
106
- const a = c.getFirstDescendantByKindOrThrow(h.ObjectLiteralExpression), s = j(a), e = (o) => typeof o == "string" || Array.isArray(o) && o.every((r) => typeof r == "string") ? o : o.reduce((r, d) => typeof d == "string" ? r : {
107
- ...r,
108
- [d.identifier]: e(d.value)
124
+ const n = c.getFirstDescendantByKindOrThrow(F.ObjectLiteralExpression), r = j(n), l = (e) => typeof e == "string" || Array.isArray(e) && e.every((t) => typeof t == "string") ? e : e.reduce((t, s) => typeof s == "string" ? t : {
125
+ ...t,
126
+ [s.identifier]: l(s.value)
109
127
  }, {});
110
- return e(s);
111
- }, V = (t) => {
112
- const n = [], c = E({
113
- sourceFile: t,
128
+ return l(r);
129
+ }, _ = (o) => {
130
+ const i = [], c = N({
131
+ sourceFile: o,
114
132
  originalName: "useExposeApiModel"
115
- }), a = E({
116
- sourceFile: t,
133
+ }), n = N({
134
+ sourceFile: o,
117
135
  originalName: "useExposeNamedApiModels"
118
136
  });
119
- return t.forEachChildAsArray().filter((s) => s.isKind(h.ExpressionStatement)).map((s) => {
120
- if (c && s.getText().startsWith(c)) {
121
- const r = (s.getFirstChild()?.getChildrenOfKind(h.SyntaxList) || [])[0].getFirstChild();
122
- if (!r)
137
+ return o.forEachChildAsArray().filter((r) => r.isKind(F.ExpressionStatement)).map((r) => {
138
+ if (c && r.getText().startsWith(c)) {
139
+ const t = (r.getFirstChild()?.getChildrenOfKind(F.SyntaxList) || [])[0].getFirstChild();
140
+ if (!t)
123
141
  return;
124
- n.push(K(r));
142
+ i.push(H(t));
125
143
  return;
126
144
  }
127
- if (a && s.getText().startsWith(a)) {
128
- const r = (s.getFirstChild()?.getChildrenOfKind(h.SyntaxList) || [])[0].getFirstChild();
129
- if (!r)
145
+ if (n && r.getText().startsWith(n)) {
146
+ const t = (r.getFirstChild()?.getChildrenOfKind(F.SyntaxList) || [])[0].getFirstChild();
147
+ if (!t)
130
148
  return;
131
- H(r).forEach((f) => n.push(f));
149
+ U(t).forEach((u) => i.push(u));
132
150
  }
133
- }), n;
151
+ }), i;
134
152
  };
135
153
  export {
136
- U as analyzeMultipleSourceFiles,
137
- B as analyzeSourceFileApiHeader,
138
- I as analyzeSourceFileEndpoints,
139
- V as analyzeSourceFileExposedModels,
140
- W as analyzeSourceFileWithCache,
141
- oe as prepareOpenApiSpec
154
+ W as analyzeMultipleSourceFiles,
155
+ V as analyzeSourceFileApiHeader,
156
+ B as analyzeSourceFileEndpoints,
157
+ _ as analyzeSourceFileExposedModels,
158
+ I as analyzeSourceFileWithCache,
159
+ se as prepareOpenApiSpec
142
160
  };
143
161
  //# sourceMappingURL=analyzerModule.mjs.map
@@ -1 +1 @@
1
- {"version":3,"file":"analyzerModule.mjs","sources":["../../../src/openapi/analyzerModule/analyzerModule.ts"],"sourcesContent":["import * as path from 'path'\nimport { SourceFile, SyntaxKind } from 'ts-morph'\nimport { Project } from 'ts-morph'\n\nimport { Logger } from '../../utils/logger'\nimport { discoverImportedName } from '../discoveryModule/discoverImports/discoverImports'\nimport {\n\tDiscoveredSourceFile,\n\tdiscoverRouterFiles,\n} from '../discoveryModule/discoverRouterFiles/discoverRouterFiles'\nimport { discoverRouters } from '../discoveryModule/discoverRouters/discoverRouters'\nimport { ApiDocsHeader, OpenApiManager } from '../manager/OpenApiManager'\nimport { EndpointData, ExposedModelData } from '../types'\nimport { getSourceFileTimestamp, TimestampCache } from './getSourceFileTimestamp'\nimport { getValuesOfObjectLiteral, resolveEndpointPath } from './nodeParsers'\nimport { parseEndpoint } from './parseEndpoint'\nimport { parseExposedModel, parseNamedExposedModels } from './parseExposedModels'\nimport { SourceFileCache } from './sourceFileCache'\n\ntype Props = {\n\tlogLevel?: Parameters<(typeof Logger)['setLevel']>[0]\n\ttsconfigPath: string\n\tsourceFilePaths?: string[]\n\tsourceFileDiscovery?: boolean | FileDiscoveryConfig\n\tincremental?:\n\t\t| boolean\n\t\t| {\n\t\t\t\tcachePath: string\n\t\t }\n}\n\ntype FileDiscoveryConfig = {\n\trootPath: string\n}\n\n/**\n * @param tsconfigPath Path to tsconfig file relative to project root\n * @param sourceFilePaths Array of router source files relative to project root\n */\nexport const prepareOpenApiSpec = ({\n\tlogLevel,\n\ttsconfigPath,\n\tsourceFilePaths,\n\tsourceFileDiscovery,\n\tincremental,\n}: Props) => {\n\tconst openApiManager = OpenApiManager.getInstance()\n\n\tif (openApiManager.isReady()) {\n\t\treturn\n\t}\n\n\tif (logLevel) {\n\t\tLogger.setLevel(logLevel)\n\t}\n\n\tLogger.info('Preparing OpenAPI spec')\n\n\tconst project = new Project({\n\t\ttsConfigFilePath: path.resolve(tsconfigPath),\n\t\tskipFileDependencyResolution: true,\n\t})\n\n\tconst { explicitRouters, discoveredRouterFiles, allSourceFiles } = (() => {\n\t\tconst sourceFilesToAdd = sourceFilePaths ?? []\n\t\tconst resolvedSourceFilePaths = sourceFilesToAdd.map((filepath) => path.resolve(filepath))\n\t\tconst sourceFiles = resolvedSourceFilePaths.map((filePath) => project.getSourceFileOrThrow(filePath))\n\t\tconst explicitRouters = sourceFiles.flatMap((file) => ({\n\t\t\tfileName: file.getFilePath(),\n\t\t\tsourceFile: file,\n\t\t\trouters: discoverRouters(file),\n\t\t}))\n\n\t\tconst { discoveredRouterFiles, discoveredSourceFiles } = (() => {\n\t\t\tif (sourceFileDiscovery === false) {\n\t\t\t\treturn { discoveredRouterFiles: [], discoveredSourceFiles: [] }\n\t\t\t}\n\n\t\t\tconst startTime = performance.now()\n\t\t\tconst files = discoverRouterFiles({\n\t\t\t\ttargetPath: typeof sourceFileDiscovery === 'object' ? sourceFileDiscovery.rootPath : '.',\n\t\t\t\ttsConfigPath: tsconfigPath,\n\t\t\t})\n\t\t\tLogger.info(`File discovery took ${Math.round(performance.now() - startTime)}ms`)\n\t\t\treturn files\n\t\t})()\n\n\t\tconst allSourceFiles = sourceFiles.reduce(\n\t\t\t(acc, current) =>\n\t\t\t\tacc.some((r) => r.getFilePath() === current.getFilePath()) ? acc : acc.concat(current),\n\t\t\tdiscoveredSourceFiles,\n\t\t)\n\n\t\treturn { explicitRouters, discoveredRouterFiles, allSourceFiles }\n\t})()\n\n\tconst filesToAnalyze = explicitRouters.reduce(\n\t\t(acc, current) => (acc.some((r) => r.fileName === current.fileName) ? acc : acc.concat(current)),\n\t\tdiscoveredRouterFiles,\n\t)\n\n\tconst apiHeaders = allSourceFiles\n\t\t.flatMap((file) => analyzeSourceFileApiHeader(file))\n\t\t.filter((headers) => !!headers)\n\tif (apiHeaders.length > 0 && apiHeaders[0]) {\n\t\topenApiManager.setHeader(apiHeaders[0])\n\t}\n\n\tconst exposedModels = allSourceFiles.flatMap((file) => analyzeSourceFileExposedModels(file))\n\n\topenApiManager.setExposedModels(exposedModels)\n\n\tconst cachePath = (() => {\n\t\tif (typeof incremental === 'object' && incremental.cachePath) {\n\t\t\treturn incremental.cachePath\n\t\t}\n\t\treturn path.resolve(process.cwd(), 'node_modules', '.cache', 'moonflower')\n\t})()\n\tconst endpoints = analyzeMultipleSourceFiles(filesToAnalyze, {\n\t\tincremental: incremental !== false,\n\t\tcachePath,\n\t\ttimestampCache: {},\n\t})\n\n\topenApiManager.setStats({\n\t\tdiscoveredRouterFiles: discoveredRouterFiles.map((file) => ({\n\t\t\tpath: file.fileName,\n\t\t\trouters: file.routers.named.map((r) => ({\n\t\t\t\tname: r,\n\t\t\t\tendpoints: endpoints\n\t\t\t\t\t.filter((e) => e.sourceFilePath === file.fileName)\n\t\t\t\t\t.map((e) => `${e.method.toUpperCase()} ${e.path}`),\n\t\t\t})),\n\t\t})),\n\t\texplicitRouterFiles: explicitRouters.map((file) => ({\n\t\t\tpath: file.fileName,\n\t\t\trouters: file.routers.named.map((r) => ({\n\t\t\t\tname: r,\n\t\t\t\tendpoints: endpoints\n\t\t\t\t\t.filter((e) => e.sourceFilePath === file.fileName)\n\t\t\t\t\t.map((e) => `${e.method.toUpperCase()} ${e.path}`),\n\t\t\t})),\n\t\t})),\n\t})\n\n\topenApiManager.setEndpoints(endpoints)\n\topenApiManager.markAsReady()\n}\n\nexport const analyzeMultipleSourceFiles = (\n\tfiles: DiscoveredSourceFile[],\n\tconfig: {\n\t\tincremental: boolean\n\t\tcachePath: string\n\t\ttimestampCache: TimestampCache\n\t},\n\tfilterEndpointPaths?: string[],\n): EndpointData[] => {\n\tconst startTime = performance.now()\n\tconst analyzedFiles = files.map((file) => analyzeSourceFileWithCache(file, config, filterEndpointPaths))\n\tLogger.info(`Router analysis took ${Math.round(performance.now() - startTime)}ms`)\n\n\tanalyzedFiles\n\t\t.map((f, index) => ({\n\t\t\tfileName: files[index].fileName,\n\t\t\ttimeTaken: f.timing,\n\t\t}))\n\t\t.sort((a, b) => b.timeTaken - a.timeTaken)\n\t\t.filter((t) => t.timeTaken > 500)\n\t\t.forEach((t) => {\n\t\t\tLogger.info(`- [${t.fileName}] Took ${Math.round(t.timeTaken)}ms to analyze`)\n\t\t})\n\n\treturn analyzedFiles.flatMap((f) => f.endpoints)\n}\n\nexport const analyzeSourceFileWithCache = (\n\tfile: DiscoveredSourceFile,\n\tconfig: {\n\t\tincremental: boolean\n\t\tcachePath: string\n\t\ttimestampCache: TimestampCache\n\t},\n\tfilterEndpointPaths?: string[],\n): { endpoints: EndpointData[]; timing: number } => {\n\tconst timestamp = getSourceFileTimestamp(file.sourceFile, config.timestampCache)\n\tconst cachedResults = SourceFileCache.getCachedResults(file.sourceFile, timestamp, config.cachePath)\n\n\tif (cachedResults) {\n\t\tLogger.debug(`[${file.fileName}] Found cached results`)\n\t\treturn { endpoints: cachedResults.endpoints, timing: 0 }\n\t}\n\tLogger.debug(`[${file.fileName}] Analyzing...`)\n\n\tconst t1 = performance.now()\n\tconst endpoints = analyzeSourceFileEndpoints(file, filterEndpointPaths)\n\tconst t2 = performance.now()\n\tLogger.debug(`[${file.fileName}] Analyzed in ${t2 - t1}ms`)\n\tSourceFileCache.cacheResults(file.sourceFile, timestamp, config.cachePath, endpoints)\n\treturn { endpoints, timing: t2 - t1 }\n}\n\nexport const analyzeSourceFileEndpoints = (\n\tfile: DiscoveredSourceFile,\n\tfilterEndpointPaths?: string[],\n): EndpointData[] => {\n\tconst endpoints: EndpointData[] = []\n\tconst operations = ['get', 'post', 'put', 'delete', 'del', 'patch']\n\tconst joinedOperations = operations.join('|')\n\n\tfile.routers.named.forEach((routerName) => {\n\t\tconst routerPattern = new RegExp(`${routerName}\\\\.(?:${joinedOperations})`)\n\t\tfile.sourceFile.forEachChild((node) => {\n\t\t\tconst nodeText = node.getText()\n\n\t\t\tif (routerPattern.test(nodeText)) {\n\t\t\t\tconst endpointPath = resolveEndpointPath(node) ?? ''\n\n\t\t\t\tif (filterEndpointPaths && !filterEndpointPaths.some((path) => endpointPath.includes(path))) {\n\t\t\t\t\treturn\n\t\t\t\t}\n\n\t\t\t\tendpoints.push(parseEndpoint(node, file.fileName))\n\t\t\t}\n\t\t})\n\t})\n\n\treturn endpoints\n}\n\nexport const analyzeSourceFileApiHeader = (sourceFile: SourceFile): ApiDocsHeader | null => {\n\tconst nameOfUseApiHeader = discoverImportedName({\n\t\tsourceFile,\n\t\toriginalName: 'useApiHeader',\n\t})\n\n\tif (!nameOfUseApiHeader) {\n\t\treturn null\n\t}\n\n\tconst node = sourceFile\n\t\t.forEachChildAsArray()\n\t\t.filter((node) => node.isKind(SyntaxKind.ExpressionStatement))\n\t\t.find((node) => nameOfUseApiHeader && node.getText().startsWith(nameOfUseApiHeader))\n\n\tif (!node) {\n\t\treturn null\n\t}\n\n\tconst targetNode = node.getFirstDescendantByKindOrThrow(SyntaxKind.ObjectLiteralExpression)\n\tconst values = getValuesOfObjectLiteral(targetNode)\n\n\tconst collapseObject = (v: string | string[] | typeof values): any => {\n\t\tif (typeof v === 'string') {\n\t\t\treturn v\n\t\t}\n\t\tif (Array.isArray(v) && v.every((value) => typeof value === 'string')) {\n\t\t\treturn v\n\t\t}\n\n\t\treturn v.reduce((acc, current) => {\n\t\t\tif (typeof current === 'string') {\n\t\t\t\treturn acc\n\t\t\t}\n\t\t\treturn {\n\t\t\t\t...acc,\n\t\t\t\t[current.identifier]: collapseObject(current.value as string[]),\n\t\t\t}\n\t\t}, {})\n\t}\n\treturn collapseObject(values)\n}\n\nexport const analyzeSourceFileExposedModels = (sourceFile: SourceFile): ExposedModelData[] => {\n\tconst models: ExposedModelData[] = []\n\n\tconst nameOfUseExposeApiModel = discoverImportedName({\n\t\tsourceFile,\n\t\toriginalName: 'useExposeApiModel',\n\t})\n\n\tconst nameOfUseExposeNamedApiModels = discoverImportedName({\n\t\tsourceFile,\n\t\toriginalName: 'useExposeNamedApiModels',\n\t})\n\n\tsourceFile\n\t\t.forEachChildAsArray()\n\t\t.filter((node) => node.isKind(SyntaxKind.ExpressionStatement))\n\t\t.map((node) => {\n\t\t\tif (nameOfUseExposeApiModel && node.getText().startsWith(nameOfUseExposeApiModel)) {\n\t\t\t\tconst callExpressionNode = node.getFirstChild()\n\t\t\t\tconst syntaxListChildren = callExpressionNode?.getChildrenOfKind(SyntaxKind.SyntaxList) || []\n\n\t\t\t\tconst firstChild = syntaxListChildren[0].getFirstChild()\n\t\t\t\tif (!firstChild) {\n\t\t\t\t\treturn\n\t\t\t\t}\n\n\t\t\t\tmodels.push(parseExposedModel(firstChild))\n\t\t\t\treturn\n\t\t\t}\n\n\t\t\tif (nameOfUseExposeNamedApiModels && node.getText().startsWith(nameOfUseExposeNamedApiModels)) {\n\t\t\t\tconst callExpressionNode = node.getFirstChild()\n\t\t\t\tconst syntaxListChildren = callExpressionNode?.getChildrenOfKind(SyntaxKind.SyntaxList) || []\n\n\t\t\t\tconst firstChild = syntaxListChildren[0].getFirstChild()\n\t\t\t\tif (!firstChild) {\n\t\t\t\t\treturn\n\t\t\t\t}\n\n\t\t\t\tconst parsedModels = parseNamedExposedModels(firstChild)\n\t\t\t\tparsedModels.forEach((model) => models.push(model))\n\t\t\t}\n\t\t})\n\treturn models\n}\n"],"names":["prepareOpenApiSpec","logLevel","tsconfigPath","sourceFilePaths","sourceFileDiscovery","incremental","openApiManager","OpenApiManager","Logger","project","Project","path","explicitRouters","discoveredRouterFiles","allSourceFiles","sourceFiles","filepath","filePath","file","discoverRouters","discoveredSourceFiles","startTime","files","discoverRouterFiles","acc","current","r","filesToAnalyze","apiHeaders","analyzeSourceFileApiHeader","headers","exposedModels","analyzeSourceFileExposedModels","cachePath","endpoints","analyzeMultipleSourceFiles","e","config","filterEndpointPaths","analyzedFiles","analyzeSourceFileWithCache","f","index","a","b","t","timestamp","getSourceFileTimestamp","cachedResults","SourceFileCache","t1","analyzeSourceFileEndpoints","t2","joinedOperations","routerName","routerPattern","node","nodeText","resolveEndpointPath","parseEndpoint","sourceFile","nameOfUseApiHeader","discoverImportedName","SyntaxKind","targetNode","values","getValuesOfObjectLiteral","collapseObject","v","value","models","nameOfUseExposeApiModel","nameOfUseExposeNamedApiModels","firstChild","parseExposedModel","parseNamedExposedModels","model"],"mappings":";;;;;;;;;;;;AAuCO,MAAMA,KAAqB,CAAC;AAAA,EAClC,UAAAC;AAAA,EACA,cAAAC;AAAA,EACA,iBAAAC;AAAA,EACA,qBAAAC;AAAA,EACA,aAAAC;AACD,MAAa;AACN,QAAAC,IAAiBC,EAAe,YAAY;AAE9C,MAAAD,EAAe;AAClB;AAGD,EAAIL,KACHO,EAAO,SAASP,CAAQ,GAGzBO,EAAO,KAAK,wBAAwB;AAE9B,QAAAC,IAAU,IAAIC,EAAQ;AAAA,IAC3B,kBAAkBC,EAAK,QAAQT,CAAY;AAAA,IAC3C,8BAA8B;AAAA,EAAA,CAC9B,GAEK,EAAE,iBAAAU,GAAiB,uBAAAC,GAAuB,gBAAAC,EAAA,KAAoB,MAAM;AAGnE,UAAAC,KAFmBZ,KAAmB,CAAC,GACI,IAAI,CAACa,MAAaL,EAAK,QAAQK,CAAQ,CAAC,EAC7C,IAAI,CAACC,MAAaR,EAAQ,qBAAqBQ,CAAQ,CAAC,GAC9FL,IAAkBG,EAAY,QAAQ,CAACG,OAAU;AAAA,MACtD,UAAUA,EAAK,YAAY;AAAA,MAC3B,YAAYA;AAAA,MACZ,SAASC,EAAgBD,CAAI;AAAA,IAAA,EAC5B,GAEI,EAAE,uBAAAL,GAAuB,uBAAAO,EAAA,KAA2B,MAAM;AAC/D,UAAIhB,MAAwB;AAC3B,eAAO,EAAE,uBAAuB,IAAI,uBAAuB,CAAA,EAAG;AAGzD,YAAAiB,IAAY,YAAY,IAAI,GAC5BC,IAAQC,EAAoB;AAAA,QACjC,YAAY,OAAOnB,KAAwB,WAAWA,EAAoB,WAAW;AAAA,QACrF,cAAcF;AAAA,MAAA,CACd;AACM,aAAAM,EAAA,KAAK,uBAAuB,KAAK,MAAM,YAAY,IAAI,IAAIa,CAAS,CAAC,IAAI,GACzEC;AAAA,IAAA,GACL,GAEGR,IAAiBC,EAAY;AAAA,MAClC,CAACS,GAAKC,MACLD,EAAI,KAAK,CAACE,MAAMA,EAAE,YAAY,MAAMD,EAAQ,YAAY,CAAC,IAAID,IAAMA,EAAI,OAAOC,CAAO;AAAA,MACtFL;AAAA,IACD;AAEA,WAAO,EAAE,iBAAAR,GAAiB,uBAAAC,GAAuB,gBAAAC,EAAe;AAAA,EAAA,GAC9D,GAEGa,IAAiBf,EAAgB;AAAA,IACtC,CAACY,GAAKC,MAAaD,EAAI,KAAK,CAACE,MAAMA,EAAE,aAAaD,EAAQ,QAAQ,IAAID,IAAMA,EAAI,OAAOC,CAAO;AAAA,IAC9FZ;AAAA,EACD,GAEMe,IAAad,EACjB,QAAQ,CAACI,MAASW,EAA2BX,CAAI,CAAC,EAClD,OAAO,CAACY,MAAY,CAAC,CAACA,CAAO;AAC/B,EAAIF,EAAW,SAAS,KAAKA,EAAW,CAAC,KACzBtB,EAAA,UAAUsB,EAAW,CAAC,CAAC;AAGvC,QAAMG,IAAgBjB,EAAe,QAAQ,CAACI,MAASc,EAA+Bd,CAAI,CAAC;AAE3F,EAAAZ,EAAe,iBAAiByB,CAAa;AAE7C,QAAME,IACD,OAAO5B,KAAgB,YAAYA,EAAY,YAC3CA,EAAY,YAEbM,EAAK,QAAQ,QAAQ,OAAO,gBAAgB,UAAU,YAAY,GAEpEuB,IAAYC,EAA2BR,GAAgB;AAAA,IAE5D,WAAAM;AAAA,IACA,gBAAgB,CAAA;AAAA,EAAC,CACjB;AAED,EAAA3B,EAAe,SAAS;AAAA,IACvB,uBAAuBO,EAAsB,IAAI,CAACK,OAAU;AAAA,MAC3D,MAAMA,EAAK;AAAA,MACX,SAASA,EAAK,QAAQ,MAAM,IAAI,CAACQ,OAAO;AAAA,QACvC,MAAMA;AAAA,QACN,WAAWQ,EACT,OAAO,CAACE,MAAMA,EAAE,mBAAmBlB,EAAK,QAAQ,EAChD,IAAI,CAACkB,MAAM,GAAGA,EAAE,OAAO,aAAa,IAAIA,EAAE,IAAI,EAAE;AAAA,MAAA,EACjD;AAAA,IAAA,EACD;AAAA,IACF,qBAAqBxB,EAAgB,IAAI,CAACM,OAAU;AAAA,MACnD,MAAMA,EAAK;AAAA,MACX,SAASA,EAAK,QAAQ,MAAM,IAAI,CAACQ,OAAO;AAAA,QACvC,MAAMA;AAAA,QACN,WAAWQ,EACT,OAAO,CAACE,MAAMA,EAAE,mBAAmBlB,EAAK,QAAQ,EAChD,IAAI,CAACkB,MAAM,GAAGA,EAAE,OAAO,aAAa,IAAIA,EAAE,IAAI,EAAE;AAAA,MAAA,EACjD;AAAA,IAAA,EACD;AAAA,EAAA,CACF,GAED9B,EAAe,aAAa4B,CAAS,GACrC5B,EAAe,YAAY;AAC5B,GAEa6B,IAA6B,CACzCb,GACAe,GAKAC,MACoB;AACd,QAAAjB,IAAY,YAAY,IAAI,GAC5BkB,IAAgBjB,EAAM,IAAI,CAACJ,MAASsB,EAA2BtB,GAAMmB,CAA2B,CAAC;AAChG,SAAA7B,EAAA,KAAK,wBAAwB,KAAK,MAAM,YAAY,IAAI,IAAIa,CAAS,CAAC,IAAI,GAG/EkB,EAAA,IAAI,CAACE,GAAGC,OAAW;AAAA,IACnB,UAAUpB,EAAMoB,CAAK,EAAE;AAAA,IACvB,WAAWD,EAAE;AAAA,EAAA,EACZ,EACD,KAAK,CAACE,GAAGC,MAAMA,EAAE,YAAYD,EAAE,SAAS,EACxC,OAAO,CAACE,MAAMA,EAAE,YAAY,GAAG,EAC/B,QAAQ,CAACA,MAAM;AACR,IAAArC,EAAA,KAAK,MAAMqC,EAAE,QAAQ,UAAU,KAAK,MAAMA,EAAE,SAAS,CAAC,eAAe;AAAA,EAAA,CAC5E,GAEKN,EAAc,QAAQ,CAACE,MAAMA,EAAE,SAAS;AAChD,GAEaD,IAA6B,CACzCtB,GACAmB,GAKAC,MACmD;AACnD,QAAMQ,IAAYC,EAAuB7B,EAAK,YAAYmB,EAAO,cAAc,GACzEW,IAAgBC,EAAgB,iBAAiB/B,EAAK,YAAY4B,GAAWT,EAAO,SAAS;AAEnG,MAAIW;AACH,WAAAxC,EAAO,MAAM,IAAIU,EAAK,QAAQ,wBAAwB,GAC/C,EAAE,WAAW8B,EAAc,WAAW,QAAQ,EAAE;AAExD,EAAAxC,EAAO,MAAM,IAAIU,EAAK,QAAQ,gBAAgB;AAExC,QAAAgC,IAAK,YAAY,IAAI,GACrBhB,IAAYiB,EAA2BjC,CAAyB,GAChEkC,IAAK,YAAY,IAAI;AAC3B,SAAA5C,EAAO,MAAM,IAAIU,EAAK,QAAQ,iBAAiBkC,IAAKF,CAAE,IAAI,GAC1DD,EAAgB,aAAa/B,EAAK,YAAY4B,GAAWT,EAAO,WAAWH,CAAS,GAC7E,EAAE,WAAAA,GAAW,QAAQkB,IAAKF,EAAG;AACrC,GAEaC,IAA6B,CACzCjC,GACAoB,MACoB;AACpB,QAAMJ,IAA4B,CAAC,GAE7BmB,IADa,CAAC,OAAO,QAAQ,OAAO,UAAU,OAAO,OAAO,EAC9B,KAAK,GAAG;AAE5C,SAAAnC,EAAK,QAAQ,MAAM,QAAQ,CAACoC,MAAe;AAC1C,UAAMC,IAAgB,IAAI,OAAO,GAAGD,CAAU,SAASD,CAAgB,GAAG;AACrE,IAAAnC,EAAA,WAAW,aAAa,CAACsC,MAAS;AAChC,YAAAC,IAAWD,EAAK,QAAQ;AAE1B,MAAAD,EAAc,KAAKE,CAAQ,MACTC,EAAoBF,CAAI,GAM7CtB,EAAU,KAAKyB,EAAcH,GAAMtC,EAAK,QAAQ,CAAC;AAAA,IAClD,CACA;AAAA,EAAA,CACD,GAEMgB;AACR,GAEaL,IAA6B,CAAC+B,MAAiD;AAC3F,QAAMC,IAAqBC,EAAqB;AAAA,IAC/C,YAAAF;AAAA,IACA,cAAc;AAAA,EAAA,CACd;AAED,MAAI,CAACC;AACG,WAAA;AAGF,QAAAL,IAAOI,EACX,oBAAoB,EACpB,OAAO,CAACJ,MAASA,EAAK,OAAOO,EAAW,mBAAmB,CAAC,EAC5D,KAAK,CAACP,MAASK,KAAsBL,EAAK,QAAQ,EAAE,WAAWK,CAAkB,CAAC;AAEpF,MAAI,CAACL;AACG,WAAA;AAGR,QAAMQ,IAAaR,EAAK,gCAAgCO,EAAW,uBAAuB,GACpFE,IAASC,EAAyBF,CAAU,GAE5CG,IAAiB,CAACC,MACnB,OAAOA,KAAM,YAGb,MAAM,QAAQA,CAAC,KAAKA,EAAE,MAAM,CAACC,MAAU,OAAOA,KAAU,QAAQ,IAC5DD,IAGDA,EAAE,OAAO,CAAC5C,GAAKC,MACjB,OAAOA,KAAY,WACfD,IAED;AAAA,IACN,GAAGA;AAAA,IACH,CAACC,EAAQ,UAAU,GAAG0C,EAAe1C,EAAQ,KAAiB;AAAA,EAC/D,GACE,EAAE;AAEN,SAAO0C,EAAeF,CAAM;AAC7B,GAEajC,IAAiC,CAAC4B,MAA+C;AAC7F,QAAMU,IAA6B,CAAC,GAE9BC,IAA0BT,EAAqB;AAAA,IACpD,YAAAF;AAAA,IACA,cAAc;AAAA,EAAA,CACd,GAEKY,IAAgCV,EAAqB;AAAA,IAC1D,YAAAF;AAAA,IACA,cAAc;AAAA,EAAA,CACd;AAED,SAAAA,EACE,oBAAoB,EACpB,OAAO,CAACJ,MAASA,EAAK,OAAOO,EAAW,mBAAmB,CAAC,EAC5D,IAAI,CAACP,MAAS;AACd,QAAIe,KAA2Bf,EAAK,QAAU,EAAA,WAAWe,CAAuB,GAAG;AAIlF,YAAME,KAHqBjB,EAAK,cAAc,GACC,kBAAkBO,EAAW,UAAU,KAAK,CAAC,GAEtD,CAAC,EAAE,cAAc;AACvD,UAAI,CAACU;AACJ;AAGM,MAAAH,EAAA,KAAKI,EAAkBD,CAAU,CAAC;AACzC;AAAA,IAAA;AAGD,QAAID,KAAiChB,EAAK,QAAU,EAAA,WAAWgB,CAA6B,GAAG;AAI9F,YAAMC,KAHqBjB,EAAK,cAAc,GACC,kBAAkBO,EAAW,UAAU,KAAK,CAAC,GAEtD,CAAC,EAAE,cAAc;AACvD,UAAI,CAACU;AACJ;AAID,MADqBE,EAAwBF,CAAU,EAC1C,QAAQ,CAACG,MAAUN,EAAO,KAAKM,CAAK,CAAC;AAAA,IAAA;AAAA,EACnD,CACA,GACKN;AACR;"}
1
+ {"version":3,"file":"analyzerModule.mjs","sources":["../../../src/openapi/analyzerModule/analyzerModule.ts"],"sourcesContent":["import * as path from 'path'\nimport { SourceFile, SyntaxKind } from 'ts-morph'\nimport { Project } from 'ts-morph'\n\nimport { Logger } from '../../utils/logger'\nimport { discoverImportedName } from '../discoveryModule/discoverImports/discoverImports'\nimport {\n\tDiscoveredSourceFile,\n\tdiscoverRouterFiles,\n} from '../discoveryModule/discoverRouterFiles/discoverRouterFiles'\nimport { discoverRouters } from '../discoveryModule/discoverRouters/discoverRouters'\nimport { ApiDocsHeader, OpenApiManager } from '../manager/OpenApiManager'\nimport { EndpointData, ExposedModelData } from '../types'\nimport { getSourceFileTimestamp, TimestampCache } from './getSourceFileTimestamp'\nimport { getValuesOfObjectLiteral, resolveEndpointPath } from './nodeParsers'\nimport { parseEndpoint, SectionTiming } from './parseEndpoint'\nimport { parseExposedModel, parseNamedExposedModels } from './parseExposedModels'\nimport { SourceFileCache } from './sourceFileCache'\n\ntype Props = {\n\tlogLevel?: Parameters<(typeof Logger)['setLevel']>[0]\n\ttsconfigPath: string\n\tsourceFilePaths?: string[]\n\tsourceFileDiscovery?: boolean | FileDiscoveryConfig\n\tincremental?:\n\t\t| boolean\n\t\t| {\n\t\t\t\tcachePath: string\n\t\t }\n\tprofiling?: 'stats' | 'off' | 'debug'\n}\n\ntype FileDiscoveryConfig = {\n\trootPath: string\n}\n\ntype EndpointTiming = { method: string; path: string; timing: number; sectionTimings: SectionTiming[] }\n\n/**\n * @param tsconfigPath Path to tsconfig file relative to project root\n * @param sourceFilePaths Array of router source files relative to project root\n */\nexport const prepareOpenApiSpec = ({\n\tlogLevel,\n\ttsconfigPath,\n\tsourceFilePaths,\n\tsourceFileDiscovery,\n\tincremental,\n\tprofiling = 'stats',\n}: Props) => {\n\tconst openApiManager = OpenApiManager.getInstance()\n\n\tif (openApiManager.isReady()) {\n\t\treturn\n\t}\n\n\tif (logLevel) {\n\t\tLogger.setLevel(logLevel)\n\t}\n\n\tLogger.info('Preparing OpenAPI spec')\n\n\tconst project = new Project({\n\t\ttsConfigFilePath: path.resolve(tsconfigPath),\n\t\tskipFileDependencyResolution: true,\n\t})\n\n\tconst { explicitRouters, discoveredRouterFiles, allSourceFiles } = (() => {\n\t\tconst sourceFilesToAdd = sourceFilePaths ?? []\n\t\tconst resolvedSourceFilePaths = sourceFilesToAdd.map((filepath) => path.resolve(filepath))\n\t\tconst sourceFiles = resolvedSourceFilePaths.map((filePath) => project.getSourceFileOrThrow(filePath))\n\t\tconst explicitRouters = sourceFiles.flatMap((file) => ({\n\t\t\tfileName: file.getFilePath(),\n\t\t\tsourceFile: file,\n\t\t\trouters: discoverRouters(file),\n\t\t}))\n\n\t\tconst { discoveredRouterFiles, discoveredSourceFiles } = (() => {\n\t\t\tif (sourceFileDiscovery === false) {\n\t\t\t\treturn { discoveredRouterFiles: [], discoveredSourceFiles: [] }\n\t\t\t}\n\n\t\t\tconst startTime = performance.now()\n\t\t\tconst files = discoverRouterFiles({\n\t\t\t\ttargetPath: typeof sourceFileDiscovery === 'object' ? sourceFileDiscovery.rootPath : '.',\n\t\t\t\ttsConfigPath: tsconfigPath,\n\t\t\t})\n\t\t\tif (profiling !== 'off') {\n\t\t\t\tLogger.info(`File discovery took ${Math.round(performance.now() - startTime)}ms`)\n\t\t\t}\n\t\t\treturn files\n\t\t})()\n\n\t\tconst allSourceFiles = sourceFiles.reduce(\n\t\t\t(acc, current) =>\n\t\t\t\tacc.some((r) => r.getFilePath() === current.getFilePath()) ? acc : acc.concat(current),\n\t\t\tdiscoveredSourceFiles,\n\t\t)\n\n\t\treturn { explicitRouters, discoveredRouterFiles, allSourceFiles }\n\t})()\n\n\tconst filesToAnalyze = explicitRouters.reduce(\n\t\t(acc, current) => (acc.some((r) => r.fileName === current.fileName) ? acc : acc.concat(current)),\n\t\tdiscoveredRouterFiles,\n\t)\n\n\tconst apiHeaders = allSourceFiles\n\t\t.flatMap((file) => analyzeSourceFileApiHeader(file))\n\t\t.filter((headers) => !!headers)\n\tif (apiHeaders.length > 0 && apiHeaders[0]) {\n\t\topenApiManager.setHeader(apiHeaders[0])\n\t}\n\n\tconst exposedModels = allSourceFiles.flatMap((file) => analyzeSourceFileExposedModels(file))\n\n\topenApiManager.setExposedModels(exposedModels)\n\n\tconst cachePath = (() => {\n\t\tif (typeof incremental === 'object' && incremental.cachePath) {\n\t\t\treturn incremental.cachePath\n\t\t}\n\t\treturn path.resolve(process.cwd(), 'node_modules', '.cache', 'moonflower')\n\t})()\n\tconst endpoints = analyzeMultipleSourceFiles(filesToAnalyze, {\n\t\tincremental: incremental !== false,\n\t\tcachePath,\n\t\ttimestampCache: {},\n\t\tprofiling,\n\t})\n\n\topenApiManager.setStats({\n\t\tdiscoveredRouterFiles: discoveredRouterFiles.map((file) => ({\n\t\t\tpath: file.fileName,\n\t\t\trouters: file.routers.named.map((r) => ({\n\t\t\t\tname: r,\n\t\t\t\tendpoints: endpoints\n\t\t\t\t\t.filter((e) => e.sourceFilePath === file.fileName)\n\t\t\t\t\t.map((e) => `${e.method.toUpperCase()} ${e.path}`),\n\t\t\t})),\n\t\t})),\n\t\texplicitRouterFiles: explicitRouters.map((file) => ({\n\t\t\tpath: file.fileName,\n\t\t\trouters: file.routers.named.map((r) => ({\n\t\t\t\tname: r,\n\t\t\t\tendpoints: endpoints\n\t\t\t\t\t.filter((e) => e.sourceFilePath === file.fileName)\n\t\t\t\t\t.map((e) => `${e.method.toUpperCase()} ${e.path}`),\n\t\t\t})),\n\t\t})),\n\t})\n\n\topenApiManager.setEndpoints(endpoints)\n\topenApiManager.markAsReady()\n}\n\nexport const analyzeMultipleSourceFiles = (\n\tfiles: DiscoveredSourceFile[],\n\tconfig: {\n\t\tincremental: boolean\n\t\tcachePath: string\n\t\ttimestampCache: TimestampCache\n\t\tprofiling?: 'stats' | 'off' | 'debug'\n\t},\n\tfilterEndpointPaths?: string[],\n): EndpointData[] => {\n\tconst profiling = config.profiling ?? 'stats'\n\tconst startTime = performance.now()\n\tconst analyzedFiles = files.map((file) => analyzeSourceFileWithCache(file, config, filterEndpointPaths))\n\n\tif (profiling !== 'off') {\n\t\tLogger.info(`Router analysis took ${Math.round(performance.now() - startTime)}ms`)\n\t}\n\n\tif (profiling === 'stats') {\n\t\tanalyzedFiles\n\t\t\t.map((f, index) => ({ fileName: files[index].fileName, timeTaken: f.timing }))\n\t\t\t.sort((a, b) => b.timeTaken - a.timeTaken)\n\t\t\t.filter((t) => t.timeTaken > 500)\n\t\t\t.forEach((t) => {\n\t\t\t\tLogger.info(`- [${t.fileName}] Took ${Math.round(t.timeTaken)}ms to analyze`)\n\t\t\t})\n\t} else if (profiling === 'debug') {\n\t\tanalyzedFiles\n\t\t\t.map((f, index) => ({\n\t\t\t\tfileName: files[index].fileName,\n\t\t\t\ttimeTaken: f.timing,\n\t\t\t\tendpointTimings: f.endpointTimings,\n\t\t\t}))\n\t\t\t.sort((a, b) => b.timeTaken - a.timeTaken)\n\t\t\t.forEach((t) => {\n\t\t\t\tLogger.info(`- [${t.fileName}] Took ${Math.round(t.timeTaken)}ms to analyze`)\n\t\t\t\tt.endpointTimings\n\t\t\t\t\t.sort((a, b) => b.timing - a.timing)\n\t\t\t\t\t.forEach((ep) => {\n\t\t\t\t\t\tLogger.info(` - ${ep.method} ${ep.path} (${Math.round(ep.timing)}ms)`)\n\t\t\t\t\t\tep.sectionTimings\n\t\t\t\t\t\t\t.filter((s) => s.timing >= 1)\n\t\t\t\t\t\t\t.sort((a, b) => b.timing - a.timing)\n\t\t\t\t\t\t\t.forEach((s) => {\n\t\t\t\t\t\t\t\tLogger.info(` - ${s.section}: ${Math.round(s.timing)}ms`)\n\t\t\t\t\t\t\t})\n\t\t\t\t\t})\n\t\t\t})\n\t}\n\n\treturn analyzedFiles.flatMap((f) => f.endpoints)\n}\n\nexport const analyzeSourceFileWithCache = (\n\tfile: DiscoveredSourceFile,\n\tconfig: {\n\t\tincremental: boolean\n\t\tcachePath: string\n\t\ttimestampCache: TimestampCache\n\t\tprofiling?: 'stats' | 'off' | 'debug'\n\t},\n\tfilterEndpointPaths?: string[],\n): { endpoints: EndpointData[]; timing: number; endpointTimings: EndpointTiming[] } => {\n\tconst timestamp = getSourceFileTimestamp(file.sourceFile, config.timestampCache)\n\tconst cachedResults = SourceFileCache.getCachedResults(file.sourceFile, timestamp, config.cachePath)\n\n\tif (cachedResults) {\n\t\tLogger.debug(`[${file.fileName}] Found cached results`)\n\t\treturn { endpoints: cachedResults.endpoints, timing: 0, endpointTimings: [] }\n\t}\n\tLogger.debug(`[${file.fileName}] Analyzing...`)\n\n\tconst t1 = performance.now()\n\tconst { endpoints, endpointTimings } = analyzeSourceFileEndpoints(file, filterEndpointPaths)\n\tconst t2 = performance.now()\n\tLogger.debug(`[${file.fileName}] Analyzed in ${t2 - t1}ms`)\n\tSourceFileCache.cacheResults(file.sourceFile, timestamp, config.cachePath, endpoints)\n\treturn { endpoints, timing: t2 - t1, endpointTimings }\n}\n\nexport const analyzeSourceFileEndpoints = (\n\tfile: DiscoveredSourceFile,\n\tfilterEndpointPaths?: string[],\n): { endpoints: EndpointData[]; endpointTimings: EndpointTiming[] } => {\n\tconst endpoints: EndpointData[] = []\n\tconst endpointTimings: EndpointTiming[] = []\n\tconst operations = ['get', 'post', 'put', 'delete', 'del', 'patch']\n\tconst joinedOperations = operations.join('|')\n\n\tfile.routers.named.forEach((routerName) => {\n\t\tconst routerPattern = new RegExp(`${routerName}\\\\.(?:${joinedOperations})`)\n\t\tfile.sourceFile.forEachChild((node) => {\n\t\t\tconst nodeText = node.getText()\n\n\t\t\tif (routerPattern.test(nodeText)) {\n\t\t\t\tconst endpointPath = resolveEndpointPath(node) ?? ''\n\n\t\t\t\tif (filterEndpointPaths && !filterEndpointPaths.some((path) => endpointPath.includes(path))) {\n\t\t\t\t\treturn\n\t\t\t\t}\n\n\t\t\t\tconst t1 = performance.now()\n\t\t\t\tconst { endpoint, sectionTimings } = parseEndpoint(node, file.fileName)\n\t\t\t\tendpointTimings.push({\n\t\t\t\t\tmethod: endpoint.method,\n\t\t\t\t\tpath: endpoint.path,\n\t\t\t\t\ttiming: performance.now() - t1,\n\t\t\t\t\tsectionTimings,\n\t\t\t\t})\n\t\t\t\tendpoints.push(endpoint)\n\t\t\t}\n\t\t})\n\t})\n\n\treturn { endpoints, endpointTimings }\n}\n\nexport const analyzeSourceFileApiHeader = (sourceFile: SourceFile): ApiDocsHeader | null => {\n\tconst nameOfUseApiHeader = discoverImportedName({\n\t\tsourceFile,\n\t\toriginalName: 'useApiHeader',\n\t})\n\n\tif (!nameOfUseApiHeader) {\n\t\treturn null\n\t}\n\n\tconst node = sourceFile\n\t\t.forEachChildAsArray()\n\t\t.filter((node) => node.isKind(SyntaxKind.ExpressionStatement))\n\t\t.find((node) => nameOfUseApiHeader && node.getText().startsWith(nameOfUseApiHeader))\n\n\tif (!node) {\n\t\treturn null\n\t}\n\n\tconst targetNode = node.getFirstDescendantByKindOrThrow(SyntaxKind.ObjectLiteralExpression)\n\tconst values = getValuesOfObjectLiteral(targetNode)\n\n\tconst collapseObject = (v: string | string[] | typeof values): any => {\n\t\tif (typeof v === 'string') {\n\t\t\treturn v\n\t\t}\n\t\tif (Array.isArray(v) && v.every((value) => typeof value === 'string')) {\n\t\t\treturn v\n\t\t}\n\n\t\treturn v.reduce((acc, current) => {\n\t\t\tif (typeof current === 'string') {\n\t\t\t\treturn acc\n\t\t\t}\n\t\t\treturn {\n\t\t\t\t...acc,\n\t\t\t\t[current.identifier]: collapseObject(current.value as string[]),\n\t\t\t}\n\t\t}, {})\n\t}\n\treturn collapseObject(values)\n}\n\nexport const analyzeSourceFileExposedModels = (sourceFile: SourceFile): ExposedModelData[] => {\n\tconst models: ExposedModelData[] = []\n\n\tconst nameOfUseExposeApiModel = discoverImportedName({\n\t\tsourceFile,\n\t\toriginalName: 'useExposeApiModel',\n\t})\n\n\tconst nameOfUseExposeNamedApiModels = discoverImportedName({\n\t\tsourceFile,\n\t\toriginalName: 'useExposeNamedApiModels',\n\t})\n\n\tsourceFile\n\t\t.forEachChildAsArray()\n\t\t.filter((node) => node.isKind(SyntaxKind.ExpressionStatement))\n\t\t.map((node) => {\n\t\t\tif (nameOfUseExposeApiModel && node.getText().startsWith(nameOfUseExposeApiModel)) {\n\t\t\t\tconst callExpressionNode = node.getFirstChild()\n\t\t\t\tconst syntaxListChildren = callExpressionNode?.getChildrenOfKind(SyntaxKind.SyntaxList) || []\n\n\t\t\t\tconst firstChild = syntaxListChildren[0].getFirstChild()\n\t\t\t\tif (!firstChild) {\n\t\t\t\t\treturn\n\t\t\t\t}\n\n\t\t\t\tmodels.push(parseExposedModel(firstChild))\n\t\t\t\treturn\n\t\t\t}\n\n\t\t\tif (nameOfUseExposeNamedApiModels && node.getText().startsWith(nameOfUseExposeNamedApiModels)) {\n\t\t\t\tconst callExpressionNode = node.getFirstChild()\n\t\t\t\tconst syntaxListChildren = callExpressionNode?.getChildrenOfKind(SyntaxKind.SyntaxList) || []\n\n\t\t\t\tconst firstChild = syntaxListChildren[0].getFirstChild()\n\t\t\t\tif (!firstChild) {\n\t\t\t\t\treturn\n\t\t\t\t}\n\n\t\t\t\tconst parsedModels = parseNamedExposedModels(firstChild)\n\t\t\t\tparsedModels.forEach((model) => models.push(model))\n\t\t\t}\n\t\t})\n\treturn models\n}\n"],"names":["prepareOpenApiSpec","logLevel","tsconfigPath","sourceFilePaths","sourceFileDiscovery","incremental","profiling","openApiManager","OpenApiManager","Logger","project","Project","path","explicitRouters","discoveredRouterFiles","allSourceFiles","sourceFiles","filepath","filePath","file","discoverRouters","discoveredSourceFiles","startTime","files","discoverRouterFiles","acc","current","r","filesToAnalyze","apiHeaders","analyzeSourceFileApiHeader","headers","exposedModels","analyzeSourceFileExposedModels","cachePath","endpoints","analyzeMultipleSourceFiles","e","config","filterEndpointPaths","analyzedFiles","analyzeSourceFileWithCache","f","index","a","b","t","ep","timestamp","getSourceFileTimestamp","cachedResults","SourceFileCache","t1","endpointTimings","analyzeSourceFileEndpoints","t2","joinedOperations","routerName","routerPattern","node","nodeText","resolveEndpointPath","endpoint","sectionTimings","parseEndpoint","sourceFile","nameOfUseApiHeader","discoverImportedName","SyntaxKind","targetNode","values","getValuesOfObjectLiteral","collapseObject","v","value","models","nameOfUseExposeApiModel","nameOfUseExposeNamedApiModels","firstChild","parseExposedModel","parseNamedExposedModels","model"],"mappings":"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"}
@@ -1,2 +1,2 @@
1
- "use strict";Object.defineProperty(exports,Symbol.toStringTag,{value:"Module"});const o=require("ts-morph"),l=require("../../utils/logger.cjs"),u=require("./nodeParsers.cjs"),y=(e,s)=>{const r=e.getFirstDescendantByKind(o.SyntaxKind.PropertyAccessExpression).getText().split(".")[1].toUpperCase(),t=r==="DEL"?"DELETE":r,d=u.resolveEndpointPath(e)??"",n={method:t,path:d,sourceFilePath:s,requestPathParams:[],requestQuery:[],requestHeaders:[],rawBody:void 0,objectBody:[],responses:[],name:void 0,summary:void 0,description:void 0,tags:void 0},a=h(e);try{f(a.useApiEndpoint).forEach(g=>{n[g.identifier]=g.value})}catch(i){l.Logger.error("Error",i)}try{n.requestPathParams=m(a.usePathParams,d)}catch(i){l.Logger.error("Error",i)}try{n.requestQuery=c(a.useQueryParams,"useQueryParams")}catch(i){l.Logger.error("Error",i)}try{n.requestHeaders=c(a.useHeaderParams,"useHeaderParams")}catch(i){l.Logger.error("Error",i)}try{const i=x(a.useRequestRawBody);i&&(n.rawBody=i)}catch(i){l.Logger.error("Error",i)}try{n.objectBody=c(a.useRequestBody,"useRequestBody")}catch(i){l.Logger.error("Error",i)}try{n.responses=K(e)}catch(i){l.Logger.error("Error",i)}return n},h=e=>{const s={useApiEndpoint:null,usePathParams:null,useQueryParams:null,useHeaderParams:null,useRequestBody:null,useRequestRawBody:null};for(const r of e.getDescendantsOfKind(o.SyntaxKind.CallExpression)){const t=r.getFirstChildByKind(o.SyntaxKind.Identifier)?.getText();t&&t in s&&s[t]===null&&(s[t]=r)}return s},f=e=>{if(!e)return[];const s=e.getFirstChildByKind(o.SyntaxKind.SyntaxList),r=u.findNodeImplementation(s.getLastChild());if(!r.isKind(o.SyntaxKind.ObjectLiteralExpression))throw new Error("Non-literal type used in useApiEndpoint");const t=r.asKind(o.SyntaxKind.ObjectLiteralExpression);return u.getValuesOfObjectLiteral(t).filter(n=>n.value!==null)},m=(e,s)=>{if(!e)return[];const r=e.getFirstChildByKind(o.SyntaxKind.SyntaxList),t=u.findNodeImplementation(r.getLastChild());if(!t.isKind(o.SyntaxKind.ObjectLiteralExpression))throw new Error("Non-literal type used in usePathParams");const d=s.split("/").filter(a=>a.startsWith(":")).map(a=>({name:a.substring(1).replace(/\?/,""),optional:a.includes("?")})),n=t.asKind(o.SyntaxKind.ObjectLiteralExpression);return u.getShapeOfValidatorLiteral(n).filter(a=>a.shape!==null).map(a=>({identifier:a.identifier,signature:a.shape,optional:d.some(i=>i.name===a.identifier&&i.optional),description:a.description,errorMessage:a.errorMessage}))},x=e=>{if(!e)return null;const s=e.getFirstChildByKind(o.SyntaxKind.SyntaxList),r=u.findNodeImplementation(s.getLastChild(t=>!t.isKind(o.SyntaxKind.CommaToken)));return{signature:u.getValidatorPropertyShape(r),optional:u.getValidatorPropertyOptionality(r),description:u.getValidatorPropertyStringValue(r,"description"),errorMessage:u.getValidatorPropertyStringValue(r,"errorMessage")}},c=(e,s)=>{if(!e)return[];const r=e.getFirstChildByKind(o.SyntaxKind.SyntaxList),t=u.findNodeImplementation(r.getLastChild());if(!t.isKind(o.SyntaxKind.ObjectLiteralExpression))throw new Error(`Non-literal type used in ${s}`);const d=t.asKind(o.SyntaxKind.ObjectLiteralExpression);return u.getShapeOfValidatorLiteral(d).filter(n=>n.shape!==null).map(n=>({identifier:n.identifier,signature:n.shape,optional:n.optional,description:n.description,errorMessage:n.errorMessage}))},K=e=>{const r=e.getFirstChildByKind(o.SyntaxKind.CallExpression).getFirstChildByKind(o.SyntaxKind.SyntaxList).getFirstChildByKind(o.SyntaxKind.ArrowFunction).getReturnType(),t=r.getText().startsWith("Promise")?r.getTypeArguments()[0]:r,d=u.getProperTypeShape(t,e);return p(d)},p=e=>{if(typeof e=="string")return[{status:e==="void"||e==="null"?204:200,contentType:"text/plain",signature:e,description:"",errorMessage:""}];if(e[0].role==="union_entry"||e[0].role==="literal_string")return p(e[0].shape);if(e[0].role==="property"&&e[0].identifier==="_isUseReturnValue"){const s=(()=>{const t=e.find(d=>d.role==="property"&&d.identifier==="status")?.shape;if(!t||typeof t=="string"||typeof t[0].shape!="string")throw new Error("Invalid useReturnValue hook");return parseInt(t[0].shape)})(),r=(()=>{const t=e.find(d=>d.role==="property"&&d.identifier==="contentType")?.shape;if(!t||typeof t=="string"||typeof t[0].shape!="string")throw new Error("Invalid useReturnValue hook");return t[0].shape})();return[{status:s,contentType:r,signature:e,description:"",errorMessage:""}]}return e[0].role==="union"?typeof e[0].shape=="string"?[{status:e[0].shape==="void"||e[0].shape==="null"?204:200,contentType:"application/json",signature:e[0].shape,description:"",errorMessage:""}]:e[0].shape.flatMap(s=>p([s])):e[0].role==="buffer"?[{status:200,contentType:"application/octet-stream",signature:e,description:"",errorMessage:""}]:[{status:200,contentType:"application/json",signature:e,description:"",errorMessage:""}]};exports.parseEndpoint=y;
1
+ "use strict";Object.defineProperty(exports,Symbol.toStringTag,{value:"Module"});const o=require("ts-morph"),c=require("../../utils/logger.cjs"),u=require("./nodeParsers.cjs"),x=(e,i)=>{const r=e.getFirstDescendantByKind(o.SyntaxKind.PropertyAccessExpression).getText().split(".")[1].toUpperCase(),t=r==="DEL"?"DELETE":r,a=u.resolveEndpointPath(e)??"",n={method:t,path:a,sourceFilePath:i,requestPathParams:[],requestQuery:[],requestHeaders:[],rawBody:void 0,objectBody:[],responses:[],name:void 0,summary:void 0,description:void 0,tags:void 0},d=[],l=(s,g)=>{const f=performance.now(),m=g();return d.push({section:s,timing:performance.now()-f}),m},p=E(e);try{l("useApiEndpoint",()=>K(p.useApiEndpoint)).forEach(g=>{n[g.identifier]=g.value})}catch(s){c.Logger.error("Error",s)}try{n.requestPathParams=l("usePathParams",()=>P(p.usePathParams,a))}catch(s){c.Logger.error("Error",s)}try{n.requestQuery=l("useQueryParams",()=>y(p.useQueryParams,"useQueryParams"))}catch(s){c.Logger.error("Error",s)}try{n.requestHeaders=l("useHeaderParams",()=>y(p.useHeaderParams,"useHeaderParams"))}catch(s){c.Logger.error("Error",s)}try{const s=l("useRequestRawBody",()=>L(p.useRequestRawBody));s&&(n.rawBody=s)}catch(s){c.Logger.error("Error",s)}try{n.objectBody=l("useRequestBody",()=>y(p.useRequestBody,"useRequestBody"))}catch(s){c.Logger.error("Error",s)}try{n.responses=l("response",()=>S(e))}catch(s){c.Logger.error("Error",s)}return{endpoint:n,sectionTimings:d}},E=e=>{const i={useApiEndpoint:null,usePathParams:null,useQueryParams:null,useHeaderParams:null,useRequestBody:null,useRequestRawBody:null};for(const r of e.getDescendantsOfKind(o.SyntaxKind.CallExpression)){const t=r.getFirstChildByKind(o.SyntaxKind.Identifier)?.getText();t&&t in i&&i[t]===null&&(i[t]=r)}return i},K=e=>{if(!e)return[];const i=e.getFirstChildByKind(o.SyntaxKind.SyntaxList),r=u.findNodeImplementation(i.getLastChild());if(!r.isKind(o.SyntaxKind.ObjectLiteralExpression))throw new Error("Non-literal type used in useApiEndpoint");const t=r.asKind(o.SyntaxKind.ObjectLiteralExpression);return u.getValuesOfObjectLiteral(t).filter(n=>n.value!==null)},P=(e,i)=>{if(!e)return[];const r=e.getFirstChildByKind(o.SyntaxKind.SyntaxList),t=u.findNodeImplementation(r.getLastChild());if(!t.isKind(o.SyntaxKind.ObjectLiteralExpression))throw new Error("Non-literal type used in usePathParams");const a=i.split("/").filter(d=>d.startsWith(":")).map(d=>({name:d.substring(1).replace(/\?/,""),optional:d.includes("?")})),n=t.asKind(o.SyntaxKind.ObjectLiteralExpression);return u.getShapeOfValidatorLiteral(n).filter(d=>d.shape!==null).map(d=>({identifier:d.identifier,signature:d.shape,optional:a.some(l=>l.name===d.identifier&&l.optional),description:d.description,errorMessage:d.errorMessage}))},L=e=>{if(!e)return null;const i=e.getFirstChildByKind(o.SyntaxKind.SyntaxList),r=u.findNodeImplementation(i.getLastChild(t=>!t.isKind(o.SyntaxKind.CommaToken)));return{signature:u.getValidatorPropertyShape(r),optional:u.getValidatorPropertyOptionality(r),description:u.getValidatorPropertyStringValue(r,"description"),errorMessage:u.getValidatorPropertyStringValue(r,"errorMessage")}},y=(e,i)=>{if(!e)return[];const r=e.getFirstChildByKind(o.SyntaxKind.SyntaxList),t=u.findNodeImplementation(r.getLastChild());if(!t.isKind(o.SyntaxKind.ObjectLiteralExpression))throw new Error(`Non-literal type used in ${i}`);const a=t.asKind(o.SyntaxKind.ObjectLiteralExpression);return u.getShapeOfValidatorLiteral(a).filter(n=>n.shape!==null).map(n=>({identifier:n.identifier,signature:n.shape,optional:n.optional,description:n.description,errorMessage:n.errorMessage}))},S=e=>{const r=e.getFirstChildByKind(o.SyntaxKind.CallExpression).getFirstChildByKind(o.SyntaxKind.SyntaxList).getFirstChildByKind(o.SyntaxKind.ArrowFunction).getReturnType(),t=r.getText().startsWith("Promise")?r.getTypeArguments()[0]:r,a=u.getProperTypeShape(t,e);return h(a)},h=e=>{if(typeof e=="string")return[{status:e==="void"||e==="null"?204:200,contentType:"text/plain",signature:e,description:"",errorMessage:""}];if(e[0].role==="union_entry"||e[0].role==="literal_string")return h(e[0].shape);if(e[0].role==="property"&&e[0].identifier==="_isUseReturnValue"){const i=(()=>{const t=e.find(a=>a.role==="property"&&a.identifier==="status")?.shape;if(!t||typeof t=="string"||typeof t[0].shape!="string")throw new Error("Invalid useReturnValue hook");return parseInt(t[0].shape)})(),r=(()=>{const t=e.find(a=>a.role==="property"&&a.identifier==="contentType")?.shape;if(!t||typeof t=="string"||typeof t[0].shape!="string")throw new Error("Invalid useReturnValue hook");return t[0].shape})();return[{status:i,contentType:r,signature:e,description:"",errorMessage:""}]}return e[0].role==="union"?typeof e[0].shape=="string"?[{status:e[0].shape==="void"||e[0].shape==="null"?204:200,contentType:"application/json",signature:e[0].shape,description:"",errorMessage:""}]:e[0].shape.flatMap(i=>h([i])):e[0].role==="buffer"?[{status:200,contentType:"application/octet-stream",signature:e,description:"",errorMessage:""}]:[{status:200,contentType:"application/json",signature:e,description:"",errorMessage:""}]};exports.parseEndpoint=x;
2
2
  //# sourceMappingURL=parseEndpoint.cjs.map