1 | ;
|
2 | Object.defineProperty(exports, "__esModule", { value: true });
|
3 | exports.annotateMatcher = exports.matcherFrom = exports.encodedJson = exports.Capture = exports.notMatching = exports.anything = exports.stringLike = exports.match = exports.failMatcher = exports.arrayWith = exports.exactValue = exports.deepObjectLike = exports.objectLike = void 0;
|
4 | const have_resource_1 = require("./have-resource");
|
5 | /**
|
6 | * A matcher for an object that contains at least the given fields with the given matchers (or literals)
|
7 | *
|
8 | * Only does lenient matching one level deep, at the next level all objects must declare the
|
9 | * exact expected keys again.
|
10 | */
|
11 | function objectLike(pattern) {
|
12 | return _objectContaining(pattern, false);
|
13 | }
|
14 | exports.objectLike = objectLike;
|
15 | /**
|
16 | * A matcher for an object that contains at least the given fields with the given matchers (or literals)
|
17 | *
|
18 | * Switches to "deep" lenient matching. Nested objects also only need to contain declared keys.
|
19 | */
|
20 | function deepObjectLike(pattern) {
|
21 | return _objectContaining(pattern, true);
|
22 | }
|
23 | exports.deepObjectLike = deepObjectLike;
|
24 | function _objectContaining(pattern, deep) {
|
25 | const anno = { [deep ? '$deepObjectLike' : '$objectLike']: pattern };
|
26 | return annotateMatcher(anno, (value, inspection) => {
|
27 | if (typeof value !== 'object' || !value) {
|
28 | return failMatcher(inspection, `Expect an object but got '${typeof value}'`);
|
29 | }
|
30 | const errors = new Array();
|
31 | for (const [patternKey, patternValue] of Object.entries(pattern)) {
|
32 | if (patternValue === have_resource_1.ABSENT) {
|
33 | if (value[patternKey] !== undefined) {
|
34 | errors.push(`Field ${patternKey} present, but shouldn't be`);
|
35 | }
|
36 | continue;
|
37 | }
|
38 | if (!(patternKey in value)) {
|
39 | errors.push(`Field ${patternKey} missing`);
|
40 | continue;
|
41 | }
|
42 | // If we are doing DEEP objectLike, translate object literals in the pattern into
|
43 | // more `deepObjectLike` matchers, even if they occur in lists.
|
44 | const matchValue = deep ? deepMatcherFromObjectLiteral(patternValue) : patternValue;
|
45 | const innerInspection = { ...inspection, failureReason: '' };
|
46 | const valueMatches = match(value[patternKey], matchValue, innerInspection);
|
47 | if (!valueMatches) {
|
48 | errors.push(`Field ${patternKey} mismatch: ${innerInspection.failureReason}`);
|
49 | }
|
50 | }
|
51 | /**
|
52 | * Transform nested object literals into more deep object matchers, if applicable
|
53 | *
|
54 | * Object literals in lists are also transformed.
|
55 | */
|
56 | function deepMatcherFromObjectLiteral(nestedPattern) {
|
57 | if (isObject(nestedPattern)) {
|
58 | return deepObjectLike(nestedPattern);
|
59 | }
|
60 | if (Array.isArray(nestedPattern)) {
|
61 | return nestedPattern.map(deepMatcherFromObjectLiteral);
|
62 | }
|
63 | return nestedPattern;
|
64 | }
|
65 | if (errors.length > 0) {
|
66 | return failMatcher(inspection, errors.join(', '));
|
67 | }
|
68 | return true;
|
69 | });
|
70 | }
|
71 | /**
|
72 | * Match exactly the given value
|
73 | *
|
74 | * This is the default, you only need this to escape from the deep lenient matching
|
75 | * of `deepObjectLike`.
|
76 | */
|
77 | function exactValue(expected) {
|
78 | const anno = { $exactValue: expected };
|
79 | return annotateMatcher(anno, (value, inspection) => {
|
80 | return matchLiteral(value, expected, inspection);
|
81 | });
|
82 | }
|
83 | exports.exactValue = exactValue;
|
84 | /**
|
85 | * A matcher for a list that contains all of the given elements in any order
|
86 | */
|
87 | function arrayWith(...elements) {
|
88 | if (elements.length === 0) {
|
89 | return anything();
|
90 | }
|
91 | const anno = { $arrayContaining: elements.length === 1 ? elements[0] : elements };
|
92 | return annotateMatcher(anno, (value, inspection) => {
|
93 | if (!Array.isArray(value)) {
|
94 | return failMatcher(inspection, `Expect an array but got '${typeof value}'`);
|
95 | }
|
96 | for (const element of elements) {
|
97 | const failure = longestFailure(value, element);
|
98 | if (failure) {
|
99 | return failMatcher(inspection, `Array did not contain expected element, closest match at index ${failure[0]}: ${failure[1]}`);
|
100 | }
|
101 | }
|
102 | return true;
|
103 | /**
|
104 | * Return 'null' if the matcher matches anywhere in the array, otherwise the longest error and its index
|
105 | */
|
106 | function longestFailure(array, matcher) {
|
107 | let fail = null;
|
108 | for (let i = 0; i < array.length; i++) {
|
109 | const innerInspection = { ...inspection, failureReason: '' };
|
110 | if (match(array[i], matcher, innerInspection)) {
|
111 | return null;
|
112 | }
|
113 | if (fail === null || innerInspection.failureReason.length > fail[1].length) {
|
114 | fail = [i, innerInspection.failureReason];
|
115 | }
|
116 | }
|
117 | return fail;
|
118 | }
|
119 | });
|
120 | }
|
121 | exports.arrayWith = arrayWith;
|
122 | /**
|
123 | * Whether a value is an object
|
124 | */
|
125 | function isObject(x) {
|
126 | // Because `typeof null === 'object'`.
|
127 | return x && typeof x === 'object';
|
128 | }
|
129 | /**
|
130 | * Helper function to make matcher failure reporting a little easier
|
131 | *
|
132 | * Our protocol is weird (change a string on a passed-in object and return 'false'),
|
133 | * but I don't want to change that right now.
|
134 | */
|
135 | function failMatcher(inspection, error) {
|
136 | inspection.failureReason = error;
|
137 | return false;
|
138 | }
|
139 | exports.failMatcher = failMatcher;
|
140 | /**
|
141 | * Match a given literal value against a matcher
|
142 | *
|
143 | * If the matcher is a callable, use that to evaluate the value. Otherwise, the values
|
144 | * must be literally the same.
|
145 | */
|
146 | function match(value, matcher, inspection) {
|
147 | if (isCallable(matcher)) {
|
148 | // Custom matcher (this mostly looks very weird because our `InspectionFailure` signature is weird)
|
149 | const innerInspection = { ...inspection, failureReason: '' };
|
150 | const result = matcher(value, innerInspection);
|
151 | if (typeof result !== 'boolean') {
|
152 | return failMatcher(inspection, `Predicate returned non-boolean return value: ${result}`);
|
153 | }
|
154 | if (!result && !innerInspection.failureReason) {
|
155 | // Custom matcher neglected to return an error
|
156 | return failMatcher(inspection, 'Predicate returned false');
|
157 | }
|
158 | // Propagate inner error in case of failure
|
159 | if (!result) {
|
160 | inspection.failureReason = innerInspection.failureReason;
|
161 | }
|
162 | return result;
|
163 | }
|
164 | return matchLiteral(value, matcher, inspection);
|
165 | }
|
166 | exports.match = match;
|
167 | /**
|
168 | * Match a literal value at the top level.
|
169 | *
|
170 | * When recursing into arrays or objects, the nested values can be either matchers
|
171 | * or literals.
|
172 | */
|
173 | function matchLiteral(value, pattern, inspection) {
|
174 | if (pattern == null) {
|
175 | return true;
|
176 | }
|
177 | const errors = new Array();
|
178 | if (Array.isArray(value) !== Array.isArray(pattern)) {
|
179 | return failMatcher(inspection, 'Array type mismatch');
|
180 | }
|
181 | if (Array.isArray(value)) {
|
182 | if (pattern.length !== value.length) {
|
183 | return failMatcher(inspection, 'Array length mismatch');
|
184 | }
|
185 | // Recurse comparison for individual objects
|
186 | for (let i = 0; i < pattern.length; i++) {
|
187 | if (!match(value[i], pattern[i], { ...inspection })) {
|
188 | errors.push(`Array element ${i} mismatch`);
|
189 | }
|
190 | }
|
191 | if (errors.length > 0) {
|
192 | return failMatcher(inspection, errors.join(', '));
|
193 | }
|
194 | return true;
|
195 | }
|
196 | if ((typeof value === 'object') !== (typeof pattern === 'object')) {
|
197 | return failMatcher(inspection, 'Object type mismatch');
|
198 | }
|
199 | if (typeof pattern === 'object') {
|
200 | // Check that all fields in the pattern have the right value
|
201 | const innerInspection = { ...inspection, failureReason: '' };
|
202 | const matcher = objectLike(pattern)(value, innerInspection);
|
203 | if (!matcher) {
|
204 | inspection.failureReason = innerInspection.failureReason;
|
205 | return false;
|
206 | }
|
207 | // Check no fields uncovered
|
208 | const realFields = new Set(Object.keys(value));
|
209 | for (const key of Object.keys(pattern)) {
|
210 | realFields.delete(key);
|
211 | }
|
212 | if (realFields.size > 0) {
|
213 | return failMatcher(inspection, `Unexpected keys present in object: ${Array.from(realFields).join(', ')}`);
|
214 | }
|
215 | return true;
|
216 | }
|
217 | if (value !== pattern) {
|
218 | return failMatcher(inspection, 'Different values');
|
219 | }
|
220 | return true;
|
221 | }
|
222 | /**
|
223 | * Whether a value is a callable
|
224 | */
|
225 | function isCallable(x) {
|
226 | return x && {}.toString.call(x) === '[object Function]';
|
227 | }
|
228 | /**
|
229 | * Do a glob-like pattern match (which only supports *s). Supports multiline strings.
|
230 | */
|
231 | function stringLike(pattern) {
|
232 | // Replace * with .* in the string, escape the rest and brace with ^...$
|
233 | const regex = new RegExp(`^${pattern.split('*').map(escapeRegex).join('.*')}$`, 'm');
|
234 | return annotateMatcher({ $stringContaining: pattern }, (value, failure) => {
|
235 | if (typeof value !== 'string') {
|
236 | failure.failureReason = `Expected a string, but got '${typeof value}'`;
|
237 | return false;
|
238 | }
|
239 | if (!regex.test(value)) {
|
240 | failure.failureReason = 'String did not match pattern';
|
241 | return false;
|
242 | }
|
243 | return true;
|
244 | });
|
245 | }
|
246 | exports.stringLike = stringLike;
|
247 | /**
|
248 | * Matches any value
|
249 | */
|
250 | function anything() {
|
251 | return annotateMatcher({ $anything: true }, () => true);
|
252 | }
|
253 | exports.anything = anything;
|
254 | /**
|
255 | * Negate an inner matcher
|
256 | */
|
257 | function notMatching(matcher) {
|
258 | return annotateMatcher({ $notMatching: matcher }, (value, failure) => {
|
259 | const result = matcherFrom(matcher)(value, failure);
|
260 | if (result) {
|
261 | failure.failureReason = 'Should not have matched, but did';
|
262 | return false;
|
263 | }
|
264 | return true;
|
265 | });
|
266 | }
|
267 | exports.notMatching = notMatching;
|
268 | /**
|
269 | * Captures a value onto an object if it matches a given inner matcher
|
270 | *
|
271 | * @example
|
272 | *
|
273 | * const someValue = Capture.aString();
|
274 | * expect(stack).toHaveResource({
|
275 | * // ...
|
276 | * Value: someValue.capture(stringMatching('*a*')),
|
277 | * });
|
278 | * console.log(someValue.capturedValue);
|
279 | */
|
280 | class Capture {
|
281 | constructor(typeValidator) {
|
282 | this.typeValidator = typeValidator;
|
283 | this._didCapture = false;
|
284 | this._wasInvoked = false;
|
285 | }
|
286 | /**
|
287 | * A Capture object that captures any type
|
288 | */
|
289 | static anyType() {
|
290 | return new Capture();
|
291 | }
|
292 | /**
|
293 | * A Capture object that captures a string type
|
294 | */
|
295 | static aString() {
|
296 | return new Capture((x) => {
|
297 | if (typeof x !== 'string') {
|
298 | throw new Error(`Expected to capture a string, got '${x}'`);
|
299 | }
|
300 | return true;
|
301 | });
|
302 | }
|
303 | /**
|
304 | * A Capture object that captures a custom type
|
305 | */
|
306 | // eslint-disable-next-line @typescript-eslint/no-shadow
|
307 | static a(validator) {
|
308 | return new Capture(validator);
|
309 | }
|
310 | /**
|
311 | * Capture the value if the inner matcher successfully matches it
|
312 | *
|
313 | * If no matcher is given, `anything()` is assumed.
|
314 | *
|
315 | * And exception will be thrown if the inner matcher returns `true` and
|
316 | * the value turns out to be of a different type than the `Capture` object
|
317 | * is expecting.
|
318 | */
|
319 | capture(matcher) {
|
320 | if (matcher === undefined) {
|
321 | matcher = anything();
|
322 | }
|
323 | return annotateMatcher({ $capture: matcher }, (value, failure) => {
|
324 | this._wasInvoked = true;
|
325 | const result = matcherFrom(matcher)(value, failure);
|
326 | if (result) {
|
327 | if (this.typeValidator && !this.typeValidator(value)) {
|
328 | throw new Error(`Value not of the expected type: ${value}`);
|
329 | }
|
330 | this._didCapture = true;
|
331 | this._value = value;
|
332 | }
|
333 | return result;
|
334 | });
|
335 | }
|
336 | /**
|
337 | * Whether a value was successfully captured
|
338 | */
|
339 | get didCapture() {
|
340 | return this._didCapture;
|
341 | }
|
342 | /**
|
343 | * Return the value that was captured
|
344 | *
|
345 | * Throws an exception if now value was captured
|
346 | */
|
347 | get capturedValue() {
|
348 | // When this module is ported to jsii, the type parameter will obviously
|
349 | // have to be dropped and this will have to turn into an `any`.
|
350 | if (!this.didCapture) {
|
351 | throw new Error(`Did not capture a value: ${this._wasInvoked ? 'inner matcher failed' : 'never invoked'}`);
|
352 | }
|
353 | return this._value;
|
354 | }
|
355 | }
|
356 | exports.Capture = Capture;
|
357 | /**
|
358 | * Match on the innards of a JSON string, instead of the complete string
|
359 | */
|
360 | function encodedJson(matcher) {
|
361 | return annotateMatcher({ $encodedJson: matcher }, (value, failure) => {
|
362 | if (typeof value !== 'string') {
|
363 | failure.failureReason = `Expected a string, but got '${typeof value}'`;
|
364 | return false;
|
365 | }
|
366 | let decoded;
|
367 | try {
|
368 | decoded = JSON.parse(value);
|
369 | }
|
370 | catch (e) {
|
371 | failure.failureReason = `String is not JSON: ${e}`;
|
372 | return false;
|
373 | }
|
374 | return matcherFrom(matcher)(decoded, failure);
|
375 | });
|
376 | }
|
377 | exports.encodedJson = encodedJson;
|
378 | function escapeRegex(s) {
|
379 | return s.replace(/[.*+?^${}()|[\]\\]/g, '\\$&'); // $& means the whole matched string
|
380 | }
|
381 | /**
|
382 | * Make a matcher out of the given argument if it's not a matcher already
|
383 | *
|
384 | * If it's not a matcher, it will be treated as a literal.
|
385 | */
|
386 | function matcherFrom(matcher) {
|
387 | return isCallable(matcher) ? matcher : exactValue(matcher);
|
388 | }
|
389 | exports.matcherFrom = matcherFrom;
|
390 | /**
|
391 | * Annotate a matcher with toJSON
|
392 | *
|
393 | * We will JSON.stringify() values if we have a match failure, but for matchers this
|
394 | * would show (in traditional JS fashion) something like '[function Function]', or more
|
395 | * accurately nothing at all since functions cannot be JSONified.
|
396 | *
|
397 | * We override to JSON() in order to produce a readadable version of the matcher.
|
398 | */
|
399 | function annotateMatcher(how, matcher) {
|
400 | matcher.toJSON = () => how;
|
401 | return matcher;
|
402 | }
|
403 | exports.annotateMatcher = annotateMatcher;
|
404 | //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"have-resource-matchers.js","sourceRoot":"","sources":["have-resource-matchers.ts"],"names":[],"mappings":"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","sourcesContent":["import { ABSENT, InspectionFailure, PropertyMatcher } from './have-resource';\n\n/**\n * A matcher for an object that contains at least the given fields with the given matchers (or literals)\n *\n * Only does lenient matching one level deep, at the next level all objects must declare the\n * exact expected keys again.\n */\nexport function objectLike<A extends object>(pattern: A): PropertyMatcher {\n  return _objectContaining(pattern, false);\n}\n\n/**\n * A matcher for an object that contains at least the given fields with the given matchers (or literals)\n *\n * Switches to \"deep\" lenient matching. Nested objects also only need to contain declared keys.\n */\nexport function deepObjectLike<A extends object>(pattern: A): PropertyMatcher {\n  return _objectContaining(pattern, true);\n}\n\nfunction _objectContaining<A extends object>(pattern: A, deep: boolean): PropertyMatcher {\n  const anno = { [deep ? '$deepObjectLike' : '$objectLike']: pattern };\n\n  return annotateMatcher(anno, (value: any, inspection: InspectionFailure): boolean => {\n    if (typeof value !== 'object' || !value) {\n      return failMatcher(inspection, `Expect an object but got '${typeof value}'`);\n    }\n\n    const errors = new Array<string>();\n\n    for (const [patternKey, patternValue] of Object.entries(pattern)) {\n      if (patternValue === ABSENT) {\n        if (value[patternKey] !== undefined) { errors.push(`Field ${patternKey} present, but shouldn't be`); }\n        continue;\n      }\n\n      if (!(patternKey in value)) {\n        errors.push(`Field ${patternKey} missing`);\n        continue;\n      }\n\n      // If we are doing DEEP objectLike, translate object literals in the pattern into\n      // more `deepObjectLike` matchers, even if they occur in lists.\n      const matchValue = deep ? deepMatcherFromObjectLiteral(patternValue) : patternValue;\n\n      const innerInspection = { ...inspection, failureReason: '' };\n      const valueMatches = match(value[patternKey], matchValue, innerInspection);\n      if (!valueMatches) {\n        errors.push(`Field ${patternKey} mismatch: ${innerInspection.failureReason}`);\n      }\n    }\n\n    /**\n     * Transform nested object literals into more deep object matchers, if applicable\n     *\n     * Object literals in lists are also transformed.\n     */\n    function deepMatcherFromObjectLiteral(nestedPattern: any): any {\n      if (isObject(nestedPattern)) {\n        return deepObjectLike(nestedPattern);\n      }\n      if (Array.isArray(nestedPattern)) {\n        return nestedPattern.map(deepMatcherFromObjectLiteral);\n      }\n      return nestedPattern;\n    }\n\n    if (errors.length > 0) {\n      return failMatcher(inspection, errors.join(', '));\n    }\n    return true;\n  });\n}\n\n/**\n * Match exactly the given value\n *\n * This is the default, you only need this to escape from the deep lenient matching\n * of `deepObjectLike`.\n */\nexport function exactValue(expected: any): PropertyMatcher {\n  const anno = { $exactValue: expected };\n  return annotateMatcher(anno, (value: any, inspection: InspectionFailure): boolean => {\n    return matchLiteral(value, expected, inspection);\n  });\n}\n\n/**\n * A matcher for a list that contains all of the given elements in any order\n */\nexport function arrayWith(...elements: any[]): PropertyMatcher {\n  if (elements.length === 0) { return anything(); }\n\n  const anno = { $arrayContaining: elements.length === 1 ? elements[0] : elements };\n  return annotateMatcher(anno, (value: any, inspection: InspectionFailure): boolean => {\n    if (!Array.isArray(value)) {\n      return failMatcher(inspection, `Expect an array but got '${typeof value}'`);\n    }\n\n    for (const element of elements) {\n      const failure = longestFailure(value, element);\n      if (failure) {\n        return failMatcher(inspection, `Array did not contain expected element, closest match at index ${failure[0]}: ${failure[1]}`);\n      }\n    }\n\n    return true;\n\n    /**\n     * Return 'null' if the matcher matches anywhere in the array, otherwise the longest error and its index\n     */\n    function longestFailure(array: any[], matcher: any): [number, string] | null {\n      let fail: [number, string] | null = null;\n      for (let i = 0; i < array.length; i++) {\n        const innerInspection = { ...inspection, failureReason: '' };\n        if (match(array[i], matcher, innerInspection)) {\n          return null;\n        }\n\n        if (fail === null || innerInspection.failureReason.length > fail[1].length) {\n          fail = [i, innerInspection.failureReason];\n        }\n      }\n      return fail;\n    }\n  });\n}\n\n/**\n * Whether a value is an object\n */\nfunction isObject(x: any): x is object {\n  // Because `typeof null === 'object'`.\n  return x && typeof x === 'object';\n}\n\n/**\n * Helper function to make matcher failure reporting a little easier\n *\n * Our protocol is weird (change a string on a passed-in object and return 'false'),\n * but I don't want to change that right now.\n */\nexport function failMatcher(inspection: InspectionFailure, error: string): boolean {\n  inspection.failureReason = error;\n  return false;\n}\n\n/**\n * Match a given literal value against a matcher\n *\n * If the matcher is a callable, use that to evaluate the value. Otherwise, the values\n * must be literally the same.\n */\nexport function match(value: any, matcher: any, inspection: InspectionFailure) {\n  if (isCallable(matcher)) {\n    // Custom matcher (this mostly looks very weird because our `InspectionFailure` signature is weird)\n    const innerInspection: InspectionFailure = { ...inspection, failureReason: '' };\n    const result = matcher(value, innerInspection);\n    if (typeof result !== 'boolean') {\n      return failMatcher(inspection, `Predicate returned non-boolean return value: ${result}`);\n    }\n    if (!result && !innerInspection.failureReason) {\n      // Custom matcher neglected to return an error\n      return failMatcher(inspection, 'Predicate returned false');\n    }\n    // Propagate inner error in case of failure\n    if (!result) { inspection.failureReason = innerInspection.failureReason; }\n    return result;\n  }\n\n  return matchLiteral(value, matcher, inspection);\n}\n\n/**\n * Match a literal value at the top level.\n *\n * When recursing into arrays or objects, the nested values can be either matchers\n * or literals.\n */\nfunction matchLiteral(value: any, pattern: any, inspection: InspectionFailure) {\n  if (pattern == null) { return true; }\n\n  const errors = new Array<string>();\n\n  if (Array.isArray(value) !== Array.isArray(pattern)) {\n    return failMatcher(inspection, 'Array type mismatch');\n  }\n  if (Array.isArray(value)) {\n    if (pattern.length !== value.length) {\n      return failMatcher(inspection, 'Array length mismatch');\n    }\n\n    // Recurse comparison for individual objects\n    for (let i = 0; i < pattern.length; i++) {\n      if (!match(value[i], pattern[i], { ...inspection })) {\n        errors.push(`Array element ${i} mismatch`);\n      }\n    }\n\n    if (errors.length > 0) {\n      return failMatcher(inspection, errors.join(', '));\n    }\n    return true;\n  }\n  if ((typeof value === 'object') !== (typeof pattern === 'object')) {\n    return failMatcher(inspection, 'Object type mismatch');\n  }\n  if (typeof pattern === 'object') {\n    // Check that all fields in the pattern have the right value\n    const innerInspection = { ...inspection, failureReason: '' };\n    const matcher = objectLike(pattern)(value, innerInspection);\n    if (!matcher) {\n      inspection.failureReason = innerInspection.failureReason;\n      return false;\n    }\n\n    // Check no fields uncovered\n    const realFields = new Set(Object.keys(value));\n    for (const key of Object.keys(pattern)) { realFields.delete(key); }\n    if (realFields.size > 0) {\n      return failMatcher(inspection, `Unexpected keys present in object: ${Array.from(realFields).join(', ')}`);\n    }\n    return true;\n  }\n\n  if (value !== pattern) {\n    return failMatcher(inspection, 'Different values');\n  }\n\n  return true;\n}\n\n/**\n * Whether a value is a callable\n */\nfunction isCallable(x: any): x is ((...args: any[]) => any) {\n  return x && {}.toString.call(x) === '[object Function]';\n}\n\n/**\n * Do a glob-like pattern match (which only supports *s). Supports multiline strings.\n */\nexport function stringLike(pattern: string): PropertyMatcher {\n  // Replace * with .* in the string, escape the rest and brace with ^...$\n  const regex = new RegExp(`^${pattern.split('*').map(escapeRegex).join('.*')}$`, 'm');\n\n  return annotateMatcher({ $stringContaining: pattern }, (value: any, failure: InspectionFailure) => {\n    if (typeof value !== 'string') {\n      failure.failureReason = `Expected a string, but got '${typeof value}'`;\n      return false;\n    }\n\n    if (!regex.test(value)) {\n      failure.failureReason = 'String did not match pattern';\n      return false;\n    }\n\n    return true;\n  });\n}\n\n/**\n * Matches any value\n */\nexport function anything(): PropertyMatcher {\n  return annotateMatcher({ $anything: true }, () => true);\n}\n\n/**\n * Negate an inner matcher\n */\nexport function notMatching(matcher: any): PropertyMatcher {\n  return annotateMatcher({ $notMatching: matcher }, (value: any, failure: InspectionFailure) => {\n    const result = matcherFrom(matcher)(value, failure);\n    if (result) {\n      failure.failureReason = 'Should not have matched, but did';\n      return false;\n    }\n    return true;\n  });\n}\n\nexport type TypeValidator<T> = (x: any) => x is T;\n\n/**\n * Captures a value onto an object if it matches a given inner matcher\n *\n * @example\n *\n * const someValue = Capture.aString();\n * expect(stack).toHaveResource({\n *    // ...\n *    Value: someValue.capture(stringMatching('*a*')),\n * });\n * console.log(someValue.capturedValue);\n */\nexport class Capture<T=any> {\n  /**\n   * A Capture object that captures any type\n   */\n  public static anyType(): Capture<any> {\n    return new Capture();\n  }\n\n  /**\n   * A Capture object that captures a string type\n   */\n  public static aString(): Capture<string> {\n    return new Capture((x: any): x is string => {\n      if (typeof x !== 'string') {\n        throw new Error(`Expected to capture a string, got '${x}'`);\n      }\n      return true;\n    });\n  }\n\n  /**\n   * A Capture object that captures a custom type\n   */\n  // eslint-disable-next-line @typescript-eslint/no-shadow\n  public static a<T>(validator: TypeValidator<T>): Capture<T> {\n    return new Capture(validator);\n  }\n\n  private _value?: T;\n  private _didCapture = false;\n  private _wasInvoked = false;\n\n  protected constructor(private readonly typeValidator?: TypeValidator<T>) {\n  }\n\n  /**\n   * Capture the value if the inner matcher successfully matches it\n   *\n   * If no matcher is given, `anything()` is assumed.\n   *\n   * And exception will be thrown if the inner matcher returns `true` and\n   * the value turns out to be of a different type than the `Capture` object\n   * is expecting.\n   */\n  public capture(matcher?: any): PropertyMatcher {\n    if (matcher === undefined) {\n      matcher = anything();\n    }\n\n    return annotateMatcher({ $capture: matcher }, (value: any, failure: InspectionFailure) => {\n      this._wasInvoked = true;\n      const result = matcherFrom(matcher)(value, failure);\n      if (result) {\n        if (this.typeValidator && !this.typeValidator(value)) {\n          throw new Error(`Value not of the expected type: ${value}`);\n        }\n        this._didCapture = true;\n        this._value = value;\n      }\n      return result;\n    });\n  }\n\n  /**\n   * Whether a value was successfully captured\n   */\n  public get didCapture() {\n    return this._didCapture;\n  }\n\n  /**\n   * Return the value that was captured\n   *\n   * Throws an exception if now value was captured\n   */\n  public get capturedValue(): T {\n    // When this module is ported to jsii, the type parameter will obviously\n    // have to be dropped and this will have to turn into an `any`.\n    if (!this.didCapture) {\n      throw new Error(`Did not capture a value: ${this._wasInvoked ? 'inner matcher failed' : 'never invoked'}`);\n    }\n    return this._value!;\n  }\n}\n\n/**\n * Match on the innards of a JSON string, instead of the complete string\n */\nexport function encodedJson(matcher: any): PropertyMatcher {\n  return annotateMatcher({ $encodedJson: matcher }, (value: any, failure: InspectionFailure) => {\n    if (typeof value !== 'string') {\n      failure.failureReason = `Expected a string, but got '${typeof value}'`;\n      return false;\n    }\n\n    let decoded;\n    try {\n      decoded = JSON.parse(value);\n    } catch (e) {\n      failure.failureReason = `String is not JSON: ${e}`;\n      return false;\n    }\n\n    return matcherFrom(matcher)(decoded, failure);\n  });\n}\n\nfunction escapeRegex(s: string) {\n  return s.replace(/[.*+?^${}()|[\\]\\\\]/g, '\\\\$&'); // $& means the whole matched string\n}\n\n/**\n * Make a matcher out of the given argument if it's not a matcher already\n *\n * If it's not a matcher, it will be treated as a literal.\n */\nexport function matcherFrom(matcher: any): PropertyMatcher {\n  return isCallable(matcher) ? matcher : exactValue(matcher);\n}\n\n/**\n * Annotate a matcher with toJSON\n *\n * We will JSON.stringify() values if we have a match failure, but for matchers this\n * would show (in traditional JS fashion) something like '[function Function]', or more\n * accurately nothing at all since functions cannot be JSONified.\n *\n * We override to JSON() in order to produce a readadable version of the matcher.\n */\nexport function annotateMatcher<A extends object>(how: A, matcher: PropertyMatcher): PropertyMatcher {\n  (matcher as any).toJSON = () => how;\n  return matcher;\n}\n"]} |
\ | No newline at end of file |