// 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_namespace (const JSONRPCRequest& request) { CWallet* const pwallet = GetWalletForJSONRPCRequest(request); if (!EnsureWalletIsAvailable (pwallet, request.fHelp)) return NullUniValue; if (request.fHelp || request.params.size () != 1) throw std::runtime_error ( "keva_namespace \"display_name\"\n" "\nStart creation of the given namespace.\n" + HelpRequiringPassphrase (pwallet) + "\nArguments:\n" "1. \"display_name\" (string, required) the display name of the namespace\n" "\nResult:\n" "[\n" " xxxxx, (string) the txid, required for keva_put\n" "]\n" "\nExamples:\n" + HelpExampleCli ("keva_namespace", "\"display name\"") ); RPCTypeCheck (request.params, {UniValue::VSTR}); ObserveSafeMode (); const std::string nameStr = request.params[0].get_str (); const valtype name = ValtypeFromString (nameStr); if (name.size () > MAX_NAME_LENGTH) throw JSONRPCError (RPC_INVALID_PARAMETER, "the name is too long"); /* No explicit locking should be necessary. CReserveKey takes care of locking the wallet, and CommitTransaction (called when sending the tx) locks cs_main as necessary. */ EnsureWalletIsUnlocked (pwallet); CReserveKey keyName(pwallet); CPubKey pubKey; const bool ok = keyName.GetReservedKey (pubKey, true); assert (ok); const CScript addrName = GetScriptForDestination (pubKey.GetID ()); const CScript newScript = CNameScript::buildNameNew (addrName, hash); valtype rand(20); GetRandBytes (&rand[0], rand.size ()); valtype toHash(rand); toHash.insert (toHash.end (), name.begin (), name.end ()); const uint160 hash = Hash160 (toHash); CCoinControl coinControl; CWalletTx wtx; SendMoneyToScript (pwallet, newScript, nullptr, NAME_LOCKED_AMOUNT, false, wtx, coinControl); keyName.KeepKey (); const std::string randStr = HexStr (rand); const std::string txid = wtx.GetHash ().GetHex (); LogPrintf ("name_new: name=%s, rand=%s, tx=%s\n", nameStr.c_str (), randStr.c_str (), txid.c_str ()); UniValue res(UniValue::VARR); res.push_back (txid); res.push_back (randStr); return res; } /* ************************************************************************** */ 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)) 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" "3. \"value\" (string, required) value for the name\n" "\nResult:\n" "\"txid\" (string) the keva_put's txid\n" "\nExamples:\n" + HelpExampleCli ("keva_put", "\"mynamespace\", \"new-key\", \"new-value\"") ); RPCTypeCheck (request.params, {UniValue::VSTR, UniValue::VSTR, UniValue::VSTR, UniValue::VSTR}); 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 (); }