hd_test.go 5.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118
  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. package accounts
  17. import (
  18. "fmt"
  19. "reflect"
  20. "testing"
  21. )
  22. // Tests that HD derivation paths can be correctly parsed into our internal binary
  23. // representation.
  24. func TestHDPathParsing(t *testing.T) {
  25. tests := []struct {
  26. input string
  27. output DerivationPath
  28. }{
  29. // Plain absolute derivation paths
  30. {"m/44'/60'/0'/0", DerivationPath{0x80000000 + 44, 0x80000000 + 60, 0x80000000 + 0, 0}},
  31. {"m/44'/60'/0'/128", DerivationPath{0x80000000 + 44, 0x80000000 + 60, 0x80000000 + 0, 128}},
  32. {"m/44'/60'/0'/0'", DerivationPath{0x80000000 + 44, 0x80000000 + 60, 0x80000000 + 0, 0x80000000 + 0}},
  33. {"m/44'/60'/0'/128'", DerivationPath{0x80000000 + 44, 0x80000000 + 60, 0x80000000 + 0, 0x80000000 + 128}},
  34. {"m/2147483692/2147483708/2147483648/0", DerivationPath{0x80000000 + 44, 0x80000000 + 60, 0x80000000 + 0, 0}},
  35. {"m/2147483692/2147483708/2147483648/2147483648", DerivationPath{0x80000000 + 44, 0x80000000 + 60, 0x80000000 + 0, 0x80000000 + 0}},
  36. // Plain relative derivation paths
  37. {"0", DerivationPath{0x80000000 + 44, 0x80000000 + 60, 0x80000000 + 0, 0, 0}},
  38. {"128", DerivationPath{0x80000000 + 44, 0x80000000 + 60, 0x80000000 + 0, 0, 128}},
  39. {"0'", DerivationPath{0x80000000 + 44, 0x80000000 + 60, 0x80000000 + 0, 0, 0x80000000 + 0}},
  40. {"128'", DerivationPath{0x80000000 + 44, 0x80000000 + 60, 0x80000000 + 0, 0, 0x80000000 + 128}},
  41. {"2147483648", DerivationPath{0x80000000 + 44, 0x80000000 + 60, 0x80000000 + 0, 0, 0x80000000 + 0}},
  42. // Hexadecimal absolute derivation paths
  43. {"m/0x2C'/0x3c'/0x00'/0x00", DerivationPath{0x80000000 + 44, 0x80000000 + 60, 0x80000000 + 0, 0}},
  44. {"m/0x2C'/0x3c'/0x00'/0x80", DerivationPath{0x80000000 + 44, 0x80000000 + 60, 0x80000000 + 0, 128}},
  45. {"m/0x2C'/0x3c'/0x00'/0x00'", DerivationPath{0x80000000 + 44, 0x80000000 + 60, 0x80000000 + 0, 0x80000000 + 0}},
  46. {"m/0x2C'/0x3c'/0x00'/0x80'", DerivationPath{0x80000000 + 44, 0x80000000 + 60, 0x80000000 + 0, 0x80000000 + 128}},
  47. {"m/0x8000002C/0x8000003c/0x80000000/0x00", DerivationPath{0x80000000 + 44, 0x80000000 + 60, 0x80000000 + 0, 0}},
  48. {"m/0x8000002C/0x8000003c/0x80000000/0x80000000", DerivationPath{0x80000000 + 44, 0x80000000 + 60, 0x80000000 + 0, 0x80000000 + 0}},
  49. // Hexadecimal relative derivation paths
  50. {"0x00", DerivationPath{0x80000000 + 44, 0x80000000 + 60, 0x80000000 + 0, 0, 0}},
  51. {"0x80", DerivationPath{0x80000000 + 44, 0x80000000 + 60, 0x80000000 + 0, 0, 128}},
  52. {"0x00'", DerivationPath{0x80000000 + 44, 0x80000000 + 60, 0x80000000 + 0, 0, 0x80000000 + 0}},
  53. {"0x80'", DerivationPath{0x80000000 + 44, 0x80000000 + 60, 0x80000000 + 0, 0, 0x80000000 + 128}},
  54. {"0x80000000", DerivationPath{0x80000000 + 44, 0x80000000 + 60, 0x80000000 + 0, 0, 0x80000000 + 0}},
  55. // Weird inputs just to ensure they work
  56. {" m / 44 '\n/\n 60 \n\n\t' /\n0 ' /\t\t 0", DerivationPath{0x80000000 + 44, 0x80000000 + 60, 0x80000000 + 0, 0}},
  57. // Invalid derivation paths
  58. {"", nil}, // Empty relative derivation path
  59. {"m", nil}, // Empty absolute derivation path
  60. {"m/", nil}, // Missing last derivation component
  61. {"/44'/60'/0'/0", nil}, // Absolute path without m prefix, might be user error
  62. {"m/2147483648'", nil}, // Overflows 32 bit integer
  63. {"m/-1'", nil}, // Cannot contain negative number
  64. }
  65. for i, tt := range tests {
  66. if path, err := ParseDerivationPath(tt.input); !reflect.DeepEqual(path, tt.output) {
  67. t.Errorf("test %d: parse mismatch: have %v (%v), want %v", i, path, err, tt.output)
  68. } else if path == nil && err == nil {
  69. t.Errorf("test %d: nil path and error: %v", i, err)
  70. }
  71. }
  72. }
  73. func testDerive(t *testing.T, next func() DerivationPath, expected []string) {
  74. t.Helper()
  75. for i, want := range expected {
  76. if have := next(); fmt.Sprintf("%v", have) != want {
  77. t.Errorf("step %d, have %v, want %v", i, have, want)
  78. }
  79. }
  80. }
  81. func TestHdPathIteration(t *testing.T) {
  82. testDerive(t, DefaultIterator(DefaultBaseDerivationPath),
  83. []string{
  84. "m/44'/60'/0'/0/0", "m/44'/60'/0'/0/1",
  85. "m/44'/60'/0'/0/2", "m/44'/60'/0'/0/3",
  86. "m/44'/60'/0'/0/4", "m/44'/60'/0'/0/5",
  87. "m/44'/60'/0'/0/6", "m/44'/60'/0'/0/7",
  88. "m/44'/60'/0'/0/8", "m/44'/60'/0'/0/9",
  89. })
  90. testDerive(t, DefaultIterator(LegacyLedgerBaseDerivationPath),
  91. []string{
  92. "m/44'/60'/0'/0", "m/44'/60'/0'/1",
  93. "m/44'/60'/0'/2", "m/44'/60'/0'/3",
  94. "m/44'/60'/0'/4", "m/44'/60'/0'/5",
  95. "m/44'/60'/0'/6", "m/44'/60'/0'/7",
  96. "m/44'/60'/0'/8", "m/44'/60'/0'/9",
  97. })
  98. testDerive(t, LedgerLiveIterator(DefaultBaseDerivationPath),
  99. []string{
  100. "m/44'/60'/0'/0/0", "m/44'/60'/1'/0/0",
  101. "m/44'/60'/2'/0/0", "m/44'/60'/3'/0/0",
  102. "m/44'/60'/4'/0/0", "m/44'/60'/5'/0/0",
  103. "m/44'/60'/6'/0/0", "m/44'/60'/7'/0/0",
  104. "m/44'/60'/8'/0/0", "m/44'/60'/9'/0/0",
  105. })
  106. }