7 changed files with 1098 additions and 0 deletions
@ -0,0 +1,186 @@ |
|||
/*
|
|||
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 Common.h
|
|||
* @author Gav Wood <i@gavwood.com> |
|||
* @date 2014 |
|||
* |
|||
* Shared algorithms and data types. |
|||
*/ |
|||
|
|||
#pragma once |
|||
|
|||
#include <vector> |
|||
#include <algorithm> |
|||
#include <type_traits> |
|||
#include <cstring> |
|||
#include <string> |
|||
#include "Common.h" |
|||
|
|||
namespace eth |
|||
{ |
|||
|
|||
// String conversion functions, mainly to/from hex/nibble/byte representations.
|
|||
|
|||
/// Convert a series of bytes to the corresponding string of hex duplets.
|
|||
/// @param _w specifies the width of each of the elements. Defaults to two - enough to represent a byte.
|
|||
/// @example toHex("A\x69") == "4169"
|
|||
template <class _T> |
|||
std::string toHex(_T const& _data, int _w = 2) |
|||
{ |
|||
std::ostringstream ret; |
|||
for (auto i: _data) |
|||
ret << std::hex << std::setfill('0') << std::setw(_w) << (int)(typename std::make_unsigned<decltype(i)>::type)i; |
|||
return ret.str(); |
|||
} |
|||
|
|||
/// Converts a (printable) ASCII hex character into the correspnding integer value.
|
|||
/// @example fromHex('A') == 10 && fromHex('f') == 15 && fromHex('5') == 5
|
|||
int fromHex(char _i); |
|||
|
|||
/// Converts a (printable) ASCII hex string into the corresponding byte stream.
|
|||
/// @example fromHex("41626261") == asBytes("Abba")
|
|||
bytes fromHex(std::string const& _s); |
|||
|
|||
/// Converts byte array to a string containing the same (binary) data. Unless
|
|||
/// the byte array happens to contain ASCII data, this won't be printable.
|
|||
inline std::string asString(bytes const& _b) |
|||
{ |
|||
return std::string((char const*)_b.data(), (char const*)(_b.data() + _b.size())); |
|||
} |
|||
|
|||
/// Converts a string to a byte array containing the string's (byte) data.
|
|||
inline bytes asBytes(std::string const& _b) |
|||
{ |
|||
return bytes((byte const*)_b.data(), (byte const*)(_b.data() + _b.size())); |
|||
} |
|||
|
|||
/// Converts a string into the big-endian base-16 stream of integers (NOT ASCII).
|
|||
/// @example asNibbles("A")[0] == 4 && asNibbles("A")[1] == 1
|
|||
bytes asNibbles(std::string const& _s); |
|||
|
|||
|
|||
// Big-endian to/from host endian conversion functions.
|
|||
|
|||
/// Converts a templated integer value to the big-endian byte-stream represented on a templated collection.
|
|||
/// The size of the collection object will be unchanged. If it is too small, it will not represent the
|
|||
/// value properly, if too big then the additional elements will be zeroed out.
|
|||
/// @a _Out will typically be either std::string or bytes.
|
|||
/// @a _T will typically by uint, u160, u256 or bigint.
|
|||
template <class _T, class _Out> |
|||
inline void toBigEndian(_T _val, _Out& o_out) |
|||
{ |
|||
for (auto i = o_out.size(); i-- != 0; _val >>= 8) |
|||
o_out[i] = (typename _Out::value_type)(uint8_t)_val; |
|||
} |
|||
|
|||
/// Converts a big-endian byte-stream represented on a templated collection to a templated integer value.
|
|||
/// @a _In will typically be either std::string or bytes.
|
|||
/// @a _T will typically by uint, u160, u256 or bigint.
|
|||
template <class _T, class _In> |
|||
inline _T fromBigEndian(_In const& _bytes) |
|||
{ |
|||
_T ret = 0; |
|||
for (auto i: _bytes) |
|||
ret = (ret << 8) | (byte)(typename std::make_unsigned<typename _In::value_type>::type)i; |
|||
return ret; |
|||
} |
|||
|
|||
/// Convenience functions for toBigEndian
|
|||
inline std::string toBigEndianString(u256 _val) { std::string ret(32, '\0'); toBigEndian(_val, ret); return ret; } |
|||
inline std::string toBigEndianString(u160 _val) { std::string ret(20, '\0'); toBigEndian(_val, ret); return ret; } |
|||
inline bytes toBigEndian(u256 _val) { bytes ret(32); toBigEndian(_val, ret); return ret; } |
|||
inline bytes toBigEndian(u160 _val) { bytes ret(20); toBigEndian(_val, ret); return ret; } |
|||
|
|||
/// Convenience function for toBigEndian.
|
|||
/// @returns a string just big enough to represent @a _val.
|
|||
template <class _T> |
|||
inline std::string toCompactBigEndianString(_T _val) |
|||
{ |
|||
int i = 0; |
|||
for (_T v = _val; v; ++i, v >>= 8) {} |
|||
std::string ret(i, '\0'); |
|||
toBigEndian(_val, ret); |
|||
return ret; |
|||
} |
|||
|
|||
|
|||
// Algorithms for string and string-like collections.
|
|||
|
|||
/// Escapes a string into the C-string representation.
|
|||
/// @p _all if true will escape all characters, not just the unprintable ones.
|
|||
std::string escaped(std::string const& _s, bool _all = true); |
|||
|
|||
/// Determines the length of the common prefix of the two collections given.
|
|||
/// @returns the number of elements both @a _t and @a _u share, in order, at the beginning.
|
|||
/// @example commonPrefix("Hello world!", "Hello, world!") == 5
|
|||
template <class _T, class _U> |
|||
uint commonPrefix(_T const& _t, _U const& _u) |
|||
{ |
|||
uint s = std::min<uint>(_t.size(), _u.size()); |
|||
for (uint i = 0;; ++i) |
|||
if (i == s || _t[i] != _u[i]) |
|||
return i; |
|||
return s; |
|||
} |
|||
|
|||
/// Creates a random, printable, word.
|
|||
std::string randomWord(); |
|||
|
|||
|
|||
// General datatype convenience functions.
|
|||
|
|||
/// Trims a given number of elements from the front of a collection.
|
|||
/// Only works for POD element types.
|
|||
template <class _T> |
|||
void trimFront(_T& _t, uint _elements) |
|||
{ |
|||
static_assert(std::is_pod<typename _T::value_type>::value, ""); |
|||
memmove(_t.data(), _t.data() + _elements, (_t.size() - _elements) * sizeof(_t[0])); |
|||
_t.resize(_t.size() - _elements); |
|||
} |
|||
|
|||
/// Pushes an element on to the front of a collection.
|
|||
/// Only works for POD element types.
|
|||
template <class _T, class _U> |
|||
void pushFront(_T& _t, _U _e) |
|||
{ |
|||
static_assert(std::is_pod<typename _T::value_type>::value, ""); |
|||
_t.push_back(_e); |
|||
memmove(_t.data() + 1, _t.data(), (_t.size() - 1) * sizeof(_e)); |
|||
_t[0] = _e; |
|||
} |
|||
|
|||
/// Concatenate two vectors of elements. _T must be POD.
|
|||
template <class _T> |
|||
inline std::vector<_T>& operator+=(std::vector<typename std::enable_if<std::is_pod<_T>::value, _T>::type>& _a, std::vector<_T> const& _b) |
|||
{ |
|||
auto s = _a.size(); |
|||
_a.resize(_a.size() + _b.size()); |
|||
memcpy(_a.data() + s, _b.data(), _b.size() * sizeof(_T)); |
|||
return _a; |
|||
|
|||
} |
|||
|
|||
/// Concatenate two vectors of elements. _T must be POD.
|
|||
template <class _T> |
|||
inline std::vector<_T> operator+(std::vector<typename std::enable_if<std::is_pod<_T>::value, _T>::type> const& _a, std::vector<_T> const& _b) |
|||
{ |
|||
std::vector<_T> ret(_a); |
|||
return ret += _b; |
|||
} |
|||
|
|||
} |
@ -0,0 +1,137 @@ |
|||
/*
|
|||
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 CommonEth.h
|
|||
* @author Gav Wood <i@gavwood.com> |
|||
* @date 2014 |
|||
* |
|||
* Ethereum-specific data structures & algorithms. |
|||
*/ |
|||
|
|||
#pragma once |
|||
|
|||
#include "Common.h" |
|||
#include "FixedHash.h" |
|||
|
|||
namespace eth |
|||
{ |
|||
|
|||
/// A secret key: 32 bytes.
|
|||
/// @NOTE This is not endian-specific; it's just a bunch of bytes.
|
|||
using Secret = h256; |
|||
|
|||
/// A public key: 64 bytes.
|
|||
/// @NOTE This is not endian-specific; it's just a bunch of bytes.
|
|||
using Public = h512; |
|||
|
|||
/// An Ethereum address: 20 bytes.
|
|||
/// @NOTE This is not endian-specific; it's just a bunch of bytes.
|
|||
using Address = h160; |
|||
|
|||
/// A vector of Ethereum addresses.
|
|||
using Addresses = h160s; |
|||
|
|||
/// User-friendly string representation of the amount _b in wei.
|
|||
std::string formatBalance(u256 _b); |
|||
|
|||
/// Get information concerning the currency denominations.
|
|||
std::vector<std::pair<u256, std::string>> const& units(); |
|||
|
|||
// The various denominations; here for ease of use where needed within code.
|
|||
static const u256 Uether = ((((u256(1000000000) * 1000000000) * 1000000000) * 1000000000) * 1000000000) * 1000000000; |
|||
static const u256 Vether = ((((u256(1000000000) * 1000000000) * 1000000000) * 1000000000) * 1000000000) * 1000000; |
|||
static const u256 Dether = ((((u256(1000000000) * 1000000000) * 1000000000) * 1000000000) * 1000000000) * 1000; |
|||
static const u256 Nether = (((u256(1000000000) * 1000000000) * 1000000000) * 1000000000) * 1000000000; |
|||
static const u256 Yether = (((u256(1000000000) * 1000000000) * 1000000000) * 1000000000) * 1000000; |
|||
static const u256 Zether = (((u256(1000000000) * 1000000000) * 1000000000) * 1000000000) * 1000; |
|||
static const u256 Eether = ((u256(1000000000) * 1000000000) * 1000000000) * 1000000000; |
|||
static const u256 Pether = ((u256(1000000000) * 1000000000) * 1000000000) * 1000000; |
|||
static const u256 Tether = ((u256(1000000000) * 1000000000) * 1000000000) * 1000; |
|||
static const u256 Gether = (u256(1000000000) * 1000000000) * 1000000000; |
|||
static const u256 Mether = (u256(1000000000) * 1000000000) * 1000000; |
|||
static const u256 Kether = (u256(1000000000) * 1000000000) * 1000; |
|||
static const u256 ether = u256(1000000000) * 1000000000; |
|||
static const u256 finney = u256(1000000000) * 1000000; |
|||
static const u256 szabo = u256(1000000000) * 1000; |
|||
static const u256 Gwei = u256(1000000000); |
|||
static const u256 Mwei = u256(1000000); |
|||
static const u256 Kwei = u256(1000); |
|||
static const u256 wei = u256(1); |
|||
|
|||
/// Convert a private key into the public key equivalent.
|
|||
/// @returns 0 if it's not a valid private key.
|
|||
Address toAddress(h256 _private); |
|||
|
|||
/// Simple class that represents a "key pair".
|
|||
/// All of the data of the class can be regenerated from the secret key (m_secret) alone.
|
|||
/// Actually stores a tuplet of secret, public and address (the right 160-bits of the public).
|
|||
class KeyPair |
|||
{ |
|||
public: |
|||
/// Null constructor.
|
|||
KeyPair() {} |
|||
|
|||
/// Normal constructor - populates object from the given secret key.
|
|||
KeyPair(Secret _k); |
|||
|
|||
/// Create a new, randomly generated object.
|
|||
static KeyPair create(); |
|||
|
|||
/// Retrieve the secret key.
|
|||
Secret const& secret() const { return m_secret; } |
|||
/// Retrieve the secret key.
|
|||
Secret const& sec() const { return m_secret; } |
|||
|
|||
/// Retrieve the public key.
|
|||
Public const& pub() const { return m_public; } |
|||
|
|||
/// Retrieve the associated address of the public key.
|
|||
Address const& address() const { return m_address; } |
|||
|
|||
private: |
|||
Secret m_secret; |
|||
Public m_public; |
|||
Address m_address; |
|||
}; |
|||
|
|||
|
|||
// SHA-3 convenience routines.
|
|||
|
|||
/// Calculate SHA3-256 hash of the given input and load it into the given output.
|
|||
void sha3(bytesConstRef _input, bytesRef _output); |
|||
|
|||
/// Calculate SHA3-256 hash of the given input, possibly interpreting it as nibbles, and return the hash as a string filled with binary data.
|
|||
std::string sha3(std::string const& _input, bool _isNibbles); |
|||
|
|||
/// Calculate SHA3-256 hash of the given input, returning as a byte array.
|
|||
bytes sha3Bytes(bytesConstRef _input); |
|||
|
|||
/// Calculate SHA3-256 hash of the given input (presented as a binary string), returning as a byte array.
|
|||
inline bytes sha3Bytes(std::string const& _input) { return sha3Bytes((std::string*)&_input); } |
|||
|
|||
/// Calculate SHA3-256 hash of the given input, returning as a byte array.
|
|||
inline bytes sha3Bytes(bytes const& _input) { return sha3Bytes((bytes*)&_input); } |
|||
|
|||
/// Calculate SHA3-256 hash of the given input, returning as a 256-bit hash.
|
|||
h256 sha3(bytesConstRef _input); |
|||
|
|||
/// Calculate SHA3-256 hash of the given input, returning as a 256-bit hash.
|
|||
inline h256 sha3(bytes const& _input) { return sha3(bytesConstRef((bytes*)&_input)); } |
|||
|
|||
/// Calculate SHA3-256 hash of the given input (presented as a binary-filled string), returning as a 256-bit hash.
|
|||
inline h256 sha3(std::string const& _input) { return sha3(bytesConstRef(_input)); } |
|||
|
|||
} |
@ -0,0 +1,223 @@ |
|||
/*
|
|||
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 CommonIO.h
|
|||
* @author Gav Wood <i@gavwood.com> |
|||
* @date 2014 |
|||
* |
|||
* File & stream I/O routines. |
|||
*/ |
|||
|
|||
#pragma once |
|||
|
|||
#include <map> |
|||
#include <set> |
|||
#include <unordered_map> |
|||
#include <unordered_set> |
|||
#include <array> |
|||
#include <list> |
|||
#include <memory> |
|||
#include <vector> |
|||
#include <array> |
|||
#include <sstream> |
|||
#include <string> |
|||
#include <iostream> |
|||
#include "Common.h" |
|||
|
|||
namespace eth |
|||
{ |
|||
|
|||
/// Retrieve and returns the contents of the given file. If the file doesn't exist or isn't readable, returns an empty bytes.
|
|||
bytes contents(std::string const& _file); |
|||
|
|||
/// Write the given binary data into the given file, replacing the file if it pre-exists.
|
|||
void writeFile(std::string const& _file, bytes const& _data); |
|||
|
|||
/// Converts arbitrary value to string representation using std::stringstream.
|
|||
template <class _T> |
|||
std::string toString(_T const& _t) |
|||
{ |
|||
std::ostringstream o; |
|||
o << _t; |
|||
return o.str(); |
|||
} |
|||
|
|||
// Stream I/O functions.
|
|||
// Provides templated stream I/O for all STL collections so they can be shifted on to any iostream-like interface.
|
|||
|
|||
template <class S, class T> struct StreamOut { static S& bypass(S& _out, T const& _t) { _out << _t; return _out; } }; |
|||
template <class S> struct StreamOut<S, uint8_t> { static S& bypass(S& _out, uint8_t const& _t) { _out << (int)_t; return _out; } }; |
|||
|
|||
template <class S, class T> |
|||
inline S& streamout(S& _out, std::vector<T> const& _e) |
|||
{ |
|||
_out << "["; |
|||
if (!_e.empty()) |
|||
{ |
|||
StreamOut<S, T>::bypass(_out, _e.front()); |
|||
for (auto i = ++_e.begin(); i != _e.end(); ++i) |
|||
StreamOut<S, T>::bypass(_out << ",", *i); |
|||
} |
|||
_out << "]"; |
|||
return _out; |
|||
} |
|||
|
|||
template <class T> inline std::ostream& operator<<(std::ostream& _out, std::vector<T> const& _e) { streamout(_out, _e); return _out; } |
|||
|
|||
template <class S, class T, unsigned Z> |
|||
inline S& streamout(S& _out, std::array<T, Z> const& _e) |
|||
{ |
|||
_out << "["; |
|||
if (!_e.empty()) |
|||
{ |
|||
StreamOut<S, T>::bypass(_out, _e.front()); |
|||
auto i = _e.begin(); |
|||
for (++i; i != _e.end(); ++i) |
|||
StreamOut<S, T>::bypass(_out << ",", *i); |
|||
} |
|||
_out << "]"; |
|||
return _out; |
|||
} |
|||
template <class T, unsigned Z> inline std::ostream& operator<<(std::ostream& _out, std::array<T, Z> const& _e) { streamout(_out, _e); return _out; } |
|||
|
|||
template <class S, class T, unsigned long Z> |
|||
inline S& streamout(S& _out, std::array<T, Z> const& _e) |
|||
{ |
|||
_out << "["; |
|||
if (!_e.empty()) |
|||
{ |
|||
StreamOut<S, T>::bypass(_out, _e.front()); |
|||
auto i = _e.begin(); |
|||
for (++i; i != _e.end(); ++i) |
|||
StreamOut<S, T>::bypass(_out << ",", *i); |
|||
} |
|||
_out << "]"; |
|||
return _out; |
|||
} |
|||
template <class T, unsigned long Z> inline std::ostream& operator<<(std::ostream& _out, std::array<T, Z> const& _e) { streamout(_out, _e); return _out; } |
|||
|
|||
template <class S, class T> |
|||
inline S& streamout(S& _out, std::list<T> const& _e) |
|||
{ |
|||
_out << "["; |
|||
if (!_e.empty()) |
|||
{ |
|||
_out << _e.front(); |
|||
for (auto i = ++_e.begin(); i != _e.end(); ++i) |
|||
_out << "," << *i; |
|||
} |
|||
_out << "]"; |
|||
return _out; |
|||
} |
|||
template <class T> inline std::ostream& operator<<(std::ostream& _out, std::list<T> const& _e) { streamout(_out, _e); return _out; } |
|||
|
|||
template <class S, class T, class U> |
|||
inline S& streamout(S& _out, std::pair<T, U> const& _e) |
|||
{ |
|||
_out << "(" << _e.first << "," << _e.second << ")"; |
|||
return _out; |
|||
} |
|||
template <class T, class U> inline std::ostream& operator<<(std::ostream& _out, std::pair<T, U> const& _e) { streamout(_out, _e); return _out; } |
|||
|
|||
template <class S, class T1, class T2, class T3> |
|||
inline S& streamout(S& _out, std::tuple<T1, T2, T3> const& _t) |
|||
{ |
|||
_out << "(" << std::get<0>(_t) << "," << std::get<1>(_t) << "," << std::get<2>(_t) << ")"; |
|||
return _out; |
|||
} |
|||
template <class T1, class T2, class T3> inline std::ostream& operator<<(std::ostream& _out, std::tuple<T1, T2, T3> const& _e) { streamout(_out, _e); return _out; } |
|||
|
|||
template <class S, class T, class U> |
|||
S& streamout(S& _out, std::map<T, U> const& _v) |
|||
{ |
|||
if (_v.empty()) |
|||
return _out << "{}"; |
|||
int i = 0; |
|||
for (auto p: _v) |
|||
_out << (!(i++) ? "{ " : "; ") << p.first << " => " << p.second; |
|||
return _out << " }"; |
|||
} |
|||
template <class T, class U> inline std::ostream& operator<<(std::ostream& _out, std::map<T, U> const& _e) { streamout(_out, _e); return _out; } |
|||
|
|||
template <class S, class T, class U> |
|||
S& streamout(S& _out, std::unordered_map<T, U> const& _v) |
|||
{ |
|||
if (_v.empty()) |
|||
return _out << "{}"; |
|||
int i = 0; |
|||
for (auto p: _v) |
|||
_out << (!(i++) ? "{ " : "; ") << p.first << " => " << p.second; |
|||
return _out << " }"; |
|||
} |
|||
template <class T, class U> inline std::ostream& operator<<(std::ostream& _out, std::unordered_map<T, U> const& _e) { streamout(_out, _e); return _out; } |
|||
|
|||
template <class S, class T> |
|||
S& streamout(S& _out, std::set<T> const& _v) |
|||
{ |
|||
if (_v.empty()) |
|||
return _out << "{}"; |
|||
int i = 0; |
|||
for (auto p: _v) |
|||
_out << (!(i++) ? "{ " : ", ") << p; |
|||
return _out << " }"; |
|||
} |
|||
template <class T> inline std::ostream& operator<<(std::ostream& _out, std::set<T> const& _e) { streamout(_out, _e); return _out; } |
|||
|
|||
template <class S, class T> |
|||
S& streamout(S& _out, std::unordered_set<T> const& _v) |
|||
{ |
|||
if (_v.empty()) |
|||
return _out << "{}"; |
|||
int i = 0; |
|||
for (auto p: _v) |
|||
_out << (!(i++) ? "{ " : ", ") << p; |
|||
return _out << " }"; |
|||
} |
|||
template <class T> inline std::ostream& operator<<(std::ostream& _out, std::unordered_set<T> const& _e) { streamout(_out, _e); return _out; } |
|||
|
|||
template <class S, class T> |
|||
S& streamout(S& _out, std::multiset<T> const& _v) |
|||
{ |
|||
if (_v.empty()) |
|||
return _out << "{}"; |
|||
int i = 0; |
|||
for (auto p: _v) |
|||
_out << (!(i++) ? "{ " : ", ") << p; |
|||
return _out << " }"; |
|||
} |
|||
template <class T> inline std::ostream& operator<<(std::ostream& _out, std::multiset<T> const& _e) { streamout(_out, _e); return _out; } |
|||
|
|||
template <class S, class T, class U> |
|||
S& streamout(S& _out, std::multimap<T, U> const& _v) |
|||
{ |
|||
if (_v.empty()) |
|||
return _out << "{}"; |
|||
T l; |
|||
int i = 0; |
|||
for (auto p: _v) |
|||
if (!(i++)) |
|||
_out << "{ " << (l = p.first) << " => " << p.second; |
|||
else if (l == p.first) |
|||
_out << ", " << p.second; |
|||
else |
|||
_out << "; " << (l = p.first) << " => " << p.second; |
|||
return _out << " }"; |
|||
} |
|||
template <class T, class U> inline std::ostream& operator<<(std::ostream& _out, std::multimap<T, U> const& _e) { streamout(_out, _e); return _out; } |
|||
|
|||
template <class _S, class _T> _S& operator<<(_S& _out, std::shared_ptr<_T> const& _p) { if (_p) _out << "@" << (*_p); else _out << "nullptr"; return _out; } |
|||
|
|||
} |
@ -0,0 +1,195 @@ |
|||
/*
|
|||
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 FixedHash.h
|
|||
* @author Gav Wood <i@gavwood.com> |
|||
* @date 2014 |
|||
* |
|||
* The FixedHash fixed-size "hash" container type. |
|||
*/ |
|||
|
|||
#pragma once |
|||
|
|||
#include <array> |
|||
#include <algorithm> |
|||
#include "CommonData.h" |
|||
|
|||
namespace eth |
|||
{ |
|||
|
|||
/// Fixed-size raw-byte array container type, with an API optimised for storing hashes.
|
|||
/// Transparently converts to/from the corresponding arithmetic type; this will
|
|||
/// assume the data contained in the hash is big-endian.
|
|||
template <unsigned N> |
|||
class FixedHash |
|||
{ |
|||
/// The corresponding arithmetic type.
|
|||
using Arith = boost::multiprecision::number<boost::multiprecision::cpp_int_backend<N * 8, N * 8, boost::multiprecision::unsigned_magnitude, boost::multiprecision::unchecked, void>>; |
|||
|
|||
public: |
|||
/// The size of the container.
|
|||
enum { size = N }; |
|||
|
|||
/// A dummy flag to avoid accidental construction from pointer.
|
|||
enum ConstructFromPointerType { ConstructFromPointer }; |
|||
|
|||
/// Method to convert from a string.
|
|||
enum ConstructFromStringType { FromHex, FromBinary }; |
|||
|
|||
/// Construct an empty hash.
|
|||
FixedHash() { m_data.fill(0); } |
|||
|
|||
/// Convert from the corresponding arithmetic type.
|
|||
FixedHash(Arith const& _arith) { toBigEndian(_arith, m_data); } |
|||
|
|||
/// Explicitly construct, copying from a byte array.
|
|||
explicit FixedHash(bytes const& _b) { if (_b.size() == N) memcpy(m_data.data(), _b.data(), std::min<uint>(_b.size(), N)); } |
|||
|
|||
/// Explicitly construct, copying from a bytes in memory with given pointer.
|
|||
explicit FixedHash(byte const* _bs, ConstructFromPointerType) { memcpy(m_data.data(), _bs, N); } |
|||
|
|||
/// Explicitly construct, copying from a string.
|
|||
explicit FixedHash(std::string const& _s, ConstructFromStringType _t = FromHex): FixedHash(_t == FromHex ? fromHex(_s) : asBytes(_s)) {} |
|||
|
|||
/// Convert to arithmetic type.
|
|||
operator Arith() const { return fromBigEndian<Arith>(m_data); } |
|||
|
|||
/// @returns true iff this is the empty hash.
|
|||
operator bool() const { return ((Arith)*this) != 0; } |
|||
|
|||
// The obvious comparison operators.
|
|||
bool operator==(FixedHash const& _c) const { return m_data == _c.m_data; } |
|||
bool operator!=(FixedHash const& _c) const { return m_data != _c.m_data; } |
|||
bool operator<(FixedHash const& _c) const { return m_data < _c.m_data; } |
|||
|
|||
// The obvious binary operators.
|
|||
FixedHash& operator^=(FixedHash const& _c) { for (auto i = 0; i < N; ++i) m_data[i] ^= _c.m_data[i]; return *this; } |
|||
FixedHash operator^(FixedHash const& _c) const { return FixedHash(*this) ^= _c; } |
|||
FixedHash& operator|=(FixedHash const& _c) { for (auto i = 0; i < N; ++i) m_data[i] |= _c.m_data[i]; return *this; } |
|||
FixedHash operator|(FixedHash const& _c) const { return FixedHash(*this) |= _c; } |
|||
FixedHash& operator&=(FixedHash const& _c) { for (auto i = 0; i < N; ++i) m_data[i] &= _c.m_data[i]; return *this; } |
|||
FixedHash operator&(FixedHash const& _c) const { return FixedHash(*this) &= _c; } |
|||
FixedHash& operator~() { for (auto i = 0; i < N; ++i) m_data[i] = ~m_data[i]; return *this; } |
|||
|
|||
/// @returns a particular byte from the hash.
|
|||
byte& operator[](unsigned _i) { return m_data[_i]; } |
|||
/// @returns a particular byte from the hash.
|
|||
byte operator[](unsigned _i) const { return m_data[_i]; } |
|||
|
|||
/// @returns an abridged version of the hash as a user-readable hex string.
|
|||
std::string abridged() const { return toHex(ref().cropped(0, 4)) + ".."; } |
|||
|
|||
/// @returns a mutable byte vector_ref to the object's data.
|
|||
bytesRef ref() { return bytesRef(m_data.data(), N); } |
|||
|
|||
/// @returns a constant byte vector_ref to the object's data.
|
|||
bytesConstRef ref() const { return bytesConstRef(m_data.data(), N); } |
|||
|
|||
/// @returns a mutable byte pointer to the object's data.
|
|||
byte* data() { return m_data.data(); } |
|||
|
|||
/// @returns a constant byte pointer to the object's data.
|
|||
byte const* data() const { return m_data.data(); } |
|||
|
|||
/// @returns a copy of the object's data as a byte vector.
|
|||
bytes asBytes() const { return bytes(data(), data() + N); } |
|||
|
|||
/// @returns a mutable reference to the object's data as an STL array.
|
|||
std::array<byte, N>& asArray() { return m_data; } |
|||
|
|||
/// @returns a constant reference to the object's data as an STL array.
|
|||
std::array<byte, N> const& asArray() const { return m_data; } |
|||
|
|||
/// A generic std::hash compatible function object.
|
|||
struct hash |
|||
{ |
|||
/// Make a hash of the object's data.
|
|||
size_t operator()(FixedHash const& value) const |
|||
{ |
|||
size_t h = 0; |
|||
for (auto i: value.m_data) |
|||
h = (h << 5 - h) + i; |
|||
return h; |
|||
} |
|||
}; |
|||
|
|||
private: |
|||
std::array<byte, N> m_data; ///< The binary data.
|
|||
}; |
|||
|
|||
|
|||
/// Fast equality operator for h256.
|
|||
template<> inline bool FixedHash<32>::operator==(FixedHash<32> const& _other) const |
|||
{ |
|||
const uint64_t* hash1 = (const uint64_t*)this->data(); |
|||
const uint64_t* hash2 = (const uint64_t*)_other.data(); |
|||
return (hash1[0] == hash2[0]) && (hash1[1] == hash2[1]) && (hash1[2] == hash2[2]) && (hash1[3] == hash2[3]); |
|||
} |
|||
|
|||
/// Fast std::hash compatible hash function object for h256.
|
|||
template<> inline size_t FixedHash<32>::hash::operator()(FixedHash<32> const& value) const |
|||
{ |
|||
const uint64_t*data = (const uint64_t*)value.data(); |
|||
uint64_t hash = data[0]; |
|||
hash ^= data[1]; |
|||
hash ^= data[2]; |
|||
hash ^= data[3]; |
|||
return (size_t)hash; |
|||
} |
|||
|
|||
/// Stream I/O for the FixedHash class.
|
|||
template <unsigned N> |
|||
inline std::ostream& operator<<(std::ostream& _out, FixedHash<N> const& _h) |
|||
{ |
|||
_out << std::noshowbase << std::hex << std::setfill('0'); |
|||
for (unsigned i = 0; i < N; ++i) |
|||
_out << std::setw(2) << (int)_h[i]; |
|||
_out << std::dec; |
|||
return _out; |
|||
} |
|||
|
|||
// Common types of FixedHash.
|
|||
using h512 = FixedHash<64>; |
|||
using h256 = FixedHash<32>; |
|||
using h160 = FixedHash<20>; |
|||
using h256s = std::vector<h256>; |
|||
using h160s = std::vector<h160>; |
|||
using h256Set = std::set<h256>; |
|||
using h160Set = std::set<h160>; |
|||
|
|||
/// Convert the given value into h160 (160-bit unsigned integer) using the right 20 bytes.
|
|||
inline h160 right160(h256 const& _t) |
|||
{ |
|||
h160 ret; |
|||
memcpy(ret.data(), _t.data() + 12, 20); |
|||
return ret; |
|||
} |
|||
|
|||
/// Convert the given value into h160 (160-bit unsigned integer) using the left 20 bytes.
|
|||
inline h160 left160(h256 const& _t) |
|||
{ |
|||
h160 ret; |
|||
memcpy(&ret[0], _t.data(), 20); |
|||
return ret; |
|||
} |
|||
|
|||
} |
|||
|
|||
namespace std |
|||
{ |
|||
/// Forward std::hash<eth::h256> to eth::h256::hash.
|
|||
template<> struct hash<eth::h256>: eth::h256::hash {}; |
|||
} |
@ -0,0 +1,134 @@ |
|||
/*
|
|||
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 Log.h
|
|||
* @author Gav Wood <i@gavwood.com> |
|||
* @date 2014 |
|||
* |
|||
* The logging subsystem. |
|||
*/ |
|||
|
|||
#pragma once |
|||
|
|||
#include <ctime> |
|||
#include <chrono> |
|||
#include <boost/thread.hpp> |
|||
#include "vector_ref.h" |
|||
|
|||
namespace eth |
|||
{ |
|||
|
|||
/// The null output stream. Used when logging is disabled.
|
|||
class NullOutputStream |
|||
{ |
|||
public: |
|||
template <class T> NullOutputStream& operator<<(T const&) { return *this; } |
|||
}; |
|||
|
|||
/// A simple log-output function that prints log messages to stdout.
|
|||
void simpleDebugOut(std::string const&, char const* ); |
|||
|
|||
/// The logging system's current verbosity.
|
|||
extern int g_logVerbosity; |
|||
|
|||
/// The current method that the logging system uses to output the log messages. Defaults to simpleDebugOut().
|
|||
extern std::function<void(std::string const&, char const*)> g_logPost; |
|||
|
|||
/// Map of Log Channel types to bool, false forces the channel to be disabled, true forces it to be enabled.
|
|||
/// If a channel has no entry, then it will output as long as its verbosity (LogChannel::verbosity) is less than
|
|||
/// or equal to the currently output verbosity (g_logVerbosity).
|
|||
extern std::map<std::type_info const*, bool> g_logOverride; |
|||
|
|||
/// Associate a name with each thread for nice logging.
|
|||
struct ThreadLocalLogName |
|||
{ |
|||
ThreadLocalLogName(std::string _name) { m_name.reset(new std::string(_name)); }; |
|||
boost::thread_specific_ptr<std::string> m_name; |
|||
}; |
|||
|
|||
/// The current thread's name.
|
|||
extern ThreadLocalLogName t_logThreadName; |
|||
|
|||
/// Set the current thread's log name.
|
|||
inline void setThreadName(char const* _n) { t_logThreadName.m_name.reset(new std::string(_n)); } |
|||
|
|||
/// The default logging channels. Each has an associated verbosity and three-letter prefix (name() ).
|
|||
/// Channels should inherit from LogChannel and define name() and verbosity.
|
|||
struct LogChannel { static const char* name() { return " "; } static const int verbosity = 1; }; |
|||
struct LeftChannel: public LogChannel { static const char* name() { return "<<<"; } }; |
|||
struct RightChannel: public LogChannel { static const char* name() { return ">>>"; } }; |
|||
struct WarnChannel: public LogChannel { static const char* name() { return "!!!"; } static const int verbosity = 0; }; |
|||
struct NoteChannel: public LogChannel { static const char* name() { return "***"; } }; |
|||
struct DebugChannel: public LogChannel { static const char* name() { return "---"; } static const int verbosity = 0; }; |
|||
|
|||
/// Logging class, iostream-like, that can be shifted to.
|
|||
template <class Id, bool _AutoSpacing = true> |
|||
class LogOutputStream |
|||
{ |
|||
public: |
|||
/// Construct a new object.
|
|||
/// If _term is true the the prefix info is terminated with a ']' character; if not it ends only with a '|' character.
|
|||
LogOutputStream(bool _term = true) |
|||
{ |
|||
std::type_info const* i = &typeid(Id); |
|||
auto it = g_logOverride.find(i); |
|||
if ((it != g_logOverride.end() && it->second == true) || (it == g_logOverride.end() && Id::verbosity <= g_logVerbosity)) |
|||
{ |
|||
time_t rawTime = std::chrono::system_clock::to_time_t(std::chrono::system_clock::now()); |
|||
char buf[24]; |
|||
if (strftime(buf, 24, "%X", localtime(&rawTime)) == 0) |
|||
buf[0] = '\0'; // empty if case strftime fails
|
|||
m_sstr << Id::name() << " [ " << buf << " | " << *(t_logThreadName.m_name.get()) << (_term ? " ] " : ""); |
|||
} |
|||
} |
|||
|
|||
/// Destructor. Posts the accrued log entry to the g_logPost function.
|
|||
~LogOutputStream() { if (Id::verbosity <= g_logVerbosity) g_logPost(m_sstr.str(), Id::name()); } |
|||
|
|||
/// Shift arbitrary data to the log. Spaces will be added between items as required.
|
|||
template <class T> LogOutputStream& operator<<(T const& _t) { if (Id::verbosity <= g_logVerbosity) { if (_AutoSpacing && m_sstr.str().size() && m_sstr.str().back() != ' ') m_sstr << " "; m_sstr << _t; } return *this; } |
|||
|
|||
private: |
|||
std::stringstream m_sstr; ///< The accrued log entry.
|
|||
}; |
|||
|
|||
// Simple cout-like stream objects for accessing common log channels.
|
|||
// Dirties the global namespace, but oh so convenient...
|
|||
#define cnote eth::LogOutputStream<eth::NoteChannel, true>() |
|||
#define cwarn eth::LogOutputStream<eth::WarnChannel, true>() |
|||
|
|||
// Null stream-like objects.
|
|||
#define ndebug if (true) {} else eth::NullOutputStream() |
|||
#define nlog(X) if (true) {} else eth::NullOutputStream() |
|||
#define nslog(X) if (true) {} else eth::NullOutputStream() |
|||
|
|||
// Kill debugging log channel when we're in release mode.
|
|||
#if NDEBUG |
|||
#define cdebug ndebug |
|||
#else |
|||
#define cdebug eth::LogOutputStream<eth::DebugChannel, true>() |
|||
#endif |
|||
|
|||
// Kill all logs when when NLOG is defined.
|
|||
#if NLOG |
|||
#define clog(X) nlog(X) |
|||
#define cslog(X) nslog(X) |
|||
#else |
|||
#define clog(X) eth::LogOutputStream<X, true>() |
|||
#define cslog(X) eth::LogOutputStream<X, false>() |
|||
#endif |
|||
|
|||
} |
@ -0,0 +1,133 @@ |
|||
/*
|
|||
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 PeerServer.h
|
|||
* @author Gav Wood <i@gavwood.com> |
|||
* @date 2014 |
|||
*/ |
|||
|
|||
#pragma once |
|||
|
|||
#include <map> |
|||
#include <vector> |
|||
#include <set> |
|||
#include <memory> |
|||
#include <utility> |
|||
#include <thread> |
|||
#include "PeerNetwork.h" |
|||
#include "CommonEth.h" |
|||
namespace ba = boost::asio; |
|||
namespace bi = boost::asio::ip; |
|||
|
|||
namespace eth |
|||
{ |
|||
|
|||
class PeerServer |
|||
{ |
|||
friend class PeerSession; |
|||
|
|||
public: |
|||
/// Start server, listening for connections on the given port.
|
|||
PeerServer(std::string const& _clientVersion, BlockChain const& _ch, uint _networkId, unsigned short _port, NodeMode _m = NodeMode::Full, std::string const& _publicAddress = std::string(), bool _upnp = true); |
|||
/// Start server, but don't listen.
|
|||
PeerServer(std::string const& _clientVersion, uint _networkId, NodeMode _m = NodeMode::Full); |
|||
~PeerServer(); |
|||
|
|||
static unsigned protocolVersion(); |
|||
unsigned networkId() { return m_networkId; } |
|||
|
|||
/// Connect to a peer explicitly.
|
|||
void connect(std::string const& _addr, unsigned short _port = 30303) noexcept; |
|||
void connect(bi::tcp::endpoint const& _ep); |
|||
|
|||
/// Sync with the BlockChain. It might contain one of our mined blocks, we might have new candidates from the network.
|
|||
bool sync(BlockChain& _bc, TransactionQueue&, Overlay& _o); |
|||
bool sync(); |
|||
|
|||
/// Conduct I/O, polling, syncing, whatever.
|
|||
/// Ideally all time-consuming I/O is done in a background thread or otherwise asynchronously, but you get this call every 100ms or so anyway.
|
|||
/// This won't touch alter the blockchain.
|
|||
void process() { if (isInitialised()) m_ioService.poll(); } |
|||
|
|||
/// Set ideal number of peers.
|
|||
void setIdealPeerCount(unsigned _n) { m_idealPeerCount = _n; } |
|||
|
|||
void setMode(NodeMode _m) { m_mode = _m; } |
|||
|
|||
/// Get peer information.
|
|||
std::vector<PeerInfo> peers() const; |
|||
|
|||
/// Get number of peers connected; equivalent to, but faster than, peers().size().
|
|||
size_t peerCount() const { return m_peers.size(); } |
|||
|
|||
/// Ping the peers, to update the latency information.
|
|||
void pingAll(); |
|||
|
|||
/// Get the port we're listening on currently.
|
|||
unsigned short listenPort() const { return m_public.port(); } |
|||
|
|||
bytes savePeers() const; |
|||
void restorePeers(bytesConstRef _b); |
|||
|
|||
private: |
|||
void seal(bytes& _b); |
|||
void populateAddresses(); |
|||
void determinePublic(std::string const& _publicAddress, bool _upnp); |
|||
void ensureAccepting(); |
|||
|
|||
/// Check to see if the network peer-state initialisation has happened.
|
|||
bool isInitialised() const { return m_latestBlockSent; } |
|||
/// Initialises the network peer-state, doing the stuff that needs to be once-only. @returns true if it really was first.
|
|||
bool ensureInitialised(BlockChain& _bc, TransactionQueue& _tq); |
|||
|
|||
std::map<Public, bi::tcp::endpoint> potentialPeers(); |
|||
|
|||
std::string m_clientVersion; |
|||
NodeMode m_mode = NodeMode::Full; |
|||
|
|||
unsigned short m_listenPort; |
|||
|
|||
BlockChain const* m_chain = nullptr; |
|||
ba::io_service m_ioService; |
|||
bi::tcp::acceptor m_acceptor; |
|||
bi::tcp::socket m_socket; |
|||
|
|||
UPnP* m_upnp = nullptr; |
|||
bi::tcp::endpoint m_public; |
|||
KeyPair m_key; |
|||
|
|||
unsigned m_networkId; |
|||
std::map<Public, std::weak_ptr<PeerSession>> m_peers; |
|||
|
|||
std::vector<bytes> m_incomingTransactions; |
|||
std::vector<bytes> m_incomingBlocks; |
|||
std::vector<bytes> m_unknownParentBlocks; |
|||
std::vector<Public> m_freePeers; |
|||
std::map<Public, std::pair<bi::tcp::endpoint, unsigned>> m_incomingPeers; |
|||
|
|||
h256 m_latestBlockSent; |
|||
std::set<h256> m_transactionsSent; |
|||
|
|||
std::chrono::steady_clock::time_point m_lastPeersRequest; |
|||
unsigned m_idealPeerCount = 5; |
|||
|
|||
std::vector<bi::address_v4> m_addresses; |
|||
std::vector<bi::address_v4> m_peerAddresses; |
|||
|
|||
bool m_accepting = false; |
|||
}; |
|||
|
|||
} |
@ -0,0 +1,90 @@ |
|||
/*
|
|||
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 PeerSession.h
|
|||
* @author Gav Wood <i@gavwood.com> |
|||
* @date 2014 |
|||
*/ |
|||
|
|||
#pragma once |
|||
|
|||
#include <array> |
|||
#include <set> |
|||
#include <memory> |
|||
#include <utility> |
|||
#include "RLP.h" |
|||
#include "CommonEth.h" |
|||
#include "PeerNetwork.h" |
|||
|
|||
namespace eth |
|||
{ |
|||
|
|||
class PeerSession: public std::enable_shared_from_this<PeerSession> |
|||
{ |
|||
friend class PeerServer; |
|||
|
|||
public: |
|||
PeerSession(PeerServer* _server, bi::tcp::socket _socket, uint _rNId, bi::address _peerAddress, unsigned short _peerPort = 0); |
|||
~PeerSession(); |
|||
|
|||
void start(); |
|||
void disconnect(int _reason); |
|||
|
|||
void ping(); |
|||
|
|||
bool isOpen() const { return m_socket.is_open(); } |
|||
|
|||
bi::tcp::endpoint endpoint() const; ///< for other peers to connect to.
|
|||
|
|||
private: |
|||
void dropped(); |
|||
void doRead(); |
|||
void doWrite(std::size_t length); |
|||
bool interpret(RLP const& _r); |
|||
|
|||
/// @returns true iff the _msg forms a valid message for sending or receiving on the network.
|
|||
static bool checkPacket(bytesConstRef _msg); |
|||
|
|||
static RLPStream& prep(RLPStream& _s); |
|||
void sealAndSend(RLPStream& _s); |
|||
void sendDestroy(bytes& _msg); |
|||
void send(bytesConstRef _msg); |
|||
PeerServer* m_server; |
|||
|
|||
bi::tcp::socket m_socket; |
|||
std::array<byte, 65536> m_data; |
|||
PeerInfo m_info; |
|||
Public m_id; |
|||
|
|||
bytes m_incoming; |
|||
uint m_protocolVersion; |
|||
uint m_networkId; |
|||
uint m_reqNetworkId; |
|||
unsigned short m_listenPort; ///< Port that the remote client is listening on for connections. Useful for giving to peers.
|
|||
uint m_caps; |
|||
|
|||
std::chrono::steady_clock::time_point m_ping; |
|||
std::chrono::steady_clock::time_point m_connect; |
|||
std::chrono::steady_clock::time_point m_disconnect; |
|||
|
|||
uint m_rating; |
|||
bool m_requireTransactions; |
|||
|
|||
std::set<h256> m_knownBlocks; |
|||
std::set<h256> m_knownTransactions; |
|||
}; |
|||
|
|||
} |
Loading…
Reference in new issue