From 17093dbf7258910b5b14e72e2fc2c07b47baee55 Mon Sep 17 00:00:00 2001 From: Craig Raw Date: Tue, 3 Sep 2024 12:03:53 +0200 Subject: [PATCH] add menu items to the message sign dialog to save a text file for signing, and load a signed message file --- .../sparrow/control/MessageSignDialog.java | 136 ++++++++++++++++-- .../sparrow/glyphfont/FontAwesome5.java | 1 + 2 files changed, 124 insertions(+), 13 deletions(-) diff --git a/src/main/java/com/sparrowwallet/sparrow/control/MessageSignDialog.java b/src/main/java/com/sparrowwallet/sparrow/control/MessageSignDialog.java index 886703e3..4c6f25a5 100644 --- a/src/main/java/com/sparrowwallet/sparrow/control/MessageSignDialog.java +++ b/src/main/java/com/sparrowwallet/sparrow/control/MessageSignDialog.java @@ -17,10 +17,13 @@ import com.sparrowwallet.sparrow.glyphfont.FontAwesome5Brands; import com.sparrowwallet.sparrow.io.Storage; import javafx.application.Platform; +import javafx.scene.Node; import javafx.scene.control.*; import javafx.scene.image.Image; import javafx.scene.image.ImageView; import javafx.scene.layout.VBox; +import javafx.stage.FileChooser; +import javafx.stage.Stage; import org.controlsfx.control.SegmentedButton; import org.controlsfx.glyphfont.Glyph; import org.controlsfx.validation.ValidationResult; @@ -32,17 +35,21 @@ import tornadofx.control.Fieldset; import tornadofx.control.Form; +import java.io.*; +import java.nio.charset.StandardCharsets; +import java.nio.file.Files; import java.security.SignatureException; -import java.util.Arrays; -import java.util.List; -import java.util.Locale; -import java.util.Optional; +import java.util.*; +import java.util.regex.Matcher; +import java.util.regex.Pattern; import static com.sparrowwallet.sparrow.AppServices.showErrorDialog; public class MessageSignDialog extends Dialog { private static final Logger log = LoggerFactory.getLogger(MessageSignDialog.class); + private static final Pattern signedMessagePattern = Pattern.compile("-----BEGIN BITCOIN SIGNED MESSAGE-----\\r?\\n(.*)\\r?\\n-----BEGIN BITCOIN SIGNATURE-----\\r?\\n(.*)\\r?\\n(.*)\\r?\\n-----END BITCOIN SIGNATURE-----\r?\n?"); + private final TextField address; private final TextArea message; private final TextArea signature; @@ -104,7 +111,8 @@ public MessageSignDialog(Wallet wallet, WalletNode walletNode, String title, Str this.wallet = wallet; this.walletNode = walletNode; - final DialogPane dialogPane = getDialogPane(); + final DialogPane dialogPane = new MessageSignDialogPane(); + setDialogPane(dialogPane); dialogPane.getStylesheets().add(AppServices.class.getResource("general.css").toExternalForm()); dialogPane.getStylesheets().add(AppServices.class.getResource("dialog.css").toExternalForm()); AppServices.setStageIcon(dialogPane.getScene().getWindow()); @@ -199,13 +207,7 @@ public MessageSignDialog(Wallet wallet, WalletNode walletNode, String title, Str } else { dialogPane.getButtonTypes().addAll(showQrButtonType, signButtonType, verifyButtonType, doneButtonType); - Button showQrButton = (Button) dialogPane.lookupButton(showQrButtonType); - showQrButton.setDisable(wallet == null); - showQrButton.setGraphic(getGlyph(new Glyph(FontAwesome5.FONT_NAME, FontAwesome5.Glyph.QRCODE))); - showQrButton.setGraphicTextGap(5); - showQrButton.setOnAction(event -> { - showQr(); - }); + Node showQrButton = dialogPane.lookupButton(showQrButtonType); Button signButton = (Button) dialogPane.lookupButton(signButtonType); signButton.setDisable(!canSign); @@ -267,7 +269,7 @@ public MessageSignDialog(Wallet wallet, WalletNode walletNode, String title, Str AppServices.onEscapePressed(dialogPane.getScene(), () -> setResult(ButtonBar.ButtonData.CANCEL_CLOSE)); AppServices.moveToActiveWindowScreen(this); - setResultConverter(dialogButton -> dialogButton == showQrButtonType || dialogButton == signButtonType || dialogButton == verifyButtonType ? ButtonBar.ButtonData.APPLY : dialogButton.getButtonData()); + setResultConverter(dialogButton -> dialogButton == signButtonType || dialogButton == verifyButtonType ? ButtonBar.ButtonData.APPLY : dialogButton.getButtonData()); Platform.runLater(() -> { if(address.getText().isEmpty()) { @@ -495,6 +497,81 @@ private void scanQr() { } } + private void exportFile() { + if(walletNode == null) { + AppServices.showErrorDialog("Address not in wallet", "The provided address is not present in the currently selected wallet."); + return; + } + + StringJoiner joiner = new StringJoiner("\n"); + joiner.add(message.getText().trim().replaceAll("\r*\n*", "")); + //Note we can expect a single keystore due to the check in the constructor + KeyDerivation firstDerivation = walletNode.getWallet().getKeystores().get(0).getKeyDerivation(); + joiner.add(KeyDerivation.writePath(firstDerivation.extend(walletNode.getDerivation()).getDerivation(), true)); + joiner.add(walletNode.getWallet().getScriptType().toString()); + + Stage window = new Stage(); + + FileChooser fileChooser = new FileChooser(); + fileChooser.setTitle("Save Text File"); + fileChooser.setInitialFileName("signmessage.txt"); + AppServices.moveToActiveWindowScreen(window, 800, 450); + File file = fileChooser.showSaveDialog(window); + if(file != null) { + if(!file.getName().toLowerCase(Locale.ROOT).endsWith(".txt")) { + file = new File(file.getAbsolutePath() + ".txt"); + } + + try(BufferedWriter writer = new BufferedWriter(new FileWriter(file, StandardCharsets.UTF_8))) { + writer.write(joiner.toString()); + } catch(IOException e) { + log.error("Error saving signing message", e); + AppServices.showErrorDialog("Error saving signing message", "Cannot write to " + file.getAbsolutePath()); + } + } + } + + private void importFile() { + Stage window = new Stage(); + + FileChooser fileChooser = new FileChooser(); + fileChooser.setTitle("Open Signed Text File"); + fileChooser.getExtensionFilters().addAll( + new FileChooser.ExtensionFilter("All Files", org.controlsfx.tools.Platform.getCurrent().equals(org.controlsfx.tools.Platform.UNIX) ? "*" : "*.*"), + new FileChooser.ExtensionFilter("Text Files", "*.txt") + ); + + AppServices.moveToActiveWindowScreen(window, 800, 450); + File file = fileChooser.showOpenDialog(window); + + if(file != null) { + try { + String content = Files.readString(file.toPath(), StandardCharsets.UTF_8); + Matcher matcher = signedMessagePattern.matcher(content); + if(matcher.matches()) { + String signedMessage = matcher.group(1); + String signedAddress = matcher.group(2); + String signedSignature = matcher.group(3); + + if(!signedMessage.trim().equals(message.getText().trim().replaceAll("\r*\n*", ""))) { + AppServices.showErrorDialog("Incorrect Message", "The file contained a different message of:\n\n" + signedMessage); + return; + } else if(!signedAddress.trim().equals(address.getText().trim())) { + AppServices.showErrorDialog("Incorrect Address", "The file contained a different address of:\n\n" + signedAddress); + return; + } + + signature.setText(signedSignature); + } else { + signature.setText(content); + } + } catch(IOException e) { + log.error("Error loading signed message", e); + AppServices.showErrorDialog("Error loading signed message", e.getMessage()); + } + } + } + protected Glyph getSignGlyph() { if(wallet != null) { if(wallet.containsSource(KeystoreSource.HW_USB)) { @@ -539,4 +616,37 @@ public void openWallets(OpenWalletsEvent event) { decryptWalletService.start(); } } + + private class MessageSignDialogPane extends DialogPane { + @Override + protected Node createButton(ButtonType buttonType) { + if(buttonType.getButtonData() == ButtonBar.ButtonData.LEFT) { + SplitMenuButton signByButton = new SplitMenuButton(); + signByButton.setText("Sign by QR"); + signByButton.setDisable(wallet == null); + signByButton.setGraphic(getGlyph(new Glyph(FontAwesome5.FONT_NAME, FontAwesome5.Glyph.QRCODE))); + signByButton.setGraphicTextGap(5); + signByButton.setOnAction(event -> { + showQr(); + }); + MenuItem exportFile = new MenuItem("Sign by File..."); + exportFile.setGraphic(getGlyph(new Glyph(FontAwesome5.FONT_NAME, FontAwesome5.Glyph.FILE_EXPORT))); + exportFile.setOnAction(event -> { + exportFile(); + }); + MenuItem importFile = new MenuItem("Load Signed File..."); + importFile.setGraphic(getGlyph(new Glyph(FontAwesome5.FONT_NAME, FontAwesome5.Glyph.FILE_IMPORT))); + importFile.setOnAction(event -> { + importFile(); + }); + signByButton.getItems().addAll(exportFile, importFile); + final ButtonBar.ButtonData buttonData = buttonType.getButtonData(); + ButtonBar.setButtonData(signByButton, buttonData); + + return signByButton; + } + + return super.createButton(buttonType); + } + } } diff --git a/src/main/java/com/sparrowwallet/sparrow/glyphfont/FontAwesome5.java b/src/main/java/com/sparrowwallet/sparrow/glyphfont/FontAwesome5.java index 7ff74ff4..c5caf9e6 100644 --- a/src/main/java/com/sparrowwallet/sparrow/glyphfont/FontAwesome5.java +++ b/src/main/java/com/sparrowwallet/sparrow/glyphfont/FontAwesome5.java @@ -40,6 +40,7 @@ public static enum Glyph implements INamedCharacter { FEATHER_ALT('\uf56b'), FILE_CSV('\uf6dd'), FILE_IMPORT('\uf56f'), + FILE_EXPORT('\uf56e'), FILE_PDF('\uf1c1'), HAND_HOLDING('\uf4bd'), HAND_HOLDING_MEDICAL('\ue05c'),