diff --git a/go/Makefile b/go/Makefile index 46be00b62c6..e2207c2e1a4 100644 --- a/go/Makefile +++ b/go/Makefile @@ -42,6 +42,7 @@ build-helpers: $(test-helpers) # List of test vectors to generate. test-vectors-targets := staking/gen_vectors \ + staking/gen_account_vectors \ registry/gen_vectors \ governance/gen_vectors @@ -49,6 +50,11 @@ $(test-vectors-targets): @$(ECHO) "$(MAGENTA)*** Generating test vectors ($@)...$(OFF)" @$(GO) run ./$@ +# Extended set of staking/gen_account_vectors test vectors. +staking/gen_account_vectors_extended: + @$(ECHO) "$(MAGENTA)*** Generating test vectors ($@)...$(OFF)" + @$(GO) run ./staking/gen_account_vectors -extended + # Format code. fmt: @$(ECHO) "$(CYAN)*** Running Go formatters...$(OFF)" diff --git a/go/staking/gen_account_vectors/main.go b/go/staking/gen_account_vectors/main.go new file mode 100644 index 00000000000..c0800d9b5ea --- /dev/null +++ b/go/staking/gen_account_vectors/main.go @@ -0,0 +1,117 @@ +// gen_account_vectors generates test vectors for ADR 0008: Standard Account Key +// Generation. +package main + +import ( + "encoding/hex" + "encoding/json" + "flag" + "fmt" + "os" + + bip39 "github.com/tyler-smith/go-bip39" + + "github.com/oasisprotocol/oasis-core/go/common/crypto/sakg" + "github.com/oasisprotocol/oasis-core/go/common/crypto/signature" + staking "github.com/oasisprotocol/oasis-core/go/staking/api" +) + +// Custom byte slice type that encodes to a hex string when marshaling to JSON. +type byteSlice []byte + +func (bs byteSlice) MarshalJSON() ([]byte, error) { + return json.Marshal(hex.EncodeToString(bs)) +} + +// oasisAccount contains information for a particular Oasis staking account: +// BIP-0032 path, private and public keys and the account's address. +type oasisAccount struct { + BIP32Path sakg.BIP32Path `json:"bip32_path"` + PrivateKey byteSlice `json:"private_key"` + PublicKey byteSlice `json:"public_key"` + Address staking.Address `json:"address"` +} + +type testVector struct { + Kind string `json:"kind"` + BIP39Mnemonic string `json:"bip39_mnemonic"` + BIP39PassPhrase string `json:"bip39_passphrase"` + BIP39Seed byteSlice `json:"bip39_seed"` + OasisAccounts []oasisAccount `json:"oasis_accounts"` +} + +func main() { + extendedSet := flag.Bool( + "extended", + false, + "Generate extended set of ADR 0008 (Standard Account Key Generation) test vectors", + ) + flag.Parse() + + // BIP-0039 passphrases. + passphrases := []string{ + "", + } + // BIP-0039 mnemonics. + mnemonics := []string{ + // The popular "standard" mnemonic used for testing. + "abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon abandon about", + // Oasis Ledger app's development mnemonic. + "equip will roof matter pink blind book anxiety banner elbow sun young", + } + // Add additional BIP-0039 passphrases and mnemonics when generating the + // extended set of ADR 0008 test vectors. + if *extendedSet { + passphrases = append(passphrases, "p4ssphr4se") + mnemonics = append(mnemonics, []string{ + // Randomly chosen mnemonics. + "network want title crash employ twelve elegant unlock arrow marriage cat april", + "grape weapon help home entire invest warfare curious advance immense group vintage", + "vibrant length learn peanut garlic boil battle jeans fan cliff alone round setup dove shoe twist dumb trophy imitate coil team grocery when else", + }...) + } + + // Generate ADR 0008 test vectors. + var vectors []testVector + for _, passphrase := range passphrases { + for _, mnemonic := range mnemonics { + // NOTE: This is only for test vector's BIP39Seed field, + // GetAccountKey() computes the BIP-0039 seed automatically. + seed := bip39.NewSeed(mnemonic, passphrase) + + nAccounts := uint32(10) + accounts := make([]oasisAccount, nAccounts) + for number := uint32(0); number < nAccounts; number++ { + signer, path, err := sakg.GetAccountSigner(mnemonic, passphrase, number) + if err != nil { + panic(fmt.Sprintf("error obtaining account key: %v", err)) + } + pubKey := signer.Public() + pubKeyBytes, _ := pubKey.MarshalBinary() + unsafeSigner := signer.(signature.UnsafeSigner) + privKey := unsafeSigner.UnsafeBytes() + accounts[number] = oasisAccount{ + BIP32Path: path, + PrivateKey: byteSlice(privKey), + PublicKey: byteSlice(pubKeyBytes), + Address: staking.NewAddress(pubKey), + } + } + vector := testVector{ + Kind: "standard account key generation", + BIP39Mnemonic: mnemonic, + BIP39PassPhrase: passphrase, + BIP39Seed: seed, + OasisAccounts: accounts, + } + vectors = append(vectors, vector) + } + } + + // Generate output. + jsonOut, err := json.MarshalIndent(&vectors, "", " ") + if err != nil { + fmt.Fprintf(os.Stderr, "Error encoding test vectors: %v\n", err) + } + fmt.Printf("%s", jsonOut) +}