/* 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 Utility.cpp * @author Gav Wood * @date 2014 */ #include "Utility.h" #include #include #include #include #include #include #include "CanonBlockChain.h" #include "Defaults.h" using namespace std; using namespace dev; using namespace dev::eth; namespace fs = boost::filesystem; bytes dev::eth::parseData(string const& _args) { bytes m_data; boost::smatch what; static const boost::regex r("(!|#|@|\\$)?\"([^\"]*)\"(\\s.*)?"); static const boost::regex d("(@|\\$)?([0-9]+)(\\s*(ether)|(finney)|(szabo))?(\\s.*)?"); static const boost::regex h("(@|\\$)?(0x)?(([a-fA-F0-9])+)(\\s.*)?"); string s = _args; while (s.size()) if (boost::regex_match(s, what, d)) { u256 v((string)what[2]); if (what[6] == "szabo") v *= dev::eth::szabo; else if (what[5] == "finney") v *= dev::eth::finney; else if (what[4] == "ether") v *= dev::eth::ether; bytes bs = dev::toCompactBigEndian(v); if (what[1] != "$") for (auto i = bs.size(); i < 32; ++i) m_data.push_back(0); for (auto b: bs) m_data.push_back(b); s = what[7]; } else if (boost::regex_match(s, what, h)) { bytes bs = fromHex(((what[3].length() & 1) ? "0" : "") + what[3]); if (what[1] != "$") for (auto i = bs.size(); i < 32; ++i) m_data.push_back(0); for (auto b: bs) m_data.push_back(b); s = what[5]; } else if (boost::regex_match(s, what, r)) { bytes d = asBytes(what[2]); if (what[1] == "!") m_data += FixedHash<4>(sha3(d)).asBytes(); else if (what[1] == "#") m_data += sha3(d).asBytes(); else if (what[1] == "$") m_data += d + bytes{0}; else m_data += d + bytes(32 - what[2].length() % 32, 0); s = what[3]; } else s = s.substr(1); return m_data; } void dev::eth::upgradeDatabase(std::string const& _basePath) { std::string path = _basePath.empty() ? Defaults::get()->dbPath() : _basePath; if (fs::exists(path + "/state") && fs::exists(path + "/details") && fs::exists(path + "/blocks")) { // upgrade cnote << "Upgrading database to new layout..."; bytes statusBytes = contents(path + "/status"); RLP status(statusBytes); try { auto minorProtocolVersion = (unsigned)status[1]; auto databaseVersion = (unsigned)status[2]; auto genesisHash = status.itemCount() > 3 ? (h256)status[3] : CanonBlockChain::genesis().hash(); string chainPath = path + "/" + toHex(genesisHash.ref().cropped(0, 4)); string extrasPath = chainPath + "/" + toString(databaseVersion); // write status if (!fs::exists(chainPath + "/blocks")) { boost::filesystem::create_directories(chainPath); fs::rename(path + "/blocks", chainPath + "/blocks"); if (!fs::exists(extrasPath + "/extras")) { boost::filesystem::create_directories(extrasPath); fs::rename(path + "/details", extrasPath + "/extras"); fs::rename(path + "/state", extrasPath + "/state"); writeFile(extrasPath + "/minor", rlp(minorProtocolVersion)); } } } catch (Exception& ex) { cwarn << "Couldn't upgrade: " << ex.what() << boost::diagnostic_information(ex); } catch (...) { cwarn << "Couldn't upgrade. Some issue with moving files around. Probably easiest just to redownload."; } fs::rename(path + "/status", path + "/status.old"); } }