How to validate HD wallet address to match BIP44

1.1k Views Asked by At

I'm generating an bitcoin address with NodeJS. I started with this incomplete code.

[https://medium.com/bitcraft/so-you-want-to-build-a-bitcoin-hd-wallet-part-1-8b27aa001ac3][1]

Down below you will find my version.

The Issue

I end up with an address that is to long. I am also uncertain about the root three and checksum concat. Uncertain because I had to deviate from the example and I can't explain it yet.

My goal is to create an HD Wallet starting with Bitcoin but is ultimately for more and or other cryptos.

To validate the value I use this app;

[https://iancoleman.io/bip39/][2]

I manage to come up with the same bip32 key by removing the last 0 in the root three

I started with

hdkey.derive("m/44'/0'/0'/0/0")

And even the derived key seamed valid, I couldn't match it in the BIP44 tab in the validator. So I decided to remove that last 0 because i noticed it wasn't an option in the validator.

hdkey.derive("m/44'/0'/0'/0")

With that last 0 gone, both of the BIP32 extended private and public key are valid. But i still end up with an address that is to long.

This is my code;

var bip39 = require('bip39');
var HDkey = require('hdkey');
var createHash = require('create-hash');
var bs58check = require('bs58check');
        
        exports.seed = async () =>{
          return new Promise(resolve => {
            const mnemonic =  "drive captain sustain winner neutral anchor congress skirt buzz usage orient wood"
            
            //const mnemonic =  bip39.generateMnemonic(); //generates the string above
            const seed =  bip39.mnemonicToSeed(mnemonic); //creates seed buffer
            resolve(seed)
          })
        }
    
       
    
     exports.key = async (seed) =>{
      return new Promise(resolve => {
    
        const hdkey = HDkey.fromMasterSeed(Buffer.from(seed, 'hex'))
    
        const masterPrivateKey = hdkey.privateExtendedKey
    
         // This key Match iancoleman BIP32 ROOT KEY
        console.log("\x1b[32m%s\x1b[0m",'PRIVATE KEY BIP32 ROOT: ', masterPrivateKey) 
    
        // this line will not give me valid BIP32 EXTENDED PRIVATE KEYS
        //const addrnode = hdkey.derive("m/44'/0'/0'/0/0")
        
        // This one will when I removed the last /0
        const addrnode = hdkey.derive("m/44'/0'/0'/0")
        
        // THESE 2 BIP32 Extended Key are valid on iancoleman's app
        console.log("\x1b[32m%s\x1b[0m",'PRIVATE EXTENDED : ', addrnode.privateExtendedKey)
        const step1 = addrnode.publicExtendedKey
        console.log("\x1b[32m%s\x1b[0m",'PUBLIC EXTENDED : ', step1.toString('hex'))
    
        // Here is what I could understand from the example
    
        //SHA256 of the public key
        const step2 = createHash('sha256').update(step1).digest()
      
        // PIPEDMD-160 of the SHA256 Hash
        const step3 = createHash('rmd160').update(step2).digest()
        
    
        // He we must add the network byte in front of that PIPEDMD result
        // 0x00 for mainnet and 0x6f for testnet
        var step4 = Buffer.allocUnsafe(21)
        step4.writeUInt8(0x00, 0)
        step3.copy(step4,1) 
    
        //step3 now holds the  Network ID +  RIPEMD160 result
        //we hash it twice
        var step5 = createHash('sha256').update(step4).digest()
        var step6 = createHash('sha256').update(step5).digest()
    
    
        //checksum first 4 byte of second hash
        var step7 = step6.slice(0,4)
        console.log("\x1b[32m%s\x1b[0m",'CHECKSUM : ', step7.toString('hex'))
    
       // Adding the checksum to the end of
        var step8 = Buffer.concat([step4, step7]);  
        console.log("\x1b[32m%s\x1b[0m",'Base + CHECKSUM : ', step8.toString('hex'))
        // Return the bitcoins address
        var step9 = bs58check.encode(step8)
    
    
       resolve(step9)
    
        // The address generated by this code;
        // 1WpGHR9UmDm7UiJuFu1H3zE7TmtK187D1yZStMQ
    
        // The address generated with the same mnemonic on iancoleman 
        // 1HaGGkWmUDTKExFxyXaiGHHPxYNaCefQrj
      })
    }
1

There are 1 best solutions below

1
On

There are two issues in your sample.

  • In step 1 you must use _publicKey instead of publicExtendedKey; and if need to have PUBLIC EXTENDED you must use a new variable for that.

  • To generate address you don't need to step 5-8.