Christian
10 years ago
8 changed files with 365 additions and 65 deletions
@ -0,0 +1,102 @@ |
|||||
|
/*
|
||||
|
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 AccountHolder.cpp
|
||||
|
* @authors: |
||||
|
* Christian R <c@ethdev.com> |
||||
|
* Lefteris Karapetsas <lefteris@ethdev.com> |
||||
|
* @date 2015 |
||||
|
*/ |
||||
|
|
||||
|
#include "AccountHolder.h" |
||||
|
#include <libethereum/Client.h> |
||||
|
|
||||
|
|
||||
|
using namespace std; |
||||
|
using namespace dev; |
||||
|
using namespace dev::eth; |
||||
|
|
||||
|
vector<TransactionSkeleton> emptyQueue; |
||||
|
|
||||
|
void AccountHolder::setAccounts(vector<KeyPair> const& _accounts) |
||||
|
{ |
||||
|
m_accounts.clear(); |
||||
|
for (auto const& keyPair: _accounts) |
||||
|
{ |
||||
|
m_accounts.push_back(keyPair.address()); |
||||
|
m_keyPairs[keyPair.address()] = keyPair; |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
vector<Address> AccountHolder::getAllAccounts() const |
||||
|
{ |
||||
|
vector<Address> accounts = m_accounts; |
||||
|
for (auto const& pair: m_proxyAccounts) |
||||
|
if (!isRealAccount(pair.first)) |
||||
|
accounts.push_back(pair.first); |
||||
|
return accounts; |
||||
|
} |
||||
|
|
||||
|
Address const& AccountHolder::getDefaultCallAccount() const |
||||
|
{ |
||||
|
if (m_accounts.empty()) |
||||
|
return ZeroAddress; |
||||
|
Address const* bestMatch = &m_accounts.front(); |
||||
|
for (auto const& account: m_accounts) |
||||
|
if (m_client()->balanceAt(account) > m_client()->balanceAt(*bestMatch)) |
||||
|
bestMatch = &account; |
||||
|
return *bestMatch; |
||||
|
} |
||||
|
|
||||
|
int AccountHolder::addProxyAccount(const Address& _account) |
||||
|
{ |
||||
|
int const c_id = m_transactionQueues.empty() ? 1 : m_transactionQueues.rbegin()->first + 1; |
||||
|
if (isProxyAccount(_account)) |
||||
|
return 0; |
||||
|
m_proxyAccounts.insert(make_pair(_account, c_id)); |
||||
|
m_transactionQueues[c_id].first = _account; |
||||
|
return c_id; |
||||
|
} |
||||
|
|
||||
|
bool AccountHolder::removeProxyAccount(unsigned _id) |
||||
|
{ |
||||
|
if (!m_transactionQueues.count(_id)) |
||||
|
return false; |
||||
|
m_proxyAccounts.erase(m_transactionQueues[_id].first); |
||||
|
m_transactionQueues.erase(_id); |
||||
|
return true; |
||||
|
} |
||||
|
|
||||
|
void AccountHolder::queueTransaction(TransactionSkeleton const& _transaction) |
||||
|
{ |
||||
|
if (!m_proxyAccounts.count(_transaction.from)) |
||||
|
return; |
||||
|
int id = m_proxyAccounts[_transaction.from]; |
||||
|
m_transactionQueues[id].second.push_back(_transaction); |
||||
|
} |
||||
|
|
||||
|
vector<TransactionSkeleton> const& AccountHolder::getQueuedTransactions(int _id) const |
||||
|
{ |
||||
|
if (!m_transactionQueues.count(_id)) |
||||
|
return emptyQueue; |
||||
|
return m_transactionQueues.at(_id).second; |
||||
|
} |
||||
|
|
||||
|
void AccountHolder::clearQueue(int _id) |
||||
|
{ |
||||
|
if (m_transactionQueues.count(_id)) |
||||
|
m_transactionQueues.at(_id).second.clear(); |
||||
|
} |
@ -0,0 +1,74 @@ |
|||||
|
/*
|
||||
|
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 AccountHolder.h
|
||||
|
* @authors: |
||||
|
* Christian R <c@ethdev.com> |
||||
|
* Lefteris Karapetsas <lefteris@ethdev.com> |
||||
|
* @date 2015 |
||||
|
*/ |
||||
|
|
||||
|
#pragma once |
||||
|
|
||||
|
#include <functional> |
||||
|
#include <vector> |
||||
|
#include <map> |
||||
|
#include <libdevcrypto/Common.h> |
||||
|
#include <libethcore/CommonJS.h> |
||||
|
|
||||
|
namespace dev |
||||
|
{ |
||||
|
namespace eth |
||||
|
{ |
||||
|
class Interface; |
||||
|
} |
||||
|
|
||||
|
/**
|
||||
|
* Manages real accounts (where we know the secret key) and proxy accounts (where transactions |
||||
|
* to be sent from these accounts are forwarded to a proxy on the other side). |
||||
|
*/ |
||||
|
class AccountHolder |
||||
|
{ |
||||
|
public: |
||||
|
explicit AccountHolder(std::function<eth::Interface*()> const& _client): m_client(_client) {} |
||||
|
|
||||
|
/// Sets or resets the list of real accounts.
|
||||
|
void setAccounts(std::vector<KeyPair> const& _accounts); |
||||
|
std::vector<Address> const& getRealAccounts() const { return m_accounts; } |
||||
|
bool isRealAccount(Address const& _account) const { return m_keyPairs.count(_account) > 0; } |
||||
|
bool isProxyAccount(Address const& _account) const { return m_proxyAccounts.count(_account) > 0; } |
||||
|
Secret const& secretKey(Address const& _account) const { return m_keyPairs.at(_account).secret(); } |
||||
|
std::vector<Address> getAllAccounts() const; |
||||
|
Address const& getDefaultCallAccount() const; |
||||
|
|
||||
|
int addProxyAccount(Address const& _account); |
||||
|
bool removeProxyAccount(unsigned _id); |
||||
|
void queueTransaction(eth::TransactionSkeleton const& _transaction); |
||||
|
|
||||
|
std::vector<eth::TransactionSkeleton> const& getQueuedTransactions(int _id) const; |
||||
|
void clearQueue(int _id); |
||||
|
|
||||
|
private: |
||||
|
using TransactionQueue = std::vector<eth::TransactionSkeleton>; |
||||
|
|
||||
|
std::map<Address, KeyPair> m_keyPairs; |
||||
|
std::vector<Address> m_accounts; |
||||
|
std::map<Address, int> m_proxyAccounts; |
||||
|
std::map<int, std::pair<Address, TransactionQueue>> m_transactionQueues; |
||||
|
std::function<eth::Interface*()> m_client; |
||||
|
}; |
||||
|
|
||||
|
} |
@ -0,0 +1,74 @@ |
|||||
|
/*
|
||||
|
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/>.
|
||||
|
*/ |
||||
|
/**
|
||||
|
* @author Christian R <c@ethdev.com> |
||||
|
* @date 2015 |
||||
|
* Unit tests for the account holder used by the WebThreeStubServer. |
||||
|
*/ |
||||
|
|
||||
|
#include <boost/test/unit_test.hpp> |
||||
|
#include <libweb3jsonrpc/AccountHolder.h> |
||||
|
|
||||
|
namespace dev |
||||
|
{ |
||||
|
namespace test |
||||
|
{ |
||||
|
|
||||
|
BOOST_AUTO_TEST_SUITE(AccountHolderTest) |
||||
|
|
||||
|
BOOST_AUTO_TEST_CASE(ProxyAccountUseCase) |
||||
|
{ |
||||
|
AccountHolder h = AccountHolder(std::function<eth::Interface*()>()); |
||||
|
BOOST_CHECK(h.getAllAccounts().empty()); |
||||
|
BOOST_CHECK(h.getRealAccounts().empty()); |
||||
|
Address addr("abababababababababababababababababababab"); |
||||
|
Address addr2("abababababababababababababababababababab"); |
||||
|
int id = h.addProxyAccount(addr); |
||||
|
BOOST_CHECK(h.getQueuedTransactions(id).empty()); |
||||
|
// register it again
|
||||
|
int secondID = h.addProxyAccount(addr); |
||||
|
BOOST_CHECK(h.getQueuedTransactions(secondID).empty()); |
||||
|
|
||||
|
eth::TransactionSkeleton t1; |
||||
|
eth::TransactionSkeleton t2; |
||||
|
t1.from = addr; |
||||
|
t1.data = fromHex("12345678"); |
||||
|
t2.from = addr; |
||||
|
t2.data = fromHex("abcdef"); |
||||
|
BOOST_CHECK(h.getQueuedTransactions(id).empty()); |
||||
|
h.queueTransaction(t1); |
||||
|
BOOST_CHECK_EQUAL(1, h.getQueuedTransactions(id).size()); |
||||
|
h.queueTransaction(t2); |
||||
|
BOOST_REQUIRE_EQUAL(2, h.getQueuedTransactions(id).size()); |
||||
|
|
||||
|
// second proxy should not see transactions
|
||||
|
BOOST_CHECK(h.getQueuedTransactions(secondID).empty()); |
||||
|
|
||||
|
BOOST_CHECK(h.getQueuedTransactions(id)[0].data == t1.data); |
||||
|
BOOST_CHECK(h.getQueuedTransactions(id)[1].data == t2.data); |
||||
|
|
||||
|
h.clearQueue(id); |
||||
|
BOOST_CHECK(h.getQueuedTransactions(id).empty()); |
||||
|
// removing fails because it never existed
|
||||
|
BOOST_CHECK(!h.removeProxyAccount(secondID)); |
||||
|
BOOST_CHECK(h.removeProxyAccount(id)); |
||||
|
} |
||||
|
|
||||
|
BOOST_AUTO_TEST_SUITE_END() |
||||
|
|
||||
|
} |
||||
|
} |
Loading…
Reference in new issue