moonflower 1.4.8 → 1.5.0

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.
Files changed (49) hide show
  1. package/dist/openapi/analyzerModule/analyzerModule.cjs +1 -1
  2. package/dist/openapi/analyzerModule/analyzerModule.cjs.map +1 -1
  3. package/dist/openapi/analyzerModule/analyzerModule.d.ts +18 -3
  4. package/dist/openapi/analyzerModule/analyzerModule.d.ts.map +1 -1
  5. package/dist/openapi/analyzerModule/analyzerModule.mjs +168 -112
  6. package/dist/openapi/analyzerModule/analyzerModule.mjs.map +1 -1
  7. package/dist/openapi/analyzerModule/analyzerWorker.cjs +2 -0
  8. package/dist/openapi/analyzerModule/analyzerWorker.cjs.map +1 -0
  9. package/dist/openapi/analyzerModule/analyzerWorker.d.ts +2 -0
  10. package/dist/openapi/analyzerModule/analyzerWorker.d.ts.map +1 -0
  11. package/dist/openapi/analyzerModule/analyzerWorker.mjs +44 -0
  12. package/dist/openapi/analyzerModule/analyzerWorker.mjs.map +1 -0
  13. package/dist/openapi/analyzerModule/nodeParsers.cjs +1 -1
  14. package/dist/openapi/analyzerModule/nodeParsers.cjs.map +1 -1
  15. package/dist/openapi/analyzerModule/nodeParsers.d.ts.map +1 -1
  16. package/dist/openapi/analyzerModule/nodeParsers.mjs +199 -264
  17. package/dist/openapi/analyzerModule/nodeParsers.mjs.map +1 -1
  18. package/dist/openapi/analyzerModule/parseEndpoint.cjs +1 -1
  19. package/dist/openapi/analyzerModule/parseEndpoint.cjs.map +1 -1
  20. package/dist/openapi/analyzerModule/parseEndpoint.d.ts +8 -1
  21. package/dist/openapi/analyzerModule/parseEndpoint.d.ts.map +1 -1
  22. package/dist/openapi/analyzerModule/parseEndpoint.mjs +121 -106
  23. package/dist/openapi/analyzerModule/parseEndpoint.mjs.map +1 -1
  24. package/dist/openapi/analyzerModule/test/TestCase.d.ts +1 -0
  25. package/dist/openapi/analyzerModule/test/TestCase.d.ts.map +1 -1
  26. package/dist/openapi/analyzerModule/test/workerGlobalSetup.d.ts +3 -0
  27. package/dist/openapi/analyzerModule/test/workerGlobalSetup.d.ts.map +1 -0
  28. package/dist/openapi/analyzerModule/workerPaths.d.ts +2 -0
  29. package/dist/openapi/analyzerModule/workerPaths.d.ts.map +1 -0
  30. package/dist/openapi/analyzerModule/workerPool.cjs +2 -0
  31. package/dist/openapi/analyzerModule/workerPool.cjs.map +1 -0
  32. package/dist/openapi/analyzerModule/workerPool.d.ts +33 -0
  33. package/dist/openapi/analyzerModule/workerPool.d.ts.map +1 -0
  34. package/dist/openapi/analyzerModule/workerPool.mjs +46 -0
  35. package/dist/openapi/analyzerModule/workerPool.mjs.map +1 -0
  36. package/package.json +1 -1
  37. package/src/openapi/analyzerModule/analyzerModule.ts +198 -25
  38. package/src/openapi/analyzerModule/analyzerWorker.ts +73 -0
  39. package/src/openapi/analyzerModule/nodeParsers.ts +4 -104
  40. package/src/openapi/analyzerModule/parseEndpoint.ts +31 -9
  41. package/src/openapi/analyzerModule/test/TestCase.ts +1 -0
  42. package/src/openapi/analyzerModule/test/openApiAnalyzer.spec.ts +2 -2
  43. package/src/openapi/analyzerModule/test/openApiAnalyzer.zod.spec.data.ts +6 -0
  44. package/src/openapi/analyzerModule/test/openApiAnalyzer.zod.spec.ts +9 -1
  45. package/src/openapi/analyzerModule/test/workerGlobalSetup.ts +25 -0
  46. package/src/openapi/analyzerModule/workerPaths.ts +4 -0
  47. package/src/openapi/analyzerModule/workerPool.ts +92 -0
  48. package/src/test/app.spec.ts +10 -1
  49. package/vite.config.ts +5 -0
@@ -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 j=require("crypto"),q=require("fs"),z=require("path"),U=require("url"),F=require("ts-morph"),f=require("../../utils/logger.cjs"),E=require("../discoveryModule/discoverImports/discoverImports.cjs"),I=require("../discoveryModule/discoverRouterFiles/discoverRouterFiles.cjs"),K=require("../discoveryModule/discoverRouters/discoverRouters.cjs"),W=require("../manager/OpenApiManager.cjs"),_=require("./getSourceFileTimestamp.cjs"),H=require("./nodeParsers.cjs"),k=require("./parseExposedModels.cjs"),A=require("./sourceFileCache.cjs"),D=require("./workerPool.cjs");var R=typeof document<"u"?document.currentScript:null;function B(s){const i=Object.create(null,{[Symbol.toStringTag]:{value:"Module"}});if(s){for(const n in s)if(n!=="default"){const a=Object.getOwnPropertyDescriptor(s,n);Object.defineProperty(i,n,a.get?a:{enumerable:!0,get:()=>s[n]})}}return i.default=s,Object.freeze(i)}const M=B(z);function V(){const s=["./analyzerWorker.mjs","./analyzerWorker.test.mjs"];for(const i of s){const n=new URL(i,typeof document>"u"?require("url").pathToFileURL(__filename).href:R&&R.tagName.toUpperCase()==="SCRIPT"&&R.src||new URL("openapi/analyzerModule/analyzerModule.cjs",document.baseURI).href);if(q.existsSync(U.fileURLToPath(n)))return n}throw new Error("analyzerWorker.mjs not found. Run yarn build, or run tests via yarn test (which compiles the worker first).")}const G=async({logLevel:s,tsconfigPath:i,sourceFilePaths:n,sourceFileDiscovery:a,incremental:l,profiling:m="stats"})=>{const r=W.OpenApiManager.getInstance();if(r.isReady())return;s&&f.Logger.setLevel(s),f.Logger.info("Preparing OpenAPI spec");const p=new F.Project({tsConfigFilePath:M.resolve(i),skipFileDependencyResolution:!0}),{explicitRouters:h,discoveredRouterFiles:g,allSourceFiles:N}=(()=>{const d=(n??[]).map(u=>M.resolve(u)).map(u=>p.getSourceFileOrThrow(u)),P=d.flatMap(u=>({fileName:u.getFilePath(),sourceFile:u,routers:K.discoverRouters(u)})),{discoveredRouterFiles:w,discoveredSourceFiles:L}=(()=>{if(a===!1)return{discoveredRouterFiles:[],discoveredSourceFiles:[]};const u=performance.now(),S=I.discoverRouterFiles({targetPath:typeof a=="object"?a.rootPath:".",tsConfigPath:i});return m!=="off"&&f.Logger.info(`File discovery took ${Math.round(performance.now()-u)}ms`),S})(),$=d.reduce((u,S)=>u.some(b=>b.getFilePath()===S.getFilePath())?u:u.concat(S),L);return{explicitRouters:P,discoveredRouterFiles:w,allSourceFiles:$}})(),T=h.reduce((t,c)=>t.some(d=>d.fileName===c.fileName)?t:t.concat(c),g),y=N.flatMap(t=>O(t)).filter(t=>!!t);y.length>0&&y[0]&&r.setHeader(y[0]);const x=N.flatMap(t=>v(t));r.setExposedModels(x);const e=typeof l=="object"&&l.cachePath?l.cachePath:M.resolve(process.cwd(),"node_modules",".cache","moonflower"),o=await C(T,{incremental:l!==!1,cachePath:e,timestampCache:{},profiling:m,tsconfigPath:M.resolve(i)});r.setStats({discoveredRouterFiles:g.map(t=>({path:t.fileName,routers:t.routers.named.map(c=>({name:c,endpoints:o.filter(d=>d.sourceFilePath===t.fileName).map(d=>`${d.method.toUpperCase()} ${d.path}`)}))})),explicitRouterFiles:h.map(t=>({path:t.fileName,routers:t.routers.named.map(c=>({name:c,endpoints:o.filter(d=>d.sourceFilePath===t.fileName).map(d=>`${d.method.toUpperCase()} ${d.path}`)}))}))}),r.setEndpoints(o),r.markAsReady()},C=async(s,i,n)=>{const a=i.profiling??"stats",l=performance.now(),m=[],r=[];for(const e of s){const o=_.getSourceFileTimestamp(e.sourceFile,i.timestampCache),t=i.incremental?A.SourceFileCache.getCachedResults(e.sourceFile,o,i.cachePath):null;t?(f.Logger.debug(`[${e.fileName}] Found cached results`),m.push({endpoints:t.endpoints,fileName:e.fileName,timing:0,endpointTimings:[]})):r.push({file:e,timestamp:o})}if(r.length===0)return a!=="off"&&f.Logger.info(`Router analysis took ${Math.round(performance.now()-l)}ms`),m.flatMap(e=>e.endpoints);const h=["get","post","put","delete","del","patch"].join("|"),g=[];for(const{file:e}of r)for(const o of e.routers.named){const t=new RegExp(`${o}\\.(?:${h})`);let c=0;e.sourceFile.forEachChild(d=>{const P=d.getText();t.test(P)&&(g.push({fileName:e.fileName,task:{taskId:j.randomUUID(),tsconfigPath:i.tsconfigPath,sourceFilePath:e.sourceFile.getFilePath(),routerName:o,endpointIndex:c}}),c++)})}const N=new D.WorkerPool(V());let T;try{T=await N.runAll(g.map(e=>e.task))}finally{N.terminate()}const y=new Map;for(const{file:e}of r)y.set(e.fileName,{endpoints:[],fileName:e.fileName,timing:0,endpointTimings:[]});for(let e=0;e<T.length;e++){const o=T[e],t=g[e].fileName,c=y.get(t);if("error"in o){f.Logger.error(`[${t}] Worker error: ${o.error}`);continue}c.endpoints.push(o.endpoint),c.timing+=o.timing,c.endpointTimings.push({method:o.endpoint.method,path:o.endpoint.path,timing:o.timing,sectionTimings:o.sectionTimings})}for(const{file:e,timestamp:o}of r){const t=y.get(e.fileName);t.endpoints.length>0&&A.SourceFileCache.cacheResults(e.sourceFile,o,i.cachePath,t.endpoints)}const x=[...m,...Array.from(y.values())];return a!=="off"&&f.Logger.info(`Router analysis took ${Math.round(performance.now()-l)}ms`),a==="stats"?x.map(e=>({fileName:e.fileName,timeTaken:e.timing})).sort((e,o)=>o.timeTaken-e.timeTaken).filter(e=>e.timeTaken>500).forEach(e=>{f.Logger.info(`- [${e.fileName}] Took ${Math.round(e.timeTaken)}ms to analyze`)}):a==="debug"&&x.map(e=>({fileName:e.fileName,timeTaken:e.timing,endpointTimings:e.endpointTimings})).sort((e,o)=>o.timeTaken-e.timeTaken).forEach(e=>{f.Logger.info(`- [${e.fileName}] Took ${Math.round(e.timeTaken)}ms to analyze`),e.endpointTimings.sort((o,t)=>t.timing-o.timing).forEach(o=>{f.Logger.info(` - ${o.method} ${o.path} (${Math.round(o.timing)}ms)`),o.sectionTimings.filter(t=>t.timing>=1).sort((t,c)=>c.timing-t.timing).forEach(t=>{f.Logger.info(` - ${t.section}: ${Math.round(t.timing)}ms`)})})}),x.flatMap(e=>e.endpoints)},O=s=>{const i=E.discoverImportedName({sourceFile:s,originalName:"useApiHeader"});if(!i)return null;const n=s.forEachChildAsArray().filter(r=>r.isKind(F.SyntaxKind.ExpressionStatement)).find(r=>i&&r.getText().startsWith(i));if(!n)return null;const a=n.getFirstDescendantByKindOrThrow(F.SyntaxKind.ObjectLiteralExpression),l=H.getValuesOfObjectLiteral(a),m=r=>typeof r=="string"||Array.isArray(r)&&r.every(p=>typeof p=="string")?r:r.reduce((p,h)=>typeof h=="string"?p:{...p,[h.identifier]:m(h.value)},{});return m(l)},v=s=>{const i=[],n=E.discoverImportedName({sourceFile:s,originalName:"useExposeApiModel"}),a=E.discoverImportedName({sourceFile:s,originalName:"useExposeNamedApiModels"});return s.forEachChildAsArray().filter(l=>l.isKind(F.SyntaxKind.ExpressionStatement)).map(l=>{if(n&&l.getText().startsWith(n)){const p=(l.getFirstChild()?.getChildrenOfKind(F.SyntaxKind.SyntaxList)||[])[0].getFirstChild();if(!p)return;i.push(k.parseExposedModel(p));return}if(a&&l.getText().startsWith(a)){const p=(l.getFirstChild()?.getChildrenOfKind(F.SyntaxKind.SyntaxList)||[])[0].getFirstChild();if(!p)return;k.parseNamedExposedModels(p).forEach(g=>i.push(g))}}),i};exports.analyzeMultipleSourceFiles=C;exports.analyzeSourceFileApiHeader=O;exports.analyzeSourceFileExposedModels=v;exports.prepareOpenApiSpec=G;
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 crypto from 'crypto'\nimport { existsSync } from 'fs'\nimport * as path from 'path'\nimport { fileURLToPath } from 'url'\n\nfunction resolveWorkerUrl(): URL {\n\tconst candidates = ['./analyzerWorker.mjs', './analyzerWorker.test.mjs']\n\tfor (const candidate of candidates) {\n\t\tconst url = new URL(candidate, import.meta.url)\n\t\tif (existsSync(fileURLToPath(url))) {\n\t\t\treturn url\n\t\t}\n\t}\n\tthrow new Error(\n\t\t'analyzerWorker.mjs not found. Run yarn build, or run tests via yarn test (which compiles the worker first).',\n\t)\n}\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'\nimport { WorkerPool, WorkerResult, WorkerTask } from './workerPool'\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 = async ({\n\tlogLevel,\n\ttsconfigPath,\n\tsourceFilePaths,\n\tsourceFileDiscovery,\n\tincremental,\n\tprofiling = 'stats',\n}: Props): Promise<void> => {\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 = await analyzeMultipleSourceFiles(filesToAnalyze, {\n\t\tincremental: incremental !== false,\n\t\tcachePath,\n\t\ttimestampCache: {},\n\t\tprofiling,\n\t\ttsconfigPath: path.resolve(tsconfigPath),\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 = async (\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\ttsconfigPath: string\n\t},\n\tfilterEndpointPaths?: string[],\n): Promise<EndpointData[]> => {\n\tconst profiling = config.profiling ?? 'stats'\n\tconst startTime = performance.now()\n\n\t// Separate cached files from those needing analysis\n\ttype CachedFile = { endpoints: EndpointData[]; fileName: string; timing: 0; endpointTimings: [] }\n\ttype UncachedFile = { file: DiscoveredSourceFile; timestamp: number }\n\n\tconst cached: CachedFile[] = []\n\tconst uncached: UncachedFile[] = []\n\n\tfor (const file of files) {\n\t\tconst timestamp = getSourceFileTimestamp(file.sourceFile, config.timestampCache)\n\t\tconst hit = config.incremental\n\t\t\t? SourceFileCache.getCachedResults(file.sourceFile, timestamp, config.cachePath)\n\t\t\t: null\n\t\tif (hit) {\n\t\t\tLogger.debug(`[${file.fileName}] Found cached results`)\n\t\t\tcached.push({ endpoints: hit.endpoints, fileName: file.fileName, timing: 0, endpointTimings: [] })\n\t\t} else {\n\t\t\tuncached.push({ file, timestamp })\n\t\t}\n\t}\n\n\tif (uncached.length === 0) {\n\t\tif (profiling !== 'off') {\n\t\t\tLogger.info(`Router analysis took ${Math.round(performance.now() - startTime)}ms`)\n\t\t}\n\t\treturn cached.flatMap((f) => f.endpoints)\n\t}\n\n\t// Build one task per endpoint across all uncached files\n\tconst OPERATIONS = ['get', 'post', 'put', 'delete', 'del', 'patch']\n\tconst operationsPattern = OPERATIONS.join('|')\n\n\ttype FileTask = { task: WorkerTask; fileName: string }\n\tconst allTasks: FileTask[] = []\n\n\tfor (const { file } of uncached) {\n\t\tfor (const routerName of file.routers.named) {\n\t\t\tconst routerPattern = new RegExp(`${routerName}\\\\.(?:${operationsPattern})`)\n\t\t\tlet endpointIndex = 0\n\t\t\tfile.sourceFile.forEachChild((node) => {\n\t\t\t\tconst nodeText = node.getText()\n\t\t\t\tif (routerPattern.test(nodeText)) {\n\t\t\t\t\tif (\n\t\t\t\t\t\t!filterEndpointPaths ||\n\t\t\t\t\t\tfilterEndpointPaths.some((p) => resolveEndpointPath(node)?.includes(p))\n\t\t\t\t\t) {\n\t\t\t\t\t\tallTasks.push({\n\t\t\t\t\t\t\tfileName: file.fileName,\n\t\t\t\t\t\t\ttask: {\n\t\t\t\t\t\t\t\ttaskId: crypto.randomUUID(),\n\t\t\t\t\t\t\t\ttsconfigPath: config.tsconfigPath,\n\t\t\t\t\t\t\t\tsourceFilePath: file.sourceFile.getFilePath(),\n\t\t\t\t\t\t\t\trouterName,\n\t\t\t\t\t\t\t\tendpointIndex,\n\t\t\t\t\t\t\t},\n\t\t\t\t\t\t})\n\t\t\t\t\t}\n\t\t\t\t\tendpointIndex++\n\t\t\t\t}\n\t\t\t})\n\t\t}\n\t}\n\n\t// Dispatch all tasks to the worker pool\n\tconst pool = new WorkerPool(resolveWorkerUrl())\n\n\ttype EndpointTiming = { method: string; path: string; timing: number; sectionTimings: SectionTiming[] }\n\ttype FileResult = {\n\t\tendpoints: EndpointData[]\n\t\tfileName: string\n\t\ttiming: number\n\t\tendpointTimings: EndpointTiming[]\n\t}\n\n\tlet results: WorkerResult[]\n\ttry {\n\t\tresults = await pool.runAll(allTasks.map((ft) => ft.task))\n\t} finally {\n\t\tpool.terminate()\n\t}\n\n\t// Group results by file\n\tconst byFile = new Map<string, FileResult>()\n\tfor (const { file } of uncached) {\n\t\tbyFile.set(file.fileName, { endpoints: [], fileName: file.fileName, timing: 0, endpointTimings: [] })\n\t}\n\n\tfor (let i = 0; i < results.length; i++) {\n\t\tconst result = results[i]\n\t\tconst fileName = allTasks[i].fileName\n\t\tconst fileResult = byFile.get(fileName)!\n\n\t\tif ('error' in result) {\n\t\t\tLogger.error(`[${fileName}] Worker error: ${result.error}`)\n\t\t\tcontinue\n\t\t}\n\n\t\tfileResult.endpoints.push(result.endpoint)\n\t\tfileResult.timing += result.timing\n\t\tfileResult.endpointTimings.push({\n\t\t\tmethod: result.endpoint.method,\n\t\t\tpath: result.endpoint.path,\n\t\t\ttiming: result.timing,\n\t\t\tsectionTimings: result.sectionTimings,\n\t\t})\n\t}\n\n\t// Write cache for each uncached file\n\tfor (const { file, timestamp } of uncached) {\n\t\tconst fileResult = byFile.get(file.fileName)!\n\t\tif (fileResult.endpoints.length > 0) {\n\t\t\tSourceFileCache.cacheResults(file.sourceFile, timestamp, config.cachePath, fileResult.endpoints)\n\t\t}\n\t}\n\n\tconst analyzedFiles = [...cached, ...Array.from(byFile.values())]\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) => ({ fileName: f.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) => ({ fileName: f.fileName, timeTaken: f.timing, endpointTimings: f.endpointTimings }))\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":["resolveWorkerUrl","candidates","candidate","url","_documentCurrentScript","existsSync","fileURLToPath","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","cached","uncached","timestamp","getSourceFileTimestamp","hit","SourceFileCache","f","operationsPattern","allTasks","routerName","routerPattern","endpointIndex","node","nodeText","crypto","pool","WorkerPool","results","ft","byFile","i","result","fileName","fileResult","analyzedFiles","a","b","t","ep","s","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,39 @@ 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) => Promise<void>;
21
29
  export declare const analyzeMultipleSourceFiles: (files: DiscoveredSourceFile[], config: {
22
30
  incremental: boolean;
23
31
  cachePath: string;
24
32
  timestampCache: TimestampCache;
25
- }, filterEndpointPaths?: string[]) => EndpointData[];
33
+ profiling?: "stats" | "off" | "debug";
34
+ tsconfigPath: string;
35
+ }, filterEndpointPaths?: string[]) => Promise<EndpointData[]>;
26
36
  export declare const analyzeSourceFileWithCache: (file: DiscoveredSourceFile, config: {
27
37
  incremental: boolean;
28
38
  cachePath: string;
29
39
  timestampCache: TimestampCache;
40
+ profiling?: "stats" | "off" | "debug";
30
41
  }, filterEndpointPaths?: string[]) => {
31
42
  endpoints: EndpointData[];
32
43
  timing: number;
44
+ endpointTimings: EndpointTiming[];
45
+ };
46
+ export declare const analyzeSourceFileEndpoints: (file: DiscoveredSourceFile, filterEndpointPaths?: string[]) => {
47
+ endpoints: EndpointData[];
48
+ endpointTimings: EndpointTiming[];
33
49
  };
34
- export declare const analyzeSourceFileEndpoints: (file: DiscoveredSourceFile, filterEndpointPaths?: string[]) => EndpointData[];
35
50
  export declare const analyzeSourceFileApiHeader: (sourceFile: SourceFile) => ApiDocsHeader | null;
36
51
  export declare const analyzeSourceFileExposedModels: (sourceFile: SourceFile) => ExposedModelData[];
37
52
  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":"AAiBA,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;AAK9D,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,KAAG,OAAO,CAAC,IAAI,CA0GtB,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;IACrC,YAAY,EAAE,MAAM,CAAA;CACpB,wBACqB,MAAM,EAAE,KAC5B,OAAO,CAAC,YAAY,EAAE,CAyJxB,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,199 @@
1
- import * as y from "path";
2
- import { Project as v, SyntaxKind as h } from "ts-morph";
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";
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";
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,
16
- sourceFilePaths: c,
17
- sourceFileDiscovery: a,
18
- incremental: s
1
+ import w from "crypto";
2
+ import { existsSync as O } from "fs";
3
+ import * as P from "path";
4
+ import { fileURLToPath as v } from "url";
5
+ import { Project as j, SyntaxKind as x } from "ts-morph";
6
+ import { Logger as u } from "../../utils/logger.mjs";
7
+ import { discoverImportedName as M } from "../discoveryModule/discoverImports/discoverImports.mjs";
8
+ import { discoverRouterFiles as L } from "../discoveryModule/discoverRouterFiles/discoverRouterFiles.mjs";
9
+ import { discoverRouters as z } from "../discoveryModule/discoverRouters/discoverRouters.mjs";
10
+ import { OpenApiManager as U } from "../manager/OpenApiManager.mjs";
11
+ import { getSourceFileTimestamp as b } from "./getSourceFileTimestamp.mjs";
12
+ import { getValuesOfObjectLiteral as W } from "./nodeParsers.mjs";
13
+ import { parseExposedModel as I, parseNamedExposedModels as K } from "./parseExposedModels.mjs";
14
+ import { SourceFileCache as R } from "./sourceFileCache.mjs";
15
+ import { WorkerPool as H } from "./workerPool.mjs";
16
+ function B() {
17
+ const l = ["./analyzerWorker.mjs", "./analyzerWorker.test.mjs"];
18
+ for (const s of l) {
19
+ const d = new URL(s, import.meta.url);
20
+ if (O(v(d)))
21
+ return d;
22
+ }
23
+ throw new Error(
24
+ "analyzerWorker.mjs not found. Run yarn build, or run tests via yarn test (which compiles the worker first)."
25
+ );
26
+ }
27
+ const ae = async ({
28
+ logLevel: l,
29
+ tsconfigPath: s,
30
+ sourceFilePaths: d,
31
+ sourceFileDiscovery: m,
32
+ incremental: r,
33
+ profiling: f = "stats"
19
34
  }) => {
20
- const e = k.getInstance();
21
- if (e.isReady())
35
+ const i = U.getInstance();
36
+ if (i.isReady())
22
37
  return;
23
- t && m.setLevel(t), m.info("Preparing OpenAPI spec");
24
- const o = new v({
25
- tsConfigFilePath: y.resolve(n),
38
+ l && u.setLevel(l), u.info("Preparing OpenAPI spec");
39
+ const c = new j({
40
+ tsConfigFilePath: P.resolve(s),
26
41
  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) => ({
42
+ }), { explicitRouters: h, discoveredRouterFiles: g, allSourceFiles: y } = (() => {
43
+ const a = (d ?? []).map((p) => P.resolve(p)).map((p) => c.getSourceFileOrThrow(p)), k = a.flatMap((p) => ({
29
44
  fileName: p.getFilePath(),
30
45
  sourceFile: p,
31
- routers: w(p)
32
- })), { discoveredRouterFiles: M, discoveredSourceFiles: R } = (() => {
33
- if (a === !1)
46
+ routers: z(p)
47
+ })), { discoveredRouterFiles: A, discoveredSourceFiles: C } = (() => {
48
+ if (m === !1)
34
49
  return { discoveredRouterFiles: [], discoveredSourceFiles: [] };
35
- const p = performance.now(), F = $({
36
- targetPath: typeof a == "object" ? a.rootPath : ".",
37
- tsConfigPath: n
50
+ const p = performance.now(), E = L({
51
+ targetPath: typeof m == "object" ? m.rootPath : ".",
52
+ tsConfigPath: s
38
53
  });
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
54
+ return f !== "off" && u.info(`File discovery took ${Math.round(performance.now() - p)}ms`), E;
55
+ })(), S = a.reduce(
56
+ (p, E) => p.some(($) => $.getFilePath() === E.getFilePath()) ? p : p.concat(E),
57
+ C
43
58
  );
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);
49
- 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: {}
59
+ return { explicitRouters: k, discoveredRouterFiles: A, allSourceFiles: S };
60
+ })(), N = h.reduce(
61
+ (t, n) => t.some((a) => a.fileName === n.fileName) ? t : t.concat(n),
62
+ g
63
+ ), F = y.flatMap((t) => V(t)).filter((t) => !!t);
64
+ F.length > 0 && F[0] && i.setHeader(F[0]);
65
+ const T = y.flatMap((t) => _(t));
66
+ i.setExposedModels(T);
67
+ const e = typeof r == "object" && r.cachePath ? r.cachePath : P.resolve(process.cwd(), "node_modules", ".cache", "moonflower"), o = await D(N, {
68
+ incremental: r !== !1,
69
+ cachePath: e,
70
+ timestampCache: {},
71
+ profiling: f,
72
+ tsconfigPath: P.resolve(s)
55
73
  });
56
- 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}`)
74
+ i.setStats({
75
+ discoveredRouterFiles: g.map((t) => ({
76
+ path: t.fileName,
77
+ routers: t.routers.named.map((n) => ({
78
+ name: n,
79
+ endpoints: o.filter((a) => a.sourceFilePath === t.fileName).map((a) => `${a.method.toUpperCase()} ${a.path}`)
62
80
  }))
63
81
  })),
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}`)
82
+ explicitRouterFiles: h.map((t) => ({
83
+ path: t.fileName,
84
+ routers: t.routers.named.map((n) => ({
85
+ name: n,
86
+ endpoints: o.filter((a) => a.sourceFilePath === t.fileName).map((a) => `${a.method.toUpperCase()} ${a.path}`)
69
87
  }))
70
88
  }))
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) => {
78
- 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)));
89
+ }), i.setEndpoints(o), i.markAsReady();
90
+ }, D = async (l, s, d) => {
91
+ const m = s.profiling ?? "stats", r = performance.now(), f = [], i = [];
92
+ for (const e of l) {
93
+ const o = b(e.sourceFile, s.timestampCache), t = s.incremental ? R.getCachedResults(e.sourceFile, o, s.cachePath) : null;
94
+ t ? (u.debug(`[${e.fileName}] Found cached results`), f.push({ endpoints: t.endpoints, fileName: e.fileName, timing: 0, endpointTimings: [] })) : i.push({ file: e, timestamp: o });
95
+ }
96
+ if (i.length === 0)
97
+ return m !== "off" && u.info(`Router analysis took ${Math.round(performance.now() - r)}ms`), f.flatMap((e) => e.endpoints);
98
+ const h = ["get", "post", "put", "delete", "del", "patch"].join("|"), g = [];
99
+ for (const { file: e } of i)
100
+ for (const o of e.routers.named) {
101
+ const t = new RegExp(`${o}\\.(?:${h})`);
102
+ let n = 0;
103
+ e.sourceFile.forEachChild((a) => {
104
+ const k = a.getText();
105
+ t.test(k) && (g.push({
106
+ fileName: e.fileName,
107
+ task: {
108
+ taskId: w.randomUUID(),
109
+ tsconfigPath: s.tsconfigPath,
110
+ sourceFilePath: e.sourceFile.getFilePath(),
111
+ routerName: o,
112
+ endpointIndex: n
113
+ }
114
+ }), n++);
115
+ });
116
+ }
117
+ const y = new H(B());
118
+ let N;
119
+ try {
120
+ N = await y.runAll(g.map((e) => e.task));
121
+ } finally {
122
+ y.terminate();
123
+ }
124
+ const F = /* @__PURE__ */ new Map();
125
+ for (const { file: e } of i)
126
+ F.set(e.fileName, { endpoints: [], fileName: e.fileName, timing: 0, endpointTimings: [] });
127
+ for (let e = 0; e < N.length; e++) {
128
+ const o = N[e], t = g[e].fileName, n = F.get(t);
129
+ if ("error" in o) {
130
+ u.error(`[${t}] Worker error: ${o.error}`);
131
+ continue;
132
+ }
133
+ n.endpoints.push(o.endpoint), n.timing += o.timing, n.endpointTimings.push({
134
+ method: o.endpoint.method,
135
+ path: o.endpoint.path,
136
+ timing: o.timing,
137
+ sectionTimings: o.sectionTimings
138
+ });
139
+ }
140
+ for (const { file: e, timestamp: o } of i) {
141
+ const t = F.get(e.fileName);
142
+ t.endpoints.length > 0 && R.cacheResults(e.sourceFile, o, s.cachePath, t.endpoints);
143
+ }
144
+ const T = [...f, ...Array.from(F.values())];
145
+ return m !== "off" && u.info(`Router analysis took ${Math.round(performance.now() - r)}ms`), m === "stats" ? T.map((e) => ({ fileName: e.fileName, timeTaken: e.timing })).sort((e, o) => o.timeTaken - e.timeTaken).filter((e) => e.timeTaken > 500).forEach((e) => {
146
+ u.info(`- [${e.fileName}] Took ${Math.round(e.timeTaken)}ms to analyze`);
147
+ }) : m === "debug" && T.map((e) => ({ fileName: e.fileName, timeTaken: e.timing, endpointTimings: e.endpointTimings })).sort((e, o) => o.timeTaken - e.timeTaken).forEach((e) => {
148
+ u.info(`- [${e.fileName}] Took ${Math.round(e.timeTaken)}ms to analyze`), e.endpointTimings.sort((o, t) => t.timing - o.timing).forEach((o) => {
149
+ u.info(` - ${o.method} ${o.path} (${Math.round(o.timing)}ms)`), o.sectionTimings.filter((t) => t.timing >= 1).sort((t, n) => n.timing - t.timing).forEach((t) => {
150
+ u.info(` - ${t.section}: ${Math.round(t.timing)}ms`);
151
+ });
94
152
  });
95
- }), c;
96
- }, B = (t) => {
97
- const n = E({
98
- sourceFile: t,
153
+ }), T.flatMap((e) => e.endpoints);
154
+ }, V = (l) => {
155
+ const s = M({
156
+ sourceFile: l,
99
157
  originalName: "useApiHeader"
100
158
  });
101
- if (!n)
159
+ if (!s)
102
160
  return null;
103
- const c = t.forEachChildAsArray().filter((o) => o.isKind(h.ExpressionStatement)).find((o) => n && o.getText().startsWith(n));
104
- if (!c)
161
+ const d = l.forEachChildAsArray().filter((i) => i.isKind(x.ExpressionStatement)).find((i) => s && i.getText().startsWith(s));
162
+ if (!d)
105
163
  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)
164
+ const m = d.getFirstDescendantByKindOrThrow(x.ObjectLiteralExpression), r = W(m), f = (i) => typeof i == "string" || Array.isArray(i) && i.every((c) => typeof c == "string") ? i : i.reduce((c, h) => typeof h == "string" ? c : {
165
+ ...c,
166
+ [h.identifier]: f(h.value)
109
167
  }, {});
110
- return e(s);
111
- }, V = (t) => {
112
- const n = [], c = E({
113
- sourceFile: t,
168
+ return f(r);
169
+ }, _ = (l) => {
170
+ const s = [], d = M({
171
+ sourceFile: l,
114
172
  originalName: "useExposeApiModel"
115
- }), a = E({
116
- sourceFile: t,
173
+ }), m = M({
174
+ sourceFile: l,
117
175
  originalName: "useExposeNamedApiModels"
118
176
  });
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)
177
+ return l.forEachChildAsArray().filter((r) => r.isKind(x.ExpressionStatement)).map((r) => {
178
+ if (d && r.getText().startsWith(d)) {
179
+ const c = (r.getFirstChild()?.getChildrenOfKind(x.SyntaxList) || [])[0].getFirstChild();
180
+ if (!c)
123
181
  return;
124
- n.push(K(r));
182
+ s.push(I(c));
125
183
  return;
126
184
  }
127
- if (a && s.getText().startsWith(a)) {
128
- const r = (s.getFirstChild()?.getChildrenOfKind(h.SyntaxList) || [])[0].getFirstChild();
129
- if (!r)
185
+ if (m && r.getText().startsWith(m)) {
186
+ const c = (r.getFirstChild()?.getChildrenOfKind(x.SyntaxList) || [])[0].getFirstChild();
187
+ if (!c)
130
188
  return;
131
- H(r).forEach((f) => n.push(f));
189
+ K(c).forEach((g) => s.push(g));
132
190
  }
133
- }), n;
191
+ }), s;
134
192
  };
135
193
  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
194
+ D as analyzeMultipleSourceFiles,
195
+ V as analyzeSourceFileApiHeader,
196
+ _ as analyzeSourceFileExposedModels,
197
+ ae as prepareOpenApiSpec
142
198
  };
143
199
  //# sourceMappingURL=analyzerModule.mjs.map