UNPKG

@pepperize/cdk-organizations

Version:

Manage AWS organizations, organizational units (OU), accounts and service control policies (SCP).

202 lines • 38.7 kB
"use strict"; var _a, _b; Object.defineProperty(exports, "__esModule", { value: true }); exports.Root = exports.Organization = exports.FeatureSet = void 0; const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti"); const aws_cdk_lib_1 = require("aws-cdk-lib"); const aws_iam = require("aws-cdk-lib/aws-iam"); const custom_resources = require("aws-cdk-lib/custom-resources"); const constructs_1 = require("constructs"); const pascal_case_1 = require("pascal-case"); const dependency_chain_1 = require("./dependency-chain"); const enable_aws_service_access_1 = require("./enable-aws-service-access"); const enable_policy_type_1 = require("./enable-policy-type"); const organization_provider_1 = require("./organization-provider"); const policy_attachment_1 = require("./policy-attachment"); const tag_resource_1 = require("./tag-resource"); /** * Specifies the feature set supported by the new organization. Each feature set supports different levels of functionality. * * @see https://docs.aws.amazon.com/organizations/latest/userguide/orgs_getting-started_concepts.html#feature-set */ var FeatureSet; (function (FeatureSet) { /** * All member accounts have their bills consolidated to and paid by the management account. For more information, see [Consolidated billing](https://docs.aws.amazon.com/organizations/latest/userguide/orgs_getting-started_concepts.html#feature-set-cb-only) in the AWS Organizations User Guide. The consolidated billing feature subset isn’t available for organizations in the AWS GovCloud (US) Region. */ FeatureSet["CONSOLIDATED_BILLING"] = "CONSOLIDATED_BILLING"; /** * In addition to all the features supported by the consolidated billing feature set, the management account can also apply any policy type to any member account in the organization. For more information, see [All features](https://docs.aws.amazon.com/organizations/latest/userguide/orgs_getting-started_concepts.html#feature-set-all) in the AWS Organizations User Guide. */ FeatureSet["ALL"] = "ALL"; })(FeatureSet || (exports.FeatureSet = FeatureSet = {})); class Organization extends constructs_1.Construct { /** * Describe the organization that the current account belongs to. * * @see https://docs.aws.amazon.com/organizations/latest/APIReference/API_DescribeOrganization.html */ static of(scope, id) { class Import extends constructs_1.Construct { constructor() { super(scope, id); const organizationsRegion = process.env.CDK_AWS_PARTITION === "aws-cn" ? "cn-northwest-1" : "us-east-1"; const resource = new custom_resources.AwsCustomResource(scope, "CustomResource", { resourceType: "Custom::Organizations_ImportOrganization", onCreate: { service: "Organizations", action: "describeOrganization", // https://docs.aws.amazon.com/AWSJavaScriptSDK/latest/AWS/Organizations.html#describeOrganization-property region: organizationsRegion, parameters: {}, physicalResourceId: custom_resources.PhysicalResourceId.fromResponse("Organization.Id"), }, onUpdate: { service: "Organizations", action: "describeOrganization", // https://docs.aws.amazon.com/AWSJavaScriptSDK/latest/AWS/Organizations.html#describeOrganization-property region: organizationsRegion, parameters: {}, physicalResourceId: custom_resources.PhysicalResourceId.fromResponse("Organization.Id"), }, installLatestAwsSdk: false, policy: custom_resources.AwsCustomResourcePolicy.fromSdkCalls({ resources: custom_resources.AwsCustomResourcePolicy.ANY_RESOURCE, }), }); this.featureSet = resource.getResponseField("Organization.FeatureSet"); this.managementAccountArn = resource.getResponseField("Organization.MasterAccountArn"); this.managementAccountEmail = resource.getResponseField("Organization.MasterAccountEmail"); this.managementAccountId = resource.getResponseField("Organization.MasterAccountId"); this.organizationArn = resource.getResponseField("Organization.Arn"); this.organizationId = resource.getResponseField("Organization.Id"); this.principal = new aws_iam.OrganizationPrincipal(this.organizationId); } } return new Import(); } constructor(scope, id, props = {}) { super(scope, id); const featureSet = props.featureSet || FeatureSet.ALL; const organizationProvider = organization_provider_1.OrganizationProvider.getOrCreate(this); this.resource = new aws_cdk_lib_1.CustomResource(this, "Organization", { serviceToken: organizationProvider.provider.serviceToken, resourceType: "Custom::Organizations_Organization", properties: { FeatureSet: featureSet, }, }); this.organizationId = this.resource.getAtt("Id").toString(); this.organizationArn = this.resource.getAtt("Arn").toString(); this.featureSet = this.resource.getAtt("FeatureSet").toString(); this.managementAccountArn = this.resource.getAtt("MasterAccountArn").toString(); this.managementAccountId = this.resource.getAtt("MasterAccountId").toString(); this.managementAccountEmail = this.resource.getAtt("MasterAccountEmail").toString(); this.principal = new aws_iam.OrganizationPrincipal(this.organizationId); this.root = new Root(this, "Root"); this.root.node.addDependency(this.resource); } /** * Enables trusted access for a supported AWS service (trusted service), which performs tasks in your organization and its accounts on your behalf. * @param servicePrincipal The supported AWS service that you specify * * @see https://docs.aws.amazon.com/organizations/latest/userguide/orgs_integrate_services_list.html */ enableAwsServiceAccess(servicePrincipal) { const enableAwsServiceAccess = new enable_aws_service_access_1.EnableAwsServiceAccess(this, `Enable${(0, pascal_case_1.pascalCase)(servicePrincipal)}`, { servicePrincipal: servicePrincipal, }); enableAwsServiceAccess.node.addDependency(this.resource); } /** * Enables policy types in the following two broad categories: Authorization policies and Management policies. * @param policyType: the type of the policy that you specify * * @see https://docs.aws.amazon.com/organizations/latest/userguide/orgs_manage_policies.html#orgs-policy-types */ enablePolicyType(policyType) { this.root.enablePolicyType(policyType); } /** * Attach a policy. Before you can attach the policy, you must enable that policy type for use. You can use policies when you have all features enabled. * * @see https://docs.aws.amazon.com/organizations/latest/userguide/orgs_manage_policies.html */ attachPolicy(policy) { this.root.attachPolicy(policy); } } exports.Organization = Organization; _a = JSII_RTTI_SYMBOL_1; Organization[_a] = { fqn: "@pepperize/cdk-organizations.Organization", version: "0.7.988" }; /** * The parent container for all the accounts for your organization. If you apply a policy to the root, it applies to all organizational units (OUs) and accounts in the organization. * <strong>Currently, you can have only one root. AWS Organizations automatically creates it for you when you create an organization.</strong> * @see https://docs.aws.amazon.com/organizations/latest/userguide/orgs_getting-started_concepts.html */ class Root extends constructs_1.Construct { constructor(scope, id) { super(scope, id); this.tags = new aws_cdk_lib_1.TagManager(aws_cdk_lib_1.TagType.KEY_VALUE, "Custom::Organizations_Root"); this.scope = scope; const organizationsRegion = process.env.CDK_AWS_PARTITION === "aws-cn" ? "cn-northwest-1" : "us-east-1"; this.resource = new custom_resources.AwsCustomResource(this, "RootCustomResource", { resourceType: "Custom::Organizations_Root", onCreate: { service: "Organizations", action: "listRoots", // https://docs.aws.amazon.com/AWSJavaScriptSDK/latest/AWS/Organizations.html#listRoots-property region: organizationsRegion, physicalResourceId: custom_resources.PhysicalResourceId.fromResponse("Roots.0.Id"), }, onUpdate: { service: "Organizations", action: "listRoots", // https://docs.aws.amazon.com/AWSJavaScriptSDK/latest/AWS/Organizations.html#listRoots-property region: organizationsRegion, physicalResourceId: custom_resources.PhysicalResourceId.fromResponse("Roots.0.Id"), }, onDelete: { service: "Organizations", action: "listRoots", // https://docs.aws.amazon.com/AWSJavaScriptSDK/latest/AWS/Organizations.html#listRoots-property region: organizationsRegion, }, installLatestAwsSdk: false, policy: custom_resources.AwsCustomResourcePolicy.fromSdkCalls({ resources: custom_resources.AwsCustomResourcePolicy.ANY_RESOURCE, }), }); this.rootId = this.resource.getResponseField("Roots.0.Id"); // Returns first root id. It seems AWS Organizations doesn't contain multiple roots. const stack = aws_cdk_lib_1.Stack.of(this); aws_cdk_lib_1.Aspects.of(stack).add(new dependency_chain_1.DependencyChain()); // sequentially chain organization resources which can't be deployed in parallel const tagResource = new tag_resource_1.TagResource(this, "Tags", { resourceId: this.rootId, tags: this.tags.renderedTags }); tagResource.node.addDependency(this.resource); } identifier() { return this.rootId; } /** * Attach a policy. Before you can attach the policy, you must enable that policy type for use. You can use policies when you have all features enabled. * * @see https://docs.aws.amazon.com/organizations/latest/userguide/orgs_manage_policies.html */ attachPolicy(policy) { const policyAttachment = new policy_attachment_1.PolicyAttachment(this.scope, `PolicyAttachment-${aws_cdk_lib_1.Names.nodeUniqueId(this.node)}-${aws_cdk_lib_1.Names.nodeUniqueId(policy.node)}`, { target: this, policy: policy, }); policyAttachment.node.addDependency(this.resource, policy); } /** * Enables and disables Enables a policy type. After you enable a policy type in a root, you can attach policies of that type to the root, any organizational unit (OU), or account in that root. * * @see https://docs.aws.amazon.com/organizations/latest/userguide/orgs_manage_policies_enable-disable.html */ enablePolicyType(policyType) { const enablePolicyType = new enable_policy_type_1.EnablePolicyType(this.scope, `Enable${(0, pascal_case_1.pascalCase)(policyType)}`, { root: this, policyType: policyType, }); enablePolicyType.node.addDependency(this.resource); } } exports.Root = Root; _b = JSII_RTTI_SYMBOL_1; Root[_b] = { fqn: "@pepperize/cdk-organizations.Root", version: "0.7.988" }; //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"organization.js","sourceRoot":"","sources":["../src/organization.ts"],"names":[],"mappings":"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","sourcesContent":["import { Aspects, CustomResource, Names, Stack, TagManager, TagType } from \"aws-cdk-lib\";\nimport * as aws_iam from \"aws-cdk-lib/aws-iam\";\nimport * as custom_resources from \"aws-cdk-lib/custom-resources\";\nimport { Construct, IConstruct } from \"constructs\";\nimport { pascalCase } from \"pascal-case\";\nimport { DependencyChain } from \"./dependency-chain\";\nimport { EnableAwsServiceAccess } from \"./enable-aws-service-access\";\nimport { EnablePolicyType } from \"./enable-policy-type\";\nimport { OrganizationProvider } from \"./organization-provider\";\nimport { IParent } from \"./parent\";\nimport { IPolicy, PolicyType } from \"./policy\";\nimport { IPolicyAttachmentTarget, PolicyAttachment } from \"./policy-attachment\";\nimport { ITaggableResource, TagResource } from \"./tag-resource\";\n\n/**\n * Specifies the feature set supported by the new organization. Each feature set supports different levels of functionality.\n *\n * @see https://docs.aws.amazon.com/organizations/latest/userguide/orgs_getting-started_concepts.html#feature-set\n */\nexport enum FeatureSet {\n  /**\n   * All member accounts have their bills consolidated to and paid by the management account. For more information, see [Consolidated billing](https://docs.aws.amazon.com/organizations/latest/userguide/orgs_getting-started_concepts.html#feature-set-cb-only) in the AWS Organizations User Guide. The consolidated billing feature subset isn’t available for organizations in the AWS GovCloud (US) Region.\n   */\n  CONSOLIDATED_BILLING = \"CONSOLIDATED_BILLING\",\n  /**\n   * In addition to all the features supported by the consolidated billing feature set, the management account can also apply any policy type to any member account in the organization. For more information, see [All features](https://docs.aws.amazon.com/organizations/latest/userguide/orgs_getting-started_concepts.html#feature-set-all) in the AWS Organizations User Guide.\n   */\n  ALL = \"ALL\",\n}\n\nexport interface OrganizationProps {\n  /**\n   * Enabling features in your organization.\n   *\n   * @see https://docs.aws.amazon.com/organizations/latest/userguide/orgs_manage_org_support-all-features.html\n   *\n   * @default ALL\n   */\n  readonly featureSet?: FeatureSet;\n}\n\n/**\n * Creates an organization to consolidate your AWS accounts so that you can administer them as a single unit. An organization has one management account along with zero or more member accounts. You can organize the accounts in a hierarchical, tree-like structure with a root at the top and organizational units nested under the root. Each account can be directly in the root, or placed in one of the OUs in the hierarchy. An organization has the functionality that is determined by the feature set that you enable.\n *\n * <strong>The account whose user is calling the CreateOrganization operation automatically becomes the management account of the new organization.</strong>\n *\n * <strong>For deletion of an organization you must previously remove all the member accounts, OUs, and policies from the organization!</strong>\n *\n * @see https://docs.aws.amazon.com/organizations/latest/userguide/orgs_manage_org_create.html#create-org\n */\nexport interface IOrganization extends IConstruct {\n  /**\n   * The unique identifier (ID) of an organization. The regex pattern for an organization ID string requires \"o-\" followed by from 10 to 32 lowercase letters or digits.\n   */\n  readonly organizationId: string;\n  /**\n   * The Amazon Resource Name (ARN) of an organization.\n   */\n  readonly organizationArn: string;\n  /**\n   * Specifies the functionality that currently is available to the organization. If set to \"ALL\", then all features are enabled and policies can be applied to accounts in the organization. If set to \"CONSOLIDATED_BILLING\", then only consolidated billing functionality is available.\n   */\n  readonly featureSet: FeatureSet;\n  /**\n   * The Amazon Resource Name (ARN) of the account that is designated as the management account for the organization.\n   */\n  readonly managementAccountArn: string;\n  /**\n   * The unique identifier (ID) of the management account of an organization.\n   */\n  readonly managementAccountId: string;\n  /**\n   * The email address that is associated with the AWS account that is designated as the management account for the organization.\n   */\n  readonly managementAccountEmail: string;\n  /**\n   * The principal that represents this AWS Organization\n   */\n  readonly principal: aws_iam.IPrincipal;\n}\n\nexport class Organization extends Construct implements IOrganization {\n  /**\n   * Describe the organization that the current account belongs to.\n   *\n   * @see https://docs.aws.amazon.com/organizations/latest/APIReference/API_DescribeOrganization.html\n   */\n  public static of(scope: Construct, id: string): IOrganization {\n    class Import extends Construct implements IOrganization {\n      readonly featureSet: FeatureSet;\n      readonly managementAccountArn: string;\n      readonly managementAccountEmail: string;\n      readonly managementAccountId: string;\n      readonly organizationArn: string;\n      readonly organizationId: string;\n      readonly principal: aws_iam.IPrincipal;\n\n      public constructor() {\n        super(scope, id);\n\n        const organizationsRegion = process.env.CDK_AWS_PARTITION === \"aws-cn\" ? \"cn-northwest-1\" : \"us-east-1\";\n\n        const resource = new custom_resources.AwsCustomResource(scope, \"CustomResource\", {\n          resourceType: \"Custom::Organizations_ImportOrganization\",\n          onCreate: {\n            service: \"Organizations\",\n            action: \"describeOrganization\", // https://docs.aws.amazon.com/AWSJavaScriptSDK/latest/AWS/Organizations.html#describeOrganization-property\n            region: organizationsRegion,\n            parameters: {},\n            physicalResourceId: custom_resources.PhysicalResourceId.fromResponse(\"Organization.Id\"),\n          },\n          onUpdate: {\n            service: \"Organizations\",\n            action: \"describeOrganization\", // https://docs.aws.amazon.com/AWSJavaScriptSDK/latest/AWS/Organizations.html#describeOrganization-property\n            region: organizationsRegion,\n            parameters: {},\n            physicalResourceId: custom_resources.PhysicalResourceId.fromResponse(\"Organization.Id\"),\n          },\n          installLatestAwsSdk: false,\n          policy: custom_resources.AwsCustomResourcePolicy.fromSdkCalls({\n            resources: custom_resources.AwsCustomResourcePolicy.ANY_RESOURCE,\n          }),\n        });\n\n        this.featureSet = resource.getResponseField(\"Organization.FeatureSet\") as FeatureSet;\n        this.managementAccountArn = resource.getResponseField(\"Organization.MasterAccountArn\");\n        this.managementAccountEmail = resource.getResponseField(\"Organization.MasterAccountEmail\");\n        this.managementAccountId = resource.getResponseField(\"Organization.MasterAccountId\");\n        this.organizationArn = resource.getResponseField(\"Organization.Arn\");\n        this.organizationId = resource.getResponseField(\"Organization.Id\");\n        this.principal = new aws_iam.OrganizationPrincipal(this.organizationId);\n      }\n    }\n\n    return new Import();\n  }\n\n  public readonly organizationId: string;\n  public readonly organizationArn: string;\n  public readonly featureSet: FeatureSet;\n  public readonly managementAccountArn: string;\n  public readonly managementAccountId: string;\n  public readonly managementAccountEmail: string;\n  readonly principal: aws_iam.IPrincipal;\n  /**\n   * The root of the current organization, which is automatically created.\n   */\n  readonly root: Root;\n\n  private readonly resource: CustomResource;\n\n  public constructor(scope: Construct, id: string, props: OrganizationProps = {}) {\n    super(scope, id);\n\n    const featureSet = props.featureSet || FeatureSet.ALL;\n\n    const organizationProvider = OrganizationProvider.getOrCreate(this);\n    this.resource = new CustomResource(this, \"Organization\", {\n      serviceToken: organizationProvider.provider.serviceToken,\n      resourceType: \"Custom::Organizations_Organization\",\n      properties: {\n        FeatureSet: featureSet,\n      },\n    });\n\n    this.organizationId = this.resource.getAtt(\"Id\").toString();\n    this.organizationArn = this.resource.getAtt(\"Arn\").toString();\n    this.featureSet = this.resource.getAtt(\"FeatureSet\").toString() as FeatureSet;\n    this.managementAccountArn = this.resource.getAtt(\"MasterAccountArn\").toString();\n    this.managementAccountId = this.resource.getAtt(\"MasterAccountId\").toString();\n    this.managementAccountEmail = this.resource.getAtt(\"MasterAccountEmail\").toString();\n    this.principal = new aws_iam.OrganizationPrincipal(this.organizationId);\n\n    this.root = new Root(this, \"Root\");\n    this.root.node.addDependency(this.resource);\n  }\n\n  /**\n   * Enables trusted access for a supported AWS service (trusted service), which performs tasks in your organization and its accounts on your behalf.\n   * @param servicePrincipal The supported AWS service that you specify\n   *\n   * @see https://docs.aws.amazon.com/organizations/latest/userguide/orgs_integrate_services_list.html\n   */\n  public enableAwsServiceAccess(servicePrincipal: string) {\n    const enableAwsServiceAccess = new EnableAwsServiceAccess(this, `Enable${pascalCase(servicePrincipal)}`, {\n      servicePrincipal: servicePrincipal,\n    });\n    enableAwsServiceAccess.node.addDependency(this.resource);\n  }\n\n  /**\n   * Enables policy types in the following two broad categories: Authorization policies and Management policies.\n   * @param policyType: the type of the policy that you specify\n   *\n   * @see https://docs.aws.amazon.com/organizations/latest/userguide/orgs_manage_policies.html#orgs-policy-types\n   */\n  public enablePolicyType(policyType: PolicyType) {\n    this.root.enablePolicyType(policyType);\n  }\n\n  /**\n   * Attach a policy. Before you can attach the policy, you must enable that policy type for use. You can use policies when you have all features enabled.\n   *\n   * @see https://docs.aws.amazon.com/organizations/latest/userguide/orgs_manage_policies.html\n   */\n  public attachPolicy(policy: IPolicy) {\n    this.root.attachPolicy(policy);\n  }\n}\n\n/**\n * The parent container for all the accounts for your organization. If you apply a policy to the root, it applies to all organizational units (OUs) and accounts in the organization.\n * <strong>Currently, you can have only one root. AWS Organizations automatically creates it for you when you create an organization.</strong>\n * @see https://docs.aws.amazon.com/organizations/latest/userguide/orgs_getting-started_concepts.html\n */\nexport class Root extends Construct implements IParent, IPolicyAttachmentTarget, ITaggableResource {\n  /**\n   * The unique identifier (ID) for the root. The regex pattern for a root ID string requires \"r-\" followed by from 4 to 32 lowercase letters or digits.\n   */\n  public readonly rootId: string;\n\n  protected readonly resource: custom_resources.AwsCustomResource;\n\n  private readonly scope: Construct;\n\n  readonly tags = new TagManager(TagType.KEY_VALUE, \"Custom::Organizations_Root\");\n\n  public constructor(scope: Construct, id: string) {\n    super(scope, id);\n    this.scope = scope;\n\n    const organizationsRegion = process.env.CDK_AWS_PARTITION === \"aws-cn\" ? \"cn-northwest-1\" : \"us-east-1\";\n\n    this.resource = new custom_resources.AwsCustomResource(this, \"RootCustomResource\", {\n      resourceType: \"Custom::Organizations_Root\",\n      onCreate: {\n        service: \"Organizations\",\n        action: \"listRoots\", // https://docs.aws.amazon.com/AWSJavaScriptSDK/latest/AWS/Organizations.html#listRoots-property\n        region: organizationsRegion,\n        physicalResourceId: custom_resources.PhysicalResourceId.fromResponse(\"Roots.0.Id\"),\n      },\n      onUpdate: {\n        service: \"Organizations\",\n        action: \"listRoots\", // https://docs.aws.amazon.com/AWSJavaScriptSDK/latest/AWS/Organizations.html#listRoots-property\n        region: organizationsRegion,\n        physicalResourceId: custom_resources.PhysicalResourceId.fromResponse(\"Roots.0.Id\"),\n      },\n      onDelete: {\n        service: \"Organizations\",\n        action: \"listRoots\", // https://docs.aws.amazon.com/AWSJavaScriptSDK/latest/AWS/Organizations.html#listRoots-property\n        region: organizationsRegion,\n      },\n      installLatestAwsSdk: false,\n      policy: custom_resources.AwsCustomResourcePolicy.fromSdkCalls({\n        resources: custom_resources.AwsCustomResourcePolicy.ANY_RESOURCE,\n      }),\n    });\n\n    this.rootId = this.resource.getResponseField(\"Roots.0.Id\"); // Returns first root id. It seems AWS Organizations doesn't contain multiple roots.\n\n    const stack = Stack.of(this);\n    Aspects.of(stack).add(new DependencyChain()); // sequentially chain organization resources which can't be deployed in parallel\n\n    const tagResource = new TagResource(this, \"Tags\", { resourceId: this.rootId, tags: this.tags.renderedTags });\n    tagResource.node.addDependency(this.resource);\n  }\n\n  public identifier(): string {\n    return this.rootId;\n  }\n\n  /**\n   * Attach a policy. Before you can attach the policy, you must enable that policy type for use. You can use policies when you have all features enabled.\n   *\n   * @see https://docs.aws.amazon.com/organizations/latest/userguide/orgs_manage_policies.html\n   */\n  public attachPolicy(policy: IPolicy) {\n    const policyAttachment = new PolicyAttachment(\n      this.scope,\n      `PolicyAttachment-${Names.nodeUniqueId(this.node)}-${Names.nodeUniqueId(policy.node)}`,\n      {\n        target: this,\n        policy: policy,\n      }\n    );\n    policyAttachment.node.addDependency(this.resource, policy);\n  }\n\n  /**\n   * Enables and disables Enables a policy type. After you enable a policy type in a root, you can attach policies of that type to the root, any organizational unit (OU), or account in that root.\n   *\n   * @see https://docs.aws.amazon.com/organizations/latest/userguide/orgs_manage_policies_enable-disable.html\n   */\n  public enablePolicyType(policyType: PolicyType) {\n    const enablePolicyType = new EnablePolicyType(this.scope, `Enable${pascalCase(policyType)}`, {\n      root: this,\n      policyType: policyType,\n    });\n    enablePolicyType.node.addDependency(this.resource);\n  }\n}\n"]}