2013-11-20 14:18:57 +01:00
|
|
|
// Copyright (c) 2010 Satoshi Nakamoto
|
2016-12-31 11:01:21 -07:00
|
|
|
// Copyright (c) 2009-2016 The Bitcoin Core developers
|
2014-11-20 10:19:29 +08:00
|
|
|
// Distributed under the MIT software license, see the accompanying
|
2013-11-20 14:18:57 +01:00
|
|
|
// file COPYING or http://www.opensource.org/licenses/mit-license.php.
|
|
|
|
|
2016-01-15 11:55:17 +11:00
|
|
|
#include "rpc/protocol.h"
|
2013-11-20 14:18:57 +01:00
|
|
|
|
2015-07-07 14:53:48 +02:00
|
|
|
#include "random.h"
|
Split up util.cpp/h
Split up util.cpp/h into:
- string utilities (hex, base32, base64): no internal dependencies, no dependency on boost (apart from foreach)
- money utilities (parsesmoney, formatmoney)
- time utilities (gettime*, sleep, format date):
- and the rest (logging, argument parsing, config file parsing)
The latter is basically the environment and OS handling,
and is stripped of all utility functions, so we may want to
rename it to something else than util.cpp/h for clarity (Matt suggested
osinterface).
Breaks dependency of sha256.cpp on all the things pulled in by util.
2014-08-21 16:11:09 +02:00
|
|
|
#include "tinyformat.h"
|
2014-09-14 12:43:56 +02:00
|
|
|
#include "util.h"
|
Split up util.cpp/h
Split up util.cpp/h into:
- string utilities (hex, base32, base64): no internal dependencies, no dependency on boost (apart from foreach)
- money utilities (parsesmoney, formatmoney)
- time utilities (gettime*, sleep, format date):
- and the rest (logging, argument parsing, config file parsing)
The latter is basically the environment and OS handling,
and is stripped of all utility functions, so we may want to
rename it to something else than util.cpp/h for clarity (Matt suggested
osinterface).
Breaks dependency of sha256.cpp on all the things pulled in by util.
2014-08-21 16:11:09 +02:00
|
|
|
#include "utilstrencodings.h"
|
|
|
|
#include "utiltime.h"
|
2014-08-21 16:11:05 +02:00
|
|
|
#include "version.h"
|
2013-11-20 14:18:57 +01:00
|
|
|
|
|
|
|
#include <stdint.h>
|
2015-07-07 14:53:48 +02:00
|
|
|
#include <fstream>
|
2013-11-20 14:18:57 +01:00
|
|
|
|
2014-11-20 10:19:29 +08:00
|
|
|
/**
|
|
|
|
* JSON-RPC protocol. Bitcoin speaks version 1.0 for maximum compatibility,
|
|
|
|
* but uses JSON-RPC 1.1/2.0 standards for parts of the 1.0 standard that were
|
|
|
|
* unspecified (HTTP errors and contents of 'error').
|
|
|
|
*
|
|
|
|
* 1.0 spec: http://json-rpc.org/wiki/specification
|
|
|
|
* 1.2 spec: http://jsonrpc.org/historical/json-rpc-over-http.html
|
|
|
|
*/
|
2013-11-20 14:18:57 +01:00
|
|
|
|
2017-01-04 13:22:19 +09:00
|
|
|
UniValue JSONRPCRequestObj(const std::string& strMethod, const UniValue& params, const UniValue& id)
|
2013-11-20 14:18:57 +01:00
|
|
|
{
|
2015-05-13 21:29:19 +02:00
|
|
|
UniValue request(UniValue::VOBJ);
|
2013-11-20 14:18:57 +01:00
|
|
|
request.push_back(Pair("method", strMethod));
|
|
|
|
request.push_back(Pair("params", params));
|
|
|
|
request.push_back(Pair("id", id));
|
2016-09-29 18:48:27 +02:00
|
|
|
return request;
|
2013-11-20 14:18:57 +01:00
|
|
|
}
|
|
|
|
|
2015-05-18 14:02:18 +02:00
|
|
|
UniValue JSONRPCReplyObj(const UniValue& result, const UniValue& error, const UniValue& id)
|
2013-11-20 14:18:57 +01:00
|
|
|
{
|
2015-05-13 21:29:19 +02:00
|
|
|
UniValue reply(UniValue::VOBJ);
|
2014-08-20 15:15:16 -04:00
|
|
|
if (!error.isNull())
|
|
|
|
reply.push_back(Pair("result", NullUniValue));
|
2013-11-20 14:18:57 +01:00
|
|
|
else
|
|
|
|
reply.push_back(Pair("result", result));
|
|
|
|
reply.push_back(Pair("error", error));
|
|
|
|
reply.push_back(Pair("id", id));
|
|
|
|
return reply;
|
|
|
|
}
|
|
|
|
|
2017-01-04 13:22:19 +09:00
|
|
|
std::string JSONRPCReply(const UniValue& result, const UniValue& error, const UniValue& id)
|
2013-11-20 14:18:57 +01:00
|
|
|
{
|
2015-05-13 21:29:19 +02:00
|
|
|
UniValue reply = JSONRPCReplyObj(result, error, id);
|
2014-08-20 15:15:16 -04:00
|
|
|
return reply.write() + "\n";
|
2013-11-20 14:18:57 +01:00
|
|
|
}
|
|
|
|
|
2017-01-04 13:22:19 +09:00
|
|
|
UniValue JSONRPCError(int code, const std::string& message)
|
2013-11-20 14:18:57 +01:00
|
|
|
{
|
2015-05-10 13:35:44 +02:00
|
|
|
UniValue error(UniValue::VOBJ);
|
2013-11-20 14:18:57 +01:00
|
|
|
error.push_back(Pair("code", code));
|
|
|
|
error.push_back(Pair("message", message));
|
|
|
|
return error;
|
|
|
|
}
|
2015-07-07 14:53:48 +02:00
|
|
|
|
|
|
|
/** Username used when cookie authentication is in use (arbitrary, only for
|
|
|
|
* recognizability in debugging/logging purposes)
|
|
|
|
*/
|
|
|
|
static const std::string COOKIEAUTH_USER = "__cookie__";
|
|
|
|
/** Default name for auth cookie file */
|
|
|
|
static const std::string COOKIEAUTH_FILE = ".cookie";
|
|
|
|
|
|
|
|
boost::filesystem::path GetAuthCookieFile()
|
|
|
|
{
|
|
|
|
boost::filesystem::path path(GetArg("-rpccookiefile", COOKIEAUTH_FILE));
|
|
|
|
if (!path.is_complete()) path = GetDataDir() / path;
|
|
|
|
return path;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool GenerateAuthCookie(std::string *cookie_out)
|
|
|
|
{
|
2016-10-01 16:57:25 +02:00
|
|
|
const size_t COOKIE_SIZE = 32;
|
|
|
|
unsigned char rand_pwd[COOKIE_SIZE];
|
|
|
|
GetRandBytes(rand_pwd, COOKIE_SIZE);
|
|
|
|
std::string cookie = COOKIEAUTH_USER + ":" + HexStr(rand_pwd, rand_pwd+COOKIE_SIZE);
|
2015-07-07 14:53:48 +02:00
|
|
|
|
|
|
|
/** the umask determines what permissions are used to create this file -
|
|
|
|
* these are set to 077 in init.cpp unless overridden with -sysperms.
|
|
|
|
*/
|
|
|
|
std::ofstream file;
|
|
|
|
boost::filesystem::path filepath = GetAuthCookieFile();
|
|
|
|
file.open(filepath.string().c_str());
|
|
|
|
if (!file.is_open()) {
|
|
|
|
LogPrintf("Unable to open cookie authentication file %s for writing\n", filepath.string());
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
file << cookie;
|
|
|
|
file.close();
|
|
|
|
LogPrintf("Generated RPC authentication cookie %s\n", filepath.string());
|
|
|
|
|
|
|
|
if (cookie_out)
|
|
|
|
*cookie_out = cookie;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool GetAuthCookie(std::string *cookie_out)
|
|
|
|
{
|
|
|
|
std::ifstream file;
|
|
|
|
std::string cookie;
|
|
|
|
boost::filesystem::path filepath = GetAuthCookieFile();
|
|
|
|
file.open(filepath.string().c_str());
|
|
|
|
if (!file.is_open())
|
|
|
|
return false;
|
|
|
|
std::getline(file, cookie);
|
|
|
|
file.close();
|
|
|
|
|
|
|
|
if (cookie_out)
|
|
|
|
*cookie_out = cookie;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void DeleteAuthCookie()
|
|
|
|
{
|
|
|
|
try {
|
|
|
|
boost::filesystem::remove(GetAuthCookieFile());
|
|
|
|
} catch (const boost::filesystem::filesystem_error& e) {
|
|
|
|
LogPrintf("%s: Unable to remove random auth cookie file: %s\n", __func__, e.what());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|