mirror of
https://github.com/bitcoin/bitcoin.git
synced 2025-01-26 11:13:23 -03:00
2068f089c8
-BEGIN VERIFY SCRIPT- mkdir -p src/util git mv src/util.h src/util/system.h git mv src/util.cpp src/util/system.cpp git mv src/utilmemory.h src/util/memory.h git mv src/utilmoneystr.h src/util/moneystr.h git mv src/utilmoneystr.cpp src/util/moneystr.cpp git mv src/utilstrencodings.h src/util/strencodings.h git mv src/utilstrencodings.cpp src/util/strencodings.cpp git mv src/utiltime.h src/util/time.h git mv src/utiltime.cpp src/util/time.cpp sed -i 's/<util\.h>/<util\/system\.h>/g' $(git ls-files 'src/*.h' 'src/*.cpp') sed -i 's/<utilmemory\.h>/<util\/memory\.h>/g' $(git ls-files 'src/*.h' 'src/*.cpp') sed -i 's/<utilmoneystr\.h>/<util\/moneystr\.h>/g' $(git ls-files 'src/*.h' 'src/*.cpp') sed -i 's/<utilstrencodings\.h>/<util\/strencodings\.h>/g' $(git ls-files 'src/*.h' 'src/*.cpp') sed -i 's/<utiltime\.h>/<util\/time\.h>/g' $(git ls-files 'src/*.h' 'src/*.cpp') sed -i 's/BITCOIN_UTIL_H/BITCOIN_UTIL_SYSTEM_H/g' src/util/system.h sed -i 's/BITCOIN_UTILMEMORY_H/BITCOIN_UTIL_MEMORY_H/g' src/util/memory.h sed -i 's/BITCOIN_UTILMONEYSTR_H/BITCOIN_UTIL_MONEYSTR_H/g' src/util/moneystr.h sed -i 's/BITCOIN_UTILSTRENCODINGS_H/BITCOIN_UTIL_STRENCODINGS_H/g' src/util/strencodings.h sed -i 's/BITCOIN_UTILTIME_H/BITCOIN_UTIL_TIME_H/g' src/util/time.h sed -i 's/ util\.\(h\|cpp\)/ util\/system\.\1/g' src/Makefile.am sed -i 's/utilmemory\.\(h\|cpp\)/util\/memory\.\1/g' src/Makefile.am sed -i 's/utilmoneystr\.\(h\|cpp\)/util\/moneystr\.\1/g' src/Makefile.am sed -i 's/utilstrencodings\.\(h\|cpp\)/util\/strencodings\.\1/g' src/Makefile.am sed -i 's/utiltime\.\(h\|cpp\)/util\/time\.\1/g' src/Makefile.am sed -i 's/-> util ->/-> util\/system ->/' test/lint/lint-circular-dependencies.sh sed -i 's/src\/util\.cpp/src\/util\/system\.cpp/g' test/lint/lint-format-strings.py test/lint/lint-locale-dependence.sh sed -i 's/src\/utilmoneystr\.cpp/src\/util\/moneystr\.cpp/g' test/lint/lint-locale-dependence.sh sed -i 's/src\/utilstrencodings\.\(h\|cpp\)/src\/util\/strencodings\.\1/g' test/lint/lint-locale-dependence.sh sed -i 's/src\\utilstrencodings\.cpp/src\\util\\strencodings\.cpp/' build_msvc/libbitcoinconsensus/libbitcoinconsensus.vcxproj -END VERIFY SCRIPT-
213 lines
7.7 KiB
C++
213 lines
7.7 KiB
C++
// Copyright (c) 2011-2018 The Bitcoin Core developers
|
|
// Distributed under the MIT software license, see the accompanying
|
|
// file COPYING or http://www.opensource.org/licenses/mit-license.php.
|
|
|
|
//
|
|
// Wraps dumb protocol buffer paymentRequest
|
|
// with some extra methods
|
|
//
|
|
|
|
#include <qt/paymentrequestplus.h>
|
|
|
|
#include <util/system.h>
|
|
|
|
#include <stdexcept>
|
|
|
|
#include <openssl/x509_vfy.h>
|
|
|
|
#include <QDateTime>
|
|
#include <QDebug>
|
|
#include <QSslCertificate>
|
|
|
|
class SSLVerifyError : public std::runtime_error
|
|
{
|
|
public:
|
|
explicit SSLVerifyError(std::string err) : std::runtime_error(err) { }
|
|
};
|
|
|
|
bool PaymentRequestPlus::parse(const QByteArray& data)
|
|
{
|
|
bool parseOK = paymentRequest.ParseFromArray(data.data(), data.size());
|
|
if (!parseOK) {
|
|
qWarning() << "PaymentRequestPlus::parse: Error parsing payment request";
|
|
return false;
|
|
}
|
|
if (paymentRequest.payment_details_version() > 1) {
|
|
qWarning() << "PaymentRequestPlus::parse: Received up-version payment details, version=" << paymentRequest.payment_details_version();
|
|
return false;
|
|
}
|
|
|
|
parseOK = details.ParseFromString(paymentRequest.serialized_payment_details());
|
|
if (!parseOK)
|
|
{
|
|
qWarning() << "PaymentRequestPlus::parse: Error parsing payment details";
|
|
paymentRequest.Clear();
|
|
return false;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
bool PaymentRequestPlus::SerializeToString(std::string* output) const
|
|
{
|
|
return paymentRequest.SerializeToString(output);
|
|
}
|
|
|
|
bool PaymentRequestPlus::IsInitialized() const
|
|
{
|
|
return paymentRequest.IsInitialized();
|
|
}
|
|
|
|
bool PaymentRequestPlus::getMerchant(X509_STORE* certStore, QString& merchant) const
|
|
{
|
|
merchant.clear();
|
|
|
|
if (!IsInitialized())
|
|
return false;
|
|
|
|
// One day we'll support more PKI types, but just
|
|
// x509 for now:
|
|
const EVP_MD* digestAlgorithm = nullptr;
|
|
if (paymentRequest.pki_type() == "x509+sha256") {
|
|
digestAlgorithm = EVP_sha256();
|
|
}
|
|
else if (paymentRequest.pki_type() == "x509+sha1") {
|
|
digestAlgorithm = EVP_sha1();
|
|
}
|
|
else if (paymentRequest.pki_type() == "none") {
|
|
qWarning() << "PaymentRequestPlus::getMerchant: Payment request: pki_type == none";
|
|
return false;
|
|
}
|
|
else {
|
|
qWarning() << "PaymentRequestPlus::getMerchant: Payment request: unknown pki_type " << QString::fromStdString(paymentRequest.pki_type());
|
|
return false;
|
|
}
|
|
|
|
payments::X509Certificates certChain;
|
|
if (!certChain.ParseFromString(paymentRequest.pki_data())) {
|
|
qWarning() << "PaymentRequestPlus::getMerchant: Payment request: error parsing pki_data";
|
|
return false;
|
|
}
|
|
|
|
std::vector<X509*> certs;
|
|
const QDateTime currentTime = QDateTime::currentDateTime();
|
|
for (int i = 0; i < certChain.certificate_size(); i++) {
|
|
QByteArray certData(certChain.certificate(i).data(), certChain.certificate(i).size());
|
|
QSslCertificate qCert(certData, QSsl::Der);
|
|
if (currentTime < qCert.effectiveDate() || currentTime > qCert.expiryDate()) {
|
|
qWarning() << "PaymentRequestPlus::getMerchant: Payment request: certificate expired or not yet active: " << qCert;
|
|
return false;
|
|
}
|
|
if (qCert.isBlacklisted()) {
|
|
qWarning() << "PaymentRequestPlus::getMerchant: Payment request: certificate blacklisted: " << qCert;
|
|
return false;
|
|
}
|
|
const unsigned char *data = (const unsigned char *)certChain.certificate(i).data();
|
|
X509 *cert = d2i_X509(nullptr, &data, certChain.certificate(i).size());
|
|
if (cert)
|
|
certs.push_back(cert);
|
|
}
|
|
if (certs.empty()) {
|
|
qWarning() << "PaymentRequestPlus::getMerchant: Payment request: empty certificate chain";
|
|
return false;
|
|
}
|
|
|
|
// The first cert is the signing cert, the rest are untrusted certs that chain
|
|
// to a valid root authority. OpenSSL needs them separately.
|
|
STACK_OF(X509) *chain = sk_X509_new_null();
|
|
for (int i = certs.size() - 1; i > 0; i--) {
|
|
sk_X509_push(chain, certs[i]);
|
|
}
|
|
X509 *signing_cert = certs[0];
|
|
|
|
// Now create a "store context", which is a single use object for checking,
|
|
// load the signing cert into it and verify.
|
|
X509_STORE_CTX *store_ctx = X509_STORE_CTX_new();
|
|
if (!store_ctx) {
|
|
qWarning() << "PaymentRequestPlus::getMerchant: Payment request: error creating X509_STORE_CTX";
|
|
return false;
|
|
}
|
|
|
|
char *website = nullptr;
|
|
bool fResult = true;
|
|
try
|
|
{
|
|
if (!X509_STORE_CTX_init(store_ctx, certStore, signing_cert, chain))
|
|
{
|
|
int error = X509_STORE_CTX_get_error(store_ctx);
|
|
throw SSLVerifyError(X509_verify_cert_error_string(error));
|
|
}
|
|
|
|
// Now do the verification!
|
|
int result = X509_verify_cert(store_ctx);
|
|
if (result != 1) {
|
|
int error = X509_STORE_CTX_get_error(store_ctx);
|
|
// For testing payment requests, we allow self signed root certs!
|
|
if (!(error == X509_V_ERR_DEPTH_ZERO_SELF_SIGNED_CERT && gArgs.GetBoolArg("-allowselfsignedrootcertificates", DEFAULT_SELFSIGNED_ROOTCERTS))) {
|
|
throw SSLVerifyError(X509_verify_cert_error_string(error));
|
|
} else {
|
|
qDebug() << "PaymentRequestPlus::getMerchant: Allowing self signed root certificate, because -allowselfsignedrootcertificates is true.";
|
|
}
|
|
}
|
|
X509_NAME *certname = X509_get_subject_name(signing_cert);
|
|
|
|
// Valid cert; check signature:
|
|
payments::PaymentRequest rcopy(paymentRequest); // Copy
|
|
rcopy.set_signature(std::string(""));
|
|
std::string data_to_verify; // Everything but the signature
|
|
rcopy.SerializeToString(&data_to_verify);
|
|
|
|
#if HAVE_DECL_EVP_MD_CTX_NEW
|
|
EVP_MD_CTX *ctx = EVP_MD_CTX_new();
|
|
if (!ctx) throw SSLVerifyError("Error allocating OpenSSL context.");
|
|
#else
|
|
EVP_MD_CTX _ctx;
|
|
EVP_MD_CTX *ctx;
|
|
ctx = &_ctx;
|
|
#endif
|
|
EVP_PKEY *pubkey = X509_get_pubkey(signing_cert);
|
|
EVP_MD_CTX_init(ctx);
|
|
if (!EVP_VerifyInit_ex(ctx, digestAlgorithm, nullptr) ||
|
|
!EVP_VerifyUpdate(ctx, data_to_verify.data(), data_to_verify.size()) ||
|
|
!EVP_VerifyFinal(ctx, (const unsigned char*)paymentRequest.signature().data(), (unsigned int)paymentRequest.signature().size(), pubkey)) {
|
|
throw SSLVerifyError("Bad signature, invalid payment request.");
|
|
}
|
|
#if HAVE_DECL_EVP_MD_CTX_NEW
|
|
EVP_MD_CTX_free(ctx);
|
|
#endif
|
|
|
|
// OpenSSL API for getting human printable strings from certs is baroque.
|
|
int textlen = X509_NAME_get_text_by_NID(certname, NID_commonName, nullptr, 0);
|
|
website = new char[textlen + 1];
|
|
if (X509_NAME_get_text_by_NID(certname, NID_commonName, website, textlen + 1) == textlen && textlen > 0) {
|
|
merchant = website;
|
|
}
|
|
else {
|
|
throw SSLVerifyError("Bad certificate, missing common name.");
|
|
}
|
|
// TODO: detect EV certificates and set merchant = business name instead of unfriendly NID_commonName ?
|
|
}
|
|
catch (const SSLVerifyError& err) {
|
|
fResult = false;
|
|
qWarning() << "PaymentRequestPlus::getMerchant: SSL error: " << err.what();
|
|
}
|
|
|
|
delete[] website;
|
|
X509_STORE_CTX_free(store_ctx);
|
|
for (unsigned int i = 0; i < certs.size(); i++)
|
|
X509_free(certs[i]);
|
|
|
|
return fResult;
|
|
}
|
|
|
|
QList<std::pair<CScript,CAmount> > PaymentRequestPlus::getPayTo() const
|
|
{
|
|
QList<std::pair<CScript,CAmount> > result;
|
|
for (int i = 0; i < details.outputs_size(); i++)
|
|
{
|
|
const unsigned char* scriptStr = (const unsigned char*)details.outputs(i).script().data();
|
|
CScript s(scriptStr, scriptStr+details.outputs(i).script().size());
|
|
|
|
result.append(std::make_pair(s, details.outputs(i).amount()));
|
|
}
|
|
return result;
|
|
}
|