2019-12-30 06:39:22 -03:00
|
|
|
// Copyright (c) 2011-2019 The Bitcoin Core developers
|
2014-12-05 05:40:58 -03:00
|
|
|
// Distributed under the MIT software license, see the accompanying
|
2013-02-11 20:52:30 -03:00
|
|
|
// file COPYING or http://www.opensource.org/licenses/mit-license.php.
|
|
|
|
|
2017-11-06 15:12:47 -03:00
|
|
|
#if defined(HAVE_CONFIG_H)
|
|
|
|
#include <config/bitcoin-config.h>
|
|
|
|
#endif
|
|
|
|
|
2017-11-09 21:57:53 -03:00
|
|
|
#include <qt/paymentserver.h>
|
|
|
|
|
|
|
|
#include <qt/bitcoinunits.h>
|
|
|
|
#include <qt/guiutil.h>
|
|
|
|
#include <qt/optionsmodel.h>
|
|
|
|
|
|
|
|
#include <chainparams.h>
|
2018-04-07 04:42:02 -03:00
|
|
|
#include <interfaces/node.h>
|
2017-11-09 21:57:53 -03:00
|
|
|
#include <policy/policy.h>
|
2017-09-19 22:12:25 -03:00
|
|
|
#include <key_io.h>
|
2017-11-09 21:57:53 -03:00
|
|
|
#include <ui_interface.h>
|
2018-10-22 19:51:11 -03:00
|
|
|
#include <util/system.h>
|
2017-11-09 21:57:53 -03:00
|
|
|
#include <wallet/wallet.h>
|
2013-04-13 02:13:08 -03:00
|
|
|
|
|
|
|
#include <cstdlib>
|
2018-04-02 15:31:40 -03:00
|
|
|
#include <memory>
|
2013-04-13 02:13:08 -03:00
|
|
|
|
2013-07-23 02:52:24 -04:00
|
|
|
#include <QApplication>
|
2013-02-11 20:52:30 -03:00
|
|
|
#include <QByteArray>
|
|
|
|
#include <QDataStream>
|
2013-07-22 02:50:39 -04:00
|
|
|
#include <QDateTime>
|
2013-02-11 20:52:30 -03:00
|
|
|
#include <QDebug>
|
2013-07-22 02:50:39 -04:00
|
|
|
#include <QFile>
|
2013-02-11 20:52:30 -03:00
|
|
|
#include <QFileOpenEvent>
|
|
|
|
#include <QHash>
|
2013-07-22 02:50:39 -04:00
|
|
|
#include <QList>
|
2013-02-11 20:52:30 -03:00
|
|
|
#include <QLocalServer>
|
|
|
|
#include <QLocalSocket>
|
2013-04-13 02:13:08 -03:00
|
|
|
#include <QStringList>
|
2013-07-22 02:50:39 -04:00
|
|
|
#include <QUrlQuery>
|
2013-02-11 20:52:30 -03:00
|
|
|
|
|
|
|
const int BITCOIN_IPC_CONNECT_TIMEOUT = 1000; // milliseconds
|
|
|
|
const QString BITCOIN_IPC_PREFIX("bitcoin:");
|
2013-07-22 02:50:39 -04:00
|
|
|
|
2013-02-11 20:52:30 -03:00
|
|
|
//
|
|
|
|
// Create a name that is unique for:
|
|
|
|
// testnet / non-testnet
|
|
|
|
// data directory
|
|
|
|
//
|
|
|
|
static QString ipcServerName()
|
|
|
|
{
|
|
|
|
QString name("BitcoinQt");
|
|
|
|
|
|
|
|
// Append a simple hash of the datadir
|
|
|
|
// Note that GetDataDir(true) returns a different path
|
|
|
|
// for -testnet versus main net
|
2016-11-07 08:11:59 -03:00
|
|
|
QString ddir(GUIUtil::boostPathToQString(GetDataDir(true)));
|
2013-02-11 20:52:30 -03:00
|
|
|
name.append(QString::number(qHash(ddir)));
|
|
|
|
|
|
|
|
return name;
|
|
|
|
}
|
|
|
|
|
|
|
|
//
|
2013-08-30 19:11:12 -04:00
|
|
|
// We store payment URIs and requests received before
|
2013-02-11 20:52:30 -03:00
|
|
|
// the main GUI window is up and ready to ask the user
|
|
|
|
// to send payment.
|
2013-07-22 02:50:39 -04:00
|
|
|
|
2019-10-02 17:55:52 -03:00
|
|
|
static QSet<QString> savedPaymentRequests;
|
2013-07-22 02:50:39 -04:00
|
|
|
|
2013-02-11 20:52:30 -03:00
|
|
|
//
|
|
|
|
// Sending to the server is done synchronously, at startup.
|
|
|
|
// If the server isn't already running, startup continues,
|
|
|
|
// and the items in savedPaymentRequest will be handled
|
|
|
|
// when uiReady() is called.
|
|
|
|
//
|
2013-11-15 21:54:29 -03:00
|
|
|
// Warning: ipcSendCommandLine() is called early in init,
|
2015-07-14 08:59:05 -03:00
|
|
|
// so don't use "Q_EMIT message()", but "QMessageBox::"!
|
2013-11-15 21:54:29 -03:00
|
|
|
//
|
2018-04-07 04:42:02 -03:00
|
|
|
void PaymentServer::ipcParseCommandLine(interfaces::Node& node, int argc, char* argv[])
|
2013-02-11 20:52:30 -03:00
|
|
|
{
|
2013-07-22 02:50:39 -04:00
|
|
|
for (int i = 1; i < argc; i++)
|
2013-02-11 20:52:30 -03:00
|
|
|
{
|
2013-07-22 02:50:39 -04:00
|
|
|
QString arg(argv[i]);
|
|
|
|
if (arg.startsWith("-"))
|
2013-02-11 20:52:30 -03:00
|
|
|
continue;
|
2013-07-22 02:50:39 -04:00
|
|
|
|
2014-11-05 07:42:51 -03:00
|
|
|
// If the bitcoin: URI contains a payment request, we are not able to detect the
|
|
|
|
// network as that would require fetching and parsing the payment request.
|
|
|
|
// That means clicking such an URI which contains a testnet payment request
|
|
|
|
// will start a mainnet instance and throw a "wrong network" error.
|
2013-10-18 06:44:05 -03:00
|
|
|
if (arg.startsWith(BITCOIN_IPC_PREFIX, Qt::CaseInsensitive)) // bitcoin: URI
|
2013-07-22 02:50:39 -04:00
|
|
|
{
|
2019-10-02 17:55:52 -03:00
|
|
|
if (savedPaymentRequests.contains(arg)) continue;
|
|
|
|
savedPaymentRequests.insert(arg);
|
2013-07-22 02:50:39 -04:00
|
|
|
|
|
|
|
SendCoinsRecipient r;
|
2014-07-02 03:42:15 -04:00
|
|
|
if (GUIUtil::parseBitcoinURI(arg, &r) && !r.address.isEmpty())
|
2013-07-22 02:50:39 -04:00
|
|
|
{
|
2015-11-28 11:04:35 -03:00
|
|
|
auto tempChainParams = CreateChainParams(CBaseChainParams::MAIN);
|
2013-07-22 02:50:39 -04:00
|
|
|
|
2017-08-22 22:02:33 -03:00
|
|
|
if (IsValidDestinationString(r.address.toStdString(), *tempChainParams)) {
|
2018-03-23 18:14:39 -03:00
|
|
|
node.selectParams(CBaseChainParams::MAIN);
|
2017-08-22 22:02:33 -03:00
|
|
|
} else {
|
2015-11-28 11:04:35 -03:00
|
|
|
tempChainParams = CreateChainParams(CBaseChainParams::TESTNET);
|
2017-08-22 22:02:33 -03:00
|
|
|
if (IsValidDestinationString(r.address.toStdString(), *tempChainParams)) {
|
2018-03-23 18:14:39 -03:00
|
|
|
node.selectParams(CBaseChainParams::TESTNET);
|
2017-08-22 22:02:33 -03:00
|
|
|
}
|
2013-07-22 02:50:39 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2013-02-11 20:52:30 -03:00
|
|
|
}
|
2014-01-05 08:37:51 -03:00
|
|
|
}
|
2013-02-11 20:52:30 -03:00
|
|
|
|
2014-01-05 08:37:51 -03:00
|
|
|
//
|
|
|
|
// Sending to the server is done synchronously, at startup.
|
|
|
|
// If the server isn't already running, startup continues,
|
|
|
|
// and the items in savedPaymentRequest will be handled
|
|
|
|
// when uiReady() is called.
|
|
|
|
//
|
|
|
|
bool PaymentServer::ipcSendCommandLine()
|
|
|
|
{
|
|
|
|
bool fResult = false;
|
2017-06-01 21:25:02 -04:00
|
|
|
for (const QString& r : savedPaymentRequests)
|
2013-02-11 20:52:30 -03:00
|
|
|
{
|
|
|
|
QLocalSocket* socket = new QLocalSocket();
|
|
|
|
socket->connectToServer(ipcServerName(), QIODevice::WriteOnly);
|
|
|
|
if (!socket->waitForConnected(BITCOIN_IPC_CONNECT_TIMEOUT))
|
2013-11-14 15:21:16 -03:00
|
|
|
{
|
|
|
|
delete socket;
|
2017-08-07 01:36:37 -04:00
|
|
|
socket = nullptr;
|
2013-02-11 20:52:30 -03:00
|
|
|
return false;
|
2013-11-14 15:21:16 -03:00
|
|
|
}
|
2013-02-11 20:52:30 -03:00
|
|
|
|
|
|
|
QByteArray block;
|
|
|
|
QDataStream out(&block, QIODevice::WriteOnly);
|
|
|
|
out.setVersion(QDataStream::Qt_4_0);
|
2013-07-22 02:50:39 -04:00
|
|
|
out << r;
|
2013-02-11 20:52:30 -03:00
|
|
|
out.device()->seek(0);
|
2014-11-05 08:05:29 -03:00
|
|
|
|
2013-02-11 20:52:30 -03:00
|
|
|
socket->write(block);
|
|
|
|
socket->flush();
|
|
|
|
socket->waitForBytesWritten(BITCOIN_IPC_CONNECT_TIMEOUT);
|
|
|
|
socket->disconnectFromServer();
|
2014-11-05 08:05:29 -03:00
|
|
|
|
2013-02-11 20:52:30 -03:00
|
|
|
delete socket;
|
2017-08-07 01:36:37 -04:00
|
|
|
socket = nullptr;
|
2013-02-11 20:52:30 -03:00
|
|
|
fResult = true;
|
|
|
|
}
|
2013-10-18 06:44:05 -03:00
|
|
|
|
2013-02-11 20:52:30 -03:00
|
|
|
return fResult;
|
|
|
|
}
|
|
|
|
|
2013-10-24 11:16:39 -03:00
|
|
|
PaymentServer::PaymentServer(QObject* parent, bool startLocalServer) :
|
|
|
|
QObject(parent),
|
|
|
|
saveURIs(true),
|
2018-07-30 06:37:09 -04:00
|
|
|
uriServer(nullptr),
|
|
|
|
optionsModel(nullptr)
|
2013-02-11 20:52:30 -03:00
|
|
|
{
|
2013-10-18 06:44:05 -03:00
|
|
|
// Install global event filter to catch QFileOpenEvents
|
|
|
|
// on Mac: sent when you click bitcoin: links
|
2015-01-12 10:26:52 -03:00
|
|
|
// other OSes: helpful when dealing with payment request files
|
2013-07-22 02:50:39 -04:00
|
|
|
if (parent)
|
|
|
|
parent->installEventFilter(this);
|
2013-02-11 20:52:30 -03:00
|
|
|
|
|
|
|
QString name = ipcServerName();
|
|
|
|
|
|
|
|
// Clean up old socket leftover from a crash:
|
|
|
|
QLocalServer::removeServer(name);
|
|
|
|
|
2013-07-22 02:50:39 -04:00
|
|
|
if (startLocalServer)
|
|
|
|
{
|
|
|
|
uriServer = new QLocalServer(this);
|
2013-02-11 20:52:30 -03:00
|
|
|
|
2013-11-14 15:21:16 -03:00
|
|
|
if (!uriServer->listen(name)) {
|
2015-07-14 08:59:05 -03:00
|
|
|
// constructor is called early in init, so don't use "Q_EMIT message()" here
|
2018-07-31 14:02:34 -04:00
|
|
|
QMessageBox::critical(nullptr, tr("Payment request error"),
|
2013-11-14 15:21:16 -03:00
|
|
|
tr("Cannot start bitcoin: click-to-pay handler"));
|
|
|
|
}
|
2013-10-22 16:27:24 -03:00
|
|
|
else {
|
2018-06-24 11:18:22 -04:00
|
|
|
connect(uriServer, &QLocalServer::newConnection, this, &PaymentServer::handleURIConnection);
|
2013-10-22 16:27:24 -03:00
|
|
|
}
|
2013-07-22 02:50:39 -04:00
|
|
|
}
|
2013-02-11 20:52:30 -03:00
|
|
|
}
|
|
|
|
|
2013-07-22 02:50:39 -04:00
|
|
|
PaymentServer::~PaymentServer()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
//
|
2019-10-18 09:35:17 -03:00
|
|
|
// OSX-specific way of handling bitcoin: URIs
|
2013-07-22 02:50:39 -04:00
|
|
|
//
|
2013-11-14 15:21:16 -03:00
|
|
|
bool PaymentServer::eventFilter(QObject *object, QEvent *event)
|
2013-02-11 20:52:30 -03:00
|
|
|
{
|
2015-01-12 10:26:52 -03:00
|
|
|
if (event->type() == QEvent::FileOpen) {
|
2013-11-14 15:21:16 -03:00
|
|
|
QFileOpenEvent *fileEvent = static_cast<QFileOpenEvent*>(event);
|
2013-07-22 02:50:39 -04:00
|
|
|
if (!fileEvent->file().isEmpty())
|
|
|
|
handleURIOrFile(fileEvent->file());
|
|
|
|
else if (!fileEvent->url().isEmpty())
|
|
|
|
handleURIOrFile(fileEvent->url().toString());
|
|
|
|
|
|
|
|
return true;
|
2013-02-11 20:52:30 -03:00
|
|
|
}
|
2013-11-14 15:21:16 -03:00
|
|
|
|
|
|
|
return QObject::eventFilter(object, event);
|
2013-02-11 20:52:30 -03:00
|
|
|
}
|
|
|
|
|
|
|
|
void PaymentServer::uiReady()
|
|
|
|
{
|
|
|
|
saveURIs = false;
|
2017-06-01 21:25:02 -04:00
|
|
|
for (const QString& s : savedPaymentRequests)
|
2013-07-22 02:50:39 -04:00
|
|
|
{
|
|
|
|
handleURIOrFile(s);
|
|
|
|
}
|
2013-02-11 20:52:30 -03:00
|
|
|
savedPaymentRequests.clear();
|
|
|
|
}
|
|
|
|
|
2013-07-22 02:50:39 -04:00
|
|
|
void PaymentServer::handleURIOrFile(const QString& s)
|
|
|
|
{
|
|
|
|
if (saveURIs)
|
|
|
|
{
|
2019-10-02 17:55:52 -03:00
|
|
|
savedPaymentRequests.insert(s);
|
2013-07-22 02:50:39 -04:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2018-03-19 08:07:17 -03:00
|
|
|
if (s.startsWith("bitcoin://", Qt::CaseInsensitive))
|
|
|
|
{
|
|
|
|
Q_EMIT message(tr("URI handling"), tr("'bitcoin://' is not a valid URI. Use 'bitcoin:' instead."),
|
|
|
|
CClientUIInterface::MSG_ERROR);
|
|
|
|
}
|
|
|
|
else if (s.startsWith(BITCOIN_IPC_PREFIX, Qt::CaseInsensitive)) // bitcoin: URI
|
2013-07-22 02:50:39 -04:00
|
|
|
{
|
2013-04-13 02:13:08 -03:00
|
|
|
QUrlQuery uri((QUrl(s)));
|
2018-12-30 07:06:53 -03:00
|
|
|
// normal URI
|
2013-12-09 06:48:14 -03:00
|
|
|
{
|
|
|
|
SendCoinsRecipient recipient;
|
|
|
|
if (GUIUtil::parseBitcoinURI(s, &recipient))
|
2013-11-15 21:54:29 -03:00
|
|
|
{
|
2017-08-22 22:02:33 -03:00
|
|
|
if (!IsValidDestinationString(recipient.address.toStdString())) {
|
2018-12-30 07:06:53 -03:00
|
|
|
if (uri.hasQueryItem("r")) { // payment request
|
|
|
|
Q_EMIT message(tr("URI handling"),
|
2019-10-12 13:56:25 -03:00
|
|
|
tr("Cannot process payment request because BIP70 is not supported.")+
|
2019-09-12 08:07:53 -03:00
|
|
|
tr("Due to widespread security flaws in BIP70 it's strongly recommended that any merchant instructions to switch wallets be ignored.")+
|
|
|
|
tr("If you are receiving this error you should request the merchant provide a BIP21 compatible URI."),
|
2018-12-30 07:06:53 -03:00
|
|
|
CClientUIInterface::ICON_WARNING);
|
|
|
|
}
|
2015-07-14 08:59:05 -03:00
|
|
|
Q_EMIT message(tr("URI handling"), tr("Invalid payment address %1").arg(recipient.address),
|
2013-11-15 21:54:29 -03:00
|
|
|
CClientUIInterface::MSG_ERROR);
|
|
|
|
}
|
|
|
|
else
|
2015-07-14 08:59:05 -03:00
|
|
|
Q_EMIT receivedPaymentRequest(recipient);
|
2013-11-15 21:54:29 -03:00
|
|
|
}
|
2013-12-09 06:48:14 -03:00
|
|
|
else
|
2015-07-14 08:59:05 -03:00
|
|
|
Q_EMIT message(tr("URI handling"),
|
2014-07-31 10:56:14 -04:00
|
|
|
tr("URI cannot be parsed! This can be caused by an invalid Bitcoin address or malformed URI parameters."),
|
2013-12-09 06:48:14 -03:00
|
|
|
CClientUIInterface::ICON_WARNING);
|
2013-07-22 02:50:39 -04:00
|
|
|
|
2013-12-09 06:48:14 -03:00
|
|
|
return;
|
|
|
|
}
|
2013-07-22 02:50:39 -04:00
|
|
|
}
|
|
|
|
|
2013-12-09 06:48:14 -03:00
|
|
|
if (QFile::exists(s)) // payment request file
|
2013-07-22 02:50:39 -04:00
|
|
|
{
|
2018-12-30 23:48:46 -03:00
|
|
|
Q_EMIT message(tr("Payment request file handling"),
|
2019-10-12 13:56:25 -03:00
|
|
|
tr("Cannot process payment request because BIP70 is not supported.")+
|
2019-09-12 08:07:53 -03:00
|
|
|
tr("Due to widespread security flaws in BIP70 it's strongly recommended that any merchant instructions to switch wallets be ignored.")+
|
|
|
|
tr("If you are receiving this error you should request the merchant provide a BIP21 compatible URI."),
|
2018-12-30 23:48:46 -03:00
|
|
|
CClientUIInterface::ICON_WARNING);
|
|
|
|
}
|
2013-07-22 02:50:39 -04:00
|
|
|
}
|
|
|
|
|
2013-02-11 20:52:30 -03:00
|
|
|
void PaymentServer::handleURIConnection()
|
|
|
|
{
|
|
|
|
QLocalSocket *clientConnection = uriServer->nextPendingConnection();
|
|
|
|
|
|
|
|
while (clientConnection->bytesAvailable() < (int)sizeof(quint32))
|
|
|
|
clientConnection->waitForReadyRead();
|
|
|
|
|
2018-06-24 11:18:22 -04:00
|
|
|
connect(clientConnection, &QLocalSocket::disconnected, clientConnection, &QLocalSocket::deleteLater);
|
2013-02-11 20:52:30 -03:00
|
|
|
|
|
|
|
QDataStream in(clientConnection);
|
|
|
|
in.setVersion(QDataStream::Qt_4_0);
|
|
|
|
if (clientConnection->bytesAvailable() < (int)sizeof(quint16)) {
|
|
|
|
return;
|
|
|
|
}
|
2013-10-18 06:44:05 -03:00
|
|
|
QString msg;
|
|
|
|
in >> msg;
|
2013-02-11 20:52:30 -03:00
|
|
|
|
2013-10-18 06:44:05 -03:00
|
|
|
handleURIOrFile(msg);
|
2013-07-22 02:50:39 -04:00
|
|
|
}
|
|
|
|
|
2017-11-09 12:41:15 -03:00
|
|
|
void PaymentServer::setOptionsModel(OptionsModel *_optionsModel)
|
|
|
|
{
|
|
|
|
this->optionsModel = _optionsModel;
|
|
|
|
}
|