1 | ;
|
2 | var _a;
|
3 | Object.defineProperty(exports, "__esModule", { value: true });
|
4 | exports.TagManager = void 0;
|
5 | const jsiiDeprecationWarnings = require("../.warnings.jsii.js");
|
6 | const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
|
7 | const cfn_resource_1 = require("./cfn-resource");
|
8 | const lazy_1 = require("./lazy");
|
9 | /**
|
10 | * Standard tags are a list of { key, value } objects
|
11 | */
|
12 | class StandardFormatter {
|
13 | parseTags(cfnPropertyTags, priority) {
|
14 | if (!Array.isArray(cfnPropertyTags)) {
|
15 | throw new Error(`Invalid tag input expected array of {key, value} have ${JSON.stringify(cfnPropertyTags)}`);
|
16 | }
|
17 | const tags = [];
|
18 | const dynamicTags = [];
|
19 | for (const tag of cfnPropertyTags) {
|
20 | if (tag.key === undefined || tag.value === undefined) {
|
21 | dynamicTags.push(tag);
|
22 | }
|
23 | else {
|
24 | // using interp to ensure Token is now string
|
25 | tags.push({
|
26 | key: `${tag.key}`,
|
27 | value: `${tag.value}`,
|
28 | priority,
|
29 | });
|
30 | }
|
31 | }
|
32 | return { tags, dynamicTags };
|
33 | }
|
34 | formatTags(tags) {
|
35 | const cfnTags = [];
|
36 | for (const tag of tags) {
|
37 | cfnTags.push({
|
38 | key: tag.key,
|
39 | value: tag.value,
|
40 | });
|
41 | }
|
42 | return cfnTags;
|
43 | }
|
44 | }
|
45 | /**
|
46 | * ASG tags are a list of { key, value, propagateAtLaunch } objects
|
47 | */
|
48 | class AsgFormatter {
|
49 | parseTags(cfnPropertyTags, priority) {
|
50 | if (!Array.isArray(cfnPropertyTags)) {
|
51 | throw new Error(`Invalid tag input expected array of {key, value, propagateAtLaunch} have ${JSON.stringify(cfnPropertyTags)}`);
|
52 | }
|
53 | const tags = [];
|
54 | const dynamicTags = [];
|
55 | for (const tag of cfnPropertyTags) {
|
56 | if (tag.key === undefined ||
|
57 | tag.value === undefined ||
|
58 | tag.propagateAtLaunch === undefined) {
|
59 | dynamicTags.push(tag);
|
60 | }
|
61 | else {
|
62 | // using interp to ensure Token is now string
|
63 | tags.push({
|
64 | key: `${tag.key}`,
|
65 | value: `${tag.value}`,
|
66 | priority,
|
67 | applyToLaunchedInstances: !!tag.propagateAtLaunch,
|
68 | });
|
69 | }
|
70 | }
|
71 | return { tags, dynamicTags };
|
72 | }
|
73 | formatTags(tags) {
|
74 | const cfnTags = [];
|
75 | for (const tag of tags) {
|
76 | cfnTags.push({
|
77 | key: tag.key,
|
78 | value: tag.value,
|
79 | propagateAtLaunch: tag.applyToLaunchedInstances !== false,
|
80 | });
|
81 | }
|
82 | return cfnTags;
|
83 | }
|
84 | }
|
85 | /**
|
86 | * Some CloudFormation constructs use a { key: value } map for tags
|
87 | */
|
88 | class MapFormatter {
|
89 | parseTags(cfnPropertyTags, priority) {
|
90 | if (Array.isArray(cfnPropertyTags) || typeof (cfnPropertyTags) !== 'object') {
|
91 | throw new Error(`Invalid tag input expected map of {key: value} have ${JSON.stringify(cfnPropertyTags)}`);
|
92 | }
|
93 | const tags = [];
|
94 | for (const [key, value] of Object.entries(cfnPropertyTags)) {
|
95 | tags.push({
|
96 | key,
|
97 | value: `${value}`,
|
98 | priority,
|
99 | });
|
100 | }
|
101 | return { tags, dynamicTags: undefined };
|
102 | }
|
103 | formatTags(tags) {
|
104 | const cfnTags = {};
|
105 | for (const tag of tags) {
|
106 | cfnTags[`${tag.key}`] = `${tag.value}`;
|
107 | }
|
108 | return cfnTags;
|
109 | }
|
110 | }
|
111 | /**
|
112 | * StackTags are of the format { Key: key, Value: value }
|
113 | */
|
114 | class KeyValueFormatter {
|
115 | parseTags(keyValueTags, priority) {
|
116 | const tags = [];
|
117 | for (const key in keyValueTags) {
|
118 | if (keyValueTags.hasOwnProperty(key)) {
|
119 | const value = keyValueTags[key];
|
120 | tags.push({
|
121 | key,
|
122 | value,
|
123 | priority,
|
124 | });
|
125 | }
|
126 | }
|
127 | return { tags, dynamicTags: undefined };
|
128 | }
|
129 | formatTags(unformattedTags) {
|
130 | const tags = [];
|
131 | unformattedTags.forEach(tag => {
|
132 | tags.push({
|
133 | Key: tag.key,
|
134 | Value: tag.value,
|
135 | });
|
136 | });
|
137 | return tags;
|
138 | }
|
139 | }
|
140 | class NoFormat {
|
141 | parseTags(_cfnPropertyTags) {
|
142 | return { tags: [], dynamicTags: undefined };
|
143 | }
|
144 | formatTags(_tags) {
|
145 | return undefined;
|
146 | }
|
147 | }
|
148 | let _tagFormattersCache;
|
149 | /**
|
150 | * Access tag formatters table
|
151 | *
|
152 | * In a function because we're in a load cycle with cfn-resource that defines `TagType`.
|
153 | */
|
154 | function TAG_FORMATTERS() {
|
155 | return _tagFormattersCache ?? (_tagFormattersCache = {
|
156 | [cfn_resource_1.TagType.AUTOSCALING_GROUP]: new AsgFormatter(),
|
157 | [cfn_resource_1.TagType.STANDARD]: new StandardFormatter(),
|
158 | [cfn_resource_1.TagType.MAP]: new MapFormatter(),
|
159 | [cfn_resource_1.TagType.KEY_VALUE]: new KeyValueFormatter(),
|
160 | [cfn_resource_1.TagType.NOT_TAGGABLE]: new NoFormat(),
|
161 | });
|
162 | }
|
163 | /**
|
164 | * TagManager facilitates a common implementation of tagging for Constructs
|
165 | *
|
166 | * Normally, you do not need to use this class, as the CloudFormation specification
|
167 | * will indicate which resources are taggable. However, sometimes you will need this
|
168 | * to make custom resources taggable. Used `tagManager.renderedTags` to obtain a
|
169 | * value that will resolve to the tags at synthesis time.
|
170 | *
|
171 | * @example
|
172 | * import * as cdk from '@aws-cdk/core';
|
173 | *
|
174 | * class MyConstruct extends cdk.Resource implements cdk.ITaggable {
|
175 | * public readonly tags = new cdk.TagManager(cdk.TagType.KEY_VALUE, 'Whatever::The::Type');
|
176 | *
|
177 | * constructor(scope: cdk.Construct, id: string) {
|
178 | * super(scope, id);
|
179 | *
|
180 | * new cdk.CfnResource(this, 'Resource', {
|
181 | * type: 'Whatever::The::Type',
|
182 | * properties: {
|
183 | * // ...
|
184 | * Tags: this.tags.renderedTags,
|
185 | * },
|
186 | * });
|
187 | * }
|
188 | * }
|
189 | *
|
190 | */
|
191 | class TagManager {
|
192 | constructor(tagType, resourceTypeName, tagStructure, options = {}) {
|
193 | this.tags = new Map();
|
194 | this.priorities = new Map();
|
195 | this.initialTagPriority = 50;
|
196 | try {
|
197 | jsiiDeprecationWarnings._aws_cdk_core_TagType(tagType);
|
198 | jsiiDeprecationWarnings._aws_cdk_core_TagManagerOptions(options);
|
199 | }
|
200 | catch (error) {
|
201 | if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") {
|
202 | Error.captureStackTrace(error, TagManager);
|
203 | }
|
204 | throw error;
|
205 | }
|
206 | this.resourceTypeName = resourceTypeName;
|
207 | this.tagFormatter = TAG_FORMATTERS()[tagType];
|
208 | if (tagStructure !== undefined) {
|
209 | const parseTagsResult = this.tagFormatter.parseTags(tagStructure, this.initialTagPriority);
|
210 | this.dynamicTags = parseTagsResult.dynamicTags;
|
211 | this._setTag(...parseTagsResult.tags);
|
212 | }
|
213 | this.tagPropertyName = options.tagPropertyName || 'tags';
|
214 | this.renderedTags = lazy_1.Lazy.any({ produce: () => this.renderTags() });
|
215 | }
|
216 | /**
|
217 | * Check whether the given construct is Taggable
|
218 | */
|
219 | static isTaggable(construct) {
|
220 | return construct.tags !== undefined;
|
221 | }
|
222 | /**
|
223 | * Adds the specified tag to the array of tags
|
224 | *
|
225 | */
|
226 | setTag(key, value, priority = 0, applyToLaunchedInstances = true) {
|
227 | // This method mostly exists because we don't want to expose the 'Tag' type used (it will be confusing
|
228 | // to users).
|
229 | this._setTag({ key, value, priority, applyToLaunchedInstances });
|
230 | }
|
231 | /**
|
232 | * Removes the specified tag from the array if it exists
|
233 | *
|
234 | * @param key The tag to remove
|
235 | * @param priority The priority of the remove operation
|
236 | */
|
237 | removeTag(key, priority) {
|
238 | if (priority >= (this.priorities.get(key) || 0)) {
|
239 | this.tags.delete(key);
|
240 | this.priorities.set(key, priority);
|
241 | }
|
242 | }
|
243 | /**
|
244 | * Renders tags into the proper format based on TagType
|
245 | *
|
246 | * This method will eagerly render the tags currently applied. In
|
247 | * most cases, you should be using `tagManager.renderedTags` instead,
|
248 | * which will return a `Lazy` value that will resolve to the correct
|
249 | * tags at synthesis time.
|
250 | */
|
251 | renderTags() {
|
252 | const formattedTags = this.tagFormatter.formatTags(this.sortedTags);
|
253 | if (Array.isArray(formattedTags) || Array.isArray(this.dynamicTags)) {
|
254 | const ret = [...formattedTags ?? [], ...this.dynamicTags ?? []];
|
255 | return ret.length > 0 ? ret : undefined;
|
256 | }
|
257 | else {
|
258 | const ret = { ...formattedTags ?? {}, ...this.dynamicTags ?? {} };
|
259 | return Object.keys(ret).length > 0 ? ret : undefined;
|
260 | }
|
261 | }
|
262 | /**
|
263 | * Render the tags in a readable format
|
264 | */
|
265 | tagValues() {
|
266 | const ret = {};
|
267 | for (const tag of this.sortedTags) {
|
268 | ret[tag.key] = tag.value;
|
269 | }
|
270 | return ret;
|
271 | }
|
272 | /**
|
273 | * Determine if the aspect applies here
|
274 | *
|
275 | * Looks at the include and exclude resourceTypeName arrays to determine if
|
276 | * the aspect applies here
|
277 | */
|
278 | applyTagAspectHere(include, exclude) {
|
279 | if (exclude && exclude.length > 0 && exclude.indexOf(this.resourceTypeName) !== -1) {
|
280 | return false;
|
281 | }
|
282 | if (include && include.length > 0 && include.indexOf(this.resourceTypeName) === -1) {
|
283 | return false;
|
284 | }
|
285 | return true;
|
286 | }
|
287 | /**
|
288 | * Returns true if there are any tags defined
|
289 | */
|
290 | hasTags() {
|
291 | return this.tags.size > 0;
|
292 | }
|
293 | _setTag(...tags) {
|
294 | for (const tag of tags) {
|
295 | if (tag.priority >= (this.priorities.get(tag.key) || 0)) {
|
296 | this.tags.set(tag.key, tag);
|
297 | this.priorities.set(tag.key, tag.priority);
|
298 | }
|
299 | }
|
300 | }
|
301 | get sortedTags() {
|
302 | return Array.from(this.tags.values())
|
303 | .sort((a, b) => a.key.localeCompare(b.key));
|
304 | }
|
305 | }
|
306 | exports.TagManager = TagManager;
|
307 | _a = JSII_RTTI_SYMBOL_1;
|
308 | TagManager[_a] = { fqn: "@aws-cdk/core.TagManager", version: "1.204.0" };
|
309 | //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"tag-manager.js","sourceRoot":"","sources":["tag-manager.ts"],"names":[],"mappings":";;;;;;AAAA,iDAAyC;AAEzC,iCAA8B;AA2D9B;;GAEG;AACH,MAAM,iBAAiB;IACd,SAAS,CAAC,eAAoB,EAAE,QAAgB;QACrD,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,eAAe,CAAC,EAAE;YACnC,MAAM,IAAI,KAAK,CAAC,yDAAyD,IAAI,CAAC,SAAS,CAAC,eAAe,CAAC,EAAE,CAAC,CAAC;SAC7G;QAED,MAAM,IAAI,GAAU,EAAE,CAAC;QACvB,MAAM,WAAW,GAAQ,EAAE,CAAC;QAC5B,KAAK,MAAM,GAAG,IAAI,eAAe,EAAE;YACjC,IAAI,GAAG,CAAC,GAAG,KAAK,SAAS,IAAI,GAAG,CAAC,KAAK,KAAK,SAAS,EAAE;gBACpD,WAAW,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;aACvB;iBAAM;gBACL,6CAA6C;gBAC7C,IAAI,CAAC,IAAI,CAAC;oBACR,GAAG,EAAE,GAAG,GAAG,CAAC,GAAG,EAAE;oBACjB,KAAK,EAAE,GAAG,GAAG,CAAC,KAAK,EAAE;oBACrB,QAAQ;iBACT,CAAC,CAAC;aACJ;SACF;QACD,OAAO,EAAE,IAAI,EAAE,WAAW,EAAE,CAAC;KAC9B;IAEM,UAAU,CAAC,IAAW;QAC3B,MAAM,OAAO,GAAa,EAAE,CAAC;QAC7B,KAAK,MAAM,GAAG,IAAI,IAAI,EAAE;YACtB,OAAO,CAAC,IAAI,CAAC;gBACX,GAAG,EAAE,GAAG,CAAC,GAAG;gBACZ,KAAK,EAAE,GAAG,CAAC,KAAK;aACjB,CAAC,CAAC;SACJ;QACD,OAAO,OAAO,CAAC;KAChB;CACF;AAED;;GAEG;AACH,MAAM,YAAY;IACT,SAAS,CAAC,eAAoB,EAAE,QAAgB;QACrD,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,eAAe,CAAC,EAAE;YACnC,MAAM,IAAI,KAAK,CAAC,4EAA4E,IAAI,CAAC,SAAS,CAAC,eAAe,CAAC,EAAE,CAAC,CAAC;SAChI;QAED,MAAM,IAAI,GAAU,EAAE,CAAC;QACvB,MAAM,WAAW,GAAQ,EAAE,CAAC;QAC5B,KAAK,MAAM,GAAG,IAAI,eAAe,EAAE;YACjC,IAAI,GAAG,CAAC,GAAG,KAAK,SAAS;gBACrB,GAAG,CAAC,KAAK,KAAK,SAAS;gBACvB,GAAG,CAAC,iBAAiB,KAAK,SAAS,EAAE;gBACvC,WAAW,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;aACvB;iBAAM;gBACL,6CAA6C;gBAC7C,IAAI,CAAC,IAAI,CAAC;oBACR,GAAG,EAAE,GAAG,GAAG,CAAC,GAAG,EAAE;oBACjB,KAAK,EAAE,GAAG,GAAG,CAAC,KAAK,EAAE;oBACrB,QAAQ;oBACR,wBAAwB,EAAE,CAAC,CAAC,GAAG,CAAC,iBAAiB;iBAClD,CAAC,CAAC;aACJ;SACF;QAED,OAAO,EAAE,IAAI,EAAE,WAAW,EAAE,CAAC;KAC9B;IAEM,UAAU,CAAC,IAAW;QAC3B,MAAM,OAAO,GAAgB,EAAE,CAAC;QAChC,KAAK,MAAM,GAAG,IAAI,IAAI,EAAE;YACtB,OAAO,CAAC,IAAI,CAAC;gBACX,GAAG,EAAE,GAAG,CAAC,GAAG;gBACZ,KAAK,EAAE,GAAG,CAAC,KAAK;gBAChB,iBAAiB,EAAE,GAAG,CAAC,wBAAwB,KAAK,KAAK;aAC1D,CAAC,CAAC;SACJ;QACD,OAAO,OAAO,CAAC;KAChB;CACF;AAED;;GAEG;AACH,MAAM,YAAY;IACT,SAAS,CAAC,eAAoB,EAAE,QAAgB;QACrD,IAAI,KAAK,CAAC,OAAO,CAAC,eAAe,CAAC,IAAI,OAAM,CAAC,eAAe,CAAC,KAAK,QAAQ,EAAE;YAC1E,MAAM,IAAI,KAAK,CAAC,uDAAuD,IAAI,CAAC,SAAS,CAAC,eAAe,CAAC,EAAE,CAAC,CAAC;SAC3G;QAED,MAAM,IAAI,GAAU,EAAE,CAAC;QACvB,KAAK,MAAM,CAAC,GAAG,EAAE,KAAK,CAAC,IAAI,MAAM,CAAC,OAAO,CAAC,eAAe,CAAC,EAAE;YAC1D,IAAI,CAAC,IAAI,CAAC;gBACR,GAAG;gBACH,KAAK,EAAE,GAAG,KAAK,EAAE;gBACjB,QAAQ;aACT,CAAC,CAAC;SACJ;QAED,OAAO,EAAE,IAAI,EAAE,WAAW,EAAE,SAAS,EAAE,CAAC;KACzC;IAEM,UAAU,CAAC,IAAW;QAC3B,MAAM,OAAO,GAA8B,EAAE,CAAC;QAC9C,KAAK,MAAM,GAAG,IAAI,IAAI,EAAE;YACtB,OAAO,CAAC,GAAG,GAAG,CAAC,GAAG,EAAE,CAAC,GAAG,GAAG,GAAG,CAAC,KAAK,EAAE,CAAC;SACxC;QACD,OAAO,OAAO,CAAC;KAChB;CACF;AAED;;GAEG;AACH,MAAM,iBAAiB;IACd,SAAS,CAAC,YAAiB,EAAE,QAAgB;QAClD,MAAM,IAAI,GAAU,EAAE,CAAC;QACvB,KAAK,MAAM,GAAG,IAAI,YAAY,EAAE;YAC9B,IAAI,YAAY,CAAC,cAAc,CAAC,GAAG,CAAC,EAAE;gBACpC,MAAM,KAAK,GAAG,YAAY,CAAC,GAAG,CAAC,CAAC;gBAChC,IAAI,CAAC,IAAI,CAAC;oBACR,GAAG;oBACH,KAAK;oBACL,QAAQ;iBACT,CAAC,CAAC;aACJ;SACF;QACD,OAAO,EAAE,IAAI,EAAE,WAAW,EAAE,SAAS,EAAE,CAAC;KACzC;IAEM,UAAU,CAAC,eAAsB;QACtC,MAAM,IAAI,GAAe,EAAE,CAAC;QAC5B,eAAe,CAAC,OAAO,CAAC,GAAG,CAAC,EAAE;YAC5B,IAAI,CAAC,IAAI,CAAC;gBACR,GAAG,EAAE,GAAG,CAAC,GAAG;gBACZ,KAAK,EAAE,GAAG,CAAC,KAAK;aACjB,CAAC,CAAC;QACL,CAAC,CAAC,CAAC;QACH,OAAO,IAAI,CAAC;KACb;CACF;AAED,MAAM,QAAQ;IACL,SAAS,CAAC,gBAAqB;QACpC,OAAO,EAAE,IAAI,EAAE,EAAE,EAAE,WAAW,EAAE,SAAS,EAAE,CAAC;KAC7C;IAEM,UAAU,CAAC,KAAY;QAC5B,OAAO,SAAS,CAAC;KAClB;CACF;AAED,IAAI,mBAA+D,CAAC;AAEpE;;;;GAIG;AACH,SAAS,cAAc;IACrB,OAAO,mBAAmB,IAAI,CAAC,mBAAmB,GAAG;QACnD,CAAC,sBAAO,CAAC,iBAAiB,CAAC,EAAE,IAAI,YAAY,EAAE;QAC/C,CAAC,sBAAO,CAAC,QAAQ,CAAC,EAAE,IAAI,iBAAiB,EAAE;QAC3C,CAAC,sBAAO,CAAC,GAAG,CAAC,EAAE,IAAI,YAAY,EAAE;QACjC,CAAC,sBAAO,CAAC,SAAS,CAAC,EAAE,IAAI,iBAAiB,EAAE;QAC5C,CAAC,sBAAO,CAAC,YAAY,CAAC,EAAE,IAAI,QAAQ,EAAE;KACvC,CAAC,CAAC;AACL,CAAC;AA0BD;;;;;;;;;;;;;;;;;;;;;;;;;;;GA2BG;AACH,MAAa,UAAU;IA+BrB,YAAY,OAAgB,EAAE,gBAAwB,EAAE,YAAkB,EAAE,UAA6B,EAAG;QAP3F,SAAI,GAAG,IAAI,GAAG,EAAe,CAAC;QAE9B,eAAU,GAAG,IAAI,GAAG,EAAkB,CAAC;QAGvC,uBAAkB,GAAG,EAAE,CAAC;;;;;;;+CA7B9B,UAAU;;;;QAgCnB,IAAI,CAAC,gBAAgB,GAAG,gBAAgB,CAAC;QACzC,IAAI,CAAC,YAAY,GAAG,cAAc,EAAE,CAAC,OAAO,CAAC,CAAC;QAC9C,IAAI,YAAY,KAAK,SAAS,EAAE;YAC9B,MAAM,eAAe,GAAG,IAAI,CAAC,YAAY,CAAC,SAAS,CAAC,YAAY,EAAE,IAAI,CAAC,kBAAkB,CAAC,CAAC;YAC3F,IAAI,CAAC,WAAW,GAAG,eAAe,CAAC,WAAW,CAAC;YAC/C,IAAI,CAAC,OAAO,CAAC,GAAG,eAAe,CAAC,IAAI,CAAC,CAAC;SACvC;QACD,IAAI,CAAC,eAAe,GAAG,OAAO,CAAC,eAAe,IAAI,MAAM,CAAC;QAEzD,IAAI,CAAC,YAAY,GAAG,WAAI,CAAC,GAAG,CAAC,EAAE,OAAO,EAAE,GAAG,EAAE,CAAC,IAAI,CAAC,UAAU,EAAE,EAAE,CAAC,CAAC;KACpE;IAzCD;;OAEG;IACI,MAAM,CAAC,UAAU,CAAC,SAAc;QACrC,OAAQ,SAAiB,CAAC,IAAI,KAAK,SAAS,CAAC;KAC9C;IAsCD;;;OAGG;IACI,MAAM,CAAC,GAAW,EAAE,KAAa,EAAE,QAAQ,GAAG,CAAC,EAAE,wBAAwB,GAAG,IAAI;QACrF,sGAAsG;QACtG,aAAa;QACb,IAAI,CAAC,OAAO,CAAC,EAAE,GAAG,EAAE,KAAK,EAAE,QAAQ,EAAE,wBAAwB,EAAE,CAAC,CAAC;KAClE;IAED;;;;;OAKG;IACI,SAAS,CAAC,GAAW,EAAE,QAAgB;QAC5C,IAAI,QAAQ,IAAI,CAAC,IAAI,CAAC,UAAU,CAAC,GAAG,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC,EAAE;YAC/C,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC;YACtB,IAAI,CAAC,UAAU,CAAC,GAAG,CAAC,GAAG,EAAE,QAAQ,CAAC,CAAC;SACpC;KACF;IAED;;;;;;;OAOG;IACI,UAAU;QACf,MAAM,aAAa,GAAG,IAAI,CAAC,YAAY,CAAC,UAAU,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC;QACpE,IAAI,KAAK,CAAC,OAAO,CAAC,aAAa,CAAC,IAAI,KAAK,CAAC,OAAO,CAAC,IAAI,CAAC,WAAW,CAAC,EAAE;YACnE,MAAM,GAAG,GAAG,CAAC,GAAG,aAAa,IAAI,EAAE,EAAE,GAAG,IAAI,CAAC,WAAW,IAAI,EAAE,CAAC,CAAC;YAChE,OAAO,GAAG,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,SAAS,CAAC;SACzC;aAAM;YACL,MAAM,GAAG,GAAG,EAAE,GAAG,aAAa,IAAI,EAAE,EAAE,GAAG,IAAI,CAAC,WAAW,IAAI,EAAE,EAAE,CAAC;YAClE,OAAO,MAAM,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,SAAS,CAAC;SACtD;KACF;IAED;;OAEG;IACI,SAAS;QACd,MAAM,GAAG,GAA2B,EAAE,CAAC;QACvC,KAAK,MAAM,GAAG,IAAI,IAAI,CAAC,UAAU,EAAE;YACjC,GAAG,CAAC,GAAG,CAAC,GAAG,CAAC,GAAG,GAAG,CAAC,KAAK,CAAC;SAC1B;QACD,OAAO,GAAG,CAAC;KACZ;IAED;;;;;OAKG;IACI,kBAAkB,CAAC,OAAkB,EAAE,OAAkB;QAC9D,IAAI,OAAO,IAAI,OAAO,CAAC,MAAM,GAAG,CAAC,IAAI,OAAO,CAAC,OAAO,CAAC,IAAI,CAAC,gBAAgB,CAAC,KAAK,CAAC,CAAC,EAAE;YAClF,OAAO,KAAK,CAAC;SACd;QACD,IAAI,OAAO,IAAI,OAAO,CAAC,MAAM,GAAG,CAAC,IAAI,OAAO,CAAC,OAAO,CAAC,IAAI,CAAC,gBAAgB,CAAC,KAAK,CAAC,CAAC,EAAE;YAClF,OAAO,KAAK,CAAC;SACd;QAED,OAAO,IAAI,CAAC;KACb;IAED;;OAEG;IACI,OAAO;QACZ,OAAO,IAAI,CAAC,IAAI,CAAC,IAAI,GAAG,CAAC,CAAC;KAC3B;IAEO,OAAO,CAAC,GAAG,IAAW;QAC5B,KAAK,MAAM,GAAG,IAAI,IAAI,EAAE;YACtB,IAAI,GAAG,CAAC,QAAQ,IAAI,CAAC,IAAI,CAAC,UAAU,CAAC,GAAG,CAAC,GAAG,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC,EAAE;gBACvD,IAAI,CAAC,IAAI,CAAC,GAAG,CAAC,GAAG,CAAC,GAAG,EAAE,GAAG,CAAC,CAAC;gBAC5B,IAAI,CAAC,UAAU,CAAC,GAAG,CAAC,GAAG,CAAC,GAAG,EAAE,GAAG,CAAC,QAAQ,CAAC,CAAC;aAC5C;SACF;KACF;IAED,IAAY,UAAU;QACpB,OAAO,KAAK,CAAC,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,MAAM,EAAE,CAAC;aAClC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,GAAG,CAAC,aAAa,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC;KAC/C;;AArIH,gCAsIC","sourcesContent":["import { TagType } from './cfn-resource';\nimport { CfnTag } from './cfn-tag';\nimport { Lazy } from './lazy';\nimport { IResolvable } from './resolvable';\n\ninterface Tag {\n  key: string;\n  value: string;\n  priority: number;\n\n  /**\n   * @default true\n   */\n  applyToLaunchedInstances?: boolean;\n}\n\ninterface CfnAsgTag {\n  key: string;\n  value: string;\n  propagateAtLaunch: boolean;\n}\n\ninterface StackTag {\n  Key: string;\n  Value: string;\n}\n\n/**\n * The results of parsing Tags.\n */\ninterface ParseTagsResult {\n  /**\n   * The \"simple\" (meaning, not including complex CloudFormation functions)\n   * tags that were found.\n   */\n  readonly tags: Tag[];\n\n  /**\n   * The collection of \"dynamic\" (meaning, including complex CloudFormation functions)\n   * tags that were found.\n   */\n  readonly dynamicTags: any;\n}\n\n/**\n * Interface for converter between CloudFormation and internal tag representations\n */\ninterface ITagFormatter {\n  /**\n   * Format the given tags as CloudFormation tags\n   */\n  formatTags(tags: Tag[]): any;\n\n  /**\n   * Parse the CloudFormation tag representation into internal representation\n   *\n   * Use the given priority.\n   */\n  parseTags(cfnPropertyTags: any, priority: number): ParseTagsResult;\n}\n\n/**\n * Standard tags are a list of { key, value } objects\n */\nclass StandardFormatter implements ITagFormatter {\n  public parseTags(cfnPropertyTags: any, priority: number): ParseTagsResult {\n    if (!Array.isArray(cfnPropertyTags)) {\n      throw new Error(`Invalid tag input expected array of {key, value} have ${JSON.stringify(cfnPropertyTags)}`);\n    }\n\n    const tags: Tag[] = [];\n    const dynamicTags: any = [];\n    for (const tag of cfnPropertyTags) {\n      if (tag.key === undefined || tag.value === undefined) {\n        dynamicTags.push(tag);\n      } else {\n        // using interp to ensure Token is now string\n        tags.push({\n          key: `${tag.key}`,\n          value: `${tag.value}`,\n          priority,\n        });\n      }\n    }\n    return { tags, dynamicTags };\n  }\n\n  public formatTags(tags: Tag[]): any {\n    const cfnTags: CfnTag[] = [];\n    for (const tag of tags) {\n      cfnTags.push({\n        key: tag.key,\n        value: tag.value,\n      });\n    }\n    return cfnTags;\n  }\n}\n\n/**\n * ASG tags are a list of { key, value, propagateAtLaunch } objects\n */\nclass AsgFormatter implements ITagFormatter {\n  public parseTags(cfnPropertyTags: any, priority: number): ParseTagsResult {\n    if (!Array.isArray(cfnPropertyTags)) {\n      throw new Error(`Invalid tag input expected array of {key, value, propagateAtLaunch} have ${JSON.stringify(cfnPropertyTags)}`);\n    }\n\n    const tags: Tag[] = [];\n    const dynamicTags: any = [];\n    for (const tag of cfnPropertyTags) {\n      if (tag.key === undefined ||\n          tag.value === undefined ||\n          tag.propagateAtLaunch === undefined) {\n        dynamicTags.push(tag);\n      } else {\n        // using interp to ensure Token is now string\n        tags.push({\n          key: `${tag.key}`,\n          value: `${tag.value}`,\n          priority,\n          applyToLaunchedInstances: !!tag.propagateAtLaunch,\n        });\n      }\n    }\n\n    return { tags, dynamicTags };\n  }\n\n  public formatTags(tags: Tag[]): any {\n    const cfnTags: CfnAsgTag[] = [];\n    for (const tag of tags) {\n      cfnTags.push({\n        key: tag.key,\n        value: tag.value,\n        propagateAtLaunch: tag.applyToLaunchedInstances !== false,\n      });\n    }\n    return cfnTags;\n  }\n}\n\n/**\n * Some CloudFormation constructs use a { key: value } map for tags\n */\nclass MapFormatter implements ITagFormatter {\n  public parseTags(cfnPropertyTags: any, priority: number): ParseTagsResult {\n    if (Array.isArray(cfnPropertyTags) || typeof(cfnPropertyTags) !== 'object') {\n      throw new Error(`Invalid tag input expected map of {key: value} have ${JSON.stringify(cfnPropertyTags)}`);\n    }\n\n    const tags: Tag[] = [];\n    for (const [key, value] of Object.entries(cfnPropertyTags)) {\n      tags.push({\n        key,\n        value: `${value}`,\n        priority,\n      });\n    }\n\n    return { tags, dynamicTags: undefined };\n  }\n\n  public formatTags(tags: Tag[]): any {\n    const cfnTags: { [key: string]: string } = {};\n    for (const tag of tags) {\n      cfnTags[`${tag.key}`] = `${tag.value}`;\n    }\n    return cfnTags;\n  }\n}\n\n/**\n * StackTags are of the format { Key: key, Value: value }\n */\nclass KeyValueFormatter implements ITagFormatter {\n  public parseTags(keyValueTags: any, priority: number): ParseTagsResult {\n    const tags: Tag[] = [];\n    for (const key in keyValueTags) {\n      if (keyValueTags.hasOwnProperty(key)) {\n        const value = keyValueTags[key];\n        tags.push({\n          key,\n          value,\n          priority,\n        });\n      }\n    }\n    return { tags, dynamicTags: undefined };\n  }\n\n  public formatTags(unformattedTags: Tag[]): any {\n    const tags: StackTag[] = [];\n    unformattedTags.forEach(tag => {\n      tags.push({\n        Key: tag.key,\n        Value: tag.value,\n      });\n    });\n    return tags;\n  }\n}\n\nclass NoFormat implements ITagFormatter {\n  public parseTags(_cfnPropertyTags: any): ParseTagsResult {\n    return { tags: [], dynamicTags: undefined };\n  }\n\n  public formatTags(_tags: Tag[]): any {\n    return undefined;\n  }\n}\n\nlet _tagFormattersCache: {[key: string]: ITagFormatter} | undefined;\n\n/**\n * Access tag formatters table\n *\n * In a function because we're in a load cycle with cfn-resource that defines `TagType`.\n */\nfunction TAG_FORMATTERS(): {[key: string]: ITagFormatter} {\n  return _tagFormattersCache ?? (_tagFormattersCache = {\n    [TagType.AUTOSCALING_GROUP]: new AsgFormatter(),\n    [TagType.STANDARD]: new StandardFormatter(),\n    [TagType.MAP]: new MapFormatter(),\n    [TagType.KEY_VALUE]: new KeyValueFormatter(),\n    [TagType.NOT_TAGGABLE]: new NoFormat(),\n  });\n}\n\n/**\n * Interface to implement tags.\n */\nexport interface ITaggable {\n  /**\n   * TagManager to set, remove and format tags\n   */\n  readonly tags: TagManager;\n}\n\n/**\n * Options to configure TagManager behavior\n */\nexport interface TagManagerOptions {\n  /**\n   * The name of the property in CloudFormation for these tags\n   *\n   * Normally this is `tags`, but Cognito UserPool uses UserPoolTags\n   *\n   * @default \"tags\"\n   */\n  readonly tagPropertyName?: string;\n}\n\n/**\n * TagManager facilitates a common implementation of tagging for Constructs\n *\n * Normally, you do not need to use this class, as the CloudFormation specification\n * will indicate which resources are taggable. However, sometimes you will need this\n * to make custom resources taggable. Used `tagManager.renderedTags` to obtain a\n * value that will resolve to the tags at synthesis time.\n *\n * @example\n * import * as cdk from '@aws-cdk/core';\n *\n * class MyConstruct extends cdk.Resource implements cdk.ITaggable {\n *   public readonly tags = new cdk.TagManager(cdk.TagType.KEY_VALUE, 'Whatever::The::Type');\n *\n *   constructor(scope: cdk.Construct, id: string) {\n *     super(scope, id);\n *\n *     new cdk.CfnResource(this, 'Resource', {\n *       type: 'Whatever::The::Type',\n *       properties: {\n *         // ...\n *         Tags: this.tags.renderedTags,\n *       },\n *     });\n *   }\n * }\n *\n */\nexport class TagManager {\n  /**\n   * Check whether the given construct is Taggable\n   */\n  public static isTaggable(construct: any): construct is ITaggable {\n    return (construct as any).tags !== undefined;\n  }\n\n  /**\n   * The property name for tag values\n   *\n   * Normally this is `tags` but some resources choose a different name. Cognito\n   * UserPool uses UserPoolTags\n   */\n  public readonly tagPropertyName: string;\n\n  /**\n   * A lazy value that represents the rendered tags at synthesis time\n   *\n   * If you need to make a custom construct taggable, use the value of this\n   * property to pass to the `tags` property of the underlying construct.\n   */\n  public readonly renderedTags: IResolvable;\n\n  private readonly tags = new Map<string, Tag>();\n  private readonly dynamicTags: any;\n  private readonly priorities = new Map<string, number>();\n  private readonly tagFormatter: ITagFormatter;\n  private readonly resourceTypeName: string;\n  private readonly initialTagPriority = 50;\n\n  constructor(tagType: TagType, resourceTypeName: string, tagStructure?: any, options: TagManagerOptions = { }) {\n    this.resourceTypeName = resourceTypeName;\n    this.tagFormatter = TAG_FORMATTERS()[tagType];\n    if (tagStructure !== undefined) {\n      const parseTagsResult = this.tagFormatter.parseTags(tagStructure, this.initialTagPriority);\n      this.dynamicTags = parseTagsResult.dynamicTags;\n      this._setTag(...parseTagsResult.tags);\n    }\n    this.tagPropertyName = options.tagPropertyName || 'tags';\n\n    this.renderedTags = Lazy.any({ produce: () => this.renderTags() });\n  }\n\n  /**\n   * Adds the specified tag to the array of tags\n   *\n   */\n  public setTag(key: string, value: string, priority = 0, applyToLaunchedInstances = true): void {\n    // This method mostly exists because we don't want to expose the 'Tag' type used (it will be confusing\n    // to users).\n    this._setTag({ key, value, priority, applyToLaunchedInstances });\n  }\n\n  /**\n   * Removes the specified tag from the array if it exists\n   *\n   * @param key The tag to remove\n   * @param priority The priority of the remove operation\n   */\n  public removeTag(key: string, priority: number): void {\n    if (priority >= (this.priorities.get(key) || 0)) {\n      this.tags.delete(key);\n      this.priorities.set(key, priority);\n    }\n  }\n\n  /**\n   * Renders tags into the proper format based on TagType\n   *\n   * This method will eagerly render the tags currently applied. In\n   * most cases, you should be using `tagManager.renderedTags` instead,\n   * which will return a `Lazy` value that will resolve to the correct\n   * tags at synthesis time.\n   */\n  public renderTags(): any {\n    const formattedTags = this.tagFormatter.formatTags(this.sortedTags);\n    if (Array.isArray(formattedTags) || Array.isArray(this.dynamicTags)) {\n      const ret = [...formattedTags ?? [], ...this.dynamicTags ?? []];\n      return ret.length > 0 ? ret : undefined;\n    } else {\n      const ret = { ...formattedTags ?? {}, ...this.dynamicTags ?? {} };\n      return Object.keys(ret).length > 0 ? ret : undefined;\n    }\n  }\n\n  /**\n   * Render the tags in a readable format\n   */\n  public tagValues(): Record<string, string> {\n    const ret: Record<string, string> = {};\n    for (const tag of this.sortedTags) {\n      ret[tag.key] = tag.value;\n    }\n    return ret;\n  }\n\n  /**\n   * Determine if the aspect applies here\n   *\n   * Looks at the include and exclude resourceTypeName arrays to determine if\n   * the aspect applies here\n   */\n  public applyTagAspectHere(include?: string[], exclude?: string[]) {\n    if (exclude && exclude.length > 0 && exclude.indexOf(this.resourceTypeName) !== -1) {\n      return false;\n    }\n    if (include && include.length > 0 && include.indexOf(this.resourceTypeName) === -1) {\n      return false;\n    }\n\n    return true;\n  }\n\n  /**\n   * Returns true if there are any tags defined\n   */\n  public hasTags(): boolean {\n    return this.tags.size > 0;\n  }\n\n  private _setTag(...tags: Tag[]) {\n    for (const tag of tags) {\n      if (tag.priority >= (this.priorities.get(tag.key) || 0)) {\n        this.tags.set(tag.key, tag);\n        this.priorities.set(tag.key, tag.priority);\n      }\n    }\n  }\n\n  private get sortedTags(): Tag[] {\n    return Array.from(this.tags.values())\n      .sort((a, b) => a.key.localeCompare(b.key));\n  }\n}\n"]} |
\ | No newline at end of file |