1 |
3a515b92
|
cagy
|
'use strict'
|
2 |
|
|
|
3 |
|
|
function oldBrowser () {
|
4 |
|
|
throw new Error('secure random number generation not supported by this browser\nuse chrome, FireFox or Internet Explorer 11')
|
5 |
|
|
}
|
6 |
|
|
var safeBuffer = require('safe-buffer')
|
7 |
|
|
var randombytes = require('randombytes')
|
8 |
|
|
var Buffer = safeBuffer.Buffer
|
9 |
|
|
var kBufferMaxLength = safeBuffer.kMaxLength
|
10 |
|
|
var crypto = global.crypto || global.msCrypto
|
11 |
|
|
var kMaxUint32 = Math.pow(2, 32) - 1
|
12 |
|
|
function assertOffset (offset, length) {
|
13 |
|
|
if (typeof offset !== 'number' || offset !== offset) { // eslint-disable-line no-self-compare
|
14 |
|
|
throw new TypeError('offset must be a number')
|
15 |
|
|
}
|
16 |
|
|
|
17 |
|
|
if (offset > kMaxUint32 || offset < 0) {
|
18 |
|
|
throw new TypeError('offset must be a uint32')
|
19 |
|
|
}
|
20 |
|
|
|
21 |
|
|
if (offset > kBufferMaxLength || offset > length) {
|
22 |
|
|
throw new RangeError('offset out of range')
|
23 |
|
|
}
|
24 |
|
|
}
|
25 |
|
|
|
26 |
|
|
function assertSize (size, offset, length) {
|
27 |
|
|
if (typeof size !== 'number' || size !== size) { // eslint-disable-line no-self-compare
|
28 |
|
|
throw new TypeError('size must be a number')
|
29 |
|
|
}
|
30 |
|
|
|
31 |
|
|
if (size > kMaxUint32 || size < 0) {
|
32 |
|
|
throw new TypeError('size must be a uint32')
|
33 |
|
|
}
|
34 |
|
|
|
35 |
|
|
if (size + offset > length || size > kBufferMaxLength) {
|
36 |
|
|
throw new RangeError('buffer too small')
|
37 |
|
|
}
|
38 |
|
|
}
|
39 |
|
|
if ((crypto && crypto.getRandomValues) || !process.browser) {
|
40 |
|
|
exports.randomFill = randomFill
|
41 |
|
|
exports.randomFillSync = randomFillSync
|
42 |
|
|
} else {
|
43 |
|
|
exports.randomFill = oldBrowser
|
44 |
|
|
exports.randomFillSync = oldBrowser
|
45 |
|
|
}
|
46 |
|
|
function randomFill (buf, offset, size, cb) {
|
47 |
|
|
if (!Buffer.isBuffer(buf) && !(buf instanceof global.Uint8Array)) {
|
48 |
|
|
throw new TypeError('"buf" argument must be a Buffer or Uint8Array')
|
49 |
|
|
}
|
50 |
|
|
|
51 |
|
|
if (typeof offset === 'function') {
|
52 |
|
|
cb = offset
|
53 |
|
|
offset = 0
|
54 |
|
|
size = buf.length
|
55 |
|
|
} else if (typeof size === 'function') {
|
56 |
|
|
cb = size
|
57 |
|
|
size = buf.length - offset
|
58 |
|
|
} else if (typeof cb !== 'function') {
|
59 |
|
|
throw new TypeError('"cb" argument must be a function')
|
60 |
|
|
}
|
61 |
|
|
assertOffset(offset, buf.length)
|
62 |
|
|
assertSize(size, offset, buf.length)
|
63 |
|
|
return actualFill(buf, offset, size, cb)
|
64 |
|
|
}
|
65 |
|
|
|
66 |
|
|
function actualFill (buf, offset, size, cb) {
|
67 |
|
|
if (process.browser) {
|
68 |
|
|
var ourBuf = buf.buffer
|
69 |
|
|
var uint = new Uint8Array(ourBuf, offset, size)
|
70 |
|
|
crypto.getRandomValues(uint)
|
71 |
|
|
if (cb) {
|
72 |
|
|
process.nextTick(function () {
|
73 |
|
|
cb(null, buf)
|
74 |
|
|
})
|
75 |
|
|
return
|
76 |
|
|
}
|
77 |
|
|
return buf
|
78 |
|
|
}
|
79 |
|
|
if (cb) {
|
80 |
|
|
randombytes(size, function (err, bytes) {
|
81 |
|
|
if (err) {
|
82 |
|
|
return cb(err)
|
83 |
|
|
}
|
84 |
|
|
bytes.copy(buf, offset)
|
85 |
|
|
cb(null, buf)
|
86 |
|
|
})
|
87 |
|
|
return
|
88 |
|
|
}
|
89 |
|
|
var bytes = randombytes(size)
|
90 |
|
|
bytes.copy(buf, offset)
|
91 |
|
|
return buf
|
92 |
|
|
}
|
93 |
|
|
function randomFillSync (buf, offset, size) {
|
94 |
|
|
if (typeof offset === 'undefined') {
|
95 |
|
|
offset = 0
|
96 |
|
|
}
|
97 |
|
|
if (!Buffer.isBuffer(buf) && !(buf instanceof global.Uint8Array)) {
|
98 |
|
|
throw new TypeError('"buf" argument must be a Buffer or Uint8Array')
|
99 |
|
|
}
|
100 |
|
|
|
101 |
|
|
assertOffset(offset, buf.length)
|
102 |
|
|
|
103 |
|
|
if (size === undefined) size = buf.length - offset
|
104 |
|
|
|
105 |
|
|
assertSize(size, offset, buf.length)
|
106 |
|
|
|
107 |
|
|
return actualFill(buf, offset, size)
|
108 |
|
|
}
|