// Copyright (c) 2009-2010 Satoshi Nakamoto // Copyright (c) 2009-2014 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 "keystore.h" #include "key.h" #include "util.h" #include bool CKeyStore::GetPubKey(const CKeyID &address, CPubKey &vchPubKeyOut) const { CKey key; if (!GetKey(address, key)) return false; vchPubKeyOut = key.GetPubKey(); return true; } bool CKeyStore::AddKey(const CKey &key) { return AddKeyPubKey(key, key.GetPubKey()); } bool CBasicKeyStore::AddKeyPubKey(const CKey& key, const CPubKey &pubkey) { LOCK(cs_KeyStore); mapKeys[pubkey.GetID()] = key; return true; } bool CBasicKeyStore::AddCScript(const CScript& redeemScript) { if (redeemScript.size() > MAX_SCRIPT_ELEMENT_SIZE) return error("CBasicKeyStore::AddCScript(): redeemScripts > %i bytes are invalid", MAX_SCRIPT_ELEMENT_SIZE); LOCK(cs_KeyStore); mapScripts[CScriptID(redeemScript)] = redeemScript; return true; } bool CBasicKeyStore::HaveCScript(const CScriptID& hash) const { LOCK(cs_KeyStore); return mapScripts.count(hash) > 0; } bool CBasicKeyStore::GetCScript(const CScriptID &hash, CScript& redeemScriptOut) const { LOCK(cs_KeyStore); ScriptMap::const_iterator mi = mapScripts.find(hash); if (mi != mapScripts.end()) { redeemScriptOut = (*mi).second; return true; } return false; } bool CBasicKeyStore::AddWatchOnly(const CScript &dest) { LOCK(cs_KeyStore); setWatchOnly.insert(dest); return true; } bool CBasicKeyStore::RemoveWatchOnly(const CScript &dest) { LOCK(cs_KeyStore); setWatchOnly.erase(dest); return true; } bool CBasicKeyStore::HaveWatchOnly(const CScript &dest) const { LOCK(cs_KeyStore); return setWatchOnly.count(dest) > 0; } bool CBasicKeyStore::HaveWatchOnly() const { LOCK(cs_KeyStore); return (!setWatchOnly.empty()); } bool CBasicKeyStore::AddSpendingKey(const libzcash::SproutSpendingKey &sk) { LOCK(cs_SpendingKeyStore); auto address = sk.address(); mapSpendingKeys[address] = sk; mapNoteDecryptors.insert(std::make_pair(address, ZCNoteDecryption(sk.receiving_key()))); return true; } //! Sapling bool CBasicKeyStore::AddSaplingSpendingKey(const libzcash::SaplingSpendingKey &sk) { LOCK(cs_SpendingKeyStore); auto fvk = sk.full_viewing_key(); mapSaplingSpendingKeys[fvk] = sk; // if SaplingFullViewingKey is not in SaplingFullViewingKeyMap, add it AddSaplingFullViewingKey(fvk); // Add addr -> SaplingIncomingViewing to SaplingIncomingViewingKeyMap auto ivk = fvk.in_viewing_key(); auto addrOpt = sk.default_address(); assert(addrOpt != boost::none); auto addr = addrOpt.value(); mapSaplingIncomingViewingKeys[addr] = ivk; return true; } bool CBasicKeyStore::AddViewingKey(const libzcash::SproutViewingKey &vk) { LOCK(cs_SpendingKeyStore); auto address = vk.address(); mapViewingKeys[address] = vk; mapNoteDecryptors.insert(std::make_pair(address, ZCNoteDecryption(vk.sk_enc))); return true; } bool CBasicKeyStore::AddSaplingFullViewingKey(const libzcash::SaplingFullViewingKey &fvk) { LOCK(cs_SpendingKeyStore); auto ivk = fvk.in_viewing_key(); mapSaplingFullViewingKeys[ivk] = fvk; //! TODO: Note decryptors for Sapling return true; } bool CBasicKeyStore::RemoveViewingKey(const libzcash::SproutViewingKey &vk) { LOCK(cs_SpendingKeyStore); mapViewingKeys.erase(vk.address()); return true; } bool CBasicKeyStore::HaveViewingKey(const libzcash::SproutPaymentAddress &address) const { LOCK(cs_SpendingKeyStore); return mapViewingKeys.count(address) > 0; } bool CBasicKeyStore::HaveSaplingFullViewingKey(const libzcash::SaplingIncomingViewingKey &ivk) const { LOCK(cs_SpendingKeyStore); return mapSaplingFullViewingKeys.count(ivk) > 0; } bool CBasicKeyStore::HaveSaplingIncomingViewingKey(const libzcash::SaplingPaymentAddress &addr) const { LOCK(cs_SpendingKeyStore); return mapSaplingIncomingViewingKeys.count(addr) > 0; } bool CBasicKeyStore::GetViewingKey(const libzcash::SproutPaymentAddress &address, libzcash::SproutViewingKey &vkOut) const { LOCK(cs_SpendingKeyStore); ViewingKeyMap::const_iterator mi = mapViewingKeys.find(address); if (mi != mapViewingKeys.end()) { vkOut = mi->second; return true; } return false; } bool CBasicKeyStore::GetSaplingFullViewingKey(const libzcash::SaplingIncomingViewingKey &ivk, libzcash::SaplingFullViewingKey &fvkOut) const { LOCK(cs_SpendingKeyStore); SaplingFullViewingKeyMap::const_iterator mi = mapSaplingFullViewingKeys.find(ivk); if (mi != mapSaplingFullViewingKeys.end()) { fvkOut = mi->second; return true; } return false; } bool CBasicKeyStore::GetSaplingIncomingViewingKey(const libzcash::SaplingPaymentAddress &addr, libzcash::SaplingIncomingViewingKey &ivkOut) const { LOCK(cs_SpendingKeyStore); SaplingIncomingViewingKeyMap::const_iterator mi = mapSaplingIncomingViewingKeys.find(addr); if (mi != mapSaplingIncomingViewingKeys.end()) { ivkOut = mi->second; return true; } return false; }