Renamed project from 'cold' to 'burrow'

This commit is contained in:
B.J. Dweck 2020-09-01 16:09:20 +02:00
parent ac99468a9d
commit 78c03988ea
11 changed files with 847 additions and 847 deletions

View File

@ -23,7 +23,7 @@ group 'io.rudefox'
version isRelease ? gitVersion.call() : gitVersion.call() + "-SNAPSHOT" version isRelease ? gitVersion.call() : gitVersion.call() + "-SNAPSHOT"
application { application {
mainClassName = 'io.rudefox.cold.RudefoxCold' mainClassName = 'io.rudefox.burrow.RudefoxBurrow'
} }
run { run {

View File

@ -1,115 +1,115 @@
package io.rudefox.cold; package io.rudefox.burrow;
import com.bjdweck.bitcoin.mnemonic.Entropy; import com.bjdweck.bitcoin.mnemonic.Entropy;
import java.util.Scanner; import java.util.Scanner;
import java.util.regex.Pattern; import java.util.regex.Pattern;
public class Dice8EntropyGenerator { public class Dice8EntropyGenerator {
public static final int DICE_PER_ROLL = 11; public static final int DICE_PER_ROLL = 11;
private final int targetBitsOfEntropy; private final int targetBitsOfEntropy;
private final Scanner inputScanner; private final Scanner inputScanner;
private Entropy entropy = new Entropy(); private Entropy entropy = new Entropy();
public Dice8EntropyGenerator(int targetBitsOfEntropy, Scanner inputScanner) { public Dice8EntropyGenerator(int targetBitsOfEntropy, Scanner inputScanner) {
this.targetBitsOfEntropy = targetBitsOfEntropy; this.targetBitsOfEntropy = targetBitsOfEntropy;
this.inputScanner = inputScanner; this.inputScanner = inputScanner;
} }
Entropy generate() { Entropy generate() {
entropy = new Entropy(); entropy = new Entropy();
for (int rollSetCount = 0; entropy.getBitLength() < targetBitsOfEntropy; rollSetCount++) for (int rollSetCount = 0; entropy.getBitLength() < targetBitsOfEntropy; rollSetCount++)
doDiceRoll(rollSetCount); doDiceRoll(rollSetCount);
return entropy.truncate(targetBitsOfEntropy).appendChecksum(); return entropy.truncate(targetBitsOfEntropy).appendChecksum();
} }
private void doDiceRoll(int currentRollSet) { private void doDiceRoll(int currentRollSet) {
String eventString = scanNextRollSetString(); String eventString = scanNextRollSetString();
StringBuilder rollValuesLine = new StringBuilder("|"); StringBuilder rollValuesLine = new StringBuilder("|");
for (int rollNumber = 0; rollNumber < DICE_PER_ROLL; rollNumber++) { for (int rollNumber = 0; rollNumber < DICE_PER_ROLL; rollNumber++) {
int rollValue = Integer.parseInt(eventString.charAt(rollNumber) + ""); int rollValue = Integer.parseInt(eventString.charAt(rollNumber) + "");
String formatString = String formatString =
rollNumber == 3 || rollNumber == 7 ? " %d |" : " %d |"; rollNumber == 3 || rollNumber == 7 ? " %d |" : " %d |";
rollValuesLine.append(String.format(formatString, rollValue)); rollValuesLine.append(String.format(formatString, rollValue));
entropy = entropy.appendBits(rollValue - 1, 3); entropy = entropy.appendBits(rollValue - 1, 3);
} }
System.out.printf("%n%s%n%s%n%s%n%n", System.out.printf("%n%s%n%s%n%s%n%n",
rollValuesLine, rollValuesLine,
getBitsLine(), getBitsLine(),
getSeedWordsLine(currentRollSet) getSeedWordsLine(currentRollSet)
); );
} }
private String scanNextRollSetString() { private String scanNextRollSetString() {
String rollSetString = ""; String rollSetString = "";
Pattern rollSetPattern = Pattern.compile(String.format("[1-8]{%d}", DICE_PER_ROLL)); Pattern rollSetPattern = Pattern.compile(String.format("[1-8]{%d}", DICE_PER_ROLL));
while (!rollSetPattern.matcher(rollSetString).matches()) { while (!rollSetPattern.matcher(rollSetString).matches()) {
System.out.print("Input 11 x 8-sided dice rolls [1-8]: "); System.out.print("Input 11 x 8-sided dice rolls [1-8]: ");
rollSetString = inputScanner.next(); rollSetString = inputScanner.next();
} }
return rollSetString; return rollSetString;
} }
private StringBuilder getBitsLine() { private StringBuilder getBitsLine() {
String entropyBitString = entropy.toString(); String entropyBitString = entropy.toString();
String currentRollSetBitString = String currentRollSetBitString =
entropyBitString.substring(entropyBitString.length() - (DICE_PER_ROLL * 3)); entropyBitString.substring(entropyBitString.length() - (DICE_PER_ROLL * 3));
StringBuilder rollSetBitsLine = new StringBuilder("|"); StringBuilder rollSetBitsLine = new StringBuilder("|");
for (int rollSetBitIndex = 0; rollSetBitIndex < currentRollSetBitString.length(); rollSetBitIndex++) { for (int rollSetBitIndex = 0; rollSetBitIndex < currentRollSetBitString.length(); rollSetBitIndex++) {
int entropyBitIndex = entropyBitString.length() - currentRollSetBitString.length() + rollSetBitIndex; int entropyBitIndex = entropyBitString.length() - currentRollSetBitString.length() + rollSetBitIndex;
if (entropyBitIndex < targetBitsOfEntropy) if (entropyBitIndex < targetBitsOfEntropy)
rollSetBitsLine.append(currentRollSetBitString.charAt(rollSetBitIndex)); rollSetBitsLine.append(currentRollSetBitString.charAt(rollSetBitIndex));
else else
rollSetBitsLine.append("-"); rollSetBitsLine.append("-");
if (rollSetBitIndex == 32) rollSetBitsLine.append("|"); if (rollSetBitIndex == 32) rollSetBitsLine.append("|");
else if (rollSetBitIndex % 3 == 2) rollSetBitsLine.append(" "); else if (rollSetBitIndex % 3 == 2) rollSetBitsLine.append(" ");
else if (rollSetBitIndex % 11 == 10) rollSetBitsLine.append(" | "); else if (rollSetBitIndex % 11 == 10) rollSetBitsLine.append(" | ");
} }
return rollSetBitsLine; return rollSetBitsLine;
} }
private String getSeedWordsLine(int rollSet) { private String getSeedWordsLine(int rollSet) {
int baseIndex = rollSet * 3; int baseIndex = rollSet * 3;
return String.format( return String.format(
"|%-15s|%-17s|%-15s|", "|%-15s|%-17s|%-15s|",
getFormattedSeedWord(baseIndex), getFormattedSeedWord(baseIndex),
getFormattedSeedWord(baseIndex + 1), getFormattedSeedWord(baseIndex + 1),
getFormattedSeedWord(baseIndex + 2)); getFormattedSeedWord(baseIndex + 2));
} }
private String getFormattedSeedWord(int index) { private String getFormattedSeedWord(int index) {
int lastWordIndex = (3 * targetBitsOfEntropy / 32) - 1; int lastWordIndex = (3 * targetBitsOfEntropy / 32) - 1;
String seedWord = index != lastWordIndex ? entropy.getWord(index) : "CHECKWORD"; String seedWord = index != lastWordIndex ? entropy.getWord(index) : "CHECKWORD";
return String.format(" %2d. %s", index + 1, seedWord); return String.format(" %2d. %s", index + 1, seedWord);
} }
} }

View File

@ -1,47 +1,47 @@
package io.rudefox.cold; package io.rudefox.burrow;
import com.bjdweck.bitcoin.mnemonic.Entropy; import com.bjdweck.bitcoin.mnemonic.Entropy;
import com.bjdweck.math.UnsignedInt; import com.bjdweck.math.UnsignedInt;
class DiceEventBuffer { class DiceEventBuffer {
private final int diceBase; private final int diceBase;
private final StringBuilder buffer; private final StringBuilder buffer;
private final int targetBitsOfEntropy; private final int targetBitsOfEntropy;
DiceEventBuffer(int targetBitsOfEntropy, int diceBase) { DiceEventBuffer(int targetBitsOfEntropy, int diceBase) {
this.diceBase = diceBase; this.diceBase = diceBase;
this.targetBitsOfEntropy = targetBitsOfEntropy; this.targetBitsOfEntropy = targetBitsOfEntropy;
this.buffer = new StringBuilder(getRequiredEvents()); this.buffer = new StringBuilder(getRequiredEvents());
} }
int getRequiredEvents() { int getRequiredEvents() {
return (int) Math.ceil(this.targetBitsOfEntropy * Math.log(2) / Math.log(diceBase)); return (int) Math.ceil(this.targetBitsOfEntropy * Math.log(2) / Math.log(diceBase));
} }
void appendEvents(String eventString) { void appendEvents(String eventString) {
for (char inChar : eventString.toCharArray()) for (char inChar : eventString.toCharArray())
if (inChar >= '1' && inChar <= '6' && events() < getRequiredEvents()) if (inChar >= '1' && inChar <= '6' && events() < getRequiredEvents())
append((char) (inChar - 1)); append((char) (inChar - 1));
} }
Entropy toEntropy() { Entropy toEntropy() {
UnsignedInt entropy = new UnsignedInt(toString(), diceBase); UnsignedInt entropy = new UnsignedInt(toString(), diceBase);
byte[] entropyBytes = entropy.getLowestOrderBits(this.targetBitsOfEntropy).toBigEndianByteArray(); byte[] entropyBytes = entropy.getLowestOrderBits(this.targetBitsOfEntropy).toBigEndianByteArray();
return Entropy.fromRawEntropy(entropyBytes); return Entropy.fromRawEntropy(entropyBytes);
} }
int events() { int events() {
return buffer.length(); return buffer.length();
} }
private void append(char c) { private void append(char c) {
buffer.append(c); buffer.append(c);
} }
@Override @Override
public String toString() { public String toString() {
return buffer.toString(); return buffer.toString();
} }
} }

View File

@ -1,125 +1,125 @@
package io.rudefox.cold; package io.rudefox.burrow;
import com.bjdweck.bitcoin.mnemonic.Entropy; import com.bjdweck.bitcoin.mnemonic.Entropy;
import picocli.CommandLine; import picocli.CommandLine;
import java.security.SecureRandom; import java.security.SecureRandom;
import java.util.Scanner; import java.util.Scanner;
@CommandLine.Command(name = "mnemonic", description = "generate mnemonic sentence") @CommandLine.Command(name = "mnemonic", description = "generate mnemonic sentence")
public class MnemonicCommand implements Runnable { public class MnemonicCommand implements Runnable {
private static final int DEFAULT_BITS_OF_ENTROPY = 256; private static final int DEFAULT_BITS_OF_ENTROPY = 256;
private boolean isDiceEntropy() { private boolean isDiceEntropy() {
return entropyOptions != null && (entropyOptions.isDice6Entropy || entropyOptions.isDice8Entropy); return entropyOptions != null && (entropyOptions.isDice6Entropy || entropyOptions.isDice8Entropy);
} }
private boolean isInteractiveMode() { private boolean isInteractiveMode() {
return entropyOptions != null && entropyOptions.eventMethod.isInteractiveMode; return entropyOptions != null && entropyOptions.eventMethod.isInteractiveMode;
} }
private boolean isDice6InteractiveMode() { private boolean isDice6InteractiveMode() {
return isInteractiveMode() && entropyOptions.isDice6Entropy; return isInteractiveMode() && entropyOptions.isDice6Entropy;
} }
private boolean isDice8InteractiveMode() { private boolean isDice8InteractiveMode() {
return isInteractiveMode() && entropyOptions.isDice8Entropy; return isInteractiveMode() && entropyOptions.isDice8Entropy;
} }
@CommandLine.ArgGroup(exclusive = false) @CommandLine.ArgGroup(exclusive = false)
EntropyOptions entropyOptions; EntropyOptions entropyOptions;
static class EntropyOptions { static class EntropyOptions {
@CommandLine.Option(names = {"-6", "--dice6-entropy"}, @CommandLine.Option(names = {"-6", "--dice6-entropy"},
description = "use 6-sided dice entropy source") description = "use 6-sided dice entropy source")
boolean isDice6Entropy; boolean isDice6Entropy;
@CommandLine.Option(names = {"-8", "--dice8-entropy"}, @CommandLine.Option(names = {"-8", "--dice8-entropy"},
description = "use 8-sided dice entropy source") description = "use 8-sided dice entropy source")
boolean isDice8Entropy; boolean isDice8Entropy;
@CommandLine.ArgGroup(multiplicity = "1") @CommandLine.ArgGroup(multiplicity = "1")
EventMethod eventMethod; EventMethod eventMethod;
static class EventMethod { static class EventMethod {
@CommandLine.Option(names = {"-e", "--events"}, paramLabel = "[1-6]{100}|[1-8]{86}", @CommandLine.Option(names = {"-e", "--events"}, paramLabel = "[1-6]{100}|[1-8]{86}",
description = "string representing events from entropy source", description = "string representing events from entropy source",
required = true) required = true)
String getEventString; String getEventString;
@CommandLine.Option(names = {"-i", "--interactive"}, @CommandLine.Option(names = {"-i", "--interactive"},
description = "use interactive command line mode", description = "use interactive command line mode",
required = true) required = true)
boolean isInteractiveMode; boolean isInteractiveMode;
} }
} }
@CommandLine.Option(names = {"-b", "--bits"}, defaultValue = DEFAULT_BITS_OF_ENTROPY+"", @CommandLine.Option(names = {"-b", "--bits"}, defaultValue = DEFAULT_BITS_OF_ENTROPY+"",
description = "bits of entropy (default: 256)", description = "bits of entropy (default: 256)",
paramLabel = "128|160|192|224|256") paramLabel = "128|160|192|224|256")
int targetBitsOfEntropy; int targetBitsOfEntropy;
public void run() { public void run() {
Entropy entropyBytes = getEntropy(); Entropy entropyBytes = getEntropy();
System.out.println(entropyBytes.toMnemonic().getSentence()); System.out.println(entropyBytes.toMnemonic().getSentence());
} }
Entropy getEntropy() { Entropy getEntropy() {
if (!isDiceEntropy()) if (!isDiceEntropy())
return getGeneratedEntropy(); return getGeneratedEntropy();
DiceEventBuffer diceEventBuffer; DiceEventBuffer diceEventBuffer;
if (entropyOptions.isDice6Entropy) if (entropyOptions.isDice6Entropy)
diceEventBuffer = new DiceEventBuffer(this.targetBitsOfEntropy, 6); diceEventBuffer = new DiceEventBuffer(this.targetBitsOfEntropy, 6);
else else
diceEventBuffer = new DiceEventBuffer(this.targetBitsOfEntropy, 8); diceEventBuffer = new DiceEventBuffer(this.targetBitsOfEntropy, 8);
if (isDice6InteractiveMode()) if (isDice6InteractiveMode())
return getDice6EntropyInteractive(diceEventBuffer); return getDice6EntropyInteractive(diceEventBuffer);
if (isDice8InteractiveMode()) if (isDice8InteractiveMode())
return new Dice8EntropyGenerator(targetBitsOfEntropy, new Scanner(System.in)).generate(); return new Dice8EntropyGenerator(targetBitsOfEntropy, new Scanner(System.in)).generate();
diceEventBuffer.appendEvents(entropyOptions.eventMethod.getEventString); diceEventBuffer.appendEvents(entropyOptions.eventMethod.getEventString);
return diceEventBuffer.toEntropy(); return diceEventBuffer.toEntropy();
} }
private Entropy getDice6EntropyInteractive(DiceEventBuffer diceEventBuffer) { private Entropy getDice6EntropyInteractive(DiceEventBuffer diceEventBuffer) {
int requiredEvents = diceEventBuffer.getRequiredEvents(); int requiredEvents = diceEventBuffer.getRequiredEvents();
Scanner scanner = new Scanner(System.in); Scanner scanner = new Scanner(System.in);
boolean firstIteration = true; boolean firstIteration = true;
while (diceEventBuffer.events() < requiredEvents) { while (diceEventBuffer.events() < requiredEvents) {
int remainingEvents = requiredEvents - diceEventBuffer.events(); int remainingEvents = requiredEvents - diceEventBuffer.events();
System.out.print(String.format("Input %d %sdice rolls [1-6]: ", remainingEvents, (firstIteration ? "" : "more "))); System.out.print(String.format("Input %d %sdice rolls [1-6]: ", remainingEvents, (firstIteration ? "" : "more ")));
String inputString = scanner.next(); String inputString = scanner.next();
diceEventBuffer.appendEvents(inputString); diceEventBuffer.appendEvents(inputString);
firstIteration = false; firstIteration = false;
} }
return diceEventBuffer.toEntropy(); return diceEventBuffer.toEntropy();
} }
private Entropy getGeneratedEntropy() { private Entropy getGeneratedEntropy() {
SecureRandom random = new SecureRandom(); SecureRandom random = new SecureRandom();
int byteLength = targetBitsOfEntropy / 8; int byteLength = targetBitsOfEntropy / 8;
byte[] randomBytes = new byte[byteLength]; byte[] randomBytes = new byte[byteLength];
random.nextBytes(randomBytes); random.nextBytes(randomBytes);
return Entropy.fromRawEntropy(randomBytes); return Entropy.fromRawEntropy(randomBytes);
} }
} }

View File

@ -1,111 +1,111 @@
package io.rudefox.cold; package io.rudefox.burrow;
import com.google.zxing.BarcodeFormat; import com.google.zxing.BarcodeFormat;
import com.google.zxing.EncodeHintType; import com.google.zxing.EncodeHintType;
import com.google.zxing.WriterException; import com.google.zxing.WriterException;
import com.google.zxing.common.BitMatrix; import com.google.zxing.common.BitMatrix;
import com.google.zxing.qrcode.QRCodeWriter; import com.google.zxing.qrcode.QRCodeWriter;
import com.google.zxing.qrcode.decoder.ErrorCorrectionLevel; import com.google.zxing.qrcode.decoder.ErrorCorrectionLevel;
import java.nio.charset.Charset; import java.nio.charset.Charset;
import java.nio.charset.StandardCharsets; import java.nio.charset.StandardCharsets;
import java.util.HashMap; import java.util.HashMap;
import java.util.Map; import java.util.Map;
class QRCode { class QRCode {
private static final String FALLBACK_SET_STRING = "##"; private static final String FALLBACK_SET_STRING = "##";
static String toQRCode(String data) { static String toQRCode(String data) {
QRCodeWriter qrCodeWriter = new QRCodeWriter(); QRCodeWriter qrCodeWriter = new QRCodeWriter();
Map<EncodeHintType, Object> hints = new HashMap<>(); Map<EncodeHintType, Object> hints = new HashMap<>();
hints.put(EncodeHintType.MARGIN, 0); hints.put(EncodeHintType.MARGIN, 0);
hints.put(EncodeHintType.ERROR_CORRECTION, ErrorCorrectionLevel.L); hints.put(EncodeHintType.ERROR_CORRECTION, ErrorCorrectionLevel.L);
BitMatrix bitMatrix = null; BitMatrix bitMatrix = null;
try { try {
bitMatrix = qrCodeWriter.encode(data, BarcodeFormat.QR_CODE, 0, 0, hints); bitMatrix = qrCodeWriter.encode(data, BarcodeFormat.QR_CODE, 0, 0, hints);
} catch (WriterException e) { } catch (WriterException e) {
e.printStackTrace(); e.printStackTrace();
} }
if (bitMatrix == null) if (bitMatrix == null)
return "[Error Generating QR Code]"; return "[Error Generating QR Code]";
if (charsetDoesNotSupportFullBlockChar()) if (charsetDoesNotSupportFullBlockChar())
return bitMatrix.toString(FALLBACK_SET_STRING, " "); return bitMatrix.toString(FALLBACK_SET_STRING, " ");
return toUTF8(bitMatrix); return toUTF8(bitMatrix);
} }
private static String toUTF8(BitMatrix bitMatrix) { private static String toUTF8(BitMatrix bitMatrix) {
char SPACE = '\u0020'; char SPACE = '\u0020';
char LOWER_BLOCK = '\u2584'; char LOWER_BLOCK = '\u2584';
char UPPER_BLOCK = '\u2580'; char UPPER_BLOCK = '\u2580';
char FULL_BLOCK = '\u2588'; char FULL_BLOCK = '\u2588';
int[] blockCodepoints = new int[] {SPACE, LOWER_BLOCK, UPPER_BLOCK, FULL_BLOCK}; int[] blockCodepoints = new int[] {SPACE, LOWER_BLOCK, UPPER_BLOCK, FULL_BLOCK};
StringBuilder qrCode = new StringBuilder(); StringBuilder qrCode = new StringBuilder();
for (int y = 0; y < bitMatrix.getHeight(); y += 2) { for (int y = 0; y < bitMatrix.getHeight(); y += 2) {
for (int x = 0; x < bitMatrix.getWidth(); x++) { for (int x = 0; x < bitMatrix.getWidth(); x++) {
int upper_bit = bitMatrix.get(x, y) ? 1 : 0; int upper_bit = bitMatrix.get(x, y) ? 1 : 0;
int lower_bit = y+1 < bitMatrix.getHeight() && bitMatrix.get(x, y+1) ? 1 : 0; int lower_bit = y+1 < bitMatrix.getHeight() && bitMatrix.get(x, y+1) ? 1 : 0;
int index = upper_bit << 1 | lower_bit; int index = upper_bit << 1 | lower_bit;
int blockCodepoint = blockCodepoints[index]; int blockCodepoint = blockCodepoints[index];
qrCode.appendCodePoint(blockCodepoint); qrCode.appendCodePoint(blockCodepoint);
} }
if (y < bitMatrix.getHeight() - 1) if (y < bitMatrix.getHeight() - 1)
qrCode.append("\n"); qrCode.append("\n");
} }
return qrCode.toString(); return qrCode.toString();
} }
private static boolean charsetDoesNotSupportFullBlockChar() { private static boolean charsetDoesNotSupportFullBlockChar() {
Charset charset = ServiceLocator.DEFAULT_CHARSET; Charset charset = ServiceLocator.DEFAULT_CHARSET;
String charsetString = charset.toString(); String charsetString = charset.toString();
return charset.equals(StandardCharsets.ISO_8859_1) || return charset.equals(StandardCharsets.ISO_8859_1) ||
charset.equals(StandardCharsets.US_ASCII) || charset.equals(StandardCharsets.US_ASCII) ||
charsetString.equalsIgnoreCase("euc-jp") || charsetString.equalsIgnoreCase("euc-jp") ||
charsetString.equalsIgnoreCase("euc-kr") || charsetString.equalsIgnoreCase("euc-kr") ||
charsetString.equalsIgnoreCase("iso-2022-jp") || charsetString.equalsIgnoreCase("iso-2022-jp") ||
charsetString.equalsIgnoreCase("iso-8859-10") || charsetString.equalsIgnoreCase("iso-8859-10") ||
charsetString.equalsIgnoreCase("iso-8859-13") || charsetString.equalsIgnoreCase("iso-8859-13") ||
charsetString.equalsIgnoreCase("iso-8859-14") || charsetString.equalsIgnoreCase("iso-8859-14") ||
charsetString.equalsIgnoreCase("iso-8859-15") || charsetString.equalsIgnoreCase("iso-8859-15") ||
charsetString.equalsIgnoreCase("iso-8859-16") || charsetString.equalsIgnoreCase("iso-8859-16") ||
charsetString.equalsIgnoreCase("iso-8859-2") || charsetString.equalsIgnoreCase("iso-8859-2") ||
charsetString.equalsIgnoreCase("iso-8859-3") || charsetString.equalsIgnoreCase("iso-8859-3") ||
charsetString.equalsIgnoreCase("iso-8859-4") || charsetString.equalsIgnoreCase("iso-8859-4") ||
charsetString.equalsIgnoreCase("iso-8859-5") || charsetString.equalsIgnoreCase("iso-8859-5") ||
charsetString.equalsIgnoreCase("iso-8859-6") || charsetString.equalsIgnoreCase("iso-8859-6") ||
charsetString.equalsIgnoreCase("iso-8859-7") || charsetString.equalsIgnoreCase("iso-8859-7") ||
charsetString.equalsIgnoreCase("iso-8859-8") || charsetString.equalsIgnoreCase("iso-8859-8") ||
charsetString.equalsIgnoreCase("iso-8859-8-i") || charsetString.equalsIgnoreCase("iso-8859-8-i") ||
charsetString.equalsIgnoreCase("macintosh") || charsetString.equalsIgnoreCase("macintosh") ||
charsetString.equalsIgnoreCase("shift_jis") || charsetString.equalsIgnoreCase("shift_jis") ||
charsetString.equalsIgnoreCase("windows-1250") || charsetString.equalsIgnoreCase("windows-1250") ||
charsetString.equalsIgnoreCase("windows-1251") || charsetString.equalsIgnoreCase("windows-1251") ||
charsetString.equalsIgnoreCase("windows-1252") || charsetString.equalsIgnoreCase("windows-1252") ||
charsetString.equalsIgnoreCase("windows-1253") || charsetString.equalsIgnoreCase("windows-1253") ||
charsetString.equalsIgnoreCase("windows-1254") || charsetString.equalsIgnoreCase("windows-1254") ||
charsetString.equalsIgnoreCase("windows-1255") || charsetString.equalsIgnoreCase("windows-1255") ||
charsetString.equalsIgnoreCase("windows-1256") || charsetString.equalsIgnoreCase("windows-1256") ||
charsetString.equalsIgnoreCase("windows-1257") || charsetString.equalsIgnoreCase("windows-1257") ||
charsetString.equalsIgnoreCase("windows-1258") || charsetString.equalsIgnoreCase("windows-1258") ||
charsetString.equalsIgnoreCase("windows-874") || charsetString.equalsIgnoreCase("windows-874") ||
charsetString.equalsIgnoreCase("x-mac-cyrillic") || charsetString.equalsIgnoreCase("x-mac-cyrillic") ||
charsetString.equalsIgnoreCase("x-user-defined"); charsetString.equalsIgnoreCase("x-user-defined");
} }
} }

View File

@ -1,32 +1,32 @@
package io.rudefox.cold; package io.rudefox.burrow;
import com.bjdweck.bitcoin.params.INetworkParameters; import com.bjdweck.bitcoin.params.INetworkParameters;
import com.bjdweck.bitcoin.params.NetworkParameters; import com.bjdweck.bitcoin.params.NetworkParameters;
import picocli.CommandLine; import picocli.CommandLine;
@CommandLine.Command( @CommandLine.Command(
name = "rudefox-cold", name = "burrow",
synopsisSubcommandLabel = "COMMAND", synopsisSubcommandLabel = "COMMAND",
description = "Offline wallet tool", description = "Offline wallet tool",
subcommands = {MnemonicCommand.class, WalletCommand.class} subcommands = {MnemonicCommand.class, WalletCommand.class}
) )
public class RudefoxCold implements Runnable { public class RudefoxBurrow implements Runnable {
@CommandLine.Option(names = "--testnet", description = "run on Bitcoin Testnet (default: Mainnet)") @CommandLine.Option(names = "--testnet", description = "run on Bitcoin Testnet (default: Mainnet)")
boolean testnet = false; boolean testnet = false;
public INetworkParameters getNetworkParameters() { public INetworkParameters getNetworkParameters() {
return testnet ? NetworkParameters.Testnet() : NetworkParameters.Mainnet(); return testnet ? NetworkParameters.Testnet() : NetworkParameters.Mainnet();
} }
public static void main(String[] args) { public static void main(String[] args) {
new CommandLine(new RudefoxCold()).execute(args); new CommandLine(new RudefoxBurrow()).execute(args);
} }
@CommandLine.Spec @CommandLine.Spec
CommandLine.Model.CommandSpec commandSpec; CommandLine.Model.CommandSpec commandSpec;
public void run() { public void run() {
throw new CommandLine.ParameterException(commandSpec.commandLine(), "Missing required subcommand"); throw new CommandLine.ParameterException(commandSpec.commandLine(), "Missing required subcommand");
} }
} }

View File

@ -1,8 +1,8 @@
package io.rudefox.cold; package io.rudefox.burrow;
import java.nio.charset.Charset; import java.nio.charset.Charset;
public class ServiceLocator { public class ServiceLocator {
public static Charset DEFAULT_CHARSET = Charset.defaultCharset(); public static Charset DEFAULT_CHARSET = Charset.defaultCharset();
} }

View File

@ -1,137 +1,137 @@
package io.rudefox.cold; package io.rudefox.burrow;
import com.bjdweck.bitcoin.address.BitcoinAddress; import com.bjdweck.bitcoin.address.BitcoinAddress;
import com.bjdweck.bitcoin.extendedkey.AddressChain; import com.bjdweck.bitcoin.extendedkey.AddressChain;
import com.bjdweck.bitcoin.extendedkey.DerivationPath; import com.bjdweck.bitcoin.extendedkey.DerivationPath;
import com.bjdweck.bitcoin.extendedkey.KeyOrigin; import com.bjdweck.bitcoin.extendedkey.KeyOrigin;
import com.bjdweck.bitcoin.extendedkey.PrivateExtendedKey; import com.bjdweck.bitcoin.extendedkey.PrivateExtendedKey;
import com.bjdweck.bitcoin.hdwallet.Bip44Purpose; import com.bjdweck.bitcoin.hdwallet.Bip44Purpose;
import com.bjdweck.bitcoin.hdwallet.Bip44SigningAccount; import com.bjdweck.bitcoin.hdwallet.Bip44SigningAccount;
import com.bjdweck.bitcoin.hdwallet.Bip44Wallet; import com.bjdweck.bitcoin.hdwallet.Bip44Wallet;
import com.bjdweck.bitcoin.mnemonic.Mnemonic; import com.bjdweck.bitcoin.mnemonic.Mnemonic;
import com.bjdweck.bitcoin.mnemonic.Seed; import com.bjdweck.bitcoin.mnemonic.Seed;
import com.bjdweck.bitcoin.psbt.Psbt; import com.bjdweck.bitcoin.psbt.Psbt;
import com.bjdweck.bitcoin.transaction.scriptpubkey.ScriptPubKey; import com.bjdweck.bitcoin.transaction.scriptpubkey.ScriptPubKey;
import picocli.CommandLine; import picocli.CommandLine;
@SuppressWarnings("ALL") @SuppressWarnings("ALL")
@CommandLine.Command(name = "wallet", description = "BIP44 wallet operations") @CommandLine.Command(name = "wallet", description = "BIP44 wallet operations")
public class WalletCommand implements Runnable { public class WalletCommand implements Runnable {
@CommandLine.ParentCommand @CommandLine.ParentCommand
private RudefoxCold globalOptions; private RudefoxBurrow globalOptions;
@CommandLine.Option(names = {"-s", "--sentence"}, description = "mnemonic sentence", required = true) @CommandLine.Option(names = {"-s", "--sentence"}, description = "mnemonic sentence", required = true)
String sentence; String sentence;
@CommandLine.Option(names = {"-p", "--passphrase"}, description = "optional mnemonic passphrase") @CommandLine.Option(names = {"-p", "--passphrase"}, description = "optional mnemonic passphrase")
String passphrase; String passphrase;
Bip44Wallet getWallet() { Bip44Wallet getWallet() {
Seed seed = Mnemonic.fromSentence(sentence).generateSeed(passphrase); Seed seed = Mnemonic.fromSentence(sentence).generateSeed(passphrase);
return new Bip44Wallet(PrivateExtendedKey.fromSeed(seed), globalOptions.getNetworkParameters()); return new Bip44Wallet(PrivateExtendedKey.fromSeed(seed), globalOptions.getNetworkParameters());
} }
Bip44SigningAccount getAccount(Bip44Purpose purpose, int accountSeq) { Bip44SigningAccount getAccount(Bip44Purpose purpose, int accountSeq) {
return getWallet().deriveAccount(purpose, accountSeq); return getWallet().deriveAccount(purpose, accountSeq);
} }
@CommandLine.Command(name = "listxpub", description = "list xpub (default: SLIP-0132)") @CommandLine.Command(name = "listxpub", description = "list xpub (default: SLIP-0132)")
public int listXpub( public int listXpub(
@CommandLine.Option(names = {"-q", "--qrcode"}, description = "output QR code") @CommandLine.Option(names = {"-q", "--qrcode"}, description = "output QR code")
boolean isQRCode, boolean isQRCode,
@CommandLine.Option(names = {"-p", "--purpose"}, description = "purpose", @CommandLine.Option(names = {"-p", "--purpose"}, description = "purpose",
paramLabel = "[P2PKH, P2SH_P2WPKH, P2WPKH, P2SH_MultiSigP2WSH, MultiSigP2WSH]", paramLabel = "[P2PKH, P2SH_P2WPKH, P2WPKH, P2SH_MultiSigP2WSH, MultiSigP2WSH]",
defaultValue = "P2PKH") defaultValue = "P2PKH")
Bip44Purpose purpose, Bip44Purpose purpose,
@CommandLine.Option(names = {"-a", "--account"}, description = "account sequence number", @CommandLine.Option(names = {"-a", "--account"}, description = "account sequence number",
paramLabel = "n", defaultValue = "0") paramLabel = "n", defaultValue = "0")
int accountSeq) { int accountSeq) {
Bip44SigningAccount bip44SigningAccount = getAccount(purpose, accountSeq); Bip44SigningAccount bip44SigningAccount = getAccount(purpose, accountSeq);
String slip0132 = bip44SigningAccount.toXpubSlip0132Base58(); String slip0132 = bip44SigningAccount.toXpubSlip0132Base58();
String keyOrigin = bip44SigningAccount.getKeyOrigin().toString(); String keyOrigin = bip44SigningAccount.getKeyOrigin().toString();
if (isQRCode) { if (isQRCode) {
System.out.println(QRCode.toQRCode(slip0132)); System.out.println(QRCode.toQRCode(slip0132));
return 0; return 0;
} }
System.out.printf("[%s]%s%n", keyOrigin, slip0132); System.out.printf("[%s]%s%n", keyOrigin, slip0132);
return 0; return 0;
} }
@CommandLine.Command(name = "listaddress", description = "list account addresses") @CommandLine.Command(name = "listaddress", description = "list account addresses")
public int listAddress( public int listAddress(
@CommandLine.Option(names = {"-p", "--purpose"}, description = "purpose", @CommandLine.Option(names = {"-p", "--purpose"}, description = "purpose",
paramLabel = "[P2PKH, P2SH_P2WPKH, P2WPKH, P2SH_MultiSigP2WSH, MultiSigP2WSH]", paramLabel = "[P2PKH, P2SH_P2WPKH, P2WPKH, P2SH_MultiSigP2WSH, MultiSigP2WSH]",
defaultValue = "P2PKH") defaultValue = "P2PKH")
Bip44Purpose purpose, Bip44Purpose purpose,
@CommandLine.Option(names = {"-a", "--account"}, description = "account sequence number", @CommandLine.Option(names = {"-a", "--account"}, description = "account sequence number",
paramLabel = "<n>", defaultValue = "0") paramLabel = "<n>", defaultValue = "0")
int accountSeq, int accountSeq,
@CommandLine.Option(names = {"-c", "--address-chain"}, description = "address chain", @CommandLine.Option(names = {"-c", "--address-chain"}, description = "address chain",
paramLabel = "[INTERNAL|EXTERNAL]", defaultValue = "EXTERNAL") paramLabel = "[INTERNAL|EXTERNAL]", defaultValue = "EXTERNAL")
AddressChain addressChain, AddressChain addressChain,
@CommandLine.Option(names = {"-s", "--address-seq"}, description = "starting address sequence number", @CommandLine.Option(names = {"-s", "--address-seq"}, description = "starting address sequence number",
paramLabel = "<n>", defaultValue = "0") paramLabel = "<n>", defaultValue = "0")
int addressSeq, int addressSeq,
@CommandLine.Option(names = {"-n", "--count"}, description = "number of addresses to list", @CommandLine.Option(names = {"-n", "--count"}, description = "number of addresses to list",
paramLabel = "<n>", defaultValue = "10") paramLabel = "<n>", defaultValue = "10")
int count) { int count) {
Bip44SigningAccount account = getAccount(purpose, accountSeq); Bip44SigningAccount account = getAccount(purpose, accountSeq);
for (int i = 0; i < count; i++) { for (int i = 0; i < count; i++) {
int seq = addressSeq + i; int seq = addressSeq + i;
DerivationPath addressPath = DerivationPath.m().slash(addressChain.getSequence()).slash(seq); DerivationPath addressPath = DerivationPath.m().slash(addressChain.getSequence()).slash(seq);
KeyOrigin keyOrigin = account.getKeyOrigin().combineRelativePath(addressPath); KeyOrigin keyOrigin = account.getKeyOrigin().combineRelativePath(addressPath);
ScriptPubKey scriptPubKey = account.deriveScriptPubKey(addressChain, seq); ScriptPubKey scriptPubKey = account.deriveScriptPubKey(addressChain, seq);
BitcoinAddress address = scriptPubKey.getAddress(globalOptions.getNetworkParameters()); BitcoinAddress address = scriptPubKey.getAddress(globalOptions.getNetworkParameters());
System.out.printf("%s: %s%n", keyOrigin, address); System.out.printf("%s: %s%n", keyOrigin, address);
} }
return 0; return 0;
} }
@CommandLine.Command(name = "signpsbt", description = "sign a PSBT") @CommandLine.Command(name = "signpsbt", description = "sign a PSBT")
public int signPsbt( public int signPsbt(
@CommandLine.Parameters() @CommandLine.Parameters()
String psbtBase64) { String psbtBase64) {
Psbt psbt = Psbt.fromBase64(psbtBase64); Psbt psbt = Psbt.fromBase64(psbtBase64);
psbt.getValidationViolationMap(); psbt.getValidationViolationMap();
/* /*
Satoshi fee = psbt.getFee(); Satoshi fee = psbt.getFee();
Satoshi spendAmount = psbt.getSpendAmount(getWallet()); Satoshi spendAmount = psbt.getSpendAmount(getWallet());
BitcoinAddress spendAddress = psbt.getSpendAddress(getWallet()); BitcoinAddress spendAddress = psbt.getSpendAddress(getWallet());
System.out.printf("Confirm spend of %s to %s with a fee of %s", spendAmount.toBitcoinString(), spendAddress.getAddressString(), fee.toBitcoinString()); System.out.printf("Confirm spend of %s to %s with a fee of %s", spendAmount.toBitcoinString(), spendAddress.getAddressString(), fee.toBitcoinString());
*/ */
psbt.signPsbt(getWallet()); psbt.signPsbt(getWallet());
System.out.println(psbt.toBase64()); System.out.println(psbt.toBase64());
return 0; return 0;
} }
@CommandLine.Spec @CommandLine.Spec
CommandLine.Model.CommandSpec commandSpec; CommandLine.Model.CommandSpec commandSpec;
public void run() { public void run() {
throw new CommandLine.ParameterException(commandSpec.commandLine(), "Missing required wallet subcommand"); throw new CommandLine.ParameterException(commandSpec.commandLine(), "Missing required wallet subcommand");
} }
} }

View File

@ -1,64 +1,64 @@
package io.rudefox.cold; package io.rudefox.burrow;
import com.bjdweck.test.CliTestFixture; import com.bjdweck.test.CliTestFixture;
import org.junit.jupiter.api.Test; import org.junit.jupiter.api.Test;
import java.io.UnsupportedEncodingException; import java.io.UnsupportedEncodingException;
class mnemonic_8_sided_dice_tests extends CliTestFixture { class mnemonic_8_sided_dice_tests extends CliTestFixture {
@Test @Test
void with_arguments_interactive_8_sided_dice_should_generate_mnemonic_sentence() throws UnsupportedEncodingException { void with_arguments_interactive_8_sided_dice_should_generate_mnemonic_sentence() throws UnsupportedEncodingException {
withArgs("mnemonic -i8 --bits 128"); withArgs("mnemonic -i8 --bits 128");
expectedOutput("Input 11 x 8-sided dice rolls [1-8]: "); expectedOutput("Input 11 x 8-sided dice rolls [1-8]: ");
provideInput("12345678123" + EOL); provideInput("12345678123" + EOL);
expectedOutput(EOL); expectedOutput(EOL);
expectedOutput("| 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 1 | 2 | 3 |" + EOL); expectedOutput("| 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 1 | 2 | 3 |" + EOL);
expectedOutput("|000 001 010 01 | 1 100 101 110 1 | 11 000 001 010|" + EOL); expectedOutput("|000 001 010 01 | 1 100 101 110 1 | 11 000 001 010|" + EOL);
expectedOutput("| 1. ahead | 2. slight | 3. scout |" + EOL); expectedOutput("| 1. ahead | 2. slight | 3. scout |" + EOL);
expectedOutput(EOL); expectedOutput(EOL);
expectedOutput("Input 11 x 8-sided dice rolls [1-8]: "); expectedOutput("Input 11 x 8-sided dice rolls [1-8]: ");
provideInput("12345678123" + EOL); provideInput("12345678123" + EOL);
expectedOutput(EOL); expectedOutput(EOL);
expectedOutput("| 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 1 | 2 | 3 |" + EOL); expectedOutput("| 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 1 | 2 | 3 |" + EOL);
expectedOutput("|000 001 010 01 | 1 100 101 110 1 | 11 000 001 010|" + EOL); expectedOutput("|000 001 010 01 | 1 100 101 110 1 | 11 000 001 010|" + EOL);
expectedOutput("| 4. ahead | 5. slight | 6. scout |" + EOL); expectedOutput("| 4. ahead | 5. slight | 6. scout |" + EOL);
expectedOutput(EOL); expectedOutput(EOL);
expectedOutput("Input 11 x 8-sided dice rolls [1-8]: "); expectedOutput("Input 11 x 8-sided dice rolls [1-8]: ");
provideInput("12345678123" + EOL); provideInput("12345678123" + EOL);
expectedOutput(EOL); expectedOutput(EOL);
expectedOutput("| 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 1 | 2 | 3 |" + EOL); expectedOutput("| 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 1 | 2 | 3 |" + EOL);
expectedOutput("|000 001 010 01 | 1 100 101 110 1 | 11 000 001 010|" + EOL); expectedOutput("|000 001 010 01 | 1 100 101 110 1 | 11 000 001 010|" + EOL);
expectedOutput("| 7. ahead | 8. slight | 9. scout |" + EOL); expectedOutput("| 7. ahead | 8. slight | 9. scout |" + EOL);
expectedOutput(EOL); expectedOutput(EOL);
expectedOutput("Input 11 x 8-sided dice rolls [1-8]: "); expectedOutput("Input 11 x 8-sided dice rolls [1-8]: ");
provideInput("12345678123" + EOL); provideInput("12345678123" + EOL);
expectedOutput(EOL); expectedOutput(EOL);
expectedOutput("| 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 1 | 2 | 3 |" + EOL); expectedOutput("| 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 1 | 2 | 3 |" + EOL);
expectedOutput("|000 001 010 01 | 1 100 101 110 1 | 11 000 00- ---|" + EOL); expectedOutput("|000 001 010 01 | 1 100 101 110 1 | 11 000 00- ---|" + EOL);
expectedOutput("| 10. ahead | 11. slight | 12. CHECKWORD |" + EOL); expectedOutput("| 10. ahead | 11. slight | 12. CHECKWORD |" + EOL);
expectedOutput(EOL); expectedOutput(EOL);
expectedOutput("ahead slight scout ahead slight scout ahead slight scout ahead slight scan" + EOL); expectedOutput("ahead slight scout ahead slight scout ahead slight scout ahead slight scan" + EOL);
doMain(); doMain();
verifyOutput(); verifyOutput();
} }
private void doMain() { private void doMain() {
setInput(); setInput();
RudefoxCold.main(getArgs()); RudefoxBurrow.main(getArgs());
} }
} }

View File

@ -1,71 +1,71 @@
package io.rudefox.cold; package io.rudefox.burrow;
import com.bjdweck.test.CliTestFixture; import com.bjdweck.test.CliTestFixture;
import org.junit.jupiter.api.Test; import org.junit.jupiter.api.Test;
import java.io.UnsupportedEncodingException; import java.io.UnsupportedEncodingException;
import static org.junit.jupiter.api.Assertions.assertEquals; import static org.junit.jupiter.api.Assertions.assertEquals;
class mnemonic_command_tests extends CliTestFixture { class mnemonic_command_tests extends CliTestFixture {
@Test @Test
void with_arguments_interactive_dice_should_generate_mnemonic_sentence() throws UnsupportedEncodingException { void with_arguments_interactive_dice_should_generate_mnemonic_sentence() throws UnsupportedEncodingException {
withArgs("mnemonic --interactive --dice6-entropy --bits 128"); withArgs("mnemonic --interactive --dice6-entropy --bits 128");
expectedOutput("Input 50 dice rolls [1-6]: "); expectedOutput("Input 50 dice rolls [1-6]: ");
provideInput("234322343242422344161254151\r\n"); provideInput("234322343242422344161254151\r\n");
expectedOutput("Input 23 more dice rolls [1-6]: "); expectedOutput("Input 23 more dice rolls [1-6]: ");
provideInput("33116265515343114314456\r\n"); provideInput("33116265515343114314456\r\n");
expectedOutput("mountain tilt wing silk rude fox almost volume wine media verify card" + EOL); expectedOutput("mountain tilt wing silk rude fox almost volume wine media verify card" + EOL);
doMain(); doMain();
verifyOutput(); verifyOutput();
} }
@Test @Test
void with_no_arguments_should_output_mnemonic() throws UnsupportedEncodingException { void with_no_arguments_should_output_mnemonic() throws UnsupportedEncodingException {
withArgs("mnemonic"); withArgs("mnemonic");
doMain(); doMain();
assertEquals(24, getOutput().split(" ").length); assertEquals(24, getOutput().split(" ").length);
} }
@Test @Test
void with_arguments_non_interactive_dice_bits_should_generate_mnemonic_sentence() throws UnsupportedEncodingException { void with_arguments_non_interactive_dice_bits_should_generate_mnemonic_sentence() throws UnsupportedEncodingException {
withArgs("mnemonic --dice6-entropy --events 23432234324242234416125415133116265515343114314456 --bits 128"); withArgs("mnemonic --dice6-entropy --events 23432234324242234416125415133116265515343114314456 --bits 128");
expectedOutput("mountain tilt wing silk rude fox almost volume wine media verify card" + EOL); expectedOutput("mountain tilt wing silk rude fox almost volume wine media verify card" + EOL);
doMain(); doMain();
verifyOutput(); verifyOutput();
} }
@Test @Test
void with_arguments_without_interactive_dice_should_generate_mnemonic_sentence() throws UnsupportedEncodingException { void with_arguments_without_interactive_dice_should_generate_mnemonic_sentence() throws UnsupportedEncodingException {
withArgs("mnemonic --dice6-entropy --events 2343223432424223441612541513311626551534311431445623432234324242234416125415133116265515343114314456"); withArgs("mnemonic --dice6-entropy --events 2343223432424223441612541513311626551534311431445623432234324242234416125415133116265515343114314456");
expectedOutput("first welcome social broccoli nasty rather weird uncle spirit horn update pencil help rescue " + expectedOutput("first welcome social broccoli nasty rather weird uncle spirit horn update pencil help rescue " +
"grape enough fork wave eight fuel ribbon pony clean couple" + EOL); "grape enough fork wave eight fuel ribbon pony clean couple" + EOL);
doMain(); doMain();
verifyOutput(); verifyOutput();
} }
private void doMain() { private void doMain() {
setInput(); setInput();
RudefoxCold.main(getArgs()); RudefoxBurrow.main(getArgs());
} }
} }

View File

@ -1,145 +1,145 @@
package io.rudefox.cold; package io.rudefox.burrow;
import com.bjdweck.test.CliTestFixture; import com.bjdweck.test.CliTestFixture;
import org.junit.jupiter.api.Test; import org.junit.jupiter.api.Test;
import java.io.UnsupportedEncodingException; import java.io.UnsupportedEncodingException;
import java.nio.charset.StandardCharsets; import java.nio.charset.StandardCharsets;
import static org.junit.jupiter.api.Assertions.assertEquals; import static org.junit.jupiter.api.Assertions.assertEquals;
class wallet_command_tests extends CliTestFixture { class wallet_command_tests extends CliTestFixture {
static { static {
OUTPUT_CHARSET = "UTF8"; OUTPUT_CHARSET = "UTF8";
} }
@Test @Test
void with_sentence_and_passphrase_should_output_xpub() throws UnsupportedEncodingException { void with_sentence_and_passphrase_should_output_xpub() throws UnsupportedEncodingException {
withArgs(new String[] {"wallet", withArgs(new String[] {"wallet",
"--sentence","stove prefer lunch collect small orphan wasp size beyond auction guilt great", "--sentence","stove prefer lunch collect small orphan wasp size beyond auction guilt great",
"--passphrase","apple", "--passphrase","apple",
"listxpub"}); "listxpub"});
doMain(); doMain();
assertEquals("[39F8B071:m/44'/0'/0']xpub6CKy5SECeJipZid8dF3bopoMGdRzd7hMJuPzMGesZCobrMSssZyASexzXuzRTPVLcqqdyAEZJKPMGvDthgZW2Z3mPHLohxEAVbkvGKAXjqx" + EOL, getOutput()); assertEquals("[39F8B071:m/44'/0'/0']xpub6CKy5SECeJipZid8dF3bopoMGdRzd7hMJuPzMGesZCobrMSssZyASexzXuzRTPVLcqqdyAEZJKPMGvDthgZW2Z3mPHLohxEAVbkvGKAXjqx" + EOL, getOutput());
} }
@Test @Test
void with_testnet_sentence_and_passphrase_and_witness_should_output_vpub() throws UnsupportedEncodingException { void with_testnet_sentence_and_passphrase_and_witness_should_output_vpub() throws UnsupportedEncodingException {
withArgs(new String[] {"--testnet", withArgs(new String[] {"--testnet",
"wallet", "wallet",
"--sentence","icon issue absorb apology price atom bread toward worry final dune dial swing armor donkey", "--sentence","icon issue absorb apology price atom bread toward worry final dune dial swing armor donkey",
"--passphrase","apple", "--passphrase","apple",
"listxpub", "listxpub",
"--purpose","P2WPKH", "--purpose","P2WPKH",
"--account","6"}); "--account","6"});
doMain(); doMain();
assertEquals("[CBACA81C:m/84'/1'/6']vpub5Z1KP7PJ8f85AqbvPMKeFHxmDfrgimiAar3KqwCzfsw65tmFLevm4MJX9cRcoiERYkjeG5Z3VwhSf1E3fVBFF86J33CHSED45Hk6kLZumqV" + EOL, getOutput()); assertEquals("[CBACA81C:m/84'/1'/6']vpub5Z1KP7PJ8f85AqbvPMKeFHxmDfrgimiAar3KqwCzfsw65tmFLevm4MJX9cRcoiERYkjeG5Z3VwhSf1E3fVBFF86J33CHSED45Hk6kLZumqV" + EOL, getOutput());
} }
@Test @Test
void with_testnet_sentence_and_passphrase_and_witness_should_output_addresses() throws UnsupportedEncodingException { void with_testnet_sentence_and_passphrase_and_witness_should_output_addresses() throws UnsupportedEncodingException {
withArgs(new String[] {"--testnet", withArgs(new String[] {"--testnet",
"wallet", "wallet",
"--sentence","icon issue absorb apology price atom bread toward worry final dune dial swing armor donkey", "--sentence","icon issue absorb apology price atom bread toward worry final dune dial swing armor donkey",
"--passphrase","apple", "--passphrase","apple",
"listaddress", "listaddress",
"--purpose","P2WPKH", "--purpose","P2WPKH",
"--account","6", "--account","6",
"--address-chain","INTERNAL", "--address-chain","INTERNAL",
"--address-seq","3", "--address-seq","3",
"--count","5"}); "--count","5"});
doMain(); doMain();
assertEquals("CBACA81C:m/84'/1'/6'/1/3: tb1qs7cnr9sql9nef5e95jk5t7pr2qwpjsqtffnuxj" + EOL + assertEquals("CBACA81C:m/84'/1'/6'/1/3: tb1qs7cnr9sql9nef5e95jk5t7pr2qwpjsqtffnuxj" + EOL +
"CBACA81C:m/84'/1'/6'/1/4: tb1qgl947tylg3tazgdy2rm3rs4gxheue707dkg3q2" + EOL + "CBACA81C:m/84'/1'/6'/1/4: tb1qgl947tylg3tazgdy2rm3rs4gxheue707dkg3q2" + EOL +
"CBACA81C:m/84'/1'/6'/1/5: tb1q03n6q2jjeks9zaneglqz5g37xkcrzy8nh8hn6s" + EOL + "CBACA81C:m/84'/1'/6'/1/5: tb1q03n6q2jjeks9zaneglqz5g37xkcrzy8nh8hn6s" + EOL +
"CBACA81C:m/84'/1'/6'/1/6: tb1q50gylhzz2mrf76ffn7dc5zvymsd9lghj24t7ns" + EOL + "CBACA81C:m/84'/1'/6'/1/6: tb1q50gylhzz2mrf76ffn7dc5zvymsd9lghj24t7ns" + EOL +
"CBACA81C:m/84'/1'/6'/1/7: tb1qvzkez0zg6xm9z780n0358xp7nx96fh508awktw" + EOL, getOutput()); "CBACA81C:m/84'/1'/6'/1/7: tb1qvzkez0zg6xm9z780n0358xp7nx96fh508awktw" + EOL, getOutput());
} }
@Test @Test
void with_testnet_sentence_and_passphrase_and_witness_can_sign_psbt() throws UnsupportedEncodingException { void with_testnet_sentence_and_passphrase_and_witness_can_sign_psbt() throws UnsupportedEncodingException {
withArgs(new String[] {"--testnet", withArgs(new String[] {"--testnet",
"wallet", "wallet",
"--sentence","vibrant delay limit tenant prepare reflect lonely pepper dragon calm dolphin prize slide inch purse term raw eternal twin kidney scan power magnet humble", "--sentence","vibrant delay limit tenant prepare reflect lonely pepper dragon calm dolphin prize slide inch purse term raw eternal twin kidney scan power magnet humble",
"signpsbt", "signpsbt",
"cHNidP8BAHcCAAAAAS/zE49gxbTp0n4mTv0wPV7KuLtQlbzTL/E9s4b4JKl/AQAAAAD/////AkAfAAAAAAAAGXapFE9U1GY" + "cHNidP8BAHcCAAAAAS/zE49gxbTp0n4mTv0wPV7KuLtQlbzTL/E9s4b4JKl/AQAAAAD/////AkAfAAAAAAAAGXapFE9U1GY" +
"KflieNl1PUIpdTTzcXnUkiKxiyR4AAAAAABl2qRR6uJziLUo4i7ooJyzxIctMZfxb5IisAAAAAE8BBDWHzwOP9tWogAAAAN/zgjOY" + "KflieNl1PUIpdTTzcXnUkiKxiyR4AAAAAABl2qRR6uJziLUo4i7ooJyzxIctMZfxb5IisAAAAAE8BBDWHzwOP9tWogAAAAN/zgjOY" +
"H39q085n7GG+5+/Jtovkvc8WQftwMSsBV7svAhPpi88y7pGhXzOdH0iJbXJsMq3F9GNRnRyqkQ7hVEtjEI2C0R8sAACAAQAAgAAAA" + "H39q085n7GG+5+/Jtovkvc8WQftwMSsBV7svAhPpi88y7pGhXzOdH0iJbXJsMq3F9GNRnRyqkQ7hVEtjEI2C0R8sAACAAQAAgAAAA" +
"IAAAQDeAQAAAAGVcHJZzeDV1JsQv869yoS/f8eSufNKON0o56ChVEnxCwEAAABqRzBEAiAD95mxgC+CHJNt1Ui+Y/raIymIiK1fyV" + "IAAAQDeAQAAAAGVcHJZzeDV1JsQv869yoS/f8eSufNKON0o56ChVEnxCwEAAABqRzBEAiAD95mxgC+CHJNt1Ui+Y/raIymIiK1fyV" +
"c9wx0EALk/uQIgdQS27tjyujeItDg96m5GR7UD5tIz+95A4RQX4YOXXZoBIQNsvOVlbTqbZoKJbx8z94F6C/xS4faQ2XsHINcd50q" + "c9wx0EALk/uQIgdQS27tjyujeItDg96m5GR7UD5tIz+95A4RQX4YOXXZoBIQNsvOVlbTqbZoKJbx8z94F6C/xS4faQ2XsHINcd50q" +
"T5P////8CQB8AAAAAAAAWABRzMqPZdHy0JMwPKY/YRhrODmyijpzpHgAAAAAAGXapFKmSAlpRxRJXr8ywpN500rXVsxIHiKwAAAAA" + "T5P////8CQB8AAAAAAAAWABRzMqPZdHy0JMwPKY/YRhrODmyijpzpHgAAAAAAGXapFKmSAlpRxRJXr8ywpN500rXVsxIHiKwAAAAA" +
"IgYCUEaaspCm8AVu5WJ9Z7RG53iW66Btq6QODz7xfMlRogoYjYLRHywAAIABAACAAAAAgAEAAAADAAAAAAAA"}); "IgYCUEaaspCm8AVu5WJ9Z7RG53iW66Btq6QODz7xfMlRogoYjYLRHywAAIABAACAAAAAgAEAAAADAAAAAAAA"});
doMain(); doMain();
assertEquals("cHNidP8BAHcCAAAAAS/zE49gxbTp0n4mTv0wPV7KuLtQlbzTL/E9s4b4JKl/AQAAAAD/////AkAfAAAAAAAAGXapFE9U1GYKflieNl1PUIpdTTzcXnUkiKxiyR4AAAAAABl2qRR6uJziLUo4i7ooJyzxIctMZfxb5IisAAAAAE8BBDWHzwOP9tWogAAAAN/zgjOYH39q085n7GG+5+/Jtovkvc8WQftwMSsBV7svAhPpi88y7pGhXzOdH0iJbXJsMq3F9GNRnRyqkQ7hVEtjEI2C0R8sAACAAQAAgAAAAIAAAQDeAQAAAAGVcHJZzeDV1JsQv869yoS/f8eSufNKON0o56ChVEnxCwEAAABqRzBEAiAD95mxgC+CHJNt1Ui+Y/raIymIiK1fyVc9wx0EALk/uQIgdQS27tjyujeItDg96m5GR7UD5tIz+95A4RQX4YOXXZoBIQNsvOVlbTqbZoKJbx8z94F6C/xS4faQ2XsHINcd50qT5P////8CQB8AAAAAAAAWABRzMqPZdHy0JMwPKY/YRhrODmyijpzpHgAAAAAAGXapFKmSAlpRxRJXr8ywpN500rXVsxIHiKwAAAAAIgYCUEaaspCm8AVu5WJ9Z7RG53iW66Btq6QODz7xfMlRogoYjYLRHywAAIABAACAAAAAgAEAAAADAAAAIgICUEaaspCm8AVu5WJ9Z7RG53iW66Btq6QODz7xfMlRogpIMEUCIQCrgKRLUmzYL8edfZ7cltQWnLdVdU2Ta3BYbu5NR8JYwgIgDeOEdkOmOkPp7HpC4dkniXvXsvfzagmJ6pkBDd2DcgwBAAAA" + EOL, getOutput()); assertEquals("cHNidP8BAHcCAAAAAS/zE49gxbTp0n4mTv0wPV7KuLtQlbzTL/E9s4b4JKl/AQAAAAD/////AkAfAAAAAAAAGXapFE9U1GYKflieNl1PUIpdTTzcXnUkiKxiyR4AAAAAABl2qRR6uJziLUo4i7ooJyzxIctMZfxb5IisAAAAAE8BBDWHzwOP9tWogAAAAN/zgjOYH39q085n7GG+5+/Jtovkvc8WQftwMSsBV7svAhPpi88y7pGhXzOdH0iJbXJsMq3F9GNRnRyqkQ7hVEtjEI2C0R8sAACAAQAAgAAAAIAAAQDeAQAAAAGVcHJZzeDV1JsQv869yoS/f8eSufNKON0o56ChVEnxCwEAAABqRzBEAiAD95mxgC+CHJNt1Ui+Y/raIymIiK1fyVc9wx0EALk/uQIgdQS27tjyujeItDg96m5GR7UD5tIz+95A4RQX4YOXXZoBIQNsvOVlbTqbZoKJbx8z94F6C/xS4faQ2XsHINcd50qT5P////8CQB8AAAAAAAAWABRzMqPZdHy0JMwPKY/YRhrODmyijpzpHgAAAAAAGXapFKmSAlpRxRJXr8ywpN500rXVsxIHiKwAAAAAIgYCUEaaspCm8AVu5WJ9Z7RG53iW66Btq6QODz7xfMlRogoYjYLRHywAAIABAACAAAAAgAEAAAADAAAAIgICUEaaspCm8AVu5WJ9Z7RG53iW66Btq6QODz7xfMlRogpIMEUCIQCrgKRLUmzYL8edfZ7cltQWnLdVdU2Ta3BYbu5NR8JYwgIgDeOEdkOmOkPp7HpC4dkniXvXsvfzagmJ6pkBDd2DcgwBAAAA" + EOL, getOutput());
} }
@Test @Test
void with_sentence_without_passphrase_should_output_xpub() throws UnsupportedEncodingException { void with_sentence_without_passphrase_should_output_xpub() throws UnsupportedEncodingException {
withArgs(new String[] {"wallet", withArgs(new String[] {"wallet",
"--sentence","ordinary debate stomach mix poverty upset amateur small sadness female general fabric", "--sentence","ordinary debate stomach mix poverty upset amateur small sadness female general fabric",
"listxpub"}); "listxpub"});
doMain(); doMain();
assertEquals("[A6932DE1:m/44'/0'/0']xpub6BjcsZDafUukJNMhD1Porw4SZB1RkiCE7EsQWJGVaMcZd9qXyawcjeVa28SJt5WNNAKJGGUbebfgzyrsWPTsLRkWmMNwrThaq8umcp7Yzn8" + EOL, getOutput()); assertEquals("[A6932DE1:m/44'/0'/0']xpub6BjcsZDafUukJNMhD1Porw4SZB1RkiCE7EsQWJGVaMcZd9qXyawcjeVa28SJt5WNNAKJGGUbebfgzyrsWPTsLRkWmMNwrThaq8umcp7Yzn8" + EOL, getOutput());
} }
@Test @Test
void with_sentence_without_passphrase_should_output_xpub_qrcode() throws UnsupportedEncodingException { void with_sentence_without_passphrase_should_output_xpub_qrcode() throws UnsupportedEncodingException {
withArgs(new String[] {"wallet", withArgs(new String[] {"wallet",
"--sentence","ordinary debate stomach mix poverty upset amateur small sadness female general fabric", "--sentence","ordinary debate stomach mix poverty upset amateur small sadness female general fabric",
"listxpub", "listxpub",
"--qrcode"}); "--qrcode"});
String ExpectedQRCode = String ExpectedQRCode =
"█▀▀▀▀▀█ █ ▄ ██▀▄▄ ▀▀ ▀ █ ██▄ █▀▀▀▀▀█\n" + "█▀▀▀▀▀█ █ ▄ ██▀▄▄ ▀▀ ▀ █ ██▄ █▀▀▀▀▀█\n" +
"█ ███ █ █▄ ▄▀▄█ ▄▄█▄▀███▄█▄▀█▄██ █ ███ █\n" + "█ ███ █ █▄ ▄▀▄█ ▄▄█▄▀███▄█▄▀█▄██ █ ███ █\n" +
"█ ▀▀▀ █ ▄▄▄█ ▀▀ ▄ ▀█▀▄▀▀ ▄▀▀▀ █ █ ▀▀▀ █\n" + "█ ▀▀▀ █ ▄▄▄█ ▀▀ ▄ ▀█▀▄▀▀ ▄▀▀▀ █ █ ▀▀▀ █\n" +
"▀▀▀▀▀▀▀ ▀ ▀ ▀ ▀▄█▄▀ ▀▄▀ ▀ ▀ ▀▄█▄█ ▀▀▀▀▀▀▀\n" + "▀▀▀▀▀▀▀ ▀ ▀ ▀ ▀▄█▄▀ ▀▄▀ ▀ ▀ ▀▄█▄█ ▀▀▀▀▀▀▀\n" +
"▀▀▄ ▀▀▀▄ ▀ ▀█▄▄▄▄▀▀█ █ ▀█▄▀█▀▄█▀ █▄▀█▀▀\n" + "▀▀▄ ▀▀▀▄ ▀ ▀█▄▄▄▄▀▀█ █ ▀█▄▀█▀▄█▀ █▄▀█▀▀\n" +
"▀▄ ▀█ ▀▄▄▀ █ ▄▀██▄▄▀▄ ▀█▀ █▀█▀▄ █▀███ █\n" + "▀▄ ▀█ ▀▄▄▀ █ ▄▀██▄▄▀▄ ▀█▀ █▀█▀▄ █▀███ █\n" +
"███▄▀▀▀▄█ █▀▀█▀▀▄▄▀█▀ █▄▀█ ▄█ ▄ ▀▄▄▄█ ▄▄\n" + "███▄▀▀▀▄█ █▀▀█▀▀▄▄▀█▀ █▄▀█ ▄█ ▄ ▀▄▄▄█ ▄▄\n" +
" ▀ ▄▀▄█▄▄ ▄ ▄ ▀█▄█▄█▀██▀ █ █ ▄▀▄█ █▀▄█\n" + " ▀ ▄▀▄█▄▄ ▄ ▄ ▀█▄█▄█▀██▀ █ █ ▄▀▄█ █▀▄█\n" +
" ▀▄ ▄█▀█▀ ▀▀ ▀█▄█ ▀▄██▄▄▀█▀ ▄█▀▀▄████ \n" + " ▀▄ ▄█▀█▀ ▀▀ ▀█▄█ ▀▄██▄▄▀█▀ ▄█▀▀▄████ \n" +
"▄ ▄▀█▄▀ █████▀▀▄ ▀ ██▀██ ▀▀ ▄▄▄▀▄▀▄▀ ▀▄▀\n" + "▄ ▄▀█▄▀ █████▀▀▄ ▀ ██▀██ ▀▀ ▄▄▄▀▄▀▄▀ ▀▄▀\n" +
" █▄▄ ▀▀█▄█ ▄█▄ ▀ ▄▄▄▀█▄▀▄▄ ▄ ▄ ▀▄▄▄▄ ██\n" + " █▄▄ ▀▀█▄█ ▄█▄ ▀ ▄▄▄▀█▄▀▄▄ ▄ ▄ ▀▄▄▄▄ ██\n" +
"▀▀█▀▀▄▀ ▄█▀▀ █▄█▄ ▀▄ ██ ▄█ █▀▄ ▀██▀▄▀▄█\n" + "▀▀█▀▀▄▀ ▄█▀▀ █▄█▄ ▀▄ ██ ▄█ █▀▄ ▀██▀▄▀▄█\n" +
"▀▄█ ▄▀ ▄ █▄▄▄▀▄▄ █▀▄▀ ▀▀ ███▀▀▀▀▄█▄▀█ ▀\n" + "▀▄█ ▄▀ ▄ █▄▄▄▀▄▄ █▀▄▀ ▀▀ ███▀▀▀▀▄█▄▀█ ▀\n" +
"▀ ▄█ ▀▀ ▄▀ █▄ ▄▀ █▄ ▀▄█▀▀ ▀██▀ █ ▀▄▀▄▀▄▀\n" + "▀ ▄█ ▀▀ ▄▀ █▄ ▄▀ █▄ ▀▄█▀▀ ▀██▀ █ ▀▄▀▄▀▄▀\n" +
"█▀█▄▀▀▀ ██▄ ▀▀▀ ▄▄▀█▀█▀▄ ▄▄▄▄ ▀ █▄▄▄▀███\n" + "█▀█▄▀▀▀ ██▄ ▀▀▀ ▄▄▀█▀█▀▄ ▄▄▄▄ ▀ █▄▄▄▀███\n" +
" ▀██▀▀▀█▀ ▄ ▀█ █▄▄██ ▀ ▀ █▄ ▀▄▀ █▀ ▀\n" + " ▀██▀▀▀█▀ ▄ ▀█ █▄▄██ ▀ ▀ █▄ ▀▄▀ █▀ ▀\n" +
"▀▀ ▀▀▄▀▀█ ▀ ███ ▀▄▀▀▀▄▄▄█ █▀▀▀███▀▀\n" + "▀▀ ▀▀▄▀▀█ ▀ ███ ▀▄▀▀▀▄▄▄█ █▀▀▀███▀▀\n" +
"█▀▀▀▀▀█ ▄█▀▀ █▀█▄ ▀▄▄▄▄█▀ ▄▀ ▄ ██ ▀ █▄▀ ▀\n" + "█▀▀▀▀▀█ ▄█▀▀ █▀█▄ ▀▄▄▄▄█▀ ▄▀ ▄ ██ ▀ █▄▀ ▀\n" +
"█ ███ █ ▀▀█▀▄█▄ ▀▄▄██▀▄█▄ ▀█▀▄▀██▀▀▀▀▄▀▄\n" + "█ ███ █ ▀▀█▀▄█▄ ▀▄▄██▀▄█▄ ▀█▀▄▀██▀▀▀▀▄▀▄\n" +
"█ ▀▀▀ █ ▄█ █▀▀▄█▄█▄ ▄ █▀▀▄▀▄█ █▄▄▀▄▄█▀▄█\n" + "█ ▀▀▀ █ ▄█ █▀▀▄█▄█▄ ▄ █▀▀▄▀▄█ █▄▄▀▄▄█▀▄█\n" +
"▀▀▀▀▀▀▀ ▀ ▀ ▀▀ ▀ ▀ ▀ ▀ ▀ ▀ ▀ ▀▀ " + EOL; "▀▀▀▀▀▀▀ ▀ ▀ ▀▀ ▀ ▀ ▀ ▀ ▀ ▀ ▀ ▀▀ " + EOL;
doMain(); doMain();
assertEquals(ExpectedQRCode, getOutput()); assertEquals(ExpectedQRCode, getOutput());
} }
private void doMain() { private void doMain() {
setInput(); setInput();
ServiceLocator.DEFAULT_CHARSET = StandardCharsets.UTF_8; ServiceLocator.DEFAULT_CHARSET = StandardCharsets.UTF_8;
RudefoxCold.main(getArgs()); RudefoxBurrow.main(getArgs());
} }
} }