- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我正在编写比特币 BIP39 的 Java 实现。到目前为止,我的代码能够正确生成随机生成的助记词短语。但是,当将 12 个单词的助记词转换为 512 位种子时,结果值与 Ian Coleman's BIP39 Tool 的结果不匹配。
首先,SecureRandom 对象生成一个随机的 512 位熵值 (ENT)。使用 SHA256 对 ENT 值进行哈希处理以计算校验和值 (CS),即哈希值的前 4 位。校验和连接到 ENT 的末尾以给出 ENT_CS。 ENT_CS被分成每个11位的部分,并且使用这11位对应的整数值作为索引号来从Word List中获取单词。这会生成我的 12 个单词的助记词。到目前为止,所有步骤都符合上述 BIP39 工具的预期结果。
为了创建种子,我将 PBKDF2 与 HmacSHA512 结合使用,将迭代设置为 2048,并将 key 大小设置为 512 位(64 字节)。我已经针对这些 Test Vectors 、Google 的“crypto”包实现和 NovaCrypto 的 Java BIP39 实现测试了 PBKDF2 的实现。根据 Bitcoin Core BIP39 Specifications,助记词(不包括分隔符)与 "mnemonic"+password
的盐一起用作输入。
public static byte[] PBKDF2(String mnemonic, String salt) {
try {
byte[] fixedSalt = ("mnemonic"+salt).getBytes(StandardCharsets.UTF_8);
KeySpec spec = new PBEKeySpec(mnemonic.toCharArray(), fixedSalt, 2048, 512);
SecretKeyFactory f = SecretKeyFactory.getInstance("PBKDF2WithHmacSHA512");
return f.generateSecret(spec).getEncoded();
} catch (NoSuchAlgorithmException | InvalidKeySpecException ex) {
throw new RuntimeException(ex);
}
}
public static String[] generateMnemonic() {
// Generate 128-bit Random Number for Entropy
byte[] ENT = getEntropy();
// Hash the Entropy value
byte[] HASH = SHA256(ENT);
// Copy first 4 bits of Hash as Checksum
boolean[] CS = Arrays.copyOfRange(bytesToBits(HASH), 0, 4);
// Add Checksum to the end of Entropy bits
boolean[] ENT_CS = Arrays.copyOf(bytesToBits(ENT), bytesToBits(ENT).length + CS.length);
System.arraycopy(CS, 0, ENT_CS, bytesToBits(ENT).length, CS.length);
// Split ENT_CS into groups of 11 bits and creates String array for
// mnemonicWords
String[] mnemonicWords = new String[12];
for (int i = 0; i < 12; i++) {
boolean[] numBits = Arrays.copyOfRange(ENT_CS, i * 11, i * 11 + 11);
mnemonicWords[i] = wordList.get(bitsToInt(numBits));
}
return mnemonicWords;
}
// Returns randomly generated, 16-byte number
public static byte[] getEntropy() {
byte[] ent = new byte[16];
sr.nextBytes(ent);
return ent;
}
// Returns bit representation of byte array
public static boolean[] bytesToBits(byte[] data) {
boolean[] bits = new boolean[data.length * 8];
for (int i = 0; i < data.length; ++i)
for (int j = 0; j < 8; ++j)
bits[(i * 8) + j] = (data[i] & (1 << (7 - j))) != 0;
return bits;
}
// Returns hex string from byte array
private final static char[] hexArray = "0123456789ABCDEF".toCharArray();
public static String bytesToHex(byte[] bytes) {
char[] hexChars = new char[bytes.length * 2];
for (int j = 0; j < bytes.length; j++) {
int v = bytes[j] & 0xFF;
hexChars[j * 2] = hexArray[v >>> 4];
hexChars[j * 2 + 1] = hexArray[v & 0x0F];
}
return new String(hexChars);
}
// Returns SHA256 hash of input data
public static byte[] SHA256(byte[] data) {
try {
MessageDigest digest = MessageDigest.getInstance("SHA-256");
System.out.println(Arrays.toString(data));
return digest.digest(data);
} catch (NoSuchAlgorithmException ex) {
throw new RuntimeException(ex);
}
}
// Returns int value of a bit array
public static int bitsToInt(boolean[] bits) {
int n = 0, l = bits.length;
for (int i = 0; i < l; ++i) {
n = (n << 1) + (bits[i] ? 1 : 0);
}
return n;
}
// Generate Mnemonic Words, Mnemonic Phrase, and Seed
String[] mnemonicWords = generateMnemonic();
String mnemonicPhrase = "";
for (String word : mnemonicWords)
mnemonicPhrase += word;
byte[] seed = PBKDF2(mnemonicPhrase, "");
System.out.println("Seed: " + bytesToHex(seed));
My Program Trial
Entropy (hex): 3CCB62D9AF76F1E8DB113E66B2D84656
Checksum bits: 1100
Raw Binary: 00111100110 01011011000 10110110011 01011110111 01101111000 11110100011 01101100010 00100111110 01100110101 10010110110 00010001100 1010110
Mnemonic: devote force reopen galaxy humor virtual hobby chief grit nothing bag pulse
Seed: 013FFA714C57AA26C59DC215880D9C2398A8B38D10D7E41A882CF98C35976F0BF26BCC08B0B196945DE8778C7FD561FB0F20A8B9BAD46B12196C963A85E3B40E
Expected Results (Derived from same Entropy)
Entropy (hex): 3CCB62D9AF76F1E8DB113E66B2D84656
Checksum bits: 1100
Raw Binary: 00111100110 01011011000 10110110011 01011110111 01101111000 11110100011 01101100010 00100111110 01100110101 10010110110 00010001100 1010110
Mnemonic: devote force reopen galaxy humor virtual hobby chief grit nothing bag pulse
Seed: 0c3c5f9ae724a2a3ed70aeb24919c10506e4962223a5375f70164be8b897d615ec9bf9f3e64a889cff03318cc5d0b3c8378ba0264d198e307c609632016ddd01
最佳答案
看来我能够回答我自己的问题了。在我的程序中,我使用
连接不带空格的种子词String mnemonicPhrase = "";
for (String word : mnemonicWords)
mnemonicPhrase += word;
但这不是正确的格式,因为要包含空格。更改此代码块以添加空格:
String mnemonicPhrase = "";
for(int i=0; i<mnemonicWords.length; i++) {
mnemonicPhrase += mnemonicWords[i];
if(i < mnemonicWords.length-1) mnemonicPhrase += " ";
}
产生预期的已发布测试 vector 结果 here使用密码“TREZOR”。
关于java - BIP39 助记符的种子与测试 vector 不匹配,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/55622851/
我想在我自己的应用程序中显示 AmazFit Bip Watch 的心率值。 UUID: 00002a37-0000-1000-8000-00805f9b34fb 示例字节数组: byte[0] =
我正在尝试通过 Web 服务在 BI Publisher 中安排报告。我希望我的报告每月自动安排一次。我正在使用重复间隔来传递我的频率。但问题是它接受以毫秒为单位的值。所以对于这个月来说,我无法传递这
我的应用程序在本地 CF 2.0 下运行,我想知道如何连接并发送内容以在 http://www.milliontech.com/home/content/view/195/95/的嵌入式打印机中打印'
我想在调制解调器拨完一个电话号码后挂断电话。我可以跟踪调制解调器何时开始拨号以及何时断开连接。但我无法跟踪拨号完成。我为此使用 TAPI。我试过 C# 和 C++。但我不知道如何管理它。 最佳答案 当
Letsencrypt 非常棒,因为它允许用户免费生成有效(非自签名)SSL 证书。我正在使用 bip作为 IRC 代理。 Bip 可以使用 SSL 证书进行加密,但是 the documentati
我是一名优秀的程序员,十分优秀!