mirror of
https://github.com/quantum5/correcthorsebatterystaple.git
synced 2025-04-24 18:21:57 -04:00
78 lines
2.1 KiB
JavaScript
78 lines
2.1 KiB
JavaScript
const words = require('./words')
|
|
const digits = '0123456789'
|
|
const symbols = '-_.,; /`~!@#$%^&*()+=<>?:|[]{}'
|
|
const defaultSymbol = '-'
|
|
|
|
function getWordList (name) {
|
|
if (['small', 'medium', 'large'].includes(name)) {
|
|
return words[name]
|
|
}
|
|
throw new Error(`Invalid word list: ${name}`)
|
|
}
|
|
|
|
// Only works when the maximum possible value in indices is divisible by list.length.
|
|
function getWords (list, indices) {
|
|
return Array.from(indices).map(index => list[index % list.length])
|
|
}
|
|
|
|
function capitalize (string) {
|
|
return string[0].toUpperCase() + string.slice(1)
|
|
}
|
|
|
|
// Only works on power-of-two sized lists.
|
|
function pickWords (list, number) {
|
|
const array = new Uint16Array(number)
|
|
window.crypto.getRandomValues(array)
|
|
return getWords(list, array)
|
|
}
|
|
|
|
// Only works when the maximum possible value of index is divisible by choices.length.
|
|
function getChar (choices, index) {
|
|
return choices[index % choices.length]
|
|
}
|
|
|
|
function pickChar (choices) {
|
|
const array = new Uint32Array(1)
|
|
const maxValid = Math.floor(4294967296 / choices.length) * choices.length;
|
|
do {
|
|
window.crypto.getRandomValues(array)
|
|
} while (array[0] >= maxValid)
|
|
return getChar(choices, array[0])
|
|
}
|
|
|
|
function generate (options) {
|
|
let words = pickWords(getWordList(options.list), options.count)
|
|
|
|
if (options.capitalize) {
|
|
words = words.map(capitalize)
|
|
}
|
|
|
|
if (options.digit) {
|
|
words.push(pickChar(digits))
|
|
}
|
|
|
|
return words.join(options.symbol ? options.separator : '')
|
|
}
|
|
|
|
function lengthBits (list) {
|
|
return Math.log2(list.length)
|
|
}
|
|
|
|
function bitsForSymbol (symbol) {
|
|
return symbol === defaultSymbol ? 1 : lengthBits(symbols)
|
|
}
|
|
|
|
function computeBits (options) {
|
|
const wordBits = lengthBits(getWordList(options.list))
|
|
const capsBits = options.capitalize ? 1 : 0
|
|
const symbolBits = options.symbol ? bitsForSymbol(options.separator) : 0
|
|
const digitBits = options.digit ? lengthBits(digits) : 0
|
|
|
|
return wordBits * options.count + capsBits + symbolBits + digitBits
|
|
}
|
|
|
|
module.exports = {
|
|
getWordList, getWords, capitalize, generate, getChar, lengthBits, computeBits,
|
|
symbols, defaultSymbol,
|
|
}
|