UNPKG

45.7 kBJavaScriptView Raw
1"use strict";
2Object.defineProperty(exports, "__esModule", { value: true });
3exports.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;
4const 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 */
11function objectLike(pattern) {
12 return _objectContaining(pattern, false);
13}
14exports.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 */
20function deepObjectLike(pattern) {
21 return _objectContaining(pattern, true);
22}
23exports.deepObjectLike = deepObjectLike;
24function _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 */
77function exactValue(expected) {
78 const anno = { $exactValue: expected };
79 return annotateMatcher(anno, (value, inspection) => {
80 return matchLiteral(value, expected, inspection);
81 });
82}
83exports.exactValue = exactValue;
84/**
85 * A matcher for a list that contains all of the given elements in any order
86 */
87function 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}
121exports.arrayWith = arrayWith;
122/**
123 * Whether a value is an object
124 */
125function 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 */
135function failMatcher(inspection, error) {
136 inspection.failureReason = error;
137 return false;
138}
139exports.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 */
146function 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}
166exports.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 */
173function 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 */
225function 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 */
231function 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}
246exports.stringLike = stringLike;
247/**
248 * Matches any value
249 */
250function anything() {
251 return annotateMatcher({ $anything: true }, () => true);
252}
253exports.anything = anything;
254/**
255 * Negate an inner matcher
256 */
257function 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}
267exports.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 */
280class 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}
356exports.Capture = Capture;
357/**
358 * Match on the innards of a JSON string, instead of the complete string
359 */
360function 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}
377exports.encodedJson = encodedJson;
378function 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 */
386function matcherFrom(matcher) {
387 return isCallable(matcher) ? matcher : exactValue(matcher);
388}
389exports.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 */
399function annotateMatcher(how, matcher) {
400 matcher.toJSON = () => how;
401 return matcher;
402}
403exports.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