2019-11-19 15:49:35 +01:00
|
|
|
// Copyright (c) 2009-2010 Satoshi Nakamoto
|
|
|
|
// Copyright (c) 2009-2020 The Bitcoin Core developers
|
|
|
|
// Distributed under the MIT software license, see the accompanying
|
|
|
|
// file COPYING or http://www.opensource.org/licenses/mit-license.php.
|
|
|
|
|
|
|
|
#include <hash.h> // For CHashWriter
|
2019-11-22 17:17:29 +01:00
|
|
|
#include <key.h> // For CKey
|
2019-11-19 15:49:35 +01:00
|
|
|
#include <key_io.h> // For DecodeDestination()
|
|
|
|
#include <pubkey.h> // For CPubKey
|
|
|
|
#include <script/standard.h> // For CTxDestination, IsValidDestination(), PKHash
|
|
|
|
#include <serialize.h> // For SER_GETHASH
|
|
|
|
#include <util/message.h>
|
|
|
|
#include <util/strencodings.h> // For DecodeBase64()
|
|
|
|
|
|
|
|
#include <string>
|
|
|
|
#include <vector>
|
|
|
|
|
2019-12-07 20:52:38 +01:00
|
|
|
/**
|
|
|
|
* Text used to signify that a signed message follows and to prevent
|
|
|
|
* inadvertently signing a transaction.
|
|
|
|
*/
|
|
|
|
const std::string MESSAGE_MAGIC = "Bitcoin Signed Message:\n";
|
2019-11-19 15:49:35 +01:00
|
|
|
|
|
|
|
MessageVerificationResult MessageVerify(
|
|
|
|
const std::string& address,
|
|
|
|
const std::string& signature,
|
|
|
|
const std::string& message)
|
|
|
|
{
|
|
|
|
CTxDestination destination = DecodeDestination(address);
|
|
|
|
if (!IsValidDestination(destination)) {
|
|
|
|
return MessageVerificationResult::ERR_INVALID_ADDRESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (boost::get<PKHash>(&destination) == nullptr) {
|
|
|
|
return MessageVerificationResult::ERR_ADDRESS_NO_KEY;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool invalid = false;
|
|
|
|
std::vector<unsigned char> signature_bytes = DecodeBase64(signature.c_str(), &invalid);
|
|
|
|
if (invalid) {
|
|
|
|
return MessageVerificationResult::ERR_MALFORMED_SIGNATURE;
|
|
|
|
}
|
|
|
|
|
|
|
|
CPubKey pubkey;
|
2019-12-07 20:52:38 +01:00
|
|
|
if (!pubkey.RecoverCompact(MessageHash(message), signature_bytes)) {
|
2019-11-19 15:49:35 +01:00
|
|
|
return MessageVerificationResult::ERR_PUBKEY_NOT_RECOVERED;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!(CTxDestination(PKHash(pubkey)) == destination)) {
|
|
|
|
return MessageVerificationResult::ERR_NOT_SIGNED;
|
|
|
|
}
|
|
|
|
|
|
|
|
return MessageVerificationResult::OK;
|
|
|
|
}
|
2019-11-22 17:17:29 +01:00
|
|
|
|
|
|
|
bool MessageSign(
|
|
|
|
const CKey& privkey,
|
|
|
|
const std::string& message,
|
|
|
|
std::string& signature)
|
|
|
|
{
|
|
|
|
std::vector<unsigned char> signature_bytes;
|
|
|
|
|
2019-12-07 20:52:38 +01:00
|
|
|
if (!privkey.SignCompact(MessageHash(message), signature_bytes)) {
|
2019-11-22 17:17:29 +01:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2020-08-07 20:25:42 +02:00
|
|
|
signature = EncodeBase64(signature_bytes);
|
2019-11-22 17:17:29 +01:00
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
2019-12-07 20:52:38 +01:00
|
|
|
|
|
|
|
uint256 MessageHash(const std::string& message)
|
|
|
|
{
|
|
|
|
CHashWriter hasher(SER_GETHASH, 0);
|
|
|
|
hasher << MESSAGE_MAGIC << message;
|
|
|
|
|
|
|
|
return hasher.GetHash();
|
|
|
|
}
|
2020-02-13 17:09:12 -05:00
|
|
|
|
|
|
|
std::string SigningResultString(const SigningResult res)
|
|
|
|
{
|
|
|
|
switch (res) {
|
|
|
|
case SigningResult::OK:
|
|
|
|
return "No error";
|
|
|
|
case SigningResult::PRIVATE_KEY_NOT_AVAILABLE:
|
|
|
|
return "Private key not available";
|
|
|
|
case SigningResult::SIGNING_FAILED:
|
|
|
|
return "Sign failed";
|
|
|
|
// no default case, so the compiler can warn about missing cases
|
|
|
|
}
|
|
|
|
assert(false);
|
|
|
|
}
|