2014-12-09 02:17:57 +01:00
|
|
|
// Copyright (c) 2010 Satoshi Nakamoto
|
|
|
|
// Copyright (c) 2009-2012 The Bitcoin developers
|
|
|
|
// Distributed under the MIT/X11 software license, see the accompanying
|
|
|
|
// file COPYING or http://www.opensource.org/licenses/mit-license.php.
|
|
|
|
|
|
|
|
#include "main.h"
|
|
|
|
#include "core.h"
|
|
|
|
#include "db.h"
|
|
|
|
#include "init.h"
|
|
|
|
#include "activemasternode.h"
|
2015-02-23 21:01:21 +01:00
|
|
|
#include "masternodeman.h"
|
2014-12-06 20:41:53 +01:00
|
|
|
#include "masternodeconfig.h"
|
2014-12-09 02:17:57 +01:00
|
|
|
#include "rpcserver.h"
|
|
|
|
#include <boost/lexical_cast.hpp>
|
|
|
|
|
|
|
|
#include <fstream>
|
|
|
|
using namespace json_spirit;
|
|
|
|
using namespace std;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
Value darksend(const Array& params, bool fHelp)
|
|
|
|
{
|
|
|
|
if (fHelp || params.size() == 0)
|
|
|
|
throw runtime_error(
|
|
|
|
"darksend <darkcoinaddress> <amount>\n"
|
|
|
|
"darkcoinaddress, reset, or auto (AutoDenominate)"
|
|
|
|
"<amount> is a real and is rounded to the nearest 0.00000001"
|
|
|
|
+ HelpRequiringPassphrase());
|
|
|
|
|
|
|
|
if (pwalletMain->IsLocked())
|
|
|
|
throw JSONRPCError(RPC_WALLET_UNLOCK_NEEDED, "Error: Please enter the wallet passphrase with walletpassphrase first.");
|
|
|
|
|
|
|
|
if(params[0].get_str() == "auto"){
|
2014-12-30 01:09:34 +01:00
|
|
|
if(fMasterNode)
|
|
|
|
return "DarkSend is not supported from masternodes";
|
|
|
|
|
2014-12-09 02:17:57 +01:00
|
|
|
darkSendPool.DoAutomaticDenominating();
|
|
|
|
return "DoAutomaticDenominating";
|
|
|
|
}
|
|
|
|
|
|
|
|
if(params[0].get_str() == "reset"){
|
|
|
|
darkSendPool.SetNull(true);
|
|
|
|
darkSendPool.UnlockCoins();
|
|
|
|
return "successfully reset darksend";
|
|
|
|
}
|
|
|
|
|
|
|
|
if (params.size() != 2)
|
|
|
|
throw runtime_error(
|
|
|
|
"darksend <darkcoinaddress> <amount>\n"
|
|
|
|
"darkcoinaddress, denominate, or auto (AutoDenominate)"
|
|
|
|
"<amount> is a real and is rounded to the nearest 0.00000001"
|
|
|
|
+ HelpRequiringPassphrase());
|
|
|
|
|
|
|
|
CBitcoinAddress address(params[0].get_str());
|
|
|
|
if (!address.IsValid())
|
|
|
|
throw JSONRPCError(RPC_INVALID_ADDRESS_OR_KEY, "Invalid DarkCoin address");
|
|
|
|
|
|
|
|
// Amount
|
|
|
|
int64_t nAmount = AmountFromValue(params[1]);
|
|
|
|
|
|
|
|
// Wallet comments
|
|
|
|
CWalletTx wtx;
|
|
|
|
string strError = pwalletMain->SendMoneyToDestination(address.Get(), nAmount, wtx, ONLY_DENOMINATED);
|
|
|
|
if (strError != "")
|
|
|
|
throw JSONRPCError(RPC_WALLET_ERROR, strError);
|
|
|
|
|
|
|
|
return wtx.GetHash().GetHex();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Value getpoolinfo(const Array& params, bool fHelp)
|
|
|
|
{
|
|
|
|
if (fHelp || params.size() != 0)
|
|
|
|
throw runtime_error(
|
|
|
|
"getpoolinfo\n"
|
|
|
|
"Returns an object containing anonymous pool-related information.");
|
|
|
|
|
|
|
|
Object obj;
|
2015-02-23 21:01:21 +01:00
|
|
|
obj.push_back(Pair("current_masternode", mnodeman.GetCurrentMasterNode()->addr.ToString()));
|
2014-12-09 02:17:57 +01:00
|
|
|
obj.push_back(Pair("state", darkSendPool.GetState()));
|
|
|
|
obj.push_back(Pair("entries", darkSendPool.GetEntriesCount()));
|
|
|
|
obj.push_back(Pair("entries_accepted", darkSendPool.GetCountEntriesAccepted()));
|
|
|
|
return obj;
|
|
|
|
}
|
|
|
|
|
2014-12-06 20:41:53 +01:00
|
|
|
|
2014-12-09 02:17:57 +01:00
|
|
|
Value masternode(const Array& params, bool fHelp)
|
|
|
|
{
|
|
|
|
string strCommand;
|
|
|
|
if (params.size() >= 1)
|
|
|
|
strCommand = params[0].get_str();
|
|
|
|
|
|
|
|
if (fHelp ||
|
2014-12-06 20:41:53 +01:00
|
|
|
(strCommand != "start" && strCommand != "start-alias" && strCommand != "start-many" && strCommand != "stop" && strCommand != "stop-alias" && strCommand != "stop-many" && strCommand != "list" && strCommand != "list-conf" && strCommand != "count" && strCommand != "enforce"
|
2014-12-26 21:00:56 +01:00
|
|
|
&& strCommand != "debug" && strCommand != "current" && strCommand != "winners" && strCommand != "genkey" && strCommand != "connect" && strCommand != "outputs"))
|
2014-12-09 02:17:57 +01:00
|
|
|
throw runtime_error(
|
2015-02-26 02:35:33 +01:00
|
|
|
"masternode \"command\"... ( \"passphrase\" )\n"
|
|
|
|
"Set of commands to execute masternode related actions\n"
|
|
|
|
"\nArguments:\n"
|
|
|
|
"1. \"command\" (string or set of strings, required) The command to execute\n"
|
|
|
|
"2. \"passphrase\" (string, optional) The wallet passphrase\n"
|
|
|
|
"\nAvailable commands:\n"
|
|
|
|
" count - Print number of all known masternodes (optional: 'enabled', 'both')\n"
|
|
|
|
" current - Print info on current masternode winner\n"
|
|
|
|
" debug - Print masternode status\n"
|
|
|
|
" genkey - Generate new masternodeprivkey\n"
|
|
|
|
" enforce - Enforce masternode payments\n"
|
|
|
|
" outputs - Print masternode compatible outputs\n"
|
|
|
|
" start - Start masternode configured in darkcoin.conf\n"
|
|
|
|
" start-alias - Start single masternode by assigned alias configured in masternode.conf\n"
|
|
|
|
" start-many - Start all masternodes configured in masternode.conf\n"
|
|
|
|
" stop - Stop masternode configured in darkcoin.conf\n"
|
|
|
|
" stop-alias - Stop single masternode by assigned alias configured in masternode.conf\n"
|
|
|
|
" stop-many - Stop all masternodes configured in masternode.conf\n"
|
|
|
|
" list - Print list of all known masternodes (see masternodelist for more info)\n"
|
|
|
|
" list-conf - Print masternode.conf in JSON format\n"
|
|
|
|
" winners - Print list of masternode winners\n"
|
|
|
|
);
|
2014-12-09 02:17:57 +01:00
|
|
|
|
|
|
|
if (strCommand == "stop")
|
|
|
|
{
|
|
|
|
if(!fMasterNode) return "you must set masternode=1 in the configuration";
|
|
|
|
|
|
|
|
if(pwalletMain->IsLocked()) {
|
|
|
|
SecureString strWalletPass;
|
|
|
|
strWalletPass.reserve(100);
|
|
|
|
|
|
|
|
if (params.size() == 2){
|
|
|
|
strWalletPass = params[1].get_str().c_str();
|
|
|
|
} else {
|
|
|
|
throw runtime_error(
|
|
|
|
"Your wallet is locked, passphrase is required\n");
|
|
|
|
}
|
|
|
|
|
|
|
|
if(!pwalletMain->Unlock(strWalletPass)){
|
|
|
|
return "incorrect passphrase";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-12-06 20:41:53 +01:00
|
|
|
std::string errorMessage;
|
|
|
|
if(!activeMasternode.StopMasterNode(errorMessage)) {
|
|
|
|
return "stop failed: " + errorMessage;
|
|
|
|
}
|
2014-12-09 02:17:57 +01:00
|
|
|
pwalletMain->Lock();
|
2014-12-26 21:00:56 +01:00
|
|
|
|
2014-12-06 20:41:53 +01:00
|
|
|
if(activeMasternode.status == MASTERNODE_STOPPED) return "successfully stopped masternode";
|
|
|
|
if(activeMasternode.status == MASTERNODE_NOT_CAPABLE) return "not capable masternode";
|
2014-12-26 21:00:56 +01:00
|
|
|
|
2014-12-06 20:41:53 +01:00
|
|
|
return "unknown";
|
|
|
|
}
|
2014-12-09 02:17:57 +01:00
|
|
|
|
2014-12-06 20:41:53 +01:00
|
|
|
if (strCommand == "stop-alias")
|
|
|
|
{
|
|
|
|
if (params.size() < 2){
|
|
|
|
throw runtime_error(
|
|
|
|
"command needs at least 2 parameters\n");
|
|
|
|
}
|
|
|
|
|
|
|
|
std::string alias = params[1].get_str().c_str();
|
|
|
|
|
|
|
|
if(pwalletMain->IsLocked()) {
|
|
|
|
SecureString strWalletPass;
|
|
|
|
strWalletPass.reserve(100);
|
|
|
|
|
|
|
|
if (params.size() == 3){
|
|
|
|
strWalletPass = params[2].get_str().c_str();
|
|
|
|
} else {
|
|
|
|
throw runtime_error(
|
|
|
|
"Your wallet is locked, passphrase is required\n");
|
|
|
|
}
|
|
|
|
|
|
|
|
if(!pwalletMain->Unlock(strWalletPass)){
|
|
|
|
return "incorrect passphrase";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
bool found = false;
|
|
|
|
|
|
|
|
Object statusObj;
|
|
|
|
statusObj.push_back(Pair("alias", alias));
|
|
|
|
|
|
|
|
BOOST_FOREACH(CMasternodeConfig::CMasternodeEntry mne, masternodeConfig.getEntries()) {
|
|
|
|
if(mne.getAlias() == alias) {
|
|
|
|
found = true;
|
|
|
|
std::string errorMessage;
|
|
|
|
bool result = activeMasternode.StopMasterNode(mne.getIp(), mne.getPrivKey(), errorMessage);
|
|
|
|
|
|
|
|
statusObj.push_back(Pair("result", result ? "successful" : "failed"));
|
|
|
|
if(!result) {
|
|
|
|
statusObj.push_back(Pair("errorMessage", errorMessage));
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if(!found) {
|
|
|
|
statusObj.push_back(Pair("result", "failed"));
|
|
|
|
statusObj.push_back(Pair("errorMessage", "could not find alias in config. Verify with list-conf."));
|
|
|
|
}
|
|
|
|
|
|
|
|
pwalletMain->Lock();
|
|
|
|
return statusObj;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (strCommand == "stop-many")
|
|
|
|
{
|
|
|
|
if(pwalletMain->IsLocked()) {
|
|
|
|
SecureString strWalletPass;
|
|
|
|
strWalletPass.reserve(100);
|
|
|
|
|
|
|
|
if (params.size() == 2){
|
|
|
|
strWalletPass = params[1].get_str().c_str();
|
|
|
|
} else {
|
|
|
|
throw runtime_error(
|
|
|
|
"Your wallet is locked, passphrase is required\n");
|
|
|
|
}
|
|
|
|
|
|
|
|
if(!pwalletMain->Unlock(strWalletPass)){
|
|
|
|
return "incorrect passphrase";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
int total = 0;
|
|
|
|
int successful = 0;
|
|
|
|
int fail = 0;
|
|
|
|
|
|
|
|
|
|
|
|
Object resultsObj;
|
|
|
|
|
|
|
|
BOOST_FOREACH(CMasternodeConfig::CMasternodeEntry mne, masternodeConfig.getEntries()) {
|
|
|
|
total++;
|
|
|
|
|
|
|
|
std::string errorMessage;
|
|
|
|
bool result = activeMasternode.StopMasterNode(mne.getIp(), mne.getPrivKey(), errorMessage);
|
|
|
|
|
|
|
|
Object statusObj;
|
|
|
|
statusObj.push_back(Pair("alias", mne.getAlias()));
|
|
|
|
statusObj.push_back(Pair("result", result ? "successful" : "failed"));
|
|
|
|
|
|
|
|
if(result) {
|
|
|
|
successful++;
|
|
|
|
} else {
|
|
|
|
fail++;
|
|
|
|
statusObj.push_back(Pair("errorMessage", errorMessage));
|
|
|
|
}
|
|
|
|
|
|
|
|
resultsObj.push_back(Pair("status", statusObj));
|
|
|
|
}
|
|
|
|
pwalletMain->Lock();
|
|
|
|
|
|
|
|
Object returnObj;
|
|
|
|
returnObj.push_back(Pair("overall", "Successfully stopped " + boost::lexical_cast<std::string>(successful) + " masternodes, failed to stop " +
|
|
|
|
boost::lexical_cast<std::string>(fail) + ", total " + boost::lexical_cast<std::string>(total)));
|
|
|
|
returnObj.push_back(Pair("detail", resultsObj));
|
|
|
|
|
|
|
|
return returnObj;
|
2014-12-09 02:17:57 +01:00
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
if (strCommand == "list")
|
|
|
|
{
|
2015-02-23 21:26:23 +01:00
|
|
|
Array newParams(params.size() - 1);
|
|
|
|
std::copy(params.begin() + 1, params.end(), newParams.begin());
|
|
|
|
return masternodelist(newParams, fHelp);
|
|
|
|
}
|
2014-12-09 02:17:57 +01:00
|
|
|
|
2015-02-23 21:26:23 +01:00
|
|
|
if (strCommand == "count")
|
|
|
|
{
|
|
|
|
if (params.size() > 2){
|
2014-12-09 02:17:57 +01:00
|
|
|
throw runtime_error(
|
2015-02-23 21:26:23 +01:00
|
|
|
"too many parameters\n");
|
2014-12-09 02:17:57 +01:00
|
|
|
}
|
2015-02-26 02:35:33 +01:00
|
|
|
if (params.size() == 2)
|
|
|
|
{
|
|
|
|
if(params[1] == "enabled") return mnodeman.CountEnabled();
|
|
|
|
if(params[1] == "both") return boost::lexical_cast<std::string>(mnodeman.CountEnabled()) + " / " + boost::lexical_cast<std::string>(mnodeman.size());
|
|
|
|
}
|
2015-02-23 21:26:23 +01:00
|
|
|
return mnodeman.size();
|
2014-12-09 02:17:57 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
if (strCommand == "start")
|
|
|
|
{
|
|
|
|
if(!fMasterNode) return "you must set masternode=1 in the configuration";
|
|
|
|
|
|
|
|
if(pwalletMain->IsLocked()) {
|
|
|
|
SecureString strWalletPass;
|
|
|
|
strWalletPass.reserve(100);
|
|
|
|
|
|
|
|
if (params.size() == 2){
|
|
|
|
strWalletPass = params[1].get_str().c_str();
|
|
|
|
} else {
|
|
|
|
throw runtime_error(
|
|
|
|
"Your wallet is locked, passphrase is required\n");
|
|
|
|
}
|
|
|
|
|
|
|
|
if(!pwalletMain->Unlock(strWalletPass)){
|
|
|
|
return "incorrect passphrase";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-01-22 00:12:04 +01:00
|
|
|
if(activeMasternode.status != MASTERNODE_REMOTELY_ENABLED && activeMasternode.status != MASTERNODE_IS_CAPABLE){
|
2015-01-21 22:57:02 +01:00
|
|
|
activeMasternode.status = MASTERNODE_NOT_PROCESSED; // TODO: consider better way
|
|
|
|
std::string errorMessage;
|
|
|
|
activeMasternode.ManageStatus();
|
|
|
|
pwalletMain->Lock();
|
|
|
|
}
|
2014-12-26 21:00:56 +01:00
|
|
|
|
2014-12-06 20:41:53 +01:00
|
|
|
if(activeMasternode.status == MASTERNODE_REMOTELY_ENABLED) return "masternode started remotely";
|
|
|
|
if(activeMasternode.status == MASTERNODE_INPUT_TOO_NEW) return "masternode input must have at least 15 confirmations";
|
|
|
|
if(activeMasternode.status == MASTERNODE_STOPPED) return "masternode is stopped";
|
|
|
|
if(activeMasternode.status == MASTERNODE_IS_CAPABLE) return "successfully started masternode";
|
|
|
|
if(activeMasternode.status == MASTERNODE_NOT_CAPABLE) return "not capable masternode: " + activeMasternode.notCapableReason;
|
|
|
|
if(activeMasternode.status == MASTERNODE_SYNC_IN_PROCESS) return "sync in process. Must wait until client is synced to start.";
|
2014-12-09 02:17:57 +01:00
|
|
|
|
|
|
|
return "unknown";
|
|
|
|
}
|
|
|
|
|
2014-12-06 20:41:53 +01:00
|
|
|
if (strCommand == "start-alias")
|
2014-12-09 02:17:57 +01:00
|
|
|
{
|
2014-12-06 20:41:53 +01:00
|
|
|
if (params.size() < 2){
|
|
|
|
throw runtime_error(
|
|
|
|
"command needs at least 2 parameters\n");
|
|
|
|
}
|
|
|
|
|
|
|
|
std::string alias = params[1].get_str().c_str();
|
|
|
|
|
|
|
|
if(pwalletMain->IsLocked()) {
|
|
|
|
SecureString strWalletPass;
|
|
|
|
strWalletPass.reserve(100);
|
|
|
|
|
|
|
|
if (params.size() == 3){
|
|
|
|
strWalletPass = params[2].get_str().c_str();
|
|
|
|
} else {
|
|
|
|
throw runtime_error(
|
|
|
|
"Your wallet is locked, passphrase is required\n");
|
|
|
|
}
|
|
|
|
|
|
|
|
if(!pwalletMain->Unlock(strWalletPass)){
|
|
|
|
return "incorrect passphrase";
|
|
|
|
}
|
2014-12-09 02:17:57 +01:00
|
|
|
}
|
|
|
|
|
2014-12-06 20:41:53 +01:00
|
|
|
bool found = false;
|
|
|
|
|
|
|
|
Object statusObj;
|
|
|
|
statusObj.push_back(Pair("alias", alias));
|
|
|
|
|
|
|
|
BOOST_FOREACH(CMasternodeConfig::CMasternodeEntry mne, masternodeConfig.getEntries()) {
|
|
|
|
if(mne.getAlias() == alias) {
|
|
|
|
found = true;
|
|
|
|
std::string errorMessage;
|
|
|
|
bool result = activeMasternode.Register(mne.getIp(), mne.getPrivKey(), mne.getTxHash(), mne.getOutputIndex(), errorMessage);
|
|
|
|
|
|
|
|
statusObj.push_back(Pair("result", result ? "successful" : "failed"));
|
|
|
|
if(!result) {
|
|
|
|
statusObj.push_back(Pair("errorMessage", errorMessage));
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if(!found) {
|
|
|
|
statusObj.push_back(Pair("result", "failed"));
|
|
|
|
statusObj.push_back(Pair("errorMessage", "could not find alias in config. Verify with list-conf."));
|
|
|
|
}
|
|
|
|
|
|
|
|
pwalletMain->Lock();
|
|
|
|
return statusObj;
|
2014-12-09 02:17:57 +01:00
|
|
|
|
|
|
|
}
|
2014-12-26 21:00:56 +01:00
|
|
|
|
2014-12-06 20:41:53 +01:00
|
|
|
if (strCommand == "start-many")
|
2014-12-26 21:00:56 +01:00
|
|
|
{
|
2014-12-06 20:41:53 +01:00
|
|
|
if(pwalletMain->IsLocked()) {
|
|
|
|
SecureString strWalletPass;
|
|
|
|
strWalletPass.reserve(100);
|
|
|
|
|
|
|
|
if (params.size() == 2){
|
|
|
|
strWalletPass = params[1].get_str().c_str();
|
|
|
|
} else {
|
|
|
|
throw runtime_error(
|
|
|
|
"Your wallet is locked, passphrase is required\n");
|
|
|
|
}
|
|
|
|
|
|
|
|
if(!pwalletMain->Unlock(strWalletPass)){
|
|
|
|
return "incorrect passphrase";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
std::vector<CMasternodeConfig::CMasternodeEntry> mnEntries;
|
|
|
|
mnEntries = masternodeConfig.getEntries();
|
|
|
|
|
|
|
|
int total = 0;
|
|
|
|
int successful = 0;
|
|
|
|
int fail = 0;
|
|
|
|
|
|
|
|
Object resultsObj;
|
|
|
|
|
|
|
|
BOOST_FOREACH(CMasternodeConfig::CMasternodeEntry mne, masternodeConfig.getEntries()) {
|
|
|
|
total++;
|
|
|
|
|
|
|
|
std::string errorMessage;
|
|
|
|
bool result = activeMasternode.Register(mne.getIp(), mne.getPrivKey(), mne.getTxHash(), mne.getOutputIndex(), errorMessage);
|
|
|
|
|
|
|
|
Object statusObj;
|
|
|
|
statusObj.push_back(Pair("alias", mne.getAlias()));
|
2015-02-17 11:27:47 +01:00
|
|
|
statusObj.push_back(Pair("result", result ? "successful" : "failed"));
|
2014-12-06 20:41:53 +01:00
|
|
|
|
|
|
|
if(result) {
|
|
|
|
successful++;
|
|
|
|
} else {
|
|
|
|
fail++;
|
|
|
|
statusObj.push_back(Pair("errorMessage", errorMessage));
|
|
|
|
}
|
|
|
|
|
|
|
|
resultsObj.push_back(Pair("status", statusObj));
|
|
|
|
}
|
|
|
|
pwalletMain->Lock();
|
|
|
|
|
|
|
|
Object returnObj;
|
|
|
|
returnObj.push_back(Pair("overall", "Successfully started " + boost::lexical_cast<std::string>(successful) + " masternodes, failed to start " +
|
|
|
|
boost::lexical_cast<std::string>(fail) + ", total " + boost::lexical_cast<std::string>(total)));
|
|
|
|
returnObj.push_back(Pair("detail", resultsObj));
|
|
|
|
|
|
|
|
return returnObj;
|
|
|
|
}
|
2014-12-09 02:17:57 +01:00
|
|
|
|
|
|
|
if (strCommand == "debug")
|
|
|
|
{
|
2014-12-06 20:41:53 +01:00
|
|
|
if(activeMasternode.status == MASTERNODE_REMOTELY_ENABLED) return "masternode started remotely";
|
|
|
|
if(activeMasternode.status == MASTERNODE_INPUT_TOO_NEW) return "masternode input must have at least 15 confirmations";
|
|
|
|
if(activeMasternode.status == MASTERNODE_IS_CAPABLE) return "successfully started masternode";
|
|
|
|
if(activeMasternode.status == MASTERNODE_STOPPED) return "masternode is stopped";
|
|
|
|
if(activeMasternode.status == MASTERNODE_NOT_CAPABLE) return "not capable masternode: " + activeMasternode.notCapableReason;
|
|
|
|
if(activeMasternode.status == MASTERNODE_SYNC_IN_PROCESS) return "sync in process. Must wait until client is synced to start.";
|
2014-12-09 02:17:57 +01:00
|
|
|
|
|
|
|
CTxIn vin = CTxIn();
|
|
|
|
CPubKey pubkey = CScript();
|
|
|
|
CKey key;
|
|
|
|
bool found = activeMasternode.GetMasterNodeVin(vin, pubkey, key);
|
|
|
|
if(!found){
|
|
|
|
return "Missing masternode input, please look at the documentation for instructions on masternode creation";
|
|
|
|
} else {
|
|
|
|
return "No problems were found";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (strCommand == "create")
|
|
|
|
{
|
|
|
|
|
|
|
|
return "Not implemented yet, please look at the documentation for instructions on masternode creation";
|
|
|
|
}
|
|
|
|
|
|
|
|
if (strCommand == "current")
|
|
|
|
{
|
2015-02-23 21:26:23 +01:00
|
|
|
CMasternode* winner = mnodeman.GetCurrentMasterNode(1);
|
|
|
|
if(winner) {
|
2015-02-26 02:35:33 +01:00
|
|
|
Object obj;
|
|
|
|
CScript pubkey;
|
|
|
|
pubkey.SetDestination(winner->pubkey.GetID());
|
|
|
|
CTxDestination address1;
|
|
|
|
ExtractDestination(pubkey, address1);
|
|
|
|
CBitcoinAddress address2(address1);
|
|
|
|
|
|
|
|
obj.push_back(Pair("IP:port", winner->addr.ToString().c_str()));
|
|
|
|
obj.push_back(Pair("protocol", (int64_t)winner->protocolVersion));
|
|
|
|
obj.push_back(Pair("vin", winner->vin.prevout.hash.ToString().c_str()));
|
|
|
|
obj.push_back(Pair("pubkey", address2.ToString().c_str()));
|
|
|
|
obj.push_back(Pair("lastseen", (int64_t)winner->lastTimeSeen));
|
2015-03-01 16:38:53 +01:00
|
|
|
obj.push_back(Pair("activeseconds", (int64_t)(winner->lastTimeSeen - winner->sigTime)));
|
2015-02-26 02:35:33 +01:00
|
|
|
return obj;
|
2014-12-09 02:17:57 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
return "unknown";
|
|
|
|
}
|
|
|
|
|
|
|
|
if (strCommand == "genkey")
|
|
|
|
{
|
|
|
|
CKey secret;
|
|
|
|
secret.MakeNewKey(false);
|
|
|
|
|
|
|
|
return CBitcoinSecret(secret).ToString();
|
|
|
|
}
|
|
|
|
|
|
|
|
if (strCommand == "winners")
|
|
|
|
{
|
|
|
|
Object obj;
|
|
|
|
|
|
|
|
for(int nHeight = chainActive.Tip()->nHeight-10; nHeight < chainActive.Tip()->nHeight+20; nHeight++)
|
|
|
|
{
|
|
|
|
CScript payee;
|
|
|
|
if(masternodePayments.GetBlockPayee(nHeight, payee)){
|
|
|
|
CTxDestination address1;
|
|
|
|
ExtractDestination(payee, address1);
|
|
|
|
CBitcoinAddress address2(address1);
|
|
|
|
obj.push_back(Pair(boost::lexical_cast<std::string>(nHeight), address2.ToString().c_str()));
|
|
|
|
} else {
|
|
|
|
obj.push_back(Pair(boost::lexical_cast<std::string>(nHeight), ""));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return obj;
|
|
|
|
}
|
|
|
|
|
|
|
|
if(strCommand == "enforce")
|
|
|
|
{
|
|
|
|
return (uint64_t)enforceMasternodePaymentsTime;
|
|
|
|
}
|
|
|
|
|
|
|
|
if(strCommand == "connect")
|
|
|
|
{
|
|
|
|
std::string strAddress = "";
|
|
|
|
if (params.size() == 2){
|
|
|
|
strAddress = params[1].get_str().c_str();
|
|
|
|
} else {
|
|
|
|
throw runtime_error(
|
|
|
|
"Masternode address required\n");
|
|
|
|
}
|
|
|
|
|
|
|
|
CService addr = CService(strAddress);
|
|
|
|
|
|
|
|
if(ConnectNode((CAddress)addr, NULL, true)){
|
|
|
|
return "successfully connected";
|
|
|
|
} else {
|
|
|
|
return "error connecting";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-12-06 20:41:53 +01:00
|
|
|
if(strCommand == "list-conf")
|
|
|
|
{
|
|
|
|
std::vector<CMasternodeConfig::CMasternodeEntry> mnEntries;
|
|
|
|
mnEntries = masternodeConfig.getEntries();
|
|
|
|
|
|
|
|
Object resultObj;
|
|
|
|
|
|
|
|
BOOST_FOREACH(CMasternodeConfig::CMasternodeEntry mne, masternodeConfig.getEntries()) {
|
|
|
|
Object mnObj;
|
|
|
|
mnObj.push_back(Pair("alias", mne.getAlias()));
|
|
|
|
mnObj.push_back(Pair("address", mne.getIp()));
|
|
|
|
mnObj.push_back(Pair("privateKey", mne.getPrivKey()));
|
|
|
|
mnObj.push_back(Pair("txHash", mne.getTxHash()));
|
|
|
|
mnObj.push_back(Pair("outputIndex", mne.getOutputIndex()));
|
|
|
|
resultObj.push_back(Pair("masternode", mnObj));
|
|
|
|
}
|
|
|
|
|
|
|
|
return resultObj;
|
|
|
|
}
|
|
|
|
|
2014-12-26 21:00:56 +01:00
|
|
|
if (strCommand == "outputs"){
|
|
|
|
// Find possible candidates
|
|
|
|
vector<COutput> possibleCoins = activeMasternode.SelectCoinsMasternode();
|
|
|
|
|
|
|
|
Object obj;
|
|
|
|
BOOST_FOREACH(COutput& out, possibleCoins) {
|
|
|
|
obj.push_back(Pair(out.tx->GetHash().ToString().c_str(), boost::lexical_cast<std::string>(out.i)));
|
|
|
|
}
|
|
|
|
|
|
|
|
return obj;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2014-12-09 02:17:57 +01:00
|
|
|
return Value::null;
|
|
|
|
}
|
|
|
|
|
2015-02-23 21:26:23 +01:00
|
|
|
Value masternodelist(const Array& params, bool fHelp)
|
|
|
|
{
|
|
|
|
std::string strMode = "active";
|
|
|
|
std::string strFilter = "";
|
|
|
|
|
|
|
|
if (params.size() >= 1) strMode = params[0].get_str();
|
|
|
|
if (params.size() == 2) strFilter = params[1].get_str();
|
|
|
|
|
|
|
|
if (fHelp ||
|
|
|
|
(strMode != "active" && strMode != "vin" && strMode != "pubkey" && strMode != "lastseen"
|
|
|
|
&& strMode != "activeseconds" && strMode != "rank" && strMode != "protocol" && strMode != "full"))
|
|
|
|
{
|
|
|
|
throw runtime_error(
|
|
|
|
"masternodelist ( \"mode\" \"filter\" )\n"
|
|
|
|
"Get a list of masternodes in different modes\n"
|
|
|
|
"\nArguments:\n"
|
2015-03-01 18:39:18 +01:00
|
|
|
"1. \"mode\" (string, optional/required to use filter, defaults = active) The mode to run list in\n"
|
|
|
|
"2. \"filter\" (string, optional) Filter results. Partial match by IP by default in all modes, additional matches in some modes\n"
|
2015-02-26 02:35:33 +01:00
|
|
|
"\nAvailable modes:\n"
|
2015-03-01 18:39:18 +01:00
|
|
|
" active - Print '1' if active and '0' otherwise (can be additionally filtered by 'true' (active only) / 'false' (non-active only))\n"
|
2015-02-23 21:26:23 +01:00
|
|
|
" activeseconds - Print number of seconds masternode recognized by the network as enabled\n"
|
2015-03-01 18:39:18 +01:00
|
|
|
" full - Print info in format 'active | protocol | pubkey | vin | lastseen | activeseconds' (can be additionally filtered, partial match)\n"
|
2015-02-23 21:26:23 +01:00
|
|
|
" lastseen - Print timestamp of when a masternode was last seen on the network\n"
|
2015-03-01 18:39:18 +01:00
|
|
|
" protocol - Print protocol of a masternode (can be additionally filtered, exact match))\n"
|
|
|
|
" pubkey - Print public key associated with a masternode (can be additionally filtered, partial match)\n"
|
2015-02-23 21:26:23 +01:00
|
|
|
" rank - Print rank of a masternode based on current block\n"
|
2015-03-01 18:39:18 +01:00
|
|
|
" vin - Print vin associated with a masternode (can be additionally filtered, partial match)\n"
|
2015-02-23 21:26:23 +01:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2015-02-24 00:48:11 +01:00
|
|
|
Object obj;
|
|
|
|
std::vector<CMasternode> vMasternodes = mnodeman.GetFullMasternodeVector();
|
|
|
|
BOOST_FOREACH(CMasternode& mn, vMasternodes) {
|
|
|
|
|
2015-03-01 20:39:03 +01:00
|
|
|
std::ostringstream addrStream;
|
|
|
|
addrStream << setw(21) << mn.addr.ToString().c_str();
|
|
|
|
std::string strAddr = addrStream.str();
|
2015-02-24 00:48:11 +01:00
|
|
|
if(strMode == "active"){
|
2015-03-01 18:39:18 +01:00
|
|
|
if(strFilter !="" && strFilter != (mn.IsEnabled() ? "true" : "false") &&
|
2015-02-26 15:33:27 +01:00
|
|
|
mn.addr.ToString().find(strFilter) == string::npos) continue;
|
2015-02-24 00:48:11 +01:00
|
|
|
obj.push_back(Pair(strAddr, (int)mn.IsEnabled()));
|
|
|
|
} else if (strMode == "activeseconds") {
|
2015-02-26 15:33:27 +01:00
|
|
|
if(strFilter !="" && mn.addr.ToString().find(strFilter) == string::npos) continue;
|
2015-03-01 16:38:53 +01:00
|
|
|
obj.push_back(Pair(strAddr, (int64_t)(mn.lastTimeSeen - mn.sigTime)));
|
2015-02-24 00:48:11 +01:00
|
|
|
} else if (strMode == "full") {
|
|
|
|
CScript pubkey;
|
|
|
|
pubkey.SetDestination(mn.pubkey.GetID());
|
|
|
|
CTxDestination address1;
|
|
|
|
ExtractDestination(pubkey, address1);
|
|
|
|
CBitcoinAddress address2(address1);
|
|
|
|
|
|
|
|
std::ostringstream stringStream;
|
|
|
|
stringStream << (mn.IsEnabled() ? "1" : "0") << " | " <<
|
|
|
|
mn.protocolVersion << " | " <<
|
|
|
|
address2.ToString() << " | " <<
|
|
|
|
mn.vin.prevout.hash.ToString() << " | " <<
|
2015-03-01 20:39:03 +01:00
|
|
|
mn.lastTimeSeen << " | " << setw(8) <<
|
2015-03-01 16:38:53 +01:00
|
|
|
(mn.lastTimeSeen - mn.sigTime);
|
2015-02-24 00:48:11 +01:00
|
|
|
std::string output = stringStream.str();
|
|
|
|
stringStream << " " << strAddr;
|
2015-03-01 18:39:18 +01:00
|
|
|
if(strFilter !="" && stringStream.str().find(strFilter) == string::npos &&
|
|
|
|
mn.addr.ToString().find(strFilter) == string::npos) continue;
|
2015-02-24 00:48:11 +01:00
|
|
|
obj.push_back(Pair(strAddr, output));
|
2015-03-01 18:39:18 +01:00
|
|
|
} else if (strMode == "lastseen") {
|
|
|
|
if(strFilter !="" && mn.addr.ToString().find(strFilter) == string::npos) continue;
|
|
|
|
obj.push_back(Pair(strAddr, (int64_t)mn.lastTimeSeen));
|
|
|
|
} else if (strMode == "protocol") {
|
|
|
|
if(strFilter !="" && strFilter != boost::lexical_cast<std::string>(mn.protocolVersion) &&
|
|
|
|
mn.addr.ToString().find(strFilter) == string::npos) continue;
|
|
|
|
obj.push_back(Pair(strAddr, (int64_t)mn.protocolVersion));
|
|
|
|
} else if (strMode == "pubkey") {
|
|
|
|
CScript pubkey;
|
|
|
|
pubkey.SetDestination(mn.pubkey.GetID());
|
|
|
|
CTxDestination address1;
|
|
|
|
ExtractDestination(pubkey, address1);
|
|
|
|
CBitcoinAddress address2(address1);
|
|
|
|
|
|
|
|
if(strFilter !="" && address2.ToString().find(strFilter) == string::npos &&
|
|
|
|
mn.addr.ToString().find(strFilter) == string::npos) continue;
|
|
|
|
obj.push_back(Pair(strAddr, address2.ToString().c_str()));
|
|
|
|
} else if (strMode == "rank") {
|
|
|
|
if(strFilter !="" && mn.addr.ToString().find(strFilter) == string::npos) continue;
|
|
|
|
obj.push_back(Pair(strAddr, (int)(mnodeman.GetMasternodeRank(mn.vin, chainActive.Tip()->nHeight))));
|
|
|
|
} else if (strMode == "vin") {
|
|
|
|
if(strFilter !="" && mn.vin.prevout.hash.ToString().find(strFilter) == string::npos &&
|
|
|
|
mn.addr.ToString().find(strFilter) == string::npos) continue;
|
|
|
|
obj.push_back(Pair(strAddr, mn.vin.prevout.hash.ToString().c_str()));
|
2015-02-24 00:48:11 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return obj;
|
|
|
|
|
2015-02-23 21:26:23 +01:00
|
|
|
}
|