@depup/aws-cdk 2.1117.0-depup.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 (308) hide show
  1. package/LICENSE +202 -0
  2. package/NOTICE +16 -0
  3. package/README.md +25 -0
  4. package/THIRD_PARTY_LICENSES +24866 -0
  5. package/bin/cdk +6 -0
  6. package/build-info.json +4 -0
  7. package/changes.json +5 -0
  8. package/db.json.gz +0 -0
  9. package/docs/deploy-architecture.md +194 -0
  10. package/lib/api/aws-auth.d.ts +3 -0
  11. package/lib/api/aws-auth.js +21 -0
  12. package/lib/api/bootstrap/bootstrap-template.yaml +855 -0
  13. package/lib/api/bootstrap.d.ts +1 -0
  14. package/lib/api/bootstrap.js +19 -0
  15. package/lib/api/cloud-assembly.d.ts +5 -0
  16. package/lib/api/cloud-assembly.js +23 -0
  17. package/lib/api/cloudformation.d.ts +1 -0
  18. package/lib/api/cloudformation.js +19 -0
  19. package/lib/api/context.d.ts +1 -0
  20. package/lib/api/context.js +19 -0
  21. package/lib/api/deployments.d.ts +1 -0
  22. package/lib/api/deployments.js +19 -0
  23. package/lib/api/hotswap.d.ts +1 -0
  24. package/lib/api/hotswap.js +19 -0
  25. package/lib/api/index.d.ts +16 -0
  26. package/lib/api/index.js +38 -0
  27. package/lib/api/network-detector.d.ts +1 -0
  28. package/lib/api/network-detector.js +19 -0
  29. package/lib/api/notices.d.ts +1 -0
  30. package/lib/api/notices.js +19 -0
  31. package/lib/api/plugin.d.ts +1 -0
  32. package/lib/api/plugin.js +19 -0
  33. package/lib/api/refactor.d.ts +1 -0
  34. package/lib/api/refactor.js +8 -0
  35. package/lib/api/settings.d.ts +1 -0
  36. package/lib/api/settings.js +19 -0
  37. package/lib/api/tags.d.ts +1 -0
  38. package/lib/api/tags.js +19 -0
  39. package/lib/api-private.d.ts +9 -0
  40. package/lib/api-private.js +29 -0
  41. package/lib/cli/cdk-toolkit.d.ts +699 -0
  42. package/lib/cli/cdk-toolkit.js +1260 -0
  43. package/lib/cli/ci-systems.d.ts +29 -0
  44. package/lib/cli/ci-systems.js +61 -0
  45. package/lib/cli/cli-config.d.ts +10 -0
  46. package/lib/cli/cli-config.js +556 -0
  47. package/lib/cli/cli-type-registry.json +1147 -0
  48. package/lib/cli/cli.d.ts +3 -0
  49. package/lib/cli/cli.js +746 -0
  50. package/lib/cli/convert-to-user-input.d.ts +3 -0
  51. package/lib/cli/convert-to-user-input.js +560 -0
  52. package/lib/cli/display-version.d.ts +11 -0
  53. package/lib/cli/display-version.js +101 -0
  54. package/lib/cli/io-host/cli-io-host.d.ts +191 -0
  55. package/lib/cli/io-host/cli-io-host.js +477 -0
  56. package/lib/cli/io-host/index.d.ts +1 -0
  57. package/lib/cli/io-host/index.js +18 -0
  58. package/lib/cli/parse-command-line-arguments.d.ts +1 -0
  59. package/lib/cli/parse-command-line-arguments.js +1067 -0
  60. package/lib/cli/platform-warnings.d.ts +3 -0
  61. package/lib/cli/platform-warnings.js +44 -0
  62. package/lib/cli/pretty-print-error.d.ts +1 -0
  63. package/lib/cli/pretty-print-error.js +37 -0
  64. package/lib/cli/proxy-agent.d.ts +30 -0
  65. package/lib/cli/proxy-agent.js +52 -0
  66. package/lib/cli/root-dir.d.ts +10 -0
  67. package/lib/cli/root-dir.js +23 -0
  68. package/lib/cli/singleton-plugin-host.d.ts +7 -0
  69. package/lib/cli/singleton-plugin-host.js +11 -0
  70. package/lib/cli/tables.d.ts +1 -0
  71. package/lib/cli/tables.js +10 -0
  72. package/lib/cli/telemetry/collect-telemetry.d.ts +5 -0
  73. package/lib/cli/telemetry/collect-telemetry.js +16 -0
  74. package/lib/cli/telemetry/error.d.ts +15 -0
  75. package/lib/cli/telemetry/error.js +68 -0
  76. package/lib/cli/telemetry/feature-flags.d.ts +96 -0
  77. package/lib/cli/telemetry/feature-flags.js +103 -0
  78. package/lib/cli/telemetry/installation-id.d.ts +5 -0
  79. package/lib/cli/telemetry/installation-id.js +47 -0
  80. package/lib/cli/telemetry/library-version.d.ts +2 -0
  81. package/lib/cli/telemetry/library-version.js +30 -0
  82. package/lib/cli/telemetry/messages.d.ts +43 -0
  83. package/lib/cli/telemetry/messages.js +60 -0
  84. package/lib/cli/telemetry/sanitation.d.ts +100 -0
  85. package/lib/cli/telemetry/sanitation.js +79 -0
  86. package/lib/cli/telemetry/schema.d.ts +85 -0
  87. package/lib/cli/telemetry/schema.js +3 -0
  88. package/lib/cli/telemetry/session.d.ts +67 -0
  89. package/lib/cli/telemetry/session.js +174 -0
  90. package/lib/cli/telemetry/sink/endpoint-sink.d.ts +44 -0
  91. package/lib/cli/telemetry/sink/endpoint-sink.js +105 -0
  92. package/lib/cli/telemetry/sink/file-sink.d.ts +32 -0
  93. package/lib/cli/telemetry/sink/file-sink.js +43 -0
  94. package/lib/cli/telemetry/sink/funnel.d.ts +16 -0
  95. package/lib/cli/telemetry/sink/funnel.js +29 -0
  96. package/lib/cli/telemetry/sink/io-host-sink.d.ts +27 -0
  97. package/lib/cli/telemetry/sink/io-host-sink.js +35 -0
  98. package/lib/cli/telemetry/sink/sink-interface.d.ts +18 -0
  99. package/lib/cli/telemetry/sink/sink-interface.js +3 -0
  100. package/lib/cli/user-configuration.d.ts +104 -0
  101. package/lib/cli/user-configuration.js +334 -0
  102. package/lib/cli/user-input.d.ts +1505 -0
  103. package/lib/cli/user-input.js +3 -0
  104. package/lib/cli/util/ci.d.ts +5 -0
  105. package/lib/cli/util/ci.js +11 -0
  106. package/lib/cli/util/console-formatters.d.ts +18 -0
  107. package/lib/cli/util/console-formatters.js +42 -0
  108. package/lib/cli/util/guess-agent.d.ts +7 -0
  109. package/lib/cli/util/guess-agent.js +32 -0
  110. package/lib/cli/util/npm.d.ts +4 -0
  111. package/lib/cli/util/npm.js +34 -0
  112. package/lib/cli/util/trap-errors.d.ts +6 -0
  113. package/lib/cli/util/trap-errors.js +17 -0
  114. package/lib/cli/util/yargs-helpers.d.ts +22 -0
  115. package/lib/cli/util/yargs-helpers.js +49 -0
  116. package/lib/cli/version.d.ts +3 -0
  117. package/lib/cli/version.js +22 -0
  118. package/lib/commands/context.d.ts +40 -0
  119. package/lib/commands/context.js +158 -0
  120. package/lib/commands/deploy.d.ts +13 -0
  121. package/lib/commands/deploy.js +18 -0
  122. package/lib/commands/docs.d.ts +18 -0
  123. package/lib/commands/docs.js +33 -0
  124. package/lib/commands/doctor.d.ts +4 -0
  125. package/lib/commands/doctor.js +69 -0
  126. package/lib/commands/flags/flags.d.ts +13 -0
  127. package/lib/commands/flags/flags.js +30 -0
  128. package/lib/commands/flags/interactive-handler.d.ts +16 -0
  129. package/lib/commands/flags/interactive-handler.js +71 -0
  130. package/lib/commands/flags/obsolete-flags.d.ts +1 -0
  131. package/lib/commands/flags/obsolete-flags.js +9 -0
  132. package/lib/commands/flags/operations.d.ts +80 -0
  133. package/lib/commands/flags/operations.js +467 -0
  134. package/lib/commands/flags/router.d.ts +18 -0
  135. package/lib/commands/flags/router.js +60 -0
  136. package/lib/commands/flags/types.d.ts +12 -0
  137. package/lib/commands/flags/types.js +12 -0
  138. package/lib/commands/flags/validator.d.ts +22 -0
  139. package/lib/commands/flags/validator.js +95 -0
  140. package/lib/commands/init/index.d.ts +1 -0
  141. package/lib/commands/init/index.js +18 -0
  142. package/lib/commands/init/init-hooks.d.ts +41 -0
  143. package/lib/commands/init/init-hooks.js +85 -0
  144. package/lib/commands/init/init.d.ts +139 -0
  145. package/lib/commands/init/init.js +788 -0
  146. package/lib/commands/init/os.d.ts +8 -0
  147. package/lib/commands/init/os.js +91 -0
  148. package/lib/commands/init/package-manager.d.ts +15 -0
  149. package/lib/commands/init/package-manager.js +14 -0
  150. package/lib/commands/language.d.ts +30 -0
  151. package/lib/commands/language.js +46 -0
  152. package/lib/commands/list-stacks.d.ts +21 -0
  153. package/lib/commands/list-stacks.js +28 -0
  154. package/lib/commands/migrate.d.ts +316 -0
  155. package/lib/commands/migrate.js +801 -0
  156. package/lib/context-providers.d.ts +1 -0
  157. package/lib/context-providers.js +19 -0
  158. package/lib/cxapp/cloud-assembly.d.ts +79 -0
  159. package/lib/cxapp/cloud-assembly.js +109 -0
  160. package/lib/cxapp/cloud-executable.d.ts +51 -0
  161. package/lib/cxapp/cloud-executable.js +123 -0
  162. package/lib/cxapp/environments.d.ts +8 -0
  163. package/lib/cxapp/environments.js +66 -0
  164. package/lib/cxapp/exec.d.ts +14 -0
  165. package/lib/cxapp/exec.js +133 -0
  166. package/lib/cxapp/index.d.ts +4 -0
  167. package/lib/cxapp/index.js +21 -0
  168. package/lib/index.d.ts +2 -0
  169. package/lib/index.js +364043 -0
  170. package/lib/index_bg.wasm +0 -0
  171. package/lib/init-templates/.init-version.json +1 -0
  172. package/lib/init-templates/.no-packagejson-validator +0 -0
  173. package/lib/init-templates/.recommended-feature-flags.json +86 -0
  174. package/lib/init-templates/LICENSE +16 -0
  175. package/lib/init-templates/app/csharp/.template.gitignore +342 -0
  176. package/lib/init-templates/app/csharp/README.md +14 -0
  177. package/lib/init-templates/app/csharp/cdk.template.json +15 -0
  178. package/lib/init-templates/app/csharp/src/%name.PascalCased%/%name.PascalCased%.template.csproj +20 -0
  179. package/lib/init-templates/app/csharp/src/%name.PascalCased%/%name.PascalCased%Stack.template.cs +13 -0
  180. package/lib/init-templates/app/csharp/src/%name.PascalCased%/GlobalSuppressions.cs +1 -0
  181. package/lib/init-templates/app/csharp/src/%name.PascalCased%/Program.template.cs +44 -0
  182. package/lib/init-templates/app/csharp/src/%name.PascalCased%.template.sln +18 -0
  183. package/lib/init-templates/app/fsharp/.template.gitignore +342 -0
  184. package/lib/init-templates/app/fsharp/README.md +18 -0
  185. package/lib/init-templates/app/fsharp/cdk.template.json +14 -0
  186. package/lib/init-templates/app/fsharp/src/%name.PascalCased%/%name.PascalCased%.template.fsproj +25 -0
  187. package/lib/init-templates/app/fsharp/src/%name.PascalCased%/%name.PascalCased%Stack.template.fs +8 -0
  188. package/lib/init-templates/app/fsharp/src/%name.PascalCased%/Program.template.fs +11 -0
  189. package/lib/init-templates/app/fsharp/src/%name.PascalCased%.template.sln +18 -0
  190. package/lib/init-templates/app/go/%name%.template.go +70 -0
  191. package/lib/init-templates/app/go/%name%_test.template.go +26 -0
  192. package/lib/init-templates/app/go/.template.gitignore +19 -0
  193. package/lib/init-templates/app/go/README.md +12 -0
  194. package/lib/init-templates/app/go/cdk.template.json +13 -0
  195. package/lib/init-templates/app/go/go.template.mod +9 -0
  196. package/lib/init-templates/app/info.json +4 -0
  197. package/lib/init-templates/app/java/.template.gitignore +13 -0
  198. package/lib/init-templates/app/java/README.md +18 -0
  199. package/lib/init-templates/app/java/cdk.json +13 -0
  200. package/lib/init-templates/app/java/pom.xml +60 -0
  201. package/lib/init-templates/app/java/src/main/java/com/myorg/%name.PascalCased%App.template.java +42 -0
  202. package/lib/init-templates/app/java/src/main/java/com/myorg/%name.PascalCased%Stack.template.java +24 -0
  203. package/lib/init-templates/app/java/src/test/java/com/myorg/%name.PascalCased%Test.template.java +26 -0
  204. package/lib/init-templates/app/javascript/.template.gitignore +5 -0
  205. package/lib/init-templates/app/javascript/.template.npmignore +3 -0
  206. package/lib/init-templates/app/javascript/README.md +12 -0
  207. package/lib/init-templates/app/javascript/bin/%name%.template.js +21 -0
  208. package/lib/init-templates/app/javascript/cdk.template.json +15 -0
  209. package/lib/init-templates/app/javascript/jest.config.js +4 -0
  210. package/lib/init-templates/app/javascript/lib/%name%-stack.template.js +23 -0
  211. package/lib/init-templates/app/javascript/package.json +20 -0
  212. package/lib/init-templates/app/javascript/test/%name%.test.template.js +17 -0
  213. package/lib/init-templates/app/python/%name.PythonModule%/%name.PythonModule%_stack.template.py +19 -0
  214. package/lib/init-templates/app/python/%name.PythonModule%/__init__.py +0 -0
  215. package/lib/init-templates/app/python/.template.gitignore +10 -0
  216. package/lib/init-templates/app/python/README.template.md +58 -0
  217. package/lib/init-templates/app/python/app.template.py +28 -0
  218. package/lib/init-templates/app/python/cdk.template.json +15 -0
  219. package/lib/init-templates/app/python/requirements-dev.txt +1 -0
  220. package/lib/init-templates/app/python/requirements.txt +2 -0
  221. package/lib/init-templates/app/python/source.bat +13 -0
  222. package/lib/init-templates/app/python/tests/__init__.py +0 -0
  223. package/lib/init-templates/app/python/tests/unit/__init__.py +0 -0
  224. package/lib/init-templates/app/python/tests/unit/test_%name.PythonModule%_stack.template.py +15 -0
  225. package/lib/init-templates/app/typescript/.template.gitignore +8 -0
  226. package/lib/init-templates/app/typescript/.template.npmignore +6 -0
  227. package/lib/init-templates/app/typescript/README.md +14 -0
  228. package/lib/init-templates/app/typescript/bin/%name%.template.ts +20 -0
  229. package/lib/init-templates/app/typescript/cdk.template.json +17 -0
  230. package/lib/init-templates/app/typescript/jest.config.js +9 -0
  231. package/lib/init-templates/app/typescript/lib/%name%-stack.template.ts +16 -0
  232. package/lib/init-templates/app/typescript/package.json +26 -0
  233. package/lib/init-templates/app/typescript/test/%name%.test.template.ts +17 -0
  234. package/lib/init-templates/app/typescript/tsconfig.json +32 -0
  235. package/lib/init-templates/lib/info.json +4 -0
  236. package/lib/init-templates/lib/typescript/.template.gitignore +8 -0
  237. package/lib/init-templates/lib/typescript/.template.npmignore +6 -0
  238. package/lib/init-templates/lib/typescript/README.template.md +12 -0
  239. package/lib/init-templates/lib/typescript/jest.config.js +9 -0
  240. package/lib/init-templates/lib/typescript/lib/index.template.ts +21 -0
  241. package/lib/init-templates/lib/typescript/package.json +24 -0
  242. package/lib/init-templates/lib/typescript/test/%name%.test.template.ts +18 -0
  243. package/lib/init-templates/lib/typescript/tsconfig.json +32 -0
  244. package/lib/init-templates/sample-app/csharp/.template.gitignore +342 -0
  245. package/lib/init-templates/sample-app/csharp/README.template.md +19 -0
  246. package/lib/init-templates/sample-app/csharp/cdk.template.json +15 -0
  247. package/lib/init-templates/sample-app/csharp/src/%name.PascalCased%/%name.PascalCased%.template.csproj +20 -0
  248. package/lib/init-templates/sample-app/csharp/src/%name.PascalCased%/%name.PascalCased%Stack.template.cs +24 -0
  249. package/lib/init-templates/sample-app/csharp/src/%name.PascalCased%/GlobalSuppressions.cs +1 -0
  250. package/lib/init-templates/sample-app/csharp/src/%name.PascalCased%/Program.template.cs +15 -0
  251. package/lib/init-templates/sample-app/csharp/src/%name.PascalCased%.template.sln +18 -0
  252. package/lib/init-templates/sample-app/fsharp/.template.gitignore +342 -0
  253. package/lib/init-templates/sample-app/fsharp/README.template.md +20 -0
  254. package/lib/init-templates/sample-app/fsharp/cdk.template.json +14 -0
  255. package/lib/init-templates/sample-app/fsharp/src/%name.PascalCased%/%name.PascalCased%.template.fsproj +25 -0
  256. package/lib/init-templates/sample-app/fsharp/src/%name.PascalCased%/%name.PascalCased%Stack.template.fs +14 -0
  257. package/lib/init-templates/sample-app/fsharp/src/%name.PascalCased%/Program.template.fs +11 -0
  258. package/lib/init-templates/sample-app/fsharp/src/%name.PascalCased%.template.sln +18 -0
  259. package/lib/init-templates/sample-app/go/%name%.template.go +73 -0
  260. package/lib/init-templates/sample-app/go/%name%_test.template.go +25 -0
  261. package/lib/init-templates/sample-app/go/.template.gitignore +19 -0
  262. package/lib/init-templates/sample-app/go/README.md +12 -0
  263. package/lib/init-templates/sample-app/go/cdk.template.json +13 -0
  264. package/lib/init-templates/sample-app/go/go.template.mod +9 -0
  265. package/lib/init-templates/sample-app/info.json +4 -0
  266. package/lib/init-templates/sample-app/java/.template.gitignore +13 -0
  267. package/lib/init-templates/sample-app/java/README.template.md +19 -0
  268. package/lib/init-templates/sample-app/java/cdk.json +13 -0
  269. package/lib/init-templates/sample-app/java/pom.xml +55 -0
  270. package/lib/init-templates/sample-app/java/src/main/java/com/myorg/%name.PascalCased%App.template.java +13 -0
  271. package/lib/init-templates/sample-app/java/src/main/java/com/myorg/%name.PascalCased%Stack.template.java +29 -0
  272. package/lib/init-templates/sample-app/java/src/test/java/com/myorg/%name.PascalCased%StackTest.template.java +27 -0
  273. package/lib/init-templates/sample-app/javascript/.template.gitignore +5 -0
  274. package/lib/init-templates/sample-app/javascript/.template.npmignore +3 -0
  275. package/lib/init-templates/sample-app/javascript/README.template.md +13 -0
  276. package/lib/init-templates/sample-app/javascript/bin/%name%.template.js +6 -0
  277. package/lib/init-templates/sample-app/javascript/cdk.template.json +15 -0
  278. package/lib/init-templates/sample-app/javascript/jest.config.js +4 -0
  279. package/lib/init-templates/sample-app/javascript/lib/%name%-stack.template.js +25 -0
  280. package/lib/init-templates/sample-app/javascript/package.json +20 -0
  281. package/lib/init-templates/sample-app/javascript/test/%name%.test.template.js +16 -0
  282. package/lib/init-templates/sample-app/javascript/tsconfig.json +34 -0
  283. package/lib/init-templates/sample-app/python/%name.PythonModule%/%name.PythonModule%_stack.template.py +26 -0
  284. package/lib/init-templates/sample-app/python/%name.PythonModule%/__init__.py +0 -0
  285. package/lib/init-templates/sample-app/python/.template.gitignore +22 -0
  286. package/lib/init-templates/sample-app/python/README.template.md +65 -0
  287. package/lib/init-templates/sample-app/python/app.template.py +11 -0
  288. package/lib/init-templates/sample-app/python/cdk.template.json +15 -0
  289. package/lib/init-templates/sample-app/python/requirements-dev.txt +1 -0
  290. package/lib/init-templates/sample-app/python/requirements.txt +2 -0
  291. package/lib/init-templates/sample-app/python/source.bat +13 -0
  292. package/lib/init-templates/sample-app/python/tests/__init__.py +0 -0
  293. package/lib/init-templates/sample-app/python/tests/unit/__init__.py +0 -0
  294. package/lib/init-templates/sample-app/python/tests/unit/test_%name.PythonModule%_stack.template.py +21 -0
  295. package/lib/init-templates/sample-app/typescript/.template.gitignore +8 -0
  296. package/lib/init-templates/sample-app/typescript/.template.npmignore +6 -0
  297. package/lib/init-templates/sample-app/typescript/README.template.md +15 -0
  298. package/lib/init-templates/sample-app/typescript/bin/%name%.template.ts +6 -0
  299. package/lib/init-templates/sample-app/typescript/cdk.template.json +17 -0
  300. package/lib/init-templates/sample-app/typescript/jest.config.js +9 -0
  301. package/lib/init-templates/sample-app/typescript/lib/%name%-stack.template.ts +19 -0
  302. package/lib/init-templates/sample-app/typescript/package.json +26 -0
  303. package/lib/init-templates/sample-app/typescript/test/%name%.test.template.ts +17 -0
  304. package/lib/init-templates/sample-app/typescript/tsconfig.json +32 -0
  305. package/lib/util.d.ts +1 -0
  306. package/lib/util.js +19 -0
  307. package/package.json +170 -0
  308. package/release.txt +2 -0
@@ -0,0 +1,467 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.FlagOperations = void 0;
4
+ exports.isEffectiveValueEqualToRecommended = isEffectiveValueEqualToRecommended;
5
+ const os = require("os");
6
+ const path = require("path");
7
+ const cloudformation_diff_1 = require("@aws-cdk/cloudformation-diff");
8
+ const toolkit_lib_1 = require("@aws-cdk/toolkit-lib");
9
+ const chalk = require("chalk");
10
+ const fs = require("fs-extra");
11
+ const p_queue_1 = require("p-queue");
12
+ const obsolete_flags_1 = require("./obsolete-flags");
13
+ const api_1 = require("../../api");
14
+ class FlagOperations {
15
+ /**
16
+ * Returns only those feature flags that need configuration
17
+ *
18
+ * That is those flags:
19
+ * - That are unconfigured
20
+ * - That are not obsolete
21
+ * - Whose default value is different from the recommended value
22
+ *
23
+ * The default value being equal to the recommended value sounds odd, but
24
+ * crops up in a number of situtations:
25
+ *
26
+ * - Security-related fixes that we want to force on people, but want to
27
+ * give them a flag to back out of the changes if they really need to.
28
+ * - Flags that changed their default value in the most recent major
29
+ * version.
30
+ * - Flags that we've introduced at some point in the past, but have gone
31
+ * back on.
32
+ */
33
+ static filterNeedsAttention(flags) {
34
+ return flags
35
+ .filter(flag => !obsolete_flags_1.OBSOLETE_FLAGS.includes(flag.name))
36
+ .filter(flag => flag.userValue === undefined)
37
+ .filter(flag => defaultValue(flag) !== flag.recommendedValue);
38
+ }
39
+ constructor(flags, toolkit, ioHelper, cliContextValues = {}) {
40
+ this.flags = flags;
41
+ this.toolkit = toolkit;
42
+ this.ioHelper = ioHelper;
43
+ this.cliContextValues = cliContextValues;
44
+ this.app = '';
45
+ this.baseContextValues = {};
46
+ this.allStacks = [];
47
+ this.queue = new p_queue_1.default({ concurrency: 4 });
48
+ }
49
+ /** Main entry point that routes to either flag setting or display operations */
50
+ async execute(params) {
51
+ if (params.set) {
52
+ if (params.FLAGNAME && params.value) {
53
+ await this.setFlag(params);
54
+ }
55
+ else {
56
+ await this.setMultipleFlags(params);
57
+ }
58
+ }
59
+ else {
60
+ await this.displayFlags(params);
61
+ }
62
+ }
63
+ /** Sets a single specific flag with validation and user confirmation */
64
+ async setFlag(params) {
65
+ const flagName = params.FLAGNAME[0];
66
+ const flag = this.flags.find(f => f.name === flagName);
67
+ if (!flag) {
68
+ await this.ioHelper.defaults.error('Flag not found.');
69
+ return;
70
+ }
71
+ if (!this.isBooleanFlag(flag)) {
72
+ await this.ioHelper.defaults.error(`Flag '${flagName}' is not a boolean flag. Only boolean flags are currently supported.`);
73
+ return;
74
+ }
75
+ const prototypeSuccess = await this.prototypeChanges([flagName], params);
76
+ if (prototypeSuccess) {
77
+ await this.handleUserResponse([flagName], params);
78
+ }
79
+ }
80
+ /** Sets multiple flags (all or unconfigured) with validation and user confirmation */
81
+ async setMultipleFlags(params) {
82
+ if (params.default && !this.flags.some(f => f.unconfiguredBehavesLike)) {
83
+ await this.ioHelper.defaults.error('The --default options are not compatible with the AWS CDK library used by your application. Please upgrade to 2.212.0 or above.');
84
+ return;
85
+ }
86
+ const flagsToSet = this.getFlagsToSet(params);
87
+ const prototypeSuccess = await this.prototypeChanges(flagsToSet, params);
88
+ if (prototypeSuccess) {
89
+ await this.handleUserResponse(flagsToSet, params);
90
+ }
91
+ }
92
+ /** Determines which flags should be set based on the provided parameters */
93
+ getFlagsToSet(params) {
94
+ if (params.all && params.default) {
95
+ return this.flags
96
+ .filter(flag => this.isBooleanFlag(flag))
97
+ .map(flag => flag.name);
98
+ }
99
+ else if (params.all) {
100
+ return this.flags
101
+ .filter(flag => flag.userValue === undefined || !isEffectiveValueEqualToRecommended(flag))
102
+ .filter(flag => this.isBooleanFlag(flag))
103
+ .map(flag => flag.name);
104
+ }
105
+ else {
106
+ return this.flags
107
+ .filter(flag => flag.userValue === undefined)
108
+ .filter(flag => this.isBooleanFlag(flag))
109
+ .map(flag => flag.name);
110
+ }
111
+ }
112
+ /** Sets flags that don't cause template changes */
113
+ async setSafeFlags(params) {
114
+ const cdkJson = await JSON.parse(await fs.readFile(path.join(process.cwd(), 'cdk.json'), 'utf-8'));
115
+ this.app = params.app || cdkJson.app;
116
+ const isUsingTsNode = this.app.includes('ts-node');
117
+ if (isUsingTsNode && !this.app.includes('-T') && !this.app.includes('--transpileOnly')) {
118
+ await this.ioHelper.defaults.info('Repeated synths with ts-node will type-check the application on every synth. Add --transpileOnly to cdk.json\'s "app" command to make this operation faster.');
119
+ }
120
+ const unconfiguredFlags = this.flags.filter(flag => flag.userValue === undefined && this.isBooleanFlag(flag));
121
+ if (unconfiguredFlags.length === 0) {
122
+ await this.ioHelper.defaults.info('All feature flags are configured.');
123
+ return;
124
+ }
125
+ await this.initializeSafetyCheck();
126
+ const safeFlags = await this.batchTestFlags(unconfiguredFlags);
127
+ await this.cleanupSafetyCheck();
128
+ if (safeFlags.length > 0) {
129
+ await this.ioHelper.defaults.info('Flags that can be set without template changes:');
130
+ for (const flag of safeFlags) {
131
+ await this.ioHelper.defaults.info(`- ${flag.name} -> ${flag.recommendedValue}`);
132
+ }
133
+ await this.handleUserResponse(safeFlags.map(flag => flag.name), { ...params, recommended: true });
134
+ }
135
+ else {
136
+ await this.ioHelper.defaults.info('No more flags can be set without causing template changes.');
137
+ }
138
+ }
139
+ /** Initializes the safety check by reading context and synthesizing baseline templates */
140
+ async initializeSafetyCheck() {
141
+ const baseContext = new toolkit_lib_1.CdkAppMultiContext(process.cwd());
142
+ this.baseContextValues = { ...await baseContext.read(), ...this.cliContextValues };
143
+ this.baselineTempDir = fs.mkdtempSync(path.join(os.tmpdir(), 'cdk-baseline-'));
144
+ const mergedContext = new toolkit_lib_1.MemoryContext(this.baseContextValues);
145
+ const baseSource = await this.toolkit.fromCdkApp(this.app, {
146
+ contextStore: mergedContext,
147
+ outdir: this.baselineTempDir,
148
+ });
149
+ const baseCx = await this.toolkit.synth(baseSource);
150
+ const baseAssembly = baseCx.cloudAssembly;
151
+ this.allStacks = baseAssembly.stacksRecursively;
152
+ this.queue = new p_queue_1.default({ concurrency: 4 });
153
+ }
154
+ /** Cleans up temporary directories created during safety checks */
155
+ async cleanupSafetyCheck() {
156
+ if (this.baselineTempDir) {
157
+ await fs.remove(this.baselineTempDir);
158
+ this.baselineTempDir = undefined;
159
+ }
160
+ }
161
+ /** Tests multiple flags together and isolates unsafe ones using binary search */
162
+ async batchTestFlags(flags) {
163
+ if (flags.length === 0)
164
+ return [];
165
+ const allFlagsContext = { ...this.baseContextValues };
166
+ flags.forEach(flag => {
167
+ allFlagsContext[flag.name] = flag.recommendedValue;
168
+ });
169
+ const allSafe = await this.testBatch(allFlagsContext);
170
+ if (allSafe)
171
+ return flags;
172
+ return this.isolateUnsafeFlags(flags);
173
+ }
174
+ /** Tests if a set of context values causes template changes by synthesizing and diffing */
175
+ async testBatch(contextValues) {
176
+ const testContext = new toolkit_lib_1.MemoryContext(contextValues);
177
+ const tempDir = fs.mkdtempSync(path.join(os.tmpdir(), 'cdk-test-'));
178
+ const testSource = await this.toolkit.fromCdkApp(this.app, {
179
+ contextStore: testContext,
180
+ outdir: tempDir,
181
+ });
182
+ const testCx = await this.toolkit.synth(testSource);
183
+ try {
184
+ for (const stack of this.allStacks) {
185
+ const templatePath = stack.templateFullPath;
186
+ const diff = await this.toolkit.diff(testCx, {
187
+ method: toolkit_lib_1.DiffMethod.LocalFile(templatePath),
188
+ stacks: {
189
+ strategy: api_1.StackSelectionStrategy.PATTERN_MUST_MATCH_SINGLE,
190
+ patterns: [stack.hierarchicalId],
191
+ },
192
+ });
193
+ for (const stackDiff of Object.values(diff)) {
194
+ if (stackDiff.differenceCount > 0) {
195
+ return false;
196
+ }
197
+ }
198
+ }
199
+ return true;
200
+ }
201
+ finally {
202
+ await fs.remove(tempDir);
203
+ }
204
+ }
205
+ /** Uses binary search to isolate which flags are safe to set without template changes */
206
+ async isolateUnsafeFlags(flags) {
207
+ const safeFlags = [];
208
+ const processBatch = async (batch, contextValues) => {
209
+ if (batch.length === 1) {
210
+ const isSafe = await this.testBatch({ ...contextValues, [batch[0].name]: batch[0].recommendedValue });
211
+ if (isSafe)
212
+ safeFlags.push(batch[0]);
213
+ return;
214
+ }
215
+ const batchContext = { ...contextValues };
216
+ batch.forEach(flag => {
217
+ batchContext[flag.name] = flag.recommendedValue;
218
+ });
219
+ const isSafeBatch = await this.testBatch(batchContext);
220
+ if (isSafeBatch) {
221
+ safeFlags.push(...batch);
222
+ return;
223
+ }
224
+ const mid = Math.floor(batch.length / 2);
225
+ const left = batch.slice(0, mid);
226
+ const right = batch.slice(mid);
227
+ void this.queue.add(() => processBatch(left, contextValues));
228
+ void this.queue.add(() => processBatch(right, contextValues));
229
+ };
230
+ void this.queue.add(() => processBatch(flags, this.baseContextValues));
231
+ await this.queue.onIdle();
232
+ return safeFlags;
233
+ }
234
+ /** Prototypes flag changes by synthesizing templates and showing diffs to the user */
235
+ async prototypeChanges(flagNames, params) {
236
+ const baseContext = new toolkit_lib_1.CdkAppMultiContext(process.cwd());
237
+ const baseContextValues = { ...await baseContext.read(), ...this.cliContextValues };
238
+ const memoryContext = new toolkit_lib_1.MemoryContext(baseContextValues);
239
+ const cdkJson = await JSON.parse(await fs.readFile(path.join(process.cwd(), 'cdk.json'), 'utf-8'));
240
+ const app = cdkJson.app;
241
+ const source = await this.toolkit.fromCdkApp(app, {
242
+ contextStore: memoryContext,
243
+ outdir: fs.mkdtempSync(path.join(os.tmpdir(), 'cdk-original-')),
244
+ });
245
+ const updateObj = await this.buildUpdateObject(flagNames, params, baseContextValues);
246
+ if (!updateObj)
247
+ return false;
248
+ await memoryContext.update(updateObj);
249
+ const cx = await this.toolkit.synth(source);
250
+ const assembly = cx.cloudAssembly;
251
+ const modifiedSource = await this.toolkit.fromCdkApp(app, {
252
+ contextStore: memoryContext,
253
+ outdir: fs.mkdtempSync(path.join(os.tmpdir(), 'cdk-temp-')),
254
+ });
255
+ const modifiedCx = await this.toolkit.synth(modifiedSource);
256
+ const allStacks = assembly.stacksRecursively;
257
+ for (const stack of allStacks) {
258
+ const templatePath = stack.templateFullPath;
259
+ await this.toolkit.diff(modifiedCx, {
260
+ method: toolkit_lib_1.DiffMethod.LocalFile(templatePath),
261
+ stacks: {
262
+ strategy: api_1.StackSelectionStrategy.PATTERN_MUST_MATCH_SINGLE,
263
+ patterns: [stack.hierarchicalId],
264
+ },
265
+ });
266
+ }
267
+ await this.displayFlagChanges(updateObj, baseContextValues);
268
+ return true;
269
+ }
270
+ /** Displays a summary of flag changes showing old and new values */
271
+ async displayFlagChanges(updateObj, baseContextValues) {
272
+ await this.ioHelper.defaults.info('\nFlag changes:');
273
+ for (const [flagName, newValue] of Object.entries(updateObj)) {
274
+ const currentValue = baseContextValues[flagName];
275
+ const currentDisplay = currentValue === undefined ? '<unset>' : String(currentValue);
276
+ await this.ioHelper.defaults.info(` ${flagName}: ${currentDisplay} → ${newValue}`);
277
+ }
278
+ }
279
+ /** Builds the update object with new flag values based on parameters and current context */
280
+ async buildUpdateObject(flagNames, params, baseContextValues) {
281
+ const updateObj = {};
282
+ if (flagNames.length === 1 && params.value !== undefined) {
283
+ const flagName = flagNames[0];
284
+ const boolValue = params.value === 'true';
285
+ if (baseContextValues[flagName] === boolValue) {
286
+ await this.ioHelper.defaults.info('Flag is already set to the specified value. No changes needed.');
287
+ return null;
288
+ }
289
+ updateObj[flagName] = boolValue;
290
+ }
291
+ else {
292
+ for (const flagName of flagNames) {
293
+ const flag = this.flags.find(f => f.name === flagName);
294
+ if (!flag) {
295
+ await this.ioHelper.defaults.error(`Flag ${flagName} not found.`);
296
+ return null;
297
+ }
298
+ const newValue = params.recommended
299
+ ? flag.recommendedValue
300
+ : String(defaultValue(flag)) === 'true';
301
+ updateObj[flagName] = newValue;
302
+ }
303
+ }
304
+ return updateObj;
305
+ }
306
+ /** Prompts user for confirmation and applies changes if accepted */
307
+ async handleUserResponse(flagNames, params) {
308
+ const userAccepted = await this.ioHelper.requestResponse({
309
+ time: new Date(),
310
+ level: 'info',
311
+ code: 'CDK_TOOLKIT_I9300',
312
+ message: 'Do you want to accept these changes?',
313
+ data: {
314
+ flagNames,
315
+ responseDescription: 'Enter "y" to apply changes or "n" to cancel',
316
+ },
317
+ defaultResponse: false,
318
+ });
319
+ if (userAccepted) {
320
+ await this.modifyValues(flagNames, params);
321
+ await this.ioHelper.defaults.info('Flag value(s) updated successfully.');
322
+ }
323
+ else {
324
+ await this.ioHelper.defaults.info('Operation cancelled');
325
+ }
326
+ await this.cleanupTempDirectories();
327
+ }
328
+ /** Removes temporary directories created during flag operations */
329
+ async cleanupTempDirectories() {
330
+ const originalDir = path.join(process.cwd(), 'original');
331
+ const tempDir = path.join(process.cwd(), 'temp');
332
+ await fs.remove(originalDir);
333
+ await fs.remove(tempDir);
334
+ }
335
+ /** Actually modifies the cdk.json file with the new flag values */
336
+ async modifyValues(flagNames, params) {
337
+ const cdkJsonPath = path.join(process.cwd(), 'cdk.json');
338
+ const cdkJsonContent = await fs.readFile(cdkJsonPath, 'utf-8');
339
+ const cdkJson = JSON.parse(cdkJsonContent);
340
+ if (flagNames.length === 1 && !params.safe && !params.all) {
341
+ const boolValue = params.value === 'true';
342
+ cdkJson.context[String(flagNames[0])] = boolValue;
343
+ await this.ioHelper.defaults.info(`Setting flag '${flagNames}' to: ${boolValue}`);
344
+ }
345
+ else {
346
+ for (const flagName of flagNames) {
347
+ const flag = this.flags.find(f => f.name === flagName);
348
+ const newValue = params.recommended || params.safe
349
+ ? flag.recommendedValue
350
+ : String(defaultValue(flag)) === 'true';
351
+ cdkJson.context[flagName] = newValue;
352
+ }
353
+ }
354
+ await fs.writeFile(cdkJsonPath, JSON.stringify(cdkJson, null, 2), 'utf-8');
355
+ }
356
+ /** Displays flags in a table format, either specific flags or filtered by criteria */
357
+ async displayFlags(params) {
358
+ const { FLAGNAME, all } = params;
359
+ if (FLAGNAME && FLAGNAME.length > 0) {
360
+ await this.displaySpecificFlags(FLAGNAME);
361
+ return;
362
+ }
363
+ const [flagsToDisplay, header] = all
364
+ ? [this.flags, 'All feature flags']
365
+ : [FlagOperations.filterNeedsAttention(this.flags), 'Unconfigured feature flags'];
366
+ await this.ioHelper.defaults.info(header);
367
+ await this.displayFlagTable(flagsToDisplay);
368
+ // Add helpful message after empty table when not using --all
369
+ if (!all && flagsToDisplay.length === 0) {
370
+ await this.ioHelper.defaults.info('');
371
+ await this.ioHelper.defaults.info('✅ All feature flags are already set to their recommended values.');
372
+ await this.ioHelper.defaults.info('Use \'cdk flags --all --unstable=flags\' to see all flags and their current values.');
373
+ }
374
+ }
375
+ /** Displays detailed information for specific flags matching the given names */
376
+ async displaySpecificFlags(flagNames) {
377
+ const matchingFlags = this.flags.filter(f => flagNames.some(searchTerm => f.name.toLowerCase().includes(searchTerm.toLowerCase())));
378
+ if (matchingFlags.length === 0) {
379
+ await this.ioHelper.defaults.error(`Flag matching "${flagNames.join(', ')}" not found.`);
380
+ return;
381
+ }
382
+ if (matchingFlags.length === 1) {
383
+ const flag = matchingFlags[0];
384
+ await this.ioHelper.defaults.info(`Flag name: ${flag.name}`);
385
+ await this.ioHelper.defaults.info(`Description: ${flag.explanation}`);
386
+ await this.ioHelper.defaults.info(`Recommended value: ${flag.recommendedValue}`);
387
+ await this.ioHelper.defaults.info(`Default value: ${defaultValue(flag)}`);
388
+ await this.ioHelper.defaults.info(`User value: ${flag.userValue}`);
389
+ await this.ioHelper.defaults.info(`Effective value: ${effectiveValue(flag)}`);
390
+ return;
391
+ }
392
+ await this.ioHelper.defaults.info(`Found ${matchingFlags.length} flags matching "${flagNames.join(', ')}"`);
393
+ await this.displayFlagTable(matchingFlags);
394
+ }
395
+ /** Returns sort order for flags */
396
+ getFlagSortOrder(flag) {
397
+ if (flag.userValue === undefined)
398
+ return 3;
399
+ if (isEffectiveValueEqualToRecommended(flag))
400
+ return 1;
401
+ return 2;
402
+ }
403
+ /** Displays flags in a formatted table grouped by module and sorted */
404
+ async displayFlagTable(flags) {
405
+ const sortedFlags = [...flags].sort((a, b) => {
406
+ const orderA = this.getFlagSortOrder(a);
407
+ const orderB = this.getFlagSortOrder(b);
408
+ if (orderA !== orderB)
409
+ return orderA - orderB;
410
+ if (a.module !== b.module)
411
+ return a.module.localeCompare(b.module);
412
+ return a.name.localeCompare(b.name);
413
+ });
414
+ const rows = [['Feature Flag', 'Recommended', 'User', 'Effective']];
415
+ let currentModule = '';
416
+ sortedFlags.forEach((flag) => {
417
+ if (flag.module !== currentModule) {
418
+ rows.push([chalk.bold(`Module: ${flag.module}`), '', '', '']);
419
+ currentModule = flag.module;
420
+ }
421
+ rows.push([
422
+ ` ${flag.name}`,
423
+ String(flag.recommendedValue),
424
+ flag.userValue === undefined ? '<unset>' : String(flag.userValue),
425
+ String(effectiveValue(flag)),
426
+ ]);
427
+ });
428
+ const formattedTable = (0, cloudformation_diff_1.formatTable)(rows, undefined, true);
429
+ await this.ioHelper.defaults.info(formattedTable);
430
+ }
431
+ /** Checks if a flag has a boolean recommended value */
432
+ isBooleanFlag(flag) {
433
+ const recommended = flag.recommendedValue;
434
+ return typeof recommended === 'boolean' ||
435
+ recommended === 'true' ||
436
+ recommended === 'false';
437
+ }
438
+ /** Shows helpful usage examples and available command options */
439
+ async displayHelpMessage() {
440
+ await this.ioHelper.defaults.info('\n' + chalk.bold('Available options:'));
441
+ await this.ioHelper.defaults.info(' cdk flags --interactive # Interactive menu to manage flags');
442
+ await this.ioHelper.defaults.info(' cdk flags --all # Show all flags (including configured ones)');
443
+ await this.ioHelper.defaults.info(' cdk flags --set --all --recommended # Set all flags to recommended values');
444
+ await this.ioHelper.defaults.info(' cdk flags --set --all --default # Set all flags to default values');
445
+ await this.ioHelper.defaults.info(' cdk flags --set --unconfigured --recommended # Set unconfigured flags to recommended');
446
+ await this.ioHelper.defaults.info(' cdk flags --set <flag-name> --value <true|false> # Set specific flag');
447
+ await this.ioHelper.defaults.info(' cdk flags --safe # Safely set flags that don\'t change templates');
448
+ }
449
+ }
450
+ exports.FlagOperations = FlagOperations;
451
+ /** Checks if the flags current effective value matches the recommended value */
452
+ function isEffectiveValueEqualToRecommended(flag) {
453
+ return String(effectiveValue(flag)) === String(flag.recommendedValue);
454
+ }
455
+ /**
456
+ * Return the effective value of a flag (user value or default)
457
+ */
458
+ function effectiveValue(flag) {
459
+ return flag.userValue ?? defaultValue(flag);
460
+ }
461
+ /**
462
+ * Return the default value for a flag, assume it's `false` if not given
463
+ */
464
+ function defaultValue(flag) {
465
+ return flag.unconfiguredBehavesLike?.v2 ?? false;
466
+ }
467
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"operations.js","sourceRoot":"","sources":["operations.ts"],"names":[],"mappings":"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","sourcesContent":["import * as os from 'os';\nimport * as path from 'path';\nimport type { CloudFormationStackArtifact } from '@aws-cdk/cloud-assembly-api';\nimport { formatTable } from '@aws-cdk/cloudformation-diff';\nimport type { FeatureFlag, Toolkit } from '@aws-cdk/toolkit-lib';\nimport { CdkAppMultiContext, MemoryContext, DiffMethod } from '@aws-cdk/toolkit-lib';\nimport * as chalk from 'chalk';\nimport * as fs from 'fs-extra';\nimport PQueue from 'p-queue';\nimport { OBSOLETE_FLAGS } from './obsolete-flags';\nimport type { FlagOperationsParams } from './types';\nimport { StackSelectionStrategy } from '../../api';\nimport type { IoHelper } from '../../api-private';\n\nexport class FlagOperations {\n  /**\n   * Returns only those feature flags that need configuration\n   *\n   * That is those flags:\n   * - That are unconfigured\n   * - That are not obsolete\n   * - Whose default value is different from the recommended value\n   *\n   * The default value being equal to the recommended value sounds odd, but\n   * crops up in a number of situtations:\n   *\n   * - Security-related fixes that we want to force on people, but want to\n   *   give them a flag to back out of the changes if they really need to.\n   * - Flags that changed their default value in the most recent major\n   *   version.\n   * - Flags that we've introduced at some point in the past, but have gone\n   *   back on.\n   */\n  public static filterNeedsAttention(flags: FeatureFlag[]): FeatureFlag[] {\n    return flags\n      .filter(flag => !OBSOLETE_FLAGS.includes(flag.name))\n      .filter(flag => flag.userValue === undefined)\n      .filter(flag => defaultValue(flag) !== flag.recommendedValue);\n  }\n\n  private app: string;\n  private baseContextValues: Record<string, any>;\n  private allStacks: CloudFormationStackArtifact[];\n  private queue: PQueue;\n  private baselineTempDir?: string;\n\n  constructor(\n    private readonly flags: FeatureFlag[],\n    private readonly toolkit: Toolkit,\n    private readonly ioHelper: IoHelper,\n    private readonly cliContextValues: Record<string, any> = {},\n  ) {\n    this.app = '';\n    this.baseContextValues = {};\n    this.allStacks = [];\n    this.queue = new PQueue({ concurrency: 4 });\n  }\n\n  /** Main entry point that routes to either flag setting or display operations */\n  async execute(params: FlagOperationsParams): Promise<void> {\n    if (params.set) {\n      if (params.FLAGNAME && params.value) {\n        await this.setFlag(params);\n      } else {\n        await this.setMultipleFlags(params);\n      }\n    } else {\n      await this.displayFlags(params);\n    }\n  }\n\n  /** Sets a single specific flag with validation and user confirmation */\n  async setFlag(params: FlagOperationsParams): Promise<void> {\n    const flagName = params.FLAGNAME![0];\n    const flag = this.flags.find(f => f.name === flagName);\n\n    if (!flag) {\n      await this.ioHelper.defaults.error('Flag not found.');\n      return;\n    }\n\n    if (!this.isBooleanFlag(flag)) {\n      await this.ioHelper.defaults.error(`Flag '${flagName}' is not a boolean flag. Only boolean flags are currently supported.`);\n      return;\n    }\n\n    const prototypeSuccess = await this.prototypeChanges([flagName], params);\n    if (prototypeSuccess) {\n      await this.handleUserResponse([flagName], params);\n    }\n  }\n\n  /** Sets multiple flags (all or unconfigured) with validation and user confirmation */\n  async setMultipleFlags(params: FlagOperationsParams): Promise<void> {\n    if (params.default && !this.flags.some(f => f.unconfiguredBehavesLike)) {\n      await this.ioHelper.defaults.error('The --default options are not compatible with the AWS CDK library used by your application. Please upgrade to 2.212.0 or above.');\n      return;\n    }\n\n    const flagsToSet = this.getFlagsToSet(params);\n    const prototypeSuccess = await this.prototypeChanges(flagsToSet, params);\n\n    if (prototypeSuccess) {\n      await this.handleUserResponse(flagsToSet, params);\n    }\n  }\n\n  /** Determines which flags should be set based on the provided parameters */\n  private getFlagsToSet(params: FlagOperationsParams): string[] {\n    if (params.all && params.default) {\n      return this.flags\n        .filter(flag => this.isBooleanFlag(flag))\n        .map(flag => flag.name);\n    } else if (params.all) {\n      return this.flags\n        .filter(flag => flag.userValue === undefined || !isEffectiveValueEqualToRecommended(flag))\n        .filter(flag => this.isBooleanFlag(flag))\n        .map(flag => flag.name);\n    } else {\n      return this.flags\n        .filter(flag => flag.userValue === undefined)\n        .filter(flag => this.isBooleanFlag(flag))\n        .map(flag => flag.name);\n    }\n  }\n\n  /** Sets flags that don't cause template changes */\n  async setSafeFlags(params: FlagOperationsParams): Promise<void> {\n    const cdkJson = await JSON.parse(await fs.readFile(path.join(process.cwd(), 'cdk.json'), 'utf-8'));\n    this.app = params.app || cdkJson.app;\n\n    const isUsingTsNode = this.app.includes('ts-node');\n    if (isUsingTsNode && !this.app.includes('-T') && !this.app.includes('--transpileOnly')) {\n      await this.ioHelper.defaults.info('Repeated synths with ts-node will type-check the application on every synth. Add --transpileOnly to cdk.json\\'s \"app\" command to make this operation faster.');\n    }\n\n    const unconfiguredFlags = this.flags.filter(flag =>\n      flag.userValue === undefined && this.isBooleanFlag(flag));\n\n    if (unconfiguredFlags.length === 0) {\n      await this.ioHelper.defaults.info('All feature flags are configured.');\n      return;\n    }\n\n    await this.initializeSafetyCheck();\n    const safeFlags = await this.batchTestFlags(unconfiguredFlags);\n    await this.cleanupSafetyCheck();\n\n    if (safeFlags.length > 0) {\n      await this.ioHelper.defaults.info('Flags that can be set without template changes:');\n      for (const flag of safeFlags) {\n        await this.ioHelper.defaults.info(`- ${flag.name} -> ${flag.recommendedValue}`);\n      }\n      await this.handleUserResponse(safeFlags.map(flag => flag.name), { ...params, recommended: true });\n    } else {\n      await this.ioHelper.defaults.info('No more flags can be set without causing template changes.');\n    }\n  }\n\n  /** Initializes the safety check by reading context and synthesizing baseline templates */\n  private async initializeSafetyCheck(): Promise<void> {\n    const baseContext = new CdkAppMultiContext(process.cwd());\n    this.baseContextValues = { ...await baseContext.read(), ...this.cliContextValues };\n\n    this.baselineTempDir = fs.mkdtempSync(path.join(os.tmpdir(), 'cdk-baseline-'));\n    const mergedContext = new MemoryContext(this.baseContextValues);\n    const baseSource = await this.toolkit.fromCdkApp(this.app, {\n      contextStore: mergedContext,\n      outdir: this.baselineTempDir,\n    });\n\n    const baseCx = await this.toolkit.synth(baseSource);\n    const baseAssembly = baseCx.cloudAssembly;\n    this.allStacks = baseAssembly.stacksRecursively;\n    this.queue = new PQueue({ concurrency: 4 });\n  }\n\n  /** Cleans up temporary directories created during safety checks */\n  private async cleanupSafetyCheck(): Promise<void> {\n    if (this.baselineTempDir) {\n      await fs.remove(this.baselineTempDir);\n      this.baselineTempDir = undefined;\n    }\n  }\n\n  /** Tests multiple flags together and isolates unsafe ones using binary search */\n  private async batchTestFlags(flags: FeatureFlag[]): Promise<FeatureFlag[]> {\n    if (flags.length === 0) return [];\n\n    const allFlagsContext = { ...this.baseContextValues };\n    flags.forEach(flag => {\n      allFlagsContext[flag.name] = flag.recommendedValue;\n    });\n\n    const allSafe = await this.testBatch(allFlagsContext);\n    if (allSafe) return flags;\n\n    return this.isolateUnsafeFlags(flags);\n  }\n\n  /** Tests if a set of context values causes template changes by synthesizing and diffing */\n  private async testBatch(contextValues: Record<string, any>): Promise<boolean> {\n    const testContext = new MemoryContext(contextValues);\n    const tempDir = fs.mkdtempSync(path.join(os.tmpdir(), 'cdk-test-'));\n    const testSource = await this.toolkit.fromCdkApp(this.app, {\n      contextStore: testContext,\n      outdir: tempDir,\n    });\n\n    const testCx = await this.toolkit.synth(testSource);\n\n    try {\n      for (const stack of this.allStacks) {\n        const templatePath = stack.templateFullPath;\n        const diff = await this.toolkit.diff(testCx, {\n          method: DiffMethod.LocalFile(templatePath),\n          stacks: {\n            strategy: StackSelectionStrategy.PATTERN_MUST_MATCH_SINGLE,\n            patterns: [stack.hierarchicalId],\n          },\n        });\n\n        for (const stackDiff of Object.values(diff)) {\n          if (stackDiff.differenceCount > 0) {\n            return false;\n          }\n        }\n      }\n      return true;\n    } finally {\n      await fs.remove(tempDir);\n    }\n  }\n\n  /** Uses binary search to isolate which flags are safe to set without template changes */\n  private async isolateUnsafeFlags(flags: FeatureFlag[]): Promise<FeatureFlag[]> {\n    const safeFlags: FeatureFlag[] = [];\n\n    const processBatch = async (batch: FeatureFlag[], contextValues: Record<string, any>): Promise<void> => {\n      if (batch.length === 1) {\n        const isSafe = await this.testBatch(\n          { ...contextValues, [batch[0].name]: batch[0].recommendedValue },\n        );\n        if (isSafe) safeFlags.push(batch[0]);\n        return;\n      }\n\n      const batchContext = { ...contextValues };\n      batch.forEach(flag => {\n        batchContext[flag.name] = flag.recommendedValue;\n      });\n\n      const isSafeBatch = await this.testBatch(batchContext);\n      if (isSafeBatch) {\n        safeFlags.push(...batch);\n        return;\n      }\n\n      const mid = Math.floor(batch.length / 2);\n      const left = batch.slice(0, mid);\n      const right = batch.slice(mid);\n\n      void this.queue.add(() => processBatch(left, contextValues));\n      void this.queue.add(() => processBatch(right, contextValues));\n    };\n\n    void this.queue.add(() => processBatch(flags, this.baseContextValues));\n    await this.queue.onIdle();\n    return safeFlags;\n  }\n\n  /** Prototypes flag changes by synthesizing templates and showing diffs to the user */\n  private async prototypeChanges(flagNames: string[], params: FlagOperationsParams): Promise<boolean> {\n    const baseContext = new CdkAppMultiContext(process.cwd());\n    const baseContextValues = { ...await baseContext.read(), ...this.cliContextValues };\n    const memoryContext = new MemoryContext(baseContextValues);\n\n    const cdkJson = await JSON.parse(await fs.readFile(path.join(process.cwd(), 'cdk.json'), 'utf-8'));\n    const app = cdkJson.app;\n\n    const source = await this.toolkit.fromCdkApp(app, {\n      contextStore: memoryContext,\n      outdir: fs.mkdtempSync(path.join(os.tmpdir(), 'cdk-original-')),\n    });\n\n    const updateObj = await this.buildUpdateObject(flagNames, params, baseContextValues);\n    if (!updateObj) return false;\n\n    await memoryContext.update(updateObj);\n    const cx = await this.toolkit.synth(source);\n    const assembly = cx.cloudAssembly;\n\n    const modifiedSource = await this.toolkit.fromCdkApp(app, {\n      contextStore: memoryContext,\n      outdir: fs.mkdtempSync(path.join(os.tmpdir(), 'cdk-temp-')),\n    });\n\n    const modifiedCx = await this.toolkit.synth(modifiedSource);\n    const allStacks = assembly.stacksRecursively;\n\n    for (const stack of allStacks) {\n      const templatePath = stack.templateFullPath;\n      await this.toolkit.diff(modifiedCx, {\n        method: DiffMethod.LocalFile(templatePath),\n        stacks: {\n          strategy: StackSelectionStrategy.PATTERN_MUST_MATCH_SINGLE,\n          patterns: [stack.hierarchicalId],\n        },\n      });\n    }\n\n    await this.displayFlagChanges(updateObj, baseContextValues);\n    return true;\n  }\n\n  /** Displays a summary of flag changes showing old and new values */\n  private async displayFlagChanges(updateObj: Record<string, boolean>, baseContextValues: Record<string, any>): Promise<void> {\n    await this.ioHelper.defaults.info('\\nFlag changes:');\n    for (const [flagName, newValue] of Object.entries(updateObj)) {\n      const currentValue = baseContextValues[flagName];\n      const currentDisplay = currentValue === undefined ? '<unset>' : String(currentValue);\n      await this.ioHelper.defaults.info(`  ${flagName}: ${currentDisplay} → ${newValue}`);\n    }\n  }\n\n  /** Builds the update object with new flag values based on parameters and current context */\n  private async buildUpdateObject(flagNames: string[], params: FlagOperationsParams, baseContextValues: Record<string, any>)\n    : Promise<Record<string, boolean> | null> {\n    const updateObj: Record<string, boolean> = {};\n\n    if (flagNames.length === 1 && params.value !== undefined) {\n      const flagName = flagNames[0];\n      const boolValue = params.value === 'true';\n      if (baseContextValues[flagName] === boolValue) {\n        await this.ioHelper.defaults.info('Flag is already set to the specified value. No changes needed.');\n        return null;\n      }\n      updateObj[flagName] = boolValue;\n    } else {\n      for (const flagName of flagNames) {\n        const flag = this.flags.find(f => f.name === flagName);\n        if (!flag) {\n          await this.ioHelper.defaults.error(`Flag ${flagName} not found.`);\n          return null;\n        }\n        const newValue = params.recommended\n          ? flag.recommendedValue as boolean\n          : String(defaultValue(flag)) === 'true';\n        updateObj[flagName] = newValue;\n      }\n    }\n    return updateObj;\n  }\n\n  /** Prompts user for confirmation and applies changes if accepted */\n  private async handleUserResponse(flagNames: string[], params: FlagOperationsParams): Promise<void> {\n    const userAccepted = await this.ioHelper.requestResponse({\n      time: new Date(),\n      level: 'info',\n      code: 'CDK_TOOLKIT_I9300',\n      message: 'Do you want to accept these changes?',\n      data: {\n        flagNames,\n        responseDescription: 'Enter \"y\" to apply changes or \"n\" to cancel',\n      },\n      defaultResponse: false,\n    });\n\n    if (userAccepted) {\n      await this.modifyValues(flagNames, params);\n      await this.ioHelper.defaults.info('Flag value(s) updated successfully.');\n    } else {\n      await this.ioHelper.defaults.info('Operation cancelled');\n    }\n\n    await this.cleanupTempDirectories();\n  }\n\n  /** Removes temporary directories created during flag operations */\n  private async cleanupTempDirectories(): Promise<void> {\n    const originalDir = path.join(process.cwd(), 'original');\n    const tempDir = path.join(process.cwd(), 'temp');\n    await fs.remove(originalDir);\n    await fs.remove(tempDir);\n  }\n\n  /** Actually modifies the cdk.json file with the new flag values */\n  private async modifyValues(flagNames: string[], params: FlagOperationsParams): Promise<void> {\n    const cdkJsonPath = path.join(process.cwd(), 'cdk.json');\n    const cdkJsonContent = await fs.readFile(cdkJsonPath, 'utf-8');\n    const cdkJson = JSON.parse(cdkJsonContent);\n\n    if (flagNames.length === 1 && !params.safe && !params.all) {\n      const boolValue = params.value === 'true';\n      cdkJson.context[String(flagNames[0])] = boolValue;\n      await this.ioHelper.defaults.info(`Setting flag '${flagNames}' to: ${boolValue}`);\n    } else {\n      for (const flagName of flagNames) {\n        const flag = this.flags.find(f => f.name === flagName)!;\n        const newValue = params.recommended || params.safe\n          ? flag.recommendedValue as boolean\n          : String(defaultValue(flag)) === 'true';\n        cdkJson.context[flagName] = newValue;\n      }\n    }\n    await fs.writeFile(cdkJsonPath, JSON.stringify(cdkJson, null, 2), 'utf-8');\n  }\n\n  /** Displays flags in a table format, either specific flags or filtered by criteria */\n  async displayFlags(params: FlagOperationsParams): Promise<void> {\n    const { FLAGNAME, all } = params;\n\n    if (FLAGNAME && FLAGNAME.length > 0) {\n      await this.displaySpecificFlags(FLAGNAME);\n      return;\n    }\n\n    const [flagsToDisplay, header] = all\n      ? [this.flags, 'All feature flags']\n      : [FlagOperations.filterNeedsAttention(this.flags), 'Unconfigured feature flags'];\n\n    await this.ioHelper.defaults.info(header);\n    await this.displayFlagTable(flagsToDisplay);\n\n    // Add helpful message after empty table when not using --all\n    if (!all && flagsToDisplay.length === 0) {\n      await this.ioHelper.defaults.info('');\n      await this.ioHelper.defaults.info('✅ All feature flags are already set to their recommended values.');\n      await this.ioHelper.defaults.info('Use \\'cdk flags --all --unstable=flags\\' to see all flags and their current values.');\n    }\n  }\n\n  /** Displays detailed information for specific flags matching the given names */\n  private async displaySpecificFlags(flagNames: string[]): Promise<void> {\n    const matchingFlags = this.flags.filter(f =>\n      flagNames.some(searchTerm => f.name.toLowerCase().includes(searchTerm.toLowerCase())));\n\n    if (matchingFlags.length === 0) {\n      await this.ioHelper.defaults.error(`Flag matching \"${flagNames.join(', ')}\" not found.`);\n      return;\n    }\n\n    if (matchingFlags.length === 1) {\n      const flag = matchingFlags[0];\n      await this.ioHelper.defaults.info(`Flag name: ${flag.name}`);\n      await this.ioHelper.defaults.info(`Description: ${flag.explanation}`);\n      await this.ioHelper.defaults.info(`Recommended value: ${flag.recommendedValue}`);\n      await this.ioHelper.defaults.info(`Default value: ${defaultValue(flag)}`);\n      await this.ioHelper.defaults.info(`User value: ${flag.userValue}`);\n      await this.ioHelper.defaults.info(`Effective value: ${effectiveValue(flag)}`);\n      return;\n    }\n\n    await this.ioHelper.defaults.info(`Found ${matchingFlags.length} flags matching \"${flagNames.join(', ')}\"`);\n    await this.displayFlagTable(matchingFlags);\n  }\n\n  /** Returns sort order for flags */\n  private getFlagSortOrder(flag: FeatureFlag): number {\n    if (flag.userValue === undefined) return 3;\n    if (isEffectiveValueEqualToRecommended(flag)) return 1;\n    return 2;\n  }\n\n  /** Displays flags in a formatted table grouped by module and sorted */\n  async displayFlagTable(flags: FeatureFlag[]): Promise<void> {\n    const sortedFlags = [...flags].sort((a, b) => {\n      const orderA = this.getFlagSortOrder(a);\n      const orderB = this.getFlagSortOrder(b);\n\n      if (orderA !== orderB) return orderA - orderB;\n      if (a.module !== b.module) return a.module.localeCompare(b.module);\n      return a.name.localeCompare(b.name);\n    });\n\n    const rows: string[][] = [['Feature Flag', 'Recommended', 'User', 'Effective']];\n    let currentModule = '';\n\n    sortedFlags.forEach((flag) => {\n      if (flag.module !== currentModule) {\n        rows.push([chalk.bold(`Module: ${flag.module}`), '', '', '']);\n        currentModule = flag.module;\n      }\n      rows.push([\n        `  ${flag.name}`,\n        String(flag.recommendedValue),\n        flag.userValue === undefined ? '<unset>' : String(flag.userValue),\n        String(effectiveValue(flag)),\n      ]);\n    });\n\n    const formattedTable = formatTable(rows, undefined, true);\n    await this.ioHelper.defaults.info(formattedTable);\n  }\n\n  /** Checks if a flag has a boolean recommended value */\n  isBooleanFlag(flag: FeatureFlag): boolean {\n    const recommended = flag.recommendedValue;\n    return typeof recommended === 'boolean' ||\n      recommended === 'true' ||\n      recommended === 'false';\n  }\n\n  /** Shows helpful usage examples and available command options */\n  async displayHelpMessage(): Promise<void> {\n    await this.ioHelper.defaults.info('\\n' + chalk.bold('Available options:'));\n    await this.ioHelper.defaults.info('  cdk flags --interactive     # Interactive menu to manage flags');\n    await this.ioHelper.defaults.info('  cdk flags --all             # Show all flags (including configured ones)');\n    await this.ioHelper.defaults.info('  cdk flags --set --all --recommended    # Set all flags to recommended values');\n    await this.ioHelper.defaults.info('  cdk flags --set --all --default       # Set all flags to default values');\n    await this.ioHelper.defaults.info('  cdk flags --set --unconfigured --recommended  # Set unconfigured flags to recommended');\n    await this.ioHelper.defaults.info('  cdk flags --set <flag-name> --value <true|false>  # Set specific flag');\n    await this.ioHelper.defaults.info('  cdk flags --safe            # Safely set flags that don\\'t change templates');\n  }\n}\n\n/** Checks if the flags current effective value matches the recommended value */\nexport function isEffectiveValueEqualToRecommended(flag: FeatureFlag): boolean {\n  return String(effectiveValue(flag)) === String(flag.recommendedValue);\n}\n\n/**\n * Return the effective value of a flag (user value or default)\n */\nfunction effectiveValue(flag: FeatureFlag) {\n  return flag.userValue ?? defaultValue(flag);\n}\n\n/**\n * Return the default value for a flag, assume it's `false` if not given\n */\nfunction defaultValue(flag: FeatureFlag) {\n  return flag.unconfiguredBehavesLike?.v2 ?? false;\n}\n"]}
@@ -0,0 +1,18 @@
1
+ import type { InteractiveHandler } from './interactive-handler';
2
+ import type { FlagOperations } from './operations.ts';
3
+ import type { FlagOperationsParams } from './types';
4
+ import type { FlagValidator } from './validator';
5
+ export declare class FlagOperationRouter {
6
+ private readonly validator;
7
+ private readonly interactiveHandler;
8
+ private readonly flagOperations;
9
+ constructor(validator: FlagValidator, interactiveHandler: InteractiveHandler, flagOperations: FlagOperations);
10
+ /** Routes flag operations to appropriate handlers based on parameters */
11
+ route(params: FlagOperationsParams): Promise<void>;
12
+ /** Handles flag setting operations, routing to single or multiple flag methods */
13
+ private handleSetOperations;
14
+ /** Manages interactive mode */
15
+ private handleInteractiveMode;
16
+ /** Shows help message when no specific options are provided */
17
+ private showHelpMessage;
18
+ }