@pepperize/cdk-organizations
Version:
Manage AWS organizations, organizational units (OU), accounts and service control policies (SCP).
101 lines • 20.9 kB
JavaScript
;
var _a;
Object.defineProperty(exports, "__esModule", { value: true });
exports.Account = exports.IamUserAccessToBilling = void 0;
const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
const aws_cdk_lib_1 = require("aws-cdk-lib");
const constructs_1 = require("constructs");
const pascal_case_1 = require("pascal-case");
const account_provider_1 = require("./account-provider");
const delegated_administrator_1 = require("./delegated-administrator");
const policy_attachment_1 = require("./policy-attachment");
const tag_resource_1 = require("./tag-resource");
const validators_1 = require("./validators");
/**
* @see https://docs.aws.amazon.com/awsaccountbilling/latest/aboutv2/control-access-billing.html#ControllingAccessWebsite-Activate
*/
var IamUserAccessToBilling;
(function (IamUserAccessToBilling) {
/**
* If set to ALLOW, the new account enables IAM users to access account billing information if they have the required permissions.
*/
IamUserAccessToBilling["ALLOW"] = "ALLOW";
/**
* If set to DENY, only the root user of the new account can access account billing information.
*/
IamUserAccessToBilling["DENY"] = "DENY";
})(IamUserAccessToBilling || (exports.IamUserAccessToBilling = IamUserAccessToBilling = {}));
/**
* Creates or imports an AWS account that is automatically a member of the organization whose credentials made the request. AWS Organizations automatically copies the information from the management account to the new member account
*/
class Account extends constructs_1.Construct {
constructor(scope, id, props) {
super(scope, id);
this.tags = new aws_cdk_lib_1.TagManager(aws_cdk_lib_1.TagType.KEY_VALUE, "Custom::Organizations_Account");
this.scope = scope;
const { email, accountName, roleName, iamUserAccessToBilling, parent, importOnDuplicate, removalPolicy } = props;
if (!validators_1.Validators.of().email(email)) {
aws_cdk_lib_1.Annotations.of(this).addError("The account's email must be of type string and between 6 and 64 characters long.");
}
if (!validators_1.Validators.of().accountName(accountName)) {
aws_cdk_lib_1.Annotations.of(this).addError("The account's name must be of type string and between 1 and 50 characters long.");
}
const createAccountProvider = account_provider_1.AccountProvider.getOrCreate(this);
const account = new aws_cdk_lib_1.CustomResource(this, "CreateAccount", {
serviceToken: createAccountProvider.provider.serviceToken,
resourceType: "Custom::Organizations_Account",
properties: {
Email: email,
AccountName: accountName,
RoleName: roleName ?? "OrganizationAccountAccessRole",
IamUserAccessToBilling: iamUserAccessToBilling ?? IamUserAccessToBilling.ALLOW,
ParentId: parent?.identifier(),
ImportOnDuplicate: String(importOnDuplicate ?? true),
RemovalPolicy: removalPolicy ?? aws_cdk_lib_1.RemovalPolicy.RETAIN,
},
});
this.accountId = account.getAtt("AccountId").toString();
this.accountArn = account.getAtt("AccountArn").toString();
this.accountName = account.getAtt("AccountName").toString();
this.email = account.getAtt("Email").toString();
this.resource = account;
const tagResource = new tag_resource_1.TagResource(this, "Tags", { resourceId: this.accountId, tags: this.tags.renderedTags });
tagResource.node.addDependency(account);
}
identifier() {
return this.accountId;
}
/**
* Enables trusted access for the AWS service (trusted service) as <strong>Delegated Administrator</strong>, which performs tasks in your organization and its accounts on your behalf.
*
* @param {string} servicePrincipal The supported AWS service that you specify
* @param {string} region The region to delegate in
* @param {DelegatedAdministratorProps} props additional DelegatedAdministrator props
*/
delegateAdministrator(servicePrincipal, region, props = {}) {
const organizationsRegion = process.env.CDK_AWS_PARTITION === "aws-cn" ? "cn-northwest-1" : "us-east-1";
const delegatedAdministrator = new delegated_administrator_1.DelegatedAdministrator(this.scope, `Delegate${(0, pascal_case_1.pascalCase)(servicePrincipal)}${region && region !== organizationsRegion ? `-${region}` : ""}-${aws_cdk_lib_1.Names.nodeUniqueId(this.node)}`, {
...props,
account: this,
servicePrincipal: servicePrincipal,
region,
});
delegatedAdministrator.node.addDependency(this.resource);
}
/**
* 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);
}
}
exports.Account = Account;
_a = JSII_RTTI_SYMBOL_1;
Account[_a] = { fqn: "@pepperize/cdk-organizations.Account", version: "0.7.988" };
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"account.js","sourceRoot":"","sources":["../src/account.ts"],"names":[],"mappings":"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","sourcesContent":["import { Annotations, CustomResource, Names, RemovalPolicy, TagManager, TagType } from \"aws-cdk-lib\";\nimport { Construct, IConstruct } from \"constructs\";\nimport { pascalCase } from \"pascal-case\";\nimport { AccountProvider } from \"./account-provider\";\nimport { DelegatedAdministrator } from \"./delegated-administrator\";\nimport { IChild, IParent } from \"./parent\";\nimport { IPolicy } from \"./policy\";\nimport { IPolicyAttachmentTarget, PolicyAttachment } from \"./policy-attachment\";\nimport { IResource } from \"./resource\";\nimport { ITaggableResource, TagResource } from \"./tag-resource\";\nimport { Validators } from \"./validators\";\n\n/**\n * @see https://docs.aws.amazon.com/awsaccountbilling/latest/aboutv2/control-access-billing.html#ControllingAccessWebsite-Activate\n */\nexport enum IamUserAccessToBilling {\n  /**\n   * If set to ALLOW, the new account enables IAM users to access account billing information if they have the required permissions.\n   */\n  ALLOW = \"ALLOW\",\n  /**\n   * If set to DENY, only the root user of the new account can access account billing information.\n   */\n  DENY = \"DENY\",\n}\n\nexport interface AccountProps {\n  /**\n   * The email address of the owner to assign to the new member account. This email address must not already be associated with another AWS account. You must use a valid email address to complete account creation. You can't access the root user of the account or remove an account that was created with an invalid email address.\n   */\n  readonly email: string;\n  /**\n   * The friendly name of the member account.\n   */\n  readonly accountName: string;\n  /**\n   * The name of an IAM role that AWS Organizations automatically preconfigures in the new member account. This role trusts the management account, allowing users in the management account to assume the role, as permitted by the management account administrator. The role has administrator permissions in the new member account.\n   *\n   * If you don't specify this parameter, the role name defaults to OrganizationAccountAccessRole.\n   */\n  readonly roleName?: string;\n  /**\n   * If set to ALLOW , the new account enables IAM users to access account billing information if they have the required permissions. If set to DENY , only the root user of the new account can access account billing information.\n   *\n   * @default ALLOW\n   */\n  readonly iamUserAccessToBilling?: IamUserAccessToBilling;\n\n  /**\n   * The parent root or OU that you want to create the new Account in.\n   */\n  readonly parent?: IParent;\n  /**\n   * Whether to import, if a duplicate account with same name and email already exists.\n   *\n   * @default true\n   */\n  readonly importOnDuplicate?: boolean;\n  /**\n   * If set to RemovalPolicy.DESTROY, the account will be moved to the root.\n   *\n   * @default RemovalPolicy.Retain\n   */\n  readonly removalPolicy?: RemovalPolicy;\n}\n\nexport interface IAccount extends IPolicyAttachmentTarget, IChild, IConstruct, IResource {\n  /**\n   * If the account was created successfully, the unique identifier (ID) of the new account. Exactly 12 digits.\n   */\n  readonly accountId: string;\n  /**\n   * The Amazon Resource Name (ARN) of the account.\n   */\n  readonly accountArn: string;\n  /**\n   * The friendly name of the account.\n   */\n  readonly accountName: string;\n  /**\n   * The email address of the owner to assign to the new member account. This email address must not already be associated with another AWS account. You must use a valid email address to complete account creation. You can't access the root user of the account or remove an account that was created with an invalid email address.\n   */\n  readonly email: string;\n\n  /**\n   * Enables trusted access for the AWS service (trusted service) as <strong>Delegated Administrator</strong>, which performs tasks in your organization and its accounts on your behalf.\n   *\n   * @param servicePrincipal The supported AWS service that you specify\n   * @param region The region to delegate in\n   * @param {DelegatedAdministratorProps} props additional DelegatedAdministrator props\n   */\n  delegateAdministrator(servicePrincipal: string, region?: string, props?: Record<string, any>): void;\n}\n\n/**\n * Creates or imports an AWS account that is automatically a member of the organization whose credentials made the request. AWS Organizations automatically copies the information from the management account to the new member account\n */\nexport class Account extends Construct implements IAccount, ITaggableResource {\n  public readonly accountId: string;\n  public readonly accountArn: string;\n  public readonly accountName: string;\n  public readonly email: string;\n\n  protected readonly resource: CustomResource;\n\n  private readonly scope: Construct;\n\n  readonly tags = new TagManager(TagType.KEY_VALUE, \"Custom::Organizations_Account\");\n\n  public constructor(scope: Construct, id: string, props: AccountProps) {\n    super(scope, id);\n    this.scope = scope;\n\n    const { email, accountName, roleName, iamUserAccessToBilling, parent, importOnDuplicate, removalPolicy } = props;\n\n    if (!Validators.of().email(email)) {\n      Annotations.of(this).addError(\"The account's email must be of type string and between 6 and 64 characters long.\");\n    }\n\n    if (!Validators.of().accountName(accountName)) {\n      Annotations.of(this).addError(\"The account's name must be of type string and between 1 and 50 characters long.\");\n    }\n\n    const createAccountProvider = AccountProvider.getOrCreate(this);\n    const account = new CustomResource(this, \"CreateAccount\", {\n      serviceToken: createAccountProvider.provider.serviceToken,\n      resourceType: \"Custom::Organizations_Account\",\n      properties: {\n        Email: email,\n        AccountName: accountName,\n        RoleName: roleName ?? \"OrganizationAccountAccessRole\",\n        IamUserAccessToBilling: iamUserAccessToBilling ?? IamUserAccessToBilling.ALLOW,\n        ParentId: parent?.identifier(),\n        ImportOnDuplicate: String(importOnDuplicate ?? true),\n        RemovalPolicy: removalPolicy ?? RemovalPolicy.RETAIN,\n      },\n    });\n\n    this.accountId = account.getAtt(\"AccountId\").toString();\n    this.accountArn = account.getAtt(\"AccountArn\").toString();\n    this.accountName = account.getAtt(\"AccountName\").toString();\n    this.email = account.getAtt(\"Email\").toString();\n\n    this.resource = account;\n\n    const tagResource = new TagResource(this, \"Tags\", { resourceId: this.accountId, tags: this.tags.renderedTags });\n    tagResource.node.addDependency(account);\n  }\n\n  identifier(): string {\n    return this.accountId;\n  }\n\n  /**\n   * Enables trusted access for the AWS service (trusted service) as <strong>Delegated Administrator</strong>, which performs tasks in your organization and its accounts on your behalf.\n   *\n   * @param {string} servicePrincipal The supported AWS service that you specify\n   * @param {string} region The region to delegate in\n   * @param {DelegatedAdministratorProps} props additional DelegatedAdministrator props\n   */\n  public delegateAdministrator(servicePrincipal: string, region?: string, props: Record<string, any> = {}) {\n    const organizationsRegion = process.env.CDK_AWS_PARTITION === \"aws-cn\" ? \"cn-northwest-1\" : \"us-east-1\";\n    const delegatedAdministrator = new DelegatedAdministrator(\n      this.scope,\n      `Delegate${pascalCase(servicePrincipal)}${\n        region && region !== organizationsRegion ? `-${region}` : \"\"\n      }-${Names.nodeUniqueId(this.node)}`,\n      {\n        ...props,\n        account: this,\n        servicePrincipal: servicePrincipal,\n        region,\n      }\n    );\n    delegatedAdministrator.node.addDependency(this.resource);\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"]}