UNPKG

16 kBJavaScriptView Raw
1"use strict";
2var _a;
3Object.defineProperty(exports, "__esModule", { value: true });
4exports.ContextProvider = void 0;
5const jsiiDeprecationWarnings = require("../.warnings.jsii.js");
6const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
7const cxapi = require("@aws-cdk/cx-api");
8const constructs_1 = require("constructs");
9const annotations_1 = require("./annotations");
10const stack_1 = require("./stack");
11const token_1 = require("./token");
12/**
13 * Base class for the model side of context providers
14 *
15 * Instances of this class communicate with context provider plugins in the 'cdk
16 * toolkit' via context variables (input), outputting specialized queries for
17 * more context variables (output).
18 *
19 * ContextProvider needs access to a Construct to hook into the context mechanism.
20 *
21 */
22class ContextProvider {
23 constructor() { }
24 /**
25 * @returns the context key or undefined if a key cannot be rendered (due to tokens used in any of the props)
26 */
27 static getKey(scope, options) {
28 try {
29 jsiiDeprecationWarnings._aws_cdk_core_GetContextKeyOptions(options);
30 }
31 catch (error) {
32 if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") {
33 Error.captureStackTrace(error, this.getKey);
34 }
35 throw error;
36 }
37 const stack = stack_1.Stack.of(scope);
38 const props = options.includeEnvironment ?? true
39 ? { account: stack.account, region: stack.region, ...options.props }
40 : (options.props ?? {});
41 if (Object.values(props).find(x => token_1.Token.isUnresolved(x))) {
42 throw new Error(`Cannot determine scope for context provider ${options.provider}.\n` +
43 'This usually happens when one or more of the provider props have unresolved tokens');
44 }
45 const propStrings = propsToArray(props);
46 return {
47 key: `${options.provider}:${propStrings.join(':')}`,
48 props,
49 };
50 }
51 static getValue(scope, options) {
52 try {
53 jsiiDeprecationWarnings._aws_cdk_core_GetContextValueOptions(options);
54 }
55 catch (error) {
56 if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") {
57 Error.captureStackTrace(error, this.getValue);
58 }
59 throw error;
60 }
61 const stack = stack_1.Stack.of(scope);
62 if (token_1.Token.isUnresolved(stack.account) || token_1.Token.isUnresolved(stack.region)) {
63 throw new Error(`Cannot retrieve value from context provider ${options.provider} since account/region ` +
64 'are not specified at the stack level. Configure "env" with an account and region when ' +
65 'you define your stack.' +
66 'See https://docs.aws.amazon.com/cdk/latest/guide/environments.html for more details.');
67 }
68 const { key, props } = this.getKey(scope, options);
69 const value = constructs_1.Node.of(scope).tryGetContext(key);
70 const providerError = extractProviderError(value);
71 // if context is missing or an error occurred during context retrieval,
72 // report and return a dummy value.
73 if (value === undefined || providerError !== undefined) {
74 stack.reportMissingContextKey({
75 key,
76 provider: options.provider,
77 props: props,
78 });
79 if (providerError !== undefined) {
80 annotations_1.Annotations.of(scope).addError(providerError);
81 }
82 return { value: options.dummyValue };
83 }
84 return { value };
85 }
86}
87exports.ContextProvider = ContextProvider;
88_a = JSII_RTTI_SYMBOL_1;
89ContextProvider[_a] = { fqn: "@aws-cdk/core.ContextProvider", version: "1.204.0" };
90/**
91 * If the context value represents an error, return the error message
92 */
93function extractProviderError(value) {
94 if (typeof value === 'object' && value !== null) {
95 return value[cxapi.PROVIDER_ERROR_KEY];
96 }
97 return undefined;
98}
99/**
100 * Quote colons in all strings so that we can undo the quoting at a later point
101 *
102 * We'll use $ as a quoting character, for no particularly good reason other
103 * than that \ is going to lead to quoting hell when the keys are stored in JSON.
104 */
105function colonQuote(xs) {
106 return xs.replace('$', '$$').replace(':', '$:');
107}
108function propsToArray(props, keyPrefix = '') {
109 const ret = [];
110 for (const key of Object.keys(props)) {
111 // skip undefined values
112 if (props[key] === undefined) {
113 continue;
114 }
115 switch (typeof props[key]) {
116 case 'object': {
117 ret.push(...propsToArray(props[key], `${keyPrefix}${key}.`));
118 break;
119 }
120 case 'string': {
121 ret.push(`${keyPrefix}${key}=${colonQuote(props[key])}`);
122 break;
123 }
124 default: {
125 ret.push(`${keyPrefix}${key}=${JSON.stringify(props[key])}`);
126 break;
127 }
128 }
129 }
130 ret.sort();
131 return ret;
132}
133//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"context-provider.js","sourceRoot":"","sources":["context-provider.ts"],"names":[],"mappings":"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","sourcesContent":["import * as cxschema from '@aws-cdk/cloud-assembly-schema';\nimport * as cxapi from '@aws-cdk/cx-api';\nimport { Construct, Node } from 'constructs';\nimport { Annotations } from './annotations';\nimport { Stack } from './stack';\nimport { Token } from './token';\n\n/**\n */\nexport interface GetContextKeyOptions {\n  /**\n   * The context provider to query.\n   */\n  readonly provider: string;\n\n  /**\n   * Provider-specific properties.\n   */\n  readonly props?: { [key: string]: any };\n\n  /**\n   * Whether to include the stack's account and region automatically.\n   *\n   * @default true\n   */\n  readonly includeEnvironment?: boolean;\n}\n\n/**\n */\nexport interface GetContextValueOptions extends GetContextKeyOptions {\n  /**\n   * The value to return if the context value was not found and a missing\n   * context is reported. This should be a dummy value that should preferably\n   * fail during deployment since it represents an invalid state.\n   */\n  readonly dummyValue: any;\n}\n\n/**\n */\nexport interface GetContextKeyResult {\n  readonly key: string;\n  readonly props: { [key: string]: any };\n}\n\n/**\n */\nexport interface GetContextValueResult {\n  readonly value?: any;\n}\n\n/**\n * Base class for the model side of context providers\n *\n * Instances of this class communicate with context provider plugins in the 'cdk\n * toolkit' via context variables (input), outputting specialized queries for\n * more context variables (output).\n *\n * ContextProvider needs access to a Construct to hook into the context mechanism.\n *\n */\nexport class ContextProvider {\n  /**\n   * @returns the context key or undefined if a key cannot be rendered (due to tokens used in any of the props)\n   */\n  public static getKey(scope: Construct, options: GetContextKeyOptions): GetContextKeyResult {\n    const stack = Stack.of(scope);\n\n    const props = options.includeEnvironment ?? true\n      ? { account: stack.account, region: stack.region, ...options.props }\n      : (options.props ?? {});\n\n    if (Object.values(props).find(x => Token.isUnresolved(x))) {\n      throw new Error(\n        `Cannot determine scope for context provider ${options.provider}.\\n` +\n        'This usually happens when one or more of the provider props have unresolved tokens');\n    }\n\n    const propStrings = propsToArray(props);\n    return {\n      key: `${options.provider}:${propStrings.join(':')}`,\n      props,\n    };\n  }\n\n  public static getValue(scope: Construct, options: GetContextValueOptions): GetContextValueResult {\n    const stack = Stack.of(scope);\n\n    if (Token.isUnresolved(stack.account) || Token.isUnresolved(stack.region)) {\n      throw new Error(`Cannot retrieve value from context provider ${options.provider} since account/region ` +\n                      'are not specified at the stack level. Configure \"env\" with an account and region when ' +\n                      'you define your stack.' +\n                      'See https://docs.aws.amazon.com/cdk/latest/guide/environments.html for more details.');\n    }\n\n    const { key, props } = this.getKey(scope, options);\n    const value = Node.of(scope).tryGetContext(key);\n    const providerError = extractProviderError(value);\n\n    // if context is missing or an error occurred during context retrieval,\n    // report and return a dummy value.\n    if (value === undefined || providerError !== undefined) {\n      stack.reportMissingContextKey({\n        key,\n        provider: options.provider as cxschema.ContextProvider,\n        props: props as cxschema.ContextQueryProperties,\n      });\n\n      if (providerError !== undefined) {\n        Annotations.of(scope).addError(providerError);\n      }\n\n      return { value: options.dummyValue };\n    }\n\n    return { value };\n  }\n\n  private constructor() { }\n}\n\n/**\n * If the context value represents an error, return the error message\n */\nfunction extractProviderError(value: any): string | undefined {\n  if (typeof value === 'object' && value !== null) {\n    return value[cxapi.PROVIDER_ERROR_KEY];\n  }\n  return undefined;\n}\n\n/**\n * Quote colons in all strings so that we can undo the quoting at a later point\n *\n * We'll use $ as a quoting character, for no particularly good reason other\n * than that \\ is going to lead to quoting hell when the keys are stored in JSON.\n */\nfunction colonQuote(xs: string): string {\n  return xs.replace('$', '$$').replace(':', '$:');\n}\n\nfunction propsToArray(props: {[key: string]: any}, keyPrefix = ''): string[] {\n  const ret: string[] = [];\n\n  for (const key of Object.keys(props)) {\n    // skip undefined values\n    if (props[key] === undefined) {\n      continue;\n    }\n\n    switch (typeof props[key]) {\n      case 'object': {\n        ret.push(...propsToArray(props[key], `${keyPrefix}${key}.`));\n        break;\n      }\n      case 'string': {\n        ret.push(`${keyPrefix}${key}=${colonQuote(props[key])}`);\n        break;\n      }\n      default: {\n        ret.push(`${keyPrefix}${key}=${JSON.stringify(props[key])}`);\n        break;\n      }\n    }\n  }\n\n  ret.sort();\n  return ret;\n}\n"]}
\No newline at end of file