-
Notifications
You must be signed in to change notification settings - Fork 129
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Add support for STX (Blockstack) address (#299)
* Add support for RUNE (ThorChain) * Add support for STX (Blockstack) address * tslint * chore: reduce size
- Loading branch information
Showing
6 changed files
with
215 additions
and
1 deletion.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.
Oops, something went wrong.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,197 @@ | ||
// https://en.wikipedia.org/wiki/Base32#Crockford's_Base32 | ||
import { sha256 } from 'js-sha256'; | ||
export const C32_ALPHABET = '0123456789ABCDEFGHJKMNPQRSTVWXYZ'; | ||
const hex = '0123456789abcdef'; | ||
|
||
function hashSha256(data: Buffer): Buffer { | ||
return Buffer.from(sha256.update(data).digest()) | ||
} | ||
|
||
function c32checksum(dataHex: string): string { | ||
const dataHash = hashSha256(hashSha256(Buffer.from(dataHex, 'hex'))); | ||
const checksum = dataHash.slice(0, 4).toString('hex'); | ||
return checksum; | ||
} | ||
|
||
export function c32checkEncode(data: Buffer): string { | ||
const dataHex = data.toString('hex'); | ||
let hash160hex = dataHex.substring(0, dataHex.length - 8); | ||
if (!hash160hex.match(/^[0-9a-fA-F]{40}$/)) { | ||
throw new Error('Invalid argument: not a hash160 hex string'); | ||
} | ||
|
||
hash160hex = hash160hex.toLowerCase(); | ||
if (hash160hex.length % 2 !== 0) { | ||
hash160hex = `0${hash160hex}`; | ||
} | ||
|
||
// p2pkh: 'P' | ||
// p2sh: 'M' | ||
const version = { p2pkh: 22, p2sh: 20 }; | ||
|
||
const checksumHex = dataHex.slice(-8); | ||
let c32str = ''; | ||
let prefix = ''; | ||
|
||
if (checksumHex === c32checksum(`${version.p2pkh.toString(16)}${hash160hex}`)) { | ||
prefix = 'P'; | ||
c32str = c32encode(`${hash160hex}${checksumHex}`); | ||
} else if ((checksumHex === c32checksum(`${version.p2sh.toString(16)}${hash160hex}`))) { | ||
prefix = 'M'; | ||
c32str = c32encode(`${hash160hex}${checksumHex}`); | ||
} | ||
|
||
return `S${prefix}${c32str}`; | ||
} | ||
|
||
function c32encode(inputHex: string): string { | ||
// must be hex | ||
if (!inputHex.match(/^[0-9a-fA-F]*$/)) { | ||
throw new Error('Not a hex-encoded string'); | ||
} | ||
|
||
if (inputHex.length % 2 !== 0) { | ||
inputHex = `0${inputHex}`; | ||
} | ||
|
||
inputHex = inputHex.toLowerCase(); | ||
|
||
let res = []; | ||
let carry = 0; | ||
for (let i = inputHex.length - 1; i >= 0; i--) { | ||
if (carry < 4) { | ||
// tslint:disable-next-line:no-bitwise | ||
const currentCode = hex.indexOf(inputHex[i]) >> carry; | ||
let nextCode = 0; | ||
if (i !== 0) { | ||
nextCode = hex.indexOf(inputHex[i - 1]); | ||
} | ||
// carry = 0, nextBits is 1, carry = 1, nextBits is 2 | ||
const nextBits = 1 + carry; | ||
// tslint:disable-next-line:no-bitwise | ||
const nextLowBits = nextCode % (1 << nextBits) << (5 - nextBits); | ||
const curC32Digit = C32_ALPHABET[currentCode + nextLowBits]; | ||
carry = nextBits; | ||
res.unshift(curC32Digit); | ||
} else { | ||
carry = 0; | ||
} | ||
} | ||
|
||
let C32leadingZeros = 0; | ||
// tslint:disable-next-line:prefer-for-of | ||
for (let i = 0; i < res.length; i++) { | ||
if (res[i] !== '0') { | ||
break; | ||
} else { | ||
C32leadingZeros++; | ||
} | ||
} | ||
|
||
res = res.slice(C32leadingZeros); | ||
|
||
const zeroPrefix = Buffer.from(inputHex, 'hex') | ||
.toString() | ||
.match(/^\u0000*/); | ||
const numLeadingZeroBytesInHex = zeroPrefix ? zeroPrefix[0].length : 0; | ||
|
||
for (let i = 0; i < numLeadingZeroBytesInHex; i++) { | ||
res.unshift(C32_ALPHABET[0]); | ||
} | ||
|
||
return res.join(''); | ||
} | ||
|
||
function c32normalize(c32input: string): string { | ||
// must be upper-case | ||
// replace all O's with 0's | ||
// replace all I's and L's with 1's | ||
return c32input.toUpperCase().replace(/O/g, '0').replace(/[IL]/g, '1'); | ||
} | ||
|
||
export function c32checkDecode(data: string): Buffer { | ||
if (data.length <= 5) { | ||
throw new Error('Invalid c32 address: invalid length'); | ||
} | ||
if (data[0] !== 'S') { | ||
throw new Error('Invalid c32 address: must start with "S"'); | ||
} | ||
|
||
const c32data = c32normalize(data.slice(1)); | ||
const versionChar = c32data[0]; | ||
const version = C32_ALPHABET.indexOf(versionChar); | ||
|
||
let versionHex = version.toString(16); | ||
if (versionHex.length === 1) { | ||
versionHex = `0${versionHex}`; | ||
} | ||
|
||
const dataHex = c32decode(c32data.slice(1)); | ||
const checksum = dataHex.slice(-8); | ||
|
||
if (c32checksum(`${versionHex}${dataHex.substring(0, dataHex.length - 8)}`) !== checksum) { | ||
throw new Error('Invalid c32check string: checksum mismatch'); | ||
} | ||
|
||
return Buffer.from(dataHex, 'hex'); | ||
} | ||
|
||
function c32decode(c32input: string): string { | ||
c32input = c32normalize(c32input); | ||
|
||
// must result in a c32 string | ||
if (!c32input.match(`^[${C32_ALPHABET}]*$`)) { | ||
throw new Error('Not a c32-encoded string'); | ||
} | ||
|
||
const zeroPrefix = c32input.match(`^${C32_ALPHABET[0]}*`); | ||
const numLeadingZeroBytes = zeroPrefix ? zeroPrefix[0].length : 0; | ||
|
||
let res = []; | ||
let carry = 0; | ||
let carryBits = 0; | ||
for (let i = c32input.length - 1; i >= 0; i--) { | ||
if (carryBits === 4) { | ||
res.unshift(hex[carry]); | ||
carryBits = 0; | ||
carry = 0; | ||
} | ||
// tslint:disable-next-line:no-bitwise | ||
const currentCode = C32_ALPHABET.indexOf(c32input[i]) << carryBits; | ||
const currentValue = currentCode + carry; | ||
const currentHexDigit = hex[currentValue % 16]; | ||
carryBits += 1; | ||
// tslint:disable-next-line:no-bitwise | ||
carry = currentValue >> 4; | ||
// tslint:disable-next-line:no-bitwise | ||
if (carry > 1 << carryBits) { | ||
throw new Error('Panic error in decoding.'); | ||
} | ||
res.unshift(currentHexDigit); | ||
} | ||
// one last carry | ||
res.unshift(hex[carry]); | ||
|
||
if (res.length % 2 === 1) { | ||
res.unshift('0'); | ||
} | ||
|
||
let hexLeadingZeros = 0; | ||
// tslint:disable-next-line:prefer-for-of | ||
for (let i = 0; i < res.length; i++) { | ||
if (res[i] !== '0') { | ||
break; | ||
} else { | ||
hexLeadingZeros++; | ||
} | ||
} | ||
|
||
res = res.slice(hexLeadingZeros - (hexLeadingZeros % 2)); | ||
|
||
let hexStr = res.join(''); | ||
for (let i = 0; i < numLeadingZeroBytes; i++) { | ||
hexStr = `00${hexStr}`; | ||
} | ||
|
||
return hexStr; | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters