1 | ;
|
2 | var _a, _b;
|
3 | Object.defineProperty(exports, "__esModule", { value: true });
|
4 | exports.unionValidator = exports.requireProperty = exports.requiredValidator = exports.propertyValidator = exports.hashValidator = exports.listValidator = exports.validateCfnTag = exports.validateObject = exports.validateDate = exports.validateBoolean = exports.validateNumber = exports.validateString = exports.canInspect = exports.VALIDATION_SUCCESS = exports.ValidationResults = exports.ValidationResult = exports.unionMapper = exports.hashMapper = exports.listMapper = exports.cfnTagToCloudFormation = exports.dateToCloudFormation = exports.numberToCloudFormation = exports.objectToCloudFormation = exports.booleanToCloudFormation = exports.stringToCloudFormation = void 0;
|
5 | const jsiiDeprecationWarnings = require("../.warnings.jsii.js");
|
6 | const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
|
7 | function identity(x) {
|
8 | return x;
|
9 | }
|
10 | exports.stringToCloudFormation = identity;
|
11 | exports.booleanToCloudFormation = identity;
|
12 | exports.objectToCloudFormation = identity;
|
13 | exports.numberToCloudFormation = identity;
|
14 | /**
|
15 | * The date needs to be formatted as an ISO date in UTC
|
16 | *
|
17 | * Some usage sites require a date, some require a timestamp. We'll
|
18 | * always output a timestamp and hope the parser on the other end
|
19 | * is smart enough to ignore the time part... (?)
|
20 | */
|
21 | function dateToCloudFormation(x) {
|
22 | if (!x) {
|
23 | return undefined;
|
24 | }
|
25 | // eslint-disable-next-line max-len
|
26 | return `${x.getUTCFullYear()}-${pad(x.getUTCMonth() + 1)}-${pad(x.getUTCDate())}T${pad(x.getUTCHours())}:${pad(x.getUTCMinutes())}:${pad(x.getUTCSeconds())}`;
|
27 | }
|
28 | exports.dateToCloudFormation = dateToCloudFormation;
|
29 | /**
|
30 | * Pad a number to 2 decimal places
|
31 | */
|
32 | function pad(x) {
|
33 | if (x < 10) {
|
34 | return '0' + x.toString();
|
35 | }
|
36 | return x.toString();
|
37 | }
|
38 | /**
|
39 | * Turn a tag object into the proper CloudFormation representation
|
40 | */
|
41 | function cfnTagToCloudFormation(x) {
|
42 | return {
|
43 | Key: x.key,
|
44 | Value: x.value,
|
45 | };
|
46 | }
|
47 | exports.cfnTagToCloudFormation = cfnTagToCloudFormation;
|
48 | function listMapper(elementMapper) {
|
49 | return (x) => {
|
50 | if (!canInspect(x)) {
|
51 | return x;
|
52 | }
|
53 | return x.map(elementMapper);
|
54 | };
|
55 | }
|
56 | exports.listMapper = listMapper;
|
57 | function hashMapper(elementMapper) {
|
58 | return (x) => {
|
59 | if (!canInspect(x)) {
|
60 | return x;
|
61 | }
|
62 | const ret = {};
|
63 | Object.keys(x).forEach((key) => {
|
64 | ret[key] = elementMapper(x[key]);
|
65 | });
|
66 | return ret;
|
67 | };
|
68 | }
|
69 | exports.hashMapper = hashMapper;
|
70 | /**
|
71 | * Return a union mapper
|
72 | *
|
73 | * Takes a list of validators and a list of mappers, which should correspond pairwise.
|
74 | *
|
75 | * The mapper of the first successful validator will be called.
|
76 | */
|
77 | function unionMapper(validators, mappers) {
|
78 | if (validators.length !== mappers.length) {
|
79 | throw Error('Not the same amount of validators and mappers passed to unionMapper()');
|
80 | }
|
81 | return (x) => {
|
82 | if (!canInspect(x)) {
|
83 | return x;
|
84 | }
|
85 | for (let i = 0; i < validators.length; i++) {
|
86 | if (validators[i](x).isSuccess) {
|
87 | return mappers[i](x);
|
88 | }
|
89 | }
|
90 | // Should not be possible because the union must have passed validation before this function
|
91 | // will be called, but catch it anyway.
|
92 | throw new TypeError('No validators matched in the union()');
|
93 | };
|
94 | }
|
95 | exports.unionMapper = unionMapper;
|
96 | // ----------------------------------------------------------------------
|
97 | // VALIDATORS
|
98 | //
|
99 | // These are used while checking that supplied property bags match the expected schema
|
100 | //
|
101 | // We have a couple of datatypes that model validation errors and collections of validation
|
102 | // errors (together forming a tree of errors so that we can trace validation errors through
|
103 | // an object graph), and validators.
|
104 | //
|
105 | // Validators are simply functions that take a value and return a validation results. Then
|
106 | // we have some combinators to turn primitive validators into more complex validators.
|
107 | //
|
108 | /**
|
109 | * Representation of validation results
|
110 | *
|
111 | * Models a tree of validation errors so that we have as much information as possible
|
112 | * about the failure that occurred.
|
113 | */
|
114 | class ValidationResult {
|
115 | constructor(errorMessage = '', results = new ValidationResults()) {
|
116 | this.errorMessage = errorMessage;
|
117 | this.results = results;
|
118 | try {
|
119 | jsiiDeprecationWarnings._aws_cdk_core_ValidationResults(results);
|
120 | }
|
121 | catch (error) {
|
122 | if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") {
|
123 | Error.captureStackTrace(error, ValidationResult);
|
124 | }
|
125 | throw error;
|
126 | }
|
127 | }
|
128 | get isSuccess() {
|
129 | return !this.errorMessage && this.results.isSuccess;
|
130 | }
|
131 | /**
|
132 | * Turn a failed validation into an exception
|
133 | */
|
134 | assertSuccess() {
|
135 | if (!this.isSuccess) {
|
136 | let message = this.errorTree();
|
137 | // The first letter will be lowercase, so uppercase it for a nicer error message
|
138 | message = message.slice(0, 1).toUpperCase() + message.slice(1);
|
139 | throw new CfnSynthesisError(message);
|
140 | }
|
141 | }
|
142 | /**
|
143 | * Return a string rendering of the tree of validation failures
|
144 | */
|
145 | errorTree() {
|
146 | const childMessages = this.results.errorTreeList();
|
147 | return this.errorMessage + (childMessages.length ? `\n ${childMessages.replace(/\n/g, '\n ')}` : '');
|
148 | }
|
149 | /**
|
150 | * Wrap this result with an error message, if it concerns an error
|
151 | */
|
152 | prefix(message) {
|
153 | if (this.isSuccess) {
|
154 | return this;
|
155 | }
|
156 | return new ValidationResult(`${message}: ${this.errorMessage}`, this.results);
|
157 | }
|
158 | }
|
159 | exports.ValidationResult = ValidationResult;
|
160 | _a = JSII_RTTI_SYMBOL_1;
|
161 | ValidationResult[_a] = { fqn: "@aws-cdk/core.ValidationResult", version: "1.204.0" };
|
162 | /**
|
163 | * A collection of validation results
|
164 | */
|
165 | class ValidationResults {
|
166 | constructor(results = []) {
|
167 | this.results = results;
|
168 | }
|
169 | collect(result) {
|
170 | try {
|
171 | jsiiDeprecationWarnings._aws_cdk_core_ValidationResult(result);
|
172 | }
|
173 | catch (error) {
|
174 | if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") {
|
175 | Error.captureStackTrace(error, this.collect);
|
176 | }
|
177 | throw error;
|
178 | }
|
179 | // Only collect failures
|
180 | if (!result.isSuccess) {
|
181 | this.results.push(result);
|
182 | }
|
183 | }
|
184 | get isSuccess() {
|
185 | return this.results.every(x => x.isSuccess);
|
186 | }
|
187 | errorTreeList() {
|
188 | return this.results.map(child => child.errorTree()).join('\n');
|
189 | }
|
190 | /**
|
191 | * Wrap up all validation results into a single tree node
|
192 | *
|
193 | * If there are failures in the collection, add a message, otherwise
|
194 | * return a success.
|
195 | */
|
196 | wrap(message) {
|
197 | if (this.isSuccess) {
|
198 | return exports.VALIDATION_SUCCESS;
|
199 | }
|
200 | return new ValidationResult(message, this);
|
201 | }
|
202 | }
|
203 | exports.ValidationResults = ValidationResults;
|
204 | _b = JSII_RTTI_SYMBOL_1;
|
205 | ValidationResults[_b] = { fqn: "@aws-cdk/core.ValidationResults", version: "1.204.0" };
|
206 | // Singleton object to save on allocations
|
207 | exports.VALIDATION_SUCCESS = new ValidationResult();
|
208 | /**
|
209 | * Return whether this object can be validated at all
|
210 | *
|
211 | * True unless it's undefined or a CloudFormation intrinsic
|
212 | */
|
213 | function canInspect(x) {
|
214 | // Note: using weak equality on purpose, we also want to catch undefined
|
215 | return (x != null && !isCloudFormationIntrinsic(x) && !isCloudFormationDynamicReference(x));
|
216 | }
|
217 | exports.canInspect = canInspect;
|
218 | // CloudFormation validators for primitive types
|
219 | function validateString(x) {
|
220 | if (canInspect(x) && typeof x !== 'string') {
|
221 | return new ValidationResult(`${JSON.stringify(x)} should be a string`);
|
222 | }
|
223 | return exports.VALIDATION_SUCCESS;
|
224 | }
|
225 | exports.validateString = validateString;
|
226 | function validateNumber(x) {
|
227 | if (canInspect(x) && typeof x !== 'number') {
|
228 | return new ValidationResult(`${JSON.stringify(x)} should be a number`);
|
229 | }
|
230 | return exports.VALIDATION_SUCCESS;
|
231 | }
|
232 | exports.validateNumber = validateNumber;
|
233 | function validateBoolean(x) {
|
234 | if (canInspect(x) && typeof x !== 'boolean') {
|
235 | return new ValidationResult(`${JSON.stringify(x)} should be a boolean`);
|
236 | }
|
237 | return exports.VALIDATION_SUCCESS;
|
238 | }
|
239 | exports.validateBoolean = validateBoolean;
|
240 | function validateDate(x) {
|
241 | if (canInspect(x) && !(x instanceof Date)) {
|
242 | return new ValidationResult(`${JSON.stringify(x)} should be a Date`);
|
243 | }
|
244 | if (x !== undefined && isNaN(x.getTime())) {
|
245 | return new ValidationResult('got an unparseable Date');
|
246 | }
|
247 | return exports.VALIDATION_SUCCESS;
|
248 | }
|
249 | exports.validateDate = validateDate;
|
250 | function validateObject(x) {
|
251 | if (canInspect(x) && typeof x !== 'object') {
|
252 | return new ValidationResult(`${JSON.stringify(x)} should be an 'object'`);
|
253 | }
|
254 | return exports.VALIDATION_SUCCESS;
|
255 | }
|
256 | exports.validateObject = validateObject;
|
257 | function validateCfnTag(x) {
|
258 | if (!canInspect(x)) {
|
259 | return exports.VALIDATION_SUCCESS;
|
260 | }
|
261 | if (x.key == null || x.value == null) {
|
262 | return new ValidationResult(`${JSON.stringify(x)} should have a 'key' and a 'value' property`);
|
263 | }
|
264 | return exports.VALIDATION_SUCCESS;
|
265 | }
|
266 | exports.validateCfnTag = validateCfnTag;
|
267 | /**
|
268 | * Return a list validator based on the given element validator
|
269 | */
|
270 | function listValidator(elementValidator) {
|
271 | return (x) => {
|
272 | if (!canInspect(x)) {
|
273 | return exports.VALIDATION_SUCCESS;
|
274 | }
|
275 | if (!x.forEach) {
|
276 | return new ValidationResult(`${JSON.stringify(x)} should be a list`);
|
277 | }
|
278 | for (let i = 0; i < x.length; i++) {
|
279 | const element = x[i];
|
280 | const result = elementValidator(element);
|
281 | if (!result.isSuccess) {
|
282 | return result.prefix(`element ${i}`);
|
283 | }
|
284 | }
|
285 | return exports.VALIDATION_SUCCESS;
|
286 | };
|
287 | }
|
288 | exports.listValidator = listValidator;
|
289 | /**
|
290 | * Return a hash validator based on the given element validator
|
291 | */
|
292 | function hashValidator(elementValidator) {
|
293 | return (x) => {
|
294 | if (!canInspect(x)) {
|
295 | return exports.VALIDATION_SUCCESS;
|
296 | }
|
297 | for (const key of Object.keys(x)) {
|
298 | const result = elementValidator(x[key]);
|
299 | if (!result.isSuccess) {
|
300 | return result.prefix(`element '${key}'`);
|
301 | }
|
302 | }
|
303 | return exports.VALIDATION_SUCCESS;
|
304 | };
|
305 | }
|
306 | exports.hashValidator = hashValidator;
|
307 | /**
|
308 | * Decorate a validator with a message clarifying the property the failure is for.
|
309 | */
|
310 | function propertyValidator(propName, validator) {
|
311 | return (x) => {
|
312 | return validator(x).prefix(propName);
|
313 | };
|
314 | }
|
315 | exports.propertyValidator = propertyValidator;
|
316 | /**
|
317 | * Return a validator that will fail if the passed property is not present
|
318 | *
|
319 | * Does not distinguish between the property actually not being present, vs being present but 'null'
|
320 | * or 'undefined' (courtesy of JavaScript), which is generally the behavior that we want.
|
321 | *
|
322 | * Empty strings are considered "present"--don't know if this agrees with how CloudFormation looks
|
323 | * at the world.
|
324 | */
|
325 | function requiredValidator(x) {
|
326 | if (x == null) {
|
327 | return new ValidationResult('required but missing');
|
328 | }
|
329 | return exports.VALIDATION_SUCCESS;
|
330 | }
|
331 | exports.requiredValidator = requiredValidator;
|
332 | /**
|
333 | * Require a property from a property bag.
|
334 | *
|
335 | * @param props the property bag from which a property is required.
|
336 | * @param name the name of the required property.
|
337 | * @param typeName the name of the construct type that requires the property
|
338 | *
|
339 | * @returns the value of ``props[name]``
|
340 | *
|
341 | * @throws if the property ``name`` is not present in ``props``.
|
342 | */
|
343 | function requireProperty(props, name, context) {
|
344 | const value = props[name];
|
345 | if (value == null) {
|
346 | throw new Error(`${context.toString()} is missing required property: ${name}`);
|
347 | }
|
348 | // Possibly add type-checking here...
|
349 | return value;
|
350 | }
|
351 | exports.requireProperty = requireProperty;
|
352 | /**
|
353 | * Validates if any of the given validators matches
|
354 | *
|
355 | * We add either/or words to the front of the error mesages so that they read
|
356 | * more nicely. Example:
|
357 | *
|
358 | * Properties not correct for 'FunctionProps'
|
359 | * codeUri: not one of the possible types
|
360 | * either: properties not correct for 'S3LocationProperty'
|
361 | * bucket: required but missing
|
362 | * key: required but missing
|
363 | * version: required but missing
|
364 | * or: '3' should be a 'string'
|
365 | *
|
366 | */
|
367 | function unionValidator(...validators) {
|
368 | return (x) => {
|
369 | const results = new ValidationResults();
|
370 | let eitherOr = 'either';
|
371 | for (const validator of validators) {
|
372 | const result = validator(x);
|
373 | if (result.isSuccess) {
|
374 | return result;
|
375 | }
|
376 | results.collect(result.prefix(eitherOr));
|
377 | eitherOr = 'or';
|
378 | }
|
379 | return results.wrap('not one of the possible types');
|
380 | };
|
381 | }
|
382 | exports.unionValidator = unionValidator;
|
383 | /**
|
384 | * Return whether the indicated value represents a CloudFormation intrinsic.
|
385 | *
|
386 | * CloudFormation intrinsics are modeled as objects with a single key, which
|
387 | * look like: { "Fn::GetAtt": [...] } or similar.
|
388 | */
|
389 | function isCloudFormationIntrinsic(x) {
|
390 | if (!(typeof x === 'object')) {
|
391 | return false;
|
392 | }
|
393 | const keys = Object.keys(x);
|
394 | if (keys.length !== 1) {
|
395 | return false;
|
396 | }
|
397 | return keys[0] === 'Ref' || keys[0].slice(0, 4) === 'Fn::';
|
398 | }
|
399 | /**
|
400 | * Check whether the indicated value is a CloudFormation dynamic reference.
|
401 | *
|
402 | * CloudFormation dynamic references take the format: '{{resolve:service-name:reference-key}}'
|
403 | */
|
404 | function isCloudFormationDynamicReference(x) {
|
405 | return (typeof x === 'string' && x.startsWith('{{resolve:') && x.endsWith('}}'));
|
406 | }
|
407 | // Cannot be public because JSII gets confused about es5.d.ts
|
408 | class CfnSynthesisError extends Error {
|
409 | constructor() {
|
410 | super(...arguments);
|
411 | this.type = 'CfnSynthesisError';
|
412 | }
|
413 | }
|
414 | //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"runtime.js","sourceRoot":"","sources":["runtime.ts"],"names":[],"mappings":"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","sourcesContent":["import { Construct } from './construct-compat';\n\n// ----------------------------------------------------------------------\n// PROPERTY MAPPERS\n//\n// These are used while converting generated classes/property bags to CloudFormation property objects\n//\n// We use identity mappers for the primitive types. These don't do anything but are there to make the code\n// generation work out nicely (so the code generator doesn't need to emit different code for primitive\n// vs. complex types).\nexport type Mapper = (x: any) => any;\n\nfunction identity(x: any) {\n  return x;\n}\n\nexport const stringToCloudFormation: Mapper = identity;\nexport const booleanToCloudFormation: Mapper = identity;\nexport const objectToCloudFormation: Mapper = identity;\nexport const numberToCloudFormation: Mapper = identity;\n\n/**\n * The date needs to be formatted as an ISO date in UTC\n *\n * Some usage sites require a date, some require a timestamp. We'll\n * always output a timestamp and hope the parser on the other end\n * is smart enough to ignore the time part... (?)\n */\nexport function dateToCloudFormation(x?: Date): any {\n  if (!x) {\n    return undefined;\n  }\n\n  // eslint-disable-next-line max-len\n  return `${x.getUTCFullYear()}-${pad(x.getUTCMonth() + 1)}-${pad(x.getUTCDate())}T${pad(x.getUTCHours())}:${pad(x.getUTCMinutes())}:${pad(x.getUTCSeconds())}`;\n}\n\n/**\n * Pad a number to 2 decimal places\n */\nfunction pad(x: number) {\n  if (x < 10) {\n    return '0' + x.toString();\n  }\n  return x.toString();\n}\n\n/**\n * Turn a tag object into the proper CloudFormation representation\n */\nexport function cfnTagToCloudFormation(x: any): any {\n  return {\n    Key: x.key,\n    Value: x.value,\n  };\n}\n\nexport function listMapper(elementMapper: Mapper): Mapper {\n  return (x: any) => {\n    if (!canInspect(x)) { return x; }\n    return x.map(elementMapper);\n  };\n}\n\nexport function hashMapper(elementMapper: Mapper): Mapper {\n  return (x: any) => {\n    if (!canInspect(x)) { return x; }\n\n    const ret: any = {};\n\n    Object.keys(x).forEach((key) => {\n      ret[key] = elementMapper(x[key]);\n    });\n\n    return ret;\n  };\n}\n\n/**\n * Return a union mapper\n *\n * Takes a list of validators and a list of mappers, which should correspond pairwise.\n *\n * The mapper of the first successful validator will be called.\n */\nexport function unionMapper(validators: Validator[], mappers: Mapper[]): Mapper {\n  if (validators.length !== mappers.length) {\n    throw Error('Not the same amount of validators and mappers passed to unionMapper()');\n  }\n\n  return (x: any) => {\n    if (!canInspect(x)) { return x; }\n\n    for (let i = 0; i < validators.length; i++) {\n      if (validators[i](x).isSuccess) {\n        return mappers[i](x);\n      }\n    }\n\n    // Should not be possible because the union must have passed validation before this function\n    // will be called, but catch it anyway.\n    throw new TypeError('No validators matched in the union()');\n  };\n}\n\n// ----------------------------------------------------------------------\n// VALIDATORS\n//\n// These are used while checking that supplied property bags match the expected schema\n//\n// We have a couple of datatypes that model validation errors and collections of validation\n// errors (together forming a tree of errors so that we can trace validation errors through\n// an object graph), and validators.\n//\n// Validators are simply functions that take a value and return a validation results. Then\n// we have some combinators to turn primitive validators into more complex validators.\n//\n\n/**\n * Representation of validation results\n *\n * Models a tree of validation errors so that we have as much information as possible\n * about the failure that occurred.\n */\nexport class ValidationResult {\n  constructor(readonly errorMessage: string = '', readonly results: ValidationResults = new ValidationResults()) {\n  }\n\n  public get isSuccess(): boolean {\n    return !this.errorMessage && this.results.isSuccess;\n  }\n\n  /**\n   * Turn a failed validation into an exception\n   */\n  public assertSuccess() {\n    if (!this.isSuccess) {\n      let message = this.errorTree();\n      // The first letter will be lowercase, so uppercase it for a nicer error message\n      message = message.slice(0, 1).toUpperCase() + message.slice(1);\n      throw new CfnSynthesisError(message);\n    }\n  }\n\n  /**\n   * Return a string rendering of the tree of validation failures\n   */\n  public errorTree(): string {\n    const childMessages = this.results.errorTreeList();\n    return this.errorMessage + (childMessages.length ? `\\n  ${childMessages.replace(/\\n/g, '\\n  ')}` : '');\n  }\n\n  /**\n   * Wrap this result with an error message, if it concerns an error\n   */\n  public prefix(message: string): ValidationResult {\n    if (this.isSuccess) { return this; }\n    return new ValidationResult(`${message}: ${this.errorMessage}`, this.results);\n  }\n}\n\n/**\n * A collection of validation results\n */\nexport class ValidationResults {\n  constructor(public results: ValidationResult[] = []) {\n  }\n\n  public collect(result: ValidationResult) {\n    // Only collect failures\n    if (!result.isSuccess) {\n      this.results.push(result);\n    }\n  }\n\n  public get isSuccess(): boolean {\n    return this.results.every(x => x.isSuccess);\n  }\n\n  public errorTreeList(): string {\n    return this.results.map(child => child.errorTree()).join('\\n');\n  }\n\n  /**\n   * Wrap up all validation results into a single tree node\n   *\n   * If there are failures in the collection, add a message, otherwise\n   * return a success.\n   */\n  public wrap(message: string): ValidationResult {\n    if (this.isSuccess) { return VALIDATION_SUCCESS; }\n    return new ValidationResult(message, this);\n  }\n}\n\n// Singleton object to save on allocations\nexport const VALIDATION_SUCCESS = new ValidationResult();\n\nexport type Validator = (x: any) => ValidationResult;\n\n/**\n * Return whether this object can be validated at all\n *\n * True unless it's undefined or a CloudFormation intrinsic\n */\nexport function canInspect(x: any) {\n  // Note: using weak equality on purpose, we also want to catch undefined\n  return (x != null && !isCloudFormationIntrinsic(x) && !isCloudFormationDynamicReference(x));\n}\n\n// CloudFormation validators for primitive types\nexport function validateString(x: any): ValidationResult {\n  if (canInspect(x) && typeof x !== 'string') {\n    return new ValidationResult(`${JSON.stringify(x)} should be a string`);\n  }\n  return VALIDATION_SUCCESS;\n}\n\nexport function validateNumber(x: any): ValidationResult {\n  if (canInspect(x) && typeof x !== 'number') {\n    return new ValidationResult(`${JSON.stringify(x)} should be a number`);\n  }\n  return VALIDATION_SUCCESS;\n}\n\nexport function validateBoolean(x: any): ValidationResult {\n  if (canInspect(x) && typeof x !== 'boolean') {\n    return new ValidationResult(`${JSON.stringify(x)} should be a boolean`);\n  }\n  return VALIDATION_SUCCESS;\n}\n\nexport function validateDate(x: any): ValidationResult {\n  if (canInspect(x) && !(x instanceof Date)) {\n    return new ValidationResult(`${JSON.stringify(x)} should be a Date`);\n  }\n\n  if (x !== undefined && isNaN(x.getTime())) {\n    return new ValidationResult('got an unparseable Date');\n  }\n\n  return VALIDATION_SUCCESS;\n}\n\nexport function validateObject(x: any): ValidationResult {\n  if (canInspect(x) && typeof x !== 'object') {\n    return new ValidationResult(`${JSON.stringify(x)} should be an 'object'`);\n  }\n  return VALIDATION_SUCCESS;\n}\n\nexport function validateCfnTag(x: any): ValidationResult {\n  if (!canInspect(x)) { return VALIDATION_SUCCESS; }\n\n  if (x.key == null || x.value == null) {\n    return new ValidationResult(`${JSON.stringify(x)} should have a 'key' and a 'value' property`);\n  }\n\n  return VALIDATION_SUCCESS;\n}\n\n/**\n * Return a list validator based on the given element validator\n */\nexport function listValidator(elementValidator: Validator): Validator {\n  return (x: any) => {\n    if (!canInspect(x)) { return VALIDATION_SUCCESS; }\n\n    if (!x.forEach) {\n      return new ValidationResult(`${JSON.stringify(x)} should be a list`);\n    }\n\n    for (let i = 0; i < x.length; i++) {\n      const element = x[i];\n      const result = elementValidator(element);\n      if (!result.isSuccess) { return result.prefix(`element ${i}`); }\n    }\n\n    return VALIDATION_SUCCESS;\n  };\n}\n\n/**\n * Return a hash validator based on the given element validator\n */\nexport function hashValidator(elementValidator: Validator): Validator {\n  return (x: any) => {\n    if (!canInspect(x)) { return VALIDATION_SUCCESS; }\n\n    for (const key of Object.keys(x)) {\n      const result = elementValidator(x[key]);\n      if (!result.isSuccess) { return result.prefix(`element '${key}'`); }\n    }\n\n    return VALIDATION_SUCCESS;\n  };\n}\n\n/**\n * Decorate a validator with a message clarifying the property the failure is for.\n */\nexport function propertyValidator(propName: string, validator: Validator): Validator {\n  return (x: any) => {\n    return validator(x).prefix(propName);\n  };\n}\n\n/**\n * Return a validator that will fail if the passed property is not present\n *\n * Does not distinguish between the property actually not being present, vs being present but 'null'\n * or 'undefined' (courtesy of JavaScript), which is generally the behavior that we want.\n *\n * Empty strings are considered \"present\"--don't know if this agrees with how CloudFormation looks\n * at the world.\n */\nexport function requiredValidator(x: any) {\n  if (x == null) {\n    return new ValidationResult('required but missing');\n  }\n  return VALIDATION_SUCCESS;\n}\n\n/**\n * Require a property from a property bag.\n *\n * @param props  the property bag from which a property is required.\n * @param name   the name of the required property.\n * @param typeName the name of the construct type that requires the property\n *\n * @returns the value of ``props[name]``\n *\n * @throws if the property ``name`` is not present in ``props``.\n */\nexport function requireProperty(props: { [name: string]: any }, name: string, context: Construct): any {\n  const value = props[name];\n  if (value == null) {\n    throw new Error(`${context.toString()} is missing required property: ${name}`);\n  }\n  // Possibly add type-checking here...\n  return value;\n}\n\n/**\n * Validates if any of the given validators matches\n *\n * We add either/or words to the front of the error mesages so that they read\n * more nicely. Example:\n *\n *   Properties not correct for 'FunctionProps'\n *     codeUri: not one of the possible types\n *       either: properties not correct for 'S3LocationProperty'\n *         bucket: required but missing\n *         key: required but missing\n *         version: required but missing\n *       or: '3' should be a 'string'\n *\n */\nexport function unionValidator(...validators: Validator[]): Validator {\n  return (x: any) => {\n    const results = new ValidationResults();\n    let eitherOr = 'either';\n\n    for (const validator of validators) {\n      const result = validator(x);\n      if (result.isSuccess) { return result; }\n      results.collect(result.prefix(eitherOr));\n      eitherOr = 'or';\n    }\n    return results.wrap('not one of the possible types');\n  };\n}\n\n/**\n * Return whether the indicated value represents a CloudFormation intrinsic.\n *\n * CloudFormation intrinsics are modeled as objects with a single key, which\n * look like: { \"Fn::GetAtt\": [...] } or similar.\n */\nfunction isCloudFormationIntrinsic(x: any) {\n  if (!(typeof x === 'object')) { return false; }\n  const keys = Object.keys(x);\n  if (keys.length !== 1) { return false; }\n\n  return keys[0] === 'Ref' || keys[0].slice(0, 4) === 'Fn::';\n}\n\n/**\n * Check whether the indicated value is a CloudFormation dynamic reference.\n *\n * CloudFormation dynamic references take the format: '{{resolve:service-name:reference-key}}'\n */\nfunction isCloudFormationDynamicReference(x: any) {\n  return (typeof x === 'string' && x.startsWith('{{resolve:') && x.endsWith('}}'));\n}\n\n// Cannot be public because JSII gets confused about es5.d.ts\nclass CfnSynthesisError extends Error {\n  public readonly type = 'CfnSynthesisError';\n}\n"]} |
\ | No newline at end of file |