iec-cli 0.6.6

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 (443) hide show
  1. package/README.md +159 -0
  2. package/dist/commands/backup.d.ts +11 -0
  3. package/dist/commands/backup.d.ts.map +1 -0
  4. package/dist/commands/backup.js +112 -0
  5. package/dist/commands/backup.js.map +1 -0
  6. package/dist/commands/config.d.ts +28 -0
  7. package/dist/commands/config.d.ts.map +1 -0
  8. package/dist/commands/config.js +190 -0
  9. package/dist/commands/config.js.map +1 -0
  10. package/dist/commands/deploy.d.ts +16 -0
  11. package/dist/commands/deploy.d.ts.map +1 -0
  12. package/dist/commands/deploy.js +235 -0
  13. package/dist/commands/deploy.js.map +1 -0
  14. package/dist/commands/destroy.d.ts +12 -0
  15. package/dist/commands/destroy.d.ts.map +1 -0
  16. package/dist/commands/destroy.js +62 -0
  17. package/dist/commands/destroy.js.map +1 -0
  18. package/dist/commands/domains.d.ts +36 -0
  19. package/dist/commands/domains.d.ts.map +1 -0
  20. package/dist/commands/domains.js +323 -0
  21. package/dist/commands/domains.js.map +1 -0
  22. package/dist/commands/env.d.ts +7 -0
  23. package/dist/commands/env.d.ts.map +1 -0
  24. package/dist/commands/env.js +42 -0
  25. package/dist/commands/env.js.map +1 -0
  26. package/dist/commands/git.d.ts +32 -0
  27. package/dist/commands/git.d.ts.map +1 -0
  28. package/dist/commands/git.js +340 -0
  29. package/dist/commands/git.js.map +1 -0
  30. package/dist/commands/init-ai.d.ts +7 -0
  31. package/dist/commands/init-ai.d.ts.map +1 -0
  32. package/dist/commands/init-ai.js +257 -0
  33. package/dist/commands/init-ai.js.map +1 -0
  34. package/dist/commands/init.d.ts +8 -0
  35. package/dist/commands/init.d.ts.map +1 -0
  36. package/dist/commands/init.js +228 -0
  37. package/dist/commands/init.js.map +1 -0
  38. package/dist/commands/link.d.ts +10 -0
  39. package/dist/commands/link.d.ts.map +1 -0
  40. package/dist/commands/link.js +193 -0
  41. package/dist/commands/link.js.map +1 -0
  42. package/dist/commands/login.d.ts +16 -0
  43. package/dist/commands/login.d.ts.map +1 -0
  44. package/dist/commands/login.js +468 -0
  45. package/dist/commands/login.js.map +1 -0
  46. package/dist/commands/logs.d.ts +16 -0
  47. package/dist/commands/logs.d.ts.map +1 -0
  48. package/dist/commands/logs.js +362 -0
  49. package/dist/commands/logs.js.map +1 -0
  50. package/dist/commands/oauth.d.ts +33 -0
  51. package/dist/commands/oauth.d.ts.map +1 -0
  52. package/dist/commands/oauth.js +181 -0
  53. package/dist/commands/oauth.js.map +1 -0
  54. package/dist/commands/preflight.d.ts +8 -0
  55. package/dist/commands/preflight.d.ts.map +1 -0
  56. package/dist/commands/preflight.js +190 -0
  57. package/dist/commands/preflight.js.map +1 -0
  58. package/dist/commands/program.d.ts +7 -0
  59. package/dist/commands/program.d.ts.map +1 -0
  60. package/dist/commands/program.js +59 -0
  61. package/dist/commands/program.js.map +1 -0
  62. package/dist/commands/programs.d.ts +7 -0
  63. package/dist/commands/programs.d.ts.map +1 -0
  64. package/dist/commands/programs.js +42 -0
  65. package/dist/commands/programs.js.map +1 -0
  66. package/dist/commands/push.d.ts +12 -0
  67. package/dist/commands/push.d.ts.map +1 -0
  68. package/dist/commands/push.js +166 -0
  69. package/dist/commands/push.js.map +1 -0
  70. package/dist/commands/register-scaffold.test.d.ts +2 -0
  71. package/dist/commands/register-scaffold.test.d.ts.map +1 -0
  72. package/dist/commands/register-scaffold.test.js +291 -0
  73. package/dist/commands/register-scaffold.test.js.map +1 -0
  74. package/dist/commands/register.d.ts +7 -0
  75. package/dist/commands/register.d.ts.map +1 -0
  76. package/dist/commands/register.js +304 -0
  77. package/dist/commands/register.js.map +1 -0
  78. package/dist/commands/rollback.d.ts +11 -0
  79. package/dist/commands/rollback.d.ts.map +1 -0
  80. package/dist/commands/rollback.js +147 -0
  81. package/dist/commands/rollback.js.map +1 -0
  82. package/dist/commands/sample.d.ts +15 -0
  83. package/dist/commands/sample.d.ts.map +1 -0
  84. package/dist/commands/sample.js +312 -0
  85. package/dist/commands/sample.js.map +1 -0
  86. package/dist/commands/scopes.d.ts +29 -0
  87. package/dist/commands/scopes.d.ts.map +1 -0
  88. package/dist/commands/scopes.js +203 -0
  89. package/dist/commands/scopes.js.map +1 -0
  90. package/dist/commands/services.d.ts +13 -0
  91. package/dist/commands/services.d.ts.map +1 -0
  92. package/dist/commands/services.js +95 -0
  93. package/dist/commands/services.js.map +1 -0
  94. package/dist/commands/settings.d.ts +10 -0
  95. package/dist/commands/settings.d.ts.map +1 -0
  96. package/dist/commands/settings.js +41 -0
  97. package/dist/commands/settings.js.map +1 -0
  98. package/dist/commands/signup.d.ts +9 -0
  99. package/dist/commands/signup.d.ts.map +1 -0
  100. package/dist/commands/signup.js +192 -0
  101. package/dist/commands/signup.js.map +1 -0
  102. package/dist/commands/status.d.ts +9 -0
  103. package/dist/commands/status.d.ts.map +1 -0
  104. package/dist/commands/status.js +170 -0
  105. package/dist/commands/status.js.map +1 -0
  106. package/dist/commands/troubleshoot.d.ts +19 -0
  107. package/dist/commands/troubleshoot.d.ts.map +1 -0
  108. package/dist/commands/troubleshoot.js +465 -0
  109. package/dist/commands/troubleshoot.js.map +1 -0
  110. package/dist/commands/update.d.ts +6 -0
  111. package/dist/commands/update.d.ts.map +1 -0
  112. package/dist/commands/update.js +64 -0
  113. package/dist/commands/update.js.map +1 -0
  114. package/dist/commands/user.d.ts +17 -0
  115. package/dist/commands/user.d.ts.map +1 -0
  116. package/dist/commands/user.js +126 -0
  117. package/dist/commands/user.js.map +1 -0
  118. package/dist/commands/versions.d.ts +16 -0
  119. package/dist/commands/versions.d.ts.map +1 -0
  120. package/dist/commands/versions.js +116 -0
  121. package/dist/commands/versions.js.map +1 -0
  122. package/dist/index.d.ts +3 -0
  123. package/dist/index.d.ts.map +1 -0
  124. package/dist/index.js +556 -0
  125. package/dist/index.js.map +1 -0
  126. package/dist/lib/__tests__/scaffold-integration.test.d.ts +2 -0
  127. package/dist/lib/__tests__/scaffold-integration.test.d.ts.map +1 -0
  128. package/dist/lib/__tests__/scaffold-integration.test.js +321 -0
  129. package/dist/lib/__tests__/scaffold-integration.test.js.map +1 -0
  130. package/dist/lib/ai/client.d.ts +39 -0
  131. package/dist/lib/ai/client.d.ts.map +1 -0
  132. package/dist/lib/ai/client.js +236 -0
  133. package/dist/lib/ai/client.js.map +1 -0
  134. package/dist/lib/ai/generator.d.ts +17 -0
  135. package/dist/lib/ai/generator.d.ts.map +1 -0
  136. package/dist/lib/ai/generator.js +338 -0
  137. package/dist/lib/ai/generator.js.map +1 -0
  138. package/dist/lib/ai/prompts.d.ts +47 -0
  139. package/dist/lib/ai/prompts.d.ts.map +1 -0
  140. package/dist/lib/ai/prompts.js +247 -0
  141. package/dist/lib/ai/prompts.js.map +1 -0
  142. package/dist/lib/ai/scanner.d.ts +50 -0
  143. package/dist/lib/ai/scanner.d.ts.map +1 -0
  144. package/dist/lib/ai/scanner.js +237 -0
  145. package/dist/lib/ai/scanner.js.map +1 -0
  146. package/dist/lib/api.d.ts +18 -0
  147. package/dist/lib/api.d.ts.map +1 -0
  148. package/dist/lib/api.js +67 -0
  149. package/dist/lib/api.js.map +1 -0
  150. package/dist/lib/auto-register.d.ts +12 -0
  151. package/dist/lib/auto-register.d.ts.map +1 -0
  152. package/dist/lib/auto-register.js +42 -0
  153. package/dist/lib/auto-register.js.map +1 -0
  154. package/dist/lib/bio.d.ts +90 -0
  155. package/dist/lib/bio.d.ts.map +1 -0
  156. package/dist/lib/bio.js +247 -0
  157. package/dist/lib/bio.js.map +1 -0
  158. package/dist/lib/builder.d.ts +65 -0
  159. package/dist/lib/builder.d.ts.map +1 -0
  160. package/dist/lib/builder.js +192 -0
  161. package/dist/lib/builder.js.map +1 -0
  162. package/dist/lib/catalog.d.ts +3 -0
  163. package/dist/lib/catalog.d.ts.map +1 -0
  164. package/dist/lib/catalog.js +17 -0
  165. package/dist/lib/catalog.js.map +1 -0
  166. package/dist/lib/config.d.ts +20 -0
  167. package/dist/lib/config.d.ts.map +1 -0
  168. package/dist/lib/config.js +124 -0
  169. package/dist/lib/config.js.map +1 -0
  170. package/dist/lib/forgejo.d.ts +188 -0
  171. package/dist/lib/forgejo.d.ts.map +1 -0
  172. package/dist/lib/forgejo.js +305 -0
  173. package/dist/lib/forgejo.js.map +1 -0
  174. package/dist/lib/output.d.ts +12 -0
  175. package/dist/lib/output.d.ts.map +1 -0
  176. package/dist/lib/output.js +39 -0
  177. package/dist/lib/output.js.map +1 -0
  178. package/dist/lib/scaffold.d.ts +30 -0
  179. package/dist/lib/scaffold.d.ts.map +1 -0
  180. package/dist/lib/scaffold.js +180 -0
  181. package/dist/lib/scaffold.js.map +1 -0
  182. package/dist/lib/scope-requests.d.ts +17 -0
  183. package/dist/lib/scope-requests.d.ts.map +1 -0
  184. package/dist/lib/scope-requests.js +78 -0
  185. package/dist/lib/scope-requests.js.map +1 -0
  186. package/dist/lib/scope-requests.test.d.ts +2 -0
  187. package/dist/lib/scope-requests.test.d.ts.map +1 -0
  188. package/dist/lib/scope-requests.test.js +239 -0
  189. package/dist/lib/scope-requests.test.js.map +1 -0
  190. package/dist/lib/troubleshoot/analyzer.d.ts +14 -0
  191. package/dist/lib/troubleshoot/analyzer.d.ts.map +1 -0
  192. package/dist/lib/troubleshoot/analyzer.js +541 -0
  193. package/dist/lib/troubleshoot/analyzer.js.map +1 -0
  194. package/dist/lib/troubleshoot/auto-fix.d.ts +29 -0
  195. package/dist/lib/troubleshoot/auto-fix.d.ts.map +1 -0
  196. package/dist/lib/troubleshoot/auto-fix.js +373 -0
  197. package/dist/lib/troubleshoot/auto-fix.js.map +1 -0
  198. package/dist/lib/troubleshoot/index.d.ts +5 -0
  199. package/dist/lib/troubleshoot/index.d.ts.map +1 -0
  200. package/dist/lib/troubleshoot/index.js +6 -0
  201. package/dist/lib/troubleshoot/index.js.map +1 -0
  202. package/dist/lib/troubleshoot/log-fetcher.d.ts +43 -0
  203. package/dist/lib/troubleshoot/log-fetcher.d.ts.map +1 -0
  204. package/dist/lib/troubleshoot/log-fetcher.js +431 -0
  205. package/dist/lib/troubleshoot/log-fetcher.js.map +1 -0
  206. package/dist/lib/troubleshoot/redactor.d.ts +35 -0
  207. package/dist/lib/troubleshoot/redactor.d.ts.map +1 -0
  208. package/dist/lib/troubleshoot/redactor.js +208 -0
  209. package/dist/lib/troubleshoot/redactor.js.map +1 -0
  210. package/dist/lib/validators/catalog-schemas.d.ts +59 -0
  211. package/dist/lib/validators/catalog-schemas.d.ts.map +1 -0
  212. package/dist/lib/validators/catalog-schemas.js +32 -0
  213. package/dist/lib/validators/catalog-schemas.js.map +1 -0
  214. package/dist/lib/validators/catalog-validator.d.ts +3 -0
  215. package/dist/lib/validators/catalog-validator.d.ts.map +1 -0
  216. package/dist/lib/validators/catalog-validator.js +247 -0
  217. package/dist/lib/validators/catalog-validator.js.map +1 -0
  218. package/dist/lib/validators/catalog-validator.test.d.ts +2 -0
  219. package/dist/lib/validators/catalog-validator.test.d.ts.map +1 -0
  220. package/dist/lib/validators/catalog-validator.test.js +420 -0
  221. package/dist/lib/validators/catalog-validator.test.js.map +1 -0
  222. package/dist/lib/validators/dockerfile-validator.d.ts +5 -0
  223. package/dist/lib/validators/dockerfile-validator.d.ts.map +1 -0
  224. package/dist/lib/validators/dockerfile-validator.js +295 -0
  225. package/dist/lib/validators/dockerfile-validator.js.map +1 -0
  226. package/dist/lib/validators/dockerfile-validator.test.d.ts +2 -0
  227. package/dist/lib/validators/dockerfile-validator.test.d.ts.map +1 -0
  228. package/dist/lib/validators/dockerfile-validator.test.js +161 -0
  229. package/dist/lib/validators/dockerfile-validator.test.js.map +1 -0
  230. package/dist/lib/validators/env-validator.d.ts +3 -0
  231. package/dist/lib/validators/env-validator.d.ts.map +1 -0
  232. package/dist/lib/validators/env-validator.js +268 -0
  233. package/dist/lib/validators/env-validator.js.map +1 -0
  234. package/dist/lib/validators/git-validator.d.ts +3 -0
  235. package/dist/lib/validators/git-validator.d.ts.map +1 -0
  236. package/dist/lib/validators/git-validator.js +236 -0
  237. package/dist/lib/validators/git-validator.js.map +1 -0
  238. package/dist/lib/validators/health-detector.d.ts +4 -0
  239. package/dist/lib/validators/health-detector.d.ts.map +1 -0
  240. package/dist/lib/validators/health-detector.js +172 -0
  241. package/dist/lib/validators/health-detector.js.map +1 -0
  242. package/dist/lib/validators/health-detector.test.d.ts +2 -0
  243. package/dist/lib/validators/health-detector.test.d.ts.map +1 -0
  244. package/dist/lib/validators/health-detector.test.js +118 -0
  245. package/dist/lib/validators/health-detector.test.js.map +1 -0
  246. package/dist/lib/validators/index.d.ts +8 -0
  247. package/dist/lib/validators/index.d.ts.map +1 -0
  248. package/dist/lib/validators/index.js +8 -0
  249. package/dist/lib/validators/index.js.map +1 -0
  250. package/dist/lib/validators/route-validator.d.ts +3 -0
  251. package/dist/lib/validators/route-validator.d.ts.map +1 -0
  252. package/dist/lib/validators/route-validator.js +238 -0
  253. package/dist/lib/validators/route-validator.js.map +1 -0
  254. package/dist/lib/validators/scope-validator.d.ts +3 -0
  255. package/dist/lib/validators/scope-validator.d.ts.map +1 -0
  256. package/dist/lib/validators/scope-validator.js +141 -0
  257. package/dist/lib/validators/scope-validator.js.map +1 -0
  258. package/dist/lib/watch.d.ts +35 -0
  259. package/dist/lib/watch.d.ts.map +1 -0
  260. package/dist/lib/watch.js +171 -0
  261. package/dist/lib/watch.js.map +1 -0
  262. package/dist/templates/api/.env.example +15 -0
  263. package/dist/templates/api/Dockerfile +45 -0
  264. package/dist/templates/api/README.md +85 -0
  265. package/dist/templates/api/catalog-info.yaml +22 -0
  266. package/dist/templates/api/gitignore +31 -0
  267. package/dist/templates/api/helm/{{name}}/Chart.yaml +9 -0
  268. package/dist/templates/api/helm/{{name}}/templates/deployment.yaml +72 -0
  269. package/dist/templates/api/helm/{{name}}/templates/ingress.yaml +43 -0
  270. package/dist/templates/api/helm/{{name}}/templates/service.yaml +17 -0
  271. package/dist/templates/api/helm/{{name}}/values.yaml +64 -0
  272. package/dist/templates/api/package.json +33 -0
  273. package/dist/templates/api/src/index.ts +61 -0
  274. package/dist/templates/api/src/routes/example.ts +165 -0
  275. package/dist/templates/api/tsconfig.json +18 -0
  276. package/dist/templates/crosspod/README.md +87 -0
  277. package/dist/templates/crosspod/service-a/Dockerfile +27 -0
  278. package/dist/templates/crosspod/service-a/catalog-info.yaml +54 -0
  279. package/dist/templates/crosspod/service-a/gitignore +5 -0
  280. package/dist/templates/crosspod/service-a/helm/{{name}}-service-a/Chart.yaml +6 -0
  281. package/dist/templates/crosspod/service-a/helm/{{name}}-service-a/templates/deployment.yaml +72 -0
  282. package/dist/templates/crosspod/service-a/helm/{{name}}-service-a/templates/ingress.yaml +43 -0
  283. package/dist/templates/crosspod/service-a/helm/{{name}}-service-a/templates/service.yaml +17 -0
  284. package/dist/templates/crosspod/service-a/helm/{{name}}-service-a/values.yaml +44 -0
  285. package/dist/templates/crosspod/service-a/package.json +29 -0
  286. package/dist/templates/crosspod/service-a/src/index.ts +89 -0
  287. package/dist/templates/crosspod/service-a/tsconfig.json +14 -0
  288. package/dist/templates/crosspod/service-b/Dockerfile +27 -0
  289. package/dist/templates/crosspod/service-b/catalog-info.yaml +27 -0
  290. package/dist/templates/crosspod/service-b/gitignore +5 -0
  291. package/dist/templates/crosspod/service-b/helm/{{name}}-service-b/Chart.yaml +6 -0
  292. package/dist/templates/crosspod/service-b/helm/{{name}}-service-b/templates/deployment.yaml +72 -0
  293. package/dist/templates/crosspod/service-b/helm/{{name}}-service-b/templates/ingress.yaml +43 -0
  294. package/dist/templates/crosspod/service-b/helm/{{name}}-service-b/templates/service.yaml +17 -0
  295. package/dist/templates/crosspod/service-b/helm/{{name}}-service-b/values.yaml +47 -0
  296. package/dist/templates/crosspod/service-b/package.json +26 -0
  297. package/dist/templates/crosspod/service-b/src/index.ts +143 -0
  298. package/dist/templates/crosspod/service-b/tsconfig.json +14 -0
  299. package/dist/templates/mga/{{name}}-api/.env.example +5 -0
  300. package/dist/templates/mga/{{name}}-api/Dockerfile +45 -0
  301. package/dist/templates/mga/{{name}}-api/catalog-info.yaml +44 -0
  302. package/dist/templates/mga/{{name}}-api/gitignore +31 -0
  303. package/dist/templates/mga/{{name}}-api/helm/{{name}}-api/Chart.yaml +9 -0
  304. package/dist/templates/mga/{{name}}-api/helm/{{name}}-api/templates/deployment.yaml +72 -0
  305. package/dist/templates/mga/{{name}}-api/helm/{{name}}-api/templates/ingress.yaml +43 -0
  306. package/dist/templates/mga/{{name}}-api/helm/{{name}}-api/templates/service.yaml +17 -0
  307. package/dist/templates/mga/{{name}}-api/helm/{{name}}-api/values.yaml +65 -0
  308. package/dist/templates/mga/{{name}}-api/package.json +37 -0
  309. package/dist/templates/mga/{{name}}-api/src/index.ts +82 -0
  310. package/dist/templates/mga/{{name}}-api/src/lib/mongo.ts +19 -0
  311. package/dist/templates/mga/{{name}}-api/src/middleware/auth.ts +133 -0
  312. package/dist/templates/mga/{{name}}-api/src/models/agent.ts +45 -0
  313. package/dist/templates/mga/{{name}}-api/src/models/application.ts +58 -0
  314. package/dist/templates/mga/{{name}}-api/src/models/invite.ts +39 -0
  315. package/dist/templates/mga/{{name}}-api/src/models/policy.ts +59 -0
  316. package/dist/templates/mga/{{name}}-api/src/models/product.ts +51 -0
  317. package/dist/templates/mga/{{name}}-api/src/models/quote.ts +68 -0
  318. package/dist/templates/mga/{{name}}-api/src/routes/agents.ts +98 -0
  319. package/dist/templates/mga/{{name}}-api/src/routes/applications.ts +170 -0
  320. package/dist/templates/mga/{{name}}-api/src/routes/invites.ts +146 -0
  321. package/dist/templates/mga/{{name}}-api/src/routes/policies.ts +70 -0
  322. package/dist/templates/mga/{{name}}-api/src/routes/products.ts +154 -0
  323. package/dist/templates/mga/{{name}}-api/src/routes/quotes.ts +263 -0
  324. package/dist/templates/mga/{{name}}-api/src/routes/reports.ts +199 -0
  325. package/dist/templates/mga/{{name}}-api/tsconfig.json +18 -0
  326. package/dist/templates/mga/{{name}}-web/.dockerignore +6 -0
  327. package/dist/templates/mga/{{name}}-web/.env.example +6 -0
  328. package/dist/templates/mga/{{name}}-web/Dockerfile +51 -0
  329. package/dist/templates/mga/{{name}}-web/catalog-info.yaml +24 -0
  330. package/dist/templates/mga/{{name}}-web/dockerignore +6 -0
  331. package/dist/templates/mga/{{name}}-web/gitignore +36 -0
  332. package/dist/templates/mga/{{name}}-web/helm/{{name}}-web/Chart.yaml +9 -0
  333. package/dist/templates/mga/{{name}}-web/helm/{{name}}-web/templates/_helpers.tpl +49 -0
  334. package/dist/templates/mga/{{name}}-web/helm/{{name}}-web/templates/configmap.yaml +10 -0
  335. package/dist/templates/mga/{{name}}-web/helm/{{name}}-web/templates/deployment.yaml +56 -0
  336. package/dist/templates/mga/{{name}}-web/helm/{{name}}-web/templates/ingress.yaml +41 -0
  337. package/dist/templates/mga/{{name}}-web/helm/{{name}}-web/templates/service.yaml +15 -0
  338. package/dist/templates/mga/{{name}}-web/helm/{{name}}-web/values.yaml +70 -0
  339. package/dist/templates/mga/{{name}}-web/next.config.js +8 -0
  340. package/dist/templates/mga/{{name}}-web/package.json +30 -0
  341. package/dist/templates/mga/{{name}}-web/public/.gitkeep +0 -0
  342. package/dist/templates/mga/{{name}}-web/public/gitkeep +0 -0
  343. package/dist/templates/mga/{{name}}-web/src/app/api/auth/callback/route.ts +77 -0
  344. package/dist/templates/mga/{{name}}-web/src/app/api/auth/login/route.ts +16 -0
  345. package/dist/templates/mga/{{name}}-web/src/app/api/auth/logout/route.ts +7 -0
  346. package/dist/templates/mga/{{name}}-web/src/app/api/auth/session/route.ts +42 -0
  347. package/dist/templates/mga/{{name}}-web/src/app/api/health/route.ts +10 -0
  348. package/dist/templates/mga/{{name}}-web/src/app/api/proxy/applications/route.ts +47 -0
  349. package/dist/templates/mga/{{name}}-web/src/app/applications/layout.tsx +17 -0
  350. package/dist/templates/mga/{{name}}-web/src/app/applications/new/page.tsx +374 -0
  351. package/dist/templates/mga/{{name}}-web/src/app/applications/page.tsx +109 -0
  352. package/dist/templates/mga/{{name}}-web/src/app/dashboard/admin/page.tsx +179 -0
  353. package/dist/templates/mga/{{name}}-web/src/app/dashboard/agent/page.tsx +125 -0
  354. package/dist/templates/mga/{{name}}-web/src/app/dashboard/insured/page.tsx +145 -0
  355. package/dist/templates/mga/{{name}}-web/src/app/dashboard/layout.tsx +17 -0
  356. package/dist/templates/mga/{{name}}-web/src/app/dashboard/page.tsx +27 -0
  357. package/dist/templates/mga/{{name}}-web/src/app/dashboard/shell.tsx +89 -0
  358. package/dist/templates/mga/{{name}}-web/src/app/globals.css +477 -0
  359. package/dist/templates/mga/{{name}}-web/src/app/layout.tsx +19 -0
  360. package/dist/templates/mga/{{name}}-web/src/app/page.tsx +74 -0
  361. package/dist/templates/mga/{{name}}-web/src/app/reports/layout.tsx +21 -0
  362. package/dist/templates/mga/{{name}}-web/src/app/reports/page.tsx +231 -0
  363. package/dist/templates/mga/{{name}}-web/src/lib/api.ts +79 -0
  364. package/dist/templates/mga/{{name}}-web/src/lib/auth.ts +285 -0
  365. package/dist/templates/mga/{{name}}-web/src/middleware.ts +71 -0
  366. package/dist/templates/mga/{{name}}-web/tsconfig.json +26 -0
  367. package/dist/templates/nextjs/.env.example +11 -0
  368. package/dist/templates/nextjs/Dockerfile +51 -0
  369. package/dist/templates/nextjs/README.md +87 -0
  370. package/dist/templates/nextjs/catalog-info.yaml +16 -0
  371. package/dist/templates/nextjs/gitignore +36 -0
  372. package/dist/templates/nextjs/helm/{{name}}/Chart.yaml +9 -0
  373. package/dist/templates/nextjs/helm/{{name}}/templates/deployment.yaml +72 -0
  374. package/dist/templates/nextjs/helm/{{name}}/templates/ingress.yaml +43 -0
  375. package/dist/templates/nextjs/helm/{{name}}/templates/service.yaml +17 -0
  376. package/dist/templates/nextjs/helm/{{name}}/values.yaml +60 -0
  377. package/dist/templates/nextjs/next.config.js +23 -0
  378. package/dist/templates/nextjs/package.json +29 -0
  379. package/dist/templates/nextjs/public/.gitkeep +0 -0
  380. package/dist/templates/nextjs/public/gitkeep +0 -0
  381. package/dist/templates/nextjs/src/app/api/example/route.ts +63 -0
  382. package/dist/templates/nextjs/src/app/api/health/route.ts +10 -0
  383. package/dist/templates/nextjs/src/app/layout.tsx +18 -0
  384. package/dist/templates/nextjs/src/app/page.tsx +49 -0
  385. package/dist/templates/nextjs/tsconfig.json +26 -0
  386. package/dist/templates/nextjs-oauth/.dockerignore +6 -0
  387. package/dist/templates/nextjs-oauth/.env.example +12 -0
  388. package/dist/templates/nextjs-oauth/Dockerfile +51 -0
  389. package/dist/templates/nextjs-oauth/README.md +115 -0
  390. package/dist/templates/nextjs-oauth/catalog-info.yaml +17 -0
  391. package/dist/templates/nextjs-oauth/dockerignore +6 -0
  392. package/dist/templates/nextjs-oauth/gitignore +36 -0
  393. package/dist/templates/nextjs-oauth/helm/Chart.yaml +6 -0
  394. package/dist/templates/nextjs-oauth/helm/templates/_helpers.tpl +49 -0
  395. package/dist/templates/nextjs-oauth/helm/templates/configmap.yaml +10 -0
  396. package/dist/templates/nextjs-oauth/helm/templates/deployment.yaml +56 -0
  397. package/dist/templates/nextjs-oauth/helm/templates/ingress.yaml +41 -0
  398. package/dist/templates/nextjs-oauth/helm/templates/service.yaml +15 -0
  399. package/dist/templates/nextjs-oauth/helm/values.yaml +69 -0
  400. package/dist/templates/nextjs-oauth/helm/{{name}}/Chart.yaml +9 -0
  401. package/dist/templates/nextjs-oauth/helm/{{name}}/templates/deployment.yaml +68 -0
  402. package/dist/templates/nextjs-oauth/helm/{{name}}/templates/service.yaml +17 -0
  403. package/dist/templates/nextjs-oauth/helm/{{name}}/values.yaml +51 -0
  404. package/dist/templates/nextjs-oauth/next.config.js +23 -0
  405. package/dist/templates/nextjs-oauth/package.json +30 -0
  406. package/dist/templates/nextjs-oauth/public/.gitkeep +0 -0
  407. package/dist/templates/nextjs-oauth/public/gitkeep +0 -0
  408. package/dist/templates/nextjs-oauth/src/app/api/auth/callback/route.ts +77 -0
  409. package/dist/templates/nextjs-oauth/src/app/api/auth/login/route.ts +16 -0
  410. package/dist/templates/nextjs-oauth/src/app/api/auth/logout/route.ts +7 -0
  411. package/dist/templates/nextjs-oauth/src/app/api/auth/session/route.ts +42 -0
  412. package/dist/templates/nextjs-oauth/src/app/api/example/route.ts +63 -0
  413. package/dist/templates/nextjs-oauth/src/app/api/health/route.ts +10 -0
  414. package/dist/templates/nextjs-oauth/src/app/dashboard/page.tsx +92 -0
  415. package/dist/templates/nextjs-oauth/src/app/layout.tsx +18 -0
  416. package/dist/templates/nextjs-oauth/src/app/page.tsx +110 -0
  417. package/dist/templates/nextjs-oauth/src/lib/auth.ts +285 -0
  418. package/dist/templates/nextjs-oauth/src/middleware.ts +71 -0
  419. package/dist/templates/nextjs-oauth/tsconfig.json +26 -0
  420. package/dist/templates/static/catalog-info.yaml +14 -0
  421. package/dist/templates/static/index.html +16 -0
  422. package/dist/templates/static/styles.css +39 -0
  423. package/dist/templates/worker/.env.example +13 -0
  424. package/dist/templates/worker/Dockerfile +26 -0
  425. package/dist/templates/worker/README.md +106 -0
  426. package/dist/templates/worker/catalog-info.yaml +19 -0
  427. package/dist/templates/worker/gitignore +5 -0
  428. package/dist/templates/worker/helm/{{name}}/Chart.yaml +9 -0
  429. package/dist/templates/worker/helm/{{name}}/templates/deployment.yaml +68 -0
  430. package/dist/templates/worker/helm/{{name}}/templates/ingress.yaml +43 -0
  431. package/dist/templates/worker/helm/{{name}}/values.yaml +64 -0
  432. package/dist/templates/worker/package.json +26 -0
  433. package/dist/templates/worker/src/index.ts +185 -0
  434. package/dist/templates/worker/tsconfig.json +14 -0
  435. package/dist/types/index.d.ts +596 -0
  436. package/dist/types/index.d.ts.map +1 -0
  437. package/dist/types/index.js +41 -0
  438. package/dist/types/index.js.map +1 -0
  439. package/dist/types/platform.d.ts +191 -0
  440. package/dist/types/platform.d.ts.map +1 -0
  441. package/dist/types/platform.js +17 -0
  442. package/dist/types/platform.js.map +1 -0
  443. package/package.json +62 -0
@@ -0,0 +1,70 @@
1
+ import { Router, type Request, type Response } from 'express'
2
+ import { authMiddleware, requireRole } from '../middleware/auth.js'
3
+ import { Policy } from '../models/policy.js'
4
+
5
+ const router = Router()
6
+
7
+ // List policies (role-filtered)
8
+ router.get('/', authMiddleware, requireRole('mga_admin', 'agent', 'insured'), async (req: Request, res: Response) => {
9
+ try {
10
+ const user = req.user!
11
+ const isAdmin = user.roles.includes('mga_admin')
12
+
13
+ const filter: Record<string, unknown> = {}
14
+ if (!isAdmin && user.roles.includes('agent')) {
15
+ filter.agentId = user.id
16
+ } else if (!isAdmin && user.roles.includes('insured')) {
17
+ filter.insuredEmail = user.email
18
+ }
19
+
20
+ const policies = await Policy.find(filter).sort({ createdAt: -1 }).lean()
21
+ res.json({
22
+ success: true,
23
+ data: policies,
24
+ meta: { total: policies.length },
25
+ })
26
+ } catch (err) {
27
+ process.stderr.write(`Failed to list policies: ${err instanceof Error ? err.message : String(err)}\n`)
28
+ res.status(500).json({
29
+ success: false,
30
+ error: { code: 'INTERNAL_ERROR', message: 'Failed to list policies' },
31
+ })
32
+ }
33
+ })
34
+
35
+ // Get single policy
36
+ router.get('/:id', authMiddleware, requireRole('mga_admin', 'agent', 'insured'), async (req: Request, res: Response) => {
37
+ try {
38
+ const policy = await Policy.findById(req.params.id).lean()
39
+ if (!policy) {
40
+ res.status(404).json({
41
+ success: false,
42
+ error: { code: 'NOT_FOUND', message: 'Policy not found' },
43
+ })
44
+ return
45
+ }
46
+
47
+ const user = req.user!
48
+ const isAdmin = user.roles.includes('mga_admin')
49
+ const isOwnAgent = user.roles.includes('agent') && policy.agentId === user.id
50
+ const isOwnInsured = user.roles.includes('insured') && policy.insuredEmail === user.email
51
+
52
+ if (!isAdmin && !isOwnAgent && !isOwnInsured) {
53
+ res.status(403).json({
54
+ success: false,
55
+ error: { code: 'FORBIDDEN', message: 'Insufficient permissions' },
56
+ })
57
+ return
58
+ }
59
+
60
+ res.json({ success: true, data: policy })
61
+ } catch (err) {
62
+ process.stderr.write(`Failed to get policy: ${err instanceof Error ? err.message : String(err)}\n`)
63
+ res.status(500).json({
64
+ success: false,
65
+ error: { code: 'INTERNAL_ERROR', message: 'Failed to get policy' },
66
+ })
67
+ }
68
+ })
69
+
70
+ export { router as policiesRouter }
@@ -0,0 +1,154 @@
1
+ import { Router, type Request, type Response } from 'express'
2
+ import { z } from 'zod'
3
+ import { authMiddleware, requireRole } from '../middleware/auth.js'
4
+ import { Product } from '../models/product.js'
5
+
6
+ const router = Router()
7
+
8
+ const CreateProductSchema = z.object({
9
+ name: z.string().min(1).max(200),
10
+ description: z.string().min(1).max(2000),
11
+ basePremiumPerEmployee: z.number().int().positive(),
12
+ minEmployees: z.number().int().positive().default(1),
13
+ maxEmployees: z.number().int().positive().default(10000),
14
+ coverageLimits: z.object({
15
+ dataBreachLimit: z.number().int().positive(),
16
+ ransomwareLimit: z.number().int().positive(),
17
+ liabilityLimit: z.number().int().positive(),
18
+ }),
19
+ riskFactors: z.object({
20
+ industryMultipliers: z.record(z.string(), z.number().positive()),
21
+ }),
22
+ })
23
+
24
+ const UpdateProductSchema = CreateProductSchema.partial().extend({
25
+ active: z.boolean().optional(),
26
+ })
27
+
28
+ // List all active products
29
+ router.get('/', authMiddleware, async (_req: Request, res: Response) => {
30
+ try {
31
+ const products = await Product.find({ active: true }).sort({ name: 1 }).lean()
32
+ res.json({
33
+ success: true,
34
+ data: products,
35
+ meta: { total: products.length },
36
+ })
37
+ } catch (err) {
38
+ process.stderr.write(`Failed to list products: ${err instanceof Error ? err.message : String(err)}\n`)
39
+ res.status(500).json({
40
+ success: false,
41
+ error: { code: 'INTERNAL_ERROR', message: 'Failed to list products' },
42
+ })
43
+ }
44
+ })
45
+
46
+ // Get single product
47
+ router.get('/:id', authMiddleware, async (req: Request, res: Response) => {
48
+ try {
49
+ const product = await Product.findById(req.params.id).lean()
50
+ if (!product) {
51
+ res.status(404).json({
52
+ success: false,
53
+ error: { code: 'NOT_FOUND', message: 'Product not found' },
54
+ })
55
+ return
56
+ }
57
+ res.json({ success: true, data: product })
58
+ } catch (err) {
59
+ process.stderr.write(`Failed to get product: ${err instanceof Error ? err.message : String(err)}\n`)
60
+ res.status(500).json({
61
+ success: false,
62
+ error: { code: 'INTERNAL_ERROR', message: 'Failed to get product' },
63
+ })
64
+ }
65
+ })
66
+
67
+ // Create product (admin only)
68
+ router.post('/', authMiddleware, requireRole('mga_admin'), async (req: Request, res: Response) => {
69
+ const result = CreateProductSchema.safeParse(req.body)
70
+ if (!result.success) {
71
+ res.status(400).json({
72
+ success: false,
73
+ error: { code: 'VALIDATION_ERROR', message: 'Invalid request body', details: result.error.flatten() },
74
+ })
75
+ return
76
+ }
77
+
78
+ try {
79
+ const product = await Product.create({
80
+ ...result.data,
81
+ createdBy: req.user!.id,
82
+ })
83
+ res.status(201).json({ success: true, data: product.toObject() })
84
+ } catch (err) {
85
+ process.stderr.write(`Failed to create product: ${err instanceof Error ? err.message : String(err)}\n`)
86
+ res.status(500).json({
87
+ success: false,
88
+ error: { code: 'INTERNAL_ERROR', message: 'Failed to create product' },
89
+ })
90
+ }
91
+ })
92
+
93
+ // Update product (admin only)
94
+ router.put('/:id', authMiddleware, requireRole('mga_admin'), async (req: Request, res: Response) => {
95
+ const result = UpdateProductSchema.safeParse(req.body)
96
+ if (!result.success) {
97
+ res.status(400).json({
98
+ success: false,
99
+ error: { code: 'VALIDATION_ERROR', message: 'Invalid request body', details: result.error.flatten() },
100
+ })
101
+ return
102
+ }
103
+
104
+ try {
105
+ const product = await Product.findByIdAndUpdate(
106
+ req.params.id,
107
+ { $set: result.data },
108
+ { new: true, runValidators: true },
109
+ ).lean()
110
+
111
+ if (!product) {
112
+ res.status(404).json({
113
+ success: false,
114
+ error: { code: 'NOT_FOUND', message: 'Product not found' },
115
+ })
116
+ return
117
+ }
118
+ res.json({ success: true, data: product })
119
+ } catch (err) {
120
+ process.stderr.write(`Failed to update product: ${err instanceof Error ? err.message : String(err)}\n`)
121
+ res.status(500).json({
122
+ success: false,
123
+ error: { code: 'INTERNAL_ERROR', message: 'Failed to update product' },
124
+ })
125
+ }
126
+ })
127
+
128
+ // Delete product (admin only, soft delete)
129
+ router.delete('/:id', authMiddleware, requireRole('mga_admin'), async (req: Request, res: Response) => {
130
+ try {
131
+ const product = await Product.findByIdAndUpdate(
132
+ req.params.id,
133
+ { $set: { active: false } },
134
+ { new: true },
135
+ ).lean()
136
+
137
+ if (!product) {
138
+ res.status(404).json({
139
+ success: false,
140
+ error: { code: 'NOT_FOUND', message: 'Product not found' },
141
+ })
142
+ return
143
+ }
144
+ res.status(204).send()
145
+ } catch (err) {
146
+ process.stderr.write(`Failed to delete product: ${err instanceof Error ? err.message : String(err)}\n`)
147
+ res.status(500).json({
148
+ success: false,
149
+ error: { code: 'INTERNAL_ERROR', message: 'Failed to delete product' },
150
+ })
151
+ }
152
+ })
153
+
154
+ export { router as productsRouter }
@@ -0,0 +1,263 @@
1
+ import { Router, type Request, type Response } from 'express'
2
+ import { z } from 'zod'
3
+ import crypto from 'node:crypto'
4
+ import { authMiddleware, requireRole } from '../middleware/auth.js'
5
+ import { Quote } from '../models/quote.js'
6
+ import { Application } from '../models/application.js'
7
+ import { Product, type IProduct } from '../models/product.js'
8
+ import { Policy } from '../models/policy.js'
9
+ import { Agent } from '../models/agent.js'
10
+
11
+ const router = Router()
12
+
13
+ const QUOTE_VALID_DAYS = 30
14
+ const DEFAULT_DEDUCTIBLE_CENTS = 500_000 // $5,000 in cents
15
+
16
+ const GenerateQuoteSchema = z.object({
17
+ applicationId: z.string().min(1),
18
+ deductible: z.number().int().positive().optional(),
19
+ })
20
+
21
+ function calculatePremium(
22
+ product: IProduct,
23
+ employeeCount: number,
24
+ requestedCoverageLimit: number,
25
+ industry: string,
26
+ ): { premiumAmount: number; limitFactor: number; riskFactor: number; breakdown: string } {
27
+ const limitFactor = requestedCoverageLimit / 1_000_000
28
+ const multipliers = product.riskFactors.industryMultipliers as unknown as Map<string, number>
29
+ const riskFactor = (multipliers instanceof Map ? multipliers.get(industry) : undefined) ?? 1.0
30
+ const premiumAmount = Math.round(product.basePremiumPerEmployee * employeeCount * limitFactor * riskFactor)
31
+ const breakdown = [
32
+ `base=${product.basePremiumPerEmployee}`,
33
+ `employees=${employeeCount}`,
34
+ `limitFactor=${limitFactor.toFixed(2)}`,
35
+ `riskFactor=${riskFactor.toFixed(2)}`,
36
+ `total=${premiumAmount}`,
37
+ ].join(' x ')
38
+
39
+ return { premiumAmount, limitFactor, riskFactor, breakdown }
40
+ }
41
+
42
+ function generatePolicyNumber(): string {
43
+ const prefix = 'CYB'
44
+ const timestamp = Date.now().toString(36).toUpperCase()
45
+ const random = crypto.randomBytes(3).toString('hex').toUpperCase()
46
+ return `${prefix}-${timestamp}-${random}`
47
+ }
48
+
49
+ // List quotes (role-filtered)
50
+ router.get('/', authMiddleware, requireRole('mga_admin', 'agent'), async (req: Request, res: Response) => {
51
+ try {
52
+ const user = req.user!
53
+ const isAdmin = user.roles.includes('mga_admin')
54
+ const filter: Record<string, unknown> = isAdmin ? {} : { agentId: user.id }
55
+
56
+ const quotes = await Quote.find(filter).sort({ createdAt: -1 }).lean()
57
+ res.json({
58
+ success: true,
59
+ data: quotes,
60
+ meta: { total: quotes.length },
61
+ })
62
+ } catch (err) {
63
+ process.stderr.write(`Failed to list quotes: ${err instanceof Error ? err.message : String(err)}\n`)
64
+ res.status(500).json({
65
+ success: false,
66
+ error: { code: 'INTERNAL_ERROR', message: 'Failed to list quotes' },
67
+ })
68
+ }
69
+ })
70
+
71
+ // Get single quote
72
+ router.get('/:id', authMiddleware, requireRole('mga_admin', 'agent'), async (req: Request, res: Response) => {
73
+ try {
74
+ const quote = await Quote.findById(req.params.id).lean()
75
+ if (!quote) {
76
+ res.status(404).json({
77
+ success: false,
78
+ error: { code: 'NOT_FOUND', message: 'Quote not found' },
79
+ })
80
+ return
81
+ }
82
+
83
+ const isAdmin = req.user!.roles.includes('mga_admin')
84
+ if (!isAdmin && quote.agentId !== req.user!.id) {
85
+ res.status(403).json({
86
+ success: false,
87
+ error: { code: 'FORBIDDEN', message: 'Insufficient permissions' },
88
+ })
89
+ return
90
+ }
91
+
92
+ res.json({ success: true, data: quote })
93
+ } catch (err) {
94
+ process.stderr.write(`Failed to get quote: ${err instanceof Error ? err.message : String(err)}\n`)
95
+ res.status(500).json({
96
+ success: false,
97
+ error: { code: 'INTERNAL_ERROR', message: 'Failed to get quote' },
98
+ })
99
+ }
100
+ })
101
+
102
+ // Generate quote from application (agent)
103
+ router.post('/', authMiddleware, requireRole('mga_admin', 'agent'), async (req: Request, res: Response) => {
104
+ const result = GenerateQuoteSchema.safeParse(req.body)
105
+ if (!result.success) {
106
+ res.status(400).json({
107
+ success: false,
108
+ error: { code: 'VALIDATION_ERROR', message: 'Invalid request body', details: result.error.flatten() },
109
+ })
110
+ return
111
+ }
112
+
113
+ try {
114
+ const application = await Application.findById(result.data.applicationId)
115
+ if (!application) {
116
+ res.status(404).json({
117
+ success: false,
118
+ error: { code: 'NOT_FOUND', message: 'Application not found' },
119
+ })
120
+ return
121
+ }
122
+
123
+ if (application.status !== 'submitted') {
124
+ res.status(400).json({
125
+ success: false,
126
+ error: { code: 'INVALID_STATUS', message: 'Application must be in submitted status to generate a quote' },
127
+ })
128
+ return
129
+ }
130
+
131
+ const product = await Product.findById(application.productId)
132
+ if (!product) {
133
+ res.status(404).json({
134
+ success: false,
135
+ error: { code: 'NOT_FOUND', message: 'Product not found' },
136
+ })
137
+ return
138
+ }
139
+
140
+ const calc = calculatePremium(
141
+ product,
142
+ application.employeeCount,
143
+ application.requestedCoverageLimit,
144
+ application.industry,
145
+ )
146
+
147
+ const validUntil = new Date()
148
+ validUntil.setDate(validUntil.getDate() + QUOTE_VALID_DAYS)
149
+
150
+ const quote = await Quote.create({
151
+ applicationId: String(application._id),
152
+ productId: String(product._id),
153
+ agentId: req.user!.id,
154
+ premiumAmount: calc.premiumAmount,
155
+ coverageLimits: {
156
+ dataBreachLimit: product.coverageLimits.dataBreachLimit,
157
+ ransomwareLimit: product.coverageLimits.ransomwareLimit,
158
+ liabilityLimit: product.coverageLimits.liabilityLimit,
159
+ },
160
+ deductible: result.data.deductible ?? DEFAULT_DEDUCTIBLE_CENTS,
161
+ validUntil,
162
+ calculationDetails: {
163
+ basePremiumPerEmployee: product.basePremiumPerEmployee,
164
+ employeeCount: application.employeeCount,
165
+ limitFactor: calc.limitFactor,
166
+ riskFactor: calc.riskFactor,
167
+ breakdown: calc.breakdown,
168
+ },
169
+ })
170
+
171
+ await Application.findByIdAndUpdate(application._id, { $set: { status: 'quoted' } })
172
+
173
+ res.status(201).json({ success: true, data: quote.toObject() })
174
+ } catch (err) {
175
+ process.stderr.write(`Failed to generate quote: ${err instanceof Error ? err.message : String(err)}\n`)
176
+ res.status(500).json({
177
+ success: false,
178
+ error: { code: 'INTERNAL_ERROR', message: 'Failed to generate quote' },
179
+ })
180
+ }
181
+ })
182
+
183
+ // Accept quote (binds to policy)
184
+ router.post('/:id/accept', authMiddleware, requireRole('mga_admin', 'agent'), async (req: Request, res: Response) => {
185
+ try {
186
+ const quote = await Quote.findById(req.params.id)
187
+ if (!quote) {
188
+ res.status(404).json({
189
+ success: false,
190
+ error: { code: 'NOT_FOUND', message: 'Quote not found' },
191
+ })
192
+ return
193
+ }
194
+
195
+ if (quote.status !== 'pending') {
196
+ res.status(400).json({
197
+ success: false,
198
+ error: { code: 'INVALID_STATUS', message: 'Quote must be in pending status to accept' },
199
+ })
200
+ return
201
+ }
202
+
203
+ if (quote.validUntil < new Date()) {
204
+ await Quote.findByIdAndUpdate(quote._id, { $set: { status: 'expired' } })
205
+ res.status(410).json({
206
+ success: false,
207
+ error: { code: 'EXPIRED', message: 'This quote has expired' },
208
+ })
209
+ return
210
+ }
211
+
212
+ const application = await Application.findById(quote.applicationId)
213
+ if (!application) {
214
+ res.status(404).json({
215
+ success: false,
216
+ error: { code: 'NOT_FOUND', message: 'Associated application not found' },
217
+ })
218
+ return
219
+ }
220
+
221
+ const effectiveDate = new Date()
222
+ const expirationDate = new Date()
223
+ expirationDate.setMonth(expirationDate.getMonth() + quote.termMonths)
224
+
225
+ const policy = await Policy.create({
226
+ quoteId: String(quote._id),
227
+ applicationId: quote.applicationId,
228
+ productId: quote.productId,
229
+ agentId: quote.agentId,
230
+ insuredName: application.applicantName,
231
+ insuredEmail: application.applicantEmail,
232
+ policyNumber: generatePolicyNumber(),
233
+ premiumAmount: quote.premiumAmount,
234
+ coverageLimits: { ...quote.coverageLimits },
235
+ effectiveDate,
236
+ expirationDate,
237
+ })
238
+
239
+ await Quote.findByIdAndUpdate(quote._id, { $set: { status: 'accepted' } })
240
+ await Application.findByIdAndUpdate(application._id, { $set: { status: 'bound' } })
241
+
242
+ // Update agent stats
243
+ await Agent.findOneAndUpdate(
244
+ { userId: quote.agentId },
245
+ {
246
+ $inc: {
247
+ policyCount: 1,
248
+ totalPremium: quote.premiumAmount,
249
+ },
250
+ },
251
+ )
252
+
253
+ res.status(201).json({ success: true, data: policy.toObject() })
254
+ } catch (err) {
255
+ process.stderr.write(`Failed to accept quote: ${err instanceof Error ? err.message : String(err)}\n`)
256
+ res.status(500).json({
257
+ success: false,
258
+ error: { code: 'INTERNAL_ERROR', message: 'Failed to accept quote' },
259
+ })
260
+ }
261
+ })
262
+
263
+ export { router as quotesRouter }
@@ -0,0 +1,199 @@
1
+ import { Router, type Request, type Response } from 'express'
2
+ import { authMiddleware, requireRole } from '../middleware/auth.js'
3
+ import { Policy } from '../models/policy.js'
4
+ import { Application } from '../models/application.js'
5
+ import { Agent } from '../models/agent.js'
6
+
7
+ const router = Router()
8
+
9
+ // Summary report
10
+ router.get('/summary', authMiddleware, requireRole('mga_admin', 'agent'), async (req: Request, res: Response) => {
11
+ try {
12
+ const user = req.user!
13
+ const isAdmin = user.roles.includes('mga_admin')
14
+ const policyFilter: Record<string, unknown> = isAdmin ? {} : { agentId: user.id }
15
+ const appFilter: Record<string, unknown> = isAdmin ? {} : { agentId: user.id }
16
+
17
+ const [policies, applicationCount] = await Promise.all([
18
+ Policy.find(policyFilter).lean(),
19
+ Application.countDocuments(appFilter),
20
+ ])
21
+
22
+ const totalPremium = policies.reduce((sum, p) => sum + p.premiumAmount, 0)
23
+ const avgPremium = policies.length > 0 ? Math.round(totalPremium / policies.length) : 0
24
+
25
+ res.json({
26
+ success: true,
27
+ data: {
28
+ totalPolicies: policies.length,
29
+ totalPremium,
30
+ totalApplications: applicationCount,
31
+ avgPremium,
32
+ },
33
+ })
34
+ } catch (err) {
35
+ process.stderr.write(`Failed to generate summary: ${err instanceof Error ? err.message : String(err)}\n`)
36
+ res.status(500).json({
37
+ success: false,
38
+ error: { code: 'INTERNAL_ERROR', message: 'Failed to generate summary report' },
39
+ })
40
+ }
41
+ })
42
+
43
+ // Premium trend (admin only)
44
+ router.get('/premium-trend', authMiddleware, requireRole('mga_admin'), async (_req: Request, res: Response) => {
45
+ try {
46
+ const trend = await Policy.aggregate([
47
+ {
48
+ $group: {
49
+ _id: {
50
+ year: { $year: '$effectiveDate' },
51
+ month: { $month: '$effectiveDate' },
52
+ },
53
+ totalPremium: { $sum: '$premiumAmount' },
54
+ policyCount: { $sum: 1 },
55
+ },
56
+ },
57
+ { $sort: { '_id.year': 1, '_id.month': 1 } },
58
+ {
59
+ $project: {
60
+ _id: 0,
61
+ month: {
62
+ $concat: [
63
+ { $toString: '$_id.year' },
64
+ '-',
65
+ {
66
+ $cond: {
67
+ if: { $lt: ['$_id.month', 10] },
68
+ then: { $concat: ['0', { $toString: '$_id.month' }] },
69
+ else: { $toString: '$_id.month' },
70
+ },
71
+ },
72
+ ],
73
+ },
74
+ totalPremium: 1,
75
+ policyCount: 1,
76
+ },
77
+ },
78
+ ])
79
+
80
+ res.json({
81
+ success: true,
82
+ data: trend,
83
+ })
84
+ } catch (err) {
85
+ process.stderr.write(`Failed to generate premium trend: ${err instanceof Error ? err.message : String(err)}\n`)
86
+ res.status(500).json({
87
+ success: false,
88
+ error: { code: 'INTERNAL_ERROR', message: 'Failed to generate premium trend report' },
89
+ })
90
+ }
91
+ })
92
+
93
+ // Pipeline report (admin only)
94
+ router.get('/pipeline', authMiddleware, requireRole('mga_admin'), async (_req: Request, res: Response) => {
95
+ try {
96
+ const pipeline = await Application.aggregate([
97
+ {
98
+ $group: {
99
+ _id: '$status',
100
+ count: { $sum: 1 },
101
+ },
102
+ },
103
+ { $sort: { count: -1 } },
104
+ {
105
+ $project: {
106
+ _id: 0,
107
+ status: '$_id',
108
+ count: 1,
109
+ },
110
+ },
111
+ ])
112
+
113
+ res.json({
114
+ success: true,
115
+ data: pipeline,
116
+ })
117
+ } catch (err) {
118
+ process.stderr.write(`Failed to generate pipeline: ${err instanceof Error ? err.message : String(err)}\n`)
119
+ res.status(500).json({
120
+ success: false,
121
+ error: { code: 'INTERNAL_ERROR', message: 'Failed to generate pipeline report' },
122
+ })
123
+ }
124
+ })
125
+
126
+ // Top agents report (admin only)
127
+ router.get('/agents', authMiddleware, requireRole('mga_admin'), async (_req: Request, res: Response) => {
128
+ try {
129
+ const agents = await Agent.find({ status: 'active' })
130
+ .sort({ policyCount: -1 })
131
+ .select('firstName lastName email policyCount totalPremium applicationCount')
132
+ .lean()
133
+
134
+ res.json({
135
+ success: true,
136
+ data: agents,
137
+ meta: { total: agents.length },
138
+ })
139
+ } catch (err) {
140
+ process.stderr.write(`Failed to generate agents report: ${err instanceof Error ? err.message : String(err)}\n`)
141
+ res.status(500).json({
142
+ success: false,
143
+ error: { code: 'INTERNAL_ERROR', message: 'Failed to generate agents report' },
144
+ })
145
+ }
146
+ })
147
+
148
+ // Geography report (admin only)
149
+ router.get('/geography', authMiddleware, requireRole('mga_admin'), async (_req: Request, res: Response) => {
150
+ try {
151
+ const geography = await Application.aggregate([
152
+ {
153
+ $match: { status: 'bound' },
154
+ },
155
+ {
156
+ $lookup: {
157
+ from: 'policies',
158
+ let: { appId: { $toString: '$_id' } },
159
+ pipeline: [
160
+ { $match: { $expr: { $eq: ['$applicationId', '$$appId'] } } },
161
+ ],
162
+ as: 'policy',
163
+ },
164
+ },
165
+ { $unwind: { path: '$policy', preserveNullAndEmptyArrays: true } },
166
+ {
167
+ $group: {
168
+ _id: '$state',
169
+ totalPremium: { $sum: { $ifNull: ['$policy.premiumAmount', 0] } },
170
+ policyCount: {
171
+ $sum: { $cond: [{ $ifNull: ['$policy', false] }, 1, 0] },
172
+ },
173
+ },
174
+ },
175
+ { $sort: { totalPremium: -1 } },
176
+ {
177
+ $project: {
178
+ _id: 0,
179
+ state: '$_id',
180
+ totalPremium: 1,
181
+ policyCount: 1,
182
+ },
183
+ },
184
+ ])
185
+
186
+ res.json({
187
+ success: true,
188
+ data: geography,
189
+ })
190
+ } catch (err) {
191
+ process.stderr.write(`Failed to generate geography report: ${err instanceof Error ? err.message : String(err)}\n`)
192
+ res.status(500).json({
193
+ success: false,
194
+ error: { code: 'INTERNAL_ERROR', message: 'Failed to generate geography report' },
195
+ })
196
+ }
197
+ })
198
+
199
+ export { router as reportsRouter }
@@ -0,0 +1,18 @@
1
+ {
2
+ "compilerOptions": {
3
+ "target": "ES2022",
4
+ "module": "NodeNext",
5
+ "moduleResolution": "NodeNext",
6
+ "outDir": "./dist",
7
+ "rootDir": "./src",
8
+ "strict": true,
9
+ "esModuleInterop": true,
10
+ "skipLibCheck": true,
11
+ "forceConsistentCasingInFileNames": true,
12
+ "declaration": true,
13
+ "declarationMap": true,
14
+ "sourceMap": true
15
+ },
16
+ "include": ["src/**/*"],
17
+ "exclude": ["node_modules", "dist"]
18
+ }