1 | ;
|
2 | var _a, _b, _c;
|
3 | Object.defineProperty(exports, "__esModule", { value: true });
|
4 | exports.SpaceDelimitedTextPattern = exports.FilterPattern = exports.JsonPattern = void 0;
|
5 | const jsiiDeprecationWarnings = require("../.warnings.jsii.js");
|
6 | const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
|
7 | /**
|
8 | * Base class for patterns that only match JSON log events.
|
9 | */
|
10 | class JsonPattern {
|
11 | // This is a separate class so we have some type safety where users can't
|
12 | // combine text patterns and JSON patterns with an 'and' operation.
|
13 | constructor(jsonPatternString) {
|
14 | this.jsonPatternString = jsonPatternString;
|
15 | }
|
16 | get logPatternString() {
|
17 | return '{ ' + this.jsonPatternString + ' }';
|
18 | }
|
19 | }
|
20 | exports.JsonPattern = JsonPattern;
|
21 | _a = JSII_RTTI_SYMBOL_1;
|
22 | JsonPattern[_a] = { fqn: "@aws-cdk/aws-logs.JsonPattern", version: "1.156.1" };
|
23 | /**
|
24 | * A collection of static methods to generate appropriate ILogPatterns
|
25 | */
|
26 | class FilterPattern {
|
27 | /**
|
28 | * Use the given string as log pattern.
|
29 | *
|
30 | * See https://docs.aws.amazon.com/AmazonCloudWatch/latest/logs/FilterAndPatternSyntax.html
|
31 | * for information on writing log patterns.
|
32 | *
|
33 | * @param logPatternString The pattern string to use.
|
34 | */
|
35 | static literal(logPatternString) {
|
36 | return new LiteralLogPattern(logPatternString);
|
37 | }
|
38 | /**
|
39 | * A log pattern that matches all events.
|
40 | */
|
41 | static allEvents() {
|
42 | return new LiteralLogPattern('');
|
43 | }
|
44 | /**
|
45 | * A log pattern that matches if all the strings given appear in the event.
|
46 | *
|
47 | * @param terms The words to search for. All terms must match.
|
48 | */
|
49 | static allTerms(...terms) {
|
50 | return new TextLogPattern([terms]);
|
51 | }
|
52 | /**
|
53 | * A log pattern that matches if any of the strings given appear in the event.
|
54 | *
|
55 | * @param terms The words to search for. Any terms must match.
|
56 | */
|
57 | static anyTerm(...terms) {
|
58 | return new TextLogPattern(terms.map(t => [t]));
|
59 | }
|
60 | /**
|
61 | * A log pattern that matches if any of the given term groups matches the event.
|
62 | *
|
63 | * A term group matches an event if all the terms in it appear in the event string.
|
64 | *
|
65 | * @param termGroups A list of term groups to search for. Any one of the clauses must match.
|
66 | */
|
67 | static anyTermGroup(...termGroups) {
|
68 | return new TextLogPattern(termGroups);
|
69 | }
|
70 | /**
|
71 | * A JSON log pattern that compares string values.
|
72 | *
|
73 | * This pattern only matches if the event is a JSON event, and the indicated field inside
|
74 | * compares with the string value.
|
75 | *
|
76 | * Use '$' to indicate the root of the JSON structure. The comparison operator can only
|
77 | * compare equality or inequality. The '*' wildcard may appear in the value may at the
|
78 | * start or at the end.
|
79 | *
|
80 | * For more information, see:
|
81 | *
|
82 | * https://docs.aws.amazon.com/AmazonCloudWatch/latest/logs/FilterAndPatternSyntax.html
|
83 | *
|
84 | * @param jsonField Field inside JSON. Example: "$.myField"
|
85 | * @param comparison Comparison to carry out. Either = or !=.
|
86 | * @param value The string value to compare to. May use '*' as wildcard at start or end of string.
|
87 | */
|
88 | static stringValue(jsonField, comparison, value) {
|
89 | return new JSONStringPattern(jsonField, comparison, value);
|
90 | }
|
91 | /**
|
92 | * A JSON log pattern that compares numerical values.
|
93 | *
|
94 | * This pattern only matches if the event is a JSON event, and the indicated field inside
|
95 | * compares with the value in the indicated way.
|
96 | *
|
97 | * Use '$' to indicate the root of the JSON structure. The comparison operator can only
|
98 | * compare equality or inequality. The '*' wildcard may appear in the value may at the
|
99 | * start or at the end.
|
100 | *
|
101 | * For more information, see:
|
102 | *
|
103 | * https://docs.aws.amazon.com/AmazonCloudWatch/latest/logs/FilterAndPatternSyntax.html
|
104 | *
|
105 | * @param jsonField Field inside JSON. Example: "$.myField"
|
106 | * @param comparison Comparison to carry out. One of =, !=, <, <=, >, >=.
|
107 | * @param value The numerical value to compare to
|
108 | */
|
109 | static numberValue(jsonField, comparison, value) {
|
110 | return new JSONNumberPattern(jsonField, comparison, value);
|
111 | }
|
112 | /**
|
113 | * A JSON log pattern that matches if the field exists and has the special value 'null'.
|
114 | *
|
115 | * @param jsonField Field inside JSON. Example: "$.myField"
|
116 | */
|
117 | static isNull(jsonField) {
|
118 | return new JSONPostfixPattern(jsonField, 'IS NULL');
|
119 | }
|
120 | /**
|
121 | * A JSON log pattern that matches if the field does not exist.
|
122 | *
|
123 | * @param jsonField Field inside JSON. Example: "$.myField"
|
124 | */
|
125 | static notExists(jsonField) {
|
126 | return new JSONPostfixPattern(jsonField, 'NOT EXISTS');
|
127 | }
|
128 | /**
|
129 | * A JSON log patter that matches if the field exists.
|
130 | *
|
131 | * This is a readable convenience wrapper over 'field = *'
|
132 | *
|
133 | * @param jsonField Field inside JSON. Example: "$.myField"
|
134 | */
|
135 | static exists(jsonField) {
|
136 | return new JSONStringPattern(jsonField, '=', '*');
|
137 | }
|
138 | /**
|
139 | * A JSON log pattern that matches if the field exists and equals the boolean value.
|
140 | *
|
141 | * @param jsonField Field inside JSON. Example: "$.myField"
|
142 | * @param value The value to match
|
143 | */
|
144 | static booleanValue(jsonField, value) {
|
145 | return new JSONPostfixPattern(jsonField, value ? 'IS TRUE' : 'IS FALSE');
|
146 | }
|
147 | /**
|
148 | * A JSON log pattern that matches if all given JSON log patterns match
|
149 | */
|
150 | static all(...patterns) {
|
151 | try {
|
152 | jsiiDeprecationWarnings._aws_cdk_aws_logs_JsonPattern(patterns);
|
153 | }
|
154 | catch (error) {
|
155 | if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") {
|
156 | Error.captureStackTrace(error, this.all);
|
157 | }
|
158 | throw error;
|
159 | }
|
160 | if (patterns.length === 0) {
|
161 | throw new Error('Must supply at least one pattern, or use allEvents() to match all events.');
|
162 | }
|
163 | if (patterns.length === 1) {
|
164 | return patterns[0];
|
165 | }
|
166 | return new JSONAggregatePattern('&&', patterns);
|
167 | }
|
168 | /**
|
169 | * A JSON log pattern that matches if any of the given JSON log patterns match
|
170 | */
|
171 | static any(...patterns) {
|
172 | try {
|
173 | jsiiDeprecationWarnings._aws_cdk_aws_logs_JsonPattern(patterns);
|
174 | }
|
175 | catch (error) {
|
176 | if (process.env.JSII_DEBUG !== "1" && error.name === "DeprecationError") {
|
177 | Error.captureStackTrace(error, this.any);
|
178 | }
|
179 | throw error;
|
180 | }
|
181 | if (patterns.length === 0) {
|
182 | throw new Error('Must supply at least one pattern');
|
183 | }
|
184 | if (patterns.length === 1) {
|
185 | return patterns[0];
|
186 | }
|
187 | return new JSONAggregatePattern('||', patterns);
|
188 | }
|
189 | /**
|
190 | * A space delimited log pattern matcher.
|
191 | *
|
192 | * The log event is divided into space-delimited columns (optionally
|
193 | * enclosed by "" or [] to capture spaces into column values), and names
|
194 | * are given to each column.
|
195 | *
|
196 | * '...' may be specified once to match any number of columns.
|
197 | *
|
198 | * Afterwards, conditions may be added to individual columns.
|
199 | *
|
200 | * @param columns The columns in the space-delimited log stream.
|
201 | */
|
202 | static spaceDelimited(...columns) {
|
203 | return SpaceDelimitedTextPattern.construct(columns);
|
204 | }
|
205 | }
|
206 | exports.FilterPattern = FilterPattern;
|
207 | _b = JSII_RTTI_SYMBOL_1;
|
208 | FilterPattern[_b] = { fqn: "@aws-cdk/aws-logs.FilterPattern", version: "1.156.1" };
|
209 | /**
|
210 | * Use a string literal as a log pattern
|
211 | */
|
212 | class LiteralLogPattern {
|
213 | constructor(logPatternString) {
|
214 | this.logPatternString = logPatternString;
|
215 | }
|
216 | }
|
217 | /**
|
218 | * Search for a set of set of terms
|
219 | */
|
220 | class TextLogPattern {
|
221 | constructor(clauses) {
|
222 | const quotedClauses = clauses.map(terms => terms.map(quoteTerm).join(' '));
|
223 | if (quotedClauses.length === 1) {
|
224 | this.logPatternString = quotedClauses[0];
|
225 | }
|
226 | else {
|
227 | this.logPatternString = quotedClauses.map(alt => '?' + alt).join(' ');
|
228 | }
|
229 | }
|
230 | }
|
231 | /**
|
232 | * A string comparison for JSON values
|
233 | */
|
234 | class JSONStringPattern extends JsonPattern {
|
235 | constructor(jsonField, comparison, value) {
|
236 | comparison = validateStringOperator(comparison);
|
237 | super(`${jsonField} ${comparison} ${quoteTerm(value)}`);
|
238 | }
|
239 | }
|
240 | /**
|
241 | * A number comparison for JSON values
|
242 | */
|
243 | class JSONNumberPattern extends JsonPattern {
|
244 | constructor(jsonField, comparison, value) {
|
245 | comparison = validateNumericalOperator(comparison);
|
246 | super(`${jsonField} ${comparison} ${value}`);
|
247 | }
|
248 | }
|
249 | /**
|
250 | * A postfix operator for JSON patterns
|
251 | */
|
252 | class JSONPostfixPattern extends JsonPattern {
|
253 | constructor(jsonField, postfix) {
|
254 | // No validation, we assume these are generated by trusted factory functions
|
255 | super(`${jsonField} ${postfix}`);
|
256 | }
|
257 | }
|
258 | /**
|
259 | * Combines multiple other JSON patterns with an operator
|
260 | */
|
261 | class JSONAggregatePattern extends JsonPattern {
|
262 | constructor(operator, patterns) {
|
263 | if (operator !== '&&' && operator !== '||') {
|
264 | throw new Error('Operator must be one of && or ||');
|
265 | }
|
266 | const clauses = patterns.map(p => '(' + p.jsonPatternString + ')');
|
267 | super(clauses.join(` ${operator} `));
|
268 | }
|
269 | }
|
270 | const COL_ELLIPSIS = '...';
|
271 | /**
|
272 | * Space delimited text pattern
|
273 | */
|
274 | class SpaceDelimitedTextPattern {
|
275 | // TODO: Temporarily changed from private to protected to unblock build. We need to think
|
276 | // about how to handle jsii types with private constructors.
|
277 | constructor(columns, restrictions) {
|
278 | this.columns = columns;
|
279 | this.restrictions = restrictions;
|
280 | }
|
281 | /**
|
282 | * Construct a new instance of a space delimited text pattern
|
283 | *
|
284 | * Since this class must be public, we can't rely on the user only creating it through
|
285 | * the `LogPattern.spaceDelimited()` factory function. We must therefore validate the
|
286 | * argument in the constructor. Since we're returning a copy on every mutation, and we
|
287 | * don't want to re-validate the same things on every construction, we provide a limited
|
288 | * set of mutator functions and only validate the new data every time.
|
289 | */
|
290 | static construct(columns) {
|
291 | // Validation happens here because a user could instantiate this object directly without
|
292 | // going through the factory
|
293 | for (const column of columns) {
|
294 | if (!validColumnName(column)) {
|
295 | throw new Error(`Invalid column name: ${column}`);
|
296 | }
|
297 | }
|
298 | if (sum(columns.map(c => c === COL_ELLIPSIS ? 1 : 0)) > 1) {
|
299 | throw new Error("Can use at most one '...' column");
|
300 | }
|
301 | return new SpaceDelimitedTextPattern(columns, {});
|
302 | }
|
303 | /**
|
304 | * Restrict where the pattern applies
|
305 | */
|
306 | whereString(columnName, comparison, value) {
|
307 | if (columnName === COL_ELLIPSIS) {
|
308 | throw new Error("Can't use '...' in a restriction");
|
309 | }
|
310 | if (this.columns.indexOf(columnName) === -1) {
|
311 | throw new Error(`Column in restrictions that is not in columns: ${columnName}`);
|
312 | }
|
313 | comparison = validateStringOperator(comparison);
|
314 | return new SpaceDelimitedTextPattern(this.columns, this.addRestriction(columnName, {
|
315 | comparison,
|
316 | stringValue: value,
|
317 | }));
|
318 | }
|
319 | /**
|
320 | * Restrict where the pattern applies
|
321 | */
|
322 | whereNumber(columnName, comparison, value) {
|
323 | if (columnName === COL_ELLIPSIS) {
|
324 | throw new Error("Can't use '...' in a restriction");
|
325 | }
|
326 | if (this.columns.indexOf(columnName) === -1) {
|
327 | throw new Error(`Column in restrictions that is not in columns: ${columnName}`);
|
328 | }
|
329 | comparison = validateNumericalOperator(comparison);
|
330 | return new SpaceDelimitedTextPattern(this.columns, this.addRestriction(columnName, {
|
331 | comparison,
|
332 | numberValue: value,
|
333 | }));
|
334 | }
|
335 | get logPatternString() {
|
336 | return '[' + this.columns.map(this.columnExpression.bind(this)).join(', ') + ']';
|
337 | }
|
338 | /**
|
339 | * Return the column expression for the given column
|
340 | */
|
341 | columnExpression(column) {
|
342 | const restrictions = this.restrictions[column];
|
343 | if (!restrictions) {
|
344 | return column;
|
345 | }
|
346 | return restrictions.map(r => renderRestriction(column, r)).join(' && ');
|
347 | }
|
348 | /**
|
349 | * Make a copy of the current restrictions and add one
|
350 | */
|
351 | addRestriction(columnName, restriction) {
|
352 | const ret = {};
|
353 | for (const key of Object.keys(this.restrictions)) {
|
354 | ret[key] = this.restrictions[key].slice();
|
355 | }
|
356 | if (!(columnName in ret)) {
|
357 | ret[columnName] = [];
|
358 | }
|
359 | ret[columnName].push(restriction);
|
360 | return ret;
|
361 | }
|
362 | }
|
363 | exports.SpaceDelimitedTextPattern = SpaceDelimitedTextPattern;
|
364 | _c = JSII_RTTI_SYMBOL_1;
|
365 | SpaceDelimitedTextPattern[_c] = { fqn: "@aws-cdk/aws-logs.SpaceDelimitedTextPattern", version: "1.156.1" };
|
366 | /**
|
367 | * Quote a term for use in a pattern expression
|
368 | *
|
369 | * It's never wrong to quote a string term, and required if the term
|
370 | * contains non-alphanumerical characters, so we just always do it.
|
371 | *
|
372 | * Inner double quotes are escaped using a backslash.
|
373 | */
|
374 | function quoteTerm(term) {
|
375 | return '"' + term.replace(/\\/g, '\\\\').replace(/"/g, '\\"') + '"';
|
376 | }
|
377 | /**
|
378 | * Return whether the given column name is valid in a space-delimited table
|
379 | */
|
380 | function validColumnName(column) {
|
381 | return column === COL_ELLIPSIS || /^[a-zA-Z0-9_-]+$/.exec(column);
|
382 | }
|
383 | /**
|
384 | * Validate and normalize the string comparison operator
|
385 | *
|
386 | * Correct for a common typo/confusion, treat '==' as '='
|
387 | */
|
388 | function validateStringOperator(operator) {
|
389 | if (operator === '==') {
|
390 | operator = '=';
|
391 | }
|
392 | if (operator !== '=' && operator !== '!=') {
|
393 | throw new Error(`Invalid comparison operator ('${operator}'), must be either '=' or '!='`);
|
394 | }
|
395 | return operator;
|
396 | }
|
397 | const VALID_OPERATORS = ['=', '!=', '<', '<=', '>', '>='];
|
398 | /**
|
399 | * Validate and normalize numerical comparison operators
|
400 | *
|
401 | * Correct for a common typo/confusion, treat '==' as '='
|
402 | */
|
403 | function validateNumericalOperator(operator) {
|
404 | // Correct for a common typo, treat '==' as '='
|
405 | if (operator === '==') {
|
406 | operator = '=';
|
407 | }
|
408 | if (VALID_OPERATORS.indexOf(operator) === -1) {
|
409 | throw new Error(`Invalid comparison operator ('${operator}'), must be one of ${VALID_OPERATORS.join(', ')}`);
|
410 | }
|
411 | return operator;
|
412 | }
|
413 | /**
|
414 | * Render a table restriction
|
415 | */
|
416 | function renderRestriction(column, restriction) {
|
417 | if (restriction.numberValue !== undefined) {
|
418 | return `${column} ${restriction.comparison} ${restriction.numberValue}`;
|
419 | }
|
420 | else if (restriction.stringValue) {
|
421 | return `${column} ${restriction.comparison} ${quoteTerm(restriction.stringValue)}`;
|
422 | }
|
423 | else {
|
424 | throw new Error('Invalid restriction');
|
425 | }
|
426 | }
|
427 | function sum(xs) {
|
428 | return xs.reduce((a, c) => a + c, 0);
|
429 | }
|
430 | //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"pattern.js","sourceRoot":"","sources":["pattern.ts"],"names":[],"mappings":"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","sourcesContent":["// Implementation of metric patterns\n\n/**\n * Interface for objects that can render themselves to log patterns.\n */\nexport interface IFilterPattern {\n  readonly logPatternString: string;\n}\n\n/**\n * Base class for patterns that only match JSON log events.\n */\nexport abstract class JsonPattern implements IFilterPattern {\n  // This is a separate class so we have some type safety where users can't\n  // combine text patterns and JSON patterns with an 'and' operation.\n  constructor(public readonly jsonPatternString: string) { }\n\n  public get logPatternString(): string {\n    return '{ ' + this.jsonPatternString + ' }';\n  }\n}\n\n/**\n * A collection of static methods to generate appropriate ILogPatterns\n */\nexport class FilterPattern {\n\n  /**\n   * Use the given string as log pattern.\n   *\n   * See https://docs.aws.amazon.com/AmazonCloudWatch/latest/logs/FilterAndPatternSyntax.html\n   * for information on writing log patterns.\n   *\n   * @param logPatternString The pattern string to use.\n   */\n  public static literal(logPatternString: string): IFilterPattern {\n    return new LiteralLogPattern(logPatternString);\n  }\n\n  /**\n   * A log pattern that matches all events.\n   */\n  public static allEvents(): IFilterPattern {\n    return new LiteralLogPattern('');\n  }\n\n  /**\n   * A log pattern that matches if all the strings given appear in the event.\n   *\n   * @param terms The words to search for. All terms must match.\n   */\n  public static allTerms(...terms: string[]): IFilterPattern {\n    return new TextLogPattern([terms]);\n  }\n\n  /**\n   * A log pattern that matches if any of the strings given appear in the event.\n   *\n   * @param terms The words to search for. Any terms must match.\n   */\n  public static anyTerm(...terms: string[]): IFilterPattern {\n    return new TextLogPattern(terms.map(t => [t]));\n  }\n\n  /**\n   * A log pattern that matches if any of the given term groups matches the event.\n   *\n   * A term group matches an event if all the terms in it appear in the event string.\n   *\n   * @param termGroups A list of term groups to search for. Any one of the clauses must match.\n   */\n  public static anyTermGroup(...termGroups: string[][]): IFilterPattern {\n    return new TextLogPattern(termGroups);\n  }\n\n  /**\n   * A JSON log pattern that compares string values.\n   *\n   * This pattern only matches if the event is a JSON event, and the indicated field inside\n   * compares with the string value.\n   *\n   * Use '$' to indicate the root of the JSON structure. The comparison operator can only\n   * compare equality or inequality. The '*' wildcard may appear in the value may at the\n   * start or at the end.\n   *\n   * For more information, see:\n   *\n   * https://docs.aws.amazon.com/AmazonCloudWatch/latest/logs/FilterAndPatternSyntax.html\n   *\n   * @param jsonField Field inside JSON. Example: \"$.myField\"\n   * @param comparison Comparison to carry out. Either = or !=.\n   * @param value The string value to compare to. May use '*' as wildcard at start or end of string.\n   */\n  public static stringValue(jsonField: string, comparison: string, value: string): JsonPattern {\n    return new JSONStringPattern(jsonField, comparison, value);\n  }\n\n  /**\n   * A JSON log pattern that compares numerical values.\n   *\n   * This pattern only matches if the event is a JSON event, and the indicated field inside\n   * compares with the value in the indicated way.\n   *\n   * Use '$' to indicate the root of the JSON structure. The comparison operator can only\n   * compare equality or inequality. The '*' wildcard may appear in the value may at the\n   * start or at the end.\n   *\n   * For more information, see:\n   *\n   * https://docs.aws.amazon.com/AmazonCloudWatch/latest/logs/FilterAndPatternSyntax.html\n   *\n   * @param jsonField Field inside JSON. Example: \"$.myField\"\n   * @param comparison Comparison to carry out. One of =, !=, <, <=, >, >=.\n   * @param value The numerical value to compare to\n   */\n  public static numberValue(jsonField: string, comparison: string, value: number): JsonPattern {\n    return new JSONNumberPattern(jsonField, comparison, value);\n  }\n\n  /**\n   * A JSON log pattern that matches if the field exists and has the special value 'null'.\n   *\n   * @param jsonField Field inside JSON. Example: \"$.myField\"\n   */\n  public static isNull(jsonField: string): JsonPattern {\n    return new JSONPostfixPattern(jsonField, 'IS NULL');\n  }\n\n  /**\n   * A JSON log pattern that matches if the field does not exist.\n   *\n   * @param jsonField Field inside JSON. Example: \"$.myField\"\n   */\n  public static notExists(jsonField: string): JsonPattern {\n    return new JSONPostfixPattern(jsonField, 'NOT EXISTS');\n  }\n\n  /**\n   * A JSON log patter that matches if the field exists.\n   *\n   * This is a readable convenience wrapper over 'field = *'\n   *\n   * @param jsonField Field inside JSON. Example: \"$.myField\"\n   */\n  public static exists(jsonField: string): JsonPattern {\n    return new JSONStringPattern(jsonField, '=', '*');\n  }\n\n  /**\n   * A JSON log pattern that matches if the field exists and equals the boolean value.\n   *\n   * @param jsonField Field inside JSON. Example: \"$.myField\"\n   * @param value The value to match\n   */\n  public static booleanValue(jsonField: string, value: boolean): JsonPattern {\n    return new JSONPostfixPattern(jsonField, value ? 'IS TRUE' : 'IS FALSE');\n  }\n\n  /**\n   * A JSON log pattern that matches if all given JSON log patterns match\n   */\n  public static all(...patterns: JsonPattern[]): JsonPattern {\n    if (patterns.length === 0) { throw new Error('Must supply at least one pattern, or use allEvents() to match all events.'); }\n    if (patterns.length === 1) { return patterns[0]; }\n    return new JSONAggregatePattern('&&', patterns);\n  }\n\n  /**\n   * A JSON log pattern that matches if any of the given JSON log patterns match\n   */\n  public static any(...patterns: JsonPattern[]): JsonPattern {\n    if (patterns.length === 0) { throw new Error('Must supply at least one pattern'); }\n    if (patterns.length === 1) { return patterns[0]; }\n    return new JSONAggregatePattern('||', patterns);\n  }\n\n  /**\n   * A space delimited log pattern matcher.\n   *\n   * The log event is divided into space-delimited columns (optionally\n   * enclosed by \"\" or [] to capture spaces into column values), and names\n   * are given to each column.\n   *\n   * '...' may be specified once to match any number of columns.\n   *\n   * Afterwards, conditions may be added to individual columns.\n   *\n   * @param columns The columns in the space-delimited log stream.\n   */\n  public static spaceDelimited(...columns: string[]): SpaceDelimitedTextPattern {\n    return SpaceDelimitedTextPattern.construct(columns);\n  }\n}\n\n/**\n * Use a string literal as a log pattern\n */\nclass LiteralLogPattern implements IFilterPattern {\n  constructor(public readonly logPatternString: string) {\n  }\n}\n\n/**\n * Search for a set of set of terms\n */\nclass TextLogPattern implements IFilterPattern {\n  public readonly logPatternString: string;\n\n  constructor(clauses: string[][]) {\n    const quotedClauses = clauses.map(terms => terms.map(quoteTerm).join(' '));\n    if (quotedClauses.length === 1) {\n      this.logPatternString = quotedClauses[0];\n    } else {\n      this.logPatternString = quotedClauses.map(alt => '?' + alt).join(' ');\n    }\n  }\n}\n\n/**\n * A string comparison for JSON values\n */\nclass JSONStringPattern extends JsonPattern {\n  public constructor(jsonField: string, comparison: string, value: string) {\n    comparison = validateStringOperator(comparison);\n    super(`${jsonField} ${comparison} ${quoteTerm(value)}`);\n  }\n}\n\n/**\n * A number comparison for JSON values\n */\nclass JSONNumberPattern extends JsonPattern {\n  public constructor(jsonField: string, comparison: string, value: number) {\n    comparison = validateNumericalOperator(comparison);\n    super(`${jsonField} ${comparison} ${value}`);\n  }\n}\n\n/**\n * A postfix operator for JSON patterns\n */\nclass JSONPostfixPattern extends JsonPattern {\n  public constructor(jsonField: string, postfix: string) {\n    // No validation, we assume these are generated by trusted factory functions\n    super(`${jsonField} ${postfix}`);\n  }\n}\n\n/**\n * Combines multiple other JSON patterns with an operator\n */\nclass JSONAggregatePattern extends JsonPattern {\n  public constructor(operator: string, patterns: JsonPattern[]) {\n    if (operator !== '&&' && operator !== '||') {\n      throw new Error('Operator must be one of && or ||');\n    }\n\n    const clauses = patterns.map(p => '(' + p.jsonPatternString + ')');\n\n    super(clauses.join(` ${operator} `));\n  }\n}\n\nexport type RestrictionMap = {[column: string]: ColumnRestriction[]};\n\nconst COL_ELLIPSIS = '...';\n\n/**\n * Space delimited text pattern\n */\nexport class SpaceDelimitedTextPattern implements IFilterPattern {\n  /**\n   * Construct a new instance of a space delimited text pattern\n   *\n   * Since this class must be public, we can't rely on the user only creating it through\n   * the `LogPattern.spaceDelimited()` factory function. We must therefore validate the\n   * argument in the constructor. Since we're returning a copy on every mutation, and we\n   * don't want to re-validate the same things on every construction, we provide a limited\n   * set of mutator functions and only validate the new data every time.\n   */\n  public static construct(columns: string[]) {\n    // Validation happens here because a user could instantiate this object directly without\n    // going through the factory\n    for (const column of columns) {\n      if (!validColumnName(column)) {\n        throw new Error(`Invalid column name: ${column}`);\n      }\n    }\n\n    if (sum(columns.map(c => c === COL_ELLIPSIS ? 1 : 0)) > 1) {\n      throw new Error(\"Can use at most one '...' column\");\n    }\n\n    return new SpaceDelimitedTextPattern(columns, {});\n  }\n\n  // TODO: Temporarily changed from private to protected to unblock build. We need to think\n  //     about how to handle jsii types with private constructors.\n  protected constructor(private readonly columns: string[], private readonly restrictions: RestrictionMap) {\n    // Private constructor so we validate in the .construct() factory function\n  }\n\n  /**\n   * Restrict where the pattern applies\n   */\n  public whereString(columnName: string, comparison: string, value: string): SpaceDelimitedTextPattern {\n    if (columnName === COL_ELLIPSIS) {\n      throw new Error(\"Can't use '...' in a restriction\");\n    }\n    if (this.columns.indexOf(columnName) === -1) {\n      throw new Error(`Column in restrictions that is not in columns: ${columnName}`);\n    }\n\n    comparison = validateStringOperator(comparison);\n\n    return new SpaceDelimitedTextPattern(this.columns, this.addRestriction(columnName, {\n      comparison,\n      stringValue: value,\n    }));\n  }\n\n  /**\n   * Restrict where the pattern applies\n   */\n  public whereNumber(columnName: string, comparison: string, value: number): SpaceDelimitedTextPattern {\n    if (columnName === COL_ELLIPSIS) {\n      throw new Error(\"Can't use '...' in a restriction\");\n    }\n    if (this.columns.indexOf(columnName) === -1) {\n      throw new Error(`Column in restrictions that is not in columns: ${columnName}`);\n    }\n\n    comparison = validateNumericalOperator(comparison);\n\n    return new SpaceDelimitedTextPattern(this.columns, this.addRestriction(columnName, {\n      comparison,\n      numberValue: value,\n    }));\n  }\n\n  public get logPatternString(): string {\n    return '[' + this.columns.map(this.columnExpression.bind(this)).join(', ') + ']';\n  }\n\n  /**\n   * Return the column expression for the given column\n   */\n  private columnExpression(column: string) {\n    const restrictions = this.restrictions[column];\n    if (!restrictions) { return column; }\n\n    return restrictions.map(r => renderRestriction(column, r)).join(' && ');\n  }\n\n  /**\n   * Make a copy of the current restrictions and add one\n   */\n  private addRestriction(columnName: string, restriction: ColumnRestriction) {\n    const ret: RestrictionMap = {};\n    for (const key of Object.keys(this.restrictions)) {\n      ret[key] = this.restrictions[key].slice();\n    }\n    if (!(columnName in ret)) { ret[columnName] = []; }\n    ret[columnName].push(restriction);\n    return ret;\n  }\n}\n\nexport interface ColumnRestriction {\n  /**\n   * Comparison operator to use\n   */\n  readonly comparison: string;\n\n  /**\n   * String value to compare to\n   *\n   * Exactly one of 'stringValue' and 'numberValue' must be set.\n   */\n  readonly stringValue?: string;\n\n  /**\n   * Number value to compare to\n   *\n   * Exactly one of 'stringValue' and 'numberValue' must be set.\n   */\n  readonly numberValue?: number;\n}\n\n/**\n * Quote a term for use in a pattern expression\n *\n * It's never wrong to quote a string term, and required if the term\n * contains non-alphanumerical characters, so we just always do it.\n *\n * Inner double quotes are escaped using a backslash.\n */\nfunction quoteTerm(term: string): string {\n  return '\"' + term.replace(/\\\\/g, '\\\\\\\\').replace(/\"/g, '\\\\\"') + '\"';\n}\n\n/**\n * Return whether the given column name is valid in a space-delimited table\n */\nfunction validColumnName(column: string) {\n  return column === COL_ELLIPSIS || /^[a-zA-Z0-9_-]+$/.exec(column);\n}\n\n/**\n * Validate and normalize the string comparison operator\n *\n * Correct for a common typo/confusion, treat '==' as '='\n */\nfunction validateStringOperator(operator: string) {\n  if (operator === '==') { operator = '='; }\n\n  if (operator !== '=' && operator !== '!=') {\n    throw new Error(`Invalid comparison operator ('${operator}'), must be either '=' or '!='`);\n  }\n\n  return operator;\n}\n\nconst VALID_OPERATORS = ['=', '!=', '<', '<=', '>', '>='];\n\n/**\n * Validate and normalize numerical comparison operators\n *\n * Correct for a common typo/confusion, treat '==' as '='\n */\nfunction validateNumericalOperator(operator: string) {\n  // Correct for a common typo, treat '==' as '='\n  if (operator === '==') { operator = '='; }\n\n  if (VALID_OPERATORS.indexOf(operator) === -1) {\n    throw new Error(`Invalid comparison operator ('${operator}'), must be one of ${VALID_OPERATORS.join(', ')}`);\n  }\n\n  return operator;\n}\n\n/**\n * Render a table restriction\n */\nfunction renderRestriction(column: string, restriction: ColumnRestriction) {\n  if (restriction.numberValue !== undefined) {\n    return `${column} ${restriction.comparison} ${restriction.numberValue}`;\n  } else if (restriction.stringValue) {\n    return `${column} ${restriction.comparison} ${quoteTerm(restriction.stringValue)}`;\n  } else {\n    throw new Error('Invalid restriction');\n  }\n}\n\nfunction sum(xs: number[]): number {\n  return xs.reduce((a, c) => a + c, 0);\n}\n"]} |
\ | No newline at end of file |