-
Notifications
You must be signed in to change notification settings - Fork 20
/
PrivateKeyTests.cs
318 lines (275 loc) · 13.1 KB
/
PrivateKeyTests.cs
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
// Autarkysoft Tests
// Copyright (c) 2020 Autarkysoft
// Distributed under the MIT software license, see the accompanying
// file LICENCE or http://www.opensource.org/licenses/mit-license.php.
using Autarkysoft.Bitcoin;
using Autarkysoft.Bitcoin.Cryptography;
using Autarkysoft.Bitcoin.Cryptography.Asymmetric.KeyPairs;
using System;
using System.Collections.Generic;
using System.Numerics;
using Xunit;
namespace Tests.Bitcoin.Cryptography.Asymmetric.KeyPairs
{
public class PrivateKeyTests
{
public static IEnumerable<object[]> GetConstructorCases()
{
yield return new object[]
{
// Min value
BigInteger.One,
Helper.HexToBytes("0000000000000000000000000000000000000000000000000000000000000001")
};
yield return new object[]
{
// Max value
BigInteger.Parse("115792089237316195423570985008687907852837564279074904382605163141518161494336"),
Helper.HexToBytes("fffffffffffffffffffffffffffffffebaaedce6af48a03bbfd25e8cd0364140")
};
yield return new object[]
{
// https://en.bitcoin.it/wiki/Wallet_import_format
BigInteger.Parse("5500171714335001507730457227127633683517613019341760098818554179534751705629"),
Helper.HexToBytes("0c28fca386c7a227600b2fe50b7cae11ec86d3bf1fbe471be89827e19d72aa1d")
};
}
[Theory]
[MemberData(nameof(GetConstructorCases))]
public void Constructor_IntAndByteTest(BigInteger big, byte[] ba)
{
using PrivateKey k1 = new PrivateKey(big);
using PrivateKey k2 = new PrivateKey(ba);
Helper.ComparePrivateField(k1, "keyBytes", ba);
Helper.ComparePrivateField(k2, "keyBytes", ba);
}
[Fact]
public void Constructor_Int_BigSmallBytesTest()
{
byte[] big = new byte[40];
big[39] = 1;
big[38] = 2;
big[37] = 3;
byte[] small = { 3, 2, 1 };
byte[] expected = new byte[32];
expected[31] = 1;
expected[30] = 2;
expected[29] = 3;
using PrivateKey k1 = new PrivateKey(big);
using PrivateKey k2 = new PrivateKey(small);
Helper.ComparePrivateField(k1, "keyBytes", expected);
Helper.ComparePrivateField(k2, "keyBytes", expected);
}
[Fact]
public void Constructor_IntExceptionTest()
{
BigInteger max = BigInteger.Parse("115792089237316195423570985008687907852837564279074904382605163141518161494336");
Assert.Throws<ArgumentOutOfRangeException>(() => new PrivateKey(BigInteger.Zero));
Assert.Throws<ArgumentOutOfRangeException>(() => new PrivateKey(max + 1));
}
[Fact]
public void Constructor_ByteExceptionTest()
{
byte[] nba = null;
Assert.Throws<ArgumentNullException>(() => new PrivateKey(nba));
Assert.Throws<ArgumentOutOfRangeException>(() => new PrivateKey(new byte[32]));
Assert.Throws<ArgumentOutOfRangeException>(() => new PrivateKey(new byte[33]));
BigInteger max = BigInteger.Parse("115792089237316195423570985008687907852837564279074904382605163141518161494336");
byte[] maxBa = (max + 1).ToByteArray(true, true);
Assert.Throws<ArgumentOutOfRangeException>(() => new PrivateKey(maxBa));
maxBa = new byte[32];
((Span<byte>)maxBa).Fill(0xff);
Assert.Throws<ArgumentOutOfRangeException>(() => new PrivateKey(maxBa));
}
class MockRng : IRandomNumberGenerator
{
public MockRng(byte[] baToReturn) => data = baToReturn;
private readonly byte[] data;
public void GetBytes(byte[] toFill) => Buffer.BlockCopy(data, 0, toFill, 0, data.Length);
public void Dispose() { }
}
[Fact]
public void Constructor_RNGTest()
{
byte[] expected = new byte[32];
expected[31] = 1;
using PrivateKey key = new PrivateKey(new MockRng(expected));
byte[] actual = key.ToBytes();
Assert.Equal(expected, actual);
}
class BrokenMockRng : IRandomNumberGenerator
{
// Will not change the given array ie. it will return all zeros.
public void GetBytes(byte[] toFill) { }
public void Dispose() { }
}
class OutOfRangeRng : IRandomNumberGenerator
{
// Returns the biggest possible value which is out of range of bitcoin curve
public void GetBytes(byte[] toFill) => ((Span<byte>)toFill).Fill(0xff);
public void Dispose() { }
}
[Fact]
public void Constructor_RNG_ExceptionTest()
{
IRandomNumberGenerator nRng = null;
Assert.Throws<ArgumentNullException>(() => new PrivateKey(nRng));
Assert.Throws<ArgumentException>(() => new PrivateKey(new BrokenMockRng()));
Assert.Throws<ArgumentException>(() => new PrivateKey(new OutOfRangeRng()));
}
public static IEnumerable<object[]> GetWifCases()
{
yield return new object[]
{
// Min value
Helper.HexToBytes("0000000000000000000000000000000000000000000000000000000000000001"),
"5HpHagT65TZzG1PH3CSu63k8DbpvD8s5ip4nEB3kEsreAnchuDf",
"KwDiBf89QgGbjEhKnhXJuH7LrciVrZi3qYjgd9M7rFU73sVHnoWn",
"91avARGdfge8E4tZfYLoxeJ5sGBdNJQH4kvjJoQFacbgwmaKkrx",
"cMahea7zqjxrtgAbB7LSGbcQUr1uX1ojuat9jZodMN87JcbXMTcA"
};
yield return new object[]
{
// Max value
Helper.HexToBytes("fffffffffffffffffffffffffffffffebaaedce6af48a03bbfd25e8cd0364140"),
"5Km2kuu7vtFDPpxywn4u3NLpbr5jKpTB3jsuDU2KYEqetqj84qw",
"L5oLkpV3aqBjhki6LmvChTCV6odsp4SXM6FfU2Gppt5kFLaHLuZ9",
"93XfLeifX7KMMtUGa7xouxtnFWSSUyzNPgjrJ6Npsyahfqjy7oJ",
"cWALDjUu1tszsCBMjBjL4mhYj2wHUWYDR8Q8aSjLKzjkW5eBtpzu"
};
yield return new object[]
{
// https://en.bitcoin.it/wiki/Wallet_import_format
Helper.HexToBytes("0c28fca386c7a227600b2fe50b7cae11ec86d3bf1fbe471be89827e19d72aa1d"),
"5HueCGU8rMjxEXxiPuD5BDku4MkFqeZyd4dZ1jvhTVqvbTLvyTJ",
"KwdMAjGmerYanjeui5SHS7JkmpZvVipYvB2LJGU1ZxJwYvP98617",
"91gGn1HgSap6CbU12F6z3pJri26xzp7Ay1VW6NHCoEayNXwRpu2",
"cMzLdeGd5vEqxB8B6VFQoRopQ3sLAAvEzDAoQgvX54xwofSWj1fx"
};
yield return new object[]
{
// https://en.bitcoin.it/wiki/Private_key
Helper.HexToBytes("e9873d79c6d87dc0fb6a5778633389f4453213303da61f20bd67fc233aa33262"),
"5Kb8kLf9zgWQnogidDA76MzPL6TsZZY36hWXMssSzNydYXYB9KF",
"L53fCHmQhbNp1B4JipfBtfeHZH7cAibzG9oK19XfiFzxHgAkz6JK",
"93MmL5UhauaYksC1FZ41xxYLykpaij5ESeNUSWDxL7igKYUR9N1",
"cVQefCmG8f55AcXa7EUKFz9MBWR1qAhgLBwn7ZzBDNexYRHvBXZm"
};
yield return new object[]
{
// https://en.bitcoin.it/wiki/Technical_background_of_version_1_Bitcoin_addresses
Helper.HexToBytes("18e14a7b6a307f426a94f8114701e7c8e774e7f9a47e2c2035db29a206321725"),
"5J1F7GHadZG3sCCKHCwg8Jvys9xUbFsjLnGec4H125Ny1V9nR6V",
"Kx45GeUBSMPReYQwgXiKhG9FzNXrnCeutJp4yjTd5kKxCitadm3C",
"91msh178DnLBqFhbuYqazuUwWpKBkRQvgj8bggdWMp81nVp9PfM",
"cNR4jZU2sR5goytD4wXT4aeKcbqGSekbxLxY69v8aryxTU1SMnJZ"
};
}
[Theory]
[MemberData(nameof(GetWifCases))]
public void Constructor_WifTest(byte[] expected, string wif, string wifC, string wifT, string wifTC)
{
using PrivateKey keyW = new PrivateKey(wif, NetworkType.MainNet);
using PrivateKey keyWC = new PrivateKey(wifC, NetworkType.MainNet);
using PrivateKey keyWT = new PrivateKey(wifT, NetworkType.TestNet);
using PrivateKey keyWTC = new PrivateKey(wifTC, NetworkType.TestNet);
// RegTest and TestNet are the same
using PrivateKey keyWRT = new PrivateKey(wifT, NetworkType.RegTest);
Assert.Equal(expected, keyW.ToBytes());
Assert.Equal(expected, keyWC.ToBytes());
Assert.Equal(expected, keyWT.ToBytes());
Assert.Equal(expected, keyWTC.ToBytes());
Assert.Equal(expected, keyWRT.ToBytes());
}
[Theory]
[InlineData(null)]
[InlineData("")]
[InlineData(" ")]
public void Constructor_Wif_NullExceptionTest(string wif)
{
Exception ex = Assert.Throws<ArgumentNullException>(() => new PrivateKey(wif));
Assert.Contains("Input WIF can not be null or empty", ex.Message);
}
[Theory]
[InlineData("5GPHYxaeAAqL2egjmyU8Kaaqh831aw12XJ92y2rgRWi1zc", "Invalid first byte")] // first byte is 3
[InlineData("L5HydKmZoMcqfoY9Rgi8BRnWGmDw9YhoUS9ArnToVxvyFbM9GyfJ", "Invalid compressed byte")] // 0x54
[InlineData("KwFAa6AumokBD2dVqQLPou42jHiVsvThY1n25HJ8Ji8REf1wxAQb", "Invalid compressed byte")] // 0xcb
public void Constructor_WIF_FormatException_Test(string wif, string error)
{
Exception ex = Assert.Throws<FormatException>(() => new PrivateKey(wif));
Assert.Contains(error, ex.Message);
}
[Theory]
[InlineData("5HpHagT65TZzG1PH3CSu63k8DbpvD8s5ip4nEB3kEsreAbuatmU", NetworkType.MainNet)] // 0
[InlineData("KwDiBf89QgGbjEhKnhXJuH7LrciVrZi3qYjgd9M7rFU73Nd2Mcv1", NetworkType.MainNet)] // 0
[InlineData("91avARGdfge8E4tZfYLoxeJ5sGBdNJQH4kvjJoQFacbgwi1C2GD", NetworkType.TestNet)] // 0
[InlineData("cMahea7zqjxrtgAbB7LSGbcQUr1uX1ojuat9jZodMN87J7g8rY9t", NetworkType.TestNet)] // 0
[InlineData("5Km2kuu7vtFDPpxywn4u3NLpbr5jKpTB3jsuDU2KYEqetwr388P", NetworkType.MainNet)] // max value + 1
[InlineData("L5oLkpV3aqBjhki6LmvChTCV6odsp4SXM6FfU2Gppt5kFqRzExJJ", NetworkType.MainNet)] // max value + 1
[InlineData("93XfLeifX7KMMtUGa7xouxtnFWSSUyzNPgjrJ6Npsyahfwcd8gd", NetworkType.TestNet)] // max value + 1
[InlineData("cWALDjUu1tszsCBMjBjL4mhYj2wHUWYDR8Q8aSjLKzjkWaXMLRaY", NetworkType.TestNet)] // max value + 1
public void Constructor_WIF_OutOfRangeException_Test(string wif, NetworkType netType)
{
Exception ex = Assert.Throws<ArgumentOutOfRangeException>(() => new PrivateKey(wif, netType));
Assert.Contains("Given key value is outside the defined range by the curve", ex.Message);
}
[Fact]
public void ToBytesTest()
{
using PrivateKey key = new PrivateKey(new byte[] { 10 });
byte[] actual = key.ToBytes();
byte[] expected = new byte[32];
expected[31] = 10;
Assert.Equal(expected, actual);
}
[Fact]
public void ToBytes_ObjectDisposedException_Test()
{
PrivateKey key = new PrivateKey(new byte[] { 10 });
key.Dispose();
Assert.Throws<ObjectDisposedException>(() => key.ToBytes());
}
[Theory]
[MemberData(nameof(GetConstructorCases))]
public void ToBigIntTest(BigInteger expected, byte[] bytesToUse)
{
using PrivateKey key = new PrivateKey(bytesToUse);
BigInteger actual = key.ToBigInt();
Assert.Equal(expected, actual);
}
[Fact]
public void ToBigInt_ObjectDisposedException_Test()
{
PrivateKey key = new PrivateKey(new byte[] { 10 });
key.Dispose();
Assert.Throws<ObjectDisposedException>(() => key.ToBigInt());
}
[Theory]
[MemberData(nameof(GetWifCases))]
public void ToWifTest(byte[] bytesToUse, string wif, string wifC, string wifT, string wifTC)
{
using PrivateKey prvKey = new PrivateKey(bytesToUse);
Assert.Equal(wif, prvKey.ToWif(false, NetworkType.MainNet));
Assert.Equal(wifC, prvKey.ToWif(true, NetworkType.MainNet));
Assert.Equal(wifT, prvKey.ToWif(false, NetworkType.TestNet));
Assert.Equal(wifTC, prvKey.ToWif(true, NetworkType.TestNet));
// RegTest is the same as TestNet
Assert.Equal(wifTC, prvKey.ToWif(true, NetworkType.RegTest));
}
[Fact]
public void ToWif_ObjectDisposedExceptionTest()
{
PrivateKey key = new PrivateKey(new byte[] { 10 });
key.Dispose();
Assert.Throws<ObjectDisposedException>(() => key.ToWif(true, NetworkType.MainNet));
}
[Fact]
public void ToWif_ArgumentException_Test()
{
using PrivateKey key = new PrivateKey(new byte[] { 10 });
NetworkType netType = (NetworkType)100;
Exception ex = Assert.Throws<ArgumentException>(() => key.ToWif(true, netType));
Assert.Contains("Network type is not defined", ex.Message);
}
}
}