UNPKG

bech32

Version:
171 lines (170 loc) 5.23 kB
'use strict'; Object.defineProperty(exports, "__esModule", { value: true }); exports.bech32m = exports.bech32 = void 0; const ALPHABET = 'qpzry9x8gf2tvdw0s3jn54khce6mua7l'; const ALPHABET_MAP = {}; for (let z = 0; z < ALPHABET.length; z++) { const x = ALPHABET.charAt(z); ALPHABET_MAP[x] = z; } function polymodStep(pre) { const 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) { let chk = 1; for (let i = 0; i < prefix.length; ++i) { const c = prefix.charCodeAt(i); if (c < 33 || c > 126) return 'Invalid prefix (' + prefix + ')'; chk = polymodStep(chk) ^ (c >> 5); } chk = polymodStep(chk); for (let i = 0; i < prefix.length; ++i) { const v = prefix.charCodeAt(i); chk = polymodStep(chk) ^ (v & 0x1f); } return chk; } function convert(data, inBits, outBits, pad) { let value = 0; let bits = 0; const maxV = (1 << outBits) - 1; const result = []; for (let 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 convert(bytes, 8, 5, true); } function fromWordsUnsafe(words) { const res = convert(words, 5, 8, false); if (Array.isArray(res)) return res; } function fromWords(words) { const res = convert(words, 5, 8, false); if (Array.isArray(res)) return res; throw new Error(res); } function getLibraryFromEncoding(encoding) { let 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 let chk = prefixChk(prefix); if (typeof chk === 'string') throw new Error(chk); let result = prefix + '1'; for (let i = 0; i < words.length; ++i) { const x = words[i]; if (x >> 5 !== 0) throw new Error('Non 5-bit word'); chk = polymodStep(chk) ^ x; result += ALPHABET.charAt(x); } for (let i = 0; i < 6; ++i) { chk = polymodStep(chk); } chk ^= ENCODING_CONST; for (let i = 0; i < 6; ++i) { const 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 const lowered = str.toLowerCase(); const uppered = str.toUpperCase(); if (str !== lowered && str !== uppered) return 'Mixed-case string ' + str; str = lowered; const split = str.lastIndexOf('1'); if (split === -1) return 'No separator character for ' + str; if (split === 0) return 'Missing prefix for ' + str; const prefix = str.slice(0, split); const wordChars = str.slice(split + 1); if (wordChars.length < 6) return 'Data too short'; let chk = prefixChk(prefix); if (typeof chk === 'string') return chk; const words = []; for (let i = 0; i < wordChars.length; ++i) { const c = wordChars.charAt(i); const 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, words }; } function decodeUnsafe(str, LIMIT) { const res = __decode(str, LIMIT); if (typeof res === 'object') return res; } function decode(str, LIMIT) { const res = __decode(str, LIMIT); if (typeof res === 'object') return res; throw new Error(res); } return { decodeUnsafe, decode, encode, toWords, fromWordsUnsafe, fromWords, }; } exports.bech32 = getLibraryFromEncoding('bech32'); exports.bech32m = getLibraryFromEncoding('bech32m');