1 | ;
|
2 | var _a, _b;
|
3 | Object.defineProperty(exports, "__esModule", { value: true });
|
4 | exports.EventField = exports.RuleTargetInput = void 0;
|
5 | const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
|
6 | const core_1 = require("@aws-cdk/core");
|
7 | /**
|
8 | * The input to send to the event target
|
9 | */
|
10 | class RuleTargetInput {
|
11 | constructor() {
|
12 | }
|
13 | /**
|
14 | * Pass text to the event target
|
15 | *
|
16 | * May contain strings returned by `EventField.from()` to substitute in parts of the
|
17 | * matched event.
|
18 | *
|
19 | * The Rule Target input value will be a single string: the string you pass
|
20 | * here. Do not use this method to pass a complex value like a JSON object to
|
21 | * a Rule Target. Use `RuleTargetInput.fromObject()` instead.
|
22 | */
|
23 | static fromText(text) {
|
24 | return new FieldAwareEventInput(text, InputType.Text);
|
25 | }
|
26 | /**
|
27 | * Pass text to the event target, splitting on newlines.
|
28 | *
|
29 | * This is only useful when passing to a target that does not
|
30 | * take a single argument.
|
31 | *
|
32 | * May contain strings returned by `EventField.from()` to substitute in parts
|
33 | * of the matched event.
|
34 | */
|
35 | static fromMultilineText(text) {
|
36 | return new FieldAwareEventInput(text, InputType.Multiline);
|
37 | }
|
38 | /**
|
39 | * Pass a JSON object to the event target
|
40 | *
|
41 | * May contain strings returned by `EventField.from()` to substitute in parts of the
|
42 | * matched event.
|
43 | */
|
44 | static fromObject(obj) {
|
45 | return new FieldAwareEventInput(obj, InputType.Object);
|
46 | }
|
47 | /**
|
48 | * Take the event target input from a path in the event JSON
|
49 | */
|
50 | static fromEventPath(path) {
|
51 | return new LiteralEventInput({ inputPath: path });
|
52 | }
|
53 | }
|
54 | exports.RuleTargetInput = RuleTargetInput;
|
55 | _a = JSII_RTTI_SYMBOL_1;
|
56 | RuleTargetInput[_a] = { fqn: "@aws-cdk/aws-events.RuleTargetInput", version: "1.204.0" };
|
57 | /**
|
58 | * Event Input that is directly derived from the construct
|
59 | */
|
60 | class LiteralEventInput extends RuleTargetInput {
|
61 | constructor(props) {
|
62 | super();
|
63 | this.props = props;
|
64 | }
|
65 | /**
|
66 | * Return the input properties for this input object
|
67 | */
|
68 | bind(_rule) {
|
69 | return this.props;
|
70 | }
|
71 | }
|
72 | /**
|
73 | * Input object that can contain field replacements
|
74 | *
|
75 | * Evaluation is done in the bind() method because token resolution
|
76 | * requires access to the construct tree.
|
77 | *
|
78 | * Multiple tokens that use the same path will use the same substitution
|
79 | * key.
|
80 | *
|
81 | * One weird exception: if we're in object context, we MUST skip the quotes
|
82 | * around the placeholder. I assume this is so once a trivial string replace is
|
83 | * done later on by EventBridge, numbers are still numbers.
|
84 | *
|
85 | * So in string context:
|
86 | *
|
87 | * "this is a string with a <field>"
|
88 | *
|
89 | * But in object context:
|
90 | *
|
91 | * "{ \"this is the\": <field> }"
|
92 | *
|
93 | * To achieve the latter, we postprocess the JSON string to remove the surrounding
|
94 | * quotes by using a string replace.
|
95 | */
|
96 | class FieldAwareEventInput extends RuleTargetInput {
|
97 | constructor(input, inputType) {
|
98 | super();
|
99 | this.input = input;
|
100 | this.inputType = inputType;
|
101 | }
|
102 | bind(rule) {
|
103 | let fieldCounter = 0;
|
104 | const pathToKey = new Map();
|
105 | const inputPathsMap = {};
|
106 | function keyForField(f) {
|
107 | const existing = pathToKey.get(f.path);
|
108 | if (existing !== undefined) {
|
109 | return existing;
|
110 | }
|
111 | fieldCounter += 1;
|
112 | const key = f.displayHint || `f${fieldCounter}`;
|
113 | pathToKey.set(f.path, key);
|
114 | return key;
|
115 | }
|
116 | class EventFieldReplacer extends core_1.DefaultTokenResolver {
|
117 | constructor() {
|
118 | super(new core_1.StringConcat());
|
119 | }
|
120 | resolveToken(t, _context) {
|
121 | if (!isEventField(t)) {
|
122 | return core_1.Token.asString(t);
|
123 | }
|
124 | const key = keyForField(t);
|
125 | if (inputPathsMap[key] && inputPathsMap[key] !== t.path) {
|
126 | throw new Error(`Single key '${key}' is used for two different JSON paths: '${t.path}' and '${inputPathsMap[key]}'`);
|
127 | }
|
128 | inputPathsMap[key] = t.path;
|
129 | return `<${key}>`;
|
130 | }
|
131 | }
|
132 | const stack = core_1.Stack.of(rule);
|
133 | let resolved;
|
134 | if (this.inputType === InputType.Multiline) {
|
135 | // JSONify individual lines
|
136 | resolved = core_1.Tokenization.resolve(this.input, {
|
137 | scope: rule,
|
138 | resolver: new EventFieldReplacer(),
|
139 | });
|
140 | resolved = resolved.split('\n').map(stack.toJsonString).join('\n');
|
141 | }
|
142 | else {
|
143 | resolved = stack.toJsonString(core_1.Tokenization.resolve(this.input, {
|
144 | scope: rule,
|
145 | resolver: new EventFieldReplacer(),
|
146 | }));
|
147 | }
|
148 | const keys = Object.keys(inputPathsMap);
|
149 | if (keys.length === 0) {
|
150 | // Nothing special, just return 'input'
|
151 | return { input: resolved };
|
152 | }
|
153 | return {
|
154 | inputTemplate: this.unquoteKeyPlaceholders(resolved, keys),
|
155 | inputPathsMap,
|
156 | };
|
157 | }
|
158 | /**
|
159 | * Removing surrounding quotes from any object placeholders
|
160 | * when key is the lone value.
|
161 | *
|
162 | * Those have been put there by JSON.stringify(), but we need to
|
163 | * remove them.
|
164 | *
|
165 | * Do not remove quotes when the key is part of a larger string.
|
166 | *
|
167 | * Valid: { "data": "Some string with \"quotes\"<key>" } // key will be string
|
168 | * Valid: { "data": <key> } // Key could be number, bool, obj, or string
|
169 | */
|
170 | unquoteKeyPlaceholders(sub, keys) {
|
171 | if (this.inputType !== InputType.Object) {
|
172 | return sub;
|
173 | }
|
174 | return core_1.Lazy.uncachedString({ produce: (ctx) => core_1.Token.asString(deepUnquote(ctx.resolve(sub))) });
|
175 | function deepUnquote(resolved) {
|
176 | if (Array.isArray(resolved)) {
|
177 | return resolved.map(deepUnquote);
|
178 | }
|
179 | else if (typeof (resolved) === 'object' && resolved !== null) {
|
180 | for (const [key, value] of Object.entries(resolved)) {
|
181 | resolved[key] = deepUnquote(value);
|
182 | }
|
183 | return resolved;
|
184 | }
|
185 | else if (typeof (resolved) === 'string') {
|
186 | return keys.reduce((r, key) => r.replace(new RegExp(`(?<!\\\\)\"\<${key}\>\"`, 'g'), `<${key}>`), resolved);
|
187 | }
|
188 | return resolved;
|
189 | }
|
190 | }
|
191 | }
|
192 | /**
|
193 | * Represents a field in the event pattern
|
194 | */
|
195 | class EventField {
|
196 | /**
|
197 | *
|
198 | * @param path the path to a field in the event pattern
|
199 | */
|
200 | constructor(path) {
|
201 | this.path = path;
|
202 | this.displayHint = this.path.replace(/^[^a-zA-Z0-9_-]+/, '').replace(/[^a-zA-Z0-9_-]/g, '-');
|
203 | Object.defineProperty(this, EVENT_FIELD_SYMBOL, { value: true });
|
204 | this.creationStack = core_1.captureStackTrace();
|
205 | }
|
206 | /**
|
207 | * Extract the event ID from the event
|
208 | */
|
209 | static get eventId() {
|
210 | return this.fromPath('$.id');
|
211 | }
|
212 | /**
|
213 | * Extract the detail type from the event
|
214 | */
|
215 | static get detailType() {
|
216 | return this.fromPath('$.detail-type');
|
217 | }
|
218 | /**
|
219 | * Extract the source from the event
|
220 | */
|
221 | static get source() {
|
222 | return this.fromPath('$.source');
|
223 | }
|
224 | /**
|
225 | * Extract the account from the event
|
226 | */
|
227 | static get account() {
|
228 | return this.fromPath('$.account');
|
229 | }
|
230 | /**
|
231 | * Extract the time from the event
|
232 | */
|
233 | static get time() {
|
234 | return this.fromPath('$.time');
|
235 | }
|
236 | /**
|
237 | * Extract the region from the event
|
238 | */
|
239 | static get region() {
|
240 | return this.fromPath('$.region');
|
241 | }
|
242 | /**
|
243 | * Extract a custom JSON path from the event
|
244 | */
|
245 | static fromPath(path) {
|
246 | return new EventField(path).toString();
|
247 | }
|
248 | resolve(_ctx) {
|
249 | return this.path;
|
250 | }
|
251 | toString() {
|
252 | return core_1.Token.asString(this, { displayHint: this.displayHint });
|
253 | }
|
254 | /**
|
255 | * Convert the path to the field in the event pattern to JSON
|
256 | */
|
257 | toJSON() {
|
258 | return `<path:${this.path}>`;
|
259 | }
|
260 | }
|
261 | exports.EventField = EventField;
|
262 | _b = JSII_RTTI_SYMBOL_1;
|
263 | EventField[_b] = { fqn: "@aws-cdk/aws-events.EventField", version: "1.204.0" };
|
264 | var InputType;
|
265 | (function (InputType) {
|
266 | InputType[InputType["Object"] = 0] = "Object";
|
267 | InputType[InputType["Text"] = 1] = "Text";
|
268 | InputType[InputType["Multiline"] = 2] = "Multiline";
|
269 | })(InputType || (InputType = {}));
|
270 | function isEventField(x) {
|
271 | return EVENT_FIELD_SYMBOL in x;
|
272 | }
|
273 | const EVENT_FIELD_SYMBOL = Symbol.for('@aws-cdk/aws-events.EventField');
|
274 | //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"input.js","sourceRoot":"","sources":["input.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  captureStackTrace, DefaultTokenResolver, IResolvable,\n  IResolveContext, Lazy, Stack, StringConcat, Token, Tokenization,\n} from '@aws-cdk/core';\nimport { IRule } from './rule-ref';\n\n/**\n * The input to send to the event target\n */\nexport abstract class RuleTargetInput {\n  /**\n   * Pass text to the event target\n   *\n   * May contain strings returned by `EventField.from()` to substitute in parts of the\n   * matched event.\n   *\n   * The Rule Target input value will be a single string: the string you pass\n   * here.  Do not use this method to pass a complex value like a JSON object to\n   * a Rule Target.  Use `RuleTargetInput.fromObject()` instead.\n   */\n  public static fromText(text: string): RuleTargetInput {\n    return new FieldAwareEventInput(text, InputType.Text);\n  }\n\n  /**\n   * Pass text to the event target, splitting on newlines.\n   *\n   * This is only useful when passing to a target that does not\n   * take a single argument.\n   *\n   * May contain strings returned by `EventField.from()` to substitute in parts\n   * of the matched event.\n   */\n  public static fromMultilineText(text: string): RuleTargetInput {\n    return new FieldAwareEventInput(text, InputType.Multiline);\n  }\n\n  /**\n   * Pass a JSON object to the event target\n   *\n   * May contain strings returned by `EventField.from()` to substitute in parts of the\n   * matched event.\n   */\n  public static fromObject(obj: any): RuleTargetInput {\n    return new FieldAwareEventInput(obj, InputType.Object);\n  }\n\n  /**\n   * Take the event target input from a path in the event JSON\n   */\n  public static fromEventPath(path: string): RuleTargetInput {\n    return new LiteralEventInput({ inputPath: path });\n  }\n\n  protected constructor() {\n  }\n\n  /**\n   * Return the input properties for this input object\n   */\n  public abstract bind(rule: IRule): RuleTargetInputProperties;\n}\n\n/**\n * The input properties for an event target\n */\nexport interface RuleTargetInputProperties {\n  /**\n   * Literal input to the target service (must be valid JSON)\n   *\n   * @default - input for the event target. If the input contains a paths map\n   *   values wil be extracted from event and inserted into the `inputTemplate`.\n   */\n  readonly input?: string;\n\n  /**\n   * JsonPath to take input from the input event\n   *\n   * @default - None. The entire matched event is passed as input\n   */\n  readonly inputPath?: string;\n\n  /**\n   * Input template to insert paths map into\n   *\n   * @default - None.\n   */\n  readonly inputTemplate?: string;\n\n  /**\n   * Paths map to extract values from event and insert into `inputTemplate`\n   *\n   * @default - No values extracted from event.\n   */\n  readonly inputPathsMap?: { [key: string]: string };\n}\n\n/**\n * Event Input that is directly derived from the construct\n */\nclass LiteralEventInput extends RuleTargetInput {\n  constructor(private readonly props: RuleTargetInputProperties) {\n    super();\n  }\n\n  /**\n   * Return the input properties for this input object\n   */\n  public bind(_rule: IRule): RuleTargetInputProperties {\n    return this.props;\n  }\n}\n\n/**\n * Input object that can contain field replacements\n *\n * Evaluation is done in the bind() method because token resolution\n * requires access to the construct tree.\n *\n * Multiple tokens that use the same path will use the same substitution\n * key.\n *\n * One weird exception: if we're in object context, we MUST skip the quotes\n * around the placeholder. I assume this is so once a trivial string replace is\n * done later on by EventBridge, numbers are still numbers.\n *\n * So in string context:\n *\n *    \"this is a string with a <field>\"\n *\n * But in object context:\n *\n *    \"{ \\\"this is the\\\": <field> }\"\n *\n * To achieve the latter, we postprocess the JSON string to remove the surrounding\n * quotes by using a string replace.\n */\nclass FieldAwareEventInput extends RuleTargetInput {\n  constructor(private readonly input: any, private readonly inputType: InputType) {\n    super();\n  }\n\n  public bind(rule: IRule): RuleTargetInputProperties {\n    let fieldCounter = 0;\n    const pathToKey = new Map<string, string>();\n    const inputPathsMap: {[key: string]: string} = {};\n\n    function keyForField(f: EventField) {\n      const existing = pathToKey.get(f.path);\n      if (existing !== undefined) { return existing; }\n\n      fieldCounter += 1;\n      const key = f.displayHint || `f${fieldCounter}`;\n      pathToKey.set(f.path, key);\n      return key;\n    }\n\n    class EventFieldReplacer extends DefaultTokenResolver {\n      constructor() {\n        super(new StringConcat());\n      }\n\n      public resolveToken(t: Token, _context: IResolveContext) {\n        if (!isEventField(t)) { return Token.asString(t); }\n\n        const key = keyForField(t);\n        if (inputPathsMap[key] && inputPathsMap[key] !== t.path) {\n          throw new Error(`Single key '${key}' is used for two different JSON paths: '${t.path}' and '${inputPathsMap[key]}'`);\n        }\n        inputPathsMap[key] = t.path;\n\n        return `<${key}>`;\n      }\n    }\n\n    const stack = Stack.of(rule);\n\n    let resolved: string;\n    if (this.inputType === InputType.Multiline) {\n      // JSONify individual lines\n      resolved = Tokenization.resolve(this.input, {\n        scope: rule,\n        resolver: new EventFieldReplacer(),\n      });\n      resolved = resolved.split('\\n').map(stack.toJsonString).join('\\n');\n    } else {\n      resolved = stack.toJsonString(Tokenization.resolve(this.input, {\n        scope: rule,\n        resolver: new EventFieldReplacer(),\n      }));\n    }\n\n    const keys = Object.keys(inputPathsMap);\n\n    if (keys.length === 0) {\n      // Nothing special, just return 'input'\n      return { input: resolved };\n    }\n\n    return {\n      inputTemplate: this.unquoteKeyPlaceholders(resolved, keys),\n      inputPathsMap,\n    };\n  }\n\n  /**\n   * Removing surrounding quotes from any object placeholders\n   * when key is the lone value.\n   *\n   * Those have been put there by JSON.stringify(), but we need to\n   * remove them.\n   *\n   * Do not remove quotes when the key is part of a larger string.\n   *\n   * Valid: { \"data\": \"Some string with \\\"quotes\\\"<key>\" } // key will be string\n   * Valid: { \"data\": <key> } // Key could be number, bool, obj, or string\n   */\n  private unquoteKeyPlaceholders(sub: string, keys: string[]) {\n    if (this.inputType !== InputType.Object) { return sub; }\n\n    return Lazy.uncachedString({ produce: (ctx: IResolveContext) => Token.asString(deepUnquote(ctx.resolve(sub))) });\n\n    function deepUnquote(resolved: any): any {\n      if (Array.isArray(resolved)) {\n        return resolved.map(deepUnquote);\n      } else if (typeof(resolved) === 'object' && resolved !== null) {\n        for (const [key, value] of Object.entries(resolved)) {\n          resolved[key] = deepUnquote(value);\n        }\n        return resolved;\n      } else if (typeof(resolved) === 'string') {\n        return keys.reduce((r, key) => r.replace(new RegExp(`(?<!\\\\\\\\)\\\"\\<${key}\\>\\\"`, 'g'), `<${key}>`), resolved);\n      }\n      return resolved;\n    }\n  }\n}\n\n/**\n * Represents a field in the event pattern\n */\nexport class EventField implements IResolvable {\n  /**\n   * Extract the event ID from the event\n   */\n  public static get eventId(): string {\n    return this.fromPath('$.id');\n  }\n\n  /**\n   * Extract the detail type from the event\n   */\n  public static get detailType(): string {\n    return this.fromPath('$.detail-type');\n  }\n\n  /**\n   * Extract the source from the event\n   */\n  public static get source(): string {\n    return this.fromPath('$.source');\n  }\n\n  /**\n   * Extract the account from the event\n   */\n  public static get account(): string {\n    return this.fromPath('$.account');\n  }\n\n  /**\n   * Extract the time from the event\n   */\n  public static get time(): string {\n    return this.fromPath('$.time');\n  }\n\n  /**\n   * Extract the region from the event\n   */\n  public static get region(): string {\n    return this.fromPath('$.region');\n  }\n\n  /**\n   * Extract a custom JSON path from the event\n   */\n  public static fromPath(path: string): string {\n    return new EventField(path).toString();\n  }\n\n  /**\n   * Human readable display hint about the event pattern\n   */\n  public readonly displayHint: string;\n  public readonly creationStack: string[];\n\n  /**\n   *\n   * @param path the path to a field in the event pattern\n   */\n  private constructor(public readonly path: string) {\n    this.displayHint = this.path.replace(/^[^a-zA-Z0-9_-]+/, '').replace(/[^a-zA-Z0-9_-]/g, '-');\n    Object.defineProperty(this, EVENT_FIELD_SYMBOL, { value: true });\n    this.creationStack = captureStackTrace();\n  }\n\n  public resolve(_ctx: IResolveContext): any {\n    return this.path;\n  }\n\n  public toString() {\n    return Token.asString(this, { displayHint: this.displayHint });\n  }\n\n  /**\n   * Convert the path to the field in the event pattern to JSON\n   */\n  public toJSON() {\n    return `<path:${this.path}>`;\n  }\n}\n\nenum InputType {\n  Object,\n  Text,\n  Multiline,\n}\n\nfunction isEventField(x: any): x is EventField {\n  return EVENT_FIELD_SYMBOL in x;\n}\n\nconst EVENT_FIELD_SYMBOL = Symbol.for('@aws-cdk/aws-events.EventField');\n"]} |
\ | No newline at end of file |