@pepperize/cdk-organizations
Version:
Manage AWS organizations, organizational units (OU), accounts and service control policies (SCP).
126 lines • 17.2 kB
JavaScript
;
Object.defineProperty(exports, "__esModule", { value: true });
exports.handler = void 0;
const aws_sdk_1 = require("aws-sdk");
let organizationsClient;
const organizationsRegion = process.env.ORGANIZATIONS_ENDPOINT_REGION ?? "us-east-1";
/**
* The onEvent handler is invoked whenever a resource lifecycle event for an organizational unit occurs
*
* @see https://docs.aws.amazon.com/cdk/api/v1/docs/custom-resources-readme.html#handling-lifecycle-events-onevent
*/
const handler = async (event) => {
console.log(`Request of type ${event.RequestType} received`);
if (!organizationsClient) {
organizationsClient = new aws_sdk_1.Organizations({ region: organizationsRegion });
}
console.log("Payload: %j", event);
const { ParentId, Name, ImportOnDuplicate, RemovalPolicy } = event.ResourceProperties;
if (event.RequestType == "Create") {
try {
const organizationalUnit = await createOrganizationalUnit(organizationsClient, ParentId, Name);
return {
PhysicalResourceId: organizationalUnit.Id,
Data: {
...organizationalUnit,
},
};
}
catch (e) {
const error = e;
console.log(error);
// https://docs.aws.amazon.com/organizations/latest/APIReference/API_CreateOrganizationalUnit.html#API_CreateOrganizationalUnit_Errors
if (error.code == "DuplicateOrganizationalUnitException" && ImportOnDuplicate == "true") {
console.log(`Organizational unit already created, trying to find existing one in parent.`);
const organizationalUnit = await findOrganizationalUnitByParentAndName(organizationsClient, ParentId, Name);
return {
PhysicalResourceId: organizationalUnit.Id,
Data: {
...organizationalUnit,
},
};
}
else {
throw error;
}
}
}
if (event.RequestType == "Update") {
const organizationalUnit = await updateOrganizationalUnit(organizationsClient, event.PhysicalResourceId, Name);
return {
PhysicalResourceId: organizationalUnit.Id,
Data: {
...organizationalUnit,
},
};
}
if (event.RequestType == "Delete" && RemovalPolicy == "destroy") {
await deleteOrganizationalUnit(organizationsClient, event.PhysicalResourceId);
}
return {
PhysicalResourceId: event.PhysicalResourceId,
Data: {
...event.ResourceProperties,
},
};
};
exports.handler = handler;
const findOrganizationalUnitByParentAndName = async (client, parentId, name) => {
let response = await client
// https://docs.aws.amazon.com/AWSJavaScriptSDK/latest/AWS/Organizations.html#listOrganizationalUnitsForParent-property
.listOrganizationalUnitsForParent({ ParentId: parentId })
.promise();
for (const organizationalUnit of response.OrganizationalUnits ?? []) {
if (organizationalUnit.Name == name) {
return organizationalUnit;
}
}
while (response.NextToken) {
response = await client
.listOrganizationalUnitsForParent({ ParentId: parentId, NextToken: response.NextToken })
.promise();
for (const organizationalUnit of response.OrganizationalUnits ?? []) {
if (organizationalUnit.Name == name) {
return organizationalUnit;
}
}
}
throw new Error(`Organizational unit '${name}' not found in '${parentId}'`);
};
const createOrganizationalUnit = async (client, parentId, name) => {
const response = await client
// https://docs.aws.amazon.com/AWSJavaScriptSDK/latest/AWS/Organizations.html#createOrganizationalUnit-property
.createOrganizationalUnit({
ParentId: parentId,
Name: name,
})
.promise();
console.log("Creating organizational unit: %j", response);
if (!response.OrganizationalUnit) {
throw new Error("Could not create organizational unit, reason: empty response");
}
return response.OrganizationalUnit;
};
const updateOrganizationalUnit = async (client, id, name) => {
const response = await client
// https://docs.aws.amazon.com/AWSJavaScriptSDK/latest/AWS/Organizations.html#updateOrganizationalUnit-property
.updateOrganizationalUnit({
OrganizationalUnitId: id,
Name: name,
})
.promise();
console.log("Updating organizational unit: %j", response);
if (!response.OrganizationalUnit) {
throw new Error("Could not update organizational unit, reason: empty response");
}
return response.OrganizationalUnit;
};
const deleteOrganizationalUnit = async (client, id) => {
// https://docs.aws.amazon.com/AWSJavaScriptSDK/latest/AWS/Organizations.html#deleteOrganizationalUnit-property
await client
.deleteOrganizationalUnit({
OrganizationalUnitId: id,
})
.promise();
};
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"on-event-handler.lambda.js","sourceRoot":"","sources":["../../src/organizational-unit-provider/on-event-handler.lambda.ts"],"names":[],"mappings":"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","sourcesContent":["import { CdkCustomResourceEvent as OnEventRequest, CdkCustomResourceResponse as OnEventResponse } from \"aws-lambda\";\nimport { AWSError, Organizations } from \"aws-sdk\";\n\nlet organizationsClient: Organizations;\nconst organizationsRegion = process.env.ORGANIZATIONS_ENDPOINT_REGION ?? \"us-east-1\";\n\n/**\n * The onEvent handler is invoked whenever a resource lifecycle event for an organizational unit occurs\n *\n * @see https://docs.aws.amazon.com/cdk/api/v1/docs/custom-resources-readme.html#handling-lifecycle-events-onevent\n */\nexport const handler = async (event: OnEventRequest): Promise<OnEventResponse> => {\n  console.log(`Request of type ${event.RequestType} received`);\n\n  if (!organizationsClient) {\n    organizationsClient = new Organizations({ region: organizationsRegion });\n  }\n\n  console.log(\"Payload: %j\", event);\n\n  const { ParentId, Name, ImportOnDuplicate, RemovalPolicy } = event.ResourceProperties;\n\n  if (event.RequestType == \"Create\") {\n    try {\n      const organizationalUnit = await createOrganizationalUnit(organizationsClient, ParentId, Name);\n      return {\n        PhysicalResourceId: organizationalUnit.Id,\n        Data: {\n          ...organizationalUnit,\n        },\n      };\n    } catch (e) {\n      const error = e as AWSError;\n      console.log(error);\n      // https://docs.aws.amazon.com/organizations/latest/APIReference/API_CreateOrganizationalUnit.html#API_CreateOrganizationalUnit_Errors\n      if (error.code == \"DuplicateOrganizationalUnitException\" && ImportOnDuplicate == \"true\") {\n        console.log(`Organizational unit already created, trying to find existing one in parent.`);\n        const organizationalUnit = await findOrganizationalUnitByParentAndName(organizationsClient, ParentId, Name);\n\n        return {\n          PhysicalResourceId: organizationalUnit.Id,\n          Data: {\n            ...organizationalUnit,\n          },\n        };\n      } else {\n        throw error;\n      }\n    }\n  }\n\n  if (event.RequestType == \"Update\") {\n    const organizationalUnit = await updateOrganizationalUnit(organizationsClient, event.PhysicalResourceId!, Name);\n\n    return {\n      PhysicalResourceId: organizationalUnit.Id,\n      Data: {\n        ...organizationalUnit,\n      },\n    };\n  }\n\n  if (event.RequestType == \"Delete\" && RemovalPolicy == \"destroy\") {\n    await deleteOrganizationalUnit(organizationsClient, event.PhysicalResourceId!);\n  }\n\n  return {\n    PhysicalResourceId: event.PhysicalResourceId,\n    Data: {\n      ...event.ResourceProperties,\n    },\n  };\n};\n\nconst findOrganizationalUnitByParentAndName = async (\n  client: Organizations,\n  parentId: string,\n  name: string\n): Promise<Organizations.OrganizationalUnit> => {\n  let response: Organizations.ListOrganizationalUnitsForParentResponse = await client\n    // https://docs.aws.amazon.com/AWSJavaScriptSDK/latest/AWS/Organizations.html#listOrganizationalUnitsForParent-property\n    .listOrganizationalUnitsForParent({ ParentId: parentId })\n    .promise();\n  for (const organizationalUnit of response.OrganizationalUnits ?? []) {\n    if (organizationalUnit.Name == name) {\n      return organizationalUnit;\n    }\n  }\n\n  while (response.NextToken) {\n    response = await client\n      .listOrganizationalUnitsForParent({ ParentId: parentId, NextToken: response.NextToken })\n      .promise();\n    for (const organizationalUnit of response.OrganizationalUnits ?? []) {\n      if (organizationalUnit.Name == name) {\n        return organizationalUnit;\n      }\n    }\n  }\n\n  throw new Error(`Organizational unit '${name}' not found in '${parentId}'`);\n};\n\nconst createOrganizationalUnit = async (\n  client: Organizations,\n  parentId: string,\n  name: string\n): Promise<Organizations.OrganizationalUnit> => {\n  const response: Organizations.CreateOrganizationalUnitResponse = await client\n    // https://docs.aws.amazon.com/AWSJavaScriptSDK/latest/AWS/Organizations.html#createOrganizationalUnit-property\n    .createOrganizationalUnit({\n      ParentId: parentId,\n      Name: name,\n    })\n    .promise();\n  console.log(\"Creating organizational unit: %j\", response);\n\n  if (!response.OrganizationalUnit) {\n    throw new Error(\"Could not create organizational unit, reason: empty response\");\n  }\n\n  return response.OrganizationalUnit;\n};\n\nconst updateOrganizationalUnit = async (\n  client: Organizations,\n  id: string,\n  name: string\n): Promise<Organizations.OrganizationalUnit> => {\n  const response: Organizations.UpdateOrganizationalUnitResponse = await client\n    // https://docs.aws.amazon.com/AWSJavaScriptSDK/latest/AWS/Organizations.html#updateOrganizationalUnit-property\n    .updateOrganizationalUnit({\n      OrganizationalUnitId: id,\n      Name: name,\n    })\n    .promise();\n  console.log(\"Updating organizational unit: %j\", response);\n\n  if (!response.OrganizationalUnit) {\n    throw new Error(\"Could not update organizational unit, reason: empty response\");\n  }\n\n  return response.OrganizationalUnit;\n};\n\nconst deleteOrganizationalUnit = async (client: Organizations, id: string): Promise<void> => {\n  // https://docs.aws.amazon.com/AWSJavaScriptSDK/latest/AWS/Organizations.html#deleteOrganizationalUnit-property\n  await client\n    .deleteOrganizationalUnit({\n      OrganizationalUnitId: id,\n    })\n    .promise();\n};\n"]}