Class: Tapyrus::ExtPubkey

Inherits:
Object
  • Object
show all
Includes:
HexConverter
Defined in:
lib/tapyrus/ext_key.rb

Overview

BIP-32 Extended public key

Instance Attribute Summary collapse

Class Method Summary collapse

Instance Method Summary collapse

Methods included from HexConverter

#to_hex

Instance Attribute Details

#chain_codeObject

Returns the value of attribute chain_code.



214
215
216
# File 'lib/tapyrus/ext_key.rb', line 214

def chain_code
  @chain_code
end

#depthObject

Returns the value of attribute depth.



212
213
214
# File 'lib/tapyrus/ext_key.rb', line 212

def depth
  @depth
end

#numberObject

Returns the value of attribute number.



213
214
215
# File 'lib/tapyrus/ext_key.rb', line 213

def number
  @number
end

#parent_fingerprintObject

Returns the value of attribute parent_fingerprint.



216
217
218
# File 'lib/tapyrus/ext_key.rb', line 216

def parent_fingerprint
  @parent_fingerprint
end

#pubkeyObject

hex format



215
216
217
# File 'lib/tapyrus/ext_key.rb', line 215

def pubkey
  @pubkey
end

#verObject

Returns the value of attribute ver.



211
212
213
# File 'lib/tapyrus/ext_key.rb', line 211

def ver
  @ver
end

Class Method Details

.encode_base58(hex) ⇒ String

Generate Base58 encoded key from BIP32 payload with hex format.

Parameters:

  • hex (String)

    BIP32 payload with hex format.

Returns:

  • (String)

    Base58 encoded extended key.



260
261
262
# File 'lib/tapyrus/ext_key.rb', line 260

def self.encode_base58(hex)
  Base58.encode(hex + Tapyrus.calc_checksum(hex))
end

.from_base58(address) ⇒ Object

import pub key from Base58 private key address



335
336
337
# File 'lib/tapyrus/ext_key.rb', line 335

def self.from_base58(address)
  ExtPubkey.parse_from_payload(ExtPubkey.validate_checksum(address))
end

.parse_from_payload(payload) ⇒ Object

Raises:

  • (ArgumentError)


312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
# File 'lib/tapyrus/ext_key.rb', line 312

def self.parse_from_payload(payload)
  buf = StringIO.new(payload)
  ext_pubkey = ExtPubkey.new
  ext_pubkey.ver = buf.read(4).bth # version
  raise ArgumentError, Errors::Messages::INVALID_BIP32_VERSION unless ExtPubkey.support_version?(ext_pubkey.ver)
  ext_pubkey.depth = buf.read(1).unpack("C").first
  ext_pubkey.parent_fingerprint = buf.read(4).bth
  ext_pubkey.number = buf.read(4).unpack("N").first
  if ext_pubkey.depth == 0
    unless ext_pubkey.parent_fingerprint == ExtKey::MASTER_FINGERPRINT
      raise ArgumentError, Errors::Messages::INVALID_BIP32_FINGERPRINT
    end
    raise ArgumentError, Errors::Messages::INVALID_BIP32_ZERO_INDEX if ext_pubkey.number > 0
  end
  if ext_pubkey.parent_fingerprint == ExtKey::MASTER_FINGERPRINT && ext_pubkey.depth > 0
    raise ArgumentError, Errors::Messages::INVALID_BIP32_ZERO_DEPTH
  end
  ext_pubkey.chain_code = buf.read(32)
  ext_pubkey.pubkey = Tapyrus::Key.new(pubkey: buf.read(33).bth).pubkey
  ext_pubkey
end

.support_version?(version) ⇒ Boolean

check whether version is supported version bytes.

Returns:

  • (Boolean)


362
363
364
365
# File 'lib/tapyrus/ext_key.rb', line 362

def self.support_version?(version)
  p = Tapyrus.chain_params
  [p.bip49_pubkey_p2wpkh_p2sh_version, p.bip84_pubkey_p2wpkh_version, p.extended_pubkey_version].include?(version)
end

.validate_checksum(base58) ⇒ String

Validate address checksum and return payload.

Parameters:

  • BIP32 (String)

    Base58 address

Returns:

  • (String)

    BIP32 payload with binary format

Raises:

  • (ArgumentError)


342
343
344
345
346
# File 'lib/tapyrus/ext_key.rb', line 342

def self.validate_checksum(base58)
  raw = Base58.decode(base58)
  raise ArgumentError, Errors::Messages::INVALID_CHECKSUM unless Tapyrus.calc_checksum(raw[0...-8]) == raw[-8..-1]
  raw[0...-8].htb
end

.version_from_purpose(purpose) ⇒ Object

get version bytes from purpose’ value.



349
350
351
352
353
354
355
356
357
358
359
# File 'lib/tapyrus/ext_key.rb', line 349

def self.version_from_purpose(purpose)
  v = purpose - Tapyrus::HARDENED_THRESHOLD
  case v
  when 49
    Tapyrus.chain_params.bip49_pubkey_p2wpkh_p2sh_version
  when 84
    Tapyrus.chain_params.bip84_pubkey_p2wpkh_version
  else
    Tapyrus.chain_params.extended_pubkey_version
  end
end

Instance Method Details

#==(other) ⇒ Object



308
309
310
# File 'lib/tapyrus/ext_key.rb', line 308

def ==(other)
  to_payload == other.to_payload
end

#addrObject

get address



232
233
234
# File 'lib/tapyrus/ext_key.rb', line 232

def addr
  key.to_p2pkh
end

#derive(number) ⇒ Object

derive child key



270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
# File 'lib/tapyrus/ext_key.rb', line 270

def derive(number)
  new_key = ExtPubkey.new
  new_key.depth = depth + 1
  new_key.number = number
  new_key.parent_fingerprint = fingerprint
  raise "hardened key is not support" if number > (Tapyrus::HARDENED_THRESHOLD - 1)
  data = pub.htb << [number].pack("N")
  l = Tapyrus.hmac_sha512(chain_code, data)
  left = l[0..31].bth.to_i(16)
  raise "invalid key" if left >= CURVE_ORDER
  l_priv = ECDSA::Format::IntegerOctetString.encode(left, 32)
  p1 = Tapyrus::Key.new(priv_key: l_priv.bth, key_type: Tapyrus::Key::TYPES[:uncompressed]).to_point
  p2 = Tapyrus::Key.new(pubkey: pubkey, key_type: key_type).to_point
  new_key.pubkey = (p1 + p2).to_hex
  new_key.chain_code = l[32..-1]
  new_key.ver = version
  new_key
end

#fingerprintObject

get fingerprint



248
249
250
# File 'lib/tapyrus/ext_key.rb', line 248

def fingerprint
  identifier.slice(0..7)
end

#hardened?Boolean

whether hardened key.

Returns:

  • (Boolean)


265
266
267
# File 'lib/tapyrus/ext_key.rb', line 265

def hardened?
  number >= Tapyrus::HARDENED_THRESHOLD
end

#hash160Object



227
228
229
# File 'lib/tapyrus/ext_key.rb', line 227

def hash160
  Tapyrus.hash160(pub)
end

#identifierObject

get key identifier



243
244
245
# File 'lib/tapyrus/ext_key.rb', line 243

def identifier
  Tapyrus.hash160(pub)
end

#keyTapyrus::Key

get key object

Returns:



238
239
240
# File 'lib/tapyrus/ext_key.rb', line 238

def key
  Tapyrus::Key.new(pubkey: pubkey, key_type: key_type)
end

#key_typeObject

get key type defined by BIP-178 using version.



296
297
298
299
300
301
302
303
304
305
306
# File 'lib/tapyrus/ext_key.rb', line 296

def key_type
  v = version
  case v
  when Tapyrus.chain_params.bip49_pubkey_p2wpkh_p2sh_version
    Tapyrus::Key::TYPES[:p2wpkh_p2sh]
  when Tapyrus.chain_params.bip84_pubkey_p2wpkh_version
    Tapyrus::Key::TYPES[:p2wpkh]
  when Tapyrus.chain_params.extended_pubkey_version
    Tapyrus::Key::TYPES[:compressed]
  end
end

#master?Boolean

Returns:

  • (Boolean)


367
368
369
# File 'lib/tapyrus/ext_key.rb', line 367

def master?
  depth == 0 && number == 0 && parent_fingerprint == "00000000"
end

#pubObject



223
224
225
# File 'lib/tapyrus/ext_key.rb', line 223

def pub
  pubkey
end

#to_base58Object

Base58 encoded extended pubkey



253
254
255
# File 'lib/tapyrus/ext_key.rb', line 253

def to_base58
  ExtPubkey.encode_base58(to_hex)
end

#to_payloadObject

serialize extended pubkey



219
220
221
# File 'lib/tapyrus/ext_key.rb', line 219

def to_payload
  version.htb << [depth].pack("C") << parent_fingerprint.htb << [number].pack("N") << chain_code << pub.htb
end

#versionObject

get version bytes using serialization format



290
291
292
293
# File 'lib/tapyrus/ext_key.rb', line 290

def version
  return ExtPubkey.version_from_purpose(number) if depth == 1
  ver ? ver : Tapyrus.chain_params.extended_pubkey_version
end