2014-12-17 02:47:57 +01:00
|
|
|
// Copyright (c) 2009-2014 The Bitcoin Core developers
|
2014-09-05 13:11:11 +02:00
|
|
|
// Distributed under the MIT software license, see the accompanying
|
|
|
|
// file COPYING or http://www.opensource.org/licenses/mit-license.php.
|
|
|
|
|
2013-04-13 00:13:08 -05:00
|
|
|
#include "paymentservertests.h"
|
2013-07-22 16:50:39 +10:00
|
|
|
|
|
|
|
#include "optionsmodel.h"
|
|
|
|
#include "paymentrequestdata.h"
|
2013-04-13 00:13:08 -05:00
|
|
|
|
2014-12-05 09:39:23 +01:00
|
|
|
#include "random.h"
|
2013-07-22 16:50:39 +10:00
|
|
|
#include "util.h"
|
2014-09-24 23:32:36 -04:00
|
|
|
#include "utilstrencodings.h"
|
2013-07-22 16:50:39 +10:00
|
|
|
|
2013-04-13 00:13:08 -05:00
|
|
|
#include <openssl/x509.h>
|
|
|
|
#include <openssl/x509_vfy.h>
|
2013-11-14 19:21:16 +01:00
|
|
|
|
2013-04-13 00:13:08 -05:00
|
|
|
#include <QFileOpenEvent>
|
|
|
|
#include <QTemporaryFile>
|
2013-07-22 16:50:39 +10: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 13:11:11 +02:00
|
|
|
&sigCatcher, SLOT(getRecipient(SendCoinsRecipient)));
|
2013-07-22 16:50:39 +10:00
|
|
|
|
|
|
|
// Write data to a temp file:
|
|
|
|
QTemporaryFile f;
|
|
|
|
f.open();
|
|
|
|
f.write((const char*)&data[0], data.size());
|
|
|
|
f.close();
|
|
|
|
|
2013-11-14 19:21:16 +01: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 16:50:39 +10:00
|
|
|
QFileOpenEvent event(f.fileName());
|
2013-11-14 19:21:16 +01: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 16:50:39 +10:00
|
|
|
|
|
|
|
QObject::disconnect(server, SIGNAL(receivedPaymentRequest(SendCoinsRecipient)),
|
2014-09-05 13:11:11 +02:00
|
|
|
&sigCatcher, SLOT(getRecipient(SendCoinsRecipient)));
|
2013-07-22 16:50:39 +10:00
|
|
|
|
|
|
|
// Return results from sigCatcher
|
|
|
|
return sigCatcher.recipient;
|
|
|
|
}
|
|
|
|
|
|
|
|
void PaymentServerTests::paymentServerTests()
|
|
|
|
{
|
2014-06-19 15:10:04 +02:00
|
|
|
SelectParams(CBaseChainParams::MAIN);
|
2013-07-22 16:50:39 +10:00
|
|
|
OptionsModel optionsModel;
|
|
|
|
PaymentServer* server = new PaymentServer(NULL, false);
|
|
|
|
X509_STORE* caStore = X509_STORE_new();
|
|
|
|
X509_STORE_add_cert(caStore, parse_b64der_cert(caCert_BASE64));
|
|
|
|
PaymentServer::LoadRootCAs(caStore);
|
2013-08-24 15:07:17 +02:00
|
|
|
server->setOptionsModel(&optionsModel);
|
2013-07-22 16:50:39 +10:00
|
|
|
server->uiReady();
|
|
|
|
|
|
|
|
// Now feed PaymentRequests to server, and observe signals it produces:
|
|
|
|
std::vector<unsigned char> data = DecodeBase64(paymentrequest1_BASE64);
|
|
|
|
SendCoinsRecipient r = handleRequest(server, data);
|
|
|
|
QString merchant;
|
|
|
|
r.paymentRequest.getMerchant(caStore, merchant);
|
|
|
|
QCOMPARE(merchant, QString("testmerchant.org"));
|
|
|
|
|
|
|
|
// Version of the above, with an expired certificate:
|
|
|
|
data = DecodeBase64(paymentrequest2_BASE64);
|
|
|
|
r = handleRequest(server, data);
|
|
|
|
r.paymentRequest.getMerchant(caStore, merchant);
|
|
|
|
QCOMPARE(merchant, QString(""));
|
|
|
|
|
|
|
|
// Long certificate chain:
|
|
|
|
data = DecodeBase64(paymentrequest3_BASE64);
|
|
|
|
r = handleRequest(server, data);
|
|
|
|
r.paymentRequest.getMerchant(caStore, merchant);
|
|
|
|
QCOMPARE(merchant, QString("testmerchant8.org"));
|
|
|
|
|
|
|
|
// Long certificate chain, with an expired certificate in the middle:
|
|
|
|
data = DecodeBase64(paymentrequest4_BASE64);
|
|
|
|
r = handleRequest(server, data);
|
|
|
|
r.paymentRequest.getMerchant(caStore, merchant);
|
|
|
|
QCOMPARE(merchant, QString(""));
|
|
|
|
|
|
|
|
// Validly signed, but by a CA not in our root CA list:
|
|
|
|
data = DecodeBase64(paymentrequest5_BASE64);
|
|
|
|
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);
|
|
|
|
data = DecodeBase64(paymentrequest1_BASE64);
|
|
|
|
r = handleRequest(server, data);
|
|
|
|
r.paymentRequest.getMerchant(caStore, merchant);
|
|
|
|
QCOMPARE(merchant, QString(""));
|
|
|
|
|
2014-12-05 09:39:23 +01:00
|
|
|
// Just get some random data big enough to trigger BIP70 DoS protection
|
|
|
|
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();
|
|
|
|
// Trigger BIP70 DoS protection
|
|
|
|
QCOMPARE(PaymentServer::readPaymentRequestFromFile(tempFile.fileName(), r.paymentRequest), false);
|
|
|
|
|
2013-07-22 16:50:39 +10:00
|
|
|
delete server;
|
|
|
|
}
|
|
|
|
|
|
|
|
void RecipientCatcher::getRecipient(SendCoinsRecipient r)
|
|
|
|
{
|
|
|
|
recipient = r;
|
|
|
|
}
|