@aifabrix/builder 2.43.0 → 2.44.1

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 (371) hide show
  1. package/.cursor/rules/anchor-docs.mdc +15 -0
  2. package/.cursor/rules/cli-layout.mdc +75 -0
  3. package/.cursor/rules/project-rules.mdc +8 -0
  4. package/.npmrc.token +1 -0
  5. package/.nyc_output/55e9d034-ddab-4579-a706-e02a91d75c91.json +1 -0
  6. package/.nyc_output/processinfo/55e9d034-ddab-4579-a706-e02a91d75c91.json +1 -0
  7. package/.nyc_output/processinfo/index.json +1 -0
  8. package/README.md +1 -1
  9. package/anchor-docs/README.md +10 -0
  10. package/anchor-docs/_TEMPLATE +24 -0
  11. package/bin/aifabrix.js +13 -4
  12. package/integration/hubspot-test/README.md +31 -0
  13. package/integration/hubspot-test/create-hubspot.js +5 -5
  14. package/integration/hubspot-test/hubspot-test-datasource-company.json +58 -462
  15. package/integration/hubspot-test/hubspot-test-datasource-contact.json +61 -555
  16. package/integration/hubspot-test/hubspot-test-datasource-deal.json +63 -506
  17. package/integration/hubspot-test/hubspot-test-datasource-users.json +42 -83
  18. package/integration/hubspot-test/hubspot-test-deploy.json +3 -3
  19. package/integration/hubspot-test/test-dataplane-down-tests.js +1 -7
  20. package/integration/hubspot-test/test-dataplane-down.js +3 -3
  21. package/integration/hubspot-test/test.js +35 -43
  22. package/integration/hubspot-test/wizard-hubspot-test-headless.yaml +23 -0
  23. package/integration/roundtrip-test-local/README.md +144 -0
  24. package/integration/roundtrip-test-local/application.yaml +13 -0
  25. package/integration/roundtrip-test-local/env.template +15 -0
  26. package/integration/roundtrip-test-local/roundtrip-test-local-datasource-roundtrip-test-company.yaml +14 -0
  27. package/integration/roundtrip-test-local/roundtrip-test-local-deploy.json +61 -0
  28. package/integration/roundtrip-test-local/roundtrip-test-local-system.yaml +25 -0
  29. package/integration/roundtrip-test-local2/README.md +144 -0
  30. package/integration/roundtrip-test-local2/application.yaml +13 -0
  31. package/integration/roundtrip-test-local2/env.template +15 -0
  32. package/integration/roundtrip-test-local2/roundtrip-test-local2-datasource-company.yaml +31 -0
  33. package/integration/roundtrip-test-local2/roundtrip-test-local2-deploy.json +86 -0
  34. package/integration/roundtrip-test-local2/roundtrip-test-local2-system.yaml +25 -0
  35. package/integration/test/wizard.yaml +8 -0
  36. package/jest.config.default.js +10 -0
  37. package/jest.config.integration.fixtures.js +22 -0
  38. package/jest.config.integration.js +21 -18
  39. package/jest.config.isolated.js +10 -0
  40. package/jest.projects.js +301 -0
  41. package/lib/api/certificates.api.js +62 -0
  42. package/lib/api/datasources-core.api.js +3 -3
  43. package/lib/api/dev-mtls-request.js +110 -0
  44. package/lib/api/dev-server-https.js +145 -0
  45. package/lib/api/dev.api.js +133 -144
  46. package/lib/api/index.js +11 -3
  47. package/lib/api/pipeline.api.js +67 -20
  48. package/lib/api/types/certificates.types.js +48 -0
  49. package/lib/api/types/dev.types.js +4 -3
  50. package/lib/api/types/pipeline.types.js +8 -5
  51. package/lib/api/types/validation-run.types.js +56 -0
  52. package/lib/api/validation-run.api.js +111 -0
  53. package/lib/api/validation-runner.js +109 -0
  54. package/lib/app/certification-show-enrich.js +129 -0
  55. package/lib/app/certification-verify-rows.js +60 -0
  56. package/lib/app/config.js +1 -1
  57. package/lib/app/deploy-status-display.js +2 -2
  58. package/lib/app/deploy.js +7 -6
  59. package/lib/app/display.js +2 -1
  60. package/lib/app/dockerfile.js +3 -2
  61. package/lib/app/down.js +2 -1
  62. package/lib/app/helpers.js +6 -5
  63. package/lib/app/index.js +27 -8
  64. package/lib/app/list.js +7 -6
  65. package/lib/app/push.js +4 -3
  66. package/lib/app/register.js +16 -7
  67. package/lib/app/rotate-secret.js +14 -13
  68. package/lib/app/run-container-start.js +184 -0
  69. package/lib/app/run-docker-fallback.js +108 -0
  70. package/lib/app/run-env-compose.js +30 -42
  71. package/lib/app/run-helpers.js +49 -126
  72. package/lib/app/run-infra-requirements.js +30 -0
  73. package/lib/app/run-resolve-image.js +21 -0
  74. package/lib/app/run.js +74 -21
  75. package/lib/app/show-display.js +44 -1
  76. package/lib/app/show.js +93 -9
  77. package/lib/build/index.js +13 -10
  78. package/lib/certification/cli-cert-sync-skip.js +21 -0
  79. package/lib/certification/merge-certification-from-artifact.js +185 -0
  80. package/lib/certification/post-unified-cert-sync.js +33 -0
  81. package/lib/certification/sync-after-external-command.js +52 -0
  82. package/lib/certification/sync-system-certification.js +197 -0
  83. package/lib/cli/index.js +2 -0
  84. package/lib/cli/setup-app.help.js +67 -0
  85. package/lib/cli/setup-app.js +61 -121
  86. package/lib/cli/setup-app.test-commands.js +195 -0
  87. package/lib/cli/setup-auth.js +19 -5
  88. package/lib/cli/setup-credential-deployment.js +22 -8
  89. package/lib/cli/setup-dev-path-commands.js +124 -0
  90. package/lib/cli/setup-dev.js +170 -113
  91. package/lib/cli/setup-environment.js +7 -1
  92. package/lib/cli/setup-external-system.js +84 -23
  93. package/lib/cli/setup-infra.js +126 -47
  94. package/lib/cli/setup-parameters.js +32 -0
  95. package/lib/cli/setup-secrets.js +137 -18
  96. package/lib/cli/setup-service-user.js +1 -1
  97. package/lib/cli/setup-utility.js +54 -22
  98. package/lib/commands/app-down.js +5 -7
  99. package/lib/commands/app-install.js +14 -7
  100. package/lib/commands/app-logs.js +13 -10
  101. package/lib/commands/app-shell.js +4 -1
  102. package/lib/commands/app-test.js +25 -19
  103. package/lib/commands/app.js +32 -11
  104. package/lib/commands/auth-config.js +6 -6
  105. package/lib/commands/auth-status.js +4 -3
  106. package/lib/commands/credential-env.js +4 -3
  107. package/lib/commands/credential-list.js +5 -4
  108. package/lib/commands/credential-push.js +4 -3
  109. package/lib/commands/datasource-unified-test-cli.js +428 -0
  110. package/lib/commands/datasource-unified-test-cli.options.js +191 -0
  111. package/lib/commands/datasource-unified-test-e2e-cli-helpers.js +106 -0
  112. package/lib/commands/datasource-validation-cli.js +143 -0
  113. package/lib/commands/datasource.js +125 -95
  114. package/lib/commands/deployment-list.js +6 -5
  115. package/lib/commands/dev-cli-handlers.js +122 -18
  116. package/lib/commands/dev-down.js +4 -3
  117. package/lib/commands/dev-init.js +231 -116
  118. package/lib/commands/dev-show-display.js +473 -0
  119. package/lib/commands/login-credentials.js +3 -2
  120. package/lib/commands/login-device.js +4 -3
  121. package/lib/commands/login.js +5 -4
  122. package/lib/commands/logout.js +8 -7
  123. package/lib/commands/parameters-validate.js +54 -0
  124. package/lib/commands/repair-datasource.js +314 -68
  125. package/lib/commands/repair-env-template.js +2 -2
  126. package/lib/commands/repair.js +21 -3
  127. package/lib/commands/secrets-list.js +23 -12
  128. package/lib/commands/secrets-remove-all.js +220 -0
  129. package/lib/commands/secrets-remove.js +21 -12
  130. package/lib/commands/secrets-set.js +21 -12
  131. package/lib/commands/secrets-validate.js +4 -4
  132. package/lib/commands/secure.js +10 -9
  133. package/lib/commands/service-user.js +26 -25
  134. package/lib/commands/test-e2e-external.js +27 -1
  135. package/lib/commands/up-common.js +3 -2
  136. package/lib/commands/up-dataplane.js +29 -16
  137. package/lib/commands/up-miso.js +19 -29
  138. package/lib/commands/upload.js +149 -39
  139. package/lib/commands/wizard-core-helpers.js +1 -1
  140. package/lib/commands/wizard-dataplane.js +4 -3
  141. package/lib/commands/wizard-helpers.js +3 -3
  142. package/lib/commands/wizard.js +2 -2
  143. package/lib/core/admin-secrets.js +14 -5
  144. package/lib/core/audit-logger.js +12 -4
  145. package/lib/core/config-attach-extensions.js +46 -0
  146. package/lib/core/config-runtime-paths.js +29 -0
  147. package/lib/core/config.js +55 -56
  148. package/lib/core/diff.js +3 -2
  149. package/lib/core/ensure-encryption-key.js +1 -1
  150. package/lib/core/secrets-ensure-infra.js +77 -0
  151. package/lib/core/secrets-ensure.js +120 -64
  152. package/lib/core/secrets-env-write.js +35 -7
  153. package/lib/core/secrets-infra-placeholder-sync.js +61 -0
  154. package/lib/core/secrets.js +200 -37
  155. package/lib/core/templates-env.js +4 -3
  156. package/lib/datasource/abac-validator.js +1 -10
  157. package/lib/datasource/deploy.js +75 -53
  158. package/lib/datasource/field-reference-validator.js +9 -6
  159. package/lib/datasource/integration-context.js +63 -0
  160. package/lib/datasource/list.js +8 -7
  161. package/lib/datasource/log-viewer.js +189 -67
  162. package/lib/datasource/resolve-app.js +4 -4
  163. package/lib/datasource/test-e2e.js +113 -146
  164. package/lib/datasource/test-integration.js +114 -122
  165. package/lib/datasource/unified-validation-run-body.js +68 -0
  166. package/lib/datasource/unified-validation-run-post.js +23 -0
  167. package/lib/datasource/unified-validation-run-resolve.js +43 -0
  168. package/lib/datasource/unified-validation-run.js +93 -0
  169. package/lib/datasource/validate.js +157 -13
  170. package/lib/deployment/deployer.js +4 -3
  171. package/lib/deployment/environment.js +7 -6
  172. package/lib/deployment/push.js +17 -8
  173. package/lib/external-system/delete.js +4 -3
  174. package/lib/external-system/deploy.js +166 -53
  175. package/lib/external-system/download-helpers.js +1 -1
  176. package/lib/external-system/download.js +7 -6
  177. package/lib/external-system/generator.js +92 -6
  178. package/lib/external-system/integration-test-dispatch.js +26 -0
  179. package/lib/external-system/test-execution.js +5 -1
  180. package/lib/external-system/test-helpers.js +0 -4
  181. package/lib/external-system/test-system-level-helpers.js +110 -0
  182. package/lib/external-system/test-system-level.js +83 -44
  183. package/lib/external-system/test.js +59 -8
  184. package/lib/generator/builders.js +23 -11
  185. package/lib/generator/deploy-manifest-azure-kv.js +81 -0
  186. package/lib/generator/external.js +16 -4
  187. package/lib/generator/helpers.js +58 -3
  188. package/lib/generator/index.js +4 -0
  189. package/lib/generator/split-readme.js +12 -7
  190. package/lib/generator/split-variables.js +2 -1
  191. package/lib/generator/split.js +1 -1
  192. package/lib/generator/wizard-readme.js +3 -3
  193. package/lib/generator/wizard.js +8 -8
  194. package/lib/infrastructure/compose.js +70 -7
  195. package/lib/infrastructure/helpers-docker-check.js +67 -0
  196. package/lib/infrastructure/helpers.js +203 -42
  197. package/lib/infrastructure/index.js +31 -18
  198. package/lib/infrastructure/services.js +21 -67
  199. package/lib/internal/fs-real-sync.js +104 -0
  200. package/lib/internal/node-fs.js +98 -0
  201. package/lib/parameters/database-secret-values.js +173 -0
  202. package/lib/parameters/infra-kv-discovery.js +121 -0
  203. package/lib/parameters/infra-parameter-catalog.js +458 -0
  204. package/lib/parameters/infra-parameter-validate.js +64 -0
  205. package/lib/schema/application-schema.json +37 -17
  206. package/lib/schema/datasource-test-run.schema.json +493 -0
  207. package/lib/schema/deployment-rules.yaml +102 -63
  208. package/lib/schema/external-datasource.schema.json +1200 -442
  209. package/lib/schema/external-system.schema.json +203 -5
  210. package/lib/schema/flag-map-validation-run.json +31 -0
  211. package/lib/schema/infra-parameter.schema.json +106 -0
  212. package/lib/schema/infra.parameter.yaml +421 -0
  213. package/lib/schema/type/credential-auth-templates.json +40 -0
  214. package/lib/schema/type/document-storage.json +226 -0
  215. package/lib/schema/type/message-service.json +123 -0
  216. package/lib/schema/type/vector-store.json +88 -0
  217. package/lib/utils/aifabrix-runtime-config-dir.js +132 -0
  218. package/lib/utils/api-error-handler.js +2 -2
  219. package/lib/utils/api.js +77 -17
  220. package/lib/utils/app-register-api.js +3 -2
  221. package/lib/utils/app-register-auth.js +1 -1
  222. package/lib/utils/app-register-config.js +4 -4
  223. package/lib/utils/app-register-display.js +3 -2
  224. package/lib/utils/app-register-validator.js +3 -2
  225. package/lib/utils/app-run-containers.js +26 -22
  226. package/lib/utils/app-scoped-config.js +31 -0
  227. package/lib/utils/app-service-env-from-builder.js +164 -0
  228. package/lib/utils/build-copy.js +1 -1
  229. package/lib/utils/build-helpers.js +20 -20
  230. package/lib/utils/build-resolve-image.js +165 -0
  231. package/lib/utils/cli-layout-chalk.js +8 -0
  232. package/lib/utils/cli-test-layout-chalk.js +267 -0
  233. package/lib/utils/cli-utils.js +88 -11
  234. package/lib/utils/compose-db-passwords.js +138 -0
  235. package/lib/utils/compose-generate-docker-compose.js +216 -0
  236. package/lib/utils/compose-generator.js +197 -291
  237. package/lib/utils/compose-miso-env.js +18 -0
  238. package/lib/utils/compose-traefik-ingress-base.js +158 -0
  239. package/lib/utils/config-paths.js +166 -7
  240. package/lib/utils/config-scoped-resources-preference.js +41 -0
  241. package/lib/utils/configuration-env-resolver.js +11 -8
  242. package/lib/utils/controller-deployment-outcome.js +68 -0
  243. package/lib/utils/credential-display.js +2 -2
  244. package/lib/utils/credential-secrets-env.js +5 -5
  245. package/lib/utils/dataplane-pipeline-warning.js +4 -3
  246. package/lib/utils/datasource-test-run-capability-scope.js +43 -0
  247. package/lib/utils/datasource-test-run-certificate-tty.js +82 -0
  248. package/lib/utils/datasource-test-run-debug-display.js +137 -0
  249. package/lib/utils/datasource-test-run-debug-slice.js +93 -0
  250. package/lib/utils/datasource-test-run-display.js +459 -0
  251. package/lib/utils/datasource-test-run-exit.js +83 -0
  252. package/lib/utils/datasource-test-run-legacy-adapter.js +93 -0
  253. package/lib/utils/datasource-test-run-report-version.js +51 -0
  254. package/lib/utils/datasource-test-run-schema-sync.js +59 -0
  255. package/lib/utils/datasource-test-run-tty-log.js +81 -0
  256. package/lib/utils/datasource-validation-watch.js +266 -0
  257. package/lib/utils/declarative-url-ports.js +47 -0
  258. package/lib/utils/derive-env-key-from-client-id.js +41 -0
  259. package/lib/utils/dev-ca-install.js +185 -23
  260. package/lib/utils/dev-cert-helper.js +266 -17
  261. package/lib/utils/dev-hosts-helper.js +307 -0
  262. package/lib/utils/dev-init-cert-hints.js +37 -0
  263. package/lib/utils/dev-init-health-messages.js +52 -0
  264. package/lib/utils/dev-init-resolve.js +86 -0
  265. package/lib/utils/dev-init-ssh-merge.js +65 -0
  266. package/lib/utils/dev-ssh-config-helper.js +196 -0
  267. package/lib/utils/dev-user-groups.js +93 -0
  268. package/lib/utils/docker-build.js +42 -17
  269. package/lib/utils/docker-exec.js +28 -0
  270. package/lib/utils/docker-manifest-public-port.js +116 -0
  271. package/lib/utils/docker-not-running-hint.js +52 -0
  272. package/lib/utils/docker.js +98 -11
  273. package/lib/utils/ensure-dev-certs-for-remote-docker.js +192 -0
  274. package/lib/utils/env-config-loader.js +10 -91
  275. package/lib/utils/env-copy.js +19 -10
  276. package/lib/utils/env-map.js +35 -8
  277. package/lib/utils/env-template.js +2 -2
  278. package/lib/utils/environment-scoped-resources.js +144 -0
  279. package/lib/utils/error-formatter.js +92 -13
  280. package/lib/utils/error-formatters/http-status-errors.js +6 -5
  281. package/lib/utils/error-formatters/network-errors.js +2 -1
  282. package/lib/utils/error-formatters/permission-errors.js +2 -1
  283. package/lib/utils/error-formatters/validation-errors.js +2 -1
  284. package/lib/utils/external-readme.js +8 -1
  285. package/lib/utils/external-system-display.js +242 -136
  286. package/lib/utils/external-system-local-test-tty.js +389 -0
  287. package/lib/utils/external-system-readiness-core.js +377 -0
  288. package/lib/utils/external-system-readiness-deploy-display.js +270 -0
  289. package/lib/utils/external-system-readiness-display-internals.js +150 -0
  290. package/lib/utils/external-system-readiness-display.js +186 -0
  291. package/lib/utils/external-system-system-test-tty-overview.js +120 -0
  292. package/lib/utils/external-system-system-test-tty.js +417 -0
  293. package/lib/utils/external-system-test-helpers.js +24 -6
  294. package/lib/utils/external-system-validators.js +30 -12
  295. package/lib/utils/health-check-url.js +119 -0
  296. package/lib/utils/health-check.js +59 -25
  297. package/lib/utils/help-builder.js +11 -8
  298. package/lib/utils/image-version.js +4 -8
  299. package/lib/utils/infra-containers.js +4 -7
  300. package/lib/utils/infra-env-defaults.js +162 -0
  301. package/lib/utils/infra-status-display.js +167 -0
  302. package/lib/utils/infra-status.js +16 -8
  303. package/lib/utils/local-secrets.js +3 -4
  304. package/lib/utils/paths.js +148 -47
  305. package/lib/utils/port-resolver.js +10 -23
  306. package/lib/utils/redis-env-scope.js +62 -0
  307. package/lib/utils/register-aifabrix-shell-env.js +204 -0
  308. package/lib/utils/remote-builder-validation.js +99 -0
  309. package/lib/utils/remote-dev-auth.js +117 -21
  310. package/lib/utils/remote-docker-env.js +67 -15
  311. package/lib/utils/remote-secrets-loader.js +13 -4
  312. package/lib/utils/resolve-docker-image-ref.js +124 -0
  313. package/lib/utils/schema-loader.js +22 -9
  314. package/lib/utils/secrets-bash-kv.js +25 -0
  315. package/lib/utils/secrets-generator.js +169 -49
  316. package/lib/utils/secrets-helpers.js +70 -59
  317. package/lib/utils/secrets-kv-scope.js +60 -0
  318. package/lib/utils/secrets-utils.js +32 -38
  319. package/lib/utils/secrets-validation.js +3 -1
  320. package/lib/utils/secrets-yaml-preserve.js +109 -0
  321. package/lib/utils/ssh-key-helper.js +4 -2
  322. package/lib/utils/template-helpers.js +2 -2
  323. package/lib/utils/test-log-writer.js +3 -3
  324. package/lib/utils/token-manager.js +1 -2
  325. package/lib/utils/url-declarative-public-base.js +188 -0
  326. package/lib/utils/url-declarative-resolve-build.js +493 -0
  327. package/lib/utils/url-declarative-resolve-load-doc.js +51 -0
  328. package/lib/utils/url-declarative-resolve.js +220 -0
  329. package/lib/utils/url-declarative-token-parse.js +74 -0
  330. package/lib/utils/url-declarative-url-flags.js +50 -0
  331. package/lib/utils/url-declarative-vdir-inactive-env.js +99 -0
  332. package/lib/utils/url-public-path-prefix.js +34 -0
  333. package/lib/utils/urls-local-registry.js +220 -0
  334. package/lib/utils/validation-report-tty-kit.js +77 -0
  335. package/lib/utils/validation-run-poll.js +112 -0
  336. package/lib/utils/validation-run-post-retry.js +85 -0
  337. package/lib/utils/validation-run-request.js +116 -0
  338. package/lib/utils/variable-transformer.js +21 -4
  339. package/lib/utils/yaml-preserve.js +33 -14
  340. package/lib/validation/datasource-warnings.js +56 -0
  341. package/lib/validation/env-template-auth.js +1 -1
  342. package/lib/validation/external-manifest-validator.js +27 -7
  343. package/lib/validation/validate-display.js +37 -31
  344. package/lib/validation/validate-external-cert-sync.js +23 -0
  345. package/lib/validation/validate.js +8 -14
  346. package/lib/validation/validator-unresolved-placeholders.js +98 -0
  347. package/lib/validation/validator.js +22 -65
  348. package/lib/validation/wizard-config-validator.js +2 -1
  349. package/package.json +9 -4
  350. package/scripts/check-datasource-test-run-schema-sync.js +34 -0
  351. package/scripts/diagnose-cli.js +150 -0
  352. package/scripts/install-local.js +307 -55
  353. package/scripts/pnpm-global-remove.js +48 -0
  354. package/templates/README.md +15 -2
  355. package/templates/applications/dataplane/application.yaml +52 -2
  356. package/templates/applications/dataplane/env.template +79 -17
  357. package/templates/applications/dataplane/rbac.yaml +8 -0
  358. package/templates/applications/keycloak/application.yaml +9 -1
  359. package/templates/applications/keycloak/env.template +15 -6
  360. package/templates/applications/miso-controller/application.yaml +10 -2
  361. package/templates/applications/miso-controller/env.template +42 -12
  362. package/templates/applications/miso-controller/rbac.yaml +5 -0
  363. package/templates/external-system/README.md.hbs +20 -7
  364. package/templates/external-system/deploy.js.hbs +5 -5
  365. package/templates/external-system/external-datasource.yaml.hbs +197 -118
  366. package/templates/infra/compose.yaml.hbs +33 -16
  367. package/templates/infra/servers.json.hbs +3 -1
  368. package/templates/python/docker-compose.hbs +16 -0
  369. package/templates/typescript/docker-compose.hbs +16 -0
  370. package/lib/api/external-test.api.js +0 -111
  371. package/lib/schema/env-config.yaml +0 -60
@@ -0,0 +1,301 @@
1
+ /**
2
+ * Shared Jest project definitions (default + isolated). Split configs run in separate processes
3
+ * so `jest.mock('fs')` in the default suite cannot leak into isolated workers.
4
+ * @fileoverview
5
+ */
6
+
7
+ // Detect CI environment (GitHub Actions, CI simulation, etc.)
8
+ const isCI = process.env.CI === 'true' || process.env.CI_SIMULATION === 'true';
9
+
10
+ const sharedTransform = {
11
+ '^.+\\.js$': ['babel-jest', {
12
+ configFile: false,
13
+ babelrc: false,
14
+ presets: [['@babel/preset-env', { targets: { node: 'current' } }]],
15
+ plugins: ['@babel/plugin-syntax-optional-chaining']
16
+ }]
17
+ };
18
+
19
+ /**
20
+ * Single-file Jest project (fresh module graph when run in a separate process).
21
+ * @param {string} displayName - Jest project display name
22
+ * @param {string[]} testMatch - Glob patterns for test files
23
+ * @returns {object} Jest project configuration object
24
+ */
25
+ function makeIsolatedProject(displayName, testMatch) {
26
+ return {
27
+ displayName,
28
+ testEnvironment: 'node',
29
+ transform: sharedTransform,
30
+ testMatch,
31
+ testPathIgnorePatterns: ['/node_modules/', '\\\\node_modules\\\\'],
32
+ setupFiles: ['<rootDir>/tests/capture-real-fs.js'],
33
+ setupFilesAfterEnv: ['<rootDir>/tests/setup.js'],
34
+ testTimeout: isCI ? 10000 : 5000,
35
+ maxWorkers: 1
36
+ };
37
+ }
38
+
39
+ const defaultProject = {
40
+ displayName: 'default',
41
+ testEnvironment: 'node',
42
+ transform: sharedTransform,
43
+ testMatch: [
44
+ '**/tests/**/*.test.js',
45
+ '**/tests/**/*.spec.js'
46
+ ],
47
+ testPathIgnorePatterns: (() => {
48
+ const patterns = [
49
+ '/node_modules/',
50
+ '\\\\node_modules\\\\',
51
+ '/tests/integration/',
52
+ '\\\\tests\\\\integration\\\\',
53
+ '/tests/manual/',
54
+ '\\\\tests\\\\manual\\\\',
55
+ 'lib/utils/cli-utils.test.js',
56
+ 'lib/utils/external-system-display.test.js',
57
+ '/tests/lib/utils/cli-utils.test.js',
58
+ '/tests/lib/utils/external-system-display.test.js',
59
+ '/tests/lib/utils/dev-hosts-helper.test.js',
60
+ '/tests/lib/utils/declarative-url-matrix-d-reload.test.js',
61
+ '/tests/lib/utils/datasource-validation-watch.test.js',
62
+ '\\\\tests\\\\lib\\\\utils\\\\cli-utils.test.js',
63
+ '\\\\tests\\\\lib\\\\utils\\\\external-system-display.test.js',
64
+ '\\\\tests\\\\lib\\\\utils\\\\dev-hosts-helper.test.js',
65
+ '\\\\tests\\\\lib\\\\utils\\\\declarative-url-matrix-d-reload.test.js',
66
+ '\\\\tests\\\\lib\\\\utils\\\\datasource-validation-watch.test.js',
67
+ 'lib/utils/dev-hosts-helper.test.js',
68
+ 'lib/utils/declarative-url-matrix-d-reload.test.js',
69
+ 'lib/utils/datasource-validation-watch.test.js',
70
+ 'dev-hosts-helper\\.test\\.js',
71
+ 'declarative-url-matrix-d-reload\\.test\\.js',
72
+ '/tests/lib/datasource/log-viewer.test.js',
73
+ '\\\\tests\\\\lib\\\\datasource\\\\log-viewer.test.js',
74
+ 'lib/datasource/log-viewer.test.js',
75
+ '/tests/lib/commands/parameters-validate.test.js',
76
+ '\\\\tests\\\\lib\\\\commands\\\\parameters-validate.test.js',
77
+ 'lib/commands/parameters-validate.test.js',
78
+ 'parameters-validate\\.test\\.js',
79
+ '/tests/lib/utils/datasource-test-run-schema-sync.test.js',
80
+ '\\\\tests\\\\lib\\\\utils\\\\datasource-test-run-schema-sync.test.js',
81
+ 'lib/utils/datasource-test-run-schema-sync.test.js',
82
+ '/tests/lib/parameters/infra-platform-contract.test.js',
83
+ '\\\\tests\\\\lib\\\\parameters\\\\infra-platform-contract.test.js',
84
+ 'lib/parameters/infra-platform-contract.test.js',
85
+ '/tests/lib/parameters/database-secret-values.test.js',
86
+ '\\\\tests\\\\lib\\\\parameters\\\\database-secret-values.test.js',
87
+ 'lib/parameters/database-secret-values.test.js',
88
+ 'database-secret-values\\.test\\.js',
89
+ '/tests/lib/parameters/infra-parameter-validate.test.js',
90
+ '\\\\tests\\\\lib\\\\parameters\\\\infra-parameter-validate.test.js',
91
+ 'lib/parameters/infra-parameter-validate.test.js',
92
+ 'infra-parameter-validate\\.test\\.js',
93
+ '/tests/lib/parameters/infra-parameter-catalog.test.js',
94
+ '\\\\tests\\\\lib\\\\parameters\\\\infra-parameter-catalog.test.js',
95
+ 'lib/parameters/infra-parameter-catalog.test.js',
96
+ 'infra-parameter-catalog\\.test\\.js',
97
+ '/tests/lib/utils/urls-local-registry.test.js',
98
+ '\\\\tests\\\\lib\\\\utils\\\\urls-local-registry.test.js',
99
+ 'lib/utils/urls-local-registry.test.js',
100
+ 'urls-local-registry\\.test\\.js',
101
+ '/tests/lib/utils/aifabrix-runtime-config-dir.test.js',
102
+ '\\\\tests\\\\lib\\\\utils\\\\aifabrix-runtime-config-dir.test.js',
103
+ 'lib/utils/aifabrix-runtime-config-dir.test.js',
104
+ 'aifabrix-runtime-config-dir\\.test\\.js',
105
+ '/tests/lib/parameters/platform-env-template-kv-catalog.test.js',
106
+ '\\\\tests\\\\lib\\\\parameters\\\\platform-env-template-kv-catalog.test.js',
107
+ 'lib/parameters/platform-env-template-kv-catalog.test.js',
108
+ 'platform-env-template-kv-catalog\\.test\\.js',
109
+ '/tests/lib/infrastructure/resolve-infra-state-paths.test.js',
110
+ '\\\\tests\\\\lib\\\\infrastructure\\\\resolve-infra-state-paths.test.js',
111
+ 'lib/infrastructure/resolve-infra-state-paths.test.js',
112
+ 'resolve-infra-state-paths\\.test\\.js',
113
+ '/tests/lib/utils/dev-ssh-config-helper.test.js',
114
+ '\\\\tests\\\\lib\\\\utils\\\\dev-ssh-config-helper.test.js',
115
+ 'lib/utils/dev-ssh-config-helper.test.js',
116
+ 'dev-ssh-config-helper\\.test\\.js',
117
+ '/tests/lib/utils/ssh-key-helper.test.js',
118
+ '\\\\tests\\\\lib\\\\utils\\\\ssh-key-helper.test.js',
119
+ 'lib/utils/ssh-key-helper.test.js',
120
+ 'ssh-key-helper\\.test\\.js',
121
+ '/tests/lib/core/secrets-ensure-catalog-fallback.test.js',
122
+ '\\\\tests\\\\lib\\\\core\\\\secrets-ensure-catalog-fallback.test.js',
123
+ 'lib/core/secrets-ensure-catalog-fallback.test.js',
124
+ 'secrets-ensure-catalog-fallback\\.test\\.js',
125
+ '/tests/lib/core/secrets-ensure.test.js',
126
+ '\\\\tests\\\\lib\\\\core\\\\secrets-ensure.test.js',
127
+ 'lib/core/secrets-ensure.test.js',
128
+ 'secrets-ensure\\.test\\.js',
129
+ '/tests/lib/utils/url-declarative-vdir-inactive-env.test.js',
130
+ '\\\\tests\\\\lib\\\\utils\\\\url-declarative-vdir-inactive-env.test.js',
131
+ 'lib/utils/url-declarative-vdir-inactive-env.test.js',
132
+ 'url-declarative-vdir-inactive-env\\.test\\.js',
133
+ '/tests/lib/utils/app-service-env-from-builder.test.js',
134
+ '\\\\tests\\\\lib\\\\utils\\\\app-service-env-from-builder.test.js',
135
+ 'lib/utils/app-service-env-from-builder.test.js',
136
+ 'app-service-env-from-builder\\.test\\.js',
137
+ '/tests/lib/parameters/infra-kv-discovery.test.js',
138
+ '\\\\tests\\\\lib\\\\parameters\\\\infra-kv-discovery.test.js',
139
+ 'lib/parameters/infra-kv-discovery.test.js',
140
+ 'infra-kv-discovery\\.test\\.js',
141
+ '/tests/lib/utils/infra-env-defaults.test.js',
142
+ '\\\\tests\\\\lib\\\\utils\\\\infra-env-defaults.test.js',
143
+ 'lib/utils/infra-env-defaults.test.js',
144
+ 'infra-env-defaults\\.test\\.js',
145
+ '/tests/lib/infrastructure/compose-traefik-template.test.js',
146
+ '\\\\tests\\\\lib\\\\infrastructure\\\\compose-traefik-template.test.js',
147
+ 'lib/infrastructure/compose-traefik-template.test.js',
148
+ 'compose-traefik-template\\.test\\.js',
149
+ '/tests/lib/utils/paths-app-listing.test.js',
150
+ '\\\\tests\\\\lib\\\\utils\\\\paths-app-listing.test.js',
151
+ 'lib/utils/paths-app-listing.test.js',
152
+ 'paths-app-listing\\.test\\.js',
153
+ '/tests/lib/utils/url-declarative-truth-table-124.test.js',
154
+ '\\\\tests\\\\lib\\\\utils\\\\url-declarative-truth-table-124.test.js',
155
+ 'lib/utils/url-declarative-truth-table-124.test.js',
156
+ 'url-declarative-truth-table-124\\.test\\.js',
157
+ '/tests/lib/generator/generator-external-rbac.test.js',
158
+ '\\\\tests\\\\lib\\\\generator\\\\generator-external-rbac.test.js',
159
+ 'lib/generator/generator-external-rbac.test.js',
160
+ 'generator-external-rbac\\.test\\.js',
161
+ '/tests/lib/infrastructure/helpers-ensure-admin-secrets.test.js',
162
+ '\\\\tests\\\\lib\\\\infrastructure\\\\helpers-ensure-admin-secrets.test.js',
163
+ 'lib/infrastructure/helpers-ensure-admin-secrets.test.js',
164
+ 'helpers-ensure-admin-secrets\\.test\\.js',
165
+ '/tests/lib/utils/secrets-generator.test.js',
166
+ '\\\\tests\\\\lib\\\\utils\\\\secrets-generator.test.js',
167
+ 'lib/utils/secrets-generator.test.js',
168
+ 'secrets-generator\\.test\\.js',
169
+ '/tests/lib/app/app-uncovered-lines.test.js',
170
+ '\\\\tests\\\\lib\\\\app\\\\app-uncovered-lines.test.js',
171
+ 'lib/app/app-uncovered-lines.test.js',
172
+ 'app-uncovered-lines\\.test\\.js',
173
+ '/tests/lib/utils/ensure-dev-certs-for-remote-docker.test.js',
174
+ '\\\\tests\\\\lib\\\\utils\\\\ensure-dev-certs-for-remote-docker.test.js',
175
+ 'lib/utils/ensure-dev-certs-for-remote-docker.test.js',
176
+ 'ensure-dev-certs-for-remote-docker\\.test\\.js',
177
+ '/tests/lib/generator/generator-error-paths.test.js',
178
+ '\\\\tests\\\\lib\\\\generator\\\\generator-error-paths.test.js',
179
+ 'lib/generator/generator-error-paths.test.js',
180
+ 'generator-error-paths\\.test\\.js',
181
+ '/tests/lib/generator/generator-validation.test.js',
182
+ '\\\\tests\\\\lib\\\\generator\\\\generator-validation.test.js',
183
+ 'lib/generator/generator-validation.test.js',
184
+ 'generator-validation\\.test\\.js',
185
+ '/tests/lib/core/secrets-databaselog.test.js',
186
+ '\\\\tests\\\\lib\\\\core\\\\secrets-databaselog.test.js',
187
+ 'lib/core/secrets-databaselog.test.js',
188
+ 'secrets-databaselog\\.test\\.js',
189
+ '/tests/lib/validation/schema-241-alignment.test.js',
190
+ '\\\\tests\\\\lib\\\\validation\\\\schema-241-alignment.test.js',
191
+ 'lib/validation/schema-241-alignment.test.js',
192
+ 'schema-241-alignment\\.test\\.js',
193
+ '/tests/lib/app/app.test.js',
194
+ '\\\\tests\\\\lib\\\\app\\\\app.test.js',
195
+ 'lib/app/app.test.js',
196
+ '/tests/lib/core/admin-secrets.test.js',
197
+ '\\\\tests\\\\lib\\\\core\\\\admin-secrets.test.js',
198
+ 'lib/core/admin-secrets.test.js'
199
+ ];
200
+ if (process.env.INCLUDE_LOCAL_TESTS !== 'true') {
201
+ patterns.push('/tests/local/');
202
+ patterns.push('\\\\tests\\\\local\\\\');
203
+ }
204
+ return patterns;
205
+ })(),
206
+ setupFiles: ['<rootDir>/tests/capture-real-fs.js'],
207
+ setupFilesAfterEnv: ['<rootDir>/tests/setup.js'],
208
+ testTimeout: isCI ? 10000 : 5000,
209
+ detectOpenHandles: true,
210
+ maxWorkers: 1
211
+ };
212
+
213
+ const isolatedProjects = [
214
+ makeIsolatedProject('cli-utils', ['**/tests/lib/utils/cli-utils.test.js']),
215
+ makeIsolatedProject('external-system-display', ['**/tests/lib/utils/external-system-display.test.js']),
216
+ makeIsolatedProject('dev-hosts-helper', ['**/tests/lib/utils/dev-hosts-helper.test.js']),
217
+ makeIsolatedProject('declarative-url-matrix-d-reload', [
218
+ '**/tests/lib/utils/declarative-url-matrix-d-reload.test.js'
219
+ ]),
220
+ makeIsolatedProject('parameters-validate', ['**/tests/lib/commands/parameters-validate.test.js']),
221
+ makeIsolatedProject('paths-app-listing', ['**/tests/lib/utils/paths-app-listing.test.js']),
222
+ makeIsolatedProject('datasource-validation-watch', [
223
+ '**/tests/lib/utils/datasource-validation-watch.test.js'
224
+ ]),
225
+ makeIsolatedProject('log-viewer', ['**/tests/lib/datasource/log-viewer.test.js']),
226
+ makeIsolatedProject('datasource-test-run-schema-sync', [
227
+ '**/tests/lib/utils/datasource-test-run-schema-sync.test.js'
228
+ ]),
229
+ makeIsolatedProject('infra-platform-contract', [
230
+ '**/tests/lib/parameters/infra-platform-contract.test.js'
231
+ ]),
232
+ makeIsolatedProject('database-secret-values', [
233
+ '**/tests/lib/parameters/database-secret-values.test.js'
234
+ ]),
235
+ makeIsolatedProject('infra-parameter-validate', [
236
+ '**/tests/lib/parameters/infra-parameter-validate.test.js'
237
+ ]),
238
+ makeIsolatedProject('infra-parameter-catalog', [
239
+ '**/tests/lib/parameters/infra-parameter-catalog.test.js'
240
+ ]),
241
+ makeIsolatedProject('urls-local-registry', ['**/tests/lib/utils/urls-local-registry.test.js']),
242
+ makeIsolatedProject('aifabrix-runtime-config-dir', [
243
+ '**/tests/lib/utils/aifabrix-runtime-config-dir.test.js'
244
+ ]),
245
+ makeIsolatedProject('platform-env-template-kv-catalog', [
246
+ '**/tests/lib/parameters/platform-env-template-kv-catalog.test.js'
247
+ ]),
248
+ makeIsolatedProject('resolve-infra-state-paths', [
249
+ '**/tests/lib/infrastructure/resolve-infra-state-paths.test.js'
250
+ ]),
251
+ makeIsolatedProject('dev-ssh-config-helper', [
252
+ '**/tests/lib/utils/dev-ssh-config-helper.test.js'
253
+ ]),
254
+ makeIsolatedProject('ssh-key-helper', ['**/tests/lib/utils/ssh-key-helper.test.js']),
255
+ makeIsolatedProject('secrets-ensure-catalog-fallback', [
256
+ '**/tests/lib/core/secrets-ensure-catalog-fallback.test.js'
257
+ ]),
258
+ makeIsolatedProject('secrets-ensure', ['**/tests/lib/core/secrets-ensure.test.js']),
259
+ makeIsolatedProject('url-declarative-vdir-inactive-env', [
260
+ '**/tests/lib/utils/url-declarative-vdir-inactive-env.test.js'
261
+ ]),
262
+ makeIsolatedProject('app-service-env-from-builder', [
263
+ '**/tests/lib/utils/app-service-env-from-builder.test.js'
264
+ ]),
265
+ makeIsolatedProject('infra-kv-discovery', ['**/tests/lib/parameters/infra-kv-discovery.test.js']),
266
+ makeIsolatedProject('infra-env-defaults', ['**/tests/lib/utils/infra-env-defaults.test.js']),
267
+ makeIsolatedProject('compose-traefik-template', [
268
+ '**/tests/lib/infrastructure/compose-traefik-template.test.js'
269
+ ]),
270
+ makeIsolatedProject('generator-external-rbac', [
271
+ '**/tests/lib/generator/generator-external-rbac.test.js'
272
+ ]),
273
+ makeIsolatedProject('helpers-ensure-admin-secrets', [
274
+ '**/tests/lib/infrastructure/helpers-ensure-admin-secrets.test.js'
275
+ ]),
276
+ makeIsolatedProject('url-declarative-truth-table-124', [
277
+ '**/tests/lib/utils/url-declarative-truth-table-124.test.js'
278
+ ]),
279
+ makeIsolatedProject('secrets-generator', ['**/tests/lib/utils/secrets-generator.test.js']),
280
+ makeIsolatedProject('app-uncovered-lines', ['**/tests/lib/app/app-uncovered-lines.test.js']),
281
+ makeIsolatedProject('ensure-dev-certs-for-remote-docker', [
282
+ '**/tests/lib/utils/ensure-dev-certs-for-remote-docker.test.js'
283
+ ]),
284
+ makeIsolatedProject('generator-error-paths', ['**/tests/lib/generator/generator-error-paths.test.js']),
285
+ makeIsolatedProject('generator-validation', ['**/tests/lib/generator/generator-validation.test.js']),
286
+ makeIsolatedProject('secrets-databaselog', ['**/tests/lib/core/secrets-databaselog.test.js']),
287
+ makeIsolatedProject('schema-241-alignment', ['**/tests/lib/validation/schema-241-alignment.test.js']),
288
+ makeIsolatedProject('app-module', ['**/tests/lib/app/app.test.js']),
289
+ makeIsolatedProject('admin-secrets', ['**/tests/lib/core/admin-secrets.test.js'])
290
+ ];
291
+
292
+ const allProjects = [defaultProject, ...isolatedProjects];
293
+
294
+ module.exports = {
295
+ isCI,
296
+ sharedTransform,
297
+ makeIsolatedProject,
298
+ defaultProject,
299
+ isolatedProjects,
300
+ allProjects
301
+ };
@@ -0,0 +1,62 @@
1
+ /**
2
+ * @fileoverview Dataplane Trust API — integration certificates (active, list, verify).
3
+ * @author AI Fabrix Team
4
+ * @version 2.0.0
5
+ */
6
+
7
+ const { ApiClient } = require('./index');
8
+
9
+ /**
10
+ * Get active trusted integration certificate for a datasource.
11
+ * GET /api/v1/systems/{systemKey}/datasources/{datasourceKey}/certificates/active
12
+ * @requiresPermission {Dataplane} external-system:read
13
+ * @async
14
+ * @param {string} dataplaneUrl - Dataplane base URL
15
+ * @param {Object} authConfig - Auth (Bearer)
16
+ * @param {string} systemKey - External system key
17
+ * @param {string} datasourceKey - Datasource key
18
+ * @returns {Promise<Object>} API envelope `{ success, data?, status, ... }`
19
+ */
20
+ async function getActiveIntegrationCertificate(dataplaneUrl, authConfig, systemKey, datasourceKey) {
21
+ const client = new ApiClient(dataplaneUrl, authConfig);
22
+ const path = `/api/v1/systems/${encodeURIComponent(systemKey)}/datasources/${encodeURIComponent(
23
+ datasourceKey
24
+ )}/certificates/active`;
25
+ return await client.get(path);
26
+ }
27
+
28
+ /**
29
+ * List integration certificates (optional filters).
30
+ * GET /api/v1/certificates
31
+ * @requiresPermission {Dataplane} external-system:read
32
+ * @async
33
+ * @param {string} dataplaneUrl - Dataplane base URL
34
+ * @param {Object} authConfig - Auth
35
+ * @param {Object} [params] - Query: datasourceKey, systemIdOrKey, page, pageSize
36
+ * @returns {Promise<Object>}
37
+ */
38
+ async function listIntegrationCertificates(dataplaneUrl, authConfig, params = {}) {
39
+ const client = new ApiClient(dataplaneUrl, authConfig);
40
+ return await client.get('/api/v1/certificates', { params });
41
+ }
42
+
43
+ /**
44
+ * Verify a stored integration certificate (signature and optional hash).
45
+ * POST /api/v1/certificates/verify
46
+ * @requiresPermission {Dataplane} external-system:read
47
+ * @async
48
+ * @param {string} dataplaneUrl - Dataplane base URL
49
+ * @param {Object} authConfig - Auth
50
+ * @param {import('./types/certificates.types').CertificateVerifyRequest} body - Verify request
51
+ * @returns {Promise<Object>}
52
+ */
53
+ async function verifyIntegrationCertificate(dataplaneUrl, authConfig, body) {
54
+ const client = new ApiClient(dataplaneUrl, authConfig);
55
+ return await client.post('/api/v1/certificates/verify', { body: body || {} });
56
+ }
57
+
58
+ module.exports = {
59
+ getActiveIntegrationCertificate,
60
+ listIntegrationCertificates,
61
+ verifyIntegrationCertificate
62
+ };
@@ -1,5 +1,5 @@
1
1
  /**
2
- * @fileoverview Datasources Core API functions (Dataplane /api/v1/external/). All functions require Dataplane auth; scope per endpoint in dataplane OpenAPI (e.g. external-system:read, external-system:create).
2
+ * @fileoverview Datasources Core API functions (Dataplane /api/v1/external). All functions require Dataplane auth; scope per endpoint in dataplane OpenAPI (e.g. external-system:read, external-system:create).
3
3
  * @author AI Fabrix Team
4
4
  * @version 2.0.0
5
5
  */
@@ -9,12 +9,12 @@ const { ApiClient } = require('./index');
9
9
  /** @requiresPermission {Dataplane} external-system:read (datasource list) */
10
10
  async function listDatasources(dataplaneUrl, authConfig, options = {}) {
11
11
  const client = new ApiClient(dataplaneUrl, authConfig);
12
- return await client.get('/api/v1/external/', { params: options });
12
+ return await client.get('/api/v1/external', { params: options });
13
13
  }
14
14
  /** @requiresPermission {Dataplane} external-system:create */
15
15
  async function createDatasource(dataplaneUrl, authConfig, datasourceData) {
16
16
  const client = new ApiClient(dataplaneUrl, authConfig);
17
- return await client.post('/api/v1/external/', { body: datasourceData });
17
+ return await client.post('/api/v1/external', { body: datasourceData });
18
18
  }
19
19
  /** @requiresPermission {Dataplane} external-system:read */
20
20
  async function getDatasource(dataplaneUrl, sourceIdOrKey, authConfig) {
@@ -0,0 +1,110 @@
1
+ /**
2
+ * @fileoverview HTTPS requests to Builder Server with TLS client certificate (mTLS).
3
+ * @author AI Fabrix Team
4
+ * @version 2.0.0
5
+ */
6
+
7
+ const https = require('https');
8
+ const { mergedCaForDevServer, DEFAULT_TIMEOUT_MS } = require('./dev-server-https');
9
+
10
+ /**
11
+ * Build request options and TLS agent for mTLS request.
12
+ * @param {string} url - Full URL
13
+ * @param {Object} options - { method, headers, body }
14
+ * @param {string} certPem - PEM client certificate
15
+ * @param {string} keyPem - PEM client key
16
+ * @param {string} [serverCaPem] - Optional dev root CA
17
+ * @returns {{ urlObj: URL, method: string, headers: Object, body: string|undefined, agent: https.Agent, tlsOptions: Object }}
18
+ */
19
+ function buildMtlsRequestOptions(url, options, certPem, keyPem, serverCaPem) {
20
+ const urlObj = new URL(url);
21
+ const method = (options.method || 'GET').toUpperCase();
22
+ const headers = { 'Content-Type': 'application/json', ...options.headers };
23
+ let body = options.body;
24
+ if (body !== undefined && typeof body !== 'string') {
25
+ body = JSON.stringify(body);
26
+ }
27
+ if (body) {
28
+ headers['Content-Length'] = Buffer.byteLength(body, 'utf8');
29
+ }
30
+ const merged = mergedCaForDevServer(serverCaPem);
31
+ const tlsOptions = { cert: certPem, key: keyPem, rejectUnauthorized: true };
32
+ if (merged) {
33
+ tlsOptions.ca = merged;
34
+ }
35
+ const agent = new https.Agent(tlsOptions);
36
+ return { urlObj, method, headers, body, agent, tlsOptions };
37
+ }
38
+
39
+ /**
40
+ * @param {import('http').IncomingMessage} res - HTTP response
41
+ * @param {Function} resolve - Promise resolve
42
+ * @param {Function} reject - Promise reject
43
+ */
44
+ function handleMtlsResponse(res, resolve, reject) {
45
+ const chunks = [];
46
+ res.on('data', (c) => chunks.push(c));
47
+ res.on('end', () => {
48
+ const raw = Buffer.concat(chunks).toString('utf8');
49
+ let data;
50
+ try {
51
+ data = raw ? JSON.parse(raw) : {};
52
+ } catch {
53
+ data = raw;
54
+ }
55
+ if (res.statusCode < 200 || res.statusCode >= 300) {
56
+ const msg = (data && (data.message || data.error)) || res.statusMessage || `Request failed (${res.statusCode})`;
57
+ const err = new Error(msg);
58
+ err.status = res.statusCode;
59
+ err.errorData = data;
60
+ reject(err);
61
+ } else {
62
+ resolve(data);
63
+ }
64
+ });
65
+ }
66
+
67
+ /**
68
+ * mTLS JSON request (https only). Sends X-Client-Cert header from options if present.
69
+ * @param {string} url - Full https URL
70
+ * @param {Object} options - { method, headers, body }
71
+ * @param {string} certPem - Client certificate PEM
72
+ * @param {string} keyPem - Client key PEM
73
+ * @param {string} [serverCaPem] - Dev root CA PEM
74
+ * @returns {Promise<Object>}
75
+ */
76
+ function requestWithClientCert(url, options, certPem, keyPem, serverCaPem) {
77
+ return new Promise((resolve, reject) => {
78
+ const urlObj = new URL(url);
79
+ if (urlObj.protocol !== 'https:') {
80
+ reject(new Error('mTLS request requires https URL'));
81
+ return;
82
+ }
83
+ const { method, headers, body, agent, tlsOptions } = buildMtlsRequestOptions(
84
+ url, options, certPem, keyPem, serverCaPem
85
+ );
86
+ const req = https.request(
87
+ {
88
+ hostname: urlObj.hostname,
89
+ port: urlObj.port || 443,
90
+ path: urlObj.pathname + urlObj.search,
91
+ method,
92
+ headers,
93
+ agent,
94
+ ...tlsOptions
95
+ },
96
+ (res) => handleMtlsResponse(res, resolve, reject)
97
+ );
98
+ req.on('error', reject);
99
+ req.setTimeout(DEFAULT_TIMEOUT_MS, () => {
100
+ req.destroy();
101
+ reject(new Error(`Request timed out after ${DEFAULT_TIMEOUT_MS}ms`));
102
+ });
103
+ if (body) {
104
+ req.write(body, 'utf8');
105
+ }
106
+ req.end();
107
+ });
108
+ }
109
+
110
+ module.exports = { requestWithClientCert };
@@ -0,0 +1,145 @@
1
+ /**
2
+ * @fileoverview HTTPS JSON requests to Builder Server with dev root CA merged into Node TLS trust.
3
+ * Node does not use the Windows/macOS user trust store for fetch; OS CA install alone is insufficient.
4
+ * @author AI Fabrix Team
5
+ * @version 2.0.0
6
+ */
7
+
8
+ const https = require('https');
9
+ const tls = require('tls');
10
+
11
+ const DEFAULT_TIMEOUT_MS = 15000;
12
+
13
+ /**
14
+ * Merge Mozilla roots (bundled in Node) with the dev Builder Server root CA.
15
+ * @param {string} serverCaPem - PEM-encoded root CA
16
+ * @returns {string[]|null} CA array for tls/https.Agent
17
+ */
18
+ function mergedCaForDevServer(serverCaPem) {
19
+ const pem = typeof serverCaPem === 'string' ? serverCaPem.trim() : '';
20
+ if (!pem) return null;
21
+ return [...tls.rootCertificates, pem];
22
+ }
23
+
24
+ /**
25
+ * Collect response body, parse JSON, resolve or reject.
26
+ * @param {import('http').IncomingMessage} res - HTTP response
27
+ * @param {Function} resolve - Promise resolve
28
+ * @param {Function} reject - Promise reject
29
+ */
30
+ function handleJsonHttpsResponse(res, resolve, reject) {
31
+ const chunks = [];
32
+ res.on('data', (c) => chunks.push(c));
33
+ res.on('end', () => {
34
+ const raw = Buffer.concat(chunks).toString('utf8');
35
+ let data;
36
+ try {
37
+ data = raw ? JSON.parse(raw) : {};
38
+ } catch {
39
+ data = raw;
40
+ }
41
+ if (res.statusCode < 200 || res.statusCode >= 300) {
42
+ const msg = (data && (data.message || data.error)) || res.statusMessage || `Request failed (${res.statusCode})`;
43
+ const err = new Error(msg);
44
+ err.status = res.statusCode;
45
+ err.errorData = data;
46
+ reject(err);
47
+ } else {
48
+ resolve(data);
49
+ }
50
+ });
51
+ }
52
+
53
+ /**
54
+ * @param {string} url
55
+ * @param {{ method?: string, headers?: Object, body?: * }} options
56
+ * @param {string} serverCaPem
57
+ * @returns {{ urlObj: URL, method: string, headers: Object, body: string|undefined, ca: string[] }}
58
+ */
59
+ function prepareHttpsJsonRequest(url, options, serverCaPem) {
60
+ const urlObj = new URL(url);
61
+ if (urlObj.protocol !== 'https:') {
62
+ throw new Error('Builder Server requests require https URL');
63
+ }
64
+ const ca = mergedCaForDevServer(serverCaPem);
65
+ if (!ca) {
66
+ throw new Error('serverCaPem is required for httpsJsonRequest');
67
+ }
68
+ const method = (options.method || 'GET').toUpperCase();
69
+ const headers = { 'Content-Type': 'application/json', ...options.headers };
70
+ let body = options.body;
71
+ if (body !== undefined && typeof body !== 'string') {
72
+ body = JSON.stringify(body);
73
+ }
74
+ if (body) {
75
+ headers['Content-Length'] = Buffer.byteLength(body, 'utf8');
76
+ }
77
+ return { urlObj, method, headers, body, ca };
78
+ }
79
+
80
+ /**
81
+ * HTTPS JSON request with dev-server root CA (no client cert).
82
+ * @param {string} url - Full https URL
83
+ * @param {{ method?: string, headers?: Object, body?: * }} options - Request options
84
+ * @param {string} serverCaPem - Dev root CA PEM
85
+ * @returns {Promise<Object>} Parsed JSON body on success
86
+ */
87
+ function httpsJsonRequest(url, options, serverCaPem) {
88
+ return new Promise((resolve, reject) => {
89
+ let parts;
90
+ try {
91
+ parts = prepareHttpsJsonRequest(url, options, serverCaPem);
92
+ } catch (e) {
93
+ reject(e);
94
+ return;
95
+ }
96
+ const { urlObj, method, headers, body, ca } = parts;
97
+ const agent = new https.Agent({ ca, rejectUnauthorized: true });
98
+ const req = https.request(
99
+ {
100
+ hostname: urlObj.hostname,
101
+ port: urlObj.port || 443,
102
+ path: urlObj.pathname + urlObj.search,
103
+ method,
104
+ headers,
105
+ agent,
106
+ ca,
107
+ rejectUnauthorized: true
108
+ },
109
+ (res) => handleJsonHttpsResponse(res, resolve, reject)
110
+ );
111
+ req.on('error', reject);
112
+ req.setTimeout(DEFAULT_TIMEOUT_MS, () => {
113
+ req.destroy();
114
+ reject(new Error(`Request timed out after ${DEFAULT_TIMEOUT_MS}ms`));
115
+ });
116
+ if (body) {
117
+ req.write(body, 'utf8');
118
+ }
119
+ req.end();
120
+ });
121
+ }
122
+
123
+ /**
124
+ * Use TLS client cert in the handshake only for https URLs (plain HTTP cannot carry mTLS).
125
+ * @param {string} fullUrl - Request URL
126
+ * @param {string} [clientKeyPem] - Client private key PEM
127
+ * @returns {boolean}
128
+ */
129
+ function shouldUseMtlsForUrl(fullUrl, clientKeyPem) {
130
+ if (!clientKeyPem || typeof clientKeyPem !== 'string') {
131
+ return false;
132
+ }
133
+ try {
134
+ return new URL(fullUrl).protocol === 'https:';
135
+ } catch {
136
+ return false;
137
+ }
138
+ }
139
+
140
+ module.exports = {
141
+ mergedCaForDevServer,
142
+ httpsJsonRequest,
143
+ DEFAULT_TIMEOUT_MS,
144
+ shouldUseMtlsForUrl
145
+ };