UNPKG

@k9securityio/k9-cdk

Version:

Provision strong AWS security policies easily using the AWS CDK.

355 lines 53.4 kB
"use strict"; var _a; Object.defineProperty(exports, "__esModule", { value: true }); exports.K9PolicyFactory = exports.SID_DENY_UNTRUSTED_ORGS = exports.AccessCapability = void 0; exports.getAccessCapabilityFromValue = getAccessCapabilityFromValue; exports.canPrincipalsManageResources = canPrincipalsManageResources; exports.hasWildcardPrincipal = hasWildcardPrincipal; exports.validateAccessSpecs = validateAccessSpecs; exports.toPascalCase = toPascalCase; const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti"); const aws_iam_1 = require("aws-cdk-lib/aws-iam"); var AccessCapability; (function (AccessCapability) { AccessCapability["ADMINISTER_RESOURCE"] = "administer-resource"; AccessCapability["READ_CONFIG"] = "read-config"; AccessCapability["READ_DATA"] = "read-data"; AccessCapability["WRITE_DATA"] = "write-data"; AccessCapability["DELETE_DATA"] = "delete-data"; })(AccessCapability || (exports.AccessCapability = AccessCapability = {})); function getAccessCapabilityFromValue(accessCapabilityStr) { //https://blog.logrocket.com/typescript-string-enums-guide/ for (let key of Object.keys(AccessCapability)) { // @ts-ignore if (AccessCapability[key] == accessCapabilityStr) { // https://stackoverflow.com/questions/17380845/how-do-i-convert-a-string-to-enum-in-typescript let typedKey = key; return AccessCapability[typedKey]; } } throw Error(`Could not get AccessCapability from value: ${accessCapabilityStr}`); } /** * Check whether the provided access specs ensure that at least one principal can both read and administer configuration. * @param accessSpecsByCapability is a map of access specs keyed by access capability * * @return true when at least one principal that can administer and read configuration exists */ function canPrincipalsManageResources(accessSpecsByCapability) { let adminSpec = accessSpecsByCapability.get(AccessCapability.ADMINISTER_RESOURCE); let readConfigSpec = accessSpecsByCapability.get(AccessCapability.READ_CONFIG); if ((adminSpec?.allowPrincipalArns && adminSpec.allowPrincipalArns.length > 0) && (readConfigSpec?.allowPrincipalArns && readConfigSpec.allowPrincipalArns.length > 0)) { const adminPrincipals = new Set(adminSpec.allowPrincipalArns); const readConfigPrincipals = new Set(readConfigSpec.allowPrincipalArns); const intersection = new Set([...adminPrincipals].filter(x => readConfigPrincipals.has(x))); return intersection.size > 0; } return false; } /** * Check if any access spec contains a wildcard principal ("*"). */ function hasWildcardPrincipal(accessSpecs) { for (let spec of accessSpecs) { if (spec.allowPrincipalArns.includes('*')) { return true; } } return false; } /** * Validate that access specs have valid principal ARN + org constraint combinations. * Throws an error for invalid combinations: * - Empty allowPrincipalArns * - Wildcard allowPrincipalArns without restrictToPrincipalOrgIDs (public access) */ function validateAccessSpecs(accessSpecs) { for (let spec of accessSpecs) { if (!spec.allowPrincipalArns || spec.allowPrincipalArns.length === 0) { throw new Error('allowPrincipalArns must not be empty; every resource policy statement requires a Principal element.'); } if (spec.allowPrincipalArns.includes('*') && (!spec.restrictToPrincipalOrgIDs || spec.restrictToPrincipalOrgIDs.length === 0)) { throw new Error('k9-cdk will not generate a resource policy that allows fully public access.' + ' Wildcard principal ("*") requires restrictToPrincipalOrgIDs to scope access.' + ' Consider specifying account principal ARNs or constraining to specific PrincipalOrgIDs.'); } } } /** * Converts a string to PascalCase, which is useful for e.g. policy types that don't * do not support spaces or hyphens in statement ids. * * @param input */ function toPascalCase(input) { // Remove placeholders like ${something} and trim whitespace const cleanedInput = input.replace(/\$\{.*?\}/g, '').trim(); // Split the input into words based on spaces, hyphens, underscores, or other delimiters const words = cleanedInput.split(/[\s_\-]+/); // Convert each word to PascalCase return words .map(word => word.charAt(0).toUpperCase() + word.slice(1).toLowerCase()) .join(''); } exports.SID_DENY_UNTRUSTED_ORGS = 'DenyUntrustedOrgs'; class K9PolicyFactory { constructor() { /** @internal */ this._SUPPORTED_SERVICES = new Set([ 'S3', 'KMS', 'DynamoDB', 'SQS', 'EventBridge', ]); /** @internal */ this._K9CapabilityMapJSON = require('../resources/capability_summary.json'); // eslint-disable-line @typescript-eslint/no-require-imports /** @internal */ this._K9CapabilityMapByService = new Map(Object.entries(this._K9CapabilityMapJSON)); } /** * Deduplicate an array of principals while preserving original order of principals. * Note that principals may contain either strings or objects, so naive array sorting * produces unstable results. * * @param principals */ static deduplicatePrincipals(principals) { const observedPrincipals = new Set(); const uniquePrincipals = new Array(); for (let principal of principals) { if (!observedPrincipals.has(principal)) { uniquePrincipals.push(principal); observedPrincipals.add(principal); } } return uniquePrincipals; } getActions(service, accessCapability) { if (!this._SUPPORTED_SERVICES.has(service) && this._K9CapabilityMapByService.has(service)) { throw Error(`unsupported service: ${service}`); } let serviceCapabilitiesObj = this._K9CapabilityMapByService.get(service) || {}; let serviceCapabilitiesMap = new Map(Object.entries(serviceCapabilitiesObj)); let accessCapabilityName = accessCapability.toString(); if (serviceCapabilitiesMap && serviceCapabilitiesMap.has(accessCapabilityName)) { return serviceCapabilitiesMap.get(accessCapabilityName) || Array(); } else { return new Array(); } } /** @internal */ _mergeAccessSpecs(target, addition) { target.allowPrincipalArns.push(...addition.allowPrincipalArns); if (target.test) { //ok, user has specified a test at some point; ensure this desiredAccessSpec.test matches if (target.test != addition.test) { let msg = 'Cannot merge AccessSpecs; test attributes do not match:' + `\n${JSON.stringify(target)}\n${JSON.stringify(addition)}`; throw Error(msg); } } else { //first explicit test preference wins if (addition.test) { target.test = addition.test; } } // Merge restrictToPrincipalOrgIDs if (addition.restrictToPrincipalOrgIDs && addition.restrictToPrincipalOrgIDs.length > 0) { if (!target.restrictToPrincipalOrgIDs) { target.restrictToPrincipalOrgIDs = []; } target.restrictToPrincipalOrgIDs.push(...addition.restrictToPrincipalOrgIDs); } } mergeDesiredAccessSpecsByCapability(supportedCapabilities, desiredAccess) { let accessSpecsByCapability = new Map(); // 1. populate accessSpecsByCapability with fresh AccessSpecs for each supported capability // 2. iterate through desiredAccess specs and merge data into what we'll use // important: detect mismatched test types // we can leave `test` unset in the default access specs // and copy the value from the spec being merged if it is set // throw Error on mismatch // 3. generate an Allow statement for each supported capability for (let supportedCapability of supportedCapabilities) { //generate a default access spec for each of the service's supported capabilities let effectiveAccessSpec = { accessCapabilities: supportedCapability, allowPrincipalArns: new Array(), // leave 'test' property unset; will populate from user-provided data }; accessSpecsByCapability.set(supportedCapability, effectiveAccessSpec); //now... merge in the user's desired access for this capability for (let desiredAccessSpec of desiredAccess) { if (desiredAccessSpec.accessCapabilities instanceof Array) { for (let desiredCapability of desiredAccessSpec.accessCapabilities) { if (supportedCapability == desiredCapability) { this._mergeAccessSpecs(effectiveAccessSpec, desiredAccessSpec); } } } else if (typeof desiredAccessSpec.accessCapabilities == 'string') { if (supportedCapability == desiredAccessSpec.accessCapabilities) { this._mergeAccessSpecs(effectiveAccessSpec, desiredAccessSpec); } } else { throw Error(`Unhandled type of accessCapabilities for ${desiredAccessSpec.accessCapabilities}`); } } } const records = {}; accessSpecsByCapability.forEach(function (value, key) { records[key] = value; }); return records; } makeAllowStatements(serviceName, supportedCapabilities, desiredAccess, resourceArns, usePascalCase = false) { let policyStatements = new Array(); let accessSpecsByCapabilityRecs = this.mergeDesiredAccessSpecsByCapability(supportedCapabilities, desiredAccess); let accessSpecsByCapability = new Map(); for (let [capabilityStr, accessSpec] of Object.entries(accessSpecsByCapabilityRecs)) { accessSpecsByCapability.set(getAccessCapabilityFromValue(capabilityStr), accessSpec); } // ok, time to actually make Allow Statements from our AccessSpecs for (let supportedCapability of supportedCapabilities) { let accessSpec = accessSpecsByCapability.get(supportedCapability) || { //generate a default access spec if none was provided accessCapabilities: [supportedCapability], allowPrincipalArns: new Array(), test: 'ArnEquals', }; let arnConditionTest = accessSpec.test || 'ArnEquals'; let sid = `Allow Restricted ${supportedCapability}`; if (usePascalCase) { sid = toPascalCase(sid); } let statement = this.makeAllowStatement(sid, this.getActions(serviceName, supportedCapability), accessSpec.allowPrincipalArns, arnConditionTest, resourceArns, accessSpec.restrictToPrincipalOrgIDs); policyStatements.push(statement); } return policyStatements; } makeAllowStatement(sid, actions, principalArns, test, resources, restrictToPrincipalOrgIDs) { const policyStatementProps = { sid: sid, effect: aws_iam_1.Effect.ALLOW, }; const statement = new aws_iam_1.PolicyStatement(policyStatementProps); statement.addActions(...actions); statement.addAnyPrincipal(); statement.addResources(...resources); const isWildcardPrincipal = principalArns.includes('*'); const hasOrgConstraint = restrictToPrincipalOrgIDs && restrictToPrincipalOrgIDs.length > 0; if (isWildcardPrincipal && hasOrgConstraint) { // Code Path B: wildcard + org constraint // Use Principal: "*" (already added via addAnyPrincipal) + aws:PrincipalOrgID condition // Do NOT add aws:PrincipalArn condition statement.addCondition('StringEquals', { 'aws:PrincipalOrgID': restrictToPrincipalOrgIDs }); } else { // Code Path A: specific principal ARNs (existing behavior) statement.addCondition(test, { 'aws:PrincipalArn': K9PolicyFactory.deduplicatePrincipals(principalArns) }); if (hasOrgConstraint) { // Specific ARNs + org constraint: both conditions must be true statement.addCondition('StringEquals', { 'aws:PrincipalOrgID': restrictToPrincipalOrgIDs }); } } return statement; } wasLikeUsed(accessSpecs) { for (let accessSpec of accessSpecs) { if ('ArnLike' == accessSpec.test) { return true; } } return false; } getAllowedPrincipalArns(accessSpecs) { let allowedPrincipalArns = new Set(); for (let accessSpec of accessSpecs) { accessSpec.allowPrincipalArns.forEach(function (value) { allowedPrincipalArns.add(value); }); } return Array.from(allowedPrincipalArns); } /** * k9 wants to deny all AWS accounts and IAM principals not explicitly allowed; this *should* * be straightforward, but it isn't because of the way aws-cdk merges and manipulates Principals. * @return list of principals for a DenyEveryoneElse statement */ makeDenyEveryoneElsePrincipals() { /** * We should be able to provide AnyPrincipal once (of course), but AWS CDK converts: * "Principal": { * "AWS": "*" // identifies all AWS accounts and IAM. * } * to: * "Principal": "*" // identifies all principals including AWS Service principals * * That's a greater scope than we want. * * So provide AnyPrincipal twice, so aws-cdk maintains the array form. * * AWS rewrites the AWS member of the policy on save so * only the unique set of principals are included * So after these machinations, we end up with what we want. */ return [new aws_iam_1.AnyPrincipal(), new aws_iam_1.AnyPrincipal()]; } /** * Create a DenyUntrustedOrgs statement that explicitly denies principals from * untrusted orgs for org-restricted actions. This provides defense-in-depth * beyond the implicit deny from org-constrained Allow statements. * * The StringNotEquals condition on aws:PrincipalOrgID is inherently safe for * AWS service principals because the key is absent from their request context, * so the condition is not satisfied and the Deny does not apply. * * @return a PolicyStatement with Effect Deny, or undefined if no access specs have org restrictions * @internal */ _makeDenyUntrustedOrgsStatement(serviceName, supportedCapabilities, accessSpecsByCapability, resourceArns) { const allActions = new Set(); const allOrgIDs = new Set(); for (let capability of supportedCapabilities) { const accessSpec = accessSpecsByCapability.get(capability); if (accessSpec?.restrictToPrincipalOrgIDs && accessSpec.restrictToPrincipalOrgIDs.length > 0) { const actions = this.getActions(serviceName, capability); for (let action of actions) { allActions.add(action); } for (let orgID of accessSpec.restrictToPrincipalOrgIDs) { allOrgIDs.add(orgID); } } } if (allActions.size === 0) { return undefined; } const statement = new aws_iam_1.PolicyStatement({ sid: exports.SID_DENY_UNTRUSTED_ORGS, effect: aws_iam_1.Effect.DENY, principals: this.makeDenyEveryoneElsePrincipals(), actions: Array.from(allActions), resources: resourceArns, }); statement.addCondition('Bool', { 'aws:PrincipalIsAWSService': ['false'], }); statement.addCondition('StringNotEquals', { 'aws:PrincipalOrgID': Array.from(allOrgIDs), }); return statement; } } exports.K9PolicyFactory = K9PolicyFactory; _a = JSII_RTTI_SYMBOL_1; K9PolicyFactory[_a] = { fqn: "@k9securityio/k9-cdk.k9policy.K9PolicyFactory", version: "2.2.1" }; //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"k9policy.js","sourceRoot":"","sources":["../src/k9policy.ts"],"names":[],"mappings":";;;;AAyBA,oEAYC;AA2CD,oEAaC;AAMD,oDAOC;AAQD,kDAgBC;AAQD,oCAaC;;AAvJD,iDAO6B;AAU7B,IAAY,gBAMX;AAND,WAAY,gBAAgB;IAC1B,+DAA2C,CAAA;IAC3C,+CAA2B,CAAA;IAC3B,2CAAuB,CAAA;IACvB,6CAAyB,CAAA;IACzB,+CAA2B,CAAA;AAC7B,CAAC,EANW,gBAAgB,gCAAhB,gBAAgB,QAM3B;AAED,SAAgB,4BAA4B,CAAC,mBAA2B;IACtE,2DAA2D;IAC3D,KAAK,IAAI,GAAG,IAAI,MAAM,CAAC,IAAI,CAAC,gBAAgB,CAAC,EAAE,CAAC;QAC9C,aAAa;QACb,IAAI,gBAAgB,CAAC,GAAG,CAAC,IAAI,mBAAmB,EAAE,CAAC;YACjD,+FAA+F;YAC/F,IAAI,QAAQ,GAAkC,GAAG,CAAC;YAClD,OAAO,gBAAgB,CAAC,QAAQ,CAAC,CAAC;QACpC,CAAC;IACH,CAAC;IAED,MAAM,KAAK,CAAC,8CAA8C,mBAAmB,EAAE,CAAC,CAAC;AACnF,CAAC;AAqCD;;;;;GAKG;AACH,SAAgB,4BAA4B,CAAC,uBAA2D;IACtG,IAAI,SAAS,GAAG,uBAAuB,CAAC,GAAG,CAAC,gBAAgB,CAAC,mBAAmB,CAAC,CAAC;IAClF,IAAI,cAAc,GAAG,uBAAuB,CAAC,GAAG,CAAC,gBAAgB,CAAC,WAAW,CAAC,CAAC;IAE/E,IAAI,CAAC,SAAS,EAAE,kBAAkB,IAAI,SAAS,CAAC,kBAAkB,CAAC,MAAM,GAAG,CAAC,CAAC;WACrE,CAAC,cAAc,EAAE,kBAAkB,IAAI,cAAc,CAAC,kBAAkB,CAAC,MAAM,GAAG,CAAC,CAAC,EAAE,CAAC;QAC9F,MAAM,eAAe,GAAG,IAAI,GAAG,CAAS,SAAS,CAAC,kBAAkB,CAAC,CAAC;QACtE,MAAM,oBAAoB,GAAG,IAAI,GAAG,CAAS,cAAc,CAAC,kBAAkB,CAAC,CAAC;QAChF,MAAM,YAAY,GAAG,IAAI,GAAG,CAC1B,CAAC,GAAG,eAAe,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,oBAAoB,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;QACjE,OAAO,YAAY,CAAC,IAAI,GAAG,CAAC,CAAC;IAC/B,CAAC;IACD,OAAO,KAAK,CAAC;AACf,CAAC;AAGD;;GAEG;AACH,SAAgB,oBAAoB,CAAC,WAA+B;IAClE,KAAK,IAAI,IAAI,IAAI,WAAW,EAAE,CAAC;QAC7B,IAAI,IAAI,CAAC,kBAAkB,CAAC,QAAQ,CAAC,GAAG,CAAC,EAAE,CAAC;YAC1C,OAAO,IAAI,CAAC;QACd,CAAC;IACH,CAAC;IACD,OAAO,KAAK,CAAC;AACf,CAAC;AAED;;;;;GAKG;AACH,SAAgB,mBAAmB,CAAC,WAA+B;IACjE,KAAK,IAAI,IAAI,IAAI,WAAW,EAAE,CAAC;QAC7B,IAAI,CAAC,IAAI,CAAC,kBAAkB,IAAI,IAAI,CAAC,kBAAkB,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;YACrE,MAAM,IAAI,KAAK,CACb,qGAAqG,CACtG,CAAC;QACJ,CAAC;QACD,IAAI,IAAI,CAAC,kBAAkB,CAAC,QAAQ,CAAC,GAAG,CAAC;YACrC,CAAC,CAAC,IAAI,CAAC,yBAAyB,IAAI,IAAI,CAAC,yBAAyB,CAAC,MAAM,KAAK,CAAC,CAAC,EAAE,CAAC;YACrF,MAAM,IAAI,KAAK,CACb,6EAA6E;gBAC7E,+EAA+E;gBAC/E,0FAA0F,CAC3F,CAAC;QACJ,CAAC;IACH,CAAC;AACH,CAAC;AAED;;;;;GAKG;AACH,SAAgB,YAAY,CAAC,KAAa;IACxC,4DAA4D;IAC5D,MAAM,YAAY,GAAG,KAAK,CAAC,OAAO,CAAC,YAAY,EAAE,EAAE,CAAC,CAAC,IAAI,EAAE,CAAC;IAE5D,wFAAwF;IACxF,MAAM,KAAK,GAAG,YAAY,CAAC,KAAK,CAAC,UAAU,CAAC,CAAC;IAE7C,kCAAkC;IAClC,OAAO,KAAK;SACT,GAAG,CACF,IAAI,CAAC,EAAE,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,WAAW,EAAE,GAAG,IAAI,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,WAAW,EAAE,CACnE;SACA,IAAI,CAAC,EAAE,CAAC,CAAC;AACd,CAAC;AAEY,QAAA,uBAAuB,GAAG,mBAAmB,CAAC;AAE3D,MAAa,eAAe;IAA5B;QAqBE,gBAAgB;QAChB,wBAAmB,GAAG,IAAI,GAAG,CAAS;YACpC,IAAI;YACJ,KAAK;YACL,UAAU;YACV,KAAK;YACL,aAAa;SACd,CAAC,CAAC;QAEH,gBAAgB;QAChB,yBAAoB,GAAW,OAAO,CAAC,sCAAsC,CAAC,CAAC,CAAC,4DAA4D;QAC5I,gBAAgB;QAChB,8BAAyB,GAAwB,IAAI,GAAG,CAAC,MAAM,CAAC,OAAO,CAAC,IAAI,CAAC,oBAAoB,CAAC,CAAC,CAAC;KA6QrG;IA5SC;;;;;;OAMG;IACH,MAAM,CAAC,qBAAqB,CAAC,UAAgC;QAC3D,MAAM,kBAAkB,GAAG,IAAI,GAAG,EAAiB,CAAC;QACpD,MAAM,gBAAgB,GAAG,IAAI,KAAK,EAAiB,CAAC;QACpD,KAAK,IAAI,SAAS,IAAI,UAAU,EAAE,CAAC;YACjC,IAAI,CAAC,kBAAkB,CAAC,GAAG,CAAC,SAAS,CAAC,EAAE,CAAC;gBACvC,gBAAgB,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC;gBACjC,kBAAkB,CAAC,GAAG,CAAC,SAAS,CAAC,CAAC;YACpC,CAAC;QACH,CAAC;QACD,OAAO,gBAAgB,CAAC;IAC1B,CAAC;IAgBD,UAAU,CAAC,OAAe,EAAE,gBAAkC;QAC5D,IAAI,CAAC,IAAI,CAAC,mBAAmB,CAAC,GAAG,CAAC,OAAO,CAAC,IAAI,IAAI,CAAC,yBAAyB,CAAC,GAAG,CAAC,OAAO,CAAC,EAAE,CAAC;YAC1F,MAAM,KAAK,CAAC,wBAAwB,OAAO,EAAE,CAAC,CAAC;QACjD,CAAC;QAED,IAAI,sBAAsB,GAAW,IAAI,CAAC,yBAAyB,CAAC,GAAG,CAAC,OAAO,CAAC,IAAI,EAAE,CAAC;QACvF,IAAI,sBAAsB,GAAG,IAAI,GAAG,CAAwB,MAAM,CAAC,OAAO,CAAC,sBAAsB,CAAC,CAAC,CAAC;QAEpG,IAAI,oBAAoB,GAAG,gBAAgB,CAAC,QAAQ,EAAE,CAAC;QACvD,IAAI,sBAAsB;YAClB,sBAAsB,CAAC,GAAG,CAAC,oBAAoB,CAAC,EAAE,CAAC;YACzD,OAAO,sBAAsB,CAAC,GAAG,CAAC,oBAAoB,CAAC,IAAI,KAAK,EAAU,CAAC;QAC7E,CAAC;aAAM,CAAC;YACN,OAAO,IAAI,KAAK,EAAU,CAAC;QAC7B,CAAC;IACH,CAAC;IAED,gBAAgB;IAChB,iBAAiB,CAAC,MAAmB,EAAE,QAAqB;QAC1D,MAAM,CAAC,kBAAkB,CAAC,IAAI,CAAC,GAAG,QAAQ,CAAC,kBAAkB,CAAC,CAAC;QAC/D,IAAI,MAAM,CAAC,IAAI,EAAE,CAAC;YAChB,yFAAyF;YACzF,IAAI,MAAM,CAAC,IAAI,IAAI,QAAQ,CAAC,IAAI,EAAE,CAAC;gBACjC,IAAI,GAAG,GAAG,yDAAyD;oBACvD,KAAK,IAAI,CAAC,SAAS,CAAC,MAAM,CAAC,KAAK,IAAI,CAAC,SAAS,CAAC,QAAQ,CAAC,EAAE,CAAC;gBACvE,MAAM,KAAK,CAAC,GAAG,CAAC,CAAC;YACnB,CAAC;QACH,CAAC;aAAM,CAAC;YACN,qCAAqC;YACrC,IAAI,QAAQ,CAAC,IAAI,EAAE,CAAC;gBAClB,MAAM,CAAC,IAAI,GAAG,QAAQ,CAAC,IAAI,CAAC;YAC9B,CAAC;QACH,CAAC;QAED,kCAAkC;QAClC,IAAI,QAAQ,CAAC,yBAAyB,IAAI,QAAQ,CAAC,yBAAyB,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;YACxF,IAAI,CAAC,MAAM,CAAC,yBAAyB,EAAE,CAAC;gBACtC,MAAM,CAAC,yBAAyB,GAAG,EAAE,CAAC;YACxC,CAAC;YACD,MAAM,CAAC,yBAAyB,CAAC,IAAI,CAAC,GAAG,QAAQ,CAAC,yBAAyB,CAAC,CAAC;QAC/E,CAAC;IAEH,CAAC;IAED,mCAAmC,CAAC,qBAA8C,EAChF,aAAiC;QAEjC,IAAI,uBAAuB,GAAuC,IAAI,GAAG,EAAiC,CAAC;QAC3G,2FAA2F;QAC3F,4EAA4E;QAC5E,6CAA6C;QAC7C,4DAA4D;QAC5D,iEAAiE;QACjE,8BAA8B;QAC9B,+DAA+D;QAE/D,KAAK,IAAI,mBAAmB,IAAI,qBAAqB,EAAE,CAAC;YACtD,iFAAiF;YACjF,IAAI,mBAAmB,GAAgB;gBACrC,kBAAkB,EAAE,mBAAmB;gBACvC,kBAAkB,EAAE,IAAI,KAAK,EAAU;gBACvC,qEAAqE;aACtE,CAAC;YACF,uBAAuB,CAAC,GAAG,CAAC,mBAAmB,EAAE,mBAAmB,CAAC,CAAC;YAEtE,+DAA+D;YAC/D,KAAK,IAAI,iBAAiB,IAAI,aAAa,EAAE,CAAC;gBAC5C,IAAI,iBAAiB,CAAC,kBAAkB,YAAY,KAAK,EAAE,CAAC;oBAC1D,KAAK,IAAI,iBAAiB,IAAI,iBAAiB,CAAC,kBAAkB,EAAE,CAAC;wBACnE,IAAI,mBAAmB,IAAI,iBAAiB,EAAE,CAAC;4BAC7C,IAAI,CAAC,iBAAiB,CAAC,mBAAmB,EAAE,iBAAiB,CAAC,CAAC;wBACjE,CAAC;oBACH,CAAC;gBACH,CAAC;qBAAM,IAAI,OAAO,iBAAiB,CAAC,kBAAkB,IAAI,QAAQ,EAAE,CAAC;oBACnE,IAAI,mBAAmB,IAAI,iBAAiB,CAAC,kBAAkB,EAAE,CAAC;wBAChE,IAAI,CAAC,iBAAiB,CAAC,mBAAmB,EAAE,iBAAiB,CAAC,CAAC;oBACjE,CAAC;gBACH,CAAC;qBAAM,CAAC;oBACN,MAAM,KAAK,CAAC,4CAA4C,iBAAiB,CAAC,kBAAkB,EAAE,CAAC,CAAC;gBAClG,CAAC;YACH,CAAC;QACH,CAAC;QAED,MAAM,OAAO,GAAgC,EAAE,CAAC;QAChD,uBAAuB,CAAC,OAAO,CAAC,UAAU,KAAK,EAAE,GAAG;YAClD,OAAO,CAAC,GAAG,CAAC,GAAG,KAAK,CAAC;QACvB,CAAC,CAAC,CAAC;QACH,OAAO,OAAO,CAAC;IACjB,CAAC;IAED,mBAAmB,CAAC,WAAmB,EACrC,qBAA8C,EAC9C,aAAiC,EACjC,YAA2B,EAC3B,gBAAyB,KAAK;QAC9B,IAAI,gBAAgB,GAAG,IAAI,KAAK,EAAmB,CAAC;QACpD,IAAI,2BAA2B,GAAG,IAAI,CAAC,mCAAmC,CAAC,qBAAqB,EAAE,aAAa,CAAC,CAAC;QACjH,IAAI,uBAAuB,GAAuC,IAAI,GAAG,EAAE,CAAC;QAE5E,KAAK,IAAI,CAAC,aAAa,EAAE,UAAU,CAAC,IAAI,MAAM,CAAC,OAAO,CAAC,2BAA2B,CAAC,EAAE,CAAC;YACpF,uBAAuB,CAAC,GAAG,CAAC,4BAA4B,CAAC,aAAa,CAAC,EAAE,UAAU,CAAC,CAAC;QACvF,CAAC;QAED,kEAAkE;QAClE,KAAK,IAAI,mBAAmB,IAAI,qBAAqB,EAAE,CAAC;YAEtD,IAAI,UAAU,GAAgB,uBAAuB,CAAC,GAAG,CAAC,mBAAmB,CAAC;gBACpE;oBACE,qDAAqD;oBACrD,kBAAkB,EAAE,CAAC,mBAAmB,CAAC;oBACzC,kBAAkB,EAAE,IAAI,KAAK,EAAU;oBACvC,IAAI,EAAE,WAAW;iBAClB,CACJ;YAEP,IAAI,gBAAgB,GAAG,UAAU,CAAC,IAAI,IAAI,WAAW,CAAC;YAEtD,IAAI,GAAG,GAAG,oBAAoB,mBAAmB,EAAE,CAAC;YACpD,IAAI,aAAa,EAAE,CAAC;gBAClB,GAAG,GAAG,YAAY,CAAC,GAAG,CAAC,CAAC;YAC1B,CAAC;YAED,IAAI,SAAS,GAAG,IAAI,CAAC,kBAAkB,CAAC,GAAG,EACzC,IAAI,CAAC,UAAU,CAAC,WAAW,EAAE,mBAAmB,CAAC,EACjD,UAAU,CAAC,kBAAkB,EAC7B,gBAAgB,EAChB,YAAY,EACZ,UAAU,CAAC,yBAAyB,CAAC,CAAC;YACxC,gBAAgB,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC;QACnC,CAAC;QACD,OAAO,gBAAgB,CAAC;IAC1B,CAAC;IAED,kBAAkB,CAAC,GAAW,EAC5B,OAAsB,EACtB,aAA4B,EAC5B,IAAsB,EACtB,SAAwB,EACxB,yBAAyC;QACzC,MAAM,oBAAoB,GAAyB;YACjD,GAAG,EAAE,GAAG;YACR,MAAM,EAAE,gBAAM,CAAC,KAAK;SACrB,CAAC;QACF,MAAM,SAAS,GAAG,IAAI,yBAAe,CAAC,oBAAoB,CAAC,CAAC;QAC5D,SAAS,CAAC,UAAU,CAAC,GAAG,OAAO,CAAC,CAAC;QACjC,SAAS,CAAC,eAAe,EAAE,CAAC;QAC5B,SAAS,CAAC,YAAY,CAAC,GAAG,SAAS,CAAC,CAAC;QAErC,MAAM,mBAAmB,GAAG,aAAa,CAAC,QAAQ,CAAC,GAAG,CAAC,CAAC;QACxD,MAAM,gBAAgB,GAAG,yBAAyB,IAAI,yBAAyB,CAAC,MAAM,GAAG,CAAC,CAAC;QAE3F,IAAI,mBAAmB,IAAI,gBAAgB,EAAE,CAAC;YAC5C,yCAAyC;YACzC,wFAAwF;YACxF,wCAAwC;YACxC,SAAS,CAAC,YAAY,CAAC,cAAc,EAAE,EAAE,oBAAoB,EAAE,yBAAyB,EAAE,CAAC,CAAC;QAC9F,CAAC;aAAM,CAAC;YACN,2DAA2D;YAC3D,SAAS,CAAC,YAAY,CAAC,IAAI,EAAE,EAAE,kBAAkB,EAAE,eAAe,CAAC,qBAAqB,CAAC,aAAa,CAAC,EAAE,CAAC,CAAC;YAC3G,IAAI,gBAAgB,EAAE,CAAC;gBACrB,+DAA+D;gBAC/D,SAAS,CAAC,YAAY,CAAC,cAAc,EAAE,EAAE,oBAAoB,EAAE,yBAAyB,EAAE,CAAC,CAAC;YAC9F,CAAC;QACH,CAAC;QAED,OAAO,SAAS,CAAC;IACnB,CAAC;IAED,WAAW,CAAC,WAA0B;QACpC,KAAK,IAAI,UAAU,IAAI,WAAW,EAAE,CAAC;YACnC,IAAI,SAAS,IAAI,UAAU,CAAC,IAAI,EAAE,CAAC;gBACjC,OAAO,IAAI,CAAC;YACd,CAAC;QACH,CAAC;QACD,OAAO,KAAK,CAAC;IACf,CAAC;IAED,uBAAuB,CAAC,WAA0B;QAChD,IAAI,oBAAoB,GAAG,IAAI,GAAG,EAAU,CAAC;QAC7C,KAAK,IAAI,UAAU,IAAI,WAAW,EAAE,CAAC;YACnC,UAAU,CAAC,kBAAkB,CAAC,OAAO,CAAC,UAAU,KAAK;gBACnD,oBAAoB,CAAC,GAAG,CAAC,KAAK,CAAC,CAAC;YAClC,CAAC,CAAC,CAAC;QACL,CAAC;QACD,OAAO,KAAK,CAAC,IAAI,CAAC,oBAAoB,CAAC,CAAC;IAC1C,CAAC;IAED;;;;SAIK;IACL,8BAA8B;QAC5B;;;;;;;;;;;;;;;eAeO;QACP,OAAO,CAAC,IAAI,sBAAY,EAAE,EAAE,IAAI,sBAAY,EAAE,CAAC,CAAC;IAClD,CAAC;IAED;;;;;;;;;;;OAWG;IACH,+BAA+B,CAC7B,WAAmB,EACnB,qBAA8C,EAC9C,uBAA2D,EAC3D,YAA2B;QAE3B,MAAM,UAAU,GAAG,IAAI,GAAG,EAAU,CAAC;QACrC,MAAM,SAAS,GAAG,IAAI,GAAG,EAAU,CAAC;QAEpC,KAAK,IAAI,UAAU,IAAI,qBAAqB,EAAE,CAAC;YAC7C,MAAM,UAAU,GAAG,uBAAuB,CAAC,GAAG,CAAC,UAAU,CAAC,CAAC;YAC3D,IAAI,UAAU,EAAE,yBAAyB,IAAI,UAAU,CAAC,yBAAyB,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;gBAC7F,MAAM,OAAO,GAAG,IAAI,CAAC,UAAU,CAAC,WAAW,EAAE,UAAU,CAAC,CAAC;gBACzD,KAAK,IAAI,MAAM,IAAI,OAAO,EAAE,CAAC;oBAC3B,UAAU,CAAC,GAAG,CAAC,MAAM,CAAC,CAAC;gBACzB,CAAC;gBACD,KAAK,IAAI,KAAK,IAAI,UAAU,CAAC,yBAAyB,EAAE,CAAC;oBACvD,SAAS,CAAC,GAAG,CAAC,KAAK,CAAC,CAAC;gBACvB,CAAC;YACH,CAAC;QACH,CAAC;QAED,IAAI,UAAU,CAAC,IAAI,KAAK,CAAC,EAAE,CAAC;YAC1B,OAAO,SAAS,CAAC;QACnB,CAAC;QAED,MAAM,SAAS,GAAG,IAAI,yBAAe,CAAC;YACpC,GAAG,EAAE,+BAAuB;YAC5B,MAAM,EAAE,gBAAM,CAAC,IAAI;YACnB,UAAU,EAAE,IAAI,CAAC,8BAA8B,EAAE;YACjD,OAAO,EAAE,KAAK,CAAC,IAAI,CAAC,UAAU,CAAC;YAC/B,SAAS,EAAE,YAAY;SACxB,CAAC,CAAC;QACH,SAAS,CAAC,YAAY,CAAC,MAAM,EAAE;YAC7B,2BAA2B,EAAE,CAAC,OAAO,CAAC;SACvC,CAAC,CAAC;QACH,SAAS,CAAC,YAAY,CAAC,iBAAiB,EAAE;YACxC,oBAAoB,EAAE,KAAK,CAAC,IAAI,CAAC,SAAS,CAAC;SAC5C,CAAC,CAAC;QAEH,OAAO,SAAS,CAAC;IACnB,CAAC;;AA5SH,0CA8SC","sourcesContent":["import {\n  AnyPrincipal,\n  ArnPrincipal,\n  Conditions,\n  Effect,\n  PolicyStatement,\n  PolicyStatementProps,\n} from 'aws-cdk-lib/aws-iam';\n\nexport type ArnEqualsTest = 'ArnEquals'\n\nexport type ArnLikeTest = 'ArnLike';\n\nexport type ArnConditionTest =\n    | ArnEqualsTest\n    | ArnLikeTest;\n\nexport enum AccessCapability {\n  ADMINISTER_RESOURCE = 'administer-resource',\n  READ_CONFIG = 'read-config',\n  READ_DATA = 'read-data',\n  WRITE_DATA = 'write-data',\n  DELETE_DATA = 'delete-data',\n}\n\nexport function getAccessCapabilityFromValue(accessCapabilityStr: string): AccessCapability {\n  //https://blog.logrocket.com/typescript-string-enums-guide/\n  for (let key of Object.keys(AccessCapability)) {\n    // @ts-ignore\n    if (AccessCapability[key] == accessCapabilityStr) {\n      // https://stackoverflow.com/questions/17380845/how-do-i-convert-a-string-to-enum-in-typescript\n      let typedKey = <keyof typeof AccessCapability>key;\n      return AccessCapability[typedKey];\n    }\n  }\n\n  throw Error(`Could not get AccessCapability from value: ${accessCapabilityStr}`);\n}\n\nexport interface IAccessSpec {\n  accessCapabilities: Array<AccessCapability> | AccessCapability;\n  allowPrincipalArns: Array<string>;\n  test?: ArnConditionTest;\n  /**\n   * Optional list of AWS Organization IDs that restrict the principals specified\n   * in `allowPrincipalArns`. When present, generated Allow statements will include\n   * a `StringEquals` condition on `aws:PrincipalOrgID` and a DenyUntrustedOrgs statement will\n   * be generated for the permissions that are restricted by org IDs.\n   *\n   * Org IDs restrict which principals are allowed — they do not replace\n   * `allowPrincipalArns`. If you want to allow an entire org, add `*` to `allowPrincipalArns` and the org ID to\n   * `restrictToPrincipalOrgIDs`.\n   */\n  restrictToPrincipalOrgIDs?: Array<string>;\n}\n\n/**\n * `IAWSServiceAccessGenerator` defines an interface that the k9 policy generators use to grant an AWS service\n * access to a protected resource.\n */\nexport interface IAWSServiceAccessGenerator {\n  /**\n   * Make an array of PolicyStatement objects that allow an AWS service, e.g. CloudFront, to access to the\n   * protected AWS resource.\n   */\n  makeAllowStatements(): Array<PolicyStatement>;\n\n  /**\n   * Make a Conditions object that creates an exception for an AWS service in a protected resource's `DenyEveryoneElse`\n   * statement.\n   */\n  makeConditionsToExceptFromDenyEveryoneElse(): Conditions;\n}\n\n/**\n * Check whether the provided access specs ensure that at least one principal can both read and administer configuration.\n * @param accessSpecsByCapability is a map of access specs keyed by access capability\n *\n * @return true when at least one principal that can administer and read configuration exists\n */\nexport function canPrincipalsManageResources(accessSpecsByCapability: Map<AccessCapability, IAccessSpec>) {\n  let adminSpec = accessSpecsByCapability.get(AccessCapability.ADMINISTER_RESOURCE);\n  let readConfigSpec = accessSpecsByCapability.get(AccessCapability.READ_CONFIG);\n\n  if ((adminSpec?.allowPrincipalArns && adminSpec.allowPrincipalArns.length > 0)\n        && (readConfigSpec?.allowPrincipalArns && readConfigSpec.allowPrincipalArns.length > 0)) {\n    const adminPrincipals = new Set<string>(adminSpec.allowPrincipalArns);\n    const readConfigPrincipals = new Set<string>(readConfigSpec.allowPrincipalArns);\n    const intersection = new Set(\n      [...adminPrincipals].filter(x => readConfigPrincipals.has(x)));\n    return intersection.size > 0;\n  }\n  return false;\n}\n\n\n/**\n * Check if any access spec contains a wildcard principal (\"*\").\n */\nexport function hasWildcardPrincipal(accessSpecs: Array<IAccessSpec>): boolean {\n  for (let spec of accessSpecs) {\n    if (spec.allowPrincipalArns.includes('*')) {\n      return true;\n    }\n  }\n  return false;\n}\n\n/**\n * Validate that access specs have valid principal ARN + org constraint combinations.\n * Throws an error for invalid combinations:\n * - Empty allowPrincipalArns\n * - Wildcard allowPrincipalArns without restrictToPrincipalOrgIDs (public access)\n */\nexport function validateAccessSpecs(accessSpecs: Array<IAccessSpec>): void {\n  for (let spec of accessSpecs) {\n    if (!spec.allowPrincipalArns || spec.allowPrincipalArns.length === 0) {\n      throw new Error(\n        'allowPrincipalArns must not be empty; every resource policy statement requires a Principal element.',\n      );\n    }\n    if (spec.allowPrincipalArns.includes('*') &&\n        (!spec.restrictToPrincipalOrgIDs || spec.restrictToPrincipalOrgIDs.length === 0)) {\n      throw new Error(\n        'k9-cdk will not generate a resource policy that allows fully public access.' +\n        ' Wildcard principal (\"*\") requires restrictToPrincipalOrgIDs to scope access.' +\n        ' Consider specifying account principal ARNs or constraining to specific PrincipalOrgIDs.',\n      );\n    }\n  }\n}\n\n/**\n * Converts a string to PascalCase, which is useful for e.g. policy types that don't\n * do not support spaces or hyphens in statement ids.\n *\n * @param input\n */\nexport function toPascalCase(input: string): string {\n  // Remove placeholders like ${something} and trim whitespace\n  const cleanedInput = input.replace(/\\$\\{.*?\\}/g, '').trim();\n\n  // Split the input into words based on spaces, hyphens, underscores, or other delimiters\n  const words = cleanedInput.split(/[\\s_\\-]+/);\n\n  // Convert each word to PascalCase\n  return words\n    .map(\n      word => word.charAt(0).toUpperCase() + word.slice(1).toLowerCase(), // Capitalize the first letter, lower the rest\n    )\n    .join('');\n}\n\nexport const SID_DENY_UNTRUSTED_ORGS = 'DenyUntrustedOrgs';\n\nexport class K9PolicyFactory {\n\n  /**\n   * Deduplicate an array of principals while preserving original order of principals.\n   * Note that principals may contain either strings or objects, so naive array sorting\n   * produces unstable results.\n   *\n   * @param principals\n   */\n  static deduplicatePrincipals(principals: Array<string|object>): Array<string|object> {\n    const observedPrincipals = new Set<string|object>();\n    const uniquePrincipals = new Array<string|object>();\n    for (let principal of principals) {\n      if (!observedPrincipals.has(principal)) {\n        uniquePrincipals.push(principal);\n        observedPrincipals.add(principal);\n      }\n    }\n    return uniquePrincipals;\n  }\n\n  /** @internal */\n  _SUPPORTED_SERVICES = new Set<string>([\n    'S3',\n    'KMS',\n    'DynamoDB',\n    'SQS',\n    'EventBridge',\n  ]);\n\n  /** @internal */\n  _K9CapabilityMapJSON: Object = require('../resources/capability_summary.json'); // eslint-disable-line @typescript-eslint/no-require-imports\n  /** @internal */\n  _K9CapabilityMapByService: Map<string, Object> = new Map(Object.entries(this._K9CapabilityMapJSON));\n\n  getActions(service: string, accessCapability: AccessCapability): Array<string> {\n    if (!this._SUPPORTED_SERVICES.has(service) && this._K9CapabilityMapByService.has(service)) {\n      throw Error(`unsupported service: ${service}`);\n    }\n\n    let serviceCapabilitiesObj: Object = this._K9CapabilityMapByService.get(service) || {};\n    let serviceCapabilitiesMap = new Map<string, Array<string>>(Object.entries(serviceCapabilitiesObj));\n\n    let accessCapabilityName = accessCapability.toString();\n    if (serviceCapabilitiesMap &&\n            serviceCapabilitiesMap.has(accessCapabilityName)) {\n      return serviceCapabilitiesMap.get(accessCapabilityName) || Array<string>();\n    } else {\n      return new Array<string>();\n    }\n  }\n\n  /** @internal */\n  _mergeAccessSpecs(target: IAccessSpec, addition: IAccessSpec) {\n    target.allowPrincipalArns.push(...addition.allowPrincipalArns);\n    if (target.test) {\n      //ok, user has specified a test at some point; ensure this desiredAccessSpec.test matches\n      if (target.test != addition.test) {\n        let msg = 'Cannot merge AccessSpecs; test attributes do not match:' +\n                    `\\n${JSON.stringify(target)}\\n${JSON.stringify(addition)}`;\n        throw Error(msg);\n      }\n    } else {\n      //first explicit test preference wins\n      if (addition.test) {\n        target.test = addition.test;\n      }\n    }\n\n    // Merge restrictToPrincipalOrgIDs\n    if (addition.restrictToPrincipalOrgIDs && addition.restrictToPrincipalOrgIDs.length > 0) {\n      if (!target.restrictToPrincipalOrgIDs) {\n        target.restrictToPrincipalOrgIDs = [];\n      }\n      target.restrictToPrincipalOrgIDs.push(...addition.restrictToPrincipalOrgIDs);\n    }\n\n  }\n\n  mergeDesiredAccessSpecsByCapability(supportedCapabilities: Array<AccessCapability>,\n    desiredAccess: Array<IAccessSpec>): Record<string, IAccessSpec> {\n\n    let accessSpecsByCapability: Map<AccessCapability, IAccessSpec> = new Map<AccessCapability, IAccessSpec>();\n    // 1. populate accessSpecsByCapability with fresh AccessSpecs for each supported capability\n    // 2. iterate through desiredAccess specs and merge data into what we'll use\n    //    important: detect mismatched test types\n    //     we can leave `test` unset in the default access specs\n    //     and copy the value from the spec being merged if it is set\n    //     throw Error on mismatch\n    // 3. generate an Allow statement for each supported capability\n\n    for (let supportedCapability of supportedCapabilities) {\n      //generate a default access spec for each of the service's supported capabilities\n      let effectiveAccessSpec: IAccessSpec = {\n        accessCapabilities: supportedCapability,\n        allowPrincipalArns: new Array<string>(),\n        // leave 'test' property unset; will populate from user-provided data\n      };\n      accessSpecsByCapability.set(supportedCapability, effectiveAccessSpec);\n\n      //now... merge in the user's desired access for this capability\n      for (let desiredAccessSpec of desiredAccess) {\n        if (desiredAccessSpec.accessCapabilities instanceof Array) {\n          for (let desiredCapability of desiredAccessSpec.accessCapabilities) {\n            if (supportedCapability == desiredCapability) {\n              this._mergeAccessSpecs(effectiveAccessSpec, desiredAccessSpec);\n            }\n          }\n        } else if (typeof desiredAccessSpec.accessCapabilities == 'string') {\n          if (supportedCapability == desiredAccessSpec.accessCapabilities) {\n            this._mergeAccessSpecs(effectiveAccessSpec, desiredAccessSpec);\n          }\n        } else {\n          throw Error(`Unhandled type of accessCapabilities for ${desiredAccessSpec.accessCapabilities}`);\n        }\n      }\n    }\n\n    const records: Record<string, IAccessSpec> = {};\n    accessSpecsByCapability.forEach(function (value, key) {\n      records[key] = value;\n    });\n    return records;\n  }\n\n  makeAllowStatements(serviceName: string,\n    supportedCapabilities: Array<AccessCapability>,\n    desiredAccess: Array<IAccessSpec>,\n    resourceArns: Array<string>,\n    usePascalCase: boolean = false): Array<PolicyStatement> {\n    let policyStatements = new Array<PolicyStatement>();\n    let accessSpecsByCapabilityRecs = this.mergeDesiredAccessSpecsByCapability(supportedCapabilities, desiredAccess);\n    let accessSpecsByCapability: Map<AccessCapability, IAccessSpec> = new Map();\n\n    for (let [capabilityStr, accessSpec] of Object.entries(accessSpecsByCapabilityRecs)) {\n      accessSpecsByCapability.set(getAccessCapabilityFromValue(capabilityStr), accessSpec);\n    }\n\n    // ok, time to actually make Allow Statements from our AccessSpecs\n    for (let supportedCapability of supportedCapabilities) {\n\n      let accessSpec: IAccessSpec = accessSpecsByCapability.get(supportedCapability) ||\n                { //satisfy compiler; should never happen, because we populate at the beginning.\n                  //generate a default access spec if none was provided\n                  accessCapabilities: [supportedCapability],\n                  allowPrincipalArns: new Array<string>(),\n                  test: 'ArnEquals',\n                }\n            ;\n\n      let arnConditionTest = accessSpec.test || 'ArnEquals';\n\n      let sid = `Allow Restricted ${supportedCapability}`;\n      if (usePascalCase) {\n        sid = toPascalCase(sid);\n      }\n\n      let statement = this.makeAllowStatement(sid,\n        this.getActions(serviceName, supportedCapability),\n        accessSpec.allowPrincipalArns,\n        arnConditionTest,\n        resourceArns,\n        accessSpec.restrictToPrincipalOrgIDs);\n      policyStatements.push(statement);\n    }\n    return policyStatements;\n  }\n\n  makeAllowStatement(sid: string,\n    actions: Array<string>,\n    principalArns: Array<string>,\n    test: ArnConditionTest,\n    resources: Array<string>,\n    restrictToPrincipalOrgIDs?: Array<string>): PolicyStatement {\n    const policyStatementProps: PolicyStatementProps = {\n      sid: sid,\n      effect: Effect.ALLOW,\n    };\n    const statement = new PolicyStatement(policyStatementProps);\n    statement.addActions(...actions);\n    statement.addAnyPrincipal();\n    statement.addResources(...resources);\n\n    const isWildcardPrincipal = principalArns.includes('*');\n    const hasOrgConstraint = restrictToPrincipalOrgIDs && restrictToPrincipalOrgIDs.length > 0;\n\n    if (isWildcardPrincipal && hasOrgConstraint) {\n      // Code Path B: wildcard + org constraint\n      // Use Principal: \"*\" (already added via addAnyPrincipal) + aws:PrincipalOrgID condition\n      // Do NOT add aws:PrincipalArn condition\n      statement.addCondition('StringEquals', { 'aws:PrincipalOrgID': restrictToPrincipalOrgIDs });\n    } else {\n      // Code Path A: specific principal ARNs (existing behavior)\n      statement.addCondition(test, { 'aws:PrincipalArn': K9PolicyFactory.deduplicatePrincipals(principalArns) });\n      if (hasOrgConstraint) {\n        // Specific ARNs + org constraint: both conditions must be true\n        statement.addCondition('StringEquals', { 'aws:PrincipalOrgID': restrictToPrincipalOrgIDs });\n      }\n    }\n\n    return statement;\n  }\n\n  wasLikeUsed(accessSpecs: IAccessSpec[]): boolean {\n    for (let accessSpec of accessSpecs) {\n      if ('ArnLike' == accessSpec.test) {\n        return true;\n      }\n    }\n    return false;\n  }\n\n  getAllowedPrincipalArns(accessSpecs: IAccessSpec[]): Array<string> {\n    let allowedPrincipalArns = new Set<string>();\n    for (let accessSpec of accessSpecs) {\n      accessSpec.allowPrincipalArns.forEach(function (value) {\n        allowedPrincipalArns.add(value);\n      });\n    }\n    return Array.from(allowedPrincipalArns);\n  }\n\n  /**\n     * k9 wants to deny all AWS accounts and IAM principals not explicitly allowed; this *should*\n     * be straightforward, but it isn't because of the way aws-cdk merges and manipulates Principals.\n     * @return list of principals for a DenyEveryoneElse statement\n     */\n  makeDenyEveryoneElsePrincipals(): ArnPrincipal[] {\n    /**\n         * We should be able to provide AnyPrincipal once (of course), but AWS CDK converts:\n         * \"Principal\": {\n         *   \"AWS\": \"*\"    // identifies all AWS accounts and IAM.\n         * }\n         * to:\n         * \"