UNPKG

14.6 kBJavaScriptView Raw
1"use strict";
2Object.defineProperty(exports, "__esModule", { value: true });
3exports.UniqueStringSet = exports.mergePrincipal = exports.AttachedPolicies = exports.generatePolicyName = exports.undefinedIfEmpty = exports.LITERAL_STRING_KEY = void 0;
4const core_1 = require("@aws-cdk/core");
5const MAX_POLICY_NAME_LEN = 128;
6exports.LITERAL_STRING_KEY = 'LiteralString';
7function undefinedIfEmpty(f) {
8 return core_1.Lazy.list({
9 produce: () => {
10 const array = f();
11 return (array && array.length > 0) ? array : undefined;
12 },
13 });
14}
15exports.undefinedIfEmpty = undefinedIfEmpty;
16/**
17 * Used to generate a unique policy name based on the policy resource construct.
18 * The logical ID of the resource is a great candidate as long as it doesn't exceed
19 * 128 characters, so we take the last 128 characters (in order to make sure the hash
20 * is there).
21 */
22function generatePolicyName(scope, logicalId) {
23 // as logicalId is itself a Token, resolve it first
24 const resolvedLogicalId = core_1.Tokenization.resolve(logicalId, {
25 scope,
26 resolver: new core_1.DefaultTokenResolver(new core_1.StringConcat()),
27 });
28 return lastNCharacters(resolvedLogicalId, MAX_POLICY_NAME_LEN);
29}
30exports.generatePolicyName = generatePolicyName;
31/**
32 * Returns a string composed of the last n characters of str.
33 * If str is shorter than n, returns str.
34 *
35 * @param str the string to return the last n characters of
36 * @param n how many characters to return
37 */
38function lastNCharacters(str, n) {
39 const startIndex = Math.max(str.length - n, 0);
40 return str.substring(startIndex, str.length);
41}
42/**
43 * Helper class that maintains the set of attached policies for a principal.
44 */
45class AttachedPolicies {
46 constructor() {
47 this.policies = new Array();
48 }
49 /**
50 * Adds a policy to the list of attached policies.
51 *
52 * If this policy is already, attached, returns false.
53 * If there is another policy attached with the same name, throws an exception.
54 */
55 attach(policy) {
56 if (this.policies.find(p => p === policy)) {
57 return; // already attached
58 }
59 if (this.policies.find(p => p.policyName === policy.policyName)) {
60 throw new Error(`A policy named "${policy.policyName}" is already attached`);
61 }
62 this.policies.push(policy);
63 }
64}
65exports.AttachedPolicies = AttachedPolicies;
66/**
67 * Merge two dictionaries that represent IAM principals
68 *
69 * Does an in-place merge.
70 */
71function mergePrincipal(target, source) {
72 var _a;
73 // If one represents a literal string, the other one must be empty
74 if ((exports.LITERAL_STRING_KEY in source && !isEmptyObject(target)) ||
75 (exports.LITERAL_STRING_KEY in target && !isEmptyObject(source))) {
76 throw new Error(`Cannot merge principals ${JSON.stringify(target)} and ${JSON.stringify(source)}; if one uses a literal principal string the other one must be empty`);
77 }
78 for (const key of Object.keys(source)) {
79 target[key] = (_a = target[key]) !== null && _a !== void 0 ? _a : [];
80 let value = source[key];
81 if (!Array.isArray(value)) {
82 value = [value];
83 }
84 target[key].push(...value);
85 }
86 return target;
87}
88exports.mergePrincipal = mergePrincipal;
89/**
90 * Lazy string set token that dedupes entries
91 *
92 * Needs to operate post-resolve, because the inputs could be
93 * `[ '${Token[TOKEN.9]}', '${Token[TOKEN.10]}', '${Token[TOKEN.20]}' ]`, which
94 * still all resolve to the same string value.
95 *
96 * Needs to JSON.stringify() results because strings could resolve to literal
97 * strings but could also resolve to `{ Fn::Join: [...] }`.
98 */
99class UniqueStringSet {
100 constructor(fn) {
101 this.fn = fn;
102 this.creationStack = core_1.captureStackTrace();
103 }
104 static from(fn) {
105 return core_1.Token.asList(new UniqueStringSet(fn));
106 }
107 resolve(context) {
108 context.registerPostProcessor(this);
109 return this.fn();
110 }
111 postProcess(input, _context) {
112 if (!Array.isArray(input)) {
113 return input;
114 }
115 if (input.length === 0) {
116 return undefined;
117 }
118 const uniq = {};
119 for (const el of input) {
120 uniq[JSON.stringify(el)] = el;
121 }
122 return Object.values(uniq);
123 }
124 toString() {
125 return core_1.Token.asString(this);
126 }
127}
128exports.UniqueStringSet = UniqueStringSet;
129function isEmptyObject(x) {
130 return Object.keys(x).length === 0;
131}
132//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"util.js","sourceRoot":"","sources":["util.ts"],"names":[],"mappings":"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","sourcesContent":["import { captureStackTrace, DefaultTokenResolver, IPostProcessor, IResolvable, IResolveContext, Lazy, StringConcat, Token, Tokenization } from '@aws-cdk/core';\nimport { IConstruct } from 'constructs';\nimport { IPolicy } from './policy';\n\nconst MAX_POLICY_NAME_LEN = 128;\n\nexport const LITERAL_STRING_KEY = 'LiteralString';\n\nexport function undefinedIfEmpty(f: () => string[]): string[] {\n  return Lazy.list({\n    produce: () => {\n      const array = f();\n      return (array && array.length > 0) ? array : undefined;\n    },\n  });\n}\n\n/**\n * Used to generate a unique policy name based on the policy resource construct.\n * The logical ID of the resource is a great candidate as long as it doesn't exceed\n * 128 characters, so we take the last 128 characters (in order to make sure the hash\n * is there).\n */\nexport function generatePolicyName(scope: IConstruct, logicalId: string): string {\n  // as logicalId is itself a Token, resolve it first\n  const resolvedLogicalId = Tokenization.resolve(logicalId, {\n    scope,\n    resolver: new DefaultTokenResolver(new StringConcat()),\n  });\n  return lastNCharacters(resolvedLogicalId, MAX_POLICY_NAME_LEN);\n}\n\n/**\n * Returns a string composed of the last n characters of str.\n * If str is shorter than n, returns str.\n *\n * @param str the string to return the last n characters of\n * @param n how many characters to return\n */\nfunction lastNCharacters(str: string, n: number) {\n  const startIndex = Math.max(str.length - n, 0);\n  return str.substring(startIndex, str.length);\n}\n\n/**\n * Helper class that maintains the set of attached policies for a principal.\n */\nexport class AttachedPolicies {\n  private policies = new Array<IPolicy>();\n\n  /**\n   * Adds a policy to the list of attached policies.\n   *\n   * If this policy is already, attached, returns false.\n   * If there is another policy attached with the same name, throws an exception.\n   */\n  public attach(policy: IPolicy) {\n    if (this.policies.find(p => p === policy)) {\n      return; // already attached\n    }\n\n    if (this.policies.find(p => p.policyName === policy.policyName)) {\n      throw new Error(`A policy named \"${policy.policyName}\" is already attached`);\n    }\n\n    this.policies.push(policy);\n  }\n}\n\n/**\n * Merge two dictionaries that represent IAM principals\n *\n * Does an in-place merge.\n */\nexport function mergePrincipal(target: { [key: string]: string[] }, source: { [key: string]: string[] }) {\n  // If one represents a literal string, the other one must be empty\n  if ((LITERAL_STRING_KEY in source && !isEmptyObject(target)) ||\n    (LITERAL_STRING_KEY in target && !isEmptyObject(source))) {\n    throw new Error(`Cannot merge principals ${JSON.stringify(target)} and ${JSON.stringify(source)}; if one uses a literal principal string the other one must be empty`);\n  }\n\n  for (const key of Object.keys(source)) {\n    target[key] = target[key] ?? [];\n\n    let value = source[key];\n    if (!Array.isArray(value)) {\n      value = [value];\n    }\n\n    target[key].push(...value);\n  }\n\n  return target;\n}\n\n/**\n * Lazy string set token that dedupes entries\n *\n * Needs to operate post-resolve, because the inputs could be\n * `[ '${Token[TOKEN.9]}', '${Token[TOKEN.10]}', '${Token[TOKEN.20]}' ]`, which\n * still all resolve to the same string value.\n *\n * Needs to JSON.stringify() results because strings could resolve to literal\n * strings but could also resolve to `{ Fn::Join: [...] }`.\n */\nexport class UniqueStringSet implements IResolvable, IPostProcessor {\n  public static from(fn: () => string[]) {\n    return Token.asList(new UniqueStringSet(fn));\n  }\n\n  public readonly creationStack: string[];\n\n  private constructor(private readonly fn: () => string[]) {\n    this.creationStack = captureStackTrace();\n  }\n\n  public resolve(context: IResolveContext) {\n    context.registerPostProcessor(this);\n    return this.fn();\n  }\n\n  public postProcess(input: any, _context: IResolveContext) {\n    if (!Array.isArray(input)) { return input; }\n    if (input.length === 0) { return undefined; }\n\n    const uniq: Record<string, any> = {};\n    for (const el of input) {\n      uniq[JSON.stringify(el)] = el;\n    }\n    return Object.values(uniq);\n  }\n\n  public toString(): string {\n    return Token.asString(this);\n  }\n}\n\nfunction isEmptyObject(x: { [key: string]: any }): boolean {\n  return Object.keys(x).length === 0;\n}\n"]}
\No newline at end of file