/* 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 <http://www.gnu.org/licenses/>. */ /** @file SecretStore.cpp * @author Gav Wood <i@gavwood.com> * @date 2014 */ #include "SecretStore.h" #include <thread> #include <mutex> #include <boost/algorithm/string.hpp> #include <boost/filesystem.hpp> #include <libdevcore/Log.h> #include <libdevcore/Guards.h> #include <libdevcore/SHA3.h> #include <libdevcore/FileSystem.h> #include <test/JsonSpiritHeaders.h> #include <libdevcrypto/Exceptions.h> using namespace std; using namespace dev; namespace js = json_spirit; namespace fs = boost::filesystem; static const int c_keyFileVersion = 3; /// Upgrade the json-format to the current version. static js::mValue upgraded(string const& _s) { js::mValue v; js::read_string(_s, v); if (v.type() != js::obj_type) return js::mValue(); js::mObject ret = v.get_obj(); unsigned version = ret.count("Version") ? stoi(ret["Version"].get_str()) : ret.count("version") ? ret["version"].get_int() : 0; if (version == 1) { // upgrade to version 2 js::mObject old; swap(old, ret); ret["id"] = old["Id"]; js::mObject c; c["ciphertext"] = old["Crypto"].get_obj()["CipherText"]; c["cipher"] = "aes-128-cbc"; { js::mObject cp; cp["iv"] = old["Crypto"].get_obj()["IV"]; c["cipherparams"] = cp; } c["kdf"] = old["Crypto"].get_obj()["KeyHeader"].get_obj()["Kdf"]; { js::mObject kp; kp["salt"] = old["Crypto"].get_obj()["Salt"]; for (auto const& i: old["Crypto"].get_obj()["KeyHeader"].get_obj()["KdfParams"].get_obj()) if (i.first != "SaltLen") kp[boost::to_lower_copy(i.first)] = i.second; c["kdfparams"] = kp; } c["sillymac"] = old["Crypto"].get_obj()["MAC"]; c["sillymacjson"] = _s; ret["crypto"] = c; version = 2; } if (version == 2) { ret["crypto"].get_obj()["cipher"] = "aes-128-ctr"; ret["crypto"].get_obj()["compat"] = "2"; version = 3; } if (version == c_keyFileVersion) return ret; return js::mValue(); } SecretStore::SecretStore(string const& _path): m_path(_path) { load(); } bytes SecretStore::secret(h128 const& _uuid, function<string()> const& _pass, bool _useCache) const { auto rit = m_cached.find(_uuid); if (_useCache && rit != m_cached.end()) return rit->second; auto it = m_keys.find(_uuid); bytes key; if (it != m_keys.end()) { key = decrypt(it->second.encryptedKey, _pass()); if (!key.empty()) m_cached[_uuid] = key; } return key; } h128 SecretStore::importSecret(bytes const& _s, string const& _pass) { h128 r; EncryptedKey key{encrypt(_s, _pass), string()}; r = h128::random(); m_cached[r] = _s; m_keys[r] = move(key); save(); return r; } void SecretStore::kill(h128 const& _uuid) { m_cached.erase(_uuid); if (m_keys.count(_uuid)) { fs::remove(m_keys[_uuid].filename); m_keys.erase(_uuid); } } void SecretStore::clearCache() const { m_cached.clear(); } void SecretStore::save(string const& _keysPath) { fs::path p(_keysPath); fs::create_directories(p); for (auto& k: m_keys) { string uuid = toUUID(k.first); string filename = (p / uuid).string() + ".json"; js::mObject v; js::mValue crypto; js::read_string(k.second.encryptedKey, crypto); v["crypto"] = crypto; v["id"] = uuid; v["version"] = c_keyFileVersion; writeFile(filename, js::write_string(js::mValue(v), true)); swap(k.second.filename, filename); if (!filename.empty() && !fs::equivalent(filename, k.second.filename)) fs::remove(filename); } } void SecretStore::load(string const& _keysPath) { fs::path p(_keysPath); fs::create_directories(p); for (fs::directory_iterator it(p); it != fs::directory_iterator(); ++it) if (fs::is_regular_file(it->path())) readKey(it->path().string(), true); } h128 SecretStore::readKey(string const& _file, bool _takeFileOwnership) { cnote << "Reading" << _file; return readKeyContent(contentsString(_file), _takeFileOwnership ? _file : string()); } h128 SecretStore::readKeyContent(string const& _content, string const& _file) { js::mValue u = upgraded(_content); if (u.type() == js::obj_type) { js::mObject& o = u.get_obj(); auto uuid = fromUUID(o["id"].get_str()); m_keys[uuid] = EncryptedKey{js::write_string(o["crypto"], false), _file}; return uuid; } else cwarn << "Invalid JSON in key file" << _file; return h128(); } bool SecretStore::recode(h128 const& _uuid, string const& _newPass, function<string()> const& _pass, KDF _kdf) { bytes s = secret(_uuid, _pass, true); if (s.empty()) return false; m_cached.erase(_uuid); m_keys[_uuid].encryptedKey = encrypt(s, _newPass, _kdf); save(); return true; } static bytes deriveNewKey(string const& _pass, KDF _kdf, js::mObject& o_ret) { unsigned dklen = 32; unsigned iterations = 1 << 18; bytes salt = h256::random().asBytes(); if (_kdf == KDF::Scrypt) { unsigned p = 1; unsigned r = 8; o_ret["kdf"] = "scrypt"; { js::mObject params; params["n"] = int64_t(iterations); params["r"] = int(r); params["p"] = int(p); params["dklen"] = int(dklen); params["salt"] = toHex(salt); o_ret["kdfparams"] = params; } return scrypt(_pass, salt, iterations, r, p, dklen); } else { o_ret["kdf"] = "pbkdf2"; { js::mObject params; params["prf"] = "hmac-sha256"; params["c"] = int(iterations); params["salt"] = toHex(salt); params["dklen"] = int(dklen); o_ret["kdfparams"] = params; } return pbkdf2(_pass, salt, iterations, dklen); } } string SecretStore::encrypt(bytes const& _v, string const& _pass, KDF _kdf) { js::mObject ret; bytes derivedKey = deriveNewKey(_pass, _kdf, ret); if (derivedKey.empty()) BOOST_THROW_EXCEPTION(crypto::CryptoException() << errinfo_comment("Key derivation failed.")); ret["cipher"] = "aes-128-ctr"; h128 key(derivedKey, h128::AlignLeft); h128 iv = h128::random(); { js::mObject params; params["iv"] = toHex(iv.ref()); ret["cipherparams"] = params; } // cipher text bytes cipherText = encryptSymNoAuth(key, iv, &_v); if (cipherText.empty()) BOOST_THROW_EXCEPTION(crypto::CryptoException() << errinfo_comment("Key encryption failed.")); ret["ciphertext"] = toHex(cipherText); // and mac. h256 mac = sha3(ref(derivedKey).cropped(16, 16).toBytes() + cipherText); ret["mac"] = toHex(mac.ref()); return js::write_string(js::mValue(ret), true); } bytes SecretStore::decrypt(string const& _v, string const& _pass) { js::mObject o; { js::mValue ov; js::read_string(_v, ov); o = ov.get_obj(); } // derive key bytes derivedKey; if (o["kdf"].get_str() == "pbkdf2") { auto params = o["kdfparams"].get_obj(); if (params["prf"].get_str() != "hmac-sha256") { cwarn << "Unknown PRF for PBKDF2" << params["prf"].get_str() << "not supported."; return bytes(); } unsigned iterations = params["c"].get_int(); bytes salt = fromHex(params["salt"].get_str()); derivedKey = pbkdf2(_pass, salt, iterations, params["dklen"].get_int()); } else if (o["kdf"].get_str() == "scrypt") { auto p = o["kdfparams"].get_obj(); derivedKey = scrypt(_pass, fromHex(p["salt"].get_str()), p["n"].get_int(), p["r"].get_int(), p["p"].get_int(), p["dklen"].get_int()); } else { cwarn << "Unknown KDF" << o["kdf"].get_str() << "not supported."; return bytes(); } if (derivedKey.size() < 32 && !(o.count("compat") && o["compat"].get_str() == "2")) { cwarn << "Derived key's length too short (<32 bytes)"; return bytes(); } bytes cipherText = fromHex(o["ciphertext"].get_str()); // check MAC if (o.count("mac")) { h256 mac(o["mac"].get_str()); h256 macExp; if (o.count("compat") && o["compat"].get_str() == "2") macExp = sha3(bytesConstRef(&derivedKey).cropped(derivedKey.size() - 16).toBytes() + cipherText); else macExp = sha3(bytesConstRef(&derivedKey).cropped(16, 16).toBytes() + cipherText); if (mac != macExp) { cwarn << "Invalid key - MAC mismatch; expected" << toString(macExp) << ", got" << toString(mac); return bytes(); } } else if (o.count("sillymac")) { h256 mac(o["sillymac"].get_str()); h256 macExp = sha3(asBytes(o["sillymacjson"].get_str()) + bytesConstRef(&derivedKey).cropped(derivedKey.size() - 16).toBytes() + cipherText); if (mac != macExp) { cwarn << "Invalid key - MAC mismatch; expected" << toString(macExp) << ", got" << toString(mac); return bytes(); } } else cwarn << "No MAC. Proceeding anyway."; // decrypt if (o["cipher"].get_str() == "aes-128-ctr") { auto params = o["cipherparams"].get_obj(); h128 iv(params["iv"].get_str()); if (o.count("compat") && o["compat"].get_str() == "2") { h128 key(sha3(h128(derivedKey, h128::AlignRight)), h128::AlignRight); return decryptSymNoAuth(key, iv, &cipherText); } else return decryptSymNoAuth(h128(derivedKey, h128::AlignLeft), iv, &cipherText); } else { cwarn << "Unknown cipher" << o["cipher"].get_str() << "not supported."; return bytes(); } }