@aifabrix/builder 2.43.0 → 2.44.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (346) hide show
  1. package/.cursor/rules/anchor-docs.mdc +15 -0
  2. package/README.md +1 -1
  3. package/anchor-docs/README.md +10 -0
  4. package/anchor-docs/_TEMPLATE +24 -0
  5. package/bin/aifabrix.js +13 -4
  6. package/integration/hubspot-test/README.md +31 -0
  7. package/integration/hubspot-test/create-hubspot.js +5 -5
  8. package/integration/hubspot-test/hubspot-test-datasource-company.json +58 -462
  9. package/integration/hubspot-test/hubspot-test-datasource-contact.json +61 -555
  10. package/integration/hubspot-test/hubspot-test-datasource-deal.json +63 -506
  11. package/integration/hubspot-test/hubspot-test-datasource-users.json +42 -83
  12. package/integration/hubspot-test/hubspot-test-deploy.json +3 -3
  13. package/integration/hubspot-test/test-dataplane-down-tests.js +1 -7
  14. package/integration/hubspot-test/test-dataplane-down.js +3 -3
  15. package/integration/hubspot-test/test.js +35 -43
  16. package/integration/hubspot-test/wizard-hubspot-test-headless.yaml +23 -0
  17. package/integration/roundtrip-test-local/README.md +144 -0
  18. package/integration/roundtrip-test-local/application.yaml +13 -0
  19. package/integration/roundtrip-test-local/env.template +15 -0
  20. package/integration/roundtrip-test-local/roundtrip-test-local-datasource-roundtrip-test-company.yaml +14 -0
  21. package/integration/roundtrip-test-local/roundtrip-test-local-deploy.json +61 -0
  22. package/integration/roundtrip-test-local/roundtrip-test-local-system.yaml +25 -0
  23. package/integration/roundtrip-test-local2/README.md +144 -0
  24. package/integration/roundtrip-test-local2/application.yaml +13 -0
  25. package/integration/roundtrip-test-local2/env.template +15 -0
  26. package/integration/roundtrip-test-local2/roundtrip-test-local2-datasource-company.yaml +31 -0
  27. package/integration/roundtrip-test-local2/roundtrip-test-local2-deploy.json +86 -0
  28. package/integration/roundtrip-test-local2/roundtrip-test-local2-system.yaml +25 -0
  29. package/integration/test/wizard.yaml +8 -0
  30. package/jest.config.default.js +10 -0
  31. package/jest.config.integration.fixtures.js +22 -0
  32. package/jest.config.integration.js +21 -18
  33. package/jest.config.isolated.js +10 -0
  34. package/jest.projects.js +288 -0
  35. package/lib/api/datasources-core.api.js +3 -3
  36. package/lib/api/dev-mtls-request.js +110 -0
  37. package/lib/api/dev-server-https.js +145 -0
  38. package/lib/api/dev.api.js +133 -144
  39. package/lib/api/index.js +0 -1
  40. package/lib/api/pipeline.api.js +67 -20
  41. package/lib/api/types/dev.types.js +4 -3
  42. package/lib/api/types/pipeline.types.js +8 -5
  43. package/lib/api/types/validation-run.types.js +56 -0
  44. package/lib/api/validation-run.api.js +99 -0
  45. package/lib/api/validation-runner.js +99 -0
  46. package/lib/app/config.js +1 -1
  47. package/lib/app/deploy-status-display.js +2 -2
  48. package/lib/app/deploy.js +7 -6
  49. package/lib/app/display.js +2 -1
  50. package/lib/app/dockerfile.js +3 -2
  51. package/lib/app/down.js +2 -1
  52. package/lib/app/helpers.js +6 -5
  53. package/lib/app/index.js +27 -8
  54. package/lib/app/list.js +7 -6
  55. package/lib/app/push.js +4 -3
  56. package/lib/app/register.js +16 -7
  57. package/lib/app/rotate-secret.js +14 -13
  58. package/lib/app/run-container-start.js +184 -0
  59. package/lib/app/run-docker-fallback.js +108 -0
  60. package/lib/app/run-env-compose.js +30 -42
  61. package/lib/app/run-helpers.js +49 -126
  62. package/lib/app/run-infra-requirements.js +30 -0
  63. package/lib/app/run-resolve-image.js +21 -0
  64. package/lib/app/run.js +74 -21
  65. package/lib/app/show-display.js +1 -1
  66. package/lib/app/show.js +1 -1
  67. package/lib/build/index.js +13 -10
  68. package/lib/cli/index.js +2 -0
  69. package/lib/cli/setup-app.help.js +67 -0
  70. package/lib/cli/setup-app.js +57 -121
  71. package/lib/cli/setup-app.test-commands.js +179 -0
  72. package/lib/cli/setup-auth.js +19 -5
  73. package/lib/cli/setup-credential-deployment.js +22 -8
  74. package/lib/cli/setup-dev-path-commands.js +124 -0
  75. package/lib/cli/setup-dev.js +170 -113
  76. package/lib/cli/setup-environment.js +7 -1
  77. package/lib/cli/setup-external-system.js +62 -22
  78. package/lib/cli/setup-infra.js +126 -47
  79. package/lib/cli/setup-parameters.js +32 -0
  80. package/lib/cli/setup-secrets.js +106 -8
  81. package/lib/cli/setup-service-user.js +1 -1
  82. package/lib/cli/setup-utility.js +36 -20
  83. package/lib/commands/app-down.js +5 -7
  84. package/lib/commands/app-install.js +14 -7
  85. package/lib/commands/app-logs.js +13 -10
  86. package/lib/commands/app-shell.js +4 -1
  87. package/lib/commands/app-test.js +25 -19
  88. package/lib/commands/app.js +22 -10
  89. package/lib/commands/auth-config.js +6 -6
  90. package/lib/commands/auth-status.js +4 -3
  91. package/lib/commands/credential-env.js +4 -3
  92. package/lib/commands/credential-list.js +5 -4
  93. package/lib/commands/credential-push.js +4 -3
  94. package/lib/commands/datasource-unified-test-cli.js +495 -0
  95. package/lib/commands/datasource-unified-test-cli.options.js +149 -0
  96. package/lib/commands/datasource-validation-cli.js +129 -0
  97. package/lib/commands/datasource.js +105 -98
  98. package/lib/commands/deployment-list.js +6 -5
  99. package/lib/commands/dev-cli-handlers.js +122 -18
  100. package/lib/commands/dev-down.js +4 -3
  101. package/lib/commands/dev-init.js +231 -116
  102. package/lib/commands/dev-show-display.js +473 -0
  103. package/lib/commands/login-credentials.js +3 -2
  104. package/lib/commands/login-device.js +4 -3
  105. package/lib/commands/login.js +5 -4
  106. package/lib/commands/logout.js +8 -7
  107. package/lib/commands/parameters-validate.js +54 -0
  108. package/lib/commands/repair-datasource.js +314 -68
  109. package/lib/commands/repair-env-template.js +2 -2
  110. package/lib/commands/repair.js +21 -3
  111. package/lib/commands/secrets-list.js +23 -12
  112. package/lib/commands/secrets-remove-all.js +220 -0
  113. package/lib/commands/secrets-remove.js +21 -12
  114. package/lib/commands/secrets-set.js +21 -12
  115. package/lib/commands/secrets-validate.js +4 -4
  116. package/lib/commands/secure.js +10 -9
  117. package/lib/commands/service-user.js +26 -25
  118. package/lib/commands/test-e2e-external.js +27 -1
  119. package/lib/commands/up-common.js +3 -2
  120. package/lib/commands/up-dataplane.js +29 -16
  121. package/lib/commands/up-miso.js +19 -29
  122. package/lib/commands/upload.js +138 -39
  123. package/lib/commands/wizard-core-helpers.js +1 -1
  124. package/lib/commands/wizard-dataplane.js +4 -3
  125. package/lib/commands/wizard-helpers.js +3 -3
  126. package/lib/commands/wizard.js +2 -2
  127. package/lib/core/admin-secrets.js +14 -5
  128. package/lib/core/audit-logger.js +12 -4
  129. package/lib/core/config-attach-extensions.js +46 -0
  130. package/lib/core/config-runtime-paths.js +29 -0
  131. package/lib/core/config.js +55 -56
  132. package/lib/core/diff.js +3 -2
  133. package/lib/core/ensure-encryption-key.js +1 -1
  134. package/lib/core/secrets-ensure-infra.js +77 -0
  135. package/lib/core/secrets-ensure.js +120 -64
  136. package/lib/core/secrets-env-write.js +35 -7
  137. package/lib/core/secrets-infra-placeholder-sync.js +61 -0
  138. package/lib/core/secrets.js +200 -37
  139. package/lib/core/templates-env.js +4 -3
  140. package/lib/datasource/abac-validator.js +1 -10
  141. package/lib/datasource/deploy.js +75 -53
  142. package/lib/datasource/field-reference-validator.js +9 -6
  143. package/lib/datasource/integration-context.js +63 -0
  144. package/lib/datasource/list.js +8 -7
  145. package/lib/datasource/log-viewer.js +84 -53
  146. package/lib/datasource/resolve-app.js +4 -4
  147. package/lib/datasource/test-e2e.js +95 -146
  148. package/lib/datasource/test-integration.js +114 -122
  149. package/lib/datasource/unified-validation-run-body.js +65 -0
  150. package/lib/datasource/unified-validation-run-post.js +23 -0
  151. package/lib/datasource/unified-validation-run-resolve.js +43 -0
  152. package/lib/datasource/unified-validation-run.js +92 -0
  153. package/lib/datasource/validate.js +157 -13
  154. package/lib/deployment/deployer.js +4 -3
  155. package/lib/deployment/environment.js +7 -6
  156. package/lib/deployment/push.js +17 -8
  157. package/lib/external-system/delete.js +4 -3
  158. package/lib/external-system/deploy.js +131 -53
  159. package/lib/external-system/download-helpers.js +1 -1
  160. package/lib/external-system/download.js +7 -6
  161. package/lib/external-system/generator.js +92 -6
  162. package/lib/external-system/integration-test-dispatch.js +26 -0
  163. package/lib/external-system/test-execution.js +5 -1
  164. package/lib/external-system/test-helpers.js +0 -4
  165. package/lib/external-system/test-system-level-helpers.js +110 -0
  166. package/lib/external-system/test-system-level.js +83 -44
  167. package/lib/external-system/test.js +59 -8
  168. package/lib/generator/builders.js +23 -11
  169. package/lib/generator/deploy-manifest-azure-kv.js +81 -0
  170. package/lib/generator/external.js +16 -4
  171. package/lib/generator/helpers.js +58 -3
  172. package/lib/generator/index.js +4 -0
  173. package/lib/generator/split-readme.js +12 -7
  174. package/lib/generator/split-variables.js +2 -1
  175. package/lib/generator/split.js +1 -1
  176. package/lib/generator/wizard-readme.js +3 -3
  177. package/lib/generator/wizard.js +8 -8
  178. package/lib/infrastructure/compose.js +60 -6
  179. package/lib/infrastructure/helpers.js +201 -29
  180. package/lib/infrastructure/index.js +28 -17
  181. package/lib/infrastructure/services.js +21 -15
  182. package/lib/internal/fs-real-sync.js +104 -0
  183. package/lib/internal/node-fs.js +98 -0
  184. package/lib/parameters/database-secret-values.js +173 -0
  185. package/lib/parameters/infra-kv-discovery.js +121 -0
  186. package/lib/parameters/infra-parameter-catalog.js +458 -0
  187. package/lib/parameters/infra-parameter-validate.js +64 -0
  188. package/lib/schema/application-schema.json +37 -17
  189. package/lib/schema/datasource-test-run.schema.json +493 -0
  190. package/lib/schema/deployment-rules.yaml +102 -63
  191. package/lib/schema/external-datasource.schema.json +1200 -442
  192. package/lib/schema/external-system.schema.json +181 -5
  193. package/lib/schema/flag-map-validation-run.json +31 -0
  194. package/lib/schema/infra-parameter.schema.json +106 -0
  195. package/lib/schema/infra.parameter.yaml +421 -0
  196. package/lib/schema/type/credential-auth-templates.json +40 -0
  197. package/lib/schema/type/document-storage.json +213 -0
  198. package/lib/schema/type/message-service.json +123 -0
  199. package/lib/schema/type/vector-store.json +88 -0
  200. package/lib/utils/aifabrix-runtime-config-dir.js +132 -0
  201. package/lib/utils/api-error-handler.js +2 -2
  202. package/lib/utils/api.js +49 -14
  203. package/lib/utils/app-register-api.js +3 -2
  204. package/lib/utils/app-register-auth.js +1 -1
  205. package/lib/utils/app-register-config.js +4 -4
  206. package/lib/utils/app-register-display.js +3 -2
  207. package/lib/utils/app-register-validator.js +3 -2
  208. package/lib/utils/app-run-containers.js +26 -22
  209. package/lib/utils/app-scoped-config.js +31 -0
  210. package/lib/utils/app-service-env-from-builder.js +164 -0
  211. package/lib/utils/build-copy.js +1 -1
  212. package/lib/utils/build-helpers.js +20 -20
  213. package/lib/utils/build-resolve-image.js +165 -0
  214. package/lib/utils/cli-layout-chalk.js +8 -0
  215. package/lib/utils/cli-test-layout-chalk.js +267 -0
  216. package/lib/utils/cli-utils.js +88 -11
  217. package/lib/utils/compose-db-passwords.js +138 -0
  218. package/lib/utils/compose-generate-docker-compose.js +216 -0
  219. package/lib/utils/compose-generator.js +197 -291
  220. package/lib/utils/compose-miso-env.js +18 -0
  221. package/lib/utils/compose-traefik-ingress-base.js +158 -0
  222. package/lib/utils/config-paths.js +166 -7
  223. package/lib/utils/config-scoped-resources-preference.js +41 -0
  224. package/lib/utils/controller-deployment-outcome.js +68 -0
  225. package/lib/utils/credential-display.js +2 -2
  226. package/lib/utils/dataplane-pipeline-warning.js +4 -3
  227. package/lib/utils/datasource-test-run-capability-scope.js +43 -0
  228. package/lib/utils/datasource-test-run-debug-display.js +137 -0
  229. package/lib/utils/datasource-test-run-debug-slice.js +93 -0
  230. package/lib/utils/datasource-test-run-display.js +442 -0
  231. package/lib/utils/datasource-test-run-exit.js +58 -0
  232. package/lib/utils/datasource-test-run-legacy-adapter.js +93 -0
  233. package/lib/utils/datasource-test-run-report-version.js +51 -0
  234. package/lib/utils/datasource-test-run-schema-sync.js +59 -0
  235. package/lib/utils/datasource-test-run-tty-log.js +81 -0
  236. package/lib/utils/datasource-validation-watch.js +266 -0
  237. package/lib/utils/declarative-url-ports.js +47 -0
  238. package/lib/utils/derive-env-key-from-client-id.js +41 -0
  239. package/lib/utils/dev-ca-install.js +185 -23
  240. package/lib/utils/dev-cert-helper.js +266 -17
  241. package/lib/utils/dev-hosts-helper.js +307 -0
  242. package/lib/utils/dev-init-cert-hints.js +37 -0
  243. package/lib/utils/dev-init-health-messages.js +52 -0
  244. package/lib/utils/dev-init-resolve.js +86 -0
  245. package/lib/utils/dev-init-ssh-merge.js +65 -0
  246. package/lib/utils/dev-ssh-config-helper.js +196 -0
  247. package/lib/utils/dev-user-groups.js +93 -0
  248. package/lib/utils/docker-build.js +42 -17
  249. package/lib/utils/docker-exec.js +28 -0
  250. package/lib/utils/docker-manifest-public-port.js +116 -0
  251. package/lib/utils/docker-not-running-hint.js +52 -0
  252. package/lib/utils/docker.js +98 -11
  253. package/lib/utils/ensure-dev-certs-for-remote-docker.js +192 -0
  254. package/lib/utils/env-config-loader.js +10 -91
  255. package/lib/utils/env-copy.js +19 -10
  256. package/lib/utils/env-map.js +35 -8
  257. package/lib/utils/env-template.js +2 -2
  258. package/lib/utils/environment-scoped-resources.js +144 -0
  259. package/lib/utils/error-formatter.js +92 -13
  260. package/lib/utils/error-formatters/http-status-errors.js +6 -5
  261. package/lib/utils/error-formatters/network-errors.js +2 -1
  262. package/lib/utils/error-formatters/permission-errors.js +2 -1
  263. package/lib/utils/error-formatters/validation-errors.js +2 -1
  264. package/lib/utils/external-readme.js +8 -1
  265. package/lib/utils/external-system-display.js +234 -136
  266. package/lib/utils/external-system-local-test-tty.js +389 -0
  267. package/lib/utils/external-system-readiness-core.js +377 -0
  268. package/lib/utils/external-system-readiness-deploy-display.js +270 -0
  269. package/lib/utils/external-system-readiness-display-internals.js +150 -0
  270. package/lib/utils/external-system-readiness-display.js +186 -0
  271. package/lib/utils/external-system-test-helpers.js +24 -6
  272. package/lib/utils/external-system-validators.js +30 -12
  273. package/lib/utils/health-check-url.js +119 -0
  274. package/lib/utils/health-check.js +59 -25
  275. package/lib/utils/help-builder.js +11 -8
  276. package/lib/utils/image-version.js +4 -8
  277. package/lib/utils/infra-containers.js +4 -7
  278. package/lib/utils/infra-env-defaults.js +162 -0
  279. package/lib/utils/infra-status-display.js +167 -0
  280. package/lib/utils/infra-status.js +16 -8
  281. package/lib/utils/local-secrets.js +3 -4
  282. package/lib/utils/paths.js +134 -47
  283. package/lib/utils/port-resolver.js +10 -23
  284. package/lib/utils/redis-env-scope.js +62 -0
  285. package/lib/utils/register-aifabrix-shell-env.js +204 -0
  286. package/lib/utils/remote-builder-validation.js +99 -0
  287. package/lib/utils/remote-dev-auth.js +117 -21
  288. package/lib/utils/remote-docker-env.js +67 -15
  289. package/lib/utils/remote-secrets-loader.js +13 -4
  290. package/lib/utils/resolve-docker-image-ref.js +124 -0
  291. package/lib/utils/schema-loader.js +22 -9
  292. package/lib/utils/secrets-bash-kv.js +25 -0
  293. package/lib/utils/secrets-generator.js +169 -49
  294. package/lib/utils/secrets-helpers.js +70 -59
  295. package/lib/utils/secrets-kv-scope.js +60 -0
  296. package/lib/utils/secrets-utils.js +32 -38
  297. package/lib/utils/secrets-validation.js +3 -1
  298. package/lib/utils/secrets-yaml-preserve.js +109 -0
  299. package/lib/utils/ssh-key-helper.js +4 -2
  300. package/lib/utils/template-helpers.js +2 -2
  301. package/lib/utils/test-log-writer.js +3 -3
  302. package/lib/utils/token-manager.js +1 -2
  303. package/lib/utils/url-declarative-public-base.js +188 -0
  304. package/lib/utils/url-declarative-resolve-build.js +493 -0
  305. package/lib/utils/url-declarative-resolve-load-doc.js +51 -0
  306. package/lib/utils/url-declarative-resolve.js +220 -0
  307. package/lib/utils/url-declarative-token-parse.js +74 -0
  308. package/lib/utils/url-declarative-url-flags.js +50 -0
  309. package/lib/utils/url-declarative-vdir-inactive-env.js +99 -0
  310. package/lib/utils/url-public-path-prefix.js +34 -0
  311. package/lib/utils/urls-local-registry.js +220 -0
  312. package/lib/utils/validation-report-tty-kit.js +77 -0
  313. package/lib/utils/validation-run-poll.js +89 -0
  314. package/lib/utils/validation-run-post-retry.js +73 -0
  315. package/lib/utils/validation-run-request.js +98 -0
  316. package/lib/utils/variable-transformer.js +21 -4
  317. package/lib/utils/yaml-preserve.js +33 -14
  318. package/lib/validation/datasource-warnings.js +56 -0
  319. package/lib/validation/env-template-auth.js +1 -1
  320. package/lib/validation/external-manifest-validator.js +27 -7
  321. package/lib/validation/validate-display.js +37 -31
  322. package/lib/validation/validate.js +4 -13
  323. package/lib/validation/validator-unresolved-placeholders.js +98 -0
  324. package/lib/validation/validator.js +22 -65
  325. package/lib/validation/wizard-config-validator.js +2 -1
  326. package/package.json +7 -3
  327. package/scripts/check-datasource-test-run-schema-sync.js +34 -0
  328. package/scripts/diagnose-cli.js +150 -0
  329. package/scripts/install-local.js +304 -55
  330. package/templates/README.md +15 -2
  331. package/templates/applications/dataplane/application.yaml +52 -2
  332. package/templates/applications/dataplane/env.template +75 -17
  333. package/templates/applications/dataplane/rbac.yaml +8 -0
  334. package/templates/applications/keycloak/application.yaml +9 -1
  335. package/templates/applications/keycloak/env.template +15 -6
  336. package/templates/applications/miso-controller/application.yaml +10 -2
  337. package/templates/applications/miso-controller/env.template +42 -12
  338. package/templates/applications/miso-controller/rbac.yaml +5 -0
  339. package/templates/external-system/README.md.hbs +20 -7
  340. package/templates/external-system/deploy.js.hbs +5 -5
  341. package/templates/external-system/external-datasource.yaml.hbs +197 -118
  342. package/templates/infra/compose.yaml.hbs +20 -4
  343. package/templates/python/docker-compose.hbs +16 -0
  344. package/templates/typescript/docker-compose.hbs +16 -0
  345. package/lib/api/external-test.api.js +0 -111
  346. package/lib/schema/env-config.yaml +0 -60
@@ -0,0 +1,220 @@
1
+ /**
2
+ * urls.local.yaml beside effective config.yaml (same directory as secrets.local.yaml).
3
+ * When AIFABRIX_HOME is POSIX $HOME but config lives in $HOME/.aifabrix/, the registry
4
+ * is $HOME/.aifabrix/urls.local.yaml (not $HOME/urls.local.yaml).
5
+ *
6
+ * @fileoverview Read/write registry; scan each builder app folder for application.yaml
7
+ * @author AI Fabrix Team
8
+ * @version 1.0.0
9
+ */
10
+
11
+ 'use strict';
12
+
13
+ const fsRealSync = require('../internal/fs-real-sync');
14
+ const path = require('path');
15
+ const yaml = require('js-yaml');
16
+ const { DECLARATIVE_URL_INFRA_DEFAULTS } = require('./infra-env-defaults');
17
+ const pathsUtil = require('./paths');
18
+
19
+ /**
20
+ * @returns {string} Absolute path to urls.local.yaml (primary; beside config.yaml)
21
+ */
22
+ function getUrlsLocalYamlPath() {
23
+ return path.join(pathsUtil.getConfigDirForPaths(), 'urls.local.yaml');
24
+ }
25
+
26
+ /** @returns {string} Legacy path when registry was stored under getAifabrixHome() only */
27
+ function getLegacyUrlsLocalYamlPath() {
28
+ return path.join(pathsUtil.getAifabrixHome(), 'urls.local.yaml');
29
+ }
30
+
31
+ function loadRegistryYamlFile(filePath) {
32
+ try {
33
+ const doc = yaml.load(fsRealSync.readFileSync(filePath, 'utf8'));
34
+ return doc && typeof doc === 'object' ? doc : {};
35
+ } catch {
36
+ return {};
37
+ }
38
+ }
39
+
40
+ /**
41
+ * @returns {Record<string, unknown>}
42
+ */
43
+ function readUrlsLocalRegistrySync() {
44
+ const primary = getUrlsLocalYamlPath();
45
+ if (fsRealSync.existsSync(primary)) {
46
+ return loadRegistryYamlFile(primary);
47
+ }
48
+ const legacy = getLegacyUrlsLocalYamlPath();
49
+ if (legacy !== primary && fsRealSync.existsSync(legacy)) {
50
+ return loadRegistryYamlFile(legacy);
51
+ }
52
+ return {};
53
+ }
54
+
55
+ /**
56
+ * @param {Object} data - Full registry object to write
57
+ */
58
+ function writeUrlsLocalRegistrySync(data) {
59
+ const p = getUrlsLocalYamlPath();
60
+ const dir = path.dirname(p);
61
+ if (!fsRealSync.existsSync(dir)) {
62
+ fsRealSync.mkdirSync(dir, { recursive: true, mode: 0o700 });
63
+ }
64
+ const body = `${yaml.dump(data, { lineWidth: 120, noRefs: true }).trim()}\n`;
65
+ fsRealSync.writeFileSync(p, body, { mode: 0o600 });
66
+ }
67
+
68
+ /**
69
+ * Normalize front-door pattern for URLs (/data/* → /data).
70
+ * @param {string} pattern
71
+ * @returns {string}
72
+ */
73
+ function normalizePatternForUrl(pattern) {
74
+ if (!pattern || typeof pattern !== 'string') {
75
+ return '/';
76
+ }
77
+ let p = pattern.trim();
78
+ if (!p.startsWith('/')) {
79
+ p = `/${p}`;
80
+ }
81
+ p = p.replace(/\*+$/, '').replace(/\/+$/, '') || '/';
82
+ return p;
83
+ }
84
+
85
+ /**
86
+ * @param {Object} merged
87
+ * @returns {Object} same object after persist
88
+ */
89
+ function writeMergedRegistry(merged) {
90
+ writeUrlsLocalRegistrySync(merged);
91
+ return merged;
92
+ }
93
+
94
+ /**
95
+ * @param {string} cfgPath
96
+ * @returns {object|null}
97
+ */
98
+ function tryLoadApplicationYaml(cfgPath) {
99
+ if (!fsRealSync.existsSync(cfgPath)) {
100
+ return null;
101
+ }
102
+ try {
103
+ const doc = yaml.load(fsRealSync.readFileSync(cfgPath, 'utf8'));
104
+ return doc && typeof doc === 'object' ? doc : null;
105
+ } catch {
106
+ return null;
107
+ }
108
+ }
109
+
110
+ /**
111
+ * @param {object} doc
112
+ * @returns {number|null}
113
+ */
114
+ function readExplicitContainerPortFromDoc(doc) {
115
+ const cpRaw = doc.build && doc.build.containerPort;
116
+ if (typeof cpRaw === 'number' && cpRaw > 0) {
117
+ return cpRaw;
118
+ }
119
+ if (typeof cpRaw === 'string' && /^\d+$/.test(cpRaw.trim())) {
120
+ return parseInt(cpRaw.trim(), 10);
121
+ }
122
+ return null;
123
+ }
124
+
125
+ /**
126
+ * @param {Object} merged
127
+ * @param {object} doc
128
+ * @param {string} folderName - builder/<folderName>
129
+ */
130
+ function mergeDocIntoRegistry(merged, doc, folderName) {
131
+ const appKey = (doc.app && doc.app.key) || folderName;
132
+ let port = null;
133
+ if (typeof doc.port === 'number' && doc.port > 0) {
134
+ port = doc.port;
135
+ } else if (typeof doc.port === 'string' && /^\d+$/.test(doc.port.trim())) {
136
+ port = parseInt(doc.port.trim(), 10);
137
+ }
138
+ if (port === null || port <= 0) {
139
+ return;
140
+ }
141
+ const rawPattern = doc.frontDoorRouting && doc.frontDoorRouting.pattern;
142
+ const pattern =
143
+ typeof rawPattern === 'string'
144
+ ? rawPattern
145
+ : DECLARATIVE_URL_INFRA_DEFAULTS.frontDoorPatternWhenUnspecified;
146
+ merged[`${appKey}-port`] = port;
147
+ merged[`${appKey}-pattern`] = pattern;
148
+
149
+ const explicitC = readExplicitContainerPortFromDoc(doc);
150
+ const ckey = `${appKey}-containerPort`;
151
+ if (explicitC !== null) {
152
+ merged[ckey] = explicitC;
153
+ } else {
154
+ delete merged[ckey];
155
+ }
156
+ }
157
+
158
+ /**
159
+ * Merge scan results into registry (does not remove stale keys).
160
+ * @param {string|null} projectRoot - getProjectRoot() or null
161
+ * @returns {Object} Updated registry
162
+ */
163
+ function refreshUrlsLocalRegistryFromBuilder(projectRoot) {
164
+ const root = projectRoot || pathsUtil.getProjectRoot();
165
+ const merged = { ...readUrlsLocalRegistrySync() };
166
+ if (!root) {
167
+ return writeMergedRegistry(merged);
168
+ }
169
+ const builderDir = path.join(root, 'builder');
170
+ if (!fsRealSync.existsSync(builderDir) || !fsRealSync.statSync(builderDir).isDirectory()) {
171
+ return writeMergedRegistry(merged);
172
+ }
173
+ for (const ent of fsRealSync.readdirSync(builderDir, { withFileTypes: true })) {
174
+ if (!ent.isDirectory()) {
175
+ continue;
176
+ }
177
+ const doc = tryLoadApplicationYaml(path.join(builderDir, ent.name, 'application.yaml'));
178
+ if (!doc) {
179
+ continue;
180
+ }
181
+ mergeDocIntoRegistry(merged, doc, ent.name);
182
+ }
183
+ return writeMergedRegistry(merged);
184
+ }
185
+
186
+ /**
187
+ * @param {string} appKey
188
+ * @param {Object} registry
189
+ * @returns {{ port: number, containerPort: number|null, pattern: string }|null}
190
+ */
191
+ function getRegistryEntryForApp(appKey, registry) {
192
+ const r = registry || {};
193
+ const portKey = `${appKey}-port`;
194
+ const patKey = `${appKey}-pattern`;
195
+ const cportKey = `${appKey}-containerPort`;
196
+ const port = r[portKey];
197
+ const pattern = r[patKey];
198
+ const cport = r[cportKey];
199
+ if (typeof port !== 'number' || port <= 0) {
200
+ return null;
201
+ }
202
+ const containerPort = typeof cport === 'number' && cport > 0 ? cport : null;
203
+ return {
204
+ port,
205
+ containerPort,
206
+ pattern:
207
+ typeof pattern === 'string'
208
+ ? pattern
209
+ : DECLARATIVE_URL_INFRA_DEFAULTS.frontDoorPatternWhenUnspecified
210
+ };
211
+ }
212
+
213
+ module.exports = {
214
+ getUrlsLocalYamlPath,
215
+ readUrlsLocalRegistrySync,
216
+ writeUrlsLocalRegistrySync,
217
+ refreshUrlsLocalRegistryFromBuilder,
218
+ normalizePatternForUrl,
219
+ getRegistryEntryForApp
220
+ };
@@ -0,0 +1,77 @@
1
+ /**
2
+ * @fileoverview Reusable TTY helpers for validation/test reports (verdict, readiness, data-quality lines).
3
+ */
4
+
5
+ 'use strict';
6
+
7
+ const { SEP, statusGlyph } = require('./datasource-test-run-display');
8
+
9
+ const TRUST_LINE_LABELS = Object.freeze({
10
+ schema: 'Schema coverage',
11
+ consistency: 'Data consistency',
12
+ reliability: 'Data reliability'
13
+ });
14
+
15
+ function rollupGlyph(r) {
16
+ return statusGlyph(r === 'fail' ? 'fail' : r === 'warn' ? 'warn' : 'ok');
17
+ }
18
+
19
+ function verdictLineFromEnvelope(rootStatus, certStatus, runType) {
20
+ if (rootStatus === 'skipped') return '⏭ Skipped';
21
+ if (rootStatus === 'warn') return '⚠ Limited production use';
22
+ if (rootStatus === 'fail') {
23
+ if (runType === 'integration') return '✖ Pipeline not working';
24
+ if (runType === 'e2e') return '✖ Not usable';
25
+ return '✖ Configuration invalid';
26
+ }
27
+ if (certStatus === 'not_passed') return '✔ Functional with certification gaps';
28
+ return '✔ Suitable for production use';
29
+ }
30
+
31
+ function verdictLineLocalExternalTest(status) {
32
+ if (status === 'ok') return '✔ Suitable for continued setup (local manifest check passed).';
33
+ if (status === 'warn') return '⚠ Limited production use';
34
+ return '✖ Configuration invalid';
35
+ }
36
+
37
+ function readinessLineFromAggregateStatus(status) {
38
+ if (status === 'fail') return `Readiness: ${statusGlyph('fail')} Not ready`;
39
+ if (status === 'warn') return `Readiness: ${statusGlyph('warn')} Partial`;
40
+ return `Readiness: ${statusGlyph('ok')} Ready`;
41
+ }
42
+
43
+ function readinessLineFromDataReadiness(dataReadiness) {
44
+ if (!dataReadiness) return null;
45
+ if (dataReadiness === 'not_ready') return `Readiness: ${statusGlyph('fail')} Not ready`;
46
+ if (dataReadiness === 'partial') return `Readiness: ${statusGlyph('warn')} Partial`;
47
+ if (dataReadiness === 'ready') return `Readiness: ${statusGlyph('ok')} Ready`;
48
+ return null;
49
+ }
50
+
51
+ function formatDataQualityLines(rollups, descriptions) {
52
+ const d = descriptions || {};
53
+ return [
54
+ `${rollupGlyph(rollups.schema)} ${TRUST_LINE_LABELS.schema}${d.schema ? ` — ${d.schema}` : ''}`,
55
+ `${rollupGlyph(rollups.consistency)} ${TRUST_LINE_LABELS.consistency}${d.consistency ? ` — ${d.consistency}` : ''}`,
56
+ `${rollupGlyph(rollups.reliability)} ${TRUST_LINE_LABELS.reliability}${d.reliability ? ` — ${d.reliability}` : ''}`
57
+ ];
58
+ }
59
+
60
+ function pushSeparatorBlock(lines) {
61
+ lines.push('');
62
+ lines.push(SEP);
63
+ lines.push('');
64
+ }
65
+
66
+ module.exports = {
67
+ SEP,
68
+ statusGlyph,
69
+ TRUST_LINE_LABELS,
70
+ rollupGlyph,
71
+ verdictLineFromEnvelope,
72
+ verdictLineLocalExternalTest,
73
+ readinessLineFromAggregateStatus,
74
+ readinessLineFromDataReadiness,
75
+ formatDataQualityLines,
76
+ pushSeparatorBlock
77
+ };
@@ -0,0 +1,89 @@
1
+ /**
2
+ * @fileoverview Poll GET /api/v1/validation/run/{testRunId} until reportCompleteness is full (plan §3.5–3.6).
3
+ * @author AI Fabrix Team
4
+ * @version 2.0.0
5
+ */
6
+
7
+ const { getValidationRunWithTransportRetry } = require('./validation-run-post-retry');
8
+
9
+ const INITIAL_INTERVAL_MS = 2000;
10
+ const MAX_INTERVAL_MS = 15000;
11
+
12
+ /**
13
+ * Delay between polls after attempt `n` (0-based): 2s, 4s, 8s, … cap 15s.
14
+ * @param {number} attemptIndex - Zero-based poll index after initial POST
15
+ * @returns {number}
16
+ */
17
+ function nextPollDelayMs(attemptIndex) {
18
+ const raw = INITIAL_INTERVAL_MS * 2 ** Math.max(0, attemptIndex);
19
+ return Math.min(raw, MAX_INTERVAL_MS);
20
+ }
21
+
22
+ function sleep(ms) {
23
+ return new Promise(resolve => setTimeout(resolve, ms));
24
+ }
25
+
26
+ /**
27
+ * Whether polling should stop on this envelope.
28
+ * @param {Object} envelope - DatasourceTestRun-like
29
+ * @returns {boolean}
30
+ */
31
+ function isTerminalReportCompleteness(envelope) {
32
+ if (!envelope || typeof envelope !== 'object') return false;
33
+ return envelope.reportCompleteness === 'full';
34
+ }
35
+
36
+ /**
37
+ * Poll until reportCompleteness === 'full' or budget exhausted.
38
+ * @async
39
+ * @param {Object} opts
40
+ * @param {string} opts.dataplaneUrl
41
+ * @param {Object} opts.authConfig
42
+ * @param {string} opts.testRunId
43
+ * @param {number} opts.budgetMs - Remaining wall-clock budget for polls only (POST excluded)
44
+ * @param {typeof getValidationRunWithTransportRetry} [opts.fetchRun] - Inject for tests (default: GET with transport retry)
45
+ * @returns {Promise<{ envelope: Object|null, timedOut: boolean, lastApiResult: Object|null }>}
46
+ */
47
+ async function pollValidationRunUntilComplete(opts) {
48
+ const {
49
+ dataplaneUrl,
50
+ authConfig,
51
+ testRunId,
52
+ budgetMs,
53
+ fetchRun = getValidationRunWithTransportRetry
54
+ } = opts;
55
+ const deadline = Date.now() + Math.max(0, budgetMs);
56
+ let attempt = 0;
57
+ let lastApiResult = null;
58
+ let envelope = null;
59
+
60
+ while (Date.now() < deadline) {
61
+ const remaining = deadline - Date.now();
62
+ if (remaining <= 0) break;
63
+
64
+ lastApiResult = await fetchRun(dataplaneUrl, authConfig, testRunId);
65
+ if (!lastApiResult.success) {
66
+ return { envelope: null, timedOut: false, lastApiResult };
67
+ }
68
+ envelope = lastApiResult.data;
69
+ if (isTerminalReportCompleteness(envelope)) {
70
+ return { envelope, timedOut: false, lastApiResult };
71
+ }
72
+
73
+ const delay = Math.min(nextPollDelayMs(attempt), Math.max(0, deadline - Date.now()));
74
+ attempt += 1;
75
+ if (delay > 0) {
76
+ await sleep(delay);
77
+ }
78
+ }
79
+
80
+ return { envelope, timedOut: true, lastApiResult };
81
+ }
82
+
83
+ module.exports = {
84
+ INITIAL_INTERVAL_MS,
85
+ MAX_INTERVAL_MS,
86
+ nextPollDelayMs,
87
+ pollValidationRunUntilComplete,
88
+ isTerminalReportCompleteness
89
+ };
@@ -0,0 +1,73 @@
1
+ /**
2
+ * @fileoverview Transient transport retries for POST validation/run and GET poll.
3
+ * @author AI Fabrix Team
4
+ * @version 2.0.0
5
+ */
6
+
7
+ const { postValidationRun, getValidationRun } = require('../api/validation-run.api');
8
+
9
+ const RETRYABLE_CODES = new Set(['ECONNRESET', 'ETIMEDOUT', 'ECONNABORTED']);
10
+
11
+ /**
12
+ * @param {Object} res - ApiClient-style result
13
+ * @returns {boolean}
14
+ */
15
+ function isRetryablePostFailure(res) {
16
+ if (!res || res.success) return false;
17
+ if (!res.network) return false;
18
+ const err = res.originalError;
19
+ if (!err) return false;
20
+ const code = err.code || (err.cause && err.cause.code);
21
+ if (code && RETRYABLE_CODES.has(code)) return true;
22
+ if (err.name === 'AbortError') return true;
23
+ return false;
24
+ }
25
+
26
+ function sleep(ms) {
27
+ return new Promise(resolve => setTimeout(resolve, ms));
28
+ }
29
+
30
+ /**
31
+ * POST validation run with up to 2 retries on transient socket/timeout errors (1s / 2s backoff).
32
+ * Does not retry HTTP 4xx/5xx.
33
+ * @param {string} dataplaneUrl
34
+ * @param {Object} authConfig
35
+ * @param {Object} body
36
+ * @returns {Promise<Object>}
37
+ */
38
+ async function postValidationRunWithTransportRetry(dataplaneUrl, authConfig, body) {
39
+ let last = await postValidationRun(dataplaneUrl, authConfig, body);
40
+ if (last.success || !isRetryablePostFailure(last)) return last;
41
+
42
+ await sleep(1000);
43
+ last = await postValidationRun(dataplaneUrl, authConfig, body);
44
+ if (last.success || !isRetryablePostFailure(last)) return last;
45
+
46
+ await sleep(2000);
47
+ return postValidationRun(dataplaneUrl, authConfig, body);
48
+ }
49
+
50
+ /**
51
+ * GET validation run poll with up to 2 retries on transient socket/timeout errors (1s / 2s backoff).
52
+ * @param {string} dataplaneUrl
53
+ * @param {Object} authConfig
54
+ * @param {string} testRunId
55
+ * @returns {Promise<Object>}
56
+ */
57
+ async function getValidationRunWithTransportRetry(dataplaneUrl, authConfig, testRunId) {
58
+ let last = await getValidationRun(dataplaneUrl, authConfig, testRunId);
59
+ if (last.success || !isRetryablePostFailure(last)) return last;
60
+
61
+ await sleep(1000);
62
+ last = await getValidationRun(dataplaneUrl, authConfig, testRunId);
63
+ if (last.success || !isRetryablePostFailure(last)) return last;
64
+
65
+ await sleep(2000);
66
+ return getValidationRun(dataplaneUrl, authConfig, testRunId);
67
+ }
68
+
69
+ module.exports = {
70
+ isRetryablePostFailure,
71
+ postValidationRunWithTransportRetry,
72
+ getValidationRunWithTransportRetry
73
+ };
@@ -0,0 +1,98 @@
1
+ /**
2
+ * @fileoverview Build ValidationRunRequest bodies for datasource-scoped CLI commands.
3
+ * @author AI Fabrix Team
4
+ * @version 2.0.0
5
+ */
6
+
7
+ /**
8
+ * Whether the unified validation request should set includeDebug (any `--debug` or `--debug <level>`).
9
+ * @param {*} debugOpt - Commander `options.debug`
10
+ * @returns {boolean}
11
+ */
12
+ function includeDebugForRequest(debugOpt) {
13
+ if (debugOpt === undefined || debugOpt === false || debugOpt === null || debugOpt === '') {
14
+ return false;
15
+ }
16
+ return true;
17
+ }
18
+
19
+ /**
20
+ * Merge E2E options from CLI flags into e2eOptions for ExternalDataSourceE2ETestRequest (dataplane).
21
+ * @param {Object} options - CLI-derived options
22
+ * @param {boolean} [options.debug]
23
+ * @param {boolean} [options.verbose]
24
+ * @param {boolean} [options.testCrud]
25
+ * @param {string} [options.recordId]
26
+ * @param {boolean} [options.cleanup]
27
+ * @param {string|Object} [options.primaryKeyValue]
28
+ * @param {Object} [options.e2eOptionsExtra] - Shallow-merged last (e.g. server-specific drill-down fields)
29
+ * @returns {Object}
30
+ */
31
+ function buildE2eOptionsFromCli(options = {}) {
32
+ const e2e = {};
33
+ if (options.debug) e2e.includeDebug = true;
34
+ if (options.verbose) e2e.audit = true;
35
+ if (options.testCrud === true) e2e.testCrud = true;
36
+ if (options.recordId !== undefined && options.recordId !== null && options.recordId !== '') {
37
+ e2e.recordId = String(options.recordId);
38
+ }
39
+ if (options.cleanup === false) e2e.cleanup = false;
40
+ else if (options.cleanup === true) e2e.cleanup = true;
41
+ if (options.primaryKeyValue !== undefined && options.primaryKeyValue !== null) {
42
+ e2e.primaryKeyValue = options.primaryKeyValue;
43
+ }
44
+ if (options.e2eOptionsExtra && typeof options.e2eOptionsExtra === 'object') {
45
+ Object.assign(e2e, options.e2eOptionsExtra);
46
+ }
47
+ return e2e;
48
+ }
49
+
50
+ /**
51
+ * Build request body for validationScope=externalDataSource (single datasource in DB).
52
+ * @param {Object} params
53
+ * @param {string} params.systemKey - External system key
54
+ * @param {string} params.datasourceKey - Datasource key
55
+ * @param {'test'|'integration'|'e2e'} params.runType
56
+ * @param {Object} [params.payloadTemplate] - Required for integration-style payload tests when runType=test
57
+ * @param {boolean} [params.asyncRun]
58
+ * @param {boolean} [params.includeDebug]
59
+ * @param {boolean} [params.explain]
60
+ * @param {Object} [params.e2eOptions] - Merged with buildE2eOptionsFromCli when both used by caller
61
+ * @returns {import('../api/types/validation-run.types').ValidationRunRequestBody}
62
+ */
63
+ function buildExternalDataSourceValidationRequest(params) {
64
+ const {
65
+ systemKey,
66
+ datasourceKey,
67
+ runType,
68
+ payloadTemplate,
69
+ asyncRun,
70
+ includeDebug,
71
+ explain,
72
+ e2eOptions
73
+ } = params;
74
+ if (!systemKey || !datasourceKey || !runType) {
75
+ throw new Error('systemKey, datasourceKey, and runType are required');
76
+ }
77
+ /** @type {import('../api/types/validation-run.types').ValidationRunRequestBody} */
78
+ const body = {
79
+ validationScope: 'externalDataSource',
80
+ systemIdOrKey: systemKey,
81
+ datasourceKey,
82
+ runType
83
+ };
84
+ if (payloadTemplate !== undefined) body.payloadTemplate = payloadTemplate;
85
+ if (asyncRun === true) body.asyncRun = true;
86
+ if (includeDebug === true) body.includeDebug = true;
87
+ if (explain === true) body.explain = true;
88
+ if (e2eOptions && typeof e2eOptions === 'object' && Object.keys(e2eOptions).length > 0) {
89
+ body.e2eOptions = e2eOptions;
90
+ }
91
+ return body;
92
+ }
93
+
94
+ module.exports = {
95
+ includeDebugForRequest,
96
+ buildE2eOptionsFromCli,
97
+ buildExternalDataSourceValidationRequest
98
+ };
@@ -105,6 +105,9 @@ function handlePartialAuthentication(authentication) {
105
105
  */
106
106
  function transformFlatStructure(variables, appName) {
107
107
  const result = buildBaseResult(variables, appName);
108
+ if (result.frontDoorRouting) {
109
+ result.frontDoorRouting = normalizeFrontDoorRoutingForValidation(result.frontDoorRouting);
110
+ }
108
111
 
109
112
  // Sanitize authentication if present
110
113
  if (result.authentication) {
@@ -190,9 +193,6 @@ function validateBuildConfig(build) {
190
193
  if (build.dockerfile && build.dockerfile.trim() !== '') {
191
194
  buildConfig.dockerfile = build.dockerfile;
192
195
  }
193
- if (build.localPort !== undefined && build.localPort !== null) {
194
- buildConfig.localPort = build.localPort;
195
- }
196
196
 
197
197
  return Object.keys(buildConfig).length > 0 ? buildConfig : null;
198
198
  }
@@ -256,6 +256,23 @@ function transformConfigSections(variables, transformed) {
256
256
  }
257
257
  }
258
258
 
259
+ /**
260
+ * Clone frontDoorRouting for schema validation: JSON Schema expects `tls` as a string
261
+ * (e.g. "${TLS_ENABLED}", "true") but YAML often uses booleans (`tls: false`).
262
+ * @param {Object} fd - frontDoorRouting block from application.yaml
263
+ * @returns {Object}
264
+ */
265
+ function normalizeFrontDoorRoutingForValidation(fd) {
266
+ if (!fd || typeof fd !== 'object') {
267
+ return fd;
268
+ }
269
+ const out = { ...fd };
270
+ if (typeof out.tls === 'boolean') {
271
+ out.tls = out.tls ? 'true' : 'false';
272
+ }
273
+ return out;
274
+ }
275
+
259
276
  /**
260
277
  * Transforms simple optional fields
261
278
  * @function transformSimpleOptionalFields
@@ -273,7 +290,7 @@ function transformSimpleOptionalFields(variables, transformed) {
273
290
  transformed.scaling = variables.scaling;
274
291
  }
275
292
  if (variables.frontDoorRouting) {
276
- transformed.frontDoorRouting = variables.frontDoorRouting;
293
+ transformed.frontDoorRouting = normalizeFrontDoorRoutingForValidation(variables.frontDoorRouting);
277
294
  }
278
295
  if (variables.roles) {
279
296
  transformed.roles = variables.roles;
@@ -251,6 +251,35 @@ function collectBlockScalarLines(lines, startIndex, keyIndentLen) {
251
251
  return { value: parts.join(' '), endIndex: i };
252
252
  }
253
253
 
254
+ /**
255
+ * Processes a single block-scalar key line and its continuation lines.
256
+ * @param {string} line - Line matching BLOCK_SCALAR_PATTERN
257
+ * @param {string[]} lines - All lines
258
+ * @param {number} lineIndex - Index of current line
259
+ * @param {string} encryptionKey - Encryption key
260
+ * @param {Object} stats - Mutable stats object
261
+ * @returns {{ resultLine: string, nextIndex: number }|null} Result or null if not a block scalar
262
+ */
263
+ function processBlockScalarLine(line, lines, lineIndex, encryptionKey, stats) {
264
+ const blockMatch = line.match(BLOCK_SCALAR_PATTERN);
265
+ if (!blockMatch) {
266
+ return null;
267
+ }
268
+ const [, indent, key, blockIndicator, trailingWhitespace, comment] = blockMatch;
269
+ const keyIndentLen = indent.length;
270
+ const folded = blockIndicator.startsWith('>');
271
+ const { value: rawValue, endIndex } = collectBlockScalarLines(lines, lineIndex + 1, keyIndentLen);
272
+ const value = folded ? rawValue.replace(/\s+/g, ' ').trim() : rawValue;
273
+ stats.total++;
274
+ const needEncrypt = shouldEncryptValue(value);
275
+ const outValue = needEncrypt ? encryptSecret(value, encryptionKey) : value;
276
+ if (needEncrypt) {
277
+ stats.encrypted++;
278
+ }
279
+ const resultLine = `${indent}${key}: ${outValue}${trailingWhitespace || ''}${comment || ''}`;
280
+ return { resultLine, nextIndex: endIndex };
281
+ }
282
+
254
283
  function encryptYamlValues(content, encryptionKey) {
255
284
  const lines = content.split(/\r?\n/);
256
285
  const encryptedLines = [];
@@ -267,20 +296,10 @@ function encryptYamlValues(content, encryptionKey) {
267
296
  continue;
268
297
  }
269
298
 
270
- const blockMatch = line.match(BLOCK_SCALAR_PATTERN);
271
- if (blockMatch) {
272
- const [, indent, key, blockIndicator, trailingWhitespace, comment] = blockMatch;
273
- const keyIndentLen = indent.length;
274
- const folded = blockIndicator.startsWith('>');
275
- const { value: rawValue, endIndex } = collectBlockScalarLines(lines, i + 1, keyIndentLen);
276
- const value = folded ? rawValue.replace(/\s+/g, ' ').trim() : rawValue;
277
- stats.total++;
278
- const outValue = shouldEncryptValue(value) ? encryptSecret(value, encryptionKey) : value;
279
- if (shouldEncryptValue(value)) {
280
- stats.encrypted++;
281
- }
282
- encryptedLines.push(`${indent}${key}: ${outValue}${trailingWhitespace || ''}${comment || ''}`);
283
- i = endIndex;
299
+ const blockResult = processBlockScalarLine(line, lines, i, encryptionKey, stats);
300
+ if (blockResult) {
301
+ encryptedLines.push(blockResult.resultLine);
302
+ i = blockResult.nextIndex;
284
303
  continue;
285
304
  }
286
305