Skip to content

Commit

Permalink
Merge #373: Make name value optional in name_firstupdate and name_update
Browse files Browse the repository at this point in the history
3ea8ba6 rpc, test: Make name value optional in name_firstupdate and name_update (yanmaani)

Pull request description:

  This patch amends `name_firstupdate` and `name_update` to make the `value` parameter optional. As proposed in #372, a missing value in `name_firstupdate` equals the empty string, and a missing value in `name_update` equals the last value of the name.

  It also amends the test suite to support writing tests that do not furnish name values. That change is intended to make it easier to write future tests; it isn't relied upon by the test of the implemented feature, which uses direct RPC calls.

  Fixes #372.

ACKs for top commit:
  domob1812:
    ACK 3ea8ba6.

Tree-SHA512: 4249475423ef5274da8e3e84e7d2897332cfe3a7f89b413390024ee9c9e2d2ebfd14b9bc8d1636e021eb4054673312207d81799244ed69bfda1f0806a9bb2ad2
  • Loading branch information
domob1812 committed Oct 20, 2020
2 parents 4a08daa + 3ea8ba6 commit 94dba95
Show file tree
Hide file tree
Showing 4 changed files with 100 additions and 19 deletions.
50 changes: 34 additions & 16 deletions src/wallet/rpcnames.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -447,7 +447,7 @@ name_firstupdate (const JSONRPCRequest& request)

/* We can not use RPCHelpMan::Check here, as we have that "hidden" sixth
argument for "allow active" names (in tests). */
if (request.fHelp || request.params.size () < 4 || request.params.size () > 6)
if (request.fHelp || request.params.size () < 3 || request.params.size () > 6)
throw std::runtime_error (
RPCHelpMan ("name_firstupdate",
"\nFinishes the registration of a name. Depends on name_new being already issued."
Expand All @@ -456,7 +456,7 @@ name_firstupdate (const JSONRPCRequest& request)
{"name", RPCArg::Type::STR, RPCArg::Optional::NO, "The name to register"},
{"rand", RPCArg::Type::STR_HEX, RPCArg::Optional::NO, "The rand value of name_new"},
{"tx", RPCArg::Type::STR_HEX, RPCArg::Optional::NO, "The name_new txid"},
{"value", RPCArg::Type::STR, RPCArg::Optional::NO, "Value for the name"},
{"value", RPCArg::Type::STR, RPCArg::Optional::OMITTED_NAMED_ARG, "Value for the name"},
optHelp.buildRpcArg (),
},
RPCResult {RPCResult::Type::STR_HEX, "", "the transaction ID"},
Expand All @@ -473,7 +473,7 @@ name_firstupdate (const JSONRPCRequest& request)

RPCTypeCheck (request.params,
{UniValue::VSTR, UniValue::VSTR, UniValue::VSTR, UniValue::VSTR,
UniValue::VOBJ});
UniValue::VOBJ}, true);

UniValue options(UniValue::VOBJ);
if (request.params.size () >= 5)
Expand All @@ -489,7 +489,11 @@ name_firstupdate (const JSONRPCRequest& request)

const uint256 prevTxid = ParseHashV (request.params[2], "txid");

const valtype value = DecodeValueFromRPCOrThrow (request.params[3], options);
const bool isDefaultVal = (request.params.size () < 4 || request.params[3].isNull ());
const valtype value = isDefaultVal ?
valtype ():
DecodeValueFromRPCOrThrow (request.params[3], options);

if (value.size () > MAX_VALUE_LENGTH_UI)
throw JSONRPCError (RPC_INVALID_PARAMETER, "the value is too long");

Expand Down Expand Up @@ -567,7 +571,7 @@ name_update ()
+ HELP_REQUIRING_PASSPHRASE,
{
{"name", RPCArg::Type::STR, RPCArg::Optional::NO, "The name to update"},
{"value", RPCArg::Type::STR, RPCArg::Optional::NO, "Value for the name"},
{"value", RPCArg::Type::STR, RPCArg::Optional::OMITTED_NAMED_ARG, "Value for the name"},
optHelp.buildRpcArg (),
},
RPCResult {RPCResult::Type::STR_HEX, "", "the transaction ID"},
Expand All @@ -583,7 +587,7 @@ name_update ()
CWallet* const pwallet = wallet.get ();

RPCTypeCheck (request.params,
{UniValue::VSTR, UniValue::VSTR, UniValue::VOBJ});
{UniValue::VSTR, UniValue::VSTR, UniValue::VOBJ}, true);

UniValue options(UniValue::VOBJ);
if (request.params.size () >= 3)
Expand All @@ -593,14 +597,21 @@ name_update ()
if (name.size () > MAX_NAME_LENGTH)
throw JSONRPCError (RPC_INVALID_PARAMETER, "the name is too long");

const valtype value = DecodeValueFromRPCOrThrow (request.params[1], options);
if (value.size () > MAX_VALUE_LENGTH_UI)
throw JSONRPCError (RPC_INVALID_PARAMETER, "the value is too long");
const bool isDefaultVal = request.params.size() < 2 || request.params[1].isNull();

/* For finding the name output to spend, we first check if there are
pending operations on the name in the mempool. If there are, then we
build upon the last one to get a valid chain. If there are none, then we
look up the last outpoint from the name database instead. */
valtype value;
if (!isDefaultVal) {
value = DecodeValueFromRPCOrThrow (request.params[1], options);
if (value.size () > MAX_VALUE_LENGTH_UI)
throw JSONRPCError (RPC_INVALID_PARAMETER, "the value is too long");
}

/* For finding the name output to spend and its value, we first check if
there are pending operations on the name in the mempool. If there
are, then we build upon the last one to get a valid chain. If there
are none, then we look up the last outpoint from the name database
instead. */
// TODO: Use name_show for this instead.

const unsigned chainLimit = gArgs.GetArg ("-limitnamechains",
DEFAULT_NAME_CHAIN_LIMIT);
Expand All @@ -616,7 +627,14 @@ name_update ()
" on this name");

if (pendingOps > 0)
outp = mempool.lastNameOutput (name);
{
outp = mempool.lastNameOutput (name);
if (isDefaultVal)
{
const auto& tx = mempool.mapTx.find(outp.hash)->GetTx();
value = CNameScript(tx.vout[outp.n].scriptPubKey).getOpValue();
}
}
}

if (outp.IsNull ())
Expand All @@ -628,10 +646,10 @@ name_update ()
if (!coinsTip.GetName (name, oldData) || oldData.isExpired ())
throw JSONRPCError (RPC_TRANSACTION_ERROR,
"this name can not be updated");

if (isDefaultVal)
value = oldData.getValue();
outp = oldData.getUpdateOutpoint ();
}

assert (!outp.IsNull ());
const CTxIn txIn(outp);

Expand Down
60 changes: 60 additions & 0 deletions test/functional/name_novalue.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,60 @@
#!/usr/bin/env python3
# Licensed under CC0 (Public domain)

# Test that name_firstupdate and name_update handle missing
# name values gracefully. This test uses direct RPC calls.
# Testing the wrappers is out of scope.

from test_framework.names import NameTestFramework
from test_framework.util import *

class NameNovalueTest(NameTestFramework):

def set_test_params(self):
self.setup_clean_chain = True
self.setup_name_test ([["-namehistory", "-limitnamechains=3"]])

def run_test(self):
node = self.nodes[0]
node.generate(200)

new_name = node.name_new("d/name")
node.generate(12)
self.log.info("Began registration of name.")

# use node.name_firstupdate - we're not testing the framework
node.name_firstupdate ("d/name", new_name[1], new_name[0])
node.generate(1)
self.log.info("Name registered; no value provided.")

self.checkName(0, "d/name", "", 30, False)
self.log.info("Value equals empty string.")

node.name_update("d/name", "1")
node.generate(1)
self.log.info('Value changed to "1"; change is in chain.')

self.checkName(0, "d/name", "1", 30, False)
self.log.info('Value equals "1".')

node.name_update("d/name")
node.generate(1)
self.log.info("Updated; no value specified.")

self.checkName(0, "d/name", "1", 30, False)
self.log.info('Name was updated. Value still equals "1".')

node.name_update("d/name", "2")
node.name_update("d/name")
node.name_update("d/name", "3")
node.generate(1)
self.log.info('Stack of 3 registrations performed.')

history = node.name_history("d/name")
reference = ["", "1", "1", "2", "2", "3"]
assert(list(map(lambda x: x['value'], history)) == reference)
self.log.info('Updates correctly consider updates in the mempool.')


if __name__ == '__main__':
NameNovalueTest ().main ()
8 changes: 5 additions & 3 deletions test/functional/test_framework/names.py
Original file line number Diff line number Diff line change
Expand Up @@ -21,7 +21,7 @@ def setup_name_test (self, args = [[]] * 4):
# test_framework.py. This is needed to get us out of IBD.
self.mocktime = 1388534400 + (201 * 10 * 60)

def firstupdateName (self, ind, name, newData, value,
def firstupdateName (self, ind, name, newData, value = None,
opt = None, allowActive = False):
"""
Utility routine to perform a name_firstupdate command. The rand
Expand All @@ -36,9 +36,11 @@ def firstupdateName (self, ind, name, newData, value,
return node.name_firstupdate (name, newData[1], newData[0],
value, opt, True)

if opt is None:
if opt is not None:
return node.name_firstupdate (name, newData[1], newData[0], value, opt)
if value is not None:
return node.name_firstupdate (name, newData[1], newData[0], value)
return node.name_firstupdate (name, newData[1], newData[0], value, opt)
return node.name_firstupdate (name, newData[1], newData[0])

def checkName (self, ind, name, value, expiresIn, expired):
"""
Expand Down
1 change: 1 addition & 0 deletions test/functional/test_runner.py
Original file line number Diff line number Diff line change
Expand Up @@ -285,6 +285,7 @@
'name_multisig.py',
'name_multisig.py --bip16-active',
'name_multiupdate.py',
'name_novalue.py',
'name_pending.py',
'name_psbt.py',
'name_rawtx.py',
Expand Down

0 comments on commit 94dba95

Please sign in to comment.