Renamed project from 'cold' to 'burrow'
This commit is contained in:
parent
ac99468a9d
commit
78c03988ea
|
@ -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 {
|
||||||
|
|
|
@ -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);
|
||||||
}
|
}
|
||||||
}
|
}
|
|
@ -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();
|
||||||
}
|
}
|
||||||
}
|
}
|
|
@ -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);
|
||||||
}
|
}
|
||||||
}
|
}
|
|
@ -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");
|
||||||
}
|
}
|
||||||
}
|
}
|
|
@ -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");
|
||||||
}
|
}
|
||||||
}
|
}
|
|
@ -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();
|
||||||
}
|
}
|
|
@ -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");
|
||||||
}
|
}
|
||||||
}
|
}
|
|
@ -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());
|
||||||
}
|
}
|
||||||
}
|
}
|
|
@ -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());
|
||||||
}
|
}
|
||||||
}
|
}
|
|
@ -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());
|
||||||
}
|
}
|
||||||
}
|
}
|
Loading…
Reference in New Issue
Block a user