@fjall/components-infrastructure 0.88.3 → 0.89.2

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 (152) hide show
  1. package/LICENSE +21 -0
  2. package/dist/lib/app.d.ts +33 -10
  3. package/dist/lib/app.js +79 -36
  4. package/dist/lib/aspects/index.d.ts +1 -0
  5. package/dist/lib/aspects/index.js +6 -0
  6. package/dist/lib/config/aws/accountAuditRole.d.ts +20 -0
  7. package/dist/lib/config/aws/accountAuditRole.js +38 -0
  8. package/dist/lib/config/aws/accountMonitoringRole.d.ts +22 -0
  9. package/dist/lib/config/aws/accountMonitoringRole.js +133 -0
  10. package/dist/lib/config/aws/cloudTrail.d.ts +0 -3
  11. package/dist/lib/config/aws/cloudTrail.js +2 -2
  12. package/dist/lib/config/aws/disasterRecovery.js +53 -20
  13. package/dist/lib/config/aws/ecrDefaultImage.js +4 -3
  14. package/dist/lib/config/aws/index.d.ts +4 -0
  15. package/dist/lib/config/aws/index.js +5 -1
  16. package/dist/lib/config/aws/oidcConnector.d.ts +8 -0
  17. package/dist/lib/config/aws/oidcConnector.js +46 -0
  18. package/dist/lib/config/aws/platform.d.ts +2 -0
  19. package/dist/lib/config/aws/platform.js +6 -0
  20. package/dist/lib/config/index.d.ts +2 -0
  21. package/dist/lib/config/index.js +21 -0
  22. package/dist/lib/patterns/aws/account.js +22 -10
  23. package/dist/lib/patterns/aws/cdn.d.ts +19 -40
  24. package/dist/lib/patterns/aws/cdn.js +21 -17
  25. package/dist/lib/patterns/aws/compute.d.ts +9 -720
  26. package/dist/lib/patterns/aws/compute.js +27 -432
  27. package/dist/lib/patterns/aws/computeEc2.d.ts +67 -0
  28. package/dist/lib/patterns/aws/computeEc2.js +46 -0
  29. package/dist/lib/patterns/aws/computeEcs.d.ts +446 -0
  30. package/dist/lib/patterns/aws/computeEcs.js +246 -0
  31. package/dist/lib/patterns/aws/computeLambda.d.ts +220 -0
  32. package/dist/lib/patterns/aws/computeLambda.js +147 -0
  33. package/dist/lib/patterns/aws/database.d.ts +7 -87
  34. package/dist/lib/patterns/aws/database.js +15 -38
  35. package/dist/lib/patterns/aws/domainDelegation.d.ts +8 -0
  36. package/dist/lib/patterns/aws/domainDelegation.js +54 -0
  37. package/dist/lib/patterns/aws/domainFactory.d.ts +8 -0
  38. package/dist/lib/patterns/aws/domainFactory.js +23 -0
  39. package/dist/lib/patterns/aws/index.d.ts +4 -1
  40. package/dist/lib/patterns/aws/index.js +6 -2
  41. package/dist/lib/patterns/aws/interfaces/cdn.d.ts +26 -0
  42. package/dist/lib/patterns/aws/interfaces/cdn.js +14 -0
  43. package/dist/lib/patterns/aws/interfaces/connector.d.ts +4 -181
  44. package/dist/lib/patterns/aws/interfaces/connector.js +16 -113
  45. package/dist/lib/patterns/aws/interfaces/domain.d.ts +2 -0
  46. package/dist/lib/patterns/aws/interfaces/domain.js +6 -0
  47. package/dist/lib/patterns/aws/interfaces/index.d.ts +2 -0
  48. package/dist/lib/patterns/aws/interfaces/index.js +5 -2
  49. package/dist/lib/patterns/aws/interfaces/pattern.d.ts +9 -6
  50. package/dist/lib/patterns/aws/interfaces/pattern.js +1 -1
  51. package/dist/lib/patterns/aws/network.js +6 -9
  52. package/dist/lib/patterns/aws/organisation.d.ts +4 -2
  53. package/dist/lib/patterns/aws/organisation.js +21 -8
  54. package/dist/lib/patterns/aws/payload.js +21 -12
  55. package/dist/lib/patterns/aws/storage.d.ts +3 -2
  56. package/dist/lib/patterns/aws/storage.js +1 -1
  57. package/dist/lib/resources/aws/audit/auditRole.js +4 -4
  58. package/dist/lib/resources/aws/audit/index.d.ts +1 -0
  59. package/dist/lib/resources/aws/audit/index.js +6 -0
  60. package/dist/lib/resources/aws/backup/backupPlan.js +3 -2
  61. package/dist/lib/resources/aws/backup/backupVault.js +5 -3
  62. package/dist/lib/resources/aws/base/awsStack.d.ts +4 -2
  63. package/dist/lib/resources/aws/base/awsStack.js +8 -2
  64. package/dist/lib/resources/aws/cdn/cloudFront.d.ts +14 -0
  65. package/dist/lib/resources/aws/cdn/cloudFront.js +52 -18
  66. package/dist/lib/resources/aws/compute/ec2.js +18 -22
  67. package/dist/lib/resources/aws/compute/ecs.d.ts +23 -10
  68. package/dist/lib/resources/aws/compute/ecs.js +121 -64
  69. package/dist/lib/resources/aws/compute/index.d.ts +1 -0
  70. package/dist/lib/resources/aws/compute/index.js +2 -1
  71. package/dist/lib/resources/aws/compute/lambda.d.ts +0 -2
  72. package/dist/lib/resources/aws/compute/lambda.js +12 -27
  73. package/dist/lib/resources/aws/database/dynamodb.js +3 -13
  74. package/dist/lib/resources/aws/database/index.d.ts +8 -2
  75. package/dist/lib/resources/aws/database/index.js +19 -3
  76. package/dist/lib/resources/aws/database/rdsAurora.d.ts +2 -3
  77. package/dist/lib/resources/aws/database/rdsAurora.js +32 -68
  78. package/dist/lib/resources/aws/database/rdsAuroraGlobal.d.ts +6 -6
  79. package/dist/lib/resources/aws/database/rdsAuroraGlobal.js +25 -29
  80. package/dist/lib/resources/aws/database/rdsDefaults.d.ts +11 -0
  81. package/dist/lib/resources/aws/database/rdsDefaults.js +15 -0
  82. package/dist/lib/resources/aws/database/rdsHelpers.d.ts +39 -0
  83. package/dist/lib/resources/aws/database/rdsHelpers.js +75 -0
  84. package/dist/lib/resources/aws/database/rdsInstance.d.ts +7 -8
  85. package/dist/lib/resources/aws/database/rdsInstance.js +40 -84
  86. package/dist/lib/resources/aws/database/rdsProxyOutput.d.ts +7 -0
  87. package/dist/lib/resources/aws/database/rdsProxyOutput.js +18 -0
  88. package/dist/lib/resources/aws/iam/index.d.ts +0 -1
  89. package/dist/lib/resources/aws/iam/index.js +1 -2
  90. package/dist/lib/resources/aws/index.d.ts +0 -1
  91. package/dist/lib/resources/aws/index.js +1 -2
  92. package/dist/lib/resources/aws/logging/cloudTrail.js +13 -3
  93. package/dist/lib/resources/aws/logging/index.d.ts +2 -0
  94. package/dist/lib/resources/aws/logging/index.js +19 -0
  95. package/dist/lib/resources/aws/messaging/index.d.ts +3 -2
  96. package/dist/lib/resources/aws/messaging/index.js +4 -3
  97. package/dist/lib/resources/aws/messaging/sqs.js +14 -11
  98. package/dist/lib/resources/aws/messaging/utils.d.ts +1 -2
  99. package/dist/lib/resources/aws/messaging/utils.js +3 -4
  100. package/dist/lib/resources/aws/monitoring/index.d.ts +0 -1
  101. package/dist/lib/resources/aws/monitoring/index.js +4 -17
  102. package/dist/lib/resources/aws/networking/domain.d.ts +13 -0
  103. package/dist/lib/resources/aws/networking/domain.js +102 -0
  104. package/dist/lib/resources/aws/networking/domainCertificate.d.ts +13 -0
  105. package/dist/lib/resources/aws/networking/domainCertificate.js +28 -0
  106. package/dist/lib/resources/aws/networking/hostedZone.d.ts +28 -0
  107. package/dist/lib/resources/aws/networking/hostedZone.js +150 -0
  108. package/dist/lib/resources/aws/networking/index.d.ts +4 -0
  109. package/dist/lib/resources/aws/networking/index.js +5 -1
  110. package/dist/lib/resources/aws/networking/ipamPool.js +57 -31
  111. package/dist/lib/resources/aws/networking/securityGroup.d.ts +5 -0
  112. package/dist/lib/resources/aws/networking/securityGroup.js +14 -0
  113. package/dist/lib/resources/aws/networking/vpc.js +9 -4
  114. package/dist/lib/resources/aws/organisation/costAllocationTagActivator.d.ts +17 -0
  115. package/dist/lib/resources/aws/organisation/costAllocationTagActivator.js +66 -0
  116. package/dist/lib/resources/aws/organisation/index.d.ts +1 -0
  117. package/dist/lib/resources/aws/organisation/index.js +4 -2
  118. package/dist/lib/resources/aws/secrets/index.d.ts +0 -1
  119. package/dist/lib/resources/aws/secrets/index.js +1 -2
  120. package/dist/lib/resources/aws/storage/ecr.d.ts +0 -1
  121. package/dist/lib/resources/aws/storage/ecr.js +5 -5
  122. package/dist/lib/resources/aws/storage/s3.d.ts +3 -3
  123. package/dist/lib/resources/aws/storage/s3.js +1 -1
  124. package/dist/lib/resources/aws/utilities/index.d.ts +5 -0
  125. package/dist/lib/resources/aws/utilities/index.js +22 -0
  126. package/dist/lib/utils/backupTierMapping.d.ts +11 -0
  127. package/dist/lib/utils/backupTierMapping.js +17 -0
  128. package/dist/lib/utils/capitaliseString.d.ts +1 -12
  129. package/dist/lib/utils/capitaliseString.js +8 -28
  130. package/dist/lib/utils/connections.d.ts +46 -0
  131. package/dist/lib/utils/connections.js +159 -0
  132. package/dist/lib/utils/connector.d.ts +183 -0
  133. package/dist/lib/utils/connector.js +117 -0
  134. package/dist/lib/utils/databaseTypes.d.ts +85 -0
  135. package/dist/lib/utils/databaseTypes.js +34 -0
  136. package/dist/lib/utils/dnsRecords.d.ts +4 -0
  137. package/dist/lib/utils/dnsRecords.js +108 -0
  138. package/dist/lib/utils/domainTypes.d.ts +37 -0
  139. package/dist/lib/utils/domainTypes.js +10 -0
  140. package/dist/lib/utils/env.d.ts +42 -0
  141. package/dist/lib/utils/env.js +122 -0
  142. package/dist/lib/utils/getConfig.d.ts +0 -5
  143. package/dist/lib/utils/getConfig.js +42 -19
  144. package/dist/lib/utils/index.d.ts +7 -0
  145. package/dist/lib/utils/index.js +8 -1
  146. package/dist/lib/utils/removalPolicy.d.ts +2 -0
  147. package/dist/lib/utils/removalPolicy.js +16 -0
  148. package/dist/lib/utils/standardTagsAspect.d.ts +4 -0
  149. package/dist/lib/utils/standardTagsAspect.js +8 -8
  150. package/dist/lib/utils/vpcUtils.d.ts +14 -0
  151. package/dist/lib/utils/vpcUtils.js +28 -0
  152. package/package.json +7 -6
package/LICENSE ADDED
@@ -0,0 +1,21 @@
1
+ MIT License
2
+
3
+ Copyright (c) 2026 Fjall
4
+
5
+ Permission is hereby granted, free of charge, to any person obtaining a copy
6
+ of this software and associated documentation files (the "Software"), to deal
7
+ in the Software without restriction, including without limitation the rights
8
+ to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
9
+ copies of the Software, and to permit persons to whom the Software is
10
+ furnished to do so, subject to the following conditions:
11
+
12
+ The above copyright notice and this permission notice shall be included in all
13
+ copies or substantial portions of the Software.
14
+
15
+ THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
16
+ IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
17
+ FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
18
+ AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
19
+ LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
20
+ OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
21
+ SOFTWARE.
package/dist/lib/app.d.ts CHANGED
@@ -12,6 +12,7 @@ import { type AnyMessaging } from "./patterns/aws/messaging";
12
12
  import { type AnyCompute } from "./patterns/aws/compute";
13
13
  import { type Storage, type StorageFactoryFn } from "./patterns/aws/storage";
14
14
  import { type AnyPattern } from "./patterns/aws/pattern";
15
+ import { type BackupTier } from "./utils/backupTierMapping";
15
16
  import { type ResourceInventory } from "./aspects/resourceInventory";
16
17
  import { type ManifestCollector } from "./utils/manifestWriter";
17
18
  /**
@@ -43,6 +44,23 @@ export interface IAppOptions {
43
44
  network?: INetworkProps | false | {
44
45
  useExisting: string;
45
46
  };
47
+ /**
48
+ * Backup configuration for automatic AWS Backup enrolment.
49
+ * - Object with tier: Tags all resources with `fjall:disasterRecovery:tier`
50
+ * - false: Explicitly disabled (no backup tag)
51
+ */
52
+ backup?: {
53
+ tier: BackupTier;
54
+ } | false;
55
+ /**
56
+ * Tunnel configuration for secure database access via SSM Session Manager.
57
+ * - true or {}: Create bastion with default t4g.micro instance
58
+ * - { instanceType: string }: Create bastion with custom instance type
59
+ * - false: Explicitly disabled (no bastion)
60
+ */
61
+ tunnel?: {
62
+ instanceType?: string;
63
+ } | boolean;
46
64
  }
47
65
  /**
48
66
  * The basic corner-stone of all Fjall-hosted applications.
@@ -58,15 +76,25 @@ export declare class App extends CdkApp {
58
76
  private vpc?;
59
77
  private additionalVpcs;
60
78
  private defaultEcr;
61
- private defaultMonitoringRole;
62
79
  private defaultAuditRole;
63
80
  private auditRoleExternalId?;
81
+ private bastion?;
64
82
  private networkDisabled;
65
83
  private globalTags;
66
84
  private aspectApplied;
67
85
  private resourceInventory?;
68
86
  private manifestCollector;
69
87
  private constructor();
88
+ /**
89
+ * Apply backup tier tag to all resources in the app.
90
+ * Maps tier names to AWS Backup plan tag values.
91
+ */
92
+ private applyBackupTag;
93
+ /**
94
+ * Initialise the tunnel bastion in the network stack.
95
+ * Creates a minimal EC2 instance for SSM port forwarding to databases.
96
+ */
97
+ private initialiseTunnel;
70
98
  /**
71
99
  * Initialise the network (VPC) for this application.
72
100
  */
@@ -181,14 +209,6 @@ export declare class App extends CdkApp {
181
209
  * it will be created.
182
210
  */
183
211
  getDefaultContainerRegistry(): Ecr;
184
- /**
185
- * Create a cross-account monitoring role in the Network stack that allows
186
- * the Fjall webapp to query CloudWatch metrics and ECS status.
187
- *
188
- * @param webappAccountId - Optional AWS account ID of the Fjall webapp. Defaults to configured platform account.
189
- * @returns {Role} The created monitoring role
190
- */
191
- createMonitoringRole(webappAccountId?: string): Role;
192
212
  /**
193
213
  * Create a cross-account audit role in the Network stack that allows
194
214
  * the Fjall platform to use CloudQuery for comprehensive AWS resource auditing.
@@ -213,7 +233,6 @@ export declare class App extends CdkApp {
213
233
  private generateAuditExternalId;
214
234
  /**
215
235
  * Add a compute resource to the default compute stack using the factory pattern.
216
- * Automatically creates monitoring role if not already created.
217
236
  *
218
237
  * Returns the appropriate compute type based on the factory configuration:
219
238
  * - EcsCompute for type: "ecs"
@@ -262,6 +281,10 @@ export declare class App extends CdkApp {
262
281
  * cache.getTableName(); // Available
263
282
  */
264
283
  addDatabase<T extends AnyDatabase>(fn: (app: App, scope: Construct) => T): T;
284
+ /**
285
+ * Type guard for IConnectable — checks if a construct has a connections property.
286
+ */
287
+ private isConnectable;
265
288
  /**
266
289
  * Add a storage resource (S3) to the default storage stack using the factory pattern.
267
290
  *
package/dist/lib/app.js CHANGED
@@ -3,19 +3,22 @@ Object.defineProperty(exports, "__esModule", { value: true });
3
3
  exports.App = void 0;
4
4
  const aws_cdk_lib_1 = require("aws-cdk-lib");
5
5
  const vpc_1 = require("./resources/aws/networking/vpc");
6
+ const aws_cdk_lib_2 = require("aws-cdk-lib");
7
+ const ec2_1 = require("./resources/aws/compute/ec2");
6
8
  const resources_1 = require("./resources");
7
9
  const ecr_1 = require("./resources/aws/storage/ecr");
8
10
  const network_1 = require("./patterns/aws/network");
9
11
  const standardTagsAspect_1 = require("./utils/standardTagsAspect");
12
+ const backupTierMapping_1 = require("./utils/backupTierMapping");
10
13
  const crypto_1 = require("crypto");
11
14
  const getConfig_1 = require("./utils/getConfig");
12
- const monitoringRole_1 = require("./resources/aws/monitoring/monitoringRole");
13
- const monitoring_1 = require("./config/monitoring");
14
15
  const auditRole_1 = require("./resources/aws/audit/auditRole");
15
16
  const audit_1 = require("./config/audit");
16
17
  const validationLogger_1 = require("./utils/validationLogger");
17
18
  const manifestWriter_1 = require("./utils/manifestWriter");
18
19
  const capitaliseString_1 = require("./utils/capitaliseString");
20
+ const COST_ALLOCATION_ENVIRONMENT_TAG = "fjall:costAllocation:environment";
21
+ const COST_ALLOCATION_SERVICE_TAG = "fjall:costAllocation:service";
19
22
  /**
20
23
  * The basic corner-stone of all Fjall-hosted applications.
21
24
  * This class is a singleton and should be used to create and manage
@@ -35,6 +38,10 @@ class App extends aws_cdk_lib_1.App {
35
38
  // Initialise manifest collector for build-time service discovery
36
39
  this.manifestCollector = (0, manifestWriter_1.getManifestCollector)(this.name);
37
40
  this.initialiseStandardTags();
41
+ // Apply backup tier tag if configured
42
+ if (options?.backup && typeof options.backup === "object") {
43
+ this.applyBackupTag(options.backup.tier);
44
+ }
38
45
  // Initialise network immediately if configured
39
46
  if (options?.network === false) {
40
47
  this.networkDisabled = true;
@@ -42,6 +49,49 @@ class App extends aws_cdk_lib_1.App {
42
49
  else if (options?.network) {
43
50
  this.initialiseNetwork(options.network);
44
51
  }
52
+ // Initialise tunnel bastion if configured
53
+ if (options?.tunnel) {
54
+ this.initialiseTunnel(options.tunnel);
55
+ }
56
+ }
57
+ /**
58
+ * Apply backup tier tag to all resources in the app.
59
+ * Maps tier names to AWS Backup plan tag values.
60
+ */
61
+ applyBackupTag(tier) {
62
+ this.globalTags[backupTierMapping_1.BACKUP_TIER_TAG_KEY] = backupTierMapping_1.BACKUP_TIER_TAG_MAP[tier];
63
+ }
64
+ /**
65
+ * Initialise the tunnel bastion in the network stack.
66
+ * Creates a minimal EC2 instance for SSM port forwarding to databases.
67
+ */
68
+ initialiseTunnel(config) {
69
+ if (this.networkDisabled || !this.vpc) {
70
+ throw new Error("Tunnel requires a network. Configure network before enabling tunnel.");
71
+ }
72
+ const instanceType = typeof config === "object" && config.instanceType
73
+ ? config.instanceType
74
+ : "t4g.micro";
75
+ const networkStack = this.getDefaultNetworkStack();
76
+ const bastionId = `${this.stackPrefix}Bastion`;
77
+ this.bastion = new ec2_1.Ec2Instance(networkStack.getStack(), bastionId, {
78
+ serviceName: `${this.stackPrefix}Bastion`,
79
+ instanceType,
80
+ vpc: this.vpc,
81
+ enableSSH: false,
82
+ minCapacity: 1,
83
+ maxCapacity: 1
84
+ });
85
+ networkStack.addConstruct(this.bastion);
86
+ const outputPrefix = (0, capitaliseString_1.toPascalCase)(this.name);
87
+ new aws_cdk_lib_2.CfnOutput(networkStack.getStack(), `${outputPrefix}BastionInstanceId`, {
88
+ value: this.bastion.getAutoScalingGroup().autoScalingGroupName,
89
+ description: "Bastion ASG name for SSM tunnel discovery"
90
+ });
91
+ new aws_cdk_lib_2.CfnOutput(networkStack.getStack(), `${outputPrefix}BastionSecurityGroupId`, {
92
+ value: this.bastion.asgSecurityGroup.securityGroupId,
93
+ description: "Bastion security group ID"
94
+ });
45
95
  }
46
96
  /**
47
97
  * Initialise the network (VPC) for this application.
@@ -90,9 +140,7 @@ class App extends aws_cdk_lib_1.App {
90
140
  * const app = App.getApp(appName, { network: false });
91
141
  */
92
142
  static getApp(name, options) {
93
- const app = App.getInstance(name, options);
94
- app.initialiseStandardTags();
95
- return app;
143
+ return App.getInstance(name, options);
96
144
  }
97
145
  /**
98
146
  * Get/Create the singleton instance of the App
@@ -129,6 +177,16 @@ class App extends aws_cdk_lib_1.App {
129
177
  !App.instance.networkDisabled) {
130
178
  App.instance.initialiseNetwork(options.network);
131
179
  }
180
+ // Apply backup tag if configured and not already set
181
+ if (options?.backup &&
182
+ typeof options.backup === "object" &&
183
+ !App.instance.globalTags[backupTierMapping_1.BACKUP_TIER_TAG_KEY]) {
184
+ App.instance.applyBackupTag(options.backup.tier);
185
+ }
186
+ // Initialise tunnel if configured and not already created
187
+ if (options?.tunnel && !App.instance.bastion) {
188
+ App.instance.initialiseTunnel(options.tunnel);
189
+ }
132
190
  }
133
191
  return App.instance;
134
192
  }
@@ -266,29 +324,6 @@ class App extends aws_cdk_lib_1.App {
266
324
  }
267
325
  return this.defaultEcr;
268
326
  }
269
- /**
270
- * Create a cross-account monitoring role in the Network stack that allows
271
- * the Fjall webapp to query CloudWatch metrics and ECS status.
272
- *
273
- * @param webappAccountId - Optional AWS account ID of the Fjall webapp. Defaults to configured platform account.
274
- * @returns {Role} The created monitoring role
275
- */
276
- createMonitoringRole(webappAccountId) {
277
- if (!this.defaultMonitoringRole) {
278
- const networkStack = this.getDefaultNetworkStack();
279
- const accountId = webappAccountId || monitoring_1.FJALL_MONITORING_CONFIG.webappAwsAccountId;
280
- const role = monitoringRole_1.default.build(`${this.stackPrefix}MonitoringRole`, {
281
- webappAccountId: accountId,
282
- appName: this.name,
283
- roleNamePrefix: monitoring_1.FJALL_MONITORING_CONFIG.roleNamePrefix,
284
- rolePath: monitoring_1.FJALL_MONITORING_CONFIG.rolePath
285
- })(this, networkStack.getStack());
286
- networkStack.addConstruct(role);
287
- this.defaultMonitoringRole = role;
288
- return role;
289
- }
290
- return this.defaultMonitoringRole;
291
- }
292
327
  /**
293
328
  * Create a cross-account audit role in the Network stack that allows
294
329
  * the Fjall platform to use CloudQuery for comprehensive AWS resource auditing.
@@ -335,7 +370,6 @@ class App extends aws_cdk_lib_1.App {
335
370
  }
336
371
  /**
337
372
  * Add a compute resource to the default compute stack using the factory pattern.
338
- * Automatically creates monitoring role if not already created.
339
373
  *
340
374
  * Returns the appropriate compute type based on the factory configuration:
341
375
  * - EcsCompute for type: "ecs"
@@ -359,10 +393,6 @@ class App extends aws_cdk_lib_1.App {
359
393
  * worker.getFunction("process"); // Available on LambdaCompute
360
394
  */
361
395
  addCompute(fn) {
362
- // Auto-create monitoring role when compute is added
363
- if (!this.defaultMonitoringRole) {
364
- this.createMonitoringRole();
365
- }
366
396
  const computeStack = this.getDefaultComputeStack();
367
397
  const construct = fn(this, computeStack.getStack());
368
398
  computeStack.addConstruct(construct);
@@ -399,8 +429,21 @@ class App extends aws_cdk_lib_1.App {
399
429
  const databaseStack = this.getDefaultDatabaseStack();
400
430
  const databaseConstruct = fn(this, databaseStack.getStack());
401
431
  databaseStack.addConstruct(databaseConstruct);
432
+ // Wire bastion → database security group for tunnel access
433
+ if (this.bastion && this.isConnectable(databaseConstruct)) {
434
+ databaseConstruct.connections.allowDefaultPortFrom(this.bastion);
435
+ }
402
436
  return databaseConstruct;
403
437
  }
438
+ /**
439
+ * Type guard for IConnectable — checks if a construct has a connections property.
440
+ */
441
+ isConnectable(construct) {
442
+ return (typeof construct === "object" &&
443
+ construct !== null &&
444
+ "connections" in construct &&
445
+ typeof construct.connections === "object");
446
+ }
404
447
  /**
405
448
  * Add a storage resource (S3) to the default storage stack using the factory pattern.
406
449
  *
@@ -565,8 +608,8 @@ class App extends aws_cdk_lib_1.App {
565
608
  initialiseStandardTags() {
566
609
  const config = (0, getConfig_1.getConfig)();
567
610
  this.globalTags = {
568
- "fjall:costAllocation:environment": config.environment,
569
- "fjall:costAllocation:service": this.name
611
+ [COST_ALLOCATION_ENVIRONMENT_TAG]: config.environment,
612
+ [COST_ALLOCATION_SERVICE_TAG]: this.name
570
613
  };
571
614
  }
572
615
  /**
@@ -648,4 +691,4 @@ class App extends aws_cdk_lib_1.App {
648
691
  exports.App = App;
649
692
  App.instance = null;
650
693
  exports.default = App;
651
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"app.js","sourceRoot":"","sources":["../../lib/app.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  App as CdkApp,\n  Aspects,\n  Tags,\n  type StageSynthesisOptions\n} from \"aws-cdk-lib\";\nimport { type CloudAssembly } from \"aws-cdk-lib/cx-api\";\nimport { type Construct } from \"constructs\";\nimport { Vpc } from \"./resources/aws/networking/vpc\";\nimport { type IVpc } from \"aws-cdk-lib/aws-ec2\";\nimport { type Role } from \"aws-cdk-lib/aws-iam\";\nimport { AwsStack } from \"./resources\";\nimport { type Ecr, EcrFactory } from \"./resources/aws/storage/ecr\";\nimport { type AnyDatabase } from \"./patterns/aws/database\";\nimport {\n  type INetworkProps,\n  NetworkFactory,\n  type Network\n} from \"./patterns/aws/network\";\nimport { type Cdn } from \"./patterns/aws/cdn\";\nimport { type AnyMessaging } from \"./patterns/aws/messaging\";\nimport { type AnyCompute } from \"./patterns/aws/compute\";\nimport { type Storage, type StorageFactoryFn } from \"./patterns/aws/storage\";\nimport { type AnyPattern } from \"./patterns/aws/pattern\";\nimport { StandardTagsAspect } from \"./utils/standardTagsAspect\";\nimport { randomBytes } from \"crypto\";\nimport {\n  type ResourceInventoryAspect,\n  type ResourceInventory\n} from \"./aspects/resourceInventory\";\nimport { getConfig } from \"./utils/getConfig\";\nimport MonitoringRoleFactory from \"./resources/aws/monitoring/monitoringRole\";\nimport { FJALL_MONITORING_CONFIG } from \"./config/monitoring\";\nimport AuditRoleFactory from \"./resources/aws/audit/auditRole\";\nimport { FJALL_AUDIT_CONFIG } from \"./config/audit\";\nimport { FjallLogger } from \"./utils/validationLogger\";\nimport {\n  getManifestCollector,\n  writeManifest,\n  type ManifestCollector\n} from \"./utils/manifestWriter\";\nimport { toPascalCase, toKebab } from \"./utils/capitaliseString\";\n\n/**\n * Configuration options for App.getApp().\n *\n * @example\n * // Create app with new VPC\n * const app = App.getApp(appName, {\n *   network: { maxAzs: 2, natGateways: false }\n * });\n *\n * @example\n * // Create app using existing VPC\n * const app = App.getApp(appName, {\n *   network: { useExisting: \"vpc-12345678\" }\n * });\n *\n * @example\n * // Create app without network (S3-only apps)\n * const app = App.getApp(appName, { network: false });\n */\nexport interface IAppOptions {\n  /**\n   * Network configuration.\n   * - Object with INetworkProps: Create new VPC with config\n   * - false: No network (for S3-only apps)\n   * - { useExisting: string }: Use existing VPC by ID\n   */\n  network?: INetworkProps | false | { useExisting: string };\n}\n\n/**\n * The basic corner-stone of all Fjall-hosted applications.\n *  This class is a singleton and should be used to create and manage\n *   all resources in a Fjall application.\n */\nexport class App extends CdkApp {\n  private static instance: App | null = null;\n\n  private name: string;\n  /** PascalCase form of name, used as stack-name prefix and construct IDs. */\n  private stackPrefix: string;\n  private stacks: { [key: string]: AwsStack } = {};\n\n  private vpc?: IVpc;\n  private additionalVpcs = new Map<string, IVpc>();\n  private defaultEcr: Ecr | undefined;\n  private defaultMonitoringRole: Role | undefined;\n  private defaultAuditRole: Role | undefined;\n  private auditRoleExternalId?: string;\n  private networkDisabled = false;\n  private globalTags: { [key: string]: string } = {};\n  private aspectApplied = false;\n  private resourceInventory?: ResourceInventoryAspect;\n  private manifestCollector: ManifestCollector;\n\n  private constructor(name?: string, options?: IAppOptions) {\n    super();\n\n    App.instance = this;\n\n    this.name = name ?? \"FjallApp\";\n    this.stackPrefix = toPascalCase(this.name);\n\n    // Initialise manifest collector for build-time service discovery\n    this.manifestCollector = getManifestCollector(this.name);\n\n    this.initialiseStandardTags();\n\n    // Initialise network immediately if configured\n    if (options?.network === false) {\n      this.networkDisabled = true;\n    } else if (options?.network) {\n      this.initialiseNetwork(options.network);\n    }\n  }\n\n  /**\n   * Initialise the network (VPC) for this application.\n   */\n  private initialiseNetwork(\n    config: INetworkProps | { useExisting: string }\n  ): void {\n    const networkStack = this.getDefaultNetworkStack();\n\n    if (\"useExisting\" in config) {\n      this.vpc = Vpc.fromLookup(\n        networkStack.getStack(),\n        `${this.stackPrefix}ImportedVpc`,\n        {\n          vpcId: config.useExisting\n        }\n      );\n    } else {\n      const networkConfig = { ...config, vpcName: config.vpcName ?? this.name };\n      const network = NetworkFactory.build(\n        `${this.stackPrefix}Vpc`,\n        networkConfig\n      )(this, networkStack.getStack());\n      this.vpc = network.getVpc();\n    }\n  }\n\n  /**\n   * Get the application name.\n   * @returns {string} The application name\n   */\n  public getName(): string {\n    return this.name;\n  }\n\n  /**\n   * Get/Create a basic Fjall Application with standard tags applied.\n   *\n   * @param name Application name\n   * @param options Configuration options including network settings\n   * @returns {App}\n   *\n   * @example\n   * // Create app with new VPC\n   * const app = App.getApp(appName, {\n   *   network: { maxAzs: 2, natGateways: false }\n   * });\n   *\n   * @example\n   * // Create app using existing VPC\n   * const app = App.getApp(appName, {\n   *   network: { useExisting: \"vpc-12345678\" }\n   * });\n   *\n   * @example\n   * // Create app without network (S3-only apps)\n   * const app = App.getApp(appName, { network: false });\n   */\n  public static getApp(name?: string, options?: IAppOptions): App {\n    const app = App.getInstance(name, options);\n    app.initialiseStandardTags();\n    return app;\n  }\n\n  /**\n   * Get/Create the singleton instance of the App\n   * @param name Application name\n   * @param options Configuration options including network settings\n   * @returns {App}\n   */\n  public static getInstance(name?: string, options?: IAppOptions): App {\n    // Despite supporting multiple stacks you can still only ever\n    // have a single Application per CDK deployment\n    if (!App.instance) {\n      App.instance = new App(name, options);\n    } else {\n      // If singleton exists but name differs, update it (before any infra is defined)\n      if (name && App.instance.name !== name) {\n        // Warn if network already initialised - changing name after network exists creates a mismatch\n        if (App.instance.vpc) {\n          FjallLogger.warn(\n            `App name changed from \"${App.instance.name}\" to \"${name}\" after network was initialised. ` +\n              `VPC will have name \"${App.instance.name}Vpc\" but app name is \"${name}\". ` +\n              `Ensure App.getApp() is called with the final name before initialising network.`\n          );\n        }\n        App.instance.name = name;\n        App.instance.stackPrefix = toPascalCase(name);\n        // Reinitialise standard tags with the new name\n        App.instance.initialiseStandardTags();\n      }\n\n      // If options provided but network not yet initialised, initialise it now\n      if (options?.network === false) {\n        App.instance.networkDisabled = true;\n      } else if (\n        options?.network &&\n        !App.instance.vpc &&\n        !App.instance.networkDisabled\n      ) {\n        App.instance.initialiseNetwork(options.network);\n      }\n    }\n\n    return App.instance;\n  }\n\n  /**\n   * Retrieve a stack by key. If the stack does not exist, it will be created.\n   *  Dependencies are only applied the first time a stack is created.\n   *\n   * @param key - The key of the stack\n   * @param dependencies - The stack(s) that this stack depends on\n   * @returns {AwsStack}\n   */\n  public getStack(key: string, dependencies?: AwsStack | AwsStack[]): AwsStack {\n    // Apply the aspect once before creating the first stack\n    if (\n      !this.aspectApplied &&\n      Object.keys(this.stacks).length === 0 &&\n      Object.keys(this.globalTags).length > 0\n    ) {\n      this.applyTagsAspect();\n    }\n\n    if (!this.stacks[key]) {\n      this.stacks[key] = new AwsStack(key, dependencies);\n    }\n\n    return this.stacks[key];\n  }\n\n  /**\n   * Retrieve default compute stack - named as `${this.name}Compute`\n   *\n   * Only depends on Network. Database dependency is added automatically\n   * by CDK when compute resources reference database resources.\n   *\n   * @returns {AwsStack}\n   */\n  public getDefaultComputeStack(): AwsStack {\n    return this.getStack(\n      `${this.stackPrefix}Compute`,\n      this.getDefaultNetworkStack()\n    );\n  }\n\n  /**\n   * Retrieve default network stack - named as `${this.name}Network`\n   * @returns {AwsStack}\n   */\n  public getDefaultNetworkStack(): AwsStack {\n    return this.getStack(`${this.stackPrefix}Network`);\n  }\n\n  /**\n   * Retrieve default database stack - named as `${this.name}Database`\n   * @returns {AwsStack}\n   */\n  public getDefaultDatabaseStack(): AwsStack {\n    return this.getStack(\n      `${this.stackPrefix}Database`,\n      this.getDefaultNetworkStack()\n    );\n  }\n\n  /**\n   * Retrieve default storage stack - named as `${this.name}Storage`\n   * @returns {AwsStack}\n   */\n  public getDefaultStorageStack(): AwsStack {\n    return this.getStack(\n      `${this.stackPrefix}Storage`,\n      this.getDefaultNetworkStack()\n    );\n  }\n\n  /**\n   * Retrieve default CDN stack - named as `${this.name}Cdn`\n   *\n   * Depends on Network. Compute/Storage dependencies are added automatically\n   * by CDK when CDN resources reference ALB or S3 bucket resources.\n   *\n   * @returns {AwsStack}\n   */\n  public getDefaultCdnStack(): AwsStack {\n    return this.getStack(\n      `${this.stackPrefix}Cdn`,\n      this.getDefaultNetworkStack()\n    );\n  }\n\n  /**\n   * Retrieve default messaging stack - named as `${this.name}Messaging`\n   *\n   * Used for SQS queues, SNS topics, and EventBridge event buses.\n   * Depends on Network only. These are regional services that don't\n   * require VPC, but we maintain consistent stack dependency patterns.\n   *\n   * @returns {AwsStack}\n   */\n  public getDefaultMessagingStack(): AwsStack {\n    return this.getStack(\n      `${this.stackPrefix}Messaging`,\n      this.getDefaultNetworkStack()\n    );\n  }\n\n  /**\n   * Get a VPC by name. If no name is provided, returns the default VPC.\n   *\n   * This is a pure getter - it never creates infrastructure.\n   * Network must be configured via App.getApp() options or app.addNetwork().\n   *\n   * @param name - Optional name of the VPC to retrieve. If not provided, returns the default VPC.\n   * @returns {IVpc} The configured VPC\n   * @throws {Error} If network is disabled, not configured, or named VPC not found\n   */\n  public getVpc(name?: string): IVpc {\n    if (this.networkDisabled) {\n      throw new Error(\n        \"Network is disabled for this app. Cannot get VPC. \" +\n          \"Pass network config to App.getApp() to enable networking.\"\n      );\n    }\n\n    // If name is provided, look in additional VPCs first\n    if (name) {\n      const additionalVpc = this.additionalVpcs.get(name);\n      if (additionalVpc) {\n        return additionalVpc;\n      }\n      throw new Error(\n        `VPC '${name}' not found. Available VPCs: ${this.getVpcNames().join(\", \")}. ` +\n          \"Create additional VPCs using app.addNetwork(NetworkFactory.build(...)).\"\n      );\n    }\n\n    // Return default VPC\n    if (!this.vpc) {\n      throw new Error(\n        \"Network not configured. Pass network config to App.getApp(). \" +\n          \"Example: App.getApp(appName, { network: { maxAzs: 2 } })\"\n      );\n    }\n    return this.vpc;\n  }\n\n  /**\n   * Get the names of all available VPCs.\n   *\n   * @returns {string[]} Array of VPC names. Includes \"default\" if the default VPC is configured.\n   */\n  public getVpcNames(): string[] {\n    const names: string[] = [];\n    if (this.vpc) {\n      names.push(\"default\");\n    }\n    names.push(...this.additionalVpcs.keys());\n    return names;\n  }\n\n  /**\n   * Retrieve the default application container registry. If the registry does not exist\n   *  it will be created.\n   */\n  public getDefaultContainerRegistry(): Ecr {\n    if (!this.defaultEcr) {\n      const networkStack = this.getDefaultNetworkStack();\n\n      const ecr = EcrFactory.build(`${this.stackPrefix}Ecr`, {\n        repositoryName: toKebab(this.name)\n      })(this, networkStack.getStack());\n      this.defaultEcr = ecr;\n      return ecr;\n    }\n\n    return this.defaultEcr;\n  }\n\n  /**\n   * Create a cross-account monitoring role in the Network stack that allows\n   * the Fjall webapp to query CloudWatch metrics and ECS status.\n   *\n   * @param webappAccountId - Optional AWS account ID of the Fjall webapp. Defaults to configured platform account.\n   * @returns {Role} The created monitoring role\n   */\n  public createMonitoringRole(webappAccountId?: string): Role {\n    if (!this.defaultMonitoringRole) {\n      const networkStack = this.getDefaultNetworkStack();\n      const accountId =\n        webappAccountId || FJALL_MONITORING_CONFIG.webappAwsAccountId;\n\n      const role = MonitoringRoleFactory.build(\n        `${this.stackPrefix}MonitoringRole`,\n        {\n          webappAccountId: accountId,\n          appName: this.name,\n          roleNamePrefix: FJALL_MONITORING_CONFIG.roleNamePrefix,\n          rolePath: FJALL_MONITORING_CONFIG.rolePath\n        }\n      )(this, networkStack.getStack());\n\n      networkStack.addConstruct(role);\n      this.defaultMonitoringRole = role;\n      return role;\n    }\n\n    return this.defaultMonitoringRole;\n  }\n\n  /**\n   * Create a cross-account audit role in the Network stack that allows\n   * the Fjall platform to use CloudQuery for comprehensive AWS resource auditing.\n   *\n   * This is an explicit opt-in feature. Call this method to enable CloudQuery\n   * audit capabilities for your application.\n   *\n   * @param webappAccountId - Optional AWS account ID of the Fjall webapp. Defaults to configured platform account.\n   * @param externalId - Optional external ID for additional security. If not provided, a unique ID will be generated.\n   * @returns {Role} The created audit role\n   */\n  public createAuditRole(webappAccountId?: string, externalId?: string): Role {\n    if (!this.defaultAuditRole) {\n      const networkStack = this.getDefaultNetworkStack();\n      const accountId =\n        webappAccountId || FJALL_AUDIT_CONFIG.webappAwsAccountId;\n\n      this.auditRoleExternalId = externalId || this.generateAuditExternalId();\n\n      const role = AuditRoleFactory.build(`${this.stackPrefix}AuditRole`, {\n        webappAccountId: accountId,\n        appName: this.name,\n        externalId: this.auditRoleExternalId,\n        roleNamePrefix: FJALL_AUDIT_CONFIG.roleNamePrefix,\n        rolePath: FJALL_AUDIT_CONFIG.rolePath\n      })(this, networkStack.getStack());\n\n      networkStack.addConstruct(role);\n      this.defaultAuditRole = role;\n      return role;\n    }\n\n    return this.defaultAuditRole;\n  }\n\n  /**\n   * Get the external ID used for the audit role\n   * @returns {string | undefined} The external ID if audit role has been created\n   */\n  public getAuditRoleExternalId(): string | undefined {\n    return this.auditRoleExternalId;\n  }\n\n  /**\n   * Generate a unique external ID for audit role\n   * Format: fjall-audit-{appName}-{timestamp}\n   */\n  private generateAuditExternalId(): string {\n    const uniqueId = randomBytes(8).toString(\"hex\");\n    return `fjall-audit-${toKebab(this.name)}-${uniqueId}`;\n  }\n\n  /**\n   * Add a compute resource to the default compute stack using the factory pattern.\n   * Automatically creates monitoring role if not already created.\n   *\n   * Returns the appropriate compute type based on the factory configuration:\n   * - EcsCompute for type: \"ecs\"\n   * - LambdaCompute for type: \"lambda\"\n   * - Ec2Compute for type: \"ec2\"\n   *\n   * @example\n   * // ECS compute - returns EcsCompute\n   * const api = app.addCompute(ComputeFactory.build(\"Api\", {\n   *   type: \"ecs\",\n   *   services: [{ name: \"api\", ... }]\n   * }));\n   * api.getLoadBalancer(); // Available on EcsCompute\n   *\n   * @example\n   * // Lambda compute - returns LambdaCompute\n   * const worker = app.addCompute(ComputeFactory.build(\"Worker\", {\n   *   type: \"lambda\",\n   *   functions: [{ name: \"process\", ... }]\n   * }));\n   * worker.getFunction(\"process\"); // Available on LambdaCompute\n   */\n  public addCompute<T extends AnyCompute>(\n    fn: (app: App, scope: Construct) => T\n  ): T {\n    // Auto-create monitoring role when compute is added\n    if (!this.defaultMonitoringRole) {\n      this.createMonitoringRole();\n    }\n\n    const computeStack = this.getDefaultComputeStack();\n    const construct = fn(this, computeStack.getStack());\n    computeStack.addConstruct(construct as unknown as Construct);\n    return construct;\n  }\n\n  /**\n   * Manually add a resource to the default compute stack.\n   */\n  public addComputeResource(resource: Construct): void {\n    const computeStack = this.getDefaultComputeStack();\n    computeStack.addConstruct(resource);\n  }\n\n  /**\n   * Add a database resource to the default database stack using the factory pattern.\n   * Returns the appropriate database type based on the factory used.\n   *\n   * @example\n   * // Aurora database - returns RelationalDatabase\n   * const db = app.addDatabase(DatabaseFactory.build(\"Main\", {\n   *   type: \"Aurora\",\n   *   databaseName: \"my_app\"\n   * }));\n   * db.getHostEndpoint(); // Available\n   *\n   * @example\n   * // DynamoDB table - returns DynamoDBDatabase\n   * const cache = app.addDatabase(DatabaseFactory.build(\"Cache\", {\n   *   type: \"DynamoDB\",\n   *   partitionKey: { name: \"id\", type: \"S\" }\n   * }));\n   * cache.getTableName(); // Available\n   */\n  public addDatabase<T extends AnyDatabase>(\n    fn: (app: App, scope: Construct) => T\n  ): T {\n    const databaseStack = this.getDefaultDatabaseStack();\n    const databaseConstruct = fn(this, databaseStack.getStack());\n    databaseStack.addConstruct(databaseConstruct as unknown as Construct);\n\n    return databaseConstruct;\n  }\n\n  /**\n   * Add a storage resource (S3) to the default storage stack using the factory pattern.\n   *\n   * @example\n   * // Private bucket (default)\n   * const assets = app.addStorage(StorageFactory.build(\"Assets\", {\n   *   versioned: true\n   * }));\n   *\n   * @example\n   * // Website bucket\n   * const site = app.addStorage(StorageFactory.build(\"Site\", {\n   *   websiteHosting: { indexDocument: \"index.html\" }\n   * }));\n   */\n  public addStorage(fn: StorageFactoryFn): Storage {\n    const placement = fn.stackPlacement ?? \"storage\";\n    const stack = this.resolveStorageStack(placement);\n    const storageConstruct = fn(this, stack.getStack());\n    stack.addConstruct(storageConstruct);\n    return storageConstruct;\n  }\n\n  private resolveStorageStack(placement: string): AwsStack {\n    switch (placement) {\n      case \"cdn\":\n        return this.getDefaultCdnStack();\n      case \"compute\":\n        return this.getDefaultComputeStack();\n      default:\n        return this.getDefaultStorageStack();\n    }\n  }\n\n  /**\n   * Add a CDN resource (CloudFront) to the default CDN stack using the factory pattern.\n   *\n   * @example\n   * // S3 origin\n   * const assets = app.addStorage(StorageFactory.build(\"Assets\", {}));\n   * app.addCdn(CdnFactory.build(\"AssetsCdn\", {\n   *   originType: \"s3\",\n   *   bucket: assets\n   * }));\n   *\n   * @example\n   * // ALB origin (ECS)\n   * const api = app.addCompute(ComputeFactory.build(\"Api\", { type: \"ecs\", ... }));\n   * app.addCdn(CdnFactory.build(\"ApiCdn\", {\n   *   originType: \"alb\",\n   *   loadBalancer: api\n   * }));\n   */\n  public addCdn(fn: (app: App, scope: Construct) => Cdn): Cdn {\n    const cdnStack = this.getDefaultCdnStack();\n    const cdn = fn(this, cdnStack.getStack());\n    cdnStack.addConstruct(cdn);\n\n    return cdn;\n  }\n\n  /**\n   * Add a messaging resource to the default queue stack using the factory pattern.\n   *\n   * Returns the appropriate messaging type based on the factory configuration.\n   * Currently supports queue (SQS), with future support planned for\n   * topic (SNS), eventbus (EventBridge), and stream (Kinesis).\n   *\n   * @example\n   * // Queue messaging - returns QueueMessaging\n   * const notifications = app.addMessaging(MessagingFactory.build(\"Notifications\", {\n   *   type: \"queue\",\n   *   queueType: \"standard\",\n   *   deadLetterQueue: { enabled: true, maxReceiveCount: 3 }\n   * }));\n   * notifications.grantSendMessages(apiFunction);\n   *\n   * @example\n   * // FIFO queue for OpenNext revalidation\n   * const revalidation = app.addMessaging(MessagingFactory.build(\"Revalidation\", {\n   *   type: \"queue\",\n   *   queueType: \"fifo\",\n   *   visibilityTimeout: 300,\n   *   contentBasedDeduplication: true\n   * }));\n   */\n  public addMessaging<T extends AnyMessaging>(\n    fn: (app: App, scope: Construct) => T\n  ): T {\n    const messagingStack = this.getDefaultMessagingStack();\n    const messaging = fn(this, messagingStack.getStack());\n    messagingStack.addConstruct(messaging as unknown as Construct);\n\n    return messaging;\n  }\n\n  /**\n   * Add a high-level infrastructure pattern to the application.\n   *\n   * Patterns are composite constructs that create multiple related resources\n   * as a single, cohesive unit. They encapsulate best practices and reduce\n   * boilerplate for common deployment scenarios.\n   *\n   * Currently supported patterns:\n   * - `payload`: Payload CMS deployment with OpenNext\n   *\n   * Returns the appropriate pattern interface based on the factory configuration,\n   * providing access to all underlying resources via escape hatches.\n   *\n   * @example\n   * // Payload CMS pattern - returns Payload\n   * const payload = app.addPattern(PatternFactory.build(\"PayloadApp\", {\n   *   type: \"payload\",\n   *   name: \"my-cms\",\n   *   source: \"../..\"\n   * }));\n   *\n   * @example\n   * // Escape hatches - access underlying resources\n   * payload.getDatabase().grantConnect(otherLambda);\n   * payload.getServer().getLambdaFunction().addEnvironment(\"CUSTOM\", \"value\");\n   * payload.getCdn().getDistribution().addBehavior(\"/custom/*\", customOrigin);\n   */\n  public addPattern<T extends AnyPattern>(\n    fn: (app: App, scope: Construct) => T\n  ): T {\n    const computeStack = this.getDefaultComputeStack();\n    const pattern = fn(this, computeStack.getStack());\n    computeStack.addConstruct(pattern as unknown as Construct);\n    return pattern;\n  }\n\n  /**\n   * Add an additional network (VPC) to the application.\n   *\n   * Use this to create additional VPCs beyond the default VPC configured via App.getApp().\n   * Additional VPCs can be retrieved by name using app.getVpc(name).\n   *\n   * @param fn - Factory function that creates the Network construct\n   * @returns {Network} The created Network construct\n   *\n   * @example\n   * const isolatedVpc = app.addNetwork(\n   *   NetworkFactory.build(\"IsolatedVpc\", { maxAzs: 2, natGateways: false })\n   * );\n   */\n  public addNetwork(fn: (app: App, scope: Construct) => Network): Network {\n    const networkStack = this.getDefaultNetworkStack();\n    const network = fn(this, networkStack.getStack());\n    const vpcName = network.node.id;\n\n    // Store in additional VPCs map for retrieval via getVpc(name)\n    this.additionalVpcs.set(vpcName, network.getVpc());\n    networkStack.addConstruct(network);\n\n    return network;\n  }\n\n  /**\n   * Manually add a resource to the default database stack.\n   */\n  public addDatabaseResource(resource: Construct): void {\n    const databaseStack = this.getDefaultDatabaseStack();\n    databaseStack.addConstruct(resource);\n  }\n\n  /**\n   * Manually add a resource to the default storage stack.\n   */\n  public addStorageResource(resource: Construct): void {\n    const storageStack = this.getDefaultStorageStack();\n    storageStack.addConstruct(resource);\n  }\n\n  /**\n   * Initialise standard tags\n   */\n  private initialiseStandardTags(): void {\n    const config = getConfig();\n    this.globalTags = {\n      \"fjall:costAllocation:environment\": config.environment,\n      \"fjall:costAllocation:service\": this.name\n    };\n  }\n\n  /**\n   * Apply all tags using CDK's native Tags.of().add() API\n   */\n  private applyTagsAspect(): void {\n    if (!this.aspectApplied && Object.keys(this.globalTags).length > 0) {\n      // Apply standard tags using Tags.of(this).add()\n      for (const [key, value] of Object.entries(this.globalTags)) {\n        Tags.of(this).add(key, value);\n      }\n\n      // Apply aspect for resource-specific tags (IPAM pool, backup tier)\n      Aspects.of(this).add(new StandardTagsAspect());\n\n      this.aspectApplied = true;\n    }\n  }\n\n  /**\n   * Add custom tags to all resources in the app.\n   *\n   * @param tags Custom tags to apply to all resources\n   * @example\n   * app.addTags({\n   *   \"fjall:costAllocation:owner\": \"platform-team\",\n   *   \"fjall:costAllocation:cost-center\": \"CC-123\",\n   *   \"team:slack-channel\": \"#platform-alerts\"\n   * });\n   */\n  public addTags(tags: { [key: string]: string }): App {\n    // Merge tags into globalTags for tracking\n    this.globalTags = {\n      ...this.globalTags,\n      ...tags\n    };\n    for (const [key, value] of Object.entries(tags)) {\n      Tags.of(this).add(key, value);\n    }\n\n    return this;\n  }\n\n  /**\n   * Export resource inventory collected during synthesis\n   * Call this after app.synth() to get complete resource inventory\n   * @returns {ResourceInventory} Complete inventory of all resources in the app\n   */\n  public exportResourceInventory(): ResourceInventory {\n    if (!this.resourceInventory) {\n      throw new Error(\n        \"Resource inventory not available. Ensure a ResourceInventoryAspect has been added before calling exportResourceInventory().\"\n      );\n    }\n    return {\n      resources: this.resourceInventory.getInventory(),\n      generatedAt: new Date().toISOString(),\n      appName: this.name\n    };\n  }\n\n  /**\n   * Get the manifest collector for registering services and patterns.\n   * Used by ComputeFactory and PatternFactory to register their configurations.\n   */\n  public getManifestCollector(): ManifestCollector {\n    return this.manifestCollector;\n  }\n\n  /**\n   * Override synth to automatically export resource inventory and manifest\n   */\n  public synth(options?: StageSynthesisOptions): CloudAssembly {\n    // Call parent synth first\n    const assembly = super.synth(options);\n\n    // After synthesis, write Fjall manifest to cdk.out\n    try {\n      writeManifest(assembly, this.manifestCollector);\n    } catch (error) {\n      // Don't fail synth if manifest export fails\n      const errorMessage =\n        error instanceof Error ? error.message : String(error);\n      FjallLogger.warn(`Failed to export Fjall manifest: ${errorMessage}`);\n    }\n\n    return assembly;\n  }\n}\n\nexport default App;\n"]}
694
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"app.js","sourceRoot":"","sources":["../../lib/app.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  App as CdkApp,\n  Aspects,\n  Tags,\n  type StageSynthesisOptions\n} from \"aws-cdk-lib\";\nimport { type CloudAssembly } from \"aws-cdk-lib/cx-api\";\nimport { type Construct } from \"constructs\";\nimport { Vpc } from \"./resources/aws/networking/vpc\";\nimport { type IVpc, type IConnectable } from \"aws-cdk-lib/aws-ec2\";\nimport { CfnOutput } from \"aws-cdk-lib\";\nimport { type Role } from \"aws-cdk-lib/aws-iam\";\nimport { Ec2Instance } from \"./resources/aws/compute/ec2\";\nimport { AwsStack } from \"./resources\";\nimport { type Ecr, EcrFactory } from \"./resources/aws/storage/ecr\";\nimport { type AnyDatabase } from \"./patterns/aws/database\";\nimport {\n  type INetworkProps,\n  NetworkFactory,\n  type Network\n} from \"./patterns/aws/network\";\nimport { type Cdn } from \"./patterns/aws/cdn\";\nimport { type AnyMessaging } from \"./patterns/aws/messaging\";\nimport { type AnyCompute } from \"./patterns/aws/compute\";\nimport { type Storage, type StorageFactoryFn } from \"./patterns/aws/storage\";\nimport { type AnyPattern } from \"./patterns/aws/pattern\";\nimport { StandardTagsAspect } from \"./utils/standardTagsAspect\";\nimport {\n  type BackupTier,\n  BACKUP_TIER_TAG_KEY,\n  BACKUP_TIER_TAG_MAP\n} from \"./utils/backupTierMapping\";\nimport { randomBytes } from \"crypto\";\nimport {\n  type ResourceInventoryAspect,\n  type ResourceInventory\n} from \"./aspects/resourceInventory\";\nimport { getConfig } from \"./utils/getConfig\";\nimport AuditRoleFactory from \"./resources/aws/audit/auditRole\";\nimport { FJALL_AUDIT_CONFIG } from \"./config/audit\";\nimport { FjallLogger } from \"./utils/validationLogger\";\nimport {\n  getManifestCollector,\n  writeManifest,\n  type ManifestCollector\n} from \"./utils/manifestWriter\";\nimport { toPascalCase, toKebab } from \"./utils/capitaliseString\";\n\nconst COST_ALLOCATION_ENVIRONMENT_TAG = \"fjall:costAllocation:environment\";\nconst COST_ALLOCATION_SERVICE_TAG = \"fjall:costAllocation:service\";\n\n/**\n * Configuration options for App.getApp().\n *\n * @example\n * // Create app with new VPC\n * const app = App.getApp(appName, {\n *   network: { maxAzs: 2, natGateways: false }\n * });\n *\n * @example\n * // Create app using existing VPC\n * const app = App.getApp(appName, {\n *   network: { useExisting: \"vpc-12345678\" }\n * });\n *\n * @example\n * // Create app without network (S3-only apps)\n * const app = App.getApp(appName, { network: false });\n */\nexport interface IAppOptions {\n  /**\n   * Network configuration.\n   * - Object with INetworkProps: Create new VPC with config\n   * - false: No network (for S3-only apps)\n   * - { useExisting: string }: Use existing VPC by ID\n   */\n  network?: INetworkProps | false | { useExisting: string };\n  /**\n   * Backup configuration for automatic AWS Backup enrolment.\n   * - Object with tier: Tags all resources with `fjall:disasterRecovery:tier`\n   * - false: Explicitly disabled (no backup tag)\n   */\n  backup?: { tier: BackupTier } | false;\n  /**\n   * Tunnel configuration for secure database access via SSM Session Manager.\n   * - true or {}: Create bastion with default t4g.micro instance\n   * - { instanceType: string }: Create bastion with custom instance type\n   * - false: Explicitly disabled (no bastion)\n   */\n  tunnel?: { instanceType?: string } | boolean;\n}\n\n/**\n * The basic corner-stone of all Fjall-hosted applications.\n *  This class is a singleton and should be used to create and manage\n *   all resources in a Fjall application.\n */\nexport class App extends CdkApp {\n  private static instance: App | null = null;\n\n  private name: string;\n  /** PascalCase form of name, used as stack-name prefix and construct IDs. */\n  private stackPrefix: string;\n  private stacks: { [key: string]: AwsStack } = {};\n\n  private vpc?: IVpc;\n  private additionalVpcs = new Map<string, IVpc>();\n  private defaultEcr: Ecr | undefined;\n  private defaultAuditRole: Role | undefined;\n  private auditRoleExternalId?: string;\n  private bastion?: Ec2Instance;\n  private networkDisabled = false;\n  private globalTags: { [key: string]: string } = {};\n  private aspectApplied = false;\n  private resourceInventory?: ResourceInventoryAspect;\n  private manifestCollector: ManifestCollector;\n\n  private constructor(name?: string, options?: IAppOptions) {\n    super();\n\n    App.instance = this;\n\n    this.name = name ?? \"FjallApp\";\n    this.stackPrefix = toPascalCase(this.name);\n\n    // Initialise manifest collector for build-time service discovery\n    this.manifestCollector = getManifestCollector(this.name);\n\n    this.initialiseStandardTags();\n\n    // Apply backup tier tag if configured\n    if (options?.backup && typeof options.backup === \"object\") {\n      this.applyBackupTag(options.backup.tier);\n    }\n\n    // Initialise network immediately if configured\n    if (options?.network === false) {\n      this.networkDisabled = true;\n    } else if (options?.network) {\n      this.initialiseNetwork(options.network);\n    }\n\n    // Initialise tunnel bastion if configured\n    if (options?.tunnel) {\n      this.initialiseTunnel(options.tunnel);\n    }\n  }\n\n  /**\n   * Apply backup tier tag to all resources in the app.\n   * Maps tier names to AWS Backup plan tag values.\n   */\n  private applyBackupTag(tier: BackupTier): void {\n    this.globalTags[BACKUP_TIER_TAG_KEY] = BACKUP_TIER_TAG_MAP[tier];\n  }\n\n  /**\n   * Initialise the tunnel bastion in the network stack.\n   * Creates a minimal EC2 instance for SSM port forwarding to databases.\n   */\n  private initialiseTunnel(config: { instanceType?: string } | true): void {\n    if (this.networkDisabled || !this.vpc) {\n      throw new Error(\n        \"Tunnel requires a network. Configure network before enabling tunnel.\"\n      );\n    }\n\n    const instanceType =\n      typeof config === \"object\" && config.instanceType\n        ? config.instanceType\n        : \"t4g.micro\";\n\n    const networkStack = this.getDefaultNetworkStack();\n    const bastionId = `${this.stackPrefix}Bastion`;\n\n    this.bastion = new Ec2Instance(networkStack.getStack(), bastionId, {\n      serviceName: `${this.stackPrefix}Bastion`,\n      instanceType,\n      vpc: this.vpc,\n      enableSSH: false,\n      minCapacity: 1,\n      maxCapacity: 1\n    });\n\n    networkStack.addConstruct(this.bastion);\n\n    const outputPrefix = toPascalCase(this.name);\n    new CfnOutput(networkStack.getStack(), `${outputPrefix}BastionInstanceId`, {\n      value: this.bastion.getAutoScalingGroup().autoScalingGroupName,\n      description: \"Bastion ASG name for SSM tunnel discovery\"\n    });\n    new CfnOutput(\n      networkStack.getStack(),\n      `${outputPrefix}BastionSecurityGroupId`,\n      {\n        value: this.bastion.asgSecurityGroup.securityGroupId,\n        description: \"Bastion security group ID\"\n      }\n    );\n  }\n\n  /**\n   * Initialise the network (VPC) for this application.\n   */\n  private initialiseNetwork(\n    config: INetworkProps | { useExisting: string }\n  ): void {\n    const networkStack = this.getDefaultNetworkStack();\n\n    if (\"useExisting\" in config) {\n      this.vpc = Vpc.fromLookup(\n        networkStack.getStack(),\n        `${this.stackPrefix}ImportedVpc`,\n        {\n          vpcId: config.useExisting\n        }\n      );\n    } else {\n      const networkConfig = { ...config, vpcName: config.vpcName ?? this.name };\n      const network = NetworkFactory.build(\n        `${this.stackPrefix}Vpc`,\n        networkConfig\n      )(this, networkStack.getStack());\n      this.vpc = network.getVpc();\n    }\n  }\n\n  /**\n   * Get the application name.\n   * @returns {string} The application name\n   */\n  public getName(): string {\n    return this.name;\n  }\n\n  /**\n   * Get/Create a basic Fjall Application with standard tags applied.\n   *\n   * @param name Application name\n   * @param options Configuration options including network settings\n   * @returns {App}\n   *\n   * @example\n   * // Create app with new VPC\n   * const app = App.getApp(appName, {\n   *   network: { maxAzs: 2, natGateways: false }\n   * });\n   *\n   * @example\n   * // Create app using existing VPC\n   * const app = App.getApp(appName, {\n   *   network: { useExisting: \"vpc-12345678\" }\n   * });\n   *\n   * @example\n   * // Create app without network (S3-only apps)\n   * const app = App.getApp(appName, { network: false });\n   */\n  public static getApp(name?: string, options?: IAppOptions): App {\n    return App.getInstance(name, options);\n  }\n\n  /**\n   * Get/Create the singleton instance of the App\n   * @param name Application name\n   * @param options Configuration options including network settings\n   * @returns {App}\n   */\n  public static getInstance(name?: string, options?: IAppOptions): App {\n    // Despite supporting multiple stacks you can still only ever\n    // have a single Application per CDK deployment\n    if (!App.instance) {\n      App.instance = new App(name, options);\n    } else {\n      // If singleton exists but name differs, update it (before any infra is defined)\n      if (name && App.instance.name !== name) {\n        // Warn if network already initialised - changing name after network exists creates a mismatch\n        if (App.instance.vpc) {\n          FjallLogger.warn(\n            `App name changed from \"${App.instance.name}\" to \"${name}\" after network was initialised. ` +\n              `VPC will have name \"${App.instance.name}Vpc\" but app name is \"${name}\". ` +\n              `Ensure App.getApp() is called with the final name before initialising network.`\n          );\n        }\n        App.instance.name = name;\n        App.instance.stackPrefix = toPascalCase(name);\n        // Reinitialise standard tags with the new name\n        App.instance.initialiseStandardTags();\n      }\n\n      // If options provided but network not yet initialised, initialise it now\n      if (options?.network === false) {\n        App.instance.networkDisabled = true;\n      } else if (\n        options?.network &&\n        !App.instance.vpc &&\n        !App.instance.networkDisabled\n      ) {\n        App.instance.initialiseNetwork(options.network);\n      }\n\n      // Apply backup tag if configured and not already set\n      if (\n        options?.backup &&\n        typeof options.backup === \"object\" &&\n        !App.instance.globalTags[BACKUP_TIER_TAG_KEY]\n      ) {\n        App.instance.applyBackupTag(options.backup.tier);\n      }\n\n      // Initialise tunnel if configured and not already created\n      if (options?.tunnel && !App.instance.bastion) {\n        App.instance.initialiseTunnel(options.tunnel);\n      }\n    }\n\n    return App.instance;\n  }\n\n  /**\n   * Retrieve a stack by key. If the stack does not exist, it will be created.\n   *  Dependencies are only applied the first time a stack is created.\n   *\n   * @param key - The key of the stack\n   * @param dependencies - The stack(s) that this stack depends on\n   * @returns {AwsStack}\n   */\n  public getStack(key: string, dependencies?: AwsStack | AwsStack[]): AwsStack {\n    // Apply the aspect once before creating the first stack\n    if (\n      !this.aspectApplied &&\n      Object.keys(this.stacks).length === 0 &&\n      Object.keys(this.globalTags).length > 0\n    ) {\n      this.applyTagsAspect();\n    }\n\n    if (!this.stacks[key]) {\n      this.stacks[key] = new AwsStack(key, dependencies);\n    }\n\n    return this.stacks[key];\n  }\n\n  /**\n   * Retrieve default compute stack - named as `${this.name}Compute`\n   *\n   * Only depends on Network. Database dependency is added automatically\n   * by CDK when compute resources reference database resources.\n   *\n   * @returns {AwsStack}\n   */\n  public getDefaultComputeStack(): AwsStack {\n    return this.getStack(\n      `${this.stackPrefix}Compute`,\n      this.getDefaultNetworkStack()\n    );\n  }\n\n  /**\n   * Retrieve default network stack - named as `${this.name}Network`\n   * @returns {AwsStack}\n   */\n  public getDefaultNetworkStack(): AwsStack {\n    return this.getStack(`${this.stackPrefix}Network`);\n  }\n\n  /**\n   * Retrieve default database stack - named as `${this.name}Database`\n   * @returns {AwsStack}\n   */\n  public getDefaultDatabaseStack(): AwsStack {\n    return this.getStack(\n      `${this.stackPrefix}Database`,\n      this.getDefaultNetworkStack()\n    );\n  }\n\n  /**\n   * Retrieve default storage stack - named as `${this.name}Storage`\n   * @returns {AwsStack}\n   */\n  public getDefaultStorageStack(): AwsStack {\n    return this.getStack(\n      `${this.stackPrefix}Storage`,\n      this.getDefaultNetworkStack()\n    );\n  }\n\n  /**\n   * Retrieve default CDN stack - named as `${this.name}Cdn`\n   *\n   * Depends on Network. Compute/Storage dependencies are added automatically\n   * by CDK when CDN resources reference ALB or S3 bucket resources.\n   *\n   * @returns {AwsStack}\n   */\n  public getDefaultCdnStack(): AwsStack {\n    return this.getStack(\n      `${this.stackPrefix}Cdn`,\n      this.getDefaultNetworkStack()\n    );\n  }\n\n  /**\n   * Retrieve default messaging stack - named as `${this.name}Messaging`\n   *\n   * Used for SQS queues, SNS topics, and EventBridge event buses.\n   * Depends on Network only. These are regional services that don't\n   * require VPC, but we maintain consistent stack dependency patterns.\n   *\n   * @returns {AwsStack}\n   */\n  public getDefaultMessagingStack(): AwsStack {\n    return this.getStack(\n      `${this.stackPrefix}Messaging`,\n      this.getDefaultNetworkStack()\n    );\n  }\n\n  /**\n   * Get a VPC by name. If no name is provided, returns the default VPC.\n   *\n   * This is a pure getter - it never creates infrastructure.\n   * Network must be configured via App.getApp() options or app.addNetwork().\n   *\n   * @param name - Optional name of the VPC to retrieve. If not provided, returns the default VPC.\n   * @returns {IVpc} The configured VPC\n   * @throws {Error} If network is disabled, not configured, or named VPC not found\n   */\n  public getVpc(name?: string): IVpc {\n    if (this.networkDisabled) {\n      throw new Error(\n        \"Network is disabled for this app. Cannot get VPC. \" +\n          \"Pass network config to App.getApp() to enable networking.\"\n      );\n    }\n\n    // If name is provided, look in additional VPCs first\n    if (name) {\n      const additionalVpc = this.additionalVpcs.get(name);\n      if (additionalVpc) {\n        return additionalVpc;\n      }\n      throw new Error(\n        `VPC '${name}' not found. Available VPCs: ${this.getVpcNames().join(\", \")}. ` +\n          \"Create additional VPCs using app.addNetwork(NetworkFactory.build(...)).\"\n      );\n    }\n\n    // Return default VPC\n    if (!this.vpc) {\n      throw new Error(\n        \"Network not configured. Pass network config to App.getApp(). \" +\n          \"Example: App.getApp(appName, { network: { maxAzs: 2 } })\"\n      );\n    }\n    return this.vpc;\n  }\n\n  /**\n   * Get the names of all available VPCs.\n   *\n   * @returns {string[]} Array of VPC names. Includes \"default\" if the default VPC is configured.\n   */\n  public getVpcNames(): string[] {\n    const names: string[] = [];\n    if (this.vpc) {\n      names.push(\"default\");\n    }\n    names.push(...this.additionalVpcs.keys());\n    return names;\n  }\n\n  /**\n   * Retrieve the default application container registry. If the registry does not exist\n   *  it will be created.\n   */\n  public getDefaultContainerRegistry(): Ecr {\n    if (!this.defaultEcr) {\n      const networkStack = this.getDefaultNetworkStack();\n\n      const ecr = EcrFactory.build(`${this.stackPrefix}Ecr`, {\n        repositoryName: toKebab(this.name)\n      })(this, networkStack.getStack());\n      this.defaultEcr = ecr;\n      return ecr;\n    }\n\n    return this.defaultEcr;\n  }\n\n  /**\n   * Create a cross-account audit role in the Network stack that allows\n   * the Fjall platform to use CloudQuery for comprehensive AWS resource auditing.\n   *\n   * This is an explicit opt-in feature. Call this method to enable CloudQuery\n   * audit capabilities for your application.\n   *\n   * @param webappAccountId - Optional AWS account ID of the Fjall webapp. Defaults to configured platform account.\n   * @param externalId - Optional external ID for additional security. If not provided, a unique ID will be generated.\n   * @returns {Role} The created audit role\n   */\n  public createAuditRole(webappAccountId?: string, externalId?: string): Role {\n    if (!this.defaultAuditRole) {\n      const networkStack = this.getDefaultNetworkStack();\n      const accountId =\n        webappAccountId || FJALL_AUDIT_CONFIG.webappAwsAccountId;\n\n      this.auditRoleExternalId = externalId || this.generateAuditExternalId();\n\n      const role = AuditRoleFactory.build(`${this.stackPrefix}AuditRole`, {\n        webappAccountId: accountId,\n        appName: this.name,\n        externalId: this.auditRoleExternalId,\n        roleNamePrefix: FJALL_AUDIT_CONFIG.roleNamePrefix,\n        rolePath: FJALL_AUDIT_CONFIG.rolePath\n      })(this, networkStack.getStack());\n\n      networkStack.addConstruct(role);\n      this.defaultAuditRole = role;\n      return role;\n    }\n\n    return this.defaultAuditRole;\n  }\n\n  /**\n   * Get the external ID used for the audit role\n   * @returns {string | undefined} The external ID if audit role has been created\n   */\n  public getAuditRoleExternalId(): string | undefined {\n    return this.auditRoleExternalId;\n  }\n\n  /**\n   * Generate a unique external ID for audit role\n   * Format: fjall-audit-{appName}-{timestamp}\n   */\n  private generateAuditExternalId(): string {\n    const uniqueId = randomBytes(8).toString(\"hex\");\n    return `fjall-audit-${toKebab(this.name)}-${uniqueId}`;\n  }\n\n  /**\n   * Add a compute resource to the default compute stack using the factory pattern.\n   *\n   * Returns the appropriate compute type based on the factory configuration:\n   * - EcsCompute for type: \"ecs\"\n   * - LambdaCompute for type: \"lambda\"\n   * - Ec2Compute for type: \"ec2\"\n   *\n   * @example\n   * // ECS compute - returns EcsCompute\n   * const api = app.addCompute(ComputeFactory.build(\"Api\", {\n   *   type: \"ecs\",\n   *   services: [{ name: \"api\", ... }]\n   * }));\n   * api.getLoadBalancer(); // Available on EcsCompute\n   *\n   * @example\n   * // Lambda compute - returns LambdaCompute\n   * const worker = app.addCompute(ComputeFactory.build(\"Worker\", {\n   *   type: \"lambda\",\n   *   functions: [{ name: \"process\", ... }]\n   * }));\n   * worker.getFunction(\"process\"); // Available on LambdaCompute\n   */\n  public addCompute<T extends AnyCompute>(\n    fn: (app: App, scope: Construct) => T\n  ): T {\n    const computeStack = this.getDefaultComputeStack();\n    const construct = fn(this, computeStack.getStack());\n    computeStack.addConstruct(construct as unknown as Construct);\n    return construct;\n  }\n\n  /**\n   * Manually add a resource to the default compute stack.\n   */\n  public addComputeResource(resource: Construct): void {\n    const computeStack = this.getDefaultComputeStack();\n    computeStack.addConstruct(resource);\n  }\n\n  /**\n   * Add a database resource to the default database stack using the factory pattern.\n   * Returns the appropriate database type based on the factory used.\n   *\n   * @example\n   * // Aurora database - returns RelationalDatabase\n   * const db = app.addDatabase(DatabaseFactory.build(\"Main\", {\n   *   type: \"Aurora\",\n   *   databaseName: \"my_app\"\n   * }));\n   * db.getHostEndpoint(); // Available\n   *\n   * @example\n   * // DynamoDB table - returns DynamoDBDatabase\n   * const cache = app.addDatabase(DatabaseFactory.build(\"Cache\", {\n   *   type: \"DynamoDB\",\n   *   partitionKey: { name: \"id\", type: \"S\" }\n   * }));\n   * cache.getTableName(); // Available\n   */\n  public addDatabase<T extends AnyDatabase>(\n    fn: (app: App, scope: Construct) => T\n  ): T {\n    const databaseStack = this.getDefaultDatabaseStack();\n    const databaseConstruct = fn(this, databaseStack.getStack());\n    databaseStack.addConstruct(databaseConstruct as unknown as Construct);\n\n    // Wire bastion → database security group for tunnel access\n    if (this.bastion && this.isConnectable(databaseConstruct)) {\n      databaseConstruct.connections.allowDefaultPortFrom(this.bastion);\n    }\n\n    return databaseConstruct;\n  }\n\n  /**\n   * Type guard for IConnectable — checks if a construct has a connections property.\n   */\n  private isConnectable(construct: unknown): construct is IConnectable {\n    return (\n      typeof construct === \"object\" &&\n      construct !== null &&\n      \"connections\" in construct &&\n      typeof (construct as IConnectable).connections === \"object\"\n    );\n  }\n\n  /**\n   * Add a storage resource (S3) to the default storage stack using the factory pattern.\n   *\n   * @example\n   * // Private bucket (default)\n   * const assets = app.addStorage(StorageFactory.build(\"Assets\", {\n   *   versioned: true\n   * }));\n   *\n   * @example\n   * // Website bucket\n   * const site = app.addStorage(StorageFactory.build(\"Site\", {\n   *   websiteHosting: { indexDocument: \"index.html\" }\n   * }));\n   */\n  public addStorage(fn: StorageFactoryFn): Storage {\n    const placement = fn.stackPlacement ?? \"storage\";\n    const stack = this.resolveStorageStack(placement);\n    const storageConstruct = fn(this, stack.getStack());\n    stack.addConstruct(storageConstruct);\n    return storageConstruct;\n  }\n\n  private resolveStorageStack(placement: string): AwsStack {\n    switch (placement) {\n      case \"cdn\":\n        return this.getDefaultCdnStack();\n      case \"compute\":\n        return this.getDefaultComputeStack();\n      default:\n        return this.getDefaultStorageStack();\n    }\n  }\n\n  /**\n   * Add a CDN resource (CloudFront) to the default CDN stack using the factory pattern.\n   *\n   * @example\n   * // S3 origin\n   * const assets = app.addStorage(StorageFactory.build(\"Assets\", {}));\n   * app.addCdn(CdnFactory.build(\"AssetsCdn\", {\n   *   originType: \"s3\",\n   *   bucket: assets\n   * }));\n   *\n   * @example\n   * // ALB origin (ECS)\n   * const api = app.addCompute(ComputeFactory.build(\"Api\", { type: \"ecs\", ... }));\n   * app.addCdn(CdnFactory.build(\"ApiCdn\", {\n   *   originType: \"alb\",\n   *   loadBalancer: api\n   * }));\n   */\n  public addCdn(fn: (app: App, scope: Construct) => Cdn): Cdn {\n    const cdnStack = this.getDefaultCdnStack();\n    const cdn = fn(this, cdnStack.getStack());\n    cdnStack.addConstruct(cdn);\n\n    return cdn;\n  }\n\n  /**\n   * Add a messaging resource to the default queue stack using the factory pattern.\n   *\n   * Returns the appropriate messaging type based on the factory configuration.\n   * Currently supports queue (SQS), with future support planned for\n   * topic (SNS), eventbus (EventBridge), and stream (Kinesis).\n   *\n   * @example\n   * // Queue messaging - returns QueueMessaging\n   * const notifications = app.addMessaging(MessagingFactory.build(\"Notifications\", {\n   *   type: \"queue\",\n   *   queueType: \"standard\",\n   *   deadLetterQueue: { enabled: true, maxReceiveCount: 3 }\n   * }));\n   * notifications.grantSendMessages(apiFunction);\n   *\n   * @example\n   * // FIFO queue for OpenNext revalidation\n   * const revalidation = app.addMessaging(MessagingFactory.build(\"Revalidation\", {\n   *   type: \"queue\",\n   *   queueType: \"fifo\",\n   *   visibilityTimeout: 300,\n   *   contentBasedDeduplication: true\n   * }));\n   */\n  public addMessaging<T extends AnyMessaging>(\n    fn: (app: App, scope: Construct) => T\n  ): T {\n    const messagingStack = this.getDefaultMessagingStack();\n    const messaging = fn(this, messagingStack.getStack());\n    messagingStack.addConstruct(messaging as unknown as Construct);\n\n    return messaging;\n  }\n\n  /**\n   * Add a high-level infrastructure pattern to the application.\n   *\n   * Patterns are composite constructs that create multiple related resources\n   * as a single, cohesive unit. They encapsulate best practices and reduce\n   * boilerplate for common deployment scenarios.\n   *\n   * Currently supported patterns:\n   * - `payload`: Payload CMS deployment with OpenNext\n   *\n   * Returns the appropriate pattern interface based on the factory configuration,\n   * providing access to all underlying resources via escape hatches.\n   *\n   * @example\n   * // Payload CMS pattern - returns Payload\n   * const payload = app.addPattern(PatternFactory.build(\"PayloadApp\", {\n   *   type: \"payload\",\n   *   name: \"my-cms\",\n   *   source: \"../..\"\n   * }));\n   *\n   * @example\n   * // Escape hatches - access underlying resources\n   * payload.getDatabase().grantConnect(otherLambda);\n   * payload.getServer().getLambdaFunction().addEnvironment(\"CUSTOM\", \"value\");\n   * payload.getCdn().getDistribution().addBehavior(\"/custom/*\", customOrigin);\n   */\n  public addPattern<T extends AnyPattern>(\n    fn: (app: App, scope: Construct) => T\n  ): T {\n    const computeStack = this.getDefaultComputeStack();\n    const pattern = fn(this, computeStack.getStack());\n    computeStack.addConstruct(pattern as unknown as Construct);\n    return pattern;\n  }\n\n  /**\n   * Add an additional network (VPC) to the application.\n   *\n   * Use this to create additional VPCs beyond the default VPC configured via App.getApp().\n   * Additional VPCs can be retrieved by name using app.getVpc(name).\n   *\n   * @param fn - Factory function that creates the Network construct\n   * @returns {Network} The created Network construct\n   *\n   * @example\n   * const isolatedVpc = app.addNetwork(\n   *   NetworkFactory.build(\"IsolatedVpc\", { maxAzs: 2, natGateways: false })\n   * );\n   */\n  public addNetwork(fn: (app: App, scope: Construct) => Network): Network {\n    const networkStack = this.getDefaultNetworkStack();\n    const network = fn(this, networkStack.getStack());\n    const vpcName = network.node.id;\n\n    // Store in additional VPCs map for retrieval via getVpc(name)\n    this.additionalVpcs.set(vpcName, network.getVpc());\n    networkStack.addConstruct(network);\n\n    return network;\n  }\n\n  /**\n   * Manually add a resource to the default database stack.\n   */\n  public addDatabaseResource(resource: Construct): void {\n    const databaseStack = this.getDefaultDatabaseStack();\n    databaseStack.addConstruct(resource);\n  }\n\n  /**\n   * Manually add a resource to the default storage stack.\n   */\n  public addStorageResource(resource: Construct): void {\n    const storageStack = this.getDefaultStorageStack();\n    storageStack.addConstruct(resource);\n  }\n\n  /**\n   * Initialise standard tags\n   */\n  private initialiseStandardTags(): void {\n    const config = getConfig();\n    this.globalTags = {\n      [COST_ALLOCATION_ENVIRONMENT_TAG]: config.environment,\n      [COST_ALLOCATION_SERVICE_TAG]: this.name\n    };\n  }\n\n  /**\n   * Apply all tags using CDK's native Tags.of().add() API\n   */\n  private applyTagsAspect(): void {\n    if (!this.aspectApplied && Object.keys(this.globalTags).length > 0) {\n      // Apply standard tags using Tags.of(this).add()\n      for (const [key, value] of Object.entries(this.globalTags)) {\n        Tags.of(this).add(key, value);\n      }\n\n      // Apply aspect for resource-specific tags (IPAM pool, backup tier)\n      Aspects.of(this).add(new StandardTagsAspect());\n\n      this.aspectApplied = true;\n    }\n  }\n\n  /**\n   * Add custom tags to all resources in the app.\n   *\n   * @param tags Custom tags to apply to all resources\n   * @example\n   * app.addTags({\n   *   \"fjall:costAllocation:owner\": \"platform-team\",\n   *   \"fjall:costAllocation:cost-center\": \"CC-123\",\n   *   \"team:slack-channel\": \"#platform-alerts\"\n   * });\n   */\n  public addTags(tags: { [key: string]: string }): App {\n    // Merge tags into globalTags for tracking\n    this.globalTags = {\n      ...this.globalTags,\n      ...tags\n    };\n    for (const [key, value] of Object.entries(tags)) {\n      Tags.of(this).add(key, value);\n    }\n\n    return this;\n  }\n\n  /**\n   * Export resource inventory collected during synthesis\n   * Call this after app.synth() to get complete resource inventory\n   * @returns {ResourceInventory} Complete inventory of all resources in the app\n   */\n  public exportResourceInventory(): ResourceInventory {\n    if (!this.resourceInventory) {\n      throw new Error(\n        \"Resource inventory not available. Ensure a ResourceInventoryAspect has been added before calling exportResourceInventory().\"\n      );\n    }\n    return {\n      resources: this.resourceInventory.getInventory(),\n      generatedAt: new Date().toISOString(),\n      appName: this.name\n    };\n  }\n\n  /**\n   * Get the manifest collector for registering services and patterns.\n   * Used by ComputeFactory and PatternFactory to register their configurations.\n   */\n  public getManifestCollector(): ManifestCollector {\n    return this.manifestCollector;\n  }\n\n  /**\n   * Override synth to automatically export resource inventory and manifest\n   */\n  public synth(options?: StageSynthesisOptions): CloudAssembly {\n    // Call parent synth first\n    const assembly = super.synth(options);\n\n    // After synthesis, write Fjall manifest to cdk.out\n    try {\n      writeManifest(assembly, this.manifestCollector);\n    } catch (error) {\n      // Don't fail synth if manifest export fails\n      const errorMessage =\n        error instanceof Error ? error.message : String(error);\n      FjallLogger.warn(`Failed to export Fjall manifest: ${errorMessage}`);\n    }\n\n    return assembly;\n  }\n}\n\nexport default App;\n"]}
@@ -0,0 +1 @@
1
+ export { ResourceInventoryAspect, type ResourceMetadata, type ResourceInventory } from "./resourceInventory";
@@ -0,0 +1,6 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.ResourceInventoryAspect = void 0;
4
+ var resourceInventory_1 = require("./resourceInventory");
5
+ Object.defineProperty(exports, "ResourceInventoryAspect", { enumerable: true, get: function () { return resourceInventory_1.ResourceInventoryAspect; } });
6
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi9saWIvYXNwZWN0cy9pbmRleC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7QUFBQSx5REFJNkI7QUFIM0IsNEhBQUEsdUJBQXVCLE9BQUEiLCJzb3VyY2VzQ29udGVudCI6WyJleHBvcnQge1xuICBSZXNvdXJjZUludmVudG9yeUFzcGVjdCxcbiAgdHlwZSBSZXNvdXJjZU1ldGFkYXRhLFxuICB0eXBlIFJlc291cmNlSW52ZW50b3J5XG59IGZyb20gXCIuL3Jlc291cmNlSW52ZW50b3J5XCI7XG4iXX0=
@@ -0,0 +1,20 @@
1
+ import { Role } from "aws-cdk-lib/aws-iam";
2
+ import { Construct } from "constructs";
3
+ export interface AccountAuditRoleProps {
4
+ /**
5
+ * The Fjall organisation ID. Used as a suffix in the role name
6
+ * (`FjallAudit{orgId}`) and as the ExternalId condition.
7
+ */
8
+ fjallOrgId: string;
9
+ }
10
+ /**
11
+ * Per-account audit role for the Fjall platform.
12
+ *
13
+ * Creates a `FjallAudit{orgId}` IAM role with ReadOnlyAccess and SecurityAudit
14
+ * managed policies, trusting the Fjall platform account. Only instantiated when
15
+ * a `fjallOrgId` context value is provided to the Account stack.
16
+ */
17
+ export declare class AccountAuditRole extends Construct {
18
+ readonly role: Role;
19
+ constructor(scope: Construct, id: string, props: AccountAuditRoleProps);
20
+ }
@@ -0,0 +1,38 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.AccountAuditRole = void 0;
4
+ const aws_cdk_lib_1 = require("aws-cdk-lib");
5
+ const aws_iam_1 = require("aws-cdk-lib/aws-iam");
6
+ const constructs_1 = require("constructs");
7
+ const platform_1 = require("./platform");
8
+ /**
9
+ * Per-account audit role for the Fjall platform.
10
+ *
11
+ * Creates a `FjallAudit{orgId}` IAM role with ReadOnlyAccess and SecurityAudit
12
+ * managed policies, trusting the Fjall platform account. Only instantiated when
13
+ * a `fjallOrgId` context value is provided to the Account stack.
14
+ */
15
+ class AccountAuditRole extends constructs_1.Construct {
16
+ constructor(scope, id, props) {
17
+ super(scope, id);
18
+ this.role = new aws_iam_1.Role(this, "Role", {
19
+ roleName: `FjallAudit${props.fjallOrgId}`,
20
+ path: "/",
21
+ assumedBy: new aws_iam_1.AccountPrincipal(platform_1.FJALL_PLATFORM_ACCOUNT_ID),
22
+ description: `Cross-account audit role for Fjall organisation ${props.fjallOrgId}. Grants read-only access for asset discovery and compliance auditing.`,
23
+ externalIds: [props.fjallOrgId],
24
+ managedPolicies: [
25
+ aws_iam_1.ManagedPolicy.fromAwsManagedPolicyName("ReadOnlyAccess"),
26
+ aws_iam_1.ManagedPolicy.fromAwsManagedPolicyName("SecurityAudit")
27
+ ]
28
+ });
29
+ new aws_cdk_lib_1.CfnOutput(this, "FjallAuditRoleArn", {
30
+ key: "FjallAuditRoleArn",
31
+ value: this.role.roleArn,
32
+ description: `ARN of the Fjall audit role for organisation ${props.fjallOrgId}`,
33
+ exportName: "FjallAuditRoleArn"
34
+ });
35
+ }
36
+ }
37
+ exports.AccountAuditRole = AccountAuditRole;
38
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYWNjb3VudEF1ZGl0Um9sZS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uL2xpYi9jb25maWcvYXdzL2FjY291bnRBdWRpdFJvbGUudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7O0FBQUEsNkNBQXdDO0FBQ3hDLGlEQUE0RTtBQUM1RSwyQ0FBdUM7QUFDdkMseUNBQXVEO0FBVXZEOzs7Ozs7R0FNRztBQUNILE1BQWEsZ0JBQWlCLFNBQVEsc0JBQVM7SUFHN0MsWUFBWSxLQUFnQixFQUFFLEVBQVUsRUFBRSxLQUE0QjtRQUNwRSxLQUFLLENBQUMsS0FBSyxFQUFFLEVBQUUsQ0FBQyxDQUFDO1FBRWpCLElBQUksQ0FBQyxJQUFJLEdBQUcsSUFBSSxjQUFJLENBQUMsSUFBSSxFQUFFLE1BQU0sRUFBRTtZQUNqQyxRQUFRLEVBQUUsYUFBYSxLQUFLLENBQUMsVUFBVSxFQUFFO1lBQ3pDLElBQUksRUFBRSxHQUFHO1lBQ1QsU0FBUyxFQUFFLElBQUksMEJBQWdCLENBQUMsb0NBQXlCLENBQUM7WUFDMUQsV0FBVyxFQUFFLG1EQUFtRCxLQUFLLENBQUMsVUFBVSx3RUFBd0U7WUFDeEosV0FBVyxFQUFFLENBQUMsS0FBSyxDQUFDLFVBQVUsQ0FBQztZQUMvQixlQUFlLEVBQUU7Z0JBQ2YsdUJBQWEsQ0FBQyx3QkFBd0IsQ0FBQyxnQkFBZ0IsQ0FBQztnQkFDeEQsdUJBQWEsQ0FBQyx3QkFBd0IsQ0FBQyxlQUFlLENBQUM7YUFDeEQ7U0FDRixDQUFDLENBQUM7UUFFSCxJQUFJLHVCQUFTLENBQUMsSUFBSSxFQUFFLG1CQUFtQixFQUFFO1lBQ3ZDLEdBQUcsRUFBRSxtQkFBbUI7WUFDeEIsS0FBSyxFQUFFLElBQUksQ0FBQyxJQUFJLENBQUMsT0FBTztZQUN4QixXQUFXLEVBQUUsZ0RBQWdELEtBQUssQ0FBQyxVQUFVLEVBQUU7WUFDL0UsVUFBVSxFQUFFLG1CQUFtQjtTQUNoQyxDQUFDLENBQUM7SUFDTCxDQUFDO0NBQ0Y7QUF6QkQsNENBeUJDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgQ2ZuT3V0cHV0IH0gZnJvbSBcImF3cy1jZGstbGliXCI7XG5pbXBvcnQgeyBSb2xlLCBBY2NvdW50UHJpbmNpcGFsLCBNYW5hZ2VkUG9saWN5IH0gZnJvbSBcImF3cy1jZGstbGliL2F3cy1pYW1cIjtcbmltcG9ydCB7IENvbnN0cnVjdCB9IGZyb20gXCJjb25zdHJ1Y3RzXCI7XG5pbXBvcnQgeyBGSkFMTF9QTEFURk9STV9BQ0NPVU5UX0lEIH0gZnJvbSBcIi4vcGxhdGZvcm1cIjtcblxuZXhwb3J0IGludGVyZmFjZSBBY2NvdW50QXVkaXRSb2xlUHJvcHMge1xuICAvKipcbiAgICogVGhlIEZqYWxsIG9yZ2FuaXNhdGlvbiBJRC4gVXNlZCBhcyBhIHN1ZmZpeCBpbiB0aGUgcm9sZSBuYW1lXG4gICAqIChgRmphbGxBdWRpdHtvcmdJZH1gKSBhbmQgYXMgdGhlIEV4dGVybmFsSWQgY29uZGl0aW9uLlxuICAgKi9cbiAgZmphbGxPcmdJZDogc3RyaW5nO1xufVxuXG4vKipcbiAqIFBlci1hY2NvdW50IGF1ZGl0IHJvbGUgZm9yIHRoZSBGamFsbCBwbGF0Zm9ybS5cbiAqXG4gKiBDcmVhdGVzIGEgYEZqYWxsQXVkaXR7b3JnSWR9YCBJQU0gcm9sZSB3aXRoIFJlYWRPbmx5QWNjZXNzIGFuZCBTZWN1cml0eUF1ZGl0XG4gKiBtYW5hZ2VkIHBvbGljaWVzLCB0cnVzdGluZyB0aGUgRmphbGwgcGxhdGZvcm0gYWNjb3VudC4gT25seSBpbnN0YW50aWF0ZWQgd2hlblxuICogYSBgZmphbGxPcmdJZGAgY29udGV4dCB2YWx1ZSBpcyBwcm92aWRlZCB0byB0aGUgQWNjb3VudCBzdGFjay5cbiAqL1xuZXhwb3J0IGNsYXNzIEFjY291bnRBdWRpdFJvbGUgZXh0ZW5kcyBDb25zdHJ1Y3Qge1xuICBwdWJsaWMgcmVhZG9ubHkgcm9sZTogUm9sZTtcblxuICBjb25zdHJ1Y3RvcihzY29wZTogQ29uc3RydWN0LCBpZDogc3RyaW5nLCBwcm9wczogQWNjb3VudEF1ZGl0Um9sZVByb3BzKSB7XG4gICAgc3VwZXIoc2NvcGUsIGlkKTtcblxuICAgIHRoaXMucm9sZSA9IG5ldyBSb2xlKHRoaXMsIFwiUm9sZVwiLCB7XG4gICAgICByb2xlTmFtZTogYEZqYWxsQXVkaXQke3Byb3BzLmZqYWxsT3JnSWR9YCxcbiAgICAgIHBhdGg6IFwiL1wiLFxuICAgICAgYXNzdW1lZEJ5OiBuZXcgQWNjb3VudFByaW5jaXBhbChGSkFMTF9QTEFURk9STV9BQ0NPVU5UX0lEKSxcbiAgICAgIGRlc2NyaXB0aW9uOiBgQ3Jvc3MtYWNjb3VudCBhdWRpdCByb2xlIGZvciBGamFsbCBvcmdhbmlzYXRpb24gJHtwcm9wcy5mamFsbE9yZ0lkfS4gR3JhbnRzIHJlYWQtb25seSBhY2Nlc3MgZm9yIGFzc2V0IGRpc2NvdmVyeSBhbmQgY29tcGxpYW5jZSBhdWRpdGluZy5gLFxuICAgICAgZXh0ZXJuYWxJZHM6IFtwcm9wcy5mamFsbE9yZ0lkXSxcbiAgICAgIG1hbmFnZWRQb2xpY2llczogW1xuICAgICAgICBNYW5hZ2VkUG9saWN5LmZyb21Bd3NNYW5hZ2VkUG9saWN5TmFtZShcIlJlYWRPbmx5QWNjZXNzXCIpLFxuICAgICAgICBNYW5hZ2VkUG9saWN5LmZyb21Bd3NNYW5hZ2VkUG9saWN5TmFtZShcIlNlY3VyaXR5QXVkaXRcIilcbiAgICAgIF1cbiAgICB9KTtcblxuICAgIG5ldyBDZm5PdXRwdXQodGhpcywgXCJGamFsbEF1ZGl0Um9sZUFyblwiLCB7XG4gICAgICBrZXk6IFwiRmphbGxBdWRpdFJvbGVBcm5cIixcbiAgICAgIHZhbHVlOiB0aGlzLnJvbGUucm9sZUFybixcbiAgICAgIGRlc2NyaXB0aW9uOiBgQVJOIG9mIHRoZSBGamFsbCBhdWRpdCByb2xlIGZvciBvcmdhbmlzYXRpb24gJHtwcm9wcy5mamFsbE9yZ0lkfWAsXG4gICAgICBleHBvcnROYW1lOiBcIkZqYWxsQXVkaXRSb2xlQXJuXCJcbiAgICB9KTtcbiAgfVxufVxuIl19