UNPKG

multichain-address-validator

Version:

Multichain address validator for Bitcoin and other blockchains.

69 lines (68 loc) 2.52 kB
import base58 from '../crypto/base58.js'; import segwit from '../crypto/segwit_addr.js'; import cryptoUtils from '../crypto/utils.js'; import { getAddress } from '../helpers.js'; import { Buffer } from 'buffer'; function getDecoded(address) { try { return base58.decode(address); } catch (e) { // if decoding fails, assume invalid address return null; } } function getChecksum(hashFunction, payload) { // Each currency may implement different hashing algorithm switch (hashFunction) { // blake then keccak hash chain case 'blake256keccak256': const blake = cryptoUtils.blake2b256(payload); return cryptoUtils.keccak256Checksum(Buffer.from(blake, 'hex')); case 'blake256': return cryptoUtils.blake256Checksum(payload); case 'keccak256': return cryptoUtils.keccak256Checksum(payload); case 'sha256': default: return cryptoUtils.sha256Checksum(payload); } } function getAddressType(address, opts) { // should be 25 bytes per btc address spec and 26 decred const expectedLength = opts.expectedLength || 25; const hashFunction = opts.hashFunction || 'sha256'; const decoded = getDecoded(address); if (decoded) { const length = decoded.length; if (length !== expectedLength) { return null; } if (opts.regex) { if (!opts.regex.test(address)) { return false; } } const checksum = cryptoUtils.toHex(decoded.slice(length - 4, length)), body = cryptoUtils.toHex(decoded.slice(0, length - 4)), goodChecksum = getChecksum(hashFunction, body); return checksum === goodChecksum ? cryptoUtils.toHex(decoded.slice(0, expectedLength - 24)) : null; } return null; } function isValidP2PKHandP2SHAddress(address, opts) { const addressType = getAddressType(address, opts); if (addressType) { return opts.addressTypes.indexOf(addressType) >= 0; } return false; } // const DefaultBTCValidatorOpts: BTCValidatorOpts = { // addressTypes: {mainnet: ['00', '05'], testnet: ['6f', 'c4', '3c', '26']}, // bech32Hrp: {mainnet: ['bc'], testnet: ['tb']}, // } export default (opts) => ({ isValidAddress(address) { const addr = getAddress(address); // const _opts = {...DefaultBTCValidatorOpts, ...opts} return isValidP2PKHandP2SHAddress(addr, opts) || segwit.isValidAddress(addr, opts); } });