@robhan-cdk-lib/aws_grafana 0.0.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.
@@ -0,0 +1,694 @@
1
+ "use strict";
2
+ var _a;
3
+ Object.defineProperty(exports, "__esModule", { value: true });
4
+ exports.Workspace = exports.Status = exports.SamlConfigurationStatuses = exports.PermissionTypes = exports.NotificationDestinations = exports.AuthenticationProviders = exports.AccountAccessType = void 0;
5
+ const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
6
+ const aws_grafana_1 = require("aws-cdk-lib/aws-grafana");
7
+ const constructs_1 = require("constructs");
8
+ /**
9
+ * Specifies whether the workspace can access AWS resources in this AWS account only, or whether it
10
+ * can also access AWS resources in other accounts in the same organization. If this is
11
+ * ORGANIZATION, the OrganizationalUnits parameter specifies which organizational units the
12
+ * workspace can access.
13
+ */
14
+ var AccountAccessType;
15
+ (function (AccountAccessType) {
16
+ /**
17
+ * Access is limited to the current AWS account only.
18
+ */
19
+ AccountAccessType["CURRENT_ACCOUNT"] = "CURRENT_ACCOUNT";
20
+ /**
21
+ * Access is extended to the entire AWS organization.
22
+ */
23
+ AccountAccessType["ORGANIZATION"] = "ORGANIZATION";
24
+ })(AccountAccessType || (exports.AccountAccessType = AccountAccessType = {}));
25
+ /**
26
+ * Specifies whether this workspace uses SAML 2.0, AWS IAM Identity Center, or both to authenticate
27
+ * users for using the Grafana console within a workspace.
28
+ *
29
+ * @see https://docs.aws.amazon.com/grafana/latest/APIReference/API_CreateWorkspace.html
30
+ */
31
+ var AuthenticationProviders;
32
+ (function (AuthenticationProviders) {
33
+ /**
34
+ * AWS Single Sign-On authentication provider.
35
+ */
36
+ AuthenticationProviders["AWS_SSO"] = "AWS_SSO";
37
+ /**
38
+ * Security Assertion Markup Language (SAML) authentication provider.
39
+ */
40
+ AuthenticationProviders["SAML"] = "SAML";
41
+ })(AuthenticationProviders || (exports.AuthenticationProviders = AuthenticationProviders = {}));
42
+ /**
43
+ * The AWS notification channels that Amazon Managed Grafana can automatically create IAM roles and
44
+ * permissions for, to allow Amazon Managed Grafana to use these channels.
45
+ */
46
+ var NotificationDestinations;
47
+ (function (NotificationDestinations) {
48
+ /**
49
+ * Amazon Simple Notification Service (SNS) as notification destination.
50
+ */
51
+ NotificationDestinations["SNS"] = "SNS";
52
+ })(NotificationDestinations || (exports.NotificationDestinations = NotificationDestinations = {}));
53
+ /**
54
+ * If this is SERVICE_MANAGED, and the workplace was created through the Amazon Managed Grafana
55
+ * console, then Amazon Managed Grafana automatically creates the IAM roles and provisions the
56
+ * permissions that the workspace needs to use AWS data sources and notification channels.
57
+ *
58
+ * If this is CUSTOMER_MANAGED, you must manage those roles and permissions yourself.
59
+
60
+ * If you are working with a workspace in a member account of an organization and that account is
61
+ * not a delegated administrator account, and you want the workspace to access data sources in
62
+ * other AWS accounts in the organization, this parameter must be set to CUSTOMER_MANAGED.
63
+ */
64
+ var PermissionTypes;
65
+ (function (PermissionTypes) {
66
+ /**
67
+ * Customer-managed permissions where you manage user access to Grafana.
68
+ */
69
+ PermissionTypes["CUSTOMER_MANAGED"] = "CUSTOMER_MANAGED";
70
+ /**
71
+ * Service-managed permissions where AWS manages user access to Grafana.
72
+ */
73
+ PermissionTypes["SERVICE_MANAGED"] = "SERVICE_MANAGED";
74
+ })(PermissionTypes || (exports.PermissionTypes = PermissionTypes = {}));
75
+ /**
76
+ * Status of SAML configuration for a Grafana workspace.
77
+ */
78
+ var SamlConfigurationStatuses;
79
+ (function (SamlConfigurationStatuses) {
80
+ /**
81
+ * SAML is configured for the workspace.
82
+ */
83
+ SamlConfigurationStatuses["CONFIGURED"] = "CONFIGURED";
84
+ /**
85
+ * SAML is not configured for the workspace.
86
+ */
87
+ SamlConfigurationStatuses["NOT_CONFIGURED"] = "NOT_CONFIGURED";
88
+ })(SamlConfigurationStatuses || (exports.SamlConfigurationStatuses = SamlConfigurationStatuses = {}));
89
+ /**
90
+ * Status of a Grafana workspace.
91
+ */
92
+ var Status;
93
+ (function (Status) {
94
+ /**
95
+ * Workspace is active and ready to use.
96
+ */
97
+ Status["ACTIVE"] = "ACTIVE";
98
+ /**
99
+ * Workspace is being created.
100
+ */
101
+ Status["CREATING"] = "CREATING";
102
+ /**
103
+ * Workspace is being deleted.
104
+ */
105
+ Status["DELETING"] = "DELETING";
106
+ /**
107
+ * Workspace operation has failed.
108
+ */
109
+ Status["FAILED"] = "FAILED";
110
+ /**
111
+ * Workspace is being updated.
112
+ */
113
+ Status["UPDATING"] = "UPDATING";
114
+ /**
115
+ * Workspace is being upgraded.
116
+ */
117
+ Status["UPGRADING"] = "UPGRADING";
118
+ /**
119
+ * Workspace deletion has failed.
120
+ */
121
+ Status["DELETION_FAILED"] = "DELETION_FAILED";
122
+ /**
123
+ * Workspace creation has failed.
124
+ */
125
+ Status["CREATION_FAILED"] = "CREATION_FAILED";
126
+ /**
127
+ * Workspace update has failed.
128
+ */
129
+ Status["UPDATE_FAILED"] = "UPDATE_FAILED";
130
+ /**
131
+ * Workspace upgrade has failed.
132
+ */
133
+ Status["UPGRADE_FAILED"] = "UPGRADE_FAILED";
134
+ /**
135
+ * License removal has failed.
136
+ */
137
+ Status["LICENSE_REMOVAL_FAILED"] = "LICENSE_REMOVAL_FAILED";
138
+ })(Status || (exports.Status = Status = {}));
139
+ /**
140
+ * Represents an Amazon Managed Grafana workspace.
141
+ *
142
+ * This class provides a high-level abstraction for creating and managing
143
+ * Amazon Managed Grafana workspaces using AWS CDK.
144
+ */
145
+ class Workspace extends constructs_1.Construct {
146
+ /**
147
+ * Validates the clientToken property.
148
+ *
149
+ * @param token - The client token to validate
150
+ * @returns An array of error messages if validation fails, or an empty array if valid
151
+ *
152
+ * Validation rules:
153
+ * - Must be a string
154
+ * - Must be between 1 and 64 characters long
155
+ * - Must contain only printable ASCII characters
156
+ */
157
+ static validateClientToken(token) {
158
+ const errors = [];
159
+ if (token === undefined) {
160
+ return errors; // Optional property can be undefined
161
+ }
162
+ if (typeof token !== 'string') {
163
+ errors.push('clientToken must be a string');
164
+ return errors; // No need to check further if not a string
165
+ }
166
+ const regex = /^[!-~]{1,64}$/;
167
+ if (!regex.test(token)) {
168
+ if (token.length < 1 || token.length > 64) {
169
+ errors.push('clientToken must be between 1 and 64 characters long');
170
+ }
171
+ if (!/^[!-~]*$/.test(token)) {
172
+ errors.push('clientToken must contain only printable ASCII characters');
173
+ }
174
+ }
175
+ return errors;
176
+ }
177
+ /**
178
+ * Validates the description property.
179
+ *
180
+ * @param description - The description to validate
181
+ * @returns An array of error messages if validation fails, or an empty array if valid
182
+ *
183
+ * Validation rules:
184
+ * - Must be a string
185
+ * - Maximum length of 2048 characters
186
+ */
187
+ static validateDescription(description) {
188
+ const errors = [];
189
+ if (description === undefined) {
190
+ return errors; // Optional property can be undefined
191
+ }
192
+ if (typeof description !== 'string') {
193
+ errors.push('description must be a string');
194
+ return errors; // No need to check further if not a string
195
+ }
196
+ if (description.length > 2048) {
197
+ errors.push('description cannot exceed 2048 characters');
198
+ }
199
+ return errors;
200
+ }
201
+ /**
202
+ * Validates the grafanaVersion property.
203
+ *
204
+ * @param version - The Grafana version to validate
205
+ * @returns An array of error messages if validation fails, or an empty array if valid
206
+ *
207
+ * Validation rules:
208
+ * - Must be a string
209
+ * - Must be between 1 and 255 characters long
210
+ */
211
+ static validateGrafanaVersion(version) {
212
+ const errors = [];
213
+ if (version === undefined) {
214
+ return errors; // Optional property can be undefined
215
+ }
216
+ if (typeof version !== 'string') {
217
+ errors.push('grafanaVersion must be a string');
218
+ return errors; // No need to check further if not a string
219
+ }
220
+ if (version.length < 1) {
221
+ errors.push('grafanaVersion cannot be empty');
222
+ }
223
+ if (version.length > 255) {
224
+ errors.push('grafanaVersion cannot exceed 255 characters');
225
+ }
226
+ return errors;
227
+ }
228
+ /**
229
+ * Validates the name property.
230
+ *
231
+ * @param name - The workspace name to validate
232
+ * @returns An array of error messages if validation fails, or an empty array if valid
233
+ *
234
+ * Validation rules:
235
+ * - Must be a string
236
+ * - Must be between 1 and 255 characters long
237
+ * - Can only contain alphanumeric characters, hyphens, dots, underscores, and tildes
238
+ */
239
+ static validateName(name) {
240
+ const errors = [];
241
+ if (name === undefined) {
242
+ return errors; // Optional property can be undefined
243
+ }
244
+ if (typeof name !== 'string') {
245
+ errors.push('name must be a string');
246
+ return errors; // No need to check further if not a string
247
+ }
248
+ const regex = /^[a-zA-Z0-9\-._~]{1,255}$/;
249
+ if (!regex.test(name)) {
250
+ if (name.length < 1 || name.length > 255) {
251
+ errors.push('name must be between 1 and 255 characters long');
252
+ }
253
+ if (!/^[a-zA-Z0-9\-._~]*$/.test(name)) {
254
+ errors.push('name can only contain alphanumeric characters, hyphens, dots, underscores, and tildes');
255
+ }
256
+ }
257
+ return errors;
258
+ }
259
+ /**
260
+ * Validates the networkAccessControl property.
261
+ *
262
+ * @param nac - The network access control configuration to validate
263
+ * @returns An array of error messages if validation fails, or an empty array if valid
264
+ *
265
+ * Validation rules:
266
+ * - Must be an object
267
+ * - prefixLists (if present) must be an array with at most 5 items
268
+ * - vpcEndpoints (if present) must be an array with at most 5 items
269
+ */
270
+ static validateNetworkAccessControl(nac) {
271
+ const errors = [];
272
+ if (nac === undefined) {
273
+ return errors; // Optional property can be undefined
274
+ }
275
+ if (!nac || typeof nac !== 'object') {
276
+ errors.push('networkAccessControl must be an object');
277
+ return errors;
278
+ }
279
+ const networkAccessControl = nac;
280
+ // Check prefixLists if present
281
+ if (networkAccessControl.prefixLists !== undefined) {
282
+ if (!Array.isArray(networkAccessControl.prefixLists)) {
283
+ errors.push('prefixLists must be an array');
284
+ }
285
+ else if (networkAccessControl.prefixLists.length > 5) {
286
+ errors.push('prefixLists can have at most 5 items');
287
+ }
288
+ }
289
+ // Check vpcEndpoints if present
290
+ if (networkAccessControl.vpcEndpoints !== undefined) {
291
+ if (!Array.isArray(networkAccessControl.vpcEndpoints)) {
292
+ errors.push('vpcEndpoints must be an array');
293
+ }
294
+ else if (networkAccessControl.vpcEndpoints.length > 5) {
295
+ errors.push('vpcEndpoints can have at most 5 items');
296
+ }
297
+ }
298
+ return errors;
299
+ }
300
+ /**
301
+ * Validates the organizationRoleName property.
302
+ *
303
+ * @param roleName - The organization role name to validate
304
+ * @returns An array of error messages if validation fails, or an empty array if valid
305
+ *
306
+ * Validation rules:
307
+ * - Must be a string
308
+ * - Must be between 1 and 2048 characters long
309
+ */
310
+ static validateOrganizationRoleName(roleName) {
311
+ const errors = [];
312
+ if (roleName === undefined) {
313
+ return errors; // Optional property can be undefined
314
+ }
315
+ if (typeof roleName !== 'string') {
316
+ errors.push('organizationRoleName must be a string');
317
+ return errors; // No need to check further if not a string
318
+ }
319
+ if (roleName.length < 1) {
320
+ errors.push('organizationRoleName cannot be empty');
321
+ }
322
+ if (roleName.length > 2048) {
323
+ errors.push('organizationRoleName cannot exceed 2048 characters');
324
+ }
325
+ return errors;
326
+ }
327
+ /**
328
+ * Validates the SAML assertion attributes.
329
+ *
330
+ * @param obj - The SAML assertion attributes to validate
331
+ * @returns An array of error messages if validation fails, or an empty array if valid
332
+ *
333
+ * Validation rules:
334
+ * - Must be an object
335
+ * - Each attribute must be a string
336
+ * - Each attribute must be between 1 and 256 characters long
337
+ * - Valid attribute keys are: 'email', 'groups', 'login', 'name', 'org', 'role'
338
+ */
339
+ static validateSamlAssertionAttributes(obj) {
340
+ const errors = [];
341
+ if (!obj || typeof obj !== 'object') {
342
+ return ['Input is not an object'];
343
+ }
344
+ const attributes = obj;
345
+ for (const key in attributes) {
346
+ const value = attributes[key];
347
+ if (value === undefined) {
348
+ continue; // Optional properties can be undefined
349
+ }
350
+ if (typeof value !== 'string') {
351
+ errors.push(`Property '${key}' must be a string`);
352
+ }
353
+ else if (value.length < 1) {
354
+ errors.push(`Property '${key}' cannot be empty`);
355
+ }
356
+ else if (value.length > 256) {
357
+ errors.push(`Property '${key}' exceeds maximum length of 256 characters`);
358
+ }
359
+ }
360
+ return errors;
361
+ }
362
+ /**
363
+ * Validates the SAML IdP metadata.
364
+ *
365
+ * @param obj - The SAML IdP metadata to validate
366
+ * @returns An array of error messages if validation fails, or an empty array if valid
367
+ *
368
+ * Validation rules:
369
+ * - Must be an object
370
+ * - url (if present) must be a string between 1 and 2048 characters long
371
+ * - xml (if present) must be a string
372
+ */
373
+ static validateSamlIdpMetadata(obj) {
374
+ const errors = [];
375
+ if (!obj || typeof obj !== 'object') {
376
+ return ['Input is not an object'];
377
+ }
378
+ const metadata = obj;
379
+ // Check url property if present
380
+ if (metadata.url !== undefined) {
381
+ if (typeof metadata.url !== 'string') {
382
+ errors.push("Property 'url' must be a string");
383
+ }
384
+ else if (metadata.url.length < 1) {
385
+ errors.push("Property 'url' cannot be empty");
386
+ }
387
+ else if (metadata.url.length > 2048) {
388
+ errors.push("Property 'url' exceeds maximum length of 2048 characters");
389
+ }
390
+ }
391
+ // Check xml property if present
392
+ if (metadata.xml !== undefined && typeof metadata.xml !== 'string') {
393
+ errors.push("Property 'xml' must be a string");
394
+ }
395
+ return errors;
396
+ }
397
+ /**
398
+ * Validates the SAML configuration.
399
+ *
400
+ * @param config - The SAML configuration to validate
401
+ * @returns An array of error messages if validation fails, or an empty array if valid
402
+ *
403
+ * Validation rules:
404
+ * - Must be an object
405
+ * - idpMetadata is required and must be valid
406
+ * - assertionAtrributes (if present) must be valid
407
+ * - allowedOrganizations (if present) must be an array of strings with 1-256 elements
408
+ * - loginValidityDuration (if present) must be a positive number
409
+ * - roleValues (if present) must be an object with valid admin and editor arrays
410
+ */
411
+ static validateSamlConfiguration(config) {
412
+ const errors = [];
413
+ if (config === undefined) {
414
+ return errors; // Optional property can be undefined
415
+ }
416
+ if (!config || typeof config !== 'object') {
417
+ errors.push('samlConfiguration must be an object');
418
+ return errors;
419
+ }
420
+ const samlConfig = config;
421
+ // Check idpMetadata (required)
422
+ if (samlConfig.idpMetadata === undefined) {
423
+ errors.push('idpMetadata is required in samlConfiguration');
424
+ }
425
+ else {
426
+ const idpMetadataErrors = Workspace.validateSamlIdpMetadata(samlConfig.idpMetadata);
427
+ if (idpMetadataErrors.length > 0) {
428
+ errors.push(...idpMetadataErrors.map((err) => `idpMetadata: ${err}`));
429
+ }
430
+ }
431
+ // Check assertionAtrributes if present
432
+ if (samlConfig.assertionAtrributes !== undefined) {
433
+ const attributeErrors = Workspace.validateSamlAssertionAttributes(samlConfig.assertionAtrributes);
434
+ if (attributeErrors.length > 0) {
435
+ errors.push(...attributeErrors.map((err) => `assertionAtrributes: ${err}`));
436
+ }
437
+ }
438
+ // Check allowedOrganizations if present
439
+ if (samlConfig.allowedOrganizations !== undefined) {
440
+ if (!Array.isArray(samlConfig.allowedOrganizations)) {
441
+ errors.push('allowedOrganizations must be an array');
442
+ }
443
+ else {
444
+ if (samlConfig.allowedOrganizations.length < 1) {
445
+ errors.push('allowedOrganizations must have at least 1 element');
446
+ }
447
+ if (samlConfig.allowedOrganizations.length > 256) {
448
+ errors.push('allowedOrganizations cannot have more than 256 elements');
449
+ }
450
+ for (let i = 0; i < samlConfig.allowedOrganizations.length; i++) {
451
+ const org = samlConfig.allowedOrganizations[i];
452
+ if (typeof org !== 'string') {
453
+ errors.push(`allowedOrganizations[${i}] must be a string`);
454
+ }
455
+ }
456
+ }
457
+ }
458
+ // Check loginValidityDuration if present
459
+ if (samlConfig.loginValidityDuration !== undefined) {
460
+ if (typeof samlConfig.loginValidityDuration !== 'number') {
461
+ errors.push('loginValidityDuration must be a number');
462
+ }
463
+ else if (samlConfig.loginValidityDuration <= 0) {
464
+ errors.push('loginValidityDuration must be positive');
465
+ }
466
+ }
467
+ // Check roleValues if present
468
+ if (samlConfig.roleValues !== undefined) {
469
+ if (!samlConfig.roleValues || typeof samlConfig.roleValues !== 'object') {
470
+ errors.push('roleValues must be an object');
471
+ }
472
+ else {
473
+ // Check admin array if present
474
+ if (samlConfig.roleValues.admin !== undefined) {
475
+ if (!Array.isArray(samlConfig.roleValues.admin)) {
476
+ errors.push('roleValues.admin must be an array');
477
+ }
478
+ else {
479
+ for (let i = 0; i < samlConfig.roleValues.admin.length; i++) {
480
+ if (typeof samlConfig.roleValues.admin[i] !== 'string') {
481
+ errors.push(`roleValues.admin[${i}] must be a string`);
482
+ }
483
+ }
484
+ }
485
+ }
486
+ // Check editor array if present
487
+ if (samlConfig.roleValues.editor !== undefined) {
488
+ if (!Array.isArray(samlConfig.roleValues.editor)) {
489
+ errors.push('roleValues.editor must be an array');
490
+ }
491
+ else {
492
+ for (let i = 0; i < samlConfig.roleValues.editor.length; i++) {
493
+ if (typeof samlConfig.roleValues.editor[i] !== 'string') {
494
+ errors.push(`roleValues.editor[${i}] must be a string`);
495
+ }
496
+ }
497
+ }
498
+ }
499
+ }
500
+ }
501
+ return errors;
502
+ }
503
+ /**
504
+ * Validates the vpcConfiguration property.
505
+ *
506
+ * @param config - The VPC configuration to validate
507
+ * @returns An array of error messages if validation fails, or an empty array if valid
508
+ *
509
+ * Validation rules:
510
+ * - Must be an object
511
+ * - securityGroups is required and must be an array with 1-5 items
512
+ * - subnets is required and must be an array with 2-6 items
513
+ */
514
+ static validateVpcConfiguration(config) {
515
+ const errors = [];
516
+ if (config === undefined) {
517
+ return errors; // Optional property can be undefined
518
+ }
519
+ if (!config || typeof config !== 'object') {
520
+ errors.push('vpcConfiguration must be an object');
521
+ return errors;
522
+ }
523
+ const vpcConfig = config;
524
+ // Check securityGroups (required)
525
+ if (vpcConfig.securityGroups === undefined) {
526
+ errors.push('securityGroups is required in vpcConfiguration');
527
+ }
528
+ else if (!Array.isArray(vpcConfig.securityGroups)) {
529
+ errors.push('securityGroups must be an array');
530
+ }
531
+ else {
532
+ if (vpcConfig.securityGroups.length < 1) {
533
+ errors.push('securityGroups must have at least 1 item');
534
+ }
535
+ if (vpcConfig.securityGroups.length > 5) {
536
+ errors.push('securityGroups cannot have more than 5 items');
537
+ }
538
+ }
539
+ // Check subnets (required)
540
+ if (vpcConfig.subnets === undefined) {
541
+ errors.push('subnets is required in vpcConfiguration');
542
+ }
543
+ else if (!Array.isArray(vpcConfig.subnets)) {
544
+ errors.push('subnets must be an array');
545
+ }
546
+ else {
547
+ if (vpcConfig.subnets.length < 2) {
548
+ errors.push('subnets must have at least 2 items');
549
+ }
550
+ if (vpcConfig.subnets.length > 6) {
551
+ errors.push('subnets cannot have more than 6 items');
552
+ }
553
+ }
554
+ return errors;
555
+ }
556
+ /**
557
+ * Validates all workspace properties.
558
+ *
559
+ * @param props - The workspace properties to validate
560
+ * @returns An array of error messages if validation fails, or an empty array if valid
561
+ *
562
+ * This method aggregates validation results from all individual property validators.
563
+ * It throws an error if props is not an object.
564
+ */
565
+ static validateProps(props) {
566
+ const errors = [];
567
+ if (!props || typeof props !== 'object') {
568
+ throw new Error('Props is not an object');
569
+ }
570
+ const workspaceProps = props;
571
+ if (workspaceProps.clientToken !== undefined) {
572
+ const clientTokenErrors = Workspace.validateClientToken(workspaceProps.clientToken);
573
+ if (clientTokenErrors.length > 0) {
574
+ errors.push(...clientTokenErrors.map((err) => `clientToken: ${err}`));
575
+ }
576
+ }
577
+ if (workspaceProps.description !== undefined) {
578
+ const descriptionErrors = Workspace.validateDescription(workspaceProps.description);
579
+ if (descriptionErrors.length > 0) {
580
+ errors.push(...descriptionErrors.map((err) => `description: ${err}`));
581
+ }
582
+ }
583
+ if (workspaceProps.grafanaVersion !== undefined) {
584
+ const grafanaVersionErrors = Workspace.validateGrafanaVersion(workspaceProps.grafanaVersion);
585
+ if (grafanaVersionErrors.length > 0) {
586
+ errors.push(...grafanaVersionErrors.map((err) => `grafanaVersion: ${err}`));
587
+ }
588
+ }
589
+ if (workspaceProps.name !== undefined) {
590
+ const nameErrors = Workspace.validateName(workspaceProps.name);
591
+ if (nameErrors.length > 0) {
592
+ errors.push(...nameErrors.map((err) => `name: ${err}`));
593
+ }
594
+ }
595
+ if (workspaceProps.networkAccessControl !== undefined) {
596
+ const networkAccessControlErrors = Workspace.validateNetworkAccessControl(workspaceProps.networkAccessControl);
597
+ if (networkAccessControlErrors.length > 0) {
598
+ errors.push(...networkAccessControlErrors.map((err) => `networkAccessControl: ${err}`));
599
+ }
600
+ }
601
+ if (workspaceProps.organizationRoleName !== undefined) {
602
+ const organizationRoleNameErrors = Workspace.validateOrganizationRoleName(workspaceProps.organizationRoleName);
603
+ if (organizationRoleNameErrors.length > 0) {
604
+ errors.push(...organizationRoleNameErrors.map((err) => `organizationRoleName: ${err}`));
605
+ }
606
+ }
607
+ if (workspaceProps.samlConfiguration !== undefined) {
608
+ const samlConfigurationErrors = Workspace.validateSamlConfiguration(workspaceProps.samlConfiguration);
609
+ if (samlConfigurationErrors.length > 0) {
610
+ errors.push(...samlConfigurationErrors.map((err) => `samlConfiguration: ${err}`));
611
+ }
612
+ }
613
+ if (workspaceProps.vpcConfiguration !== undefined) {
614
+ const vpcConfigurationErrors = Workspace.validateVpcConfiguration(workspaceProps.vpcConfiguration);
615
+ if (vpcConfigurationErrors.length > 0) {
616
+ errors.push(...vpcConfigurationErrors.map((err) => `vpcConfiguration: ${err}`));
617
+ }
618
+ }
619
+ return errors;
620
+ }
621
+ /**
622
+ * Creates a new Amazon Managed Grafana workspace.
623
+ *
624
+ * @param scope - The scope in which to define this construct
625
+ * @param id - The scoped construct ID
626
+ * @param props - Configuration properties for the workspace
627
+ *
628
+ * @throws Error if any of the provided properties fail validation
629
+ */
630
+ constructor(scope, id, props) {
631
+ super(scope, id);
632
+ const errors = Workspace.validateProps(props);
633
+ if (errors.length > 0) {
634
+ throw new Error(`Invalid props:\n${errors.join('\n')}`);
635
+ }
636
+ if (props.accountAccessType === AccountAccessType.CURRENT_ACCOUNT &&
637
+ !props.role) {
638
+ throw new Error('Role must be provided when accountAccessType is CURRENT_ACCOUNT');
639
+ }
640
+ this.accountAccessType = props.accountAccessType;
641
+ this.authenticationProviders = props.authenticationProviders;
642
+ this.clientToken = props.clientToken;
643
+ this.dataSources = props.dataSources;
644
+ this.description = props.description;
645
+ this.networkAccessControl = props.networkAccessControl;
646
+ this.notificationDestinations = props.notificationDestinations;
647
+ this.organizationalUnits = props.organizationalUnits;
648
+ this.organizationRoleName = props.organizationRoleName;
649
+ this.permissionType = props.permissionType;
650
+ this.pluginAdminEnabled = props.pluginAdminEnabled;
651
+ this.name = props.name;
652
+ this.role = props.role;
653
+ this.samlConfiguration = props.samlConfiguration;
654
+ this.stackSetName = props.stackSetName;
655
+ this.vpcConfiguration = props.vpcConfiguration;
656
+ let cfnWorkspaceProps = {
657
+ accountAccessType: props.accountAccessType,
658
+ authenticationProviders: props.authenticationProviders,
659
+ clientToken: props.clientToken,
660
+ dataSources: props.dataSources,
661
+ description: props.description,
662
+ grafanaVersion: props.grafanaVersion,
663
+ name: props.name,
664
+ notificationDestinations: props.notificationDestinations,
665
+ organizationalUnits: props.organizationalUnits,
666
+ organizationRoleName: props.organizationRoleName,
667
+ permissionType: props.permissionType,
668
+ pluginAdminEnabled: props.pluginAdminEnabled,
669
+ roleArn: props.role?.roleArn,
670
+ samlConfiguration: props.samlConfiguration,
671
+ stackSetName: props.stackSetName,
672
+ vpcConfiguration: props.vpcConfiguration
673
+ ? {
674
+ securityGroupIds: props.vpcConfiguration.securityGroups.map((sg) => sg.securityGroupId),
675
+ subnetIds: props.vpcConfiguration.subnets.map((subnet) => subnet.subnetId),
676
+ }
677
+ : undefined,
678
+ };
679
+ this.workspace = new aws_grafana_1.CfnWorkspace(this, 'Resource', cfnWorkspaceProps);
680
+ this.creationTimestamp = this.workspace.attrCreationTimestamp;
681
+ this.endpoint = this.workspace.attrEndpoint;
682
+ this.grafanaVersion = this.workspace.attrGrafanaVersion;
683
+ this.id = this.workspace.attrId;
684
+ this.modificationTimestamp = this.workspace.attrModificationTimestamp;
685
+ this.samlConfigurationStatus = this.workspace
686
+ .attrSamlConfigurationStatus;
687
+ this.ssoClientId = this.workspace.attrSsoClientId;
688
+ this.status = this.workspace.attrStatus;
689
+ }
690
+ }
691
+ exports.Workspace = Workspace;
692
+ _a = JSII_RTTI_SYMBOL_1;
693
+ Workspace[_a] = { fqn: "@robhan-cdk-lib/aws_grafana.Workspace", version: "0.0.0" };
694
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"workspace.js","sourceRoot":"","sources":["../src/workspace.ts"],"names":[],"mappings":";;;;;AAMA,yDAA0E;AAE1E,2CAAuC;AAEvC;;;;;GAKG;AACH,IAAY,iBAUX;AAVD,WAAY,iBAAiB;IAC3B;;OAEG;IACH,wDAAmC,CAAA;IAEnC;;OAEG;IACH,kDAA6B,CAAA;AAC/B,CAAC,EAVW,iBAAiB,iCAAjB,iBAAiB,QAU5B;AAED;;;;;GAKG;AACH,IAAY,uBAUX;AAVD,WAAY,uBAAuB;IACjC;;OAEG;IACH,8CAAmB,CAAA;IAEnB;;OAEG;IACH,wCAAa,CAAA;AACf,CAAC,EAVW,uBAAuB,uCAAvB,uBAAuB,QAUlC;AA2BD;;;GAGG;AACH,IAAY,wBAKX;AALD,WAAY,wBAAwB;IAClC;;OAEG;IACH,uCAAW,CAAA;AACb,CAAC,EALW,wBAAwB,wCAAxB,wBAAwB,QAKnC;AAED;;;;;;;;;;GAUG;AACH,IAAY,eAUX;AAVD,WAAY,eAAe;IACzB;;OAEG;IACH,wDAAqC,CAAA;IAErC;;OAEG;IACH,sDAAmC,CAAA;AACrC,CAAC,EAVW,eAAe,+BAAf,eAAe,QAU1B;AA+PD;;GAEG;AACH,IAAY,yBAUX;AAVD,WAAY,yBAAyB;IACnC;;OAEG;IACH,sDAAyB,CAAA;IAEzB;;OAEG;IACH,8DAAiC,CAAA;AACnC,CAAC,EAVW,yBAAyB,yCAAzB,yBAAyB,QAUpC;AAED;;GAEG;AACH,IAAY,MAuDX;AAvDD,WAAY,MAAM;IAChB;;OAEG;IACH,2BAAiB,CAAA;IAEjB;;OAEG;IACH,+BAAqB,CAAA;IAErB;;OAEG;IACH,+BAAqB,CAAA;IAErB;;OAEG;IACH,2BAAiB,CAAA;IAEjB;;OAEG;IACH,+BAAqB,CAAA;IAErB;;OAEG;IACH,iCAAuB,CAAA;IAEvB;;OAEG;IACH,6CAAmC,CAAA;IAEnC;;OAEG;IACH,6CAAmC,CAAA;IAEnC;;OAEG;IACH,yCAA+B,CAAA;IAE/B;;OAEG;IACH,2CAAiC,CAAA;IAEjC;;OAEG;IACH,2DAAiD,CAAA;AACnD,CAAC,EAvDW,MAAM,sBAAN,MAAM,QAuDjB;AAED;;;;;GAKG;AACH,MAAa,SAAU,SAAQ,sBAAS;IACtC;;;;;;;;;;OAUG;IACK,MAAM,CAAC,mBAAmB,CAAC,KAAc;QAC/C,MAAM,MAAM,GAAa,EAAE,CAAC;QAE5B,IAAI,KAAK,KAAK,SAAS,EAAE,CAAC;YACxB,OAAO,MAAM,CAAC,CAAC,qCAAqC;QACtD,CAAC;QAED,IAAI,OAAO,KAAK,KAAK,QAAQ,EAAE,CAAC;YAC9B,MAAM,CAAC,IAAI,CAAC,8BAA8B,CAAC,CAAC;YAC5C,OAAO,MAAM,CAAC,CAAC,2CAA2C;QAC5D,CAAC;QAED,MAAM,KAAK,GAAG,eAAe,CAAC;QAC9B,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,KAAK,CAAC,EAAE,CAAC;YACvB,IAAI,KAAK,CAAC,MAAM,GAAG,CAAC,IAAI,KAAK,CAAC,MAAM,GAAG,EAAE,EAAE,CAAC;gBAC1C,MAAM,CAAC,IAAI,CAAC,sDAAsD,CAAC,CAAC;YACtE,CAAC;YAED,IAAI,CAAC,UAAU,CAAC,IAAI,CAAC,KAAK,CAAC,EAAE,CAAC;gBAC5B,MAAM,CAAC,IAAI,CAAC,0DAA0D,CAAC,CAAC;YAC1E,CAAC;QACH,CAAC;QAED,OAAO,MAAM,CAAC;IAChB,CAAC;IAED;;;;;;;;;OASG;IACK,MAAM,CAAC,mBAAmB,CAAC,WAAoB;QACrD,MAAM,MAAM,GAAa,EAAE,CAAC;QAE5B,IAAI,WAAW,KAAK,SAAS,EAAE,CAAC;YAC9B,OAAO,MAAM,CAAC,CAAC,qCAAqC;QACtD,CAAC;QAED,IAAI,OAAO,WAAW,KAAK,QAAQ,EAAE,CAAC;YACpC,MAAM,CAAC,IAAI,CAAC,8BAA8B,CAAC,CAAC;YAC5C,OAAO,MAAM,CAAC,CAAC,2CAA2C;QAC5D,CAAC;QAED,IAAI,WAAW,CAAC,MAAM,GAAG,IAAI,EAAE,CAAC;YAC9B,MAAM,CAAC,IAAI,CAAC,2CAA2C,CAAC,CAAC;QAC3D,CAAC;QAED,OAAO,MAAM,CAAC;IAChB,CAAC;IAED;;;;;;;;;OASG;IACK,MAAM,CAAC,sBAAsB,CAAC,OAAgB;QACpD,MAAM,MAAM,GAAa,EAAE,CAAC;QAE5B,IAAI,OAAO,KAAK,SAAS,EAAE,CAAC;YAC1B,OAAO,MAAM,CAAC,CAAC,qCAAqC;QACtD,CAAC;QAED,IAAI,OAAO,OAAO,KAAK,QAAQ,EAAE,CAAC;YAChC,MAAM,CAAC,IAAI,CAAC,iCAAiC,CAAC,CAAC;YAC/C,OAAO,MAAM,CAAC,CAAC,2CAA2C;QAC5D,CAAC;QAED,IAAI,OAAO,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;YACvB,MAAM,CAAC,IAAI,CAAC,gCAAgC,CAAC,CAAC;QAChD,CAAC;QAED,IAAI,OAAO,CAAC,MAAM,GAAG,GAAG,EAAE,CAAC;YACzB,MAAM,CAAC,IAAI,CAAC,6CAA6C,CAAC,CAAC;QAC7D,CAAC;QAED,OAAO,MAAM,CAAC;IAChB,CAAC;IAED;;;;;;;;;;OAUG;IACK,MAAM,CAAC,YAAY,CAAC,IAAa;QACvC,MAAM,MAAM,GAAa,EAAE,CAAC;QAE5B,IAAI,IAAI,KAAK,SAAS,EAAE,CAAC;YACvB,OAAO,MAAM,CAAC,CAAC,qCAAqC;QACtD,CAAC;QAED,IAAI,OAAO,IAAI,KAAK,QAAQ,EAAE,CAAC;YAC7B,MAAM,CAAC,IAAI,CAAC,uBAAuB,CAAC,CAAC;YACrC,OAAO,MAAM,CAAC,CAAC,2CAA2C;QAC5D,CAAC;QAED,MAAM,KAAK,GAAG,2BAA2B,CAAC;QAC1C,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,IAAI,CAAC,EAAE,CAAC;YACtB,IAAI,IAAI,CAAC,MAAM,GAAG,CAAC,IAAI,IAAI,CAAC,MAAM,GAAG,GAAG,EAAE,CAAC;gBACzC,MAAM,CAAC,IAAI,CAAC,gDAAgD,CAAC,CAAC;YAChE,CAAC;YAED,IAAI,CAAC,qBAAqB,CAAC,IAAI,CAAC,IAAI,CAAC,EAAE,CAAC;gBACtC,MAAM,CAAC,IAAI,CACT,uFAAuF,CACxF,CAAC;YACJ,CAAC;QACH,CAAC;QAED,OAAO,MAAM,CAAC;IAChB,CAAC;IAED;;;;;;;;;;OAUG;IACK,MAAM,CAAC,4BAA4B,CAAC,GAAY;QACtD,MAAM,MAAM,GAAa,EAAE,CAAC;QAE5B,IAAI,GAAG,KAAK,SAAS,EAAE,CAAC;YACtB,OAAO,MAAM,CAAC,CAAC,qCAAqC;QACtD,CAAC;QAED,IAAI,CAAC,GAAG,IAAI,OAAO,GAAG,KAAK,QAAQ,EAAE,CAAC;YACpC,MAAM,CAAC,IAAI,CAAC,wCAAwC,CAAC,CAAC;YACtD,OAAO,MAAM,CAAC;QAChB,CAAC;QAED,MAAM,oBAAoB,GAAG,GAA2B,CAAC;QAEzD,+BAA+B;QAC/B,IAAI,oBAAoB,CAAC,WAAW,KAAK,SAAS,EAAE,CAAC;YACnD,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,oBAAoB,CAAC,WAAW,CAAC,EAAE,CAAC;gBACrD,MAAM,CAAC,IAAI,CAAC,8BAA8B,CAAC,CAAC;YAC9C,CAAC;iBAAM,IAAI,oBAAoB,CAAC,WAAW,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;gBACvD,MAAM,CAAC,IAAI,CAAC,sCAAsC,CAAC,CAAC;YACtD,CAAC;QACH,CAAC;QAED,gCAAgC;QAChC,IAAI,oBAAoB,CAAC,YAAY,KAAK,SAAS,EAAE,CAAC;YACpD,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,oBAAoB,CAAC,YAAY,CAAC,EAAE,CAAC;gBACtD,MAAM,CAAC,IAAI,CAAC,+BAA+B,CAAC,CAAC;YAC/C,CAAC;iBAAM,IAAI,oBAAoB,CAAC,YAAY,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;gBACxD,MAAM,CAAC,IAAI,CAAC,uCAAuC,CAAC,CAAC;YACvD,CAAC;QACH,CAAC;QAED,OAAO,MAAM,CAAC;IAChB,CAAC;IAED;;;;;;;;;OASG;IACK,MAAM,CAAC,4BAA4B,CAAC,QAAiB;QAC3D,MAAM,MAAM,GAAa,EAAE,CAAC;QAE5B,IAAI,QAAQ,KAAK,SAAS,EAAE,CAAC;YAC3B,OAAO,MAAM,CAAC,CAAC,qCAAqC;QACtD,CAAC;QAED,IAAI,OAAO,QAAQ,KAAK,QAAQ,EAAE,CAAC;YACjC,MAAM,CAAC,IAAI,CAAC,uCAAuC,CAAC,CAAC;YACrD,OAAO,MAAM,CAAC,CAAC,2CAA2C;QAC5D,CAAC;QAED,IAAI,QAAQ,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;YACxB,MAAM,CAAC,IAAI,CAAC,sCAAsC,CAAC,CAAC;QACtD,CAAC;QAED,IAAI,QAAQ,CAAC,MAAM,GAAG,IAAI,EAAE,CAAC;YAC3B,MAAM,CAAC,IAAI,CAAC,oDAAoD,CAAC,CAAC;QACpE,CAAC;QAED,OAAO,MAAM,CAAC;IAChB,CAAC;IAED;;;;;;;;;;;OAWG;IACK,MAAM,CAAC,+BAA+B,CAAC,GAAY;QACzD,MAAM,MAAM,GAAa,EAAE,CAAC;QAE5B,IAAI,CAAC,GAAG,IAAI,OAAO,GAAG,KAAK,QAAQ,EAAE,CAAC;YACpC,OAAO,CAAC,wBAAwB,CAAC,CAAC;QACpC,CAAC;QAED,MAAM,UAAU,GAAG,GAA8B,CAAC;QAElD,KAAK,MAAM,GAAG,IAAI,UAAU,EAAE,CAAC;YAC7B,MAAM,KAAK,GAAG,UAAU,CAAC,GAAG,CAAC,CAAC;YAC9B,IAAI,KAAK,KAAK,SAAS,EAAE,CAAC;gBACxB,SAAS,CAAC,uCAAuC;YACnD,CAAC;YAED,IAAI,OAAO,KAAK,KAAK,QAAQ,EAAE,CAAC;gBAC9B,MAAM,CAAC,IAAI,CAAC,aAAa,GAAG,oBAAoB,CAAC,CAAC;YACpD,CAAC;iBAAM,IAAI,KAAK,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;gBAC5B,MAAM,CAAC,IAAI,CAAC,aAAa,GAAG,mBAAmB,CAAC,CAAC;YACnD,CAAC;iBAAM,IAAI,KAAK,CAAC,MAAM,GAAG,GAAG,EAAE,CAAC;gBAC9B,MAAM,CAAC,IAAI,CACT,aAAa,GAAG,4CAA4C,CAC7D,CAAC;YACJ,CAAC;QACH,CAAC;QAED,OAAO,MAAM,CAAC;IAChB,CAAC;IAED;;;;;;;;;;OAUG;IACK,MAAM,CAAC,uBAAuB,CAAC,GAAY;QACjD,MAAM,MAAM,GAAa,EAAE,CAAC;QAE5B,IAAI,CAAC,GAAG,IAAI,OAAO,GAAG,KAAK,QAAQ,EAAE,CAAC;YACpC,OAAO,CAAC,wBAAwB,CAAC,CAAC;QACpC,CAAC;QAED,MAAM,QAAQ,GAAG,GAA8B,CAAC;QAEhD,gCAAgC;QAChC,IAAI,QAAQ,CAAC,GAAG,KAAK,SAAS,EAAE,CAAC;YAC/B,IAAI,OAAO,QAAQ,CAAC,GAAG,KAAK,QAAQ,EAAE,CAAC;gBACrC,MAAM,CAAC,IAAI,CAAC,iCAAiC,CAAC,CAAC;YACjD,CAAC;iBAAM,IAAI,QAAQ,CAAC,GAAG,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;gBACnC,MAAM,CAAC,IAAI,CAAC,gCAAgC,CAAC,CAAC;YAChD,CAAC;iBAAM,IAAI,QAAQ,CAAC,GAAG,CAAC,MAAM,GAAG,IAAI,EAAE,CAAC;gBACtC,MAAM,CAAC,IAAI,CAAC,0DAA0D,CAAC,CAAC;YAC1E,CAAC;QACH,CAAC;QAED,gCAAgC;QAChC,IAAI,QAAQ,CAAC,GAAG,KAAK,SAAS,IAAI,OAAO,QAAQ,CAAC,GAAG,KAAK,QAAQ,EAAE,CAAC;YACnE,MAAM,CAAC,IAAI,CAAC,iCAAiC,CAAC,CAAC;QACjD,CAAC;QAED,OAAO,MAAM,CAAC;IAChB,CAAC;IAED;;;;;;;;;;;;;OAaG;IACK,MAAM,CAAC,yBAAyB,CAAC,MAAe;QACtD,MAAM,MAAM,GAAa,EAAE,CAAC;QAE5B,IAAI,MAAM,KAAK,SAAS,EAAE,CAAC;YACzB,OAAO,MAAM,CAAC,CAAC,qCAAqC;QACtD,CAAC;QAED,IAAI,CAAC,MAAM,IAAI,OAAO,MAAM,KAAK,QAAQ,EAAE,CAAC;YAC1C,MAAM,CAAC,IAAI,CAAC,qCAAqC,CAAC,CAAC;YACnD,OAAO,MAAM,CAAC;QAChB,CAAC;QAED,MAAM,UAAU,GAAG,MAA2B,CAAC;QAE/C,+BAA+B;QAC/B,IAAI,UAAU,CAAC,WAAW,KAAK,SAAS,EAAE,CAAC;YACzC,MAAM,CAAC,IAAI,CAAC,8CAA8C,CAAC,CAAC;QAC9D,CAAC;aAAM,CAAC;YACN,MAAM,iBAAiB,GAAG,SAAS,CAAC,uBAAuB,CACzD,UAAU,CAAC,WAAW,CACvB,CAAC;YACF,IAAI,iBAAiB,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;gBACjC,MAAM,CAAC,IAAI,CAAC,GAAG,iBAAiB,CAAC,GAAG,CAAC,CAAC,GAAG,EAAE,EAAE,CAAC,gBAAgB,GAAG,EAAE,CAAC,CAAC,CAAC;YACxE,CAAC;QACH,CAAC;QAED,uCAAuC;QACvC,IAAI,UAAU,CAAC,mBAAmB,KAAK,SAAS,EAAE,CAAC;YACjD,MAAM,eAAe,GAAG,SAAS,CAAC,+BAA+B,CAC/D,UAAU,CAAC,mBAAmB,CAC/B,CAAC;YACF,IAAI,eAAe,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;gBAC/B,MAAM,CAAC,IAAI,CACT,GAAG,eAAe,CAAC,GAAG,CAAC,CAAC,GAAG,EAAE,EAAE,CAAC,wBAAwB,GAAG,EAAE,CAAC,CAC/D,CAAC;YACJ,CAAC;QACH,CAAC;QAED,wCAAwC;QACxC,IAAI,UAAU,CAAC,oBAAoB,KAAK,SAAS,EAAE,CAAC;YAClD,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,UAAU,CAAC,oBAAoB,CAAC,EAAE,CAAC;gBACpD,MAAM,CAAC,IAAI,CAAC,uCAAuC,CAAC,CAAC;YACvD,CAAC;iBAAM,CAAC;gBACN,IAAI,UAAU,CAAC,oBAAoB,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;oBAC/C,MAAM,CAAC,IAAI,CAAC,mDAAmD,CAAC,CAAC;gBACnE,CAAC;gBACD,IAAI,UAAU,CAAC,oBAAoB,CAAC,MAAM,GAAG,GAAG,EAAE,CAAC;oBACjD,MAAM,CAAC,IAAI,CACT,yDAAyD,CAC1D,CAAC;gBACJ,CAAC;gBAED,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,UAAU,CAAC,oBAAoB,CAAC,MAAM,EAAE,CAAC,EAAE,EAAE,CAAC;oBAChE,MAAM,GAAG,GAAG,UAAU,CAAC,oBAAoB,CAAC,CAAC,CAAC,CAAC;oBAC/C,IAAI,OAAO,GAAG,KAAK,QAAQ,EAAE,CAAC;wBAC5B,MAAM,CAAC,IAAI,CAAC,wBAAwB,CAAC,oBAAoB,CAAC,CAAC;oBAC7D,CAAC;gBACH,CAAC;YACH,CAAC;QACH,CAAC;QAED,yCAAyC;QACzC,IAAI,UAAU,CAAC,qBAAqB,KAAK,SAAS,EAAE,CAAC;YACnD,IAAI,OAAO,UAAU,CAAC,qBAAqB,KAAK,QAAQ,EAAE,CAAC;gBACzD,MAAM,CAAC,IAAI,CAAC,wCAAwC,CAAC,CAAC;YACxD,CAAC;iBAAM,IAAI,UAAU,CAAC,qBAAqB,IAAI,CAAC,EAAE,CAAC;gBACjD,MAAM,CAAC,IAAI,CAAC,wCAAwC,CAAC,CAAC;YACxD,CAAC;QACH,CAAC;QAED,8BAA8B;QAC9B,IAAI,UAAU,CAAC,UAAU,KAAK,SAAS,EAAE,CAAC;YACxC,IAAI,CAAC,UAAU,CAAC,UAAU,IAAI,OAAO,UAAU,CAAC,UAAU,KAAK,QAAQ,EAAE,CAAC;gBACxE,MAAM,CAAC,IAAI,CAAC,8BAA8B,CAAC,CAAC;YAC9C,CAAC;iBAAM,CAAC;gBACN,+BAA+B;gBAC/B,IAAI,UAAU,CAAC,UAAU,CAAC,KAAK,KAAK,SAAS,EAAE,CAAC;oBAC9C,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,UAAU,CAAC,UAAU,CAAC,KAAK,CAAC,EAAE,CAAC;wBAChD,MAAM,CAAC,IAAI,CAAC,mCAAmC,CAAC,CAAC;oBACnD,CAAC;yBAAM,CAAC;wBACN,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,UAAU,CAAC,UAAU,CAAC,KAAK,CAAC,MAAM,EAAE,CAAC,EAAE,EAAE,CAAC;4BAC5D,IAAI,OAAO,UAAU,CAAC,UAAU,CAAC,KAAK,CAAC,CAAC,CAAC,KAAK,QAAQ,EAAE,CAAC;gCACvD,MAAM,CAAC,IAAI,CAAC,oBAAoB,CAAC,oBAAoB,CAAC,CAAC;4BACzD,CAAC;wBACH,CAAC;oBACH,CAAC;gBACH,CAAC;gBAED,gCAAgC;gBAChC,IAAI,UAAU,CAAC,UAAU,CAAC,MAAM,KAAK,SAAS,EAAE,CAAC;oBAC/C,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,UAAU,CAAC,UAAU,CAAC,MAAM,CAAC,EAAE,CAAC;wBACjD,MAAM,CAAC,IAAI,CAAC,oCAAoC,CAAC,CAAC;oBACpD,CAAC;yBAAM,CAAC;wBACN,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,UAAU,CAAC,UAAU,CAAC,MAAM,CAAC,MAAM,EAAE,CAAC,EAAE,EAAE,CAAC;4BAC7D,IAAI,OAAO,UAAU,CAAC,UAAU,CAAC,MAAM,CAAC,CAAC,CAAC,KAAK,QAAQ,EAAE,CAAC;gCACxD,MAAM,CAAC,IAAI,CAAC,qBAAqB,CAAC,oBAAoB,CAAC,CAAC;4BAC1D,CAAC;wBACH,CAAC;oBACH,CAAC;gBACH,CAAC;YACH,CAAC;QACH,CAAC;QAED,OAAO,MAAM,CAAC;IAChB,CAAC;IAED;;;;;;;;;;OAUG;IACK,MAAM,CAAC,wBAAwB,CAAC,MAAe;QACrD,MAAM,MAAM,GAAa,EAAE,CAAC;QAE5B,IAAI,MAAM,KAAK,SAAS,EAAE,CAAC;YACzB,OAAO,MAAM,CAAC,CAAC,qCAAqC;QACtD,CAAC;QAED,IAAI,CAAC,MAAM,IAAI,OAAO,MAAM,KAAK,QAAQ,EAAE,CAAC;YAC1C,MAAM,CAAC,IAAI,CAAC,oCAAoC,CAAC,CAAC;YAClD,OAAO,MAAM,CAAC;QAChB,CAAC;QAED,MAAM,SAAS,GAAG,MAA0B,CAAC;QAE7C,kCAAkC;QAClC,IAAI,SAAS,CAAC,cAAc,KAAK,SAAS,EAAE,CAAC;YAC3C,MAAM,CAAC,IAAI,CAAC,gDAAgD,CAAC,CAAC;QAChE,CAAC;aAAM,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,SAAS,CAAC,cAAc,CAAC,EAAE,CAAC;YACpD,MAAM,CAAC,IAAI,CAAC,iCAAiC,CAAC,CAAC;QACjD,CAAC;aAAM,CAAC;YACN,IAAI,SAAS,CAAC,cAAc,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;gBACxC,MAAM,CAAC,IAAI,CAAC,0CAA0C,CAAC,CAAC;YAC1D,CAAC;YACD,IAAI,SAAS,CAAC,cAAc,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;gBACxC,MAAM,CAAC,IAAI,CAAC,8CAA8C,CAAC,CAAC;YAC9D,CAAC;QACH,CAAC;QAED,2BAA2B;QAC3B,IAAI,SAAS,CAAC,OAAO,KAAK,SAAS,EAAE,CAAC;YACpC,MAAM,CAAC,IAAI,CAAC,yCAAyC,CAAC,CAAC;QACzD,CAAC;aAAM,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,SAAS,CAAC,OAAO,CAAC,EAAE,CAAC;YAC7C,MAAM,CAAC,IAAI,CAAC,0BAA0B,CAAC,CAAC;QAC1C,CAAC;aAAM,CAAC;YACN,IAAI,SAAS,CAAC,OAAO,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;gBACjC,MAAM,CAAC,IAAI,CAAC,oCAAoC,CAAC,CAAC;YACpD,CAAC;YACD,IAAI,SAAS,CAAC,OAAO,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;gBACjC,MAAM,CAAC,IAAI,CAAC,uCAAuC,CAAC,CAAC;YACvD,CAAC;QACH,CAAC;QAED,OAAO,MAAM,CAAC;IAChB,CAAC;IAED;;;;;;;;OAQG;IACK,MAAM,CAAC,aAAa,CAAC,KAAc;QACzC,MAAM,MAAM,GAAa,EAAE,CAAC;QAE5B,IAAI,CAAC,KAAK,IAAI,OAAO,KAAK,KAAK,QAAQ,EAAE,CAAC;YACxC,MAAM,IAAI,KAAK,CAAC,wBAAwB,CAAC,CAAC;QAC5C,CAAC;QACD,MAAM,cAAc,GAAG,KAAuB,CAAC;QAE/C,IAAI,cAAc,CAAC,WAAW,KAAK,SAAS,EAAE,CAAC;YAC7C,MAAM,iBAAiB,GAAG,SAAS,CAAC,mBAAmB,CACrD,cAAc,CAAC,WAAW,CAC3B,CAAC;YACF,IAAI,iBAAiB,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;gBACjC,MAAM,CAAC,IAAI,CAAC,GAAG,iBAAiB,CAAC,GAAG,CAAC,CAAC,GAAG,EAAE,EAAE,CAAC,gBAAgB,GAAG,EAAE,CAAC,CAAC,CAAC;YACxE,CAAC;QACH,CAAC;QAED,IAAI,cAAc,CAAC,WAAW,KAAK,SAAS,EAAE,CAAC;YAC7C,MAAM,iBAAiB,GAAG,SAAS,CAAC,mBAAmB,CACrD,cAAc,CAAC,WAAW,CAC3B,CAAC;YACF,IAAI,iBAAiB,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;gBACjC,MAAM,CAAC,IAAI,CAAC,GAAG,iBAAiB,CAAC,GAAG,CAAC,CAAC,GAAG,EAAE,EAAE,CAAC,gBAAgB,GAAG,EAAE,CAAC,CAAC,CAAC;YACxE,CAAC;QACH,CAAC;QAED,IAAI,cAAc,CAAC,cAAc,KAAK,SAAS,EAAE,CAAC;YAChD,MAAM,oBAAoB,GAAG,SAAS,CAAC,sBAAsB,CAC3D,cAAc,CAAC,cAAc,CAC9B,CAAC;YACF,IAAI,oBAAoB,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;gBACpC,MAAM,CAAC,IAAI,CACT,GAAG,oBAAoB,CAAC,GAAG,CAAC,CAAC,GAAG,EAAE,EAAE,CAAC,mBAAmB,GAAG,EAAE,CAAC,CAC/D,CAAC;YACJ,CAAC;QACH,CAAC;QAED,IAAI,cAAc,CAAC,IAAI,KAAK,SAAS,EAAE,CAAC;YACtC,MAAM,UAAU,GAAG,SAAS,CAAC,YAAY,CAAC,cAAc,CAAC,IAAI,CAAC,CAAC;YAC/D,IAAI,UAAU,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;gBAC1B,MAAM,CAAC,IAAI,CAAC,GAAG,UAAU,CAAC,GAAG,CAAC,CAAC,GAAG,EAAE,EAAE,CAAC,SAAS,GAAG,EAAE,CAAC,CAAC,CAAC;YAC1D,CAAC;QACH,CAAC;QAED,IAAI,cAAc,CAAC,oBAAoB,KAAK,SAAS,EAAE,CAAC;YACtD,MAAM,0BAA0B,GAAG,SAAS,CAAC,4BAA4B,CACvE,cAAc,CAAC,oBAAoB,CACpC,CAAC;YACF,IAAI,0BAA0B,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;gBAC1C,MAAM,CAAC,IAAI,CACT,GAAG,0BAA0B,CAAC,GAAG,CAC/B,CAAC,GAAG,EAAE,EAAE,CAAC,yBAAyB,GAAG,EAAE,CACxC,CACF,CAAC;YACJ,CAAC;QACH,CAAC;QAED,IAAI,cAAc,CAAC,oBAAoB,KAAK,SAAS,EAAE,CAAC;YACtD,MAAM,0BAA0B,GAAG,SAAS,CAAC,4BAA4B,CACvE,cAAc,CAAC,oBAAoB,CACpC,CAAC;YACF,IAAI,0BAA0B,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;gBAC1C,MAAM,CAAC,IAAI,CACT,GAAG,0BAA0B,CAAC,GAAG,CAC/B,CAAC,GAAG,EAAE,EAAE,CAAC,yBAAyB,GAAG,EAAE,CACxC,CACF,CAAC;YACJ,CAAC;QACH,CAAC;QAED,IAAI,cAAc,CAAC,iBAAiB,KAAK,SAAS,EAAE,CAAC;YACnD,MAAM,uBAAuB,GAAG,SAAS,CAAC,yBAAyB,CACjE,cAAc,CAAC,iBAAiB,CACjC,CAAC;YACF,IAAI,uBAAuB,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;gBACvC,MAAM,CAAC,IAAI,CACT,GAAG,uBAAuB,CAAC,GAAG,CAAC,CAAC,GAAG,EAAE,EAAE,CAAC,sBAAsB,GAAG,EAAE,CAAC,CACrE,CAAC;YACJ,CAAC;QACH,CAAC;QAED,IAAI,cAAc,CAAC,gBAAgB,KAAK,SAAS,EAAE,CAAC;YAClD,MAAM,sBAAsB,GAAG,SAAS,CAAC,wBAAwB,CAC/D,cAAc,CAAC,gBAAgB,CAChC,CAAC;YACF,IAAI,sBAAsB,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;gBACtC,MAAM,CAAC,IAAI,CACT,GAAG,sBAAsB,CAAC,GAAG,CAAC,CAAC,GAAG,EAAE,EAAE,CAAC,qBAAqB,GAAG,EAAE,CAAC,CACnE,CAAC;YACJ,CAAC;QACH,CAAC;QAED,OAAO,MAAM,CAAC;IAChB,CAAC;IA+HD;;;;;;;;OAQG;IACH,YAAY,KAAgB,EAAE,EAAU,EAAE,KAAqB;QAC7D,KAAK,CAAC,KAAK,EAAE,EAAE,CAAC,CAAC;QAEjB,MAAM,MAAM,GAAG,SAAS,CAAC,aAAa,CAAC,KAAK,CAAC,CAAC;QAC9C,IAAI,MAAM,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;YACtB,MAAM,IAAI,KAAK,CAAC,mBAAmB,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;QAC1D,CAAC;QAED,IACE,KAAK,CAAC,iBAAiB,KAAK,iBAAiB,CAAC,eAAe;YAC7D,CAAC,KAAK,CAAC,IAAI,EACX,CAAC;YACD,MAAM,IAAI,KAAK,CACb,iEAAiE,CAClE,CAAC;QACJ,CAAC;QAED,IAAI,CAAC,iBAAiB,GAAG,KAAK,CAAC,iBAAiB,CAAC;QACjD,IAAI,CAAC,uBAAuB,GAAG,KAAK,CAAC,uBAAuB,CAAC;QAC7D,IAAI,CAAC,WAAW,GAAG,KAAK,CAAC,WAAW,CAAC;QACrC,IAAI,CAAC,WAAW,GAAG,KAAK,CAAC,WAAW,CAAC;QACrC,IAAI,CAAC,WAAW,GAAG,KAAK,CAAC,WAAW,CAAC;QACrC,IAAI,CAAC,oBAAoB,GAAG,KAAK,CAAC,oBAAoB,CAAC;QACvD,IAAI,CAAC,wBAAwB,GAAG,KAAK,CAAC,wBAAwB,CAAC;QAC/D,IAAI,CAAC,mBAAmB,GAAG,KAAK,CAAC,mBAAmB,CAAC;QACrD,IAAI,CAAC,oBAAoB,GAAG,KAAK,CAAC,oBAAoB,CAAC;QACvD,IAAI,CAAC,cAAc,GAAG,KAAK,CAAC,cAAc,CAAC;QAC3C,IAAI,CAAC,kBAAkB,GAAG,KAAK,CAAC,kBAAkB,CAAC;QACnD,IAAI,CAAC,IAAI,GAAG,KAAK,CAAC,IAAI,CAAC;QACvB,IAAI,CAAC,IAAI,GAAG,KAAK,CAAC,IAAI,CAAC;QACvB,IAAI,CAAC,iBAAiB,GAAG,KAAK,CAAC,iBAAiB,CAAC;QACjD,IAAI,CAAC,YAAY,GAAG,KAAK,CAAC,YAAY,CAAC;QACvC,IAAI,CAAC,gBAAgB,GAAG,KAAK,CAAC,gBAAgB,CAAC;QAE/C,IAAI,iBAAiB,GAAsB;YACzC,iBAAiB,EAAE,KAAK,CAAC,iBAAiB;YAC1C,uBAAuB,EAAE,KAAK,CAAC,uBAAuB;YACtD,WAAW,EAAE,KAAK,CAAC,WAAW;YAC9B,WAAW,EAAE,KAAK,CAAC,WAAW;YAC9B,WAAW,EAAE,KAAK,CAAC,WAAW;YAC9B,cAAc,EAAE,KAAK,CAAC,cAAc;YACpC,IAAI,EAAE,KAAK,CAAC,IAAI;YAChB,wBAAwB,EAAE,KAAK,CAAC,wBAAwB;YACxD,mBAAmB,EAAE,KAAK,CAAC,mBAAmB;YAC9C,oBAAoB,EAAE,KAAK,CAAC,oBAAoB;YAChD,cAAc,EAAE,KAAK,CAAC,cAAc;YACpC,kBAAkB,EAAE,KAAK,CAAC,kBAAkB;YAC5C,OAAO,EAAE,KAAK,CAAC,IAAI,EAAE,OAAO;YAC5B,iBAAiB,EAAE,KAAK,CAAC,iBAAiB;YAC1C,YAAY,EAAE,KAAK,CAAC,YAAY;YAChC,gBAAgB,EAAE,KAAK,CAAC,gBAAgB;gBACtC,CAAC,CAAC;oBACA,gBAAgB,EAAE,KAAK,CAAC,gBAAgB,CAAC,cAAc,CAAC,GAAG,CACzD,CAAC,EAAE,EAAE,EAAE,CAAC,EAAE,CAAC,eAAe,CAC3B;oBACD,SAAS,EAAE,KAAK,CAAC,gBAAgB,CAAC,OAAO,CAAC,GAAG,CAC3C,CAAC,MAAM,EAAE,EAAE,CAAC,MAAM,CAAC,QAAQ,CAC5B;iBACF;gBACD,CAAC,CAAC,SAAS;SACd,CAAC;QAEF,IAAI,CAAC,SAAS,GAAG,IAAI,0BAAY,CAAC,IAAI,EAAE,UAAU,EAAE,iBAAiB,CAAC,CAAC;QACvE,IAAI,CAAC,iBAAiB,GAAG,IAAI,CAAC,SAAS,CAAC,qBAAqB,CAAC;QAC9D,IAAI,CAAC,QAAQ,GAAG,IAAI,CAAC,SAAS,CAAC,YAAY,CAAC;QAC5C,IAAI,CAAC,cAAc,GAAG,IAAI,CAAC,SAAS,CAAC,kBAAkB,CAAC;QACxD,IAAI,CAAC,EAAE,GAAG,IAAI,CAAC,SAAS,CAAC,MAAM,CAAC;QAChC,IAAI,CAAC,qBAAqB,GAAG,IAAI,CAAC,SAAS,CAAC,yBAAyB,CAAC;QACtE,IAAI,CAAC,uBAAuB,GAAG,IAAI,CAAC,SAAS;aAC1C,2BAAwD,CAAC;QAC5D,IAAI,CAAC,WAAW,GAAG,IAAI,CAAC,SAAS,CAAC,eAAe,CAAC;QAClD,IAAI,CAAC,MAAM,GAAG,IAAI,CAAC,SAAS,CAAC,UAAoB,CAAC;IACpD,CAAC;;AAhxBH,8BAixBC","sourcesContent":["import {\n  IPrefixList,\n  ISecurityGroup,\n  ISubnet,\n  IVpcEndpoint,\n} from 'aws-cdk-lib/aws-ec2';\nimport { CfnWorkspace, CfnWorkspaceProps } from 'aws-cdk-lib/aws-grafana';\nimport { IRole } from 'aws-cdk-lib/aws-iam';\nimport { Construct } from 'constructs';\n\n/**\n * Specifies whether the workspace can access AWS resources in this AWS account only, or whether it\n * can also access AWS resources in other accounts in the same organization. If this is\n * ORGANIZATION, the OrganizationalUnits parameter specifies which organizational units the\n * workspace can access.\n */\nexport enum AccountAccessType {\n  /**\n   * Access is limited to the current AWS account only.\n   */\n  CURRENT_ACCOUNT = 'CURRENT_ACCOUNT',\n\n  /**\n   * Access is extended to the entire AWS organization.\n   */\n  ORGANIZATION = 'ORGANIZATION',\n}\n\n/**\n * Specifies whether this workspace uses SAML 2.0, AWS IAM Identity Center, or both to authenticate\n * users for using the Grafana console within a workspace.\n *\n * @see https://docs.aws.amazon.com/grafana/latest/APIReference/API_CreateWorkspace.html\n */\nexport enum AuthenticationProviders {\n  /**\n   * AWS Single Sign-On authentication provider.\n   */\n  AWS_SSO = 'AWS_SSO',\n\n  /**\n   * Security Assertion Markup Language (SAML) authentication provider.\n   */\n  SAML = 'SAML',\n}\n\n/**\n * The configuration settings for network access to your workspace.\n */\nexport interface NetworkAccessControl {\n  /**\n   * An array of prefix list IDs. A prefix list is a list of CIDR ranges of IP addresses. The IP\n   * addresses specified are allowed to access your workspace. If the list is not included in the\n   * configuration (passed an empty array) then no IP addresses are allowed to access the\n   * workspace.\n   *\n   * Maximum of 5 prefix lists allowed.\n   */\n  readonly prefixLists?: IPrefixList[];\n\n  /**\n   * An array of Amazon VPC endpoint IDs for the workspace. You can create VPC endpoints to your\n   * Amazon Managed Grafana workspace for access from within a VPC. If a NetworkAccessConfiguration\n   * is specified then only VPC endpoints specified here are allowed to access the workspace. If\n   * you pass in an empty array of strings, then no VPCs are allowed to access the workspace.\n   *\n   * Maximum of 5 VPC endpoints allowed.\n   */\n  readonly vpcEndpoints?: IVpcEndpoint[];\n}\n\n/**\n * The AWS notification channels that Amazon Managed Grafana can automatically create IAM roles and\n * permissions for, to allow Amazon Managed Grafana to use these channels.\n */\nexport enum NotificationDestinations {\n  /**\n   * Amazon Simple Notification Service (SNS) as notification destination.\n   */\n  SNS = 'SNS',\n}\n\n/**\n * If this is SERVICE_MANAGED, and the workplace was created through the Amazon Managed Grafana\n * console, then Amazon Managed Grafana automatically creates the IAM roles and provisions the\n * permissions that the workspace needs to use AWS data sources and notification channels.\n *\n * If this is CUSTOMER_MANAGED, you must manage those roles and permissions yourself.\n\n * If you are working with a workspace in a member account of an organization and that account is\n * not a delegated administrator account, and you want the workspace to access data sources in\n * other AWS accounts in the organization, this parameter must be set to CUSTOMER_MANAGED.\n */\nexport enum PermissionTypes {\n  /**\n   * Customer-managed permissions where you manage user access to Grafana.\n   */\n  CUSTOMER_MANAGED = 'CUSTOMER_MANAGED',\n\n  /**\n   * Service-managed permissions where AWS manages user access to Grafana.\n   */\n  SERVICE_MANAGED = 'SERVICE_MANAGED',\n}\n\n/**\n * A structure that defines which attributes in the IdP assertion are to be used to define\n * information about the users authenticated by the IdP to use the workspace.\n *\n * Each attribute must be a string with length between 1 and 256 characters.\n */\nexport interface SamlAssertionAttributes {\n  /**\n   * The name of the attribute within the SAML assertion to use as the email names for SAML users.\n   *\n   * Must be between 1 and 256 characters long.\n   */\n  readonly email: string;\n  /**\n   * The name of the attribute within the SAML assertion to use as the user full \"friendly\" names\n   * for user groups.\n   *\n   * Must be between 1 and 256 characters long.\n   */\n  readonly groups: string;\n  /**\n   * The name of the attribute within the SAML assertion to use as the login names for SAML users.\n   *\n   * Must be between 1 and 256 characters long.\n   */\n  readonly login: string;\n  /**\n   * The name of the attribute within the SAML assertion to use as the user full \"friendly\" names\n   * for SAML users.\n   *\n   * Must be between 1 and 256 characters long.\n   */\n  readonly name: string;\n  /**\n   * The name of the attribute within the SAML assertion to use as the user full \"friendly\" names\n   * for the users' organizations.\n   *\n   * Must be between 1 and 256 characters long.\n   */\n  readonly org: string;\n  /**\n   * The name of the attribute within the SAML assertion to use as the user roles.\n   *\n   * Must be between 1 and 256 characters long.\n   */\n  readonly role: string;\n}\n\n/**\n * A structure containing the identity provider (IdP) metadata used to integrate the identity\n * provider with this workspace.\n */\nexport interface SamlIdpMetadata {\n  /**\n   * The URL of the location containing the IdP metadata.\n   *\n   * Must be a string with length between 1 and 2048 characters.\n   */\n  readonly url?: string;\n\n  /**\n   * The full IdP metadata, in XML format.\n   */\n  readonly xml?: string;\n}\n\n/**\n * A structure containing arrays that map group names in the SAML assertion to the Grafana Admin\n * and Editor roles in the workspace.\n */\nexport interface SamlRoleValues {\n  /**\n   * A list of groups from the SAML assertion attribute to grant the Grafana Admin role to.\n   *\n   * Maximum of 256 elements.\n   */\n  readonly admin?: string[];\n\n  /**\n   * A list of groups from the SAML assertion attribute to grant the Grafana Editor role to.\n   *\n   * Maximum of 256 elements.\n   */\n  readonly editor?: string[];\n}\n\n/**\n * If the workspace uses SAML, use this structure to map SAML assertion attributes to workspace\n * user information and define which groups in the assertion attribute are to have the Admin and\n * Editor roles in the workspace.\n */\nexport interface SamlConfiguration {\n  /**\n   * Lists which organizations defined in the SAML assertion are allowed to use the Amazon Managed\n   * Grafana workspace. If this is empty, all organizations in the assertion attribute have access.\n   *\n   * Must have between 1 and 256 elements.\n   */\n  readonly allowedOrganizations?: string[];\n\n  /**\n   * A structure that defines which attributes in the SAML assertion are to be used to define\n   * information about the users authenticated by that IdP to use the workspace.\n   */\n  readonly assertionAtrributes?: SamlAssertionAttributes;\n\n  /**\n   * A structure containing the identity provider (IdP) metadata used to integrate the identity\n   * provider with this workspace.\n   *\n   * Required field for SAML configuration.\n   */\n  readonly idpMetadata: SamlIdpMetadata;\n\n  /**\n   * How long a sign-on session by a SAML user is valid, before the user has to sign on again.\n   *\n   * Must be a positive number.\n   */\n  readonly loginValidityDuration?: number;\n\n  /**\n   * A structure containing arrays that map group names in the SAML assertion to the Grafana Admin\n   * and Editor roles in the workspace.\n   */\n  readonly roleValues?: SamlRoleValues;\n}\n\n/**\n * The configuration settings for an Amazon VPC that contains data sources for your Grafana\n * workspace to connect to.\n */\nexport interface VpcConfiguration {\n  /**\n   * The list of Amazon EC2 security groups attached to the Amazon VPC for your Grafana\n   * workspace to connect. Duplicates not allowed.\n   *\n   * Array Members: Minimum number of 1 items. Maximum number of 5 items.\n   *\n   * Required for VPC configuration.\n   */\n  readonly securityGroups: ISecurityGroup[];\n\n  /**\n   * The list of Amazon EC2 subnets created in the Amazon VPC for your Grafana workspace to\n   * connect. Duplicates not allowed.\n   *\n   * Array Members: Minimum number of 2 items. Maximum number of 6 items.\n   *\n   * Required for VPC configuration.\n   */\n  readonly subnets: ISubnet[];\n}\n\n/**\n * Properties for creating an Amazon Managed Grafana workspace.\n */\nexport interface WorkspaceProps {\n  /**\n   * Type of account access for the workspace.\n   * Required field.\n   */\n  readonly accountAccessType: AccountAccessType;\n\n  /**\n   * Authentication providers to enable for the workspace.\n   * Required field.\n   */\n  readonly authenticationProviders: AuthenticationProviders[];\n\n  /**\n   * Client token for idempotent workspace creation.\n   * Must be 1-64 characters long and contain only printable ASCII characters.\n   */\n  readonly clientToken?: string;\n\n  /**\n   * List of data sources to enable for the workspace.\n   */\n  readonly dataSources?: string[];\n\n  /**\n   * Description of the workspace.\n   * Maximum length of 2048 characters.\n   */\n  readonly description?: string;\n\n  /**\n   * Grafana version for the workspace.\n   * Must be 1-255 characters long.\n   */\n  readonly grafanaVersion?: string;\n\n  /**\n   * Name of the workspace.\n   * Must be 1-255 characters long and contain only alphanumeric characters, hyphens, dots, underscores, and tildes.\n   */\n  readonly name?: string;\n\n  /**\n   * Network access control configuration for the workspace.\n   */\n  readonly networkAccessControl?: NetworkAccessControl;\n\n  /**\n   * Notification destinations to enable for the workspace.\n   */\n  readonly notificationDestinations?: NotificationDestinations[];\n\n  /**\n   * List of organizational units to include in the workspace.\n   */\n  readonly organizationalUnits?: string[];\n\n  /**\n   * Name of the IAM role to use for the organization.\n   * Maximum length of 2048 characters.\n   */\n  readonly organizationRoleName?: string;\n\n  /**\n   * Permission type for the workspace.\n   * Required field.\n   */\n  readonly permissionType: PermissionTypes;\n\n  /**\n   * Whether to enable the Grafana plugin admin page.\n   * Default: false\n   */\n  readonly pluginAdminEnabled?: boolean;\n\n  /**\n   * IAM role to use for the workspace.\n   */\n  readonly role?: IRole;\n\n  /**\n   * SAML configuration for the workspace.\n   */\n  readonly samlConfiguration?: SamlConfiguration;\n\n  /**\n   * Name of the CloudFormation stack set to use.\n   */\n  readonly stackSetName?: string;\n\n  /**\n   * VPC configuration for the workspace.\n   */\n  readonly vpcConfiguration?: VpcConfiguration;\n}\n\n/**\n * Status of SAML configuration for a Grafana workspace.\n */\nexport enum SamlConfigurationStatuses {\n  /**\n   * SAML is configured for the workspace.\n   */\n  CONFIGURED = 'CONFIGURED',\n\n  /**\n   * SAML is not configured for the workspace.\n   */\n  NOT_CONFIGURED = 'NOT_CONFIGURED',\n}\n\n/**\n * Status of a Grafana workspace.\n */\nexport enum Status {\n  /**\n   * Workspace is active and ready to use.\n   */\n  ACTIVE = 'ACTIVE',\n\n  /**\n   * Workspace is being created.\n   */\n  CREATING = 'CREATING',\n\n  /**\n   * Workspace is being deleted.\n   */\n  DELETING = 'DELETING',\n\n  /**\n   * Workspace operation has failed.\n   */\n  FAILED = 'FAILED',\n\n  /**\n   * Workspace is being updated.\n   */\n  UPDATING = 'UPDATING',\n\n  /**\n   * Workspace is being upgraded.\n   */\n  UPGRADING = 'UPGRADING',\n\n  /**\n   * Workspace deletion has failed.\n   */\n  DELETION_FAILED = 'DELETION_FAILED',\n\n  /**\n   * Workspace creation has failed.\n   */\n  CREATION_FAILED = 'CREATION_FAILED',\n\n  /**\n   * Workspace update has failed.\n   */\n  UPDATE_FAILED = 'UPDATE_FAILED',\n\n  /**\n   * Workspace upgrade has failed.\n   */\n  UPGRADE_FAILED = 'UPGRADE_FAILED',\n\n  /**\n   * License removal has failed.\n   */\n  LICENSE_REMOVAL_FAILED = 'LICENSE_REMOVAL_FAILED',\n}\n\n/**\n * Represents an Amazon Managed Grafana workspace.\n *\n * This class provides a high-level abstraction for creating and managing\n * Amazon Managed Grafana workspaces using AWS CDK.\n */\nexport class Workspace extends Construct {\n  /**\n   * Validates the clientToken property.\n   *\n   * @param token - The client token to validate\n   * @returns An array of error messages if validation fails, or an empty array if valid\n   *\n   * Validation rules:\n   * - Must be a string\n   * - Must be between 1 and 64 characters long\n   * - Must contain only printable ASCII characters\n   */\n  private static validateClientToken(token: unknown): string[] {\n    const errors: string[] = [];\n\n    if (token === undefined) {\n      return errors; // Optional property can be undefined\n    }\n\n    if (typeof token !== 'string') {\n      errors.push('clientToken must be a string');\n      return errors; // No need to check further if not a string\n    }\n\n    const regex = /^[!-~]{1,64}$/;\n    if (!regex.test(token)) {\n      if (token.length < 1 || token.length > 64) {\n        errors.push('clientToken must be between 1 and 64 characters long');\n      }\n\n      if (!/^[!-~]*$/.test(token)) {\n        errors.push('clientToken must contain only printable ASCII characters');\n      }\n    }\n\n    return errors;\n  }\n\n  /**\n   * Validates the description property.\n   *\n   * @param description - The description to validate\n   * @returns An array of error messages if validation fails, or an empty array if valid\n   *\n   * Validation rules:\n   * - Must be a string\n   * - Maximum length of 2048 characters\n   */\n  private static validateDescription(description: unknown): string[] {\n    const errors: string[] = [];\n\n    if (description === undefined) {\n      return errors; // Optional property can be undefined\n    }\n\n    if (typeof description !== 'string') {\n      errors.push('description must be a string');\n      return errors; // No need to check further if not a string\n    }\n\n    if (description.length > 2048) {\n      errors.push('description cannot exceed 2048 characters');\n    }\n\n    return errors;\n  }\n\n  /**\n   * Validates the grafanaVersion property.\n   *\n   * @param version - The Grafana version to validate\n   * @returns An array of error messages if validation fails, or an empty array if valid\n   *\n   * Validation rules:\n   * - Must be a string\n   * - Must be between 1 and 255 characters long\n   */\n  private static validateGrafanaVersion(version: unknown): string[] {\n    const errors: string[] = [];\n\n    if (version === undefined) {\n      return errors; // Optional property can be undefined\n    }\n\n    if (typeof version !== 'string') {\n      errors.push('grafanaVersion must be a string');\n      return errors; // No need to check further if not a string\n    }\n\n    if (version.length < 1) {\n      errors.push('grafanaVersion cannot be empty');\n    }\n\n    if (version.length > 255) {\n      errors.push('grafanaVersion cannot exceed 255 characters');\n    }\n\n    return errors;\n  }\n\n  /**\n   * Validates the name property.\n   *\n   * @param name - The workspace name to validate\n   * @returns An array of error messages if validation fails, or an empty array if valid\n   *\n   * Validation rules:\n   * - Must be a string\n   * - Must be between 1 and 255 characters long\n   * - Can only contain alphanumeric characters, hyphens, dots, underscores, and tildes\n   */\n  private static validateName(name: unknown): string[] {\n    const errors: string[] = [];\n\n    if (name === undefined) {\n      return errors; // Optional property can be undefined\n    }\n\n    if (typeof name !== 'string') {\n      errors.push('name must be a string');\n      return errors; // No need to check further if not a string\n    }\n\n    const regex = /^[a-zA-Z0-9\\-._~]{1,255}$/;\n    if (!regex.test(name)) {\n      if (name.length < 1 || name.length > 255) {\n        errors.push('name must be between 1 and 255 characters long');\n      }\n\n      if (!/^[a-zA-Z0-9\\-._~]*$/.test(name)) {\n        errors.push(\n          'name can only contain alphanumeric characters, hyphens, dots, underscores, and tildes',\n        );\n      }\n    }\n\n    return errors;\n  }\n\n  /**\n   * Validates the networkAccessControl property.\n   *\n   * @param nac - The network access control configuration to validate\n   * @returns An array of error messages if validation fails, or an empty array if valid\n   *\n   * Validation rules:\n   * - Must be an object\n   * - prefixLists (if present) must be an array with at most 5 items\n   * - vpcEndpoints (if present) must be an array with at most 5 items\n   */\n  private static validateNetworkAccessControl(nac: unknown): string[] {\n    const errors: string[] = [];\n\n    if (nac === undefined) {\n      return errors; // Optional property can be undefined\n    }\n\n    if (!nac || typeof nac !== 'object') {\n      errors.push('networkAccessControl must be an object');\n      return errors;\n    }\n\n    const networkAccessControl = nac as NetworkAccessControl;\n\n    // Check prefixLists if present\n    if (networkAccessControl.prefixLists !== undefined) {\n      if (!Array.isArray(networkAccessControl.prefixLists)) {\n        errors.push('prefixLists must be an array');\n      } else if (networkAccessControl.prefixLists.length > 5) {\n        errors.push('prefixLists can have at most 5 items');\n      }\n    }\n\n    // Check vpcEndpoints if present\n    if (networkAccessControl.vpcEndpoints !== undefined) {\n      if (!Array.isArray(networkAccessControl.vpcEndpoints)) {\n        errors.push('vpcEndpoints must be an array');\n      } else if (networkAccessControl.vpcEndpoints.length > 5) {\n        errors.push('vpcEndpoints can have at most 5 items');\n      }\n    }\n\n    return errors;\n  }\n\n  /**\n   * Validates the organizationRoleName property.\n   *\n   * @param roleName - The organization role name to validate\n   * @returns An array of error messages if validation fails, or an empty array if valid\n   *\n   * Validation rules:\n   * - Must be a string\n   * - Must be between 1 and 2048 characters long\n   */\n  private static validateOrganizationRoleName(roleName: unknown): string[] {\n    const errors: string[] = [];\n\n    if (roleName === undefined) {\n      return errors; // Optional property can be undefined\n    }\n\n    if (typeof roleName !== 'string') {\n      errors.push('organizationRoleName must be a string');\n      return errors; // No need to check further if not a string\n    }\n\n    if (roleName.length < 1) {\n      errors.push('organizationRoleName cannot be empty');\n    }\n\n    if (roleName.length > 2048) {\n      errors.push('organizationRoleName cannot exceed 2048 characters');\n    }\n\n    return errors;\n  }\n\n  /**\n   * Validates the SAML assertion attributes.\n   *\n   * @param obj - The SAML assertion attributes to validate\n   * @returns An array of error messages if validation fails, or an empty array if valid\n   *\n   * Validation rules:\n   * - Must be an object\n   * - Each attribute must be a string\n   * - Each attribute must be between 1 and 256 characters long\n   * - Valid attribute keys are: 'email', 'groups', 'login', 'name', 'org', 'role'\n   */\n  private static validateSamlAssertionAttributes(obj: unknown): string[] {\n    const errors: string[] = [];\n\n    if (!obj || typeof obj !== 'object') {\n      return ['Input is not an object'];\n    }\n\n    const attributes = obj as Record<string, unknown>;\n\n    for (const key in attributes) {\n      const value = attributes[key];\n      if (value === undefined) {\n        continue; // Optional properties can be undefined\n      }\n\n      if (typeof value !== 'string') {\n        errors.push(`Property '${key}' must be a string`);\n      } else if (value.length < 1) {\n        errors.push(`Property '${key}' cannot be empty`);\n      } else if (value.length > 256) {\n        errors.push(\n          `Property '${key}' exceeds maximum length of 256 characters`,\n        );\n      }\n    }\n\n    return errors;\n  }\n\n  /**\n   * Validates the SAML IdP metadata.\n   *\n   * @param obj - The SAML IdP metadata to validate\n   * @returns An array of error messages if validation fails, or an empty array if valid\n   *\n   * Validation rules:\n   * - Must be an object\n   * - url (if present) must be a string between 1 and 2048 characters long\n   * - xml (if present) must be a string\n   */\n  private static validateSamlIdpMetadata(obj: unknown): string[] {\n    const errors: string[] = [];\n\n    if (!obj || typeof obj !== 'object') {\n      return ['Input is not an object'];\n    }\n\n    const metadata = obj as Record<string, unknown>;\n\n    // Check url property if present\n    if (metadata.url !== undefined) {\n      if (typeof metadata.url !== 'string') {\n        errors.push(\"Property 'url' must be a string\");\n      } else if (metadata.url.length < 1) {\n        errors.push(\"Property 'url' cannot be empty\");\n      } else if (metadata.url.length > 2048) {\n        errors.push(\"Property 'url' exceeds maximum length of 2048 characters\");\n      }\n    }\n\n    // Check xml property if present\n    if (metadata.xml !== undefined && typeof metadata.xml !== 'string') {\n      errors.push(\"Property 'xml' must be a string\");\n    }\n\n    return errors;\n  }\n\n  /**\n   * Validates the SAML configuration.\n   *\n   * @param config - The SAML configuration to validate\n   * @returns An array of error messages if validation fails, or an empty array if valid\n   *\n   * Validation rules:\n   * - Must be an object\n   * - idpMetadata is required and must be valid\n   * - assertionAtrributes (if present) must be valid\n   * - allowedOrganizations (if present) must be an array of strings with 1-256 elements\n   * - loginValidityDuration (if present) must be a positive number\n   * - roleValues (if present) must be an object with valid admin and editor arrays\n   */\n  private static validateSamlConfiguration(config: unknown): string[] {\n    const errors: string[] = [];\n\n    if (config === undefined) {\n      return errors; // Optional property can be undefined\n    }\n\n    if (!config || typeof config !== 'object') {\n      errors.push('samlConfiguration must be an object');\n      return errors;\n    }\n\n    const samlConfig = config as SamlConfiguration;\n\n    // Check idpMetadata (required)\n    if (samlConfig.idpMetadata === undefined) {\n      errors.push('idpMetadata is required in samlConfiguration');\n    } else {\n      const idpMetadataErrors = Workspace.validateSamlIdpMetadata(\n        samlConfig.idpMetadata,\n      );\n      if (idpMetadataErrors.length > 0) {\n        errors.push(...idpMetadataErrors.map((err) => `idpMetadata: ${err}`));\n      }\n    }\n\n    // Check assertionAtrributes if present\n    if (samlConfig.assertionAtrributes !== undefined) {\n      const attributeErrors = Workspace.validateSamlAssertionAttributes(\n        samlConfig.assertionAtrributes,\n      );\n      if (attributeErrors.length > 0) {\n        errors.push(\n          ...attributeErrors.map((err) => `assertionAtrributes: ${err}`),\n        );\n      }\n    }\n\n    // Check allowedOrganizations if present\n    if (samlConfig.allowedOrganizations !== undefined) {\n      if (!Array.isArray(samlConfig.allowedOrganizations)) {\n        errors.push('allowedOrganizations must be an array');\n      } else {\n        if (samlConfig.allowedOrganizations.length < 1) {\n          errors.push('allowedOrganizations must have at least 1 element');\n        }\n        if (samlConfig.allowedOrganizations.length > 256) {\n          errors.push(\n            'allowedOrganizations cannot have more than 256 elements',\n          );\n        }\n\n        for (let i = 0; i < samlConfig.allowedOrganizations.length; i++) {\n          const org = samlConfig.allowedOrganizations[i];\n          if (typeof org !== 'string') {\n            errors.push(`allowedOrganizations[${i}] must be a string`);\n          }\n        }\n      }\n    }\n\n    // Check loginValidityDuration if present\n    if (samlConfig.loginValidityDuration !== undefined) {\n      if (typeof samlConfig.loginValidityDuration !== 'number') {\n        errors.push('loginValidityDuration must be a number');\n      } else if (samlConfig.loginValidityDuration <= 0) {\n        errors.push('loginValidityDuration must be positive');\n      }\n    }\n\n    // Check roleValues if present\n    if (samlConfig.roleValues !== undefined) {\n      if (!samlConfig.roleValues || typeof samlConfig.roleValues !== 'object') {\n        errors.push('roleValues must be an object');\n      } else {\n        // Check admin array if present\n        if (samlConfig.roleValues.admin !== undefined) {\n          if (!Array.isArray(samlConfig.roleValues.admin)) {\n            errors.push('roleValues.admin must be an array');\n          } else {\n            for (let i = 0; i < samlConfig.roleValues.admin.length; i++) {\n              if (typeof samlConfig.roleValues.admin[i] !== 'string') {\n                errors.push(`roleValues.admin[${i}] must be a string`);\n              }\n            }\n          }\n        }\n\n        // Check editor array if present\n        if (samlConfig.roleValues.editor !== undefined) {\n          if (!Array.isArray(samlConfig.roleValues.editor)) {\n            errors.push('roleValues.editor must be an array');\n          } else {\n            for (let i = 0; i < samlConfig.roleValues.editor.length; i++) {\n              if (typeof samlConfig.roleValues.editor[i] !== 'string') {\n                errors.push(`roleValues.editor[${i}] must be a string`);\n              }\n            }\n          }\n        }\n      }\n    }\n\n    return errors;\n  }\n\n  /**\n   * Validates the vpcConfiguration property.\n   *\n   * @param config - The VPC configuration to validate\n   * @returns An array of error messages if validation fails, or an empty array if valid\n   *\n   * Validation rules:\n   * - Must be an object\n   * - securityGroups is required and must be an array with 1-5 items\n   * - subnets is required and must be an array with 2-6 items\n   */\n  private static validateVpcConfiguration(config: unknown): string[] {\n    const errors: string[] = [];\n\n    if (config === undefined) {\n      return errors; // Optional property can be undefined\n    }\n\n    if (!config || typeof config !== 'object') {\n      errors.push('vpcConfiguration must be an object');\n      return errors;\n    }\n\n    const vpcConfig = config as VpcConfiguration;\n\n    // Check securityGroups (required)\n    if (vpcConfig.securityGroups === undefined) {\n      errors.push('securityGroups is required in vpcConfiguration');\n    } else if (!Array.isArray(vpcConfig.securityGroups)) {\n      errors.push('securityGroups must be an array');\n    } else {\n      if (vpcConfig.securityGroups.length < 1) {\n        errors.push('securityGroups must have at least 1 item');\n      }\n      if (vpcConfig.securityGroups.length > 5) {\n        errors.push('securityGroups cannot have more than 5 items');\n      }\n    }\n\n    // Check subnets (required)\n    if (vpcConfig.subnets === undefined) {\n      errors.push('subnets is required in vpcConfiguration');\n    } else if (!Array.isArray(vpcConfig.subnets)) {\n      errors.push('subnets must be an array');\n    } else {\n      if (vpcConfig.subnets.length < 2) {\n        errors.push('subnets must have at least 2 items');\n      }\n      if (vpcConfig.subnets.length > 6) {\n        errors.push('subnets cannot have more than 6 items');\n      }\n    }\n\n    return errors;\n  }\n\n  /**\n   * Validates all workspace properties.\n   *\n   * @param props - The workspace properties to validate\n   * @returns An array of error messages if validation fails, or an empty array if valid\n   *\n   * This method aggregates validation results from all individual property validators.\n   * It throws an error if props is not an object.\n   */\n  private static validateProps(props: unknown): string[] {\n    const errors: string[] = [];\n\n    if (!props || typeof props !== 'object') {\n      throw new Error('Props is not an object');\n    }\n    const workspaceProps = props as WorkspaceProps;\n\n    if (workspaceProps.clientToken !== undefined) {\n      const clientTokenErrors = Workspace.validateClientToken(\n        workspaceProps.clientToken,\n      );\n      if (clientTokenErrors.length > 0) {\n        errors.push(...clientTokenErrors.map((err) => `clientToken: ${err}`));\n      }\n    }\n\n    if (workspaceProps.description !== undefined) {\n      const descriptionErrors = Workspace.validateDescription(\n        workspaceProps.description,\n      );\n      if (descriptionErrors.length > 0) {\n        errors.push(...descriptionErrors.map((err) => `description: ${err}`));\n      }\n    }\n\n    if (workspaceProps.grafanaVersion !== undefined) {\n      const grafanaVersionErrors = Workspace.validateGrafanaVersion(\n        workspaceProps.grafanaVersion,\n      );\n      if (grafanaVersionErrors.length > 0) {\n        errors.push(\n          ...grafanaVersionErrors.map((err) => `grafanaVersion: ${err}`),\n        );\n      }\n    }\n\n    if (workspaceProps.name !== undefined) {\n      const nameErrors = Workspace.validateName(workspaceProps.name);\n      if (nameErrors.length > 0) {\n        errors.push(...nameErrors.map((err) => `name: ${err}`));\n      }\n    }\n\n    if (workspaceProps.networkAccessControl !== undefined) {\n      const networkAccessControlErrors = Workspace.validateNetworkAccessControl(\n        workspaceProps.networkAccessControl,\n      );\n      if (networkAccessControlErrors.length > 0) {\n        errors.push(\n          ...networkAccessControlErrors.map(\n            (err) => `networkAccessControl: ${err}`,\n          ),\n        );\n      }\n    }\n\n    if (workspaceProps.organizationRoleName !== undefined) {\n      const organizationRoleNameErrors = Workspace.validateOrganizationRoleName(\n        workspaceProps.organizationRoleName,\n      );\n      if (organizationRoleNameErrors.length > 0) {\n        errors.push(\n          ...organizationRoleNameErrors.map(\n            (err) => `organizationRoleName: ${err}`,\n          ),\n        );\n      }\n    }\n\n    if (workspaceProps.samlConfiguration !== undefined) {\n      const samlConfigurationErrors = Workspace.validateSamlConfiguration(\n        workspaceProps.samlConfiguration,\n      );\n      if (samlConfigurationErrors.length > 0) {\n        errors.push(\n          ...samlConfigurationErrors.map((err) => `samlConfiguration: ${err}`),\n        );\n      }\n    }\n\n    if (workspaceProps.vpcConfiguration !== undefined) {\n      const vpcConfigurationErrors = Workspace.validateVpcConfiguration(\n        workspaceProps.vpcConfiguration,\n      );\n      if (vpcConfigurationErrors.length > 0) {\n        errors.push(\n          ...vpcConfigurationErrors.map((err) => `vpcConfiguration: ${err}`),\n        );\n      }\n    }\n\n    return errors;\n  }\n\n  /**\n   * The type of account access for the workspace.\n   */\n  public readonly accountAccessType: AccountAccessType;\n\n  /**\n   * Authentication providers enabled for the workspace.\n   */\n  public readonly authenticationProviders: AuthenticationProviders[];\n\n  /**\n   * Client token used for idempotent workspace creation.\n   */\n  public readonly clientToken?: string;\n\n  /**\n   * Data sources enabled for the workspace.\n   */\n  public readonly dataSources?: string[];\n\n  /**\n   * Description of the workspace.\n   */\n  public readonly description?: string;\n\n  /**\n   * Name of the workspace.\n   */\n  public readonly name?: string;\n\n  /**\n   * Network access control configuration for the workspace.\n   */\n  public readonly networkAccessControl?: NetworkAccessControl;\n\n  /**\n   * Notification destinations enabled for the workspace.\n   */\n  public readonly notificationDestinations?: NotificationDestinations[];\n\n  /**\n   * Organizational units included in the workspace.\n   */\n  public readonly organizationalUnits?: string[];\n\n  /**\n   * Name of the IAM role used for the organization.\n   */\n  public readonly organizationRoleName?: string;\n\n  /**\n   * Permission type for the workspace.\n   */\n  public readonly permissionType: PermissionTypes;\n\n  /**\n   * Whether the Grafana plugin admin page is enabled.\n   */\n  public readonly pluginAdminEnabled?: boolean;\n\n  /**\n   * IAM role used for the workspace.\n   */\n  public readonly role?: IRole;\n\n  /**\n   * SAML configuration for the workspace.\n   */\n  public readonly samlConfiguration?: SamlConfiguration;\n\n  /**\n   * Name of the CloudFormation stack set used.\n   */\n  public readonly stackSetName?: string;\n\n  /**\n   * VPC configuration for the workspace.\n   */\n  public readonly vpcConfiguration?: VpcConfiguration;\n\n  /**\n   * The underlying CloudFormation resource.\n   */\n  private readonly workspace: CfnWorkspace;\n\n  /**\n   * Timestamp when the workspace was created.\n   */\n  public readonly creationTimestamp: string;\n\n  /**\n   * Endpoint URL for the Grafana workspace.\n   */\n  public readonly endpoint: string;\n\n  /**\n   * Grafana version running in the workspace.\n   */\n  public readonly grafanaVersion: string;\n\n  /**\n   * Unique identifier for the workspace.\n   */\n  public readonly id: string;\n\n  /**\n   * Timestamp when the workspace was last modified.\n   */\n  public readonly modificationTimestamp: string;\n\n  /**\n   * Status of SAML configuration for the workspace.\n   */\n  public readonly samlConfigurationStatus: SamlConfigurationStatuses;\n\n  /**\n   * SSO client ID for the workspace.\n   */\n  public readonly ssoClientId: string;\n\n  /**\n   * Current status of the workspace.\n   */\n  public readonly status: Status;\n\n  /**\n   * Creates a new Amazon Managed Grafana workspace.\n   *\n   * @param scope - The scope in which to define this construct\n   * @param id - The scoped construct ID\n   * @param props - Configuration properties for the workspace\n   *\n   * @throws Error if any of the provided properties fail validation\n   */\n  constructor(scope: Construct, id: string, props: WorkspaceProps) {\n    super(scope, id);\n\n    const errors = Workspace.validateProps(props);\n    if (errors.length > 0) {\n      throw new Error(`Invalid props:\\n${errors.join('\\n')}`);\n    }\n\n    if (\n      props.accountAccessType === AccountAccessType.CURRENT_ACCOUNT &&\n      !props.role\n    ) {\n      throw new Error(\n        'Role must be provided when accountAccessType is CURRENT_ACCOUNT',\n      );\n    }\n\n    this.accountAccessType = props.accountAccessType;\n    this.authenticationProviders = props.authenticationProviders;\n    this.clientToken = props.clientToken;\n    this.dataSources = props.dataSources;\n    this.description = props.description;\n    this.networkAccessControl = props.networkAccessControl;\n    this.notificationDestinations = props.notificationDestinations;\n    this.organizationalUnits = props.organizationalUnits;\n    this.organizationRoleName = props.organizationRoleName;\n    this.permissionType = props.permissionType;\n    this.pluginAdminEnabled = props.pluginAdminEnabled;\n    this.name = props.name;\n    this.role = props.role;\n    this.samlConfiguration = props.samlConfiguration;\n    this.stackSetName = props.stackSetName;\n    this.vpcConfiguration = props.vpcConfiguration;\n\n    let cfnWorkspaceProps: CfnWorkspaceProps = {\n      accountAccessType: props.accountAccessType,\n      authenticationProviders: props.authenticationProviders,\n      clientToken: props.clientToken,\n      dataSources: props.dataSources,\n      description: props.description,\n      grafanaVersion: props.grafanaVersion,\n      name: props.name,\n      notificationDestinations: props.notificationDestinations,\n      organizationalUnits: props.organizationalUnits,\n      organizationRoleName: props.organizationRoleName,\n      permissionType: props.permissionType,\n      pluginAdminEnabled: props.pluginAdminEnabled,\n      roleArn: props.role?.roleArn,\n      samlConfiguration: props.samlConfiguration,\n      stackSetName: props.stackSetName,\n      vpcConfiguration: props.vpcConfiguration\n        ? {\n          securityGroupIds: props.vpcConfiguration.securityGroups.map(\n            (sg) => sg.securityGroupId,\n          ),\n          subnetIds: props.vpcConfiguration.subnets.map(\n            (subnet) => subnet.subnetId,\n          ),\n        }\n        : undefined,\n    };\n\n    this.workspace = new CfnWorkspace(this, 'Resource', cfnWorkspaceProps);\n    this.creationTimestamp = this.workspace.attrCreationTimestamp;\n    this.endpoint = this.workspace.attrEndpoint;\n    this.grafanaVersion = this.workspace.attrGrafanaVersion;\n    this.id = this.workspace.attrId;\n    this.modificationTimestamp = this.workspace.attrModificationTimestamp;\n    this.samlConfigurationStatus = this.workspace\n      .attrSamlConfigurationStatus as SamlConfigurationStatuses;\n    this.ssoClientId = this.workspace.attrSsoClientId;\n    this.status = this.workspace.attrStatus as Status;\n  }\n}\n"]}