Add SignerList to account_info response (RIPD-1061):

The caller of the account_info RPC command can optionally
specify that they want the account's SignerList returned by
adding the argument:

"signer_lists": "true"

The returned SignerList is in an array.  This leaves us room to
support multiple signer lists on an account in the future without
changing the syntax of the result.

The command-line version of account_info does not support the new
option.
This commit is contained in:
Scott Schurr
2016-01-07 09:53:59 -08:00
committed by Vinnie Falco
parent fbf736f169
commit 2f94e16359
6 changed files with 195 additions and 3 deletions

View File

@@ -3275,6 +3275,10 @@
</ClInclude>
<ClInclude Include="..\..\src\ripple\rpc\Status.h">
</ClInclude>
<ClCompile Include="..\..\src\ripple\rpc\tests\AccountInfo_test.cpp">
<ExcludedFromBuild Condition="'$(Configuration)|$(Platform)'=='debug|x64'">True</ExcludedFromBuild>
<ExcludedFromBuild Condition="'$(Configuration)|$(Platform)'=='release|x64'">True</ExcludedFromBuild>
</ClCompile>
<ClCompile Include="..\..\src\ripple\rpc\tests\AccountLinesRPC.test.cpp">
<ExcludedFromBuild Condition="'$(Configuration)|$(Platform)'=='debug|x64'">True</ExcludedFromBuild>
<ExcludedFromBuild Condition="'$(Configuration)|$(Platform)'=='release|x64'">True</ExcludedFromBuild>

View File

@@ -3765,6 +3765,9 @@
<ClInclude Include="..\..\src\ripple\rpc\Status.h">
<Filter>ripple\rpc</Filter>
</ClInclude>
<ClCompile Include="..\..\src\ripple\rpc\tests\AccountInfo_test.cpp">
<Filter>ripple\rpc\tests</Filter>
</ClCompile>
<ClCompile Include="..\..\src\ripple\rpc\tests\AccountLinesRPC.test.cpp">
<Filter>ripple\rpc\tests</Filter>
</ClCompile>

View File

@@ -344,6 +344,7 @@ JSS ( severity ); // in: LogLevel
JSS ( signature ); // out: NetworkOPs
JSS ( signing_key ); // out: NetworkOPs
JSS ( signer_list ); // in: AccountObjects
JSS ( signer_lists ); // in/out: AccountInfo
JSS ( snapshot ); // in: Subscribe
JSS ( source_account ); // in: PathRequest, RipplePathFind
JSS ( source_amount ); // in: PathRequest, RipplePathFind

View File

@@ -34,11 +34,13 @@
namespace ripple {
// {
// account: <indent>,
// strict: <bool>
// if true, only allow public keys and addresses. false, default.
// account: <ident>,
// strict: <bool> // optional (default false)
// // if true only allow public keys and addresses.
// ledger_hash : <ledger>
// ledger_index : <ledger_index>
// signer_lists : <bool> // optional (default false)
// // if true return SignerList(s).
// }
// TODO(tom): what is that "default"?
@@ -72,6 +74,22 @@ Json::Value doAccountInfo (RPC::Context& context)
{
RPC::injectSLE(jvAccepted, *sleAccepted);
result[jss::account_data] = jvAccepted;
// Return SignerList(s) if that is requested.
if (params.isMember (jss::signer_lists) &&
params[jss::signer_lists].asBool ())
{
// We put the SignerList in an array because of an anticipated
// future when we support multiple signer lists on one account.
auto& jvSignerList = result[jss::account_data][jss::signer_lists];
jvSignerList = Json::arrayValue;
// This code will need to be revisited if in the future we support
// multiple SignerLists on one account.
auto const sleSigners = ledger->read (keylet::signers (accountID));
if (sleSigners)
jvSignerList.append (sleSigners->getJson (0));
}
}
else
{

View File

@@ -0,0 +1,165 @@
//------------------------------------------------------------------------------
/*
This file is part of rippled: https://github.com/ripple/rippled
Copyright (c) 2016 Ripple Labs Inc.
Permission to use, copy, modify, and/or distribute this software for any
purpose with or without fee is hereby granted, provided that the above
copyright notice and this permission notice appear in all copies.
THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
ANY SPECIAL , DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
*/
//==============================================================================
#include <ripple/protocol/JsonFields.h> // jss:: definitions
#include <ripple/protocol/Feature.h>
#include <ripple/test/jtx.h>
namespace ripple {
namespace test {
class AccountInfo_test : public beast::unit_test::suite
{
public:
void testErrors()
{
using namespace jtx;
Env env(*this);
{
// account_info with no account.
auto const info = env.rpc ("json", "account_info", "{ }");
expect (info[jss::result][jss::error_message] ==
"Missing field 'account'.");
}
{
// account_info with a malformed account sting.
auto const info = env.rpc ("json", "account_info", "{\"account\": "
"\"n94JNrQYkDrpt62bbSR7nVEhdyAvcJXRAsjEkFYyqRkh9SUTYEqV\"}");
expect (info[jss::result][jss::error_message] ==
"Disallowed seed.");
}
{
// account_info with an account that's not in the ledger.
Account const bogie {"bogie"};
auto const info = env.rpc ("json", "account_info",
std::string ("{ ") + "\"account\": \"" + bogie.human() + "\"}");
expect (info[jss::result][jss::error_message] ==
"Account not found.");
}
}
// Test the "signer_lists" argument in account_info.
void testSignerLists()
{
using namespace jtx;
Env env(*this, features(featureMultiSign));
Account const alice {"alice"};
env.fund(XRP(1000), alice);
auto const withoutSigners = std::string ("{ ") +
"\"account\": \"" + alice.human() + "\"}";
auto const withSigners = std::string ("{ ") +
"\"account\": \"" + alice.human() + "\", " +
"\"signer_lists\": true }";
// Alice has no SignerList yet.
{
// account_info without the "signer_lists" argument.
auto const info = env.rpc ("json", "account_info", withoutSigners);
expect (! info[jss::result][jss::account_data].
isMember (jss::signer_lists));
}
{
// account_info with the "signer_lists" argument.
auto const info = env.rpc ("json", "account_info", withSigners);
auto const& data = info[jss::result][jss::account_data];
expect (data.isMember (jss::signer_lists));
auto const& signerLists = data[jss::signer_lists];
expect (signerLists.isArray());
expect (signerLists.size() == 0);
}
// Give alice a SignerList.
Account const bogie {"bogie"};
Json::Value const smallSigners = signers(alice, 2, { { bogie, 3 } });
env(smallSigners);
{
// account_info without the "signer_lists" argument.
auto const info = env.rpc ("json", "account_info", withoutSigners);
expect (! info[jss::result][jss::account_data].
isMember (jss::signer_lists));
}
{
// account_info with the "signer_lists" argument.
auto const info = env.rpc ("json", "account_info", withSigners);
auto const& data = info[jss::result][jss::account_data];
expect (data.isMember (jss::signer_lists));
auto const& signerLists = data[jss::signer_lists];
expect (signerLists.isArray());
expect (signerLists.size() == 1);
auto const& signers = signerLists[0u];
expect (signers.isObject());
expect (signers[sfSignerQuorum.jsonName] == 2);
auto const& signerEntries = signers[sfSignerEntries.jsonName];
expect (signerEntries.size() == 1);
auto const& entry0 = signerEntries[0u][sfSignerEntry.jsonName];
expect (entry0[sfSignerWeight.jsonName] == 3);
}
// Give alice a big signer list
Account const demon {"demon"};
Account const ghost {"ghost"};
Account const haunt {"haunt"};
Account const jinni {"jinni"};
Account const phase {"phase"};
Account const shade {"shade"};
Account const spook {"spook"};
Json::Value const bigSigners = signers(alice, 4, {
{bogie, 1}, {demon, 1}, {ghost, 1}, {haunt, 1},
{jinni, 1}, {phase, 1}, {shade, 1}, {spook, 1}, });
env(bigSigners);
{
// account_info with the "signer_lists" argument.
auto const info = env.rpc ("json", "account_info", withSigners);
auto const& data = info[jss::result][jss::account_data];
expect (data.isMember (jss::signer_lists));
auto const& signerLists = data[jss::signer_lists];
expect (signerLists.isArray());
expect (signerLists.size() == 1);
auto const& signers = signerLists[0u];
expect (signers.isObject());
expect (signers[sfSignerQuorum.jsonName] == 4);
auto const& signerEntries = signers[sfSignerEntries.jsonName];
expect (signerEntries.size() == 8);
for (unsigned i = 0u; i < 8; ++i)
{
auto const& entry = signerEntries[i][sfSignerEntry.jsonName];
expect (entry.size() == 2);
expect (entry.isMember(sfAccount.jsonName));
expect (entry[sfSignerWeight.jsonName] == 1);
}
}
}
void run()
{
testErrors();
testSignerLists();
}
};
BEAST_DEFINE_TESTSUITE(AccountInfo,app,ripple);
}
}

View File

@@ -99,6 +99,7 @@
#include <ripple/rpc/impl/TransactionSign.cpp>
#include <ripple/rpc/impl/RPCVersion.cpp>
#include <ripple/rpc/tests/AccountInfo_test.cpp>
#include <ripple/rpc/tests/AccountLinesRPC.test.cpp>
#include <ripple/rpc/tests/JSONRPC.test.cpp>
#include <ripple/rpc/tests/LedgerRequestRPC.test.cpp>