1 | ;
|
2 | var _a, _b, _c;
|
3 | Object.defineProperty(exports, "__esModule", { value: true });
|
4 | exports.withResolved = exports.isResolvableObject = exports.Tokenization = exports.Token = exports.TokenComparison = void 0;
|
5 | const jsiiDeprecationWarnings = require("../.warnings.jsii.js");
|
6 | const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
|
7 | const lazy_1 = require("./lazy");
|
8 | const encoding_1 = require("./private/encoding");
|
9 | const intrinsic_1 = require("./private/intrinsic");
|
10 | const resolve_1 = require("./private/resolve");
|
11 | const token_map_1 = require("./private/token-map");
|
12 | /**
|
13 | * An enum-like class that represents the result of comparing two Tokens.
|
14 | * The return type of {@link Token.compareStrings}.
|
15 | */
|
16 | class TokenComparison {
|
17 | constructor() {
|
18 | }
|
19 | }
|
20 | exports.TokenComparison = TokenComparison;
|
21 | _a = JSII_RTTI_SYMBOL_1;
|
22 | TokenComparison[_a] = { fqn: "@aws-cdk/core.TokenComparison", version: "1.204.0" };
|
23 | /**
|
24 | * This means we're certain the two components are NOT
|
25 | * Tokens, and identical.
|
26 | */
|
27 | TokenComparison.SAME = new TokenComparison();
|
28 | /**
|
29 | * This means we're certain the two components are NOT
|
30 | * Tokens, and different.
|
31 | */
|
32 | TokenComparison.DIFFERENT = new TokenComparison();
|
33 | /** This means exactly one of the components is a Token. */
|
34 | TokenComparison.ONE_UNRESOLVED = new TokenComparison();
|
35 | /** This means both components are Tokens. */
|
36 | TokenComparison.BOTH_UNRESOLVED = new TokenComparison();
|
37 | /**
|
38 | * Represents a special or lazily-evaluated value.
|
39 | *
|
40 | * Can be used to delay evaluation of a certain value in case, for example,
|
41 | * that it requires some context or late-bound data. Can also be used to
|
42 | * mark values that need special processing at document rendering time.
|
43 | *
|
44 | * Tokens can be embedded into strings while retaining their original
|
45 | * semantics.
|
46 | */
|
47 | class Token {
|
48 | constructor() {
|
49 | }
|
50 | /**
|
51 | * Returns true if obj represents an unresolved value
|
52 | *
|
53 | * One of these must be true:
|
54 | *
|
55 | * - `obj` is an IResolvable
|
56 | * - `obj` is a string containing at least one encoded `IResolvable`
|
57 | * - `obj` is either an encoded number or list
|
58 | *
|
59 | * This does NOT recurse into lists or objects to see if they
|
60 | * containing resolvables.
|
61 | *
|
62 | * @param obj The object to test.
|
63 | */
|
64 | static isUnresolved(obj) {
|
65 | return encoding_1.unresolved(obj);
|
66 | }
|
67 | /**
|
68 | * Return a reversible string representation of this token
|
69 | *
|
70 | * If the Token is initialized with a literal, the stringified value of the
|
71 | * literal is returned. Otherwise, a special quoted string representation
|
72 | * of the Token is returned that can be embedded into other strings.
|
73 | *
|
74 | * Strings with quoted Tokens in them can be restored back into
|
75 | * complex values with the Tokens restored by calling `resolve()`
|
76 | * on the string.
|
77 | */
|
78 | static asString(value, options = {}) {
|
79 | try {
|
80 | jsiiDeprecationWarnings._aws_cdk_core_EncodingOptions(options);
|
81 | }
|
82 | catch (error) {
|
83 | if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") {
|
84 | Error.captureStackTrace(error, this.asString);
|
85 | }
|
86 | throw error;
|
87 | }
|
88 | if (typeof value === 'string') {
|
89 | return value;
|
90 | }
|
91 | return token_map_1.TokenMap.instance().registerString(Token.asAny(value), options.displayHint);
|
92 | }
|
93 | /**
|
94 | * Return a reversible number representation of this token
|
95 | */
|
96 | static asNumber(value) {
|
97 | if (typeof value === 'number') {
|
98 | return value;
|
99 | }
|
100 | return token_map_1.TokenMap.instance().registerNumber(Token.asAny(value));
|
101 | }
|
102 | /**
|
103 | * Return a reversible list representation of this token
|
104 | */
|
105 | static asList(value, options = {}) {
|
106 | try {
|
107 | jsiiDeprecationWarnings._aws_cdk_core_EncodingOptions(options);
|
108 | }
|
109 | catch (error) {
|
110 | if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") {
|
111 | Error.captureStackTrace(error, this.asList);
|
112 | }
|
113 | throw error;
|
114 | }
|
115 | if (Array.isArray(value) && value.every(x => typeof x === 'string')) {
|
116 | return value;
|
117 | }
|
118 | return token_map_1.TokenMap.instance().registerList(Token.asAny(value), options.displayHint);
|
119 | }
|
120 | /**
|
121 | * Return a resolvable representation of the given value
|
122 | */
|
123 | static asAny(value) {
|
124 | // First, reverse any encoding that was already done (so we end up with an IResolvable
|
125 | // if it was a token).
|
126 | value = Tokenization.reverse(value) ?? value;
|
127 | // Then, either return the IResolvable we resolved to, or wrap in an Intrinsic
|
128 | return isResolvableObject(value) ? value : new intrinsic_1.Intrinsic(value);
|
129 | }
|
130 | /** Compare two strings that might contain Tokens with each other. */
|
131 | static compareStrings(possibleToken1, possibleToken2) {
|
132 | const firstIsUnresolved = Token.isUnresolved(possibleToken1);
|
133 | const secondIsUnresolved = Token.isUnresolved(possibleToken2);
|
134 | if (firstIsUnresolved && secondIsUnresolved) {
|
135 | return TokenComparison.BOTH_UNRESOLVED;
|
136 | }
|
137 | if (firstIsUnresolved || secondIsUnresolved) {
|
138 | return TokenComparison.ONE_UNRESOLVED;
|
139 | }
|
140 | return possibleToken1 === possibleToken2 ? TokenComparison.SAME : TokenComparison.DIFFERENT;
|
141 | }
|
142 | }
|
143 | exports.Token = Token;
|
144 | _b = JSII_RTTI_SYMBOL_1;
|
145 | Token[_b] = { fqn: "@aws-cdk/core.Token", version: "1.204.0" };
|
146 | /**
|
147 | * Less oft-needed functions to manipulate Tokens
|
148 | */
|
149 | class Tokenization {
|
150 | constructor() {
|
151 | }
|
152 | /**
|
153 | * Un-encode a string potentially containing encoded tokens
|
154 | */
|
155 | static reverseString(s) {
|
156 | return token_map_1.TokenMap.instance().splitString(s);
|
157 | }
|
158 | /**
|
159 | * Un-encode a string which is either a complete encoded token, or doesn't contain tokens at all
|
160 | *
|
161 | * It's illegal for the string to be a concatenation of an encoded token and something else.
|
162 | */
|
163 | static reverseCompleteString(s) {
|
164 | const fragments = Tokenization.reverseString(s);
|
165 | if (fragments.length !== 1) {
|
166 | throw new Error(`Tokenzation.reverseCompleteString: argument must not be a concatentation, got '${s}'`);
|
167 | }
|
168 | return fragments.firstToken;
|
169 | }
|
170 | /**
|
171 | * Un-encode a Tokenized value from a number
|
172 | */
|
173 | static reverseNumber(n) {
|
174 | return token_map_1.TokenMap.instance().lookupNumberToken(n);
|
175 | }
|
176 | /**
|
177 | * Un-encode a Tokenized value from a list
|
178 | */
|
179 | static reverseList(l) {
|
180 | return token_map_1.TokenMap.instance().lookupList(l);
|
181 | }
|
182 | /**
|
183 | * Reverse any value into a Resolvable, if possible
|
184 | *
|
185 | * In case of a string, the string must not be a concatenation.
|
186 | */
|
187 | static reverse(x, options = {}) {
|
188 | try {
|
189 | jsiiDeprecationWarnings._aws_cdk_core_ReverseOptions(options);
|
190 | }
|
191 | catch (error) {
|
192 | if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") {
|
193 | Error.captureStackTrace(error, this.reverse);
|
194 | }
|
195 | throw error;
|
196 | }
|
197 | if (Tokenization.isResolvable(x)) {
|
198 | return x;
|
199 | }
|
200 | if (typeof x === 'string') {
|
201 | if (options.failConcat === false) {
|
202 | // Handle this specially because reverseCompleteString might fail
|
203 | const fragments = Tokenization.reverseString(x);
|
204 | return fragments.length === 1 ? fragments.firstToken : undefined;
|
205 | }
|
206 | return Tokenization.reverseCompleteString(x);
|
207 | }
|
208 | if (Array.isArray(x)) {
|
209 | return Tokenization.reverseList(x);
|
210 | }
|
211 | if (typeof x === 'number') {
|
212 | return Tokenization.reverseNumber(x);
|
213 | }
|
214 | return undefined;
|
215 | }
|
216 | /**
|
217 | * Resolves an object by evaluating all tokens and removing any undefined or empty objects or arrays.
|
218 | * Values can only be primitives, arrays or tokens. Other objects (i.e. with methods) will be rejected.
|
219 | *
|
220 | * @param obj The object to resolve.
|
221 | * @param options Prefix key path components for diagnostics.
|
222 | */
|
223 | static resolve(obj, options) {
|
224 | try {
|
225 | jsiiDeprecationWarnings._aws_cdk_core_ResolveOptions(options);
|
226 | }
|
227 | catch (error) {
|
228 | if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") {
|
229 | Error.captureStackTrace(error, this.resolve);
|
230 | }
|
231 | throw error;
|
232 | }
|
233 | return resolve_1.resolve(obj, {
|
234 | scope: options.scope,
|
235 | resolver: options.resolver,
|
236 | preparing: (options.preparing ?? false),
|
237 | removeEmpty: options.removeEmpty,
|
238 | });
|
239 | }
|
240 | /**
|
241 | * Return whether the given object is an IResolvable object
|
242 | *
|
243 | * This is different from Token.isUnresolved() which will also check for
|
244 | * encoded Tokens, whereas this method will only do a type check on the given
|
245 | * object.
|
246 | */
|
247 | static isResolvable(obj) {
|
248 | return isResolvableObject(obj);
|
249 | }
|
250 | /**
|
251 | * Stringify a number directly or lazily if it's a Token. If it is an object (i.e., { Ref: 'SomeLogicalId' }), return it as-is.
|
252 | */
|
253 | static stringifyNumber(x) {
|
254 | // only convert numbers to strings so that Refs, conditions, and other things don't end up synthesizing as [object object]
|
255 | if (Token.isUnresolved(x)) {
|
256 | return lazy_1.Lazy.uncachedString({
|
257 | produce: context => {
|
258 | const resolved = context.resolve(x);
|
259 | return typeof resolved !== 'number' ? resolved : `${resolved}`;
|
260 | },
|
261 | });
|
262 | }
|
263 | else {
|
264 | return typeof x !== 'number' ? x : `${x}`;
|
265 | }
|
266 | }
|
267 | }
|
268 | exports.Tokenization = Tokenization;
|
269 | _c = JSII_RTTI_SYMBOL_1;
|
270 | Tokenization[_c] = { fqn: "@aws-cdk/core.Tokenization", version: "1.204.0" };
|
271 | function isResolvableObject(x) {
|
272 | return typeof (x) === 'object' && x !== null && typeof x.resolve === 'function';
|
273 | }
|
274 | exports.isResolvableObject = isResolvableObject;
|
275 | function withResolved(...args) {
|
276 | if (args.length < 2) {
|
277 | return;
|
278 | }
|
279 | const argArray = args.slice(0, args.length - 1);
|
280 | if (argArray.some(Token.isUnresolved)) {
|
281 | return;
|
282 | }
|
283 | args[args.length - 1].apply(arguments, argArray);
|
284 | }
|
285 | exports.withResolved = withResolved;
|
286 | //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"token.js","sourceRoot":"","sources":["token.ts"],"names":[],"mappings":"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","sourcesContent":["import { IConstruct } from 'constructs';\nimport { Lazy } from './lazy';\nimport { unresolved } from './private/encoding';\nimport { Intrinsic } from './private/intrinsic';\nimport { resolve } from './private/resolve';\nimport { TokenMap } from './private/token-map';\nimport { IResolvable, ITokenResolver } from './resolvable';\nimport { TokenizedStringFragments } from './string-fragments';\n\n/**\n * An enum-like class that represents the result of comparing two Tokens.\n * The return type of {@link Token.compareStrings}.\n */\nexport class TokenComparison {\n  /**\n   * This means we're certain the two components are NOT\n   * Tokens, and identical.\n   */\n  public static readonly SAME = new TokenComparison();\n\n  /**\n   * This means we're certain the two components are NOT\n   * Tokens, and different.\n   */\n  public static readonly DIFFERENT = new TokenComparison();\n\n  /** This means exactly one of the components is a Token. */\n  public static readonly ONE_UNRESOLVED = new TokenComparison();\n\n  /** This means both components are Tokens. */\n  public static readonly BOTH_UNRESOLVED = new TokenComparison();\n\n  private constructor() {\n  }\n}\n\n/**\n * Represents a special or lazily-evaluated value.\n *\n * Can be used to delay evaluation of a certain value in case, for example,\n * that it requires some context or late-bound data. Can also be used to\n * mark values that need special processing at document rendering time.\n *\n * Tokens can be embedded into strings while retaining their original\n * semantics.\n */\nexport class Token {\n  /**\n   * Returns true if obj represents an unresolved value\n   *\n   * One of these must be true:\n   *\n   * - `obj` is an IResolvable\n   * - `obj` is a string containing at least one encoded `IResolvable`\n   * - `obj` is either an encoded number or list\n   *\n   * This does NOT recurse into lists or objects to see if they\n   * containing resolvables.\n   *\n   * @param obj The object to test.\n   */\n  public static isUnresolved(obj: any): boolean {\n    return unresolved(obj);\n  }\n\n  /**\n   * Return a reversible string representation of this token\n   *\n   * If the Token is initialized with a literal, the stringified value of the\n   * literal is returned. Otherwise, a special quoted string representation\n   * of the Token is returned that can be embedded into other strings.\n   *\n   * Strings with quoted Tokens in them can be restored back into\n   * complex values with the Tokens restored by calling `resolve()`\n   * on the string.\n   */\n  public static asString(value: any, options: EncodingOptions = {}): string {\n    if (typeof value === 'string') { return value; }\n    return TokenMap.instance().registerString(Token.asAny(value), options.displayHint);\n  }\n\n  /**\n   * Return a reversible number representation of this token\n   */\n  public static asNumber(value: any): number {\n    if (typeof value === 'number') { return value; }\n    return TokenMap.instance().registerNumber(Token.asAny(value));\n  }\n\n  /**\n   * Return a reversible list representation of this token\n   */\n  public static asList(value: any, options: EncodingOptions = {}): string[] {\n    if (Array.isArray(value) && value.every(x => typeof x === 'string')) { return value; }\n    return TokenMap.instance().registerList(Token.asAny(value), options.displayHint);\n  }\n\n  /**\n   * Return a resolvable representation of the given value\n   */\n  public static asAny(value: any): IResolvable {\n    // First, reverse any encoding that was already done (so we end up with an IResolvable\n    // if it was a token).\n    value = Tokenization.reverse(value) ?? value;\n    // Then,  either return the IResolvable we resolved to, or wrap in an Intrinsic\n    return isResolvableObject(value) ? value : new Intrinsic(value);\n  }\n\n  /** Compare two strings that might contain Tokens with each other. */\n  public static compareStrings(possibleToken1: string, possibleToken2: string): TokenComparison {\n    const firstIsUnresolved = Token.isUnresolved(possibleToken1);\n    const secondIsUnresolved = Token.isUnresolved(possibleToken2);\n\n    if (firstIsUnresolved && secondIsUnresolved) {\n      return TokenComparison.BOTH_UNRESOLVED;\n    }\n    if (firstIsUnresolved || secondIsUnresolved) {\n      return TokenComparison.ONE_UNRESOLVED;\n    }\n\n    return possibleToken1 === possibleToken2 ? TokenComparison.SAME : TokenComparison.DIFFERENT;\n  }\n\n  private constructor() {\n  }\n}\n\n/**\n * Less oft-needed functions to manipulate Tokens\n */\nexport class Tokenization {\n  /**\n   * Un-encode a string potentially containing encoded tokens\n   */\n  public static reverseString(s: string): TokenizedStringFragments {\n    return TokenMap.instance().splitString(s);\n  }\n\n  /**\n   * Un-encode a string which is either a complete encoded token, or doesn't contain tokens at all\n   *\n   * It's illegal for the string to be a concatenation of an encoded token and something else.\n   */\n  public static reverseCompleteString(s: string): IResolvable | undefined {\n    const fragments = Tokenization.reverseString(s);\n    if (fragments.length !== 1) {\n      throw new Error(`Tokenzation.reverseCompleteString: argument must not be a concatentation, got '${s}'`);\n    }\n    return fragments.firstToken;\n  }\n\n  /**\n   * Un-encode a Tokenized value from a number\n   */\n  public static reverseNumber(n: number): IResolvable | undefined {\n    return TokenMap.instance().lookupNumberToken(n);\n  }\n\n  /**\n   * Un-encode a Tokenized value from a list\n   */\n  public static reverseList(l: string[]): IResolvable | undefined {\n    return TokenMap.instance().lookupList(l);\n  }\n\n  /**\n   * Reverse any value into a Resolvable, if possible\n   *\n   * In case of a string, the string must not be a concatenation.\n   */\n  public static reverse(x: any, options: ReverseOptions = {}): IResolvable | undefined {\n    if (Tokenization.isResolvable(x)) { return x; }\n    if (typeof x === 'string') {\n      if (options.failConcat === false) {\n        // Handle this specially because reverseCompleteString might fail\n        const fragments = Tokenization.reverseString(x);\n        return fragments.length === 1 ? fragments.firstToken : undefined;\n      }\n      return Tokenization.reverseCompleteString(x);\n    }\n    if (Array.isArray(x)) { return Tokenization.reverseList(x); }\n    if (typeof x === 'number') { return Tokenization.reverseNumber(x); }\n    return undefined;\n  }\n\n  /**\n   * Resolves an object by evaluating all tokens and removing any undefined or empty objects or arrays.\n   * Values can only be primitives, arrays or tokens. Other objects (i.e. with methods) will be rejected.\n   *\n   * @param obj The object to resolve.\n   * @param options Prefix key path components for diagnostics.\n   */\n  public static resolve(obj: any, options: ResolveOptions): any {\n    return resolve(obj, {\n      scope: options.scope,\n      resolver: options.resolver,\n      preparing: (options.preparing ?? false),\n      removeEmpty: options.removeEmpty,\n    });\n  }\n\n  /**\n   * Return whether the given object is an IResolvable object\n   *\n   * This is different from Token.isUnresolved() which will also check for\n   * encoded Tokens, whereas this method will only do a type check on the given\n   * object.\n   */\n  public static isResolvable(obj: any): obj is IResolvable {\n    return isResolvableObject(obj);\n  }\n\n  /**\n   * Stringify a number directly or lazily if it's a Token. If it is an object (i.e., { Ref: 'SomeLogicalId' }), return it as-is.\n   */\n  public static stringifyNumber(x: number) {\n    // only convert numbers to strings so that Refs, conditions, and other things don't end up synthesizing as [object object]\n\n    if (Token.isUnresolved(x)) {\n      return Lazy.uncachedString({\n        produce: context => {\n          const resolved = context.resolve(x);\n          return typeof resolved !== 'number' ? resolved : `${resolved}`;\n        },\n      });\n    } else {\n      return typeof x !== 'number' ? x : `${x}`;\n    }\n  }\n\n  private constructor() {\n  }\n}\n\n/**\n * Options for the 'reverse()' operation\n */\nexport interface ReverseOptions {\n  /**\n   * Fail if the given string is a concatenation\n   *\n   * If `false`, just return `undefined`.\n   *\n   * @default true\n   */\n  readonly failConcat?: boolean;\n}\n\n/**\n * Options to the resolve() operation\n *\n * NOT the same as the ResolveContext; ResolveContext is exposed to Token\n * implementors and resolution hooks, whereas this struct is just to bundle\n * a number of things that would otherwise be arguments to resolve() in a\n * readable way.\n */\nexport interface ResolveOptions {\n  /**\n   * The scope from which resolution is performed\n   */\n  readonly scope: IConstruct;\n\n  /**\n   * The resolver to apply to any resolvable tokens found\n   */\n  readonly resolver: ITokenResolver;\n\n  /**\n   * Whether the resolution is being executed during the prepare phase or not.\n   * @default false\n   */\n  readonly preparing?: boolean;\n\n  /**\n   * Whether to remove undefined elements from arrays and objects when resolving.\n   *\n   * @default true\n   */\n  readonly removeEmpty?: boolean;\n}\n\n/**\n * Properties to string encodings\n */\nexport interface EncodingOptions {\n  /**\n   * A hint for the Token's purpose when stringifying it\n   */\n  readonly displayHint?: string;\n}\n\nexport function isResolvableObject(x: any): x is IResolvable {\n  return typeof(x) === 'object' && x !== null && typeof x.resolve === 'function';\n}\n\n/**\n * Call the given function only if all given values are resolved\n *\n * Exported as a function since it will be used by TypeScript modules, but\n * can't be exposed via JSII because of the generics.\n */\nexport function withResolved<A>(a: A, fn: (a: A) => void): void;\nexport function withResolved<A, B>(a: A, b: B, fn: (a: A, b: B) => void): void;\nexport function withResolved<A, B, C>(a: A, b: B, c: C, fn: (a: A, b: B, c: C) => void): void;\nexport function withResolved(...args: any[]) {\n  if (args.length < 2) { return; }\n  const argArray = args.slice(0, args.length - 1);\n  if (argArray.some(Token.isUnresolved)) { return; }\n  args[args.length - 1].apply(arguments, argArray);\n}\n"]} |
\ | No newline at end of file |