github.com/ethw3/go-ethereuma@v0.0.0-20221013053120-c14602a4c23c/accounts/hd_test.go (about) 1 // Copyright 2017 The go-ethereum Authors 2 // This file is part of the go-ethereum library. 3 // 4 // The go-ethereum library is free software: you can redistribute it and/or modify 5 // it under the terms of the GNU Lesser General Public License as published by 6 // the Free Software Foundation, either version 3 of the License, or 7 // (at your option) any later version. 8 // 9 // The go-ethereum library is distributed in the hope that it will be useful, 10 // but WITHOUT ANY WARRANTY; without even the implied warranty of 11 // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the 12 // GNU Lesser General Public License for more details. 13 // 14 // You should have received a copy of the GNU Lesser General Public License 15 // along with the go-ethereum library. If not, see <http://www.gnu.org/licenses/>. 16 17 package accounts 18 19 import ( 20 "fmt" 21 "reflect" 22 "testing" 23 ) 24 25 // Tests that HD derivation paths can be correctly parsed into our internal binary 26 // representation. 27 func TestHDPathParsing(t *testing.T) { 28 tests := []struct { 29 input string 30 output DerivationPath 31 }{ 32 // Plain absolute derivation paths 33 {"m/44'/60'/0'/0", DerivationPath{0x80000000 + 44, 0x80000000 + 60, 0x80000000 + 0, 0}}, 34 {"m/44'/60'/0'/128", DerivationPath{0x80000000 + 44, 0x80000000 + 60, 0x80000000 + 0, 128}}, 35 {"m/44'/60'/0'/0'", DerivationPath{0x80000000 + 44, 0x80000000 + 60, 0x80000000 + 0, 0x80000000 + 0}}, 36 {"m/44'/60'/0'/128'", DerivationPath{0x80000000 + 44, 0x80000000 + 60, 0x80000000 + 0, 0x80000000 + 128}}, 37 {"m/2147483692/2147483708/2147483648/0", DerivationPath{0x80000000 + 44, 0x80000000 + 60, 0x80000000 + 0, 0}}, 38 {"m/2147483692/2147483708/2147483648/2147483648", DerivationPath{0x80000000 + 44, 0x80000000 + 60, 0x80000000 + 0, 0x80000000 + 0}}, 39 40 // Plain relative derivation paths 41 {"0", DerivationPath{0x80000000 + 44, 0x80000000 + 60, 0x80000000 + 0, 0, 0}}, 42 {"128", DerivationPath{0x80000000 + 44, 0x80000000 + 60, 0x80000000 + 0, 0, 128}}, 43 {"0'", DerivationPath{0x80000000 + 44, 0x80000000 + 60, 0x80000000 + 0, 0, 0x80000000 + 0}}, 44 {"128'", DerivationPath{0x80000000 + 44, 0x80000000 + 60, 0x80000000 + 0, 0, 0x80000000 + 128}}, 45 {"2147483648", DerivationPath{0x80000000 + 44, 0x80000000 + 60, 0x80000000 + 0, 0, 0x80000000 + 0}}, 46 47 // Hexadecimal absolute derivation paths 48 {"m/0x2C'/0x3c'/0x00'/0x00", DerivationPath{0x80000000 + 44, 0x80000000 + 60, 0x80000000 + 0, 0}}, 49 {"m/0x2C'/0x3c'/0x00'/0x80", DerivationPath{0x80000000 + 44, 0x80000000 + 60, 0x80000000 + 0, 128}}, 50 {"m/0x2C'/0x3c'/0x00'/0x00'", DerivationPath{0x80000000 + 44, 0x80000000 + 60, 0x80000000 + 0, 0x80000000 + 0}}, 51 {"m/0x2C'/0x3c'/0x00'/0x80'", DerivationPath{0x80000000 + 44, 0x80000000 + 60, 0x80000000 + 0, 0x80000000 + 128}}, 52 {"m/0x8000002C/0x8000003c/0x80000000/0x00", DerivationPath{0x80000000 + 44, 0x80000000 + 60, 0x80000000 + 0, 0}}, 53 {"m/0x8000002C/0x8000003c/0x80000000/0x80000000", DerivationPath{0x80000000 + 44, 0x80000000 + 60, 0x80000000 + 0, 0x80000000 + 0}}, 54 55 // Hexadecimal relative derivation paths 56 {"0x00", DerivationPath{0x80000000 + 44, 0x80000000 + 60, 0x80000000 + 0, 0, 0}}, 57 {"0x80", DerivationPath{0x80000000 + 44, 0x80000000 + 60, 0x80000000 + 0, 0, 128}}, 58 {"0x00'", DerivationPath{0x80000000 + 44, 0x80000000 + 60, 0x80000000 + 0, 0, 0x80000000 + 0}}, 59 {"0x80'", DerivationPath{0x80000000 + 44, 0x80000000 + 60, 0x80000000 + 0, 0, 0x80000000 + 128}}, 60 {"0x80000000", DerivationPath{0x80000000 + 44, 0x80000000 + 60, 0x80000000 + 0, 0, 0x80000000 + 0}}, 61 62 // Weird inputs just to ensure they work 63 {" m / 44 '\n/\n 60 \n\n\t' /\n0 ' /\t\t 0", DerivationPath{0x80000000 + 44, 0x80000000 + 60, 0x80000000 + 0, 0}}, 64 65 // Invalid derivation paths 66 {"", nil}, // Empty relative derivation path 67 {"m", nil}, // Empty absolute derivation path 68 {"m/", nil}, // Missing last derivation component 69 {"/44'/60'/0'/0", nil}, // Absolute path without m prefix, might be user error 70 {"m/2147483648'", nil}, // Overflows 32 bit integer 71 {"m/-1'", nil}, // Cannot contain negative number 72 } 73 for i, tt := range tests { 74 if path, err := ParseDerivationPath(tt.input); !reflect.DeepEqual(path, tt.output) { 75 t.Errorf("test %d: parse mismatch: have %v (%v), want %v", i, path, err, tt.output) 76 } else if path == nil && err == nil { 77 t.Errorf("test %d: nil path and error: %v", i, err) 78 } 79 } 80 } 81 82 func testDerive(t *testing.T, next func() DerivationPath, expected []string) { 83 t.Helper() 84 for i, want := range expected { 85 if have := next(); fmt.Sprintf("%v", have) != want { 86 t.Errorf("step %d, have %v, want %v", i, have, want) 87 } 88 } 89 } 90 91 func TestHdPathIteration(t *testing.T) { 92 testDerive(t, DefaultIterator(DefaultBaseDerivationPath), 93 []string{ 94 "m/44'/60'/0'/0/0", "m/44'/60'/0'/0/1", 95 "m/44'/60'/0'/0/2", "m/44'/60'/0'/0/3", 96 "m/44'/60'/0'/0/4", "m/44'/60'/0'/0/5", 97 "m/44'/60'/0'/0/6", "m/44'/60'/0'/0/7", 98 "m/44'/60'/0'/0/8", "m/44'/60'/0'/0/9", 99 }) 100 101 testDerive(t, DefaultIterator(LegacyLedgerBaseDerivationPath), 102 []string{ 103 "m/44'/60'/0'/0", "m/44'/60'/0'/1", 104 "m/44'/60'/0'/2", "m/44'/60'/0'/3", 105 "m/44'/60'/0'/4", "m/44'/60'/0'/5", 106 "m/44'/60'/0'/6", "m/44'/60'/0'/7", 107 "m/44'/60'/0'/8", "m/44'/60'/0'/9", 108 }) 109 110 testDerive(t, LedgerLiveIterator(DefaultBaseDerivationPath), 111 []string{ 112 "m/44'/60'/0'/0/0", "m/44'/60'/1'/0/0", 113 "m/44'/60'/2'/0/0", "m/44'/60'/3'/0/0", 114 "m/44'/60'/4'/0/0", "m/44'/60'/5'/0/0", 115 "m/44'/60'/6'/0/0", "m/44'/60'/7'/0/0", 116 "m/44'/60'/8'/0/0", "m/44'/60'/9'/0/0", 117 }) 118 }