2017-11-09 19:34:51 +01:00
|
|
|
/*
|
|
|
|
|
* This file is part of the Flowee project
|
|
|
|
|
* Copyright (c) 2009-2010 Satoshi Nakamoto
|
|
|
|
|
* Copyright (c) 2009-2015 The Bitcoin Core developers
|
2021-06-20 22:44:44 +02:00
|
|
|
* Copyright (C) 2017,2019 Tom Zander <tom@flowee.org>
|
2017-11-09 19:34:51 +01:00
|
|
|
*
|
|
|
|
|
* This program is free software: you can redistribute it and/or modify
|
|
|
|
|
* it under the terms of the GNU General Public License as published by
|
|
|
|
|
* the Free Software Foundation, either version 3 of the License, or
|
|
|
|
|
* (at your option) any later version.
|
|
|
|
|
*
|
|
|
|
|
* This program is distributed in the hope that it will be useful,
|
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
|
* GNU General Public License for more details.
|
|
|
|
|
*
|
|
|
|
|
* You should have received a copy of the GNU General Public License
|
|
|
|
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
|
*/
|
2013-10-11 23:09:59 +02:00
|
|
|
|
2014-09-14 12:43:56 +02:00
|
|
|
#include "chainparamsbase.h"
|
2014-10-28 21:33:23 -04:00
|
|
|
#include "clientversion.h"
|
2013-11-20 14:18:57 +01:00
|
|
|
#include "rpcclient.h"
|
2019-11-13 11:46:09 +01:00
|
|
|
#include <server/util.h>
|
2014-02-24 14:08:56 +01:00
|
|
|
#include "rpcprotocol.h"
|
2019-08-24 13:20:37 +02:00
|
|
|
#include <serverutil.h>
|
2014-08-21 16:11:09 +02:00
|
|
|
#include "utilstrencodings.h"
|
2026-05-14 13:13:40 +02:00
|
|
|
#include "utiltime.h"
|
2013-10-11 23:09:59 +02:00
|
|
|
|
2013-04-13 00:13:08 -05:00
|
|
|
#include <boost/filesystem/operations.hpp>
|
2017-08-17 20:53:23 -06:00
|
|
|
#include <cstdio>
|
2015-01-23 07:53:17 +01:00
|
|
|
|
|
|
|
|
#include <event2/event.h>
|
|
|
|
|
#include <event2/http.h>
|
|
|
|
|
#include <event2/buffer.h>
|
|
|
|
|
#include <event2/keyvalq_struct.h>
|
2013-04-13 00:13:08 -05:00
|
|
|
|
2015-09-04 16:11:34 +02:00
|
|
|
#include <univalue.h>
|
2015-05-18 14:02:18 +02:00
|
|
|
|
2013-10-11 23:09:59 +02:00
|
|
|
//////////////////////////////////////////////////////////////////////////////
|
|
|
|
|
//
|
|
|
|
|
// Start
|
|
|
|
|
//
|
2014-10-29 18:08:31 +01:00
|
|
|
|
|
|
|
|
//
|
|
|
|
|
// Exception thrown on connection error. This error is used to determine
|
|
|
|
|
// when to wait if -rpcwait is given.
|
|
|
|
|
//
|
|
|
|
|
class CConnectionFailed : public std::runtime_error
|
|
|
|
|
{
|
|
|
|
|
public:
|
|
|
|
|
|
|
|
|
|
explicit inline CConnectionFailed(const std::string& msg) :
|
|
|
|
|
std::runtime_error(msg)
|
|
|
|
|
{}
|
|
|
|
|
|
|
|
|
|
};
|
|
|
|
|
|
2013-10-11 23:09:59 +02:00
|
|
|
static bool AppInitRPC(int argc, char* argv[])
|
|
|
|
|
{
|
|
|
|
|
//
|
|
|
|
|
// Parameters
|
|
|
|
|
//
|
2019-04-02 17:45:04 +02:00
|
|
|
Settings::HubCli allowedArgs;
|
2017-01-23 01:56:55 -08:00
|
|
|
try {
|
2017-02-14 15:07:49 -08:00
|
|
|
ParseParameters(argc, argv, allowedArgs);
|
2017-01-23 01:56:55 -08:00
|
|
|
} catch (const std::exception& e) {
|
|
|
|
|
fprintf(stderr, "Error parsing program options: %s\n", e.what());
|
|
|
|
|
return false;
|
|
|
|
|
}
|
2015-10-18 21:02:36 +11:00
|
|
|
if (argc<2 || mapArgs.count("-?") || mapArgs.count("-h") || mapArgs.count("-help") || mapArgs.count("-version")) {
|
2017-11-13 23:09:33 +01:00
|
|
|
std::string strUsage = _("Flowee RPC client version") + " " + FormatFullVersion() + "\n";
|
2014-11-22 19:56:25 +01:00
|
|
|
if (!mapArgs.count("-version")) {
|
|
|
|
|
strUsage += "\n" + _("Usage:") + "\n" +
|
2017-11-13 23:09:33 +01:00
|
|
|
" bitcoin-cli [options] <command> [params] " + _("Send command to the hub") + "\n" +
|
2014-11-22 19:56:25 +01:00
|
|
|
" bitcoin-cli [options] help " + _("List commands") + "\n" +
|
|
|
|
|
" bitcoin-cli [options] help <command> " + _("Get help for a command") + "\n";
|
|
|
|
|
|
2017-02-14 17:34:40 -08:00
|
|
|
strUsage += "\n" + allowedArgs.helpMessage();
|
2014-11-22 19:56:25 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
fprintf(stdout, "%s", strUsage.c_str());
|
|
|
|
|
return false;
|
|
|
|
|
}
|
2014-06-25 21:09:36 -04:00
|
|
|
if (!boost::filesystem::is_directory(GetDataDir(false))) {
|
2013-10-11 23:09:59 +02:00
|
|
|
fprintf(stderr, "Error: Specified data directory \"%s\" does not exist.\n", mapArgs["-datadir"].c_str());
|
|
|
|
|
return false;
|
|
|
|
|
}
|
2015-05-25 09:00:17 +02:00
|
|
|
try {
|
|
|
|
|
SelectBaseParams(ChainNameFromCommandLine());
|
2015-10-27 17:39:42 +01:00
|
|
|
} catch (const std::exception& e) {
|
2015-05-25 09:00:17 +02:00
|
|
|
fprintf(stderr, "Error: %s\n", e.what());
|
2013-11-28 17:28:27 +01:00
|
|
|
return false;
|
|
|
|
|
}
|
2018-10-06 21:08:22 +02:00
|
|
|
try {
|
|
|
|
|
ReadConfigFile(mapArgs, mapMultiArgs);
|
|
|
|
|
} catch (const std::exception& e) {
|
|
|
|
|
fprintf(stderr,"Error reading configuration file: %s\n", e.what());
|
|
|
|
|
return false;
|
|
|
|
|
}
|
2013-10-11 23:09:59 +02:00
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
2015-01-23 07:53:17 +01:00
|
|
|
|
|
|
|
|
/** Reply structure for request_done to fill in */
|
|
|
|
|
struct HTTPReply
|
|
|
|
|
{
|
|
|
|
|
int status;
|
|
|
|
|
std::string body;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
static void http_request_done(struct evhttp_request *req, void *ctx)
|
|
|
|
|
{
|
|
|
|
|
HTTPReply *reply = static_cast<HTTPReply*>(ctx);
|
|
|
|
|
|
|
|
|
|
if (req == NULL) {
|
|
|
|
|
/* If req is NULL, it means an error occurred while connecting, but
|
|
|
|
|
* I'm not sure how to find out which one. We also don't really care.
|
|
|
|
|
*/
|
|
|
|
|
reply->status = 0;
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
reply->status = evhttp_request_get_response_code(req);
|
|
|
|
|
|
|
|
|
|
struct evbuffer *buf = evhttp_request_get_input_buffer(req);
|
|
|
|
|
if (buf)
|
|
|
|
|
{
|
|
|
|
|
size_t size = evbuffer_get_length(buf);
|
|
|
|
|
const char *data = (const char*)evbuffer_pullup(buf, size);
|
|
|
|
|
if (data)
|
|
|
|
|
reply->body = std::string(data, size);
|
|
|
|
|
evbuffer_drain(buf, size);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2017-06-18 16:55:32 -04:00
|
|
|
UniValue CallRPC(const std::string& strMethod, const UniValue& params)
|
2014-05-26 11:38:44 +02:00
|
|
|
{
|
2015-06-27 19:21:41 +00:00
|
|
|
std::string host = GetArg("-rpcconnect", DEFAULT_RPCCONNECT);
|
2015-01-23 07:53:17 +01:00
|
|
|
int port = GetArg("-rpcport", BaseParams().RPCPort());
|
2014-05-26 11:38:44 +02:00
|
|
|
|
2015-01-23 07:53:17 +01:00
|
|
|
// Create event base
|
|
|
|
|
struct event_base *base = event_base_new(); // TODO RAII
|
|
|
|
|
if (!base)
|
2017-06-18 16:59:29 -04:00
|
|
|
throw std::runtime_error("cannot create event_base");
|
2014-05-26 11:38:44 +02:00
|
|
|
|
2015-01-23 07:53:17 +01:00
|
|
|
// Synchronously look up hostname
|
|
|
|
|
struct evhttp_connection *evcon = evhttp_connection_base_new(base, NULL, host.c_str(), port); // TODO RAII
|
|
|
|
|
if (evcon == NULL)
|
2017-06-18 16:59:29 -04:00
|
|
|
throw std::runtime_error("create connection failed");
|
2015-09-18 15:45:38 +02:00
|
|
|
evhttp_connection_set_timeout(evcon, GetArg("-rpcclienttimeout", DEFAULT_HTTP_CLIENT_TIMEOUT));
|
2015-01-23 07:53:17 +01:00
|
|
|
|
|
|
|
|
HTTPReply response;
|
|
|
|
|
struct evhttp_request *req = evhttp_request_new(http_request_done, (void*)&response); // TODO RAII
|
|
|
|
|
if (req == NULL)
|
2017-06-18 16:59:29 -04:00
|
|
|
throw std::runtime_error("create http request failed");
|
2015-01-23 07:53:17 +01:00
|
|
|
|
|
|
|
|
// Get credentials
|
2015-07-07 14:53:48 +02:00
|
|
|
std::string strRPCUserColonPass;
|
|
|
|
|
if (mapArgs["-rpcpassword"] == "") {
|
|
|
|
|
// Try fall back to cookie-based authentication if no password is provided
|
2017-01-03 18:51:07 +01:00
|
|
|
boost::filesystem::path path = GetAuthCookieFile();
|
|
|
|
|
bool found = GetAuthCookie(path, &strRPCUserColonPass) ;
|
|
|
|
|
|
|
|
|
|
#ifndef WIN32
|
|
|
|
|
if (!found) // The default place where cookies are stored in our packaging standard
|
2018-03-08 15:31:28 +01:00
|
|
|
found = GetAuthCookie("/etc/flowee/.cookie", &strRPCUserColonPass) ;
|
2017-01-03 18:51:07 +01:00
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
if (!found) {
|
2017-06-18 16:59:29 -04:00
|
|
|
throw std::runtime_error(strprintf(
|
2015-01-23 07:53:17 +01:00
|
|
|
_("Could not locate RPC credentials. No authentication cookie could be found, and no rpcpassword is set in the configuration file (%s)"),
|
2015-07-07 14:53:48 +02:00
|
|
|
GetConfigFile().string().c_str()));
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
} else {
|
2019-11-15 19:14:34 +01:00
|
|
|
std::string user = mapArgs["-rpcuser"];
|
|
|
|
|
if (user.empty())
|
|
|
|
|
user = "__cookie__";
|
|
|
|
|
strRPCUserColonPass = user + ":" + mapArgs["-rpcpassword"];
|
2015-07-07 14:53:48 +02:00
|
|
|
}
|
|
|
|
|
|
2015-01-23 07:53:17 +01:00
|
|
|
struct evkeyvalq *output_headers = evhttp_request_get_output_headers(req);
|
|
|
|
|
assert(output_headers);
|
|
|
|
|
evhttp_add_header(output_headers, "Host", host.c_str());
|
|
|
|
|
evhttp_add_header(output_headers, "Connection", "close");
|
|
|
|
|
evhttp_add_header(output_headers, "Authorization", (std::string("Basic ") + EncodeBase64(strRPCUserColonPass)).c_str());
|
2014-05-26 11:38:44 +02:00
|
|
|
|
2015-01-23 07:53:17 +01:00
|
|
|
// Attach request data
|
|
|
|
|
std::string strRequest = JSONRPCRequest(strMethod, params, 1);
|
|
|
|
|
struct evbuffer * output_buffer = evhttp_request_get_output_buffer(req);
|
|
|
|
|
assert(output_buffer);
|
|
|
|
|
evbuffer_add(output_buffer, strRequest.data(), strRequest.size());
|
2014-05-26 11:38:44 +02:00
|
|
|
|
2015-01-23 07:53:17 +01:00
|
|
|
int r = evhttp_make_request(evcon, req, EVHTTP_REQ_POST, "/");
|
|
|
|
|
if (r != 0) {
|
|
|
|
|
evhttp_connection_free(evcon);
|
|
|
|
|
event_base_free(base);
|
|
|
|
|
throw CConnectionFailed("send http request failed");
|
|
|
|
|
}
|
2014-05-26 11:38:44 +02:00
|
|
|
|
2015-01-23 07:53:17 +01:00
|
|
|
event_base_dispatch(base);
|
|
|
|
|
evhttp_connection_free(evcon);
|
|
|
|
|
event_base_free(base);
|
2014-05-26 11:38:44 +02:00
|
|
|
|
2015-01-23 07:53:17 +01:00
|
|
|
if (response.status == 0)
|
|
|
|
|
throw CConnectionFailed("couldn't connect to server");
|
|
|
|
|
else if (response.status == HTTP_UNAUTHORIZED)
|
2017-06-18 16:59:29 -04:00
|
|
|
throw std::runtime_error("incorrect rpcuser or rpcpassword (authorization failed)");
|
2015-01-23 07:53:17 +01:00
|
|
|
else if (response.status >= 400 && response.status != HTTP_BAD_REQUEST && response.status != HTTP_NOT_FOUND && response.status != HTTP_INTERNAL_SERVER_ERROR)
|
2017-06-18 16:59:29 -04:00
|
|
|
throw std::runtime_error(strprintf("server returned HTTP error %d", response.status));
|
2015-01-23 07:53:17 +01:00
|
|
|
else if (response.body.empty())
|
2017-06-18 16:59:29 -04:00
|
|
|
throw std::runtime_error("no response from server");
|
2014-05-26 11:38:44 +02:00
|
|
|
|
|
|
|
|
// Parse reply
|
2015-05-13 21:29:19 +02:00
|
|
|
UniValue valReply(UniValue::VSTR);
|
2015-01-23 07:53:17 +01:00
|
|
|
if (!valReply.read(response.body))
|
2017-06-18 16:59:29 -04:00
|
|
|
throw std::runtime_error("couldn't parse reply from server");
|
2015-05-18 14:02:18 +02:00
|
|
|
const UniValue& reply = valReply.get_obj();
|
2014-05-26 11:38:44 +02:00
|
|
|
if (reply.empty())
|
2017-06-18 16:59:29 -04:00
|
|
|
throw std::runtime_error("expected reply to have result, error and id properties");
|
2014-05-26 11:38:44 +02:00
|
|
|
|
|
|
|
|
return reply;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
int CommandLineRPC(int argc, char *argv[])
|
|
|
|
|
{
|
2017-06-18 16:55:32 -04:00
|
|
|
std::string strPrint;
|
2014-05-26 11:38:44 +02:00
|
|
|
int nRet = 0;
|
2014-06-25 21:09:36 -04:00
|
|
|
try {
|
2014-05-26 11:38:44 +02:00
|
|
|
// Skip switches
|
2014-06-25 21:09:36 -04:00
|
|
|
while (argc > 1 && IsSwitchChar(argv[1][0])) {
|
2014-05-26 11:38:44 +02:00
|
|
|
argc--;
|
|
|
|
|
argv++;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Method
|
|
|
|
|
if (argc < 2)
|
2017-06-18 16:59:29 -04:00
|
|
|
throw std::runtime_error("too few parameters");
|
2017-06-18 16:55:32 -04:00
|
|
|
std::string strMethod = argv[1];
|
2014-05-26 11:38:44 +02:00
|
|
|
|
|
|
|
|
// Parameters default to strings
|
|
|
|
|
std::vector<std::string> strParams(&argv[2], &argv[argc]);
|
2015-05-13 21:29:19 +02:00
|
|
|
UniValue params = RPCConvertValues(strMethod, strParams);
|
2014-05-26 11:38:44 +02:00
|
|
|
|
2014-10-29 18:08:31 +01:00
|
|
|
// Execute and handle connection failures with -rpcwait
|
|
|
|
|
const bool fWait = GetBoolArg("-rpcwait", false);
|
|
|
|
|
do {
|
|
|
|
|
try {
|
2015-05-13 21:29:19 +02:00
|
|
|
const UniValue reply = CallRPC(strMethod, params);
|
2014-05-26 11:38:44 +02:00
|
|
|
|
2014-10-29 18:08:31 +01:00
|
|
|
// Parse reply
|
2015-05-18 14:02:18 +02:00
|
|
|
const UniValue& result = find_value(reply, "result");
|
|
|
|
|
const UniValue& error = find_value(reply, "error");
|
2014-05-26 11:38:44 +02:00
|
|
|
|
2014-08-20 15:15:16 -04:00
|
|
|
if (!error.isNull()) {
|
2014-10-29 18:08:31 +01:00
|
|
|
// Error
|
2014-08-20 15:15:16 -04:00
|
|
|
int code = error["code"].get_int();
|
2015-05-18 14:02:18 +02:00
|
|
|
if (fWait && code == RPC_IN_WARMUP)
|
|
|
|
|
throw CConnectionFailed("server in warmup");
|
|
|
|
|
strPrint = "error: " + error.write();
|
2014-10-29 18:08:31 +01:00
|
|
|
nRet = abs(code);
|
2015-07-07 12:15:44 +02:00
|
|
|
if (error.isObject())
|
|
|
|
|
{
|
|
|
|
|
UniValue errCode = find_value(error, "code");
|
|
|
|
|
UniValue errMsg = find_value(error, "message");
|
|
|
|
|
strPrint = errCode.isNull() ? "" : "error code: "+errCode.getValStr()+"\n";
|
|
|
|
|
|
|
|
|
|
if (errMsg.isStr())
|
|
|
|
|
strPrint += "error message:\n"+errMsg.get_str();
|
|
|
|
|
}
|
2014-10-29 18:08:31 +01:00
|
|
|
} else {
|
|
|
|
|
// Result
|
2014-08-20 15:15:16 -04:00
|
|
|
if (result.isNull())
|
2014-10-29 18:08:31 +01:00
|
|
|
strPrint = "";
|
2014-08-20 15:15:16 -04:00
|
|
|
else if (result.isStr())
|
2014-10-29 18:08:31 +01:00
|
|
|
strPrint = result.get_str();
|
|
|
|
|
else
|
2014-08-20 15:15:16 -04:00
|
|
|
strPrint = result.write(2);
|
2014-10-29 18:08:31 +01:00
|
|
|
}
|
|
|
|
|
// Connection succeeded, no need to retry.
|
|
|
|
|
break;
|
|
|
|
|
}
|
2014-12-07 13:29:06 +01:00
|
|
|
catch (const CConnectionFailed&) {
|
2014-10-29 18:08:31 +01:00
|
|
|
if (fWait)
|
|
|
|
|
MilliSleep(1000);
|
|
|
|
|
else
|
|
|
|
|
throw;
|
|
|
|
|
}
|
|
|
|
|
} while (fWait);
|
2014-05-26 11:38:44 +02:00
|
|
|
}
|
2014-12-07 13:29:06 +01:00
|
|
|
catch (const boost::thread_interrupted&) {
|
2014-05-26 11:38:44 +02:00
|
|
|
throw;
|
|
|
|
|
}
|
2014-12-07 13:29:06 +01:00
|
|
|
catch (const std::exception& e) {
|
2017-06-18 17:00:55 -04:00
|
|
|
strPrint = std::string("error: ") + e.what();
|
2014-05-26 11:38:44 +02:00
|
|
|
nRet = EXIT_FAILURE;
|
|
|
|
|
}
|
|
|
|
|
catch (...) {
|
|
|
|
|
PrintExceptionContinue(NULL, "CommandLineRPC()");
|
|
|
|
|
throw;
|
|
|
|
|
}
|
|
|
|
|
|
2014-06-25 21:09:36 -04:00
|
|
|
if (strPrint != "") {
|
2014-05-26 11:38:44 +02:00
|
|
|
fprintf((nRet == 0 ? stdout : stderr), "%s\n", strPrint.c_str());
|
|
|
|
|
}
|
|
|
|
|
return nRet;
|
|
|
|
|
}
|
|
|
|
|
|
2013-10-11 23:09:59 +02:00
|
|
|
int main(int argc, char* argv[])
|
|
|
|
|
{
|
2014-05-13 10:15:00 +00:00
|
|
|
SetupEnvironment();
|
2015-09-02 16:18:16 +02:00
|
|
|
if (!SetupNetworking()) {
|
|
|
|
|
fprintf(stderr, "Error: Initializing networking failed\n");
|
|
|
|
|
exit(1);
|
|
|
|
|
}
|
2014-05-13 10:15:00 +00:00
|
|
|
|
2014-06-25 21:09:36 -04:00
|
|
|
try {
|
2013-10-11 23:09:59 +02:00
|
|
|
if(!AppInitRPC(argc, argv))
|
2014-06-12 22:26:46 -04:00
|
|
|
return EXIT_FAILURE;
|
2013-10-11 23:09:59 +02:00
|
|
|
}
|
2014-12-07 13:29:06 +01:00
|
|
|
catch (const std::exception& e) {
|
2013-10-11 23:09:59 +02:00
|
|
|
PrintExceptionContinue(&e, "AppInitRPC()");
|
2014-06-12 22:26:46 -04:00
|
|
|
return EXIT_FAILURE;
|
2013-10-11 23:09:59 +02:00
|
|
|
} catch (...) {
|
|
|
|
|
PrintExceptionContinue(NULL, "AppInitRPC()");
|
2014-06-12 22:26:46 -04:00
|
|
|
return EXIT_FAILURE;
|
2013-10-11 23:09:59 +02:00
|
|
|
}
|
|
|
|
|
|
2014-06-12 22:26:46 -04:00
|
|
|
int ret = EXIT_FAILURE;
|
2014-06-25 21:09:36 -04:00
|
|
|
try {
|
2014-02-24 14:08:56 +01:00
|
|
|
ret = CommandLineRPC(argc, argv);
|
2013-10-11 23:09:59 +02:00
|
|
|
}
|
2014-12-07 13:29:06 +01:00
|
|
|
catch (const std::exception& e) {
|
2013-10-11 23:09:59 +02:00
|
|
|
PrintExceptionContinue(&e, "CommandLineRPC()");
|
|
|
|
|
} catch (...) {
|
|
|
|
|
PrintExceptionContinue(NULL, "CommandLineRPC()");
|
|
|
|
|
}
|
2014-02-24 14:08:56 +01:00
|
|
|
return ret;
|
2013-10-11 23:09:59 +02:00
|
|
|
}
|