/* This file is part of cpp-ethereum. cpp-ethereum is free software: you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation, either version 3 of the License, or (at your option) any later version. cpp-ethereum is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with cpp-ethereum. If not, see . */ /** @file WhisperDB.cpp * @author Vladislav Gluhovsky * @date July 2015 */ #include "WhisperDB.h" #include #include using namespace std; using namespace dev; using namespace dev::shh; WhisperDB::WhisperDB() { m_readOptions.verify_checksums = true; string path = dev::getDataDir("shh"); boost::filesystem::create_directories(path); leveldb::Options op; op.create_if_missing = true; op.max_open_files = 256; leveldb::DB* p = nullptr; leveldb::Status status = leveldb::DB::Open(op, path + "/messages", &p); m_db.reset(p); if (!status.ok()) BOOST_THROW_EXCEPTION(FailedToOpenLevelDB(status.ToString())); } string WhisperDB::lookup(dev::h256 const& _key) const { string ret; leveldb::Slice slice((char const*)_key.data(), _key.size); leveldb::Status status = m_db->Get(m_readOptions, slice, &ret); if (!status.ok() && !status.IsNotFound()) BOOST_THROW_EXCEPTION(FailedLookupInLevelDB(status.ToString())); return ret; } void WhisperDB::insert(dev::h256 const& _key, string const& _value) { leveldb::Slice slice((char const*)_key.data(), _key.size); leveldb::Status status = m_db->Put(m_writeOptions, slice, _value); if (!status.ok()) BOOST_THROW_EXCEPTION(FailedInsertInLevelDB(status.ToString())); } void WhisperDB::insert(dev::h256 const& _key, bytes const& _value) { leveldb::Slice k((char const*)_key.data(), _key.size); leveldb::Slice v((char const*)_value.data(), _value.size()); leveldb::Status status = m_db->Put(m_writeOptions, k, v); if (!status.ok()) BOOST_THROW_EXCEPTION(FailedInsertInLevelDB(status.ToString())); } void WhisperDB::kill(dev::h256 const& _key) { leveldb::Slice slice((char const*)_key.data(), _key.size); leveldb::Status status = m_db->Delete(m_writeOptions, slice); if (!status.ok()) BOOST_THROW_EXCEPTION(FailedDeleteInLevelDB(status.ToString())); } void WhisperDB::loadAll(std::map& o_dst) { leveldb::ReadOptions op; op.fill_cache = false; op.verify_checksums = true; vector wasted; unique_ptr it(m_db->NewIterator(op)); for (it->SeekToFirst(); it->Valid(); it->Next()) { leveldb::Slice const k = it->key(); leveldb::Slice const v = it->value(); bool useless = true; try { RLP rlp((byte const*)v.data(), v.size()); Envelope e(rlp[1]); h256 h2 = e.sha3(); h256 h1; readMetaInfo(rlp[0], e); if (k.size() == h256::size) h1 = h256((byte const*)k.data(), h256::ConstructFromPointer); if (h1 != h2) cwarn << "Corrupted data in Level DB:" << h1.hex() << "versus" << h2.hex(); else if (!e.isExpired()) { o_dst[h1] = e; useless = false; } } catch(RLPException const& ex) { cwarn << "RLPException in WhisperDB::loadAll():" << ex.what(); } catch(Exception const& ex) { cwarn << "Exception in WhisperDB::loadAll():" << ex.what(); } if (useless) wasted.push_back(k.ToString()); } cdebug << "WhisperDB::loadAll(): loaded " << o_dst.size() << ", deleted " << wasted.size() << "messages"; for (auto const& k: wasted) { leveldb::Status status = m_db->Delete(m_writeOptions, k); if (!status.ok()) cwarn << "Failed to delete an entry from Level DB:" << k; } } void WhisperDB::save(h256 const& _key, Envelope const& _e) { try { RLPStream meta; streamMetaInfo(meta, _e); RLPStream msg; _e.streamRLP(msg); RLPStream res(2); res.appendRaw(meta.out()); res.appendRaw(msg.out()); bytes b; res.swapOut(b); insert(_key, b); } catch(RLPException const& ex) { cwarn << "RLPException in WhisperDB::save():" << ex.what(); } catch(FailedInsertInLevelDB const& ex) { cwarn << "Exception in WhisperDB::save() - failed to insert:" << ex.what(); } } void WhisperDB::streamMetaInfo(RLPStream& _rlp, Envelope const& _e) { uint32_t x = 0; if (_e.isStoreForever()) x |= StoreForeverFlag; if (_e.isWatched()) x |= WatchedFlag; _rlp.append(x); } void WhisperDB::readMetaInfo(RLP const& _rlp, Envelope& _e) { unsigned x = (uint32_t)_rlp; if (x & StoreForeverFlag) _e.setStoreForever(); if (x & WatchedFlag) _e.setWatched(); }