UNPKG

@pepperize/cdk-organizations

Version:

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

144 lines • 20.7 kB
"use strict"; Object.defineProperty(exports, "__esModule", { value: true }); exports.handler = handler; const AWS = require("aws-sdk"); let organizationsClient; const organizationsRegion = process.env.ORGANIZATIONS_ENDPOINT_REGION ?? "us-east-1"; /** * The isComplete handler is repeatedly invoked checking CreateAccountStatus until SUCCEEDED or FAILED. * @see https://docs.aws.amazon.com/cdk/api/v1/docs/custom-resources-readme.html#asynchronous-providers-iscomplete */ async function handler(event) { console.log(`Request of type ${event.RequestType} received`); if (!organizationsClient) { organizationsClient = new AWS.Organizations({ region: organizationsRegion }); } console.log("Payload: %j", event); let accountId; if (event.RequestType == "Create" || isLegacyPhysicalResourceId(event)) { const response = await organizationsClient // https://docs.aws.amazon.com/AWSJavaScriptSDK/latest/AWS/Organizations.html#describeCreateAccountStatus-property .describeCreateAccountStatus({ CreateAccountRequestId: isLegacyPhysicalResourceId(event) ? event.PhysicalResourceId : event.Data?.CreateAccountStatusId, }) .promise(); if (response.CreateAccountStatus?.State == "IN_PROGRESS") { // @ts-ignore return { IsComplete: false, Data: {} }; } if (response.CreateAccountStatus?.State == "FAILED" && response.CreateAccountStatus?.FailureReason != "EMAIL_ALREADY_EXISTS") { throw new Error(`Failed ${event.RequestType} Account ${response.CreateAccountStatus?.AccountName}, reason: ${response.CreateAccountStatus?.FailureReason}`); } if (response.CreateAccountStatus?.FailureReason == "EMAIL_ALREADY_EXISTS" && event.ResourceProperties.ImportOnDuplicate) { const account = await findAccountByEmail(organizationsClient, event.ResourceProperties.Email); if (!account) { throw new Error(`Failed ${event.RequestType} Account ${response.CreateAccountStatus?.AccountName}, reason: ${response.CreateAccountStatus?.FailureReason}; could not find account in organization.`); } accountId = account.Id; } else if (response.CreateAccountStatus?.FailureReason == "EMAIL_ALREADY_EXISTS" && !event.ResourceProperties.ImportOnDuplicate) { throw new Error(`Failed ${event.RequestType} Account ${response.CreateAccountStatus?.AccountName}, reason: ${response.CreateAccountStatus?.FailureReason}.`); } else { // State == SUCCEEDED accountId = response.CreateAccountStatus?.AccountId; } } else { accountId = event.PhysicalResourceId; } const response = await organizationsClient // https://docs.aws.amazon.com/AWSJavaScriptSDK/latest/AWS/Organizations.html#describeAccount-property .describeAccount({ AccountId: accountId }) .promise(); // On delete, update or create move account to destination parent await move(organizationsClient, accountId, event.ResourceProperties?.ParentId); // On delete close account if (event.RequestType == "Delete" && event.ResourceProperties?.RemovalPolicy == "destroy") { await close(organizationsClient, accountId); } return { IsComplete: true, // @ts-ignore PhysicalResourceId: accountId, Data: { ...event.ResourceProperties, ...event.Data, AccountId: accountId, AccountArn: response.Account?.Arn, AccountName: response.Account?.Name, Email: response.Account?.Email, }, }; } const findCurrentParent = async (client, id) => { const response = await client // https://docs.aws.amazon.com/AWSJavaScriptSDK/latest/AWS/Organizations.html#listParents-property .listParents({ ChildId: id, }) .promise(); if (response.Parents?.length) { return response.Parents[0]; } throw new Error(`Could not find parent for id '${id}'`); }; const move = async (client, accountId, destinationParentId) => { if (!destinationParentId) { return; } const currentParent = await findCurrentParent(organizationsClient, accountId); if (destinationParentId == currentParent.Id) { return; } await client // https://docs.aws.amazon.com/AWSJavaScriptSDK/latest/AWS/Organizations.html#moveAccount-property .moveAccount({ AccountId: accountId, SourceParentId: currentParent.Id, DestinationParentId: destinationParentId, }) .promise(); }; /** * Before aws-cdk-lib 2.15.0 the physical resource was determined in the onEventHandler and therefor the physical resource id was the account's CreateAccountStatusId. */ const isLegacyPhysicalResourceId = (event) => { return /car-[a-z0-9]{8,32}/.test(event.PhysicalResourceId); }; const findAccountByEmail = async (client, email) => { let response = await client // https://docs.aws.amazon.com/AWSJavaScriptSDK/latest/AWS/Organizations.html#listAccounts-property .listAccounts() .promise(); for (const account of response.Accounts ?? []) { if (account.Email == email) { return account; } } while (response.NextToken) { response = await client // https://docs.aws.amazon.com/AWSJavaScriptSDK/latest/AWS/Organizations.html#listAccounts-property .listAccounts({ NextToken: response.NextToken }) .promise(); for (const account of response.Accounts ?? []) { if (account.Email == email) { return account; } } } return undefined; }; const close = async (client, accountId) => { await client // https://docs.aws.amazon.com/AWSJavaScriptSDK/latest/AWS/Organizations.html#closeAccount-property .closeAccount({ AccountId: accountId, }); }; //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"is-complete-handler.lambda.js","sourceRoot":"","sources":["../../src/account-provider/is-complete-handler.lambda.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  CdkCustomResourceIsCompleteEvent as IsCompleteRequest,\n  CdkCustomResourceIsCompleteResponse as IsCompleteResponse,\n} from \"aws-lambda\";\nimport * as AWS from \"aws-sdk\";\nimport { Organizations } from \"aws-sdk\";\n\nlet organizationsClient: AWS.Organizations;\nconst organizationsRegion = process.env.ORGANIZATIONS_ENDPOINT_REGION ?? \"us-east-1\";\n\n/**\n * The isComplete handler is repeatedly invoked checking CreateAccountStatus until SUCCEEDED or FAILED.\n * @see https://docs.aws.amazon.com/cdk/api/v1/docs/custom-resources-readme.html#asynchronous-providers-iscomplete\n */\nexport async function handler(event: IsCompleteRequest): Promise<IsCompleteResponse> {\n  console.log(`Request of type ${event.RequestType} received`);\n\n  if (!organizationsClient) {\n    organizationsClient = new AWS.Organizations({ region: organizationsRegion });\n  }\n\n  console.log(\"Payload: %j\", event);\n\n  let accountId: string;\n  if (event.RequestType == \"Create\" || isLegacyPhysicalResourceId(event)) {\n    const response: AWS.Organizations.DescribeCreateAccountStatusResponse = await organizationsClient\n      // https://docs.aws.amazon.com/AWSJavaScriptSDK/latest/AWS/Organizations.html#describeCreateAccountStatus-property\n      .describeCreateAccountStatus({\n        CreateAccountRequestId: isLegacyPhysicalResourceId(event)\n          ? event.PhysicalResourceId!\n          : event.Data?.CreateAccountStatusId,\n      })\n      .promise();\n\n    if (response.CreateAccountStatus?.State == \"IN_PROGRESS\") {\n      // @ts-ignore\n      return { IsComplete: false, Data: {} };\n    }\n\n    if (\n      response.CreateAccountStatus?.State == \"FAILED\" &&\n      response.CreateAccountStatus?.FailureReason != \"EMAIL_ALREADY_EXISTS\"\n    ) {\n      throw new Error(\n        `Failed ${event.RequestType} Account ${response.CreateAccountStatus?.AccountName}, reason: ${response.CreateAccountStatus?.FailureReason}`\n      );\n    }\n\n    if (\n      response.CreateAccountStatus?.FailureReason == \"EMAIL_ALREADY_EXISTS\" &&\n      event.ResourceProperties.ImportOnDuplicate\n    ) {\n      const account = await findAccountByEmail(organizationsClient, event.ResourceProperties.Email);\n\n      if (!account) {\n        throw new Error(\n          `Failed ${event.RequestType} Account ${response.CreateAccountStatus?.AccountName}, reason: ${response.CreateAccountStatus?.FailureReason}; could not find account in organization.`\n        );\n      }\n\n      accountId = account.Id!;\n    } else if (\n      response.CreateAccountStatus?.FailureReason == \"EMAIL_ALREADY_EXISTS\" &&\n      !event.ResourceProperties.ImportOnDuplicate\n    ) {\n      throw new Error(\n        `Failed ${event.RequestType} Account ${response.CreateAccountStatus?.AccountName}, reason: ${response.CreateAccountStatus?.FailureReason}.`\n      );\n    } else {\n      // State == SUCCEEDED\n      accountId = response.CreateAccountStatus?.AccountId!;\n    }\n  } else {\n    accountId = event.PhysicalResourceId!;\n  }\n\n  const response = await organizationsClient\n    // https://docs.aws.amazon.com/AWSJavaScriptSDK/latest/AWS/Organizations.html#describeAccount-property\n    .describeAccount({ AccountId: accountId })\n    .promise();\n\n  // On delete, update or create move account to destination parent\n  await move(organizationsClient, accountId, event.ResourceProperties?.ParentId);\n\n  // On delete close account\n  if (event.RequestType == \"Delete\" && event.ResourceProperties?.RemovalPolicy == \"destroy\") {\n    await close(organizationsClient, accountId);\n  }\n\n  return {\n    IsComplete: true,\n    // @ts-ignore\n    PhysicalResourceId: accountId,\n    Data: {\n      ...event.ResourceProperties,\n      ...event.Data,\n      AccountId: accountId,\n      AccountArn: response.Account?.Arn,\n      AccountName: response.Account?.Name,\n      Email: response.Account?.Email,\n    },\n  };\n}\n\nconst findCurrentParent = async (client: Organizations, id: string): Promise<Organizations.Parent> => {\n  const response: Organizations.ListParentsResponse = await client\n    // https://docs.aws.amazon.com/AWSJavaScriptSDK/latest/AWS/Organizations.html#listParents-property\n    .listParents({\n      ChildId: id,\n    })\n    .promise();\n\n  if (response.Parents?.length) {\n    return response.Parents[0];\n  }\n\n  throw new Error(`Could not find parent for id '${id}'`);\n};\n\nconst move = async (\n  client: Organizations,\n  accountId: string,\n  destinationParentId: string | undefined\n): Promise<void> => {\n  if (!destinationParentId) {\n    return;\n  }\n\n  const currentParent = await findCurrentParent(organizationsClient, accountId);\n\n  if (destinationParentId == currentParent.Id) {\n    return;\n  }\n\n  await client\n    // https://docs.aws.amazon.com/AWSJavaScriptSDK/latest/AWS/Organizations.html#moveAccount-property\n    .moveAccount({\n      AccountId: accountId,\n      SourceParentId: currentParent.Id!,\n      DestinationParentId: destinationParentId,\n    })\n    .promise();\n};\n\n/**\n * Before aws-cdk-lib 2.15.0 the physical resource was determined in the onEventHandler and therefor the physical resource id was the account's CreateAccountStatusId.\n */\nconst isLegacyPhysicalResourceId = (event: IsCompleteRequest): boolean => {\n  return /car-[a-z0-9]{8,32}/.test(event.PhysicalResourceId!);\n};\n\nconst findAccountByEmail = async (client: Organizations, email: string): Promise<Organizations.Account | undefined> => {\n  let response: Organizations.ListAccountsResponse = await client\n    // https://docs.aws.amazon.com/AWSJavaScriptSDK/latest/AWS/Organizations.html#listAccounts-property\n    .listAccounts()\n    .promise();\n  for (const account of response.Accounts ?? []) {\n    if (account.Email == email) {\n      return account;\n    }\n  }\n\n  while (response.NextToken) {\n    response = await client\n      // https://docs.aws.amazon.com/AWSJavaScriptSDK/latest/AWS/Organizations.html#listAccounts-property\n      .listAccounts({ NextToken: response.NextToken })\n      .promise();\n    for (const account of response.Accounts ?? []) {\n      if (account.Email == email) {\n        return account;\n      }\n    }\n  }\n\n  return undefined;\n};\n\nconst close = async (client: Organizations, accountId: string): Promise<void> => {\n  await client\n    // https://docs.aws.amazon.com/AWSJavaScriptSDK/latest/AWS/Organizations.html#closeAccount-property\n    .closeAccount({\n      AccountId: accountId,\n    });\n};\n"]}