// Copyright (c) 2018 Jianping Wu // Distributed under the MIT/X11 software license, see the accompanying // file COPYING or http://www.opensource.org/licenses/mit-license.php. #include "base58.h" #include "coins.h" #include "init.h" #include "keva/common.h" #include "keva/main.h" #include "primitives/transaction.h" #include "random.h" #include "rpc/mining.h" #include "rpc/safemode.h" #include "rpc/server.h" #include "script/kava.h" #include "txmempool.h" #include "util.h" #include "validation.h" #include "wallet/coincontrol.h" #include "wallet/wallet.h" #include /* ************************************************************************** */ UniValue keva_put (const JSONRPCRequest& request) { CWallet* const pwallet = GetWalletForJSONRPCRequest(request); if (!EnsureWalletIsAvailable (pwallet, request.fHelp)) return NullUniValue; if (request.fHelp || (request.params.size () != 3 && request.params.size () != 4)) throw std::runtime_error ( "keva_put \"namespace\" \"key\" \"value\" (\"create_namespace\")\n" "\nUpdate a name and possibly transfer it.\n" + HelpRequiringPassphrase (pwallet) + "\nArguments:\n" "1. \"namespace\" (string, required) the namespace to insert the key to\n" "2. \"key\" (string, required) value for the key\n" "4. \"value\" (string, required) value for the name\n" "5. \"create_namespace\" (boolean, optional, default=false) create the namespace if it does not exist yet\n" "\nResult:\n" "\"txid\" (string) the keva_put's txid\n" "\nExamples:\n" + HelpExampleCli ("keva_put", "\"mynamespace\", \"new-key\", \"new-value\"") + HelpExampleCli ("keva_put", "\"mynamespace\", \"new-key\", \"new-value\", true") ); RPCTypeCheck (request.params, {UniValue::VSTR, UniValue::VSTR, UniValue::VSTR, UniValue::VSTR, UniValue::VBOOL}); ObserveSafeMode (); const std::string namespaceStr = request.params[0].get_str (); const valtype namespaceVal = ValtypeFromString (namespaceStr); if (namespaceVal.size () > MAX_NAMESPACE_LENGTH) throw JSONRPCError (RPC_INVALID_PARAMETER, "the namespace is too long"); const std::string keyStr = request.params[1].get_str (); const valtype key = ValtypeFromString (keyStr); if (key.size () > MAX_KEY_LENGTH) throw JSONRPCError (RPC_INVALID_PARAMETER, "the key is too long"); const std::string valueStr = request.params[2].get_str (); const valtype value = ValtypeFromString (valueStr); if (value.size () > MAX_VALUE_LENGTH) throw JSONRPCError (RPC_INVALID_PARAMETER, "the value is too long"); bool createNamespace = false; if (!request.params[3].isNull()) { createNamespace = request.params[3].get_bool(); } /* Reject updates to a name for which the mempool already has a pending update. This is not a hard rule enforced by network rules, but it is necessary with the current mempool implementation. */ { LOCK (mempool.cs); if (mempool.updatesName (name)) throw JSONRPCError (RPC_TRANSACTION_ERROR, "there is already a pending update for this name"); } CNameData oldData; { LOCK (cs_main); if (!pcoinsTip->GetName (name, oldData) || oldData.isExpired ()) throw JSONRPCError (RPC_TRANSACTION_ERROR, "this name can not be updated"); } const COutPoint outp = oldData.getUpdateOutpoint (); const CTxIn txIn(outp); /* No more locking required, similarly to name_new. */ EnsureWalletIsUnlocked (pwallet); CReserveKey keyName(pwallet); CPubKey pubKeyReserve; const bool ok = keyName.GetReservedKey (pubKeyReserve, true); assert (ok); bool usedKey = false; CScript addrName; if (request.params.size () == 3) { keyName.ReturnKey (); const CTxDestination dest = DecodeDestination (request.params[2].get_str ()); if (!IsValidDestination (dest)) throw JSONRPCError (RPC_INVALID_ADDRESS_OR_KEY, "invalid address"); addrName = GetScriptForDestination (dest); } else { usedKey = true; addrName = GetScriptForDestination (pubKeyReserve.GetID ()); } const CScript nameScript = CNameScript::buildNameUpdate (addrName, name, value); CCoinControl coinControl; CWalletTx wtx; SendMoneyToScript (pwallet, nameScript, &txIn, NAME_LOCKED_AMOUNT, false, wtx, coinControl); if (usedKey) keyName.KeepKey (); return wtx.GetHash ().GetHex (); }