mirror of
https://github.com/feather-wallet/feather.git
synced 2024-11-16 17:27:38 +00:00
dice: add entropy from system
This commit is contained in:
parent
60eda75b55
commit
9056047b1b
6 changed files with 63 additions and 16 deletions
|
@ -6,6 +6,10 @@
|
|||
|
||||
#include <cmath>
|
||||
#include <algorithm>
|
||||
#include <sodium/core.h>
|
||||
#include <sodium/utils.h>
|
||||
#include <sodium/randombytes.h>
|
||||
#include "polyseed/polyseed.h"
|
||||
|
||||
#include <QPasswordDigestor>
|
||||
|
||||
|
@ -17,6 +21,10 @@ SeedDiceDialog::SeedDiceDialog(QWidget *parent)
|
|||
{
|
||||
ui->setupUi(this);
|
||||
|
||||
if (sodium_init() == -1) {
|
||||
throw std::runtime_error("sodium_init failed");
|
||||
}
|
||||
|
||||
ui->frame_dice->hide();
|
||||
ui->frame_coinflip->hide();
|
||||
|
||||
|
@ -77,13 +85,28 @@ SeedDiceDialog::SeedDiceDialog(QWidget *parent)
|
|||
});
|
||||
|
||||
connect(ui->btn_createPolyseed, &QPushButton::clicked, [this]{
|
||||
QByteArray salt = "POLYSEED";
|
||||
QByteArray data = m_rolls.join(" ").toUtf8();
|
||||
qsizetype rolls_length = 0;
|
||||
for (const auto& roll : m_rolls) {
|
||||
rolls_length += roll.length() + 1;
|
||||
}
|
||||
|
||||
QByteArray data;
|
||||
data.reserve(rolls_length + POLYSEED_RANDBYTES);
|
||||
data = m_rolls.join(" ").toUtf8();
|
||||
|
||||
// Get 19 bytes of entropy from the system
|
||||
char random[POLYSEED_RANDBYTES] = {};
|
||||
randombytes_buf(&random, POLYSEED_RANDBYTES);
|
||||
|
||||
data.append(random, POLYSEED_RANDBYTES);
|
||||
|
||||
// We already have enough entropy assuming unbiased throws, but a few extra rounds can't hurt
|
||||
// Polyseed requests 19 bytes of random data and discards two bits (for a total of 150 bits)
|
||||
QByteArray salt = "POLYSEED";
|
||||
m_key = QPasswordDigestor::deriveKeyPbkdf2(QCryptographicHash::Sha256, data, salt, 2048, 19);
|
||||
|
||||
sodium_memzero(data.data(), data.size());
|
||||
sodium_memzero(&random, POLYSEED_RANDBYTES);
|
||||
|
||||
this->accept();
|
||||
});
|
||||
|
||||
|
@ -146,7 +169,7 @@ bool SeedDiceDialog::updateEntropy() {
|
|||
double entropy = entropyPerRoll() * m_rolls.length();
|
||||
ui->label_entropy->setText(QString("%1 / %2 bits").arg(QString::number(entropy, 'f', 2), QString::number(entropyNeeded)));
|
||||
|
||||
return entropy > entropyNeeded;
|
||||
return entropy >= entropyNeeded;
|
||||
}
|
||||
|
||||
void SeedDiceDialog::updateRolls() {
|
||||
|
@ -172,12 +195,16 @@ void SeedDiceDialog::setEnableMethodSelection(bool enabled) {
|
|||
ui->spin_sides->setEnabled(enabled);
|
||||
}
|
||||
|
||||
bool SeedDiceDialog::finished() {
|
||||
return updateEntropy();
|
||||
}
|
||||
|
||||
const char* SeedDiceDialog::getSecret() {
|
||||
return m_key.data();
|
||||
}
|
||||
|
||||
const QString& SeedDiceDialog::getMnemonic() {
|
||||
return m_mnemonic;
|
||||
void SeedDiceDialog::wipeSecret() {
|
||||
sodium_memzero(m_key.data(), m_key.length());
|
||||
}
|
||||
|
||||
SeedDiceDialog::~SeedDiceDialog() = default;
|
|
@ -20,9 +20,9 @@ public:
|
|||
explicit SeedDiceDialog(QWidget *parent);
|
||||
~SeedDiceDialog() override;
|
||||
|
||||
bool finished();
|
||||
const char* getSecret();
|
||||
|
||||
const QString& getMnemonic();
|
||||
void wipeSecret();
|
||||
|
||||
private:
|
||||
void addFlip(bool heads);
|
||||
|
@ -40,7 +40,6 @@ private:
|
|||
QStringList m_rolls;
|
||||
QByteArray m_key;
|
||||
int entropyNeeded = 152; // Polyseed requests 19 bytes of random data
|
||||
QString m_mnemonic;
|
||||
};
|
||||
|
||||
|
||||
|
|
|
@ -13,8 +13,6 @@
|
|||
|
||||
#include <QString>
|
||||
|
||||
#define POLYSEED_RANDBYTES 19
|
||||
|
||||
namespace polyseed {
|
||||
|
||||
static std::locale locale;
|
||||
|
|
|
@ -1,11 +1,16 @@
|
|||
// SPDX-License-Identifier: BSD-2-Clause
|
||||
// SPDX-FileCopyrightText: Copyright 2021 tevador <tevador@gmail.com>
|
||||
|
||||
#ifndef FEATHER_POLYSEED_H
|
||||
#define FEATHER_POLYSEED_H
|
||||
|
||||
#include <polyseed.h>
|
||||
#include <vector>
|
||||
#include <stdexcept>
|
||||
#include <string>
|
||||
|
||||
#define POLYSEED_RANDBYTES 19
|
||||
|
||||
namespace polyseed {
|
||||
|
||||
class data;
|
||||
|
@ -123,3 +128,5 @@ namespace polyseed {
|
|||
polyseed_coin m_coin;
|
||||
};
|
||||
}
|
||||
|
||||
#endif // FEATHER_POLYSEED_H
|
|
@ -30,14 +30,21 @@ PageWalletSeed::PageWalletSeed(WizardFields *fields, QWidget *parent)
|
|||
ui->frame_invalidSeed->setInfo(icons()->icon("warning"), "Feather was unable to generate a valid seed.\n"
|
||||
"This should never happen.\n"
|
||||
"Please contact the developers immediately.");
|
||||
ui->frame_invalidSeed->hide();
|
||||
|
||||
QShortcut *shortcut = new QShortcut(QKeySequence("Ctrl+K"), this);
|
||||
QObject::connect(shortcut, &QShortcut::activated, [&](){
|
||||
SeedDiceDialog dialog{this};
|
||||
int r = dialog.exec();
|
||||
if (r == QDialog::Accepted) {
|
||||
if (!dialog.finished()) {
|
||||
this->onError();
|
||||
Utils::showError(this, "Unable to create polyseed using additional entropy", "Not enough entropy was collected", {"You have found a bug. Please contact the developers."});
|
||||
return;
|
||||
}
|
||||
|
||||
this->generateSeed(dialog.getSecret());
|
||||
dialog.wipeSecret();
|
||||
Utils::showInfo(this, "Polyseed created successfully using additional entropy");
|
||||
}
|
||||
});
|
||||
|
||||
|
@ -51,6 +58,9 @@ PageWalletSeed::PageWalletSeed(WizardFields *fields, QWidget *parent)
|
|||
}
|
||||
|
||||
void PageWalletSeed::initializePage() {
|
||||
ui->frame_invalidSeed->hide();
|
||||
ui->frame_seedDisplay->show();
|
||||
|
||||
this->generateSeed();
|
||||
this->setTitle(m_fields->modeText);
|
||||
}
|
||||
|
@ -77,9 +87,7 @@ void PageWalletSeed::generateSeed(const char* secret) {
|
|||
this->displaySeed(mnemonic);
|
||||
|
||||
if (!m_seed.errorString.isEmpty()) {
|
||||
ui->frame_invalidSeed->show();
|
||||
ui->frame_seedDisplay->hide();
|
||||
m_seedError = true;
|
||||
this->onError();
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -122,6 +130,13 @@ void PageWalletSeed::onOptionsClicked() {
|
|||
m_fields->showSetSeedPassphrasePage = checkbox.isChecked();
|
||||
}
|
||||
|
||||
void PageWalletSeed::onError() {
|
||||
ui->frame_invalidSeed->show();
|
||||
ui->frame_seedDisplay->hide();
|
||||
m_seedError = true;
|
||||
this->completeChanged();
|
||||
}
|
||||
|
||||
int PageWalletSeed::nextId() const {
|
||||
if (m_fields->showSetSeedPassphrasePage) {
|
||||
return WalletWizard::Page_SetSeedPassphrase;
|
||||
|
|
|
@ -32,6 +32,7 @@ private:
|
|||
void seedRoulette(int count);
|
||||
void generateSeed(const char* secret = nullptr);
|
||||
void onOptionsClicked();
|
||||
void onError();
|
||||
|
||||
signals:
|
||||
void createWallet();
|
||||
|
|
Loading…
Reference in a new issue