
I did this because they are 2 separate concerns and I have diverged so far from the original repo. I didn't start a new repo because it contains valuable history of all the people who contributed to it. Additionally if there is an issue with this we can revert, but I rather not. Forks are forks.
169 lines
5 KiB
JavaScript
169 lines
5 KiB
JavaScript
var ALPHABET = 'qpzry9x8gf2tvdw0s3jn54khce6mua7l';
|
|
var ALPHABET_MAP = {};
|
|
for (var z = 0; z < ALPHABET.length; z++) {
|
|
var x = ALPHABET.charAt(z);
|
|
ALPHABET_MAP[x] = z;
|
|
}
|
|
function polymodStep(pre) {
|
|
var b = pre >> 25;
|
|
return (((pre & 0x1ffffff) << 5) ^
|
|
(-((b >> 0) & 1) & 0x3b6a57b2) ^
|
|
(-((b >> 1) & 1) & 0x26508e6d) ^
|
|
(-((b >> 2) & 1) & 0x1ea119fa) ^
|
|
(-((b >> 3) & 1) & 0x3d4233dd) ^
|
|
(-((b >> 4) & 1) & 0x2a1462b3));
|
|
}
|
|
function prefixChk(prefix) {
|
|
var chk = 1;
|
|
for (var i = 0; i < prefix.length; ++i) {
|
|
var c = prefix.charCodeAt(i);
|
|
if (c < 33 || c > 126)
|
|
return 'Invalid prefix (' + prefix + ')';
|
|
chk = polymodStep(chk) ^ (c >> 5);
|
|
}
|
|
chk = polymodStep(chk);
|
|
for (var i = 0; i < prefix.length; ++i) {
|
|
var v = prefix.charCodeAt(i);
|
|
chk = polymodStep(chk) ^ (v & 0x1f);
|
|
}
|
|
return chk;
|
|
}
|
|
function convertbits(data, inBits, outBits, pad) {
|
|
var value = 0;
|
|
var bits = 0;
|
|
var maxV = (1 << outBits) - 1;
|
|
var result = [];
|
|
for (var i = 0; i < data.length; ++i) {
|
|
value = (value << inBits) | data[i];
|
|
bits += inBits;
|
|
while (bits >= outBits) {
|
|
bits -= outBits;
|
|
result.push((value >> bits) & maxV);
|
|
}
|
|
}
|
|
if (pad) {
|
|
if (bits > 0) {
|
|
result.push((value << (outBits - bits)) & maxV);
|
|
}
|
|
}
|
|
else {
|
|
if (bits >= inBits)
|
|
return 'Excess padding';
|
|
if ((value << (outBits - bits)) & maxV)
|
|
return 'Non-zero padding';
|
|
}
|
|
return result;
|
|
}
|
|
function toWords(bytes) {
|
|
return convertbits(bytes, 8, 5, true);
|
|
}
|
|
function fromWordsUnsafe(words) {
|
|
var res = convertbits(words, 5, 8, false);
|
|
if (Array.isArray(res))
|
|
return res;
|
|
}
|
|
function fromWords(words) {
|
|
var res = convertbits(words, 5, 8, false);
|
|
if (Array.isArray(res))
|
|
return res;
|
|
throw new Error(res);
|
|
}
|
|
function getLibraryFromEncoding(encoding) {
|
|
var ENCODING_CONST;
|
|
if (encoding === 'bech32') {
|
|
ENCODING_CONST = 1;
|
|
}
|
|
else {
|
|
ENCODING_CONST = 0x2bc830a3;
|
|
}
|
|
function encode(prefix, words, LIMIT) {
|
|
LIMIT = LIMIT || 90;
|
|
if (prefix.length + 7 + words.length > LIMIT)
|
|
throw new TypeError('Exceeds length limit');
|
|
prefix = prefix.toLowerCase();
|
|
// determine chk mod
|
|
var chk = prefixChk(prefix);
|
|
if (typeof chk === 'string')
|
|
throw new Error(chk);
|
|
var result = prefix + '1';
|
|
for (var i = 0; i < words.length; ++i) {
|
|
var x = words[i];
|
|
if (x >> 5 !== 0)
|
|
throw new Error('Non 5-bit word');
|
|
chk = polymodStep(chk) ^ x;
|
|
result += ALPHABET.charAt(x);
|
|
}
|
|
for (var i = 0; i < 6; ++i) {
|
|
chk = polymodStep(chk);
|
|
}
|
|
chk ^= ENCODING_CONST;
|
|
for (var i = 0; i < 6; ++i) {
|
|
var v = (chk >> ((5 - i) * 5)) & 0x1f;
|
|
result += ALPHABET.charAt(v);
|
|
}
|
|
return result;
|
|
}
|
|
function __decode(str, LIMIT) {
|
|
LIMIT = LIMIT || 90;
|
|
if (str.length < 8)
|
|
return str + ' too short';
|
|
if (str.length > LIMIT)
|
|
return 'Exceeds length limit';
|
|
// don't allow mixed case
|
|
var lowered = str.toLowerCase();
|
|
var uppered = str.toUpperCase();
|
|
if (str !== lowered && str !== uppered)
|
|
return 'Mixed-case string ' + str;
|
|
str = lowered;
|
|
var split = str.lastIndexOf('1');
|
|
if (split === -1)
|
|
return 'No separator character for ' + str;
|
|
if (split === 0)
|
|
return 'Missing prefix for ' + str;
|
|
var prefix = str.slice(0, split);
|
|
var wordChars = str.slice(split + 1);
|
|
if (wordChars.length < 6)
|
|
return 'Data too short';
|
|
var chk = prefixChk(prefix);
|
|
if (typeof chk === 'string')
|
|
return chk;
|
|
var words = [];
|
|
for (var i = 0; i < wordChars.length; ++i) {
|
|
var c = wordChars.charAt(i);
|
|
var v = ALPHABET_MAP[c];
|
|
if (v === undefined)
|
|
return 'Unknown character ' + c;
|
|
chk = polymodStep(chk) ^ v;
|
|
// not in the checksum?
|
|
if (i + 6 >= wordChars.length)
|
|
continue;
|
|
words.push(v);
|
|
}
|
|
if (chk !== ENCODING_CONST)
|
|
return 'Invalid checksum for ' + str;
|
|
return { prefix: prefix, words: words };
|
|
}
|
|
function decodeUnsafe(str, LIMIT) {
|
|
var res = __decode(str, LIMIT);
|
|
if (typeof res === 'object')
|
|
return res;
|
|
}
|
|
function decode(str, LIMIT) {
|
|
var res = __decode(str, LIMIT);
|
|
if (typeof res === 'object')
|
|
return res;
|
|
throw new Error(res);
|
|
}
|
|
return {
|
|
decodeUnsafe: decodeUnsafe,
|
|
decode: decode,
|
|
encode: encode,
|
|
toWords: toWords,
|
|
fromWordsUnsafe: fromWordsUnsafe,
|
|
fromWords: fromWords
|
|
};
|
|
}
|
|
|
|
const bech32 = getLibraryFromEncoding('bech32');
|
|
const bech32m = getLibraryFromEncoding('bech32m');
|
|
|