2014-12-17 01:47:57 +00:00
|
|
|
// Copyright (c) 2009-2014 The Bitcoin Core developers
|
2014-09-05 11:11:11 +00:00
|
|
|
// Distributed under the MIT software license, see the accompanying
|
|
|
|
// file COPYING or http://www.opensource.org/licenses/mit-license.php.
|
|
|
|
|
2013-04-13 05:13:08 +00:00
|
|
|
#include "paymentservertests.h"
|
2013-07-22 06:50:39 +00:00
|
|
|
|
|
|
|
#include "optionsmodel.h"
|
|
|
|
#include "paymentrequestdata.h"
|
2013-04-13 05:13:08 +00:00
|
|
|
|
2015-01-09 13:25:43 +00:00
|
|
|
#include "amount.h"
|
2014-12-05 08:39:23 +00:00
|
|
|
#include "random.h"
|
2015-01-09 13:25:43 +00:00
|
|
|
#include "script/script.h"
|
|
|
|
#include "script/standard.h"
|
2013-07-22 06:50:39 +00:00
|
|
|
#include "util.h"
|
2014-09-25 03:32:36 +00:00
|
|
|
#include "utilstrencodings.h"
|
2013-07-22 06:50:39 +00:00
|
|
|
|
2013-04-13 05:13:08 +00:00
|
|
|
#include <openssl/x509.h>
|
|
|
|
#include <openssl/x509_vfy.h>
|
2013-11-14 18:21:16 +00:00
|
|
|
|
2013-04-13 05:13:08 +00:00
|
|
|
#include <QFileOpenEvent>
|
|
|
|
#include <QTemporaryFile>
|
2013-07-22 06:50:39 +00:00
|
|
|
|
|
|
|
X509 *parse_b64der_cert(const char* cert_data)
|
|
|
|
{
|
|
|
|
std::vector<unsigned char> data = DecodeBase64(cert_data);
|
|
|
|
assert(data.size() > 0);
|
|
|
|
const unsigned char* dptr = &data[0];
|
|
|
|
X509 *cert = d2i_X509(NULL, &dptr, data.size());
|
|
|
|
assert(cert);
|
|
|
|
return cert;
|
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// Test payment request handling
|
|
|
|
//
|
|
|
|
|
|
|
|
static SendCoinsRecipient handleRequest(PaymentServer* server, std::vector<unsigned char>& data)
|
|
|
|
{
|
|
|
|
RecipientCatcher sigCatcher;
|
|
|
|
QObject::connect(server, SIGNAL(receivedPaymentRequest(SendCoinsRecipient)),
|
2014-09-05 11:11:11 +00:00
|
|
|
&sigCatcher, SLOT(getRecipient(SendCoinsRecipient)));
|
2013-07-22 06:50:39 +00:00
|
|
|
|
|
|
|
// Write data to a temp file:
|
|
|
|
QTemporaryFile f;
|
|
|
|
f.open();
|
|
|
|
f.write((const char*)&data[0], data.size());
|
|
|
|
f.close();
|
|
|
|
|
2013-11-14 18:21:16 +00:00
|
|
|
// Create a QObject, install event filter from PaymentServer
|
|
|
|
// and send a file open event to the object
|
|
|
|
QObject object;
|
|
|
|
object.installEventFilter(server);
|
2013-07-22 06:50:39 +00:00
|
|
|
QFileOpenEvent event(f.fileName());
|
2013-11-14 18:21:16 +00:00
|
|
|
// If sending the event fails, this will cause sigCatcher to be empty,
|
|
|
|
// which will lead to a test failure anyway.
|
|
|
|
QCoreApplication::sendEvent(&object, &event);
|
2013-07-22 06:50:39 +00:00
|
|
|
|
|
|
|
QObject::disconnect(server, SIGNAL(receivedPaymentRequest(SendCoinsRecipient)),
|
2014-09-05 11:11:11 +00:00
|
|
|
&sigCatcher, SLOT(getRecipient(SendCoinsRecipient)));
|
2013-07-22 06:50:39 +00:00
|
|
|
|
|
|
|
// Return results from sigCatcher
|
|
|
|
return sigCatcher.recipient;
|
|
|
|
}
|
|
|
|
|
|
|
|
void PaymentServerTests::paymentServerTests()
|
|
|
|
{
|
2014-06-19 13:10:04 +00:00
|
|
|
SelectParams(CBaseChainParams::MAIN);
|
2013-07-22 06:50:39 +00:00
|
|
|
OptionsModel optionsModel;
|
|
|
|
PaymentServer* server = new PaymentServer(NULL, false);
|
|
|
|
X509_STORE* caStore = X509_STORE_new();
|
2015-01-11 18:42:47 +00:00
|
|
|
X509_STORE_add_cert(caStore, parse_b64der_cert(caCert1_BASE64));
|
2013-07-22 06:50:39 +00:00
|
|
|
PaymentServer::LoadRootCAs(caStore);
|
2013-08-24 13:07:17 +00:00
|
|
|
server->setOptionsModel(&optionsModel);
|
2013-07-22 06:50:39 +00:00
|
|
|
server->uiReady();
|
|
|
|
|
2015-01-11 18:42:47 +00:00
|
|
|
std::vector<unsigned char> data;
|
|
|
|
SendCoinsRecipient r;
|
2013-07-22 06:50:39 +00:00
|
|
|
QString merchant;
|
2015-01-11 18:42:47 +00:00
|
|
|
|
|
|
|
// Now feed PaymentRequests to server, and observe signals it produces
|
|
|
|
|
|
|
|
// This payment request validates directly against the
|
|
|
|
// caCert1 certificate authority:
|
|
|
|
data = DecodeBase64(paymentrequest1_cert1_BASE64);
|
|
|
|
r = handleRequest(server, data);
|
2013-07-22 06:50:39 +00:00
|
|
|
r.paymentRequest.getMerchant(caStore, merchant);
|
|
|
|
QCOMPARE(merchant, QString("testmerchant.org"));
|
|
|
|
|
2015-01-11 18:42:47 +00:00
|
|
|
// Signed, but expired, merchant cert in the request:
|
|
|
|
data = DecodeBase64(paymentrequest2_cert1_BASE64);
|
2013-07-22 06:50:39 +00:00
|
|
|
r = handleRequest(server, data);
|
|
|
|
r.paymentRequest.getMerchant(caStore, merchant);
|
|
|
|
QCOMPARE(merchant, QString(""));
|
|
|
|
|
2015-01-11 18:42:47 +00:00
|
|
|
// 10-long certificate chain, all intermediates valid:
|
|
|
|
data = DecodeBase64(paymentrequest3_cert1_BASE64);
|
2013-07-22 06:50:39 +00:00
|
|
|
r = handleRequest(server, data);
|
|
|
|
r.paymentRequest.getMerchant(caStore, merchant);
|
|
|
|
QCOMPARE(merchant, QString("testmerchant8.org"));
|
|
|
|
|
|
|
|
// Long certificate chain, with an expired certificate in the middle:
|
2015-01-11 18:42:47 +00:00
|
|
|
data = DecodeBase64(paymentrequest4_cert1_BASE64);
|
2013-07-22 06:50:39 +00:00
|
|
|
r = handleRequest(server, data);
|
|
|
|
r.paymentRequest.getMerchant(caStore, merchant);
|
|
|
|
QCOMPARE(merchant, QString(""));
|
|
|
|
|
|
|
|
// Validly signed, but by a CA not in our root CA list:
|
2015-01-11 18:42:47 +00:00
|
|
|
data = DecodeBase64(paymentrequest5_cert1_BASE64);
|
2013-07-22 06:50:39 +00:00
|
|
|
r = handleRequest(server, data);
|
|
|
|
r.paymentRequest.getMerchant(caStore, merchant);
|
|
|
|
QCOMPARE(merchant, QString(""));
|
|
|
|
|
|
|
|
// Try again with no root CA's, verifiedMerchant should be empty:
|
|
|
|
caStore = X509_STORE_new();
|
|
|
|
PaymentServer::LoadRootCAs(caStore);
|
2015-01-11 18:42:47 +00:00
|
|
|
data = DecodeBase64(paymentrequest1_cert1_BASE64);
|
2013-07-22 06:50:39 +00:00
|
|
|
r = handleRequest(server, data);
|
|
|
|
r.paymentRequest.getMerchant(caStore, merchant);
|
|
|
|
QCOMPARE(merchant, QString(""));
|
|
|
|
|
2015-01-12 07:43:08 +00:00
|
|
|
// Load second root certificate
|
|
|
|
caStore = X509_STORE_new();
|
|
|
|
X509_STORE_add_cert(caStore, parse_b64der_cert(caCert2_BASE64));
|
|
|
|
PaymentServer::LoadRootCAs(caStore);
|
|
|
|
|
|
|
|
QByteArray byteArray;
|
|
|
|
|
|
|
|
// For the tests below we just need the payment request data from
|
|
|
|
// paymentrequestdata.h parsed + stored in r.paymentRequest.
|
|
|
|
//
|
|
|
|
// These tests require us to bypass the following normal client execution flow
|
|
|
|
// shown below to be able to explicitly just trigger a certain condition!
|
|
|
|
//
|
|
|
|
// handleRequest()
|
|
|
|
// -> PaymentServer::eventFilter()
|
|
|
|
// -> PaymentServer::handleURIOrFile()
|
|
|
|
// -> PaymentServer::readPaymentRequestFromFile()
|
|
|
|
// -> PaymentServer::processPaymentRequest()
|
|
|
|
|
|
|
|
// Contains a testnet paytoaddress, so payment request network doesn't match client network:
|
|
|
|
data = DecodeBase64(paymentrequest1_cert2_BASE64);
|
|
|
|
byteArray = QByteArray((const char*)&data[0], data.size());
|
|
|
|
r.paymentRequest.parse(byteArray);
|
|
|
|
// Ensure the request is initialized, because network "main" is default, even for
|
|
|
|
// uninizialized payment requests and that will fail our test here.
|
|
|
|
QVERIFY(r.paymentRequest.IsInitialized());
|
|
|
|
QCOMPARE(PaymentServer::verifyNetwork(r.paymentRequest.getDetails()), false);
|
|
|
|
|
2015-01-08 13:42:04 +00:00
|
|
|
// Expired payment request (expires is set to 1 = 1970-01-01 00:00:01):
|
|
|
|
data = DecodeBase64(paymentrequest2_cert2_BASE64);
|
|
|
|
byteArray = QByteArray((const char*)&data[0], data.size());
|
|
|
|
r.paymentRequest.parse(byteArray);
|
|
|
|
// Ensure the request is initialized
|
|
|
|
QVERIFY(r.paymentRequest.IsInitialized());
|
|
|
|
// compares 1 < GetTime() == false (treated as expired payment request)
|
|
|
|
QCOMPARE(PaymentServer::verifyExpired(r.paymentRequest.getDetails()), true);
|
|
|
|
|
|
|
|
// Unexpired payment request (expires is set to 0x7FFFFFFFFFFFFFFF = max. int64_t):
|
|
|
|
// 9223372036854775807 (uint64), 9223372036854775807 (int64_t) and -1 (int32_t)
|
|
|
|
// -1 is 1969-12-31 23:59:59 (for a 32 bit time values)
|
|
|
|
data = DecodeBase64(paymentrequest3_cert2_BASE64);
|
|
|
|
byteArray = QByteArray((const char*)&data[0], data.size());
|
|
|
|
r.paymentRequest.parse(byteArray);
|
|
|
|
// Ensure the request is initialized
|
|
|
|
QVERIFY(r.paymentRequest.IsInitialized());
|
|
|
|
// compares 9223372036854775807 < GetTime() == false (treated as unexpired payment request)
|
|
|
|
QCOMPARE(PaymentServer::verifyExpired(r.paymentRequest.getDetails()), false);
|
|
|
|
|
|
|
|
// Unexpired payment request (expires is set to 0x8000000000000000 > max. int64_t, allowed uint64):
|
|
|
|
// 9223372036854775808 (uint64), -9223372036854775808 (int64_t) and 0 (int32_t)
|
|
|
|
// 0 is 1970-01-01 00:00:00 (for a 32 bit time values)
|
|
|
|
data = DecodeBase64(paymentrequest4_cert2_BASE64);
|
|
|
|
byteArray = QByteArray((const char*)&data[0], data.size());
|
|
|
|
r.paymentRequest.parse(byteArray);
|
|
|
|
// Ensure the request is initialized
|
|
|
|
QVERIFY(r.paymentRequest.IsInitialized());
|
|
|
|
// compares -9223372036854775808 < GetTime() == true (treated as expired payment request)
|
|
|
|
QCOMPARE(PaymentServer::verifyExpired(r.paymentRequest.getDetails()), true);
|
|
|
|
|
|
|
|
// Test BIP70 DoS protection:
|
2014-12-05 08:39:23 +00:00
|
|
|
unsigned char randData[BIP70_MAX_PAYMENTREQUEST_SIZE + 1];
|
|
|
|
GetRandBytes(randData, sizeof(randData));
|
|
|
|
// Write data to a temp file:
|
|
|
|
QTemporaryFile tempFile;
|
|
|
|
tempFile.open();
|
|
|
|
tempFile.write((const char*)randData, sizeof(randData));
|
|
|
|
tempFile.close();
|
|
|
|
QCOMPARE(PaymentServer::readPaymentRequestFromFile(tempFile.fileName(), r.paymentRequest), false);
|
|
|
|
|
2015-01-09 13:25:43 +00:00
|
|
|
// Payment request with amount overflow (amount is set to 21000001 BTC):
|
|
|
|
data = DecodeBase64(paymentrequest5_cert2_BASE64);
|
|
|
|
byteArray = QByteArray((const char*)&data[0], data.size());
|
|
|
|
r.paymentRequest.parse(byteArray);
|
|
|
|
// Ensure the request is initialized
|
|
|
|
QVERIFY(r.paymentRequest.IsInitialized());
|
|
|
|
// Extract address and amount from the request
|
|
|
|
QList<std::pair<CScript, CAmount> > sendingTos = r.paymentRequest.getPayTo();
|
2015-07-14 11:59:05 +00:00
|
|
|
Q_FOREACH (const PAIRTYPE(CScript, CAmount)& sendingTo, sendingTos) {
|
2015-01-09 13:25:43 +00:00
|
|
|
CTxDestination dest;
|
|
|
|
if (ExtractDestination(sendingTo.first, dest))
|
|
|
|
QCOMPARE(PaymentServer::verifyAmount(sendingTo.second), false);
|
|
|
|
}
|
|
|
|
|
2013-07-22 06:50:39 +00:00
|
|
|
delete server;
|
|
|
|
}
|
|
|
|
|
|
|
|
void RecipientCatcher::getRecipient(SendCoinsRecipient r)
|
|
|
|
{
|
|
|
|
recipient = r;
|
|
|
|
}
|