From 92abf08f0b71c4fcfa898e41c795e43c4127e5a6 Mon Sep 17 00:00:00 2001 From: Ruben Bridgewater Date: Wed, 2 May 2018 19:26:32 +0200 Subject: [PATCH 1/3] assert: accept regular expressions to validate This makes sure regular expressions on validation objects validate against strings when used with `assert.throws` and `assert.rejects`. --- doc/api/assert.md | 47 +++++++++++++++++++++++++++++++----- lib/assert.js | 23 ++++++++++++++---- test/parallel/test-assert.js | 26 ++++++++++++++++++++ 3 files changed, 85 insertions(+), 11 deletions(-) diff --git a/doc/api/assert.md b/doc/api/assert.md index 468293b208a90d..0d2f989cd06b63 100644 --- a/doc/api/assert.md +++ b/doc/api/assert.md @@ -1069,18 +1069,26 @@ changes: Expects the function `block` to throw an error. If specified, `error` can be a [`Class`][], [`RegExp`][], a validation function, -an object where each property will be tested for, or an instance of error where -each property will be tested for including the non-enumerable `message` and -`name` properties. +an validation object where each property will be tested for strict deep +equality, or an instance of error where each property will be tested for strict +deep equality including the non-enumerable `message` and `name` properties. When +using an validation object, it is also possible to use a regular expression, +when validating against a string property. See below for examples. If specified, `message` will be the message provided by the `AssertionError` if the block fails to throw. -Custom error object / error instance: +Custom validation object / error instance: ```js const err = new TypeError('Wrong value'); err.code = 404; +err.foo = 'bar'; +err.info = { + nested: true, + baz: 'text' +}; +err.reg = /abc/i; assert.throws( () => { @@ -1088,8 +1096,35 @@ assert.throws( }, { name: 'TypeError', - message: 'Wrong value' - // Note that only properties on the error object will be tested! + message: 'Wrong value', + info: { + nested: true, + baz: 'text' + } + // Note that only properties on the validation object will be tested for. + // Using nested objects requires all properties to be present. Otherwise + // the validation is going to fail. + } +); + +// Using regular expressions to validate the error properties: +assert.throws( + () => { + throw err; + }, + { + name: /^TypeError$/, + message: /Wrong/, + foo: 'bar', + info: { + nested: true, + // Note that it is not possible to use regular expressions for nested + // properties! + baz: 'text' + }, + // This will strictly validate that the regular expression is identical to + // the regular expression on the error. + reg: /abc/i } ); diff --git a/lib/assert.js b/lib/assert.js index 05b43b5b54d074..ac9fd30793aaf7 100644 --- a/lib/assert.js +++ b/lib/assert.js @@ -35,7 +35,7 @@ const { } } = require('internal/errors'); const { openSync, closeSync, readSync } = require('fs'); -const { inspect, types: { isPromise } } = require('util'); +const { inspect, types: { isPromise, isRegExp } } = require('util'); const { EOL } = require('os'); const { NativeModule } = require('internal/bootstrap/loaders'); @@ -362,10 +362,18 @@ assert.notStrictEqual = function notStrictEqual(actual, expected, message) { }; class Comparison { - constructor(obj, keys) { + constructor(obj, keys, actual) { for (const key of keys) { - if (key in obj) - this[key] = obj[key]; + if (key in obj) { + if (actual !== undefined && + typeof actual[key] === 'string' && + isRegExp(obj[key]) && + obj[key].test(actual[key])) { + this[key] = actual[key]; + } else { + this[key] = obj[key]; + } + } } } } @@ -375,7 +383,7 @@ function compareExceptionKey(actual, expected, key, message, keys) { if (!message) { // Create placeholder objects to create a nice output. const a = new Comparison(actual, keys); - const b = new Comparison(expected, keys); + const b = new Comparison(expected, keys, actual); const tmpLimit = Error.stackTraceLimit; Error.stackTraceLimit = 0; @@ -415,6 +423,11 @@ function expectedException(actual, expected, msg) { keys.push('name', 'message'); } for (const key of keys) { + if (typeof actual[key] === 'string' && + isRegExp(expected[key]) && + expected[key].test(actual[key])) { + continue; + } compareExceptionKey(actual, expected, key, msg, keys); } return true; diff --git a/test/parallel/test-assert.js b/test/parallel/test-assert.js index d892122ed62ff2..cbcda17ab7188d 100644 --- a/test/parallel/test-assert.js +++ b/test/parallel/test-assert.js @@ -914,3 +914,29 @@ assert.throws( } ); } + +assert.throws( + () => { throw new TypeError('foobar'); }, + { + message: /foo/, + name: /^TypeError$/ + } +); + +assert.throws( + () => assert.throws( + () => { throw new TypeError('foobar'); }, + { + message: /fooa/, + name: /^TypeError$/ + } + ), + { + message: `${start}\n${actExp}\n\n` + + ' Comparison {\n' + + "- message: 'foobar',\n" + + '+ message: /fooa/,\n' + + " name: 'TypeError'\n" + + ' }' + } +); From e20115a13d6af02168a3e531e10415e2a5314f68 Mon Sep 17 00:00:00 2001 From: Ruben Bridgewater Date: Mon, 7 May 2018 13:42:24 +0200 Subject: [PATCH 2/3] fixup: address comments --- doc/api/assert.md | 23 +++++++++++++---------- 1 file changed, 13 insertions(+), 10 deletions(-) diff --git a/doc/api/assert.md b/doc/api/assert.md index 0d2f989cd06b63..bcbeb5ad6c9d25 100644 --- a/doc/api/assert.md +++ b/doc/api/assert.md @@ -1069,11 +1069,11 @@ changes: Expects the function `block` to throw an error. If specified, `error` can be a [`Class`][], [`RegExp`][], a validation function, -an validation object where each property will be tested for strict deep -equality, or an instance of error where each property will be tested for strict -deep equality including the non-enumerable `message` and `name` properties. When -using an validation object, it is also possible to use a regular expression, -when validating against a string property. See below for examples. +a validation object where each property will be tested for strict deep equality, +or an instance of error where each property will be tested for strict deep +equality including the non-enumerable `message` and `name` properties. When +using a object, it is also possible to use a regular expression, when validating +against a string property. See below for examples. If specified, `message` will be the message provided by the `AssertionError` if the block fails to throw. @@ -1107,23 +1107,26 @@ assert.throws( } ); -// Using regular expressions to validate the error properties: +// Using regular expressions to validate error properties: assert.throws( () => { throw err; }, { + // The `name` and `message` properties are strings and using regular + // expressions on those will match against the string. If they fail, an + // error is thrown. name: /^TypeError$/, message: /Wrong/, foo: 'bar', info: { nested: true, - // Note that it is not possible to use regular expressions for nested - // properties! + // It is not possible to use regular expressions for nested properties! baz: 'text' }, - // This will strictly validate that the regular expression is identical to - // the regular expression on the error. + // The `reg` property contains a regular expression and only if the + // validation object contains an identical regular expression, it is going + // to pass. reg: /abc/i } ); From 40aa105fc84866e5e34e26de62932f01b354133f Mon Sep 17 00:00:00 2001 From: Ruben Bridgewater Date: Mon, 7 May 2018 14:00:13 +0200 Subject: [PATCH 3/3] fixup --- doc/api/assert.md | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/doc/api/assert.md b/doc/api/assert.md index bcbeb5ad6c9d25..6bd16c66637016 100644 --- a/doc/api/assert.md +++ b/doc/api/assert.md @@ -1072,8 +1072,8 @@ If specified, `error` can be a [`Class`][], [`RegExp`][], a validation function, a validation object where each property will be tested for strict deep equality, or an instance of error where each property will be tested for strict deep equality including the non-enumerable `message` and `name` properties. When -using a object, it is also possible to use a regular expression, when validating -against a string property. See below for examples. +using an object, it is also possible to use a regular expression, when +validating against a string property. See below for examples. If specified, `message` will be the message provided by the `AssertionError` if the block fails to throw.