Skip to content

randomshinichi/bip32

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 

History

2 Commits
 
 
 
 
 
 
 
 
 
 
 
 

Repository files navigation

go-bip32

An implementation of the BIP32 spec for Hierarchical Deterministic Bitcoin addresses as a simple Go library. The semantics of derived keys are up to the user. BIP43 and BIP44 are good schemes to implement with this library. An additional library for either or both of those on top of this library should be developed.

Modified for use in Skycoin, derived from https://github.com/tyler-smith/go-bip32

Example

It's very unlikely, but possible, that a given index does not produce a valid private key. Error checking is skipped in this example for brevity but should be handled in real code. In such a case, an error of type Error is returned and it's method ImpossibleChild() will return true.

An example for handling this:

func tryNewPrivateKeyFromPath() (*bip32.PrivateKey, error) {
	k, err := bip32.NewPrivateKeyFromPath([]byte("abc123"), "m/1'/1'/0/1")
	if err != nil {
		if bip32.IsImpossibleChildError(err) {
			fmt.Println("Child number 1 generated an invalid key, use the next child number")
		}
		return err
	}

	return k, nil
}

Any valid private key will have a valid public key so that PrivateKey.PublicKey() method never returns an error.

package main

import (
	"github.com/skycoin/skycoin/src/cipher/bip32"
	"github.com/skycoin/skycoin/src/cipher/bip39"
	"fmt"
	"log"
)

// Example address creation for a fictitious company ComputerVoice Inc. where
// each department has their own wallet to manage
func main(){
	// Generate a mnemonic to determine all keys from. Don't lose it.
	mnemonic := bip39.MustNewDefaultMnemonic()

	// Derivce a seed from the mnemonic
	seed, err := bip39.NewSeed(mnemonic, "")
	if err != nil {
		log.Fatalln("Error generating seed:", err)
	}

	// Create master private key from seed
	computerVoiceMasterKey, err := bip32.NewMasterKey(seed)
	if err != nil {
		log.Fatalln("NewMasterKey failed:", err)
	}

	// Map departments to keys
	// There is a very small chance a given child index is invalid (less than 1 in 2^127)
	// If so your real program should handle this by skipping the index
	departmentKeys := map[string]*bip32.Key{}
	departmentKeys["Sales"], err = computerVoiceMasterKey.NewChildKey(0)
	if err != nil {
		log.Fatalln("NewChildKey:", err)
	}
	departmentKeys["Marketing"], err = computerVoiceMasterKey.NewChildKey(1)
	if err != nil {
		log.Fatalln("NewChildKey:", err)
	}
	departmentKeys["Engineering"], err = computerVoiceMasterKey.NewChildKey(2)
	if err != nil {
		log.Fatalln("NewChildKey:", err)
	}
	departmentKeys["Customer Support"], err = computerVoiceMasterKey.NewChildKey(3)
	if err != nil {
		log.Fatalln("NewChildKey:", err)
	}

	// Create public keys for record keeping, auditors, payroll, etc
	departmentAuditKeys := map[string]*bip32.Key{}
	departmentAuditKeys["Sales"] = departmentKeys["Sales"].PublicKey()
	departmentAuditKeys["Marketing"] = departmentKeys["Marketing"].PublicKey()
	departmentAuditKeys["Engineering"] = departmentKeys["Engineering"].PublicKey()
	departmentAuditKeys["Customer Support"] = departmentKeys["Customer Support"].PublicKey()

	// Print public keys
	for department, pubKey := range departmentAuditKeys {
		fmt.Println(department, pubKey)
	}
}

Thanks

The developers at Factom have contributed a lot to this library and have made many great improvements to it. Please check out their project(s) and give them a thanks if you use this library.

Thanks to bartekn from Stellar for some important bug catches.

About

Skycoin's fork of tyler-smith's go-bip32 with path derivation, customized for ed25519 curve

Resources

License

Stars

Watchers

Forks

Releases

No releases published

Packages

 
 
 

Contributors

Languages