From f3570f201bcf3e815e9b42bd0880620d56d800ac Mon Sep 17 00:00:00 2001 From: Ben Noordhuis Date: Wed, 30 May 2018 16:01:00 +0200 Subject: [PATCH] lib: replace checkUint() with validateInt32() MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit PR-URL: https://github.com/nodejs/node/pull/20816 Reviewed-By: Anna Henningsen Reviewed-By: Colin Ihrig Reviewed-By: James M Snell Reviewed-By: Tobias Nießen --- lib/internal/crypto/pbkdf2.js | 10 ++++------ lib/internal/crypto/scrypt.js | 19 +++++++++++-------- lib/internal/crypto/util.js | 15 --------------- lib/internal/validators.js | 6 ++++++ test/parallel/test-crypto-pbkdf2.js | 17 +++++++++++++++-- 5 files changed, 36 insertions(+), 31 deletions(-) diff --git a/lib/internal/crypto/pbkdf2.js b/lib/internal/crypto/pbkdf2.js index f842ab82b9a26a..7dd51e8f72881a 100644 --- a/lib/internal/crypto/pbkdf2.js +++ b/lib/internal/crypto/pbkdf2.js @@ -2,7 +2,8 @@ const { AsyncWrap, Providers } = process.binding('async_wrap'); const { Buffer } = require('buffer'); -const { pbkdf2: _pbkdf2 } = process.binding('crypto'); +const { INT_MAX, pbkdf2: _pbkdf2 } = process.binding('crypto'); +const { validateInt32 } = require('internal/validators'); const { ERR_CRYPTO_INVALID_DIGEST, ERR_CRYPTO_PBKDF2_ERROR, @@ -11,7 +12,6 @@ const { } = require('internal/errors').codes; const { checkIsArrayBufferView, - checkIsUint, getDefaultEncoding, } = require('internal/crypto/util'); @@ -59,10 +59,8 @@ function check(password, salt, iterations, keylen, digest, callback) { password = checkIsArrayBufferView('password', password); salt = checkIsArrayBufferView('salt', salt); - // FIXME(bnoordhuis) The error message is in fact wrong since |iterations| - // cannot be > INT_MAX. Adjust in the next major release. - iterations = checkIsUint('iterations', iterations, 'a non-negative number'); - keylen = checkIsUint('keylen', keylen); + iterations = validateInt32(iterations, 'iterations', 0, INT_MAX); + keylen = validateInt32(keylen, 'keylen', 0, INT_MAX); return { password, salt, iterations, keylen, digest }; } diff --git a/lib/internal/crypto/scrypt.js b/lib/internal/crypto/scrypt.js index 09771455ac2d9d..a512af0f810bc3 100644 --- a/lib/internal/crypto/scrypt.js +++ b/lib/internal/crypto/scrypt.js @@ -2,7 +2,8 @@ const { AsyncWrap, Providers } = process.binding('async_wrap'); const { Buffer } = require('buffer'); -const { scrypt: _scrypt } = process.binding('crypto'); +const { INT_MAX, scrypt: _scrypt } = process.binding('crypto'); +const { validateInt32 } = require('internal/validators'); const { ERR_CRYPTO_SCRYPT_INVALID_PARAMETER, ERR_CRYPTO_SCRYPT_NOT_SUPPORTED, @@ -10,7 +11,6 @@ const { } = require('internal/errors').codes; const { checkIsArrayBufferView, - checkIsUint, getDefaultEncoding, } = require('internal/crypto/util'); @@ -75,16 +75,19 @@ function check(password, salt, keylen, options, callback) { throw new ERR_CRYPTO_SCRYPT_NOT_SUPPORTED(); password = checkIsArrayBufferView('password', password); - salt = checkIsArrayBufferView('salt', salt); - keylen = checkIsUint('keylen', keylen); + salt = checkIsArrayBufferView(salt, 'salt'); + keylen = validateInt32(keylen, 'keylen', 0, INT_MAX); let { N, r, p, maxmem } = defaults; if (options && options !== defaults) { - if (options.hasOwnProperty('N')) N = checkIsUint('N', options.N); - if (options.hasOwnProperty('r')) r = checkIsUint('r', options.r); - if (options.hasOwnProperty('p')) p = checkIsUint('p', options.p); + if (options.hasOwnProperty('N')) + N = validateInt32(options.N, 'N', 0, INT_MAX); + if (options.hasOwnProperty('r')) + r = validateInt32(options.r, 'r', 0, INT_MAX); + if (options.hasOwnProperty('p')) + p = validateInt32(options.p, 'p', 0, INT_MAX); if (options.hasOwnProperty('maxmem')) - maxmem = checkIsUint('maxmem', options.maxmem); + maxmem = validateInt32(options.maxmem, 'maxmem', 0, INT_MAX); if (N === 0) N = defaults.N; if (r === 0) r = defaults.r; if (p === 0) p = defaults.p; diff --git a/lib/internal/crypto/util.js b/lib/internal/crypto/util.js index cda973addcfe76..e2cd810a011bfa 100644 --- a/lib/internal/crypto/util.js +++ b/lib/internal/crypto/util.js @@ -16,7 +16,6 @@ const { ERR_CRYPTO_ENGINE_UNKNOWN, ERR_CRYPTO_TIMING_SAFE_EQUAL_LENGTH, ERR_INVALID_ARG_TYPE, - ERR_OUT_OF_RANGE, } = require('internal/errors').codes; const { Buffer } = require('buffer'); const { @@ -26,9 +25,6 @@ const { const { isArrayBufferView } = require('internal/util/types'); -const { - INT_MAX -} = process.binding('constants').crypto; var defaultEncoding = 'buffer'; @@ -99,19 +95,8 @@ function checkIsArrayBufferView(name, buffer) { return buffer; } -function checkIsUint(name, value, errmsg = `>= 0 && <= ${INT_MAX}`) { - if (typeof value !== 'number') - throw new ERR_INVALID_ARG_TYPE(name, 'number', value); - - if (value < 0 || !Number.isInteger(value) || value > INT_MAX) - throw new ERR_OUT_OF_RANGE(name, errmsg, value); - - return value; -} - module.exports = { checkIsArrayBufferView, - checkIsUint, getCiphers, getCurves, getDefaultEncoding, diff --git a/lib/internal/validators.js b/lib/internal/validators.js index b7b21d29bfa097..ab8643b9af673c 100644 --- a/lib/internal/validators.js +++ b/lib/internal/validators.js @@ -71,6 +71,8 @@ function validateInteger(value, name) { Error.captureStackTrace(err, validateInteger); throw err; } + + return value; } function validateInt32(value, name, min = -2147483648, max = 2147483647) { @@ -91,6 +93,8 @@ function validateInt32(value, name, min = -2147483648, max = 2147483647) { Error.captureStackTrace(err, validateInt32); throw err; } + + return value; } function validateUint32(value, name, positive) { @@ -112,6 +116,8 @@ function validateUint32(value, name, positive) { Error.captureStackTrace(err, validateUint32); throw err; } + + return value; } module.exports = { diff --git a/test/parallel/test-crypto-pbkdf2.js b/test/parallel/test-crypto-pbkdf2.js index f65176132ae5c4..e1fa1e3d86f5fb 100644 --- a/test/parallel/test-crypto-pbkdf2.js +++ b/test/parallel/test-crypto-pbkdf2.js @@ -71,7 +71,7 @@ assert.throws( code: 'ERR_OUT_OF_RANGE', name: 'RangeError [ERR_OUT_OF_RANGE]', message: 'The value of "iterations" is out of range. ' + - 'It must be a non-negative number. Received -1' + 'It must be >= 0 && <= 2147483647. Received -1' } ); @@ -87,7 +87,20 @@ assert.throws( }); }); -[Infinity, -Infinity, NaN, -1, 4073741824, INT_MAX + 1].forEach((input) => { +[Infinity, -Infinity, NaN].forEach((input) => { + assert.throws( + () => { + crypto.pbkdf2('password', 'salt', 1, input, 'sha256', + common.mustNotCall()); + }, { + code: 'ERR_OUT_OF_RANGE', + name: 'RangeError [ERR_OUT_OF_RANGE]', + message: 'The value of "keylen" is out of range. It ' + + `must be an integer. Received ${input}` + }); +}); + +[-1, 4073741824, INT_MAX + 1].forEach((input) => { assert.throws( () => { crypto.pbkdf2('password', 'salt', 1, input, 'sha256',