config option -rpcallowip= to accept json-rpc connections from another machine

git-svn-id: https://bitcoin.svn.sourceforge.net/svnroot/bitcoin/trunk@155 1a98c847-1fd6-4fd8-948a-caf3550aa51b
This commit is contained in:
s_nakamoto 2010-09-15 22:10:13 +00:00
parent 71cc095cb2
commit fe37c35018
5 changed files with 76 additions and 22 deletions

View File

@ -3170,11 +3170,6 @@ int64 GetBalance()
} }
int GetRandInt(int nMax)
{
return GetRand(nMax);
}
bool SelectCoins(int64 nTargetValue, set<CWalletTx*>& setCoinsRet) bool SelectCoins(int64 nTargetValue, set<CWalletTx*>& setCoinsRet)
{ {
setCoinsRet.clear(); setCoinsRet.clear();

24
rpc.cpp
View File

@ -783,8 +783,9 @@ int ReadHTTPStatus(tcp::iostream& stream)
getline(stream, str); getline(stream, str);
vector<string> vWords; vector<string> vWords;
boost::split(vWords, str, boost::is_any_of(" ")); boost::split(vWords, str, boost::is_any_of(" "));
int nStatus = atoi(vWords[1].c_str()); if (vWords.size() < 2)
return nStatus; return 500;
return atoi(vWords[1].c_str());
} }
int ReadHTTPHeader(tcp::iostream& stream, map<string, string>& mapHeadersRet) int ReadHTTPHeader(tcp::iostream& stream, map<string, string>& mapHeadersRet)
@ -918,6 +919,17 @@ string JSONRPCReply(const Value& result, const Value& error, const Value& id)
return write_string(Value(reply), false) + "\n"; return write_string(Value(reply), false) + "\n";
} }
bool ClientAllowed(const string& strAddress)
{
if (strAddress == asio::ip::address_v4::loopback().to_string())
return true;
const vector<string>& vAllow = mapMultiArgs["-rpcallowip"];
foreach(string strAllow, vAllow)
if (WildcardMatch(strAddress, strAllow))
return true;
return false;
}
@ -962,7 +974,7 @@ void ThreadRPCServer2(void* parg)
// Bind to loopback 127.0.0.1 so the socket can only be accessed locally // Bind to loopback 127.0.0.1 so the socket can only be accessed locally
boost::asio::io_service io_service; boost::asio::io_service io_service;
tcp::endpoint endpoint(boost::asio::ip::address_v4::loopback(), 8332); tcp::endpoint endpoint(mapArgs.count("-rpcallowip") ? asio::ip::address_v4::any() : asio::ip::address_v4::loopback(), 8332);
tcp::acceptor acceptor(io_service, endpoint); tcp::acceptor acceptor(io_service, endpoint);
loop loop
@ -976,8 +988,8 @@ void ThreadRPCServer2(void* parg)
if (fShutdown) if (fShutdown)
return; return;
// Shouldn't be possible for anyone else to connect, but just in case // Restrict callers by IP
if (peer.address().to_string() != "127.0.0.1") if (!ClientAllowed(peer.address().to_string()))
continue; continue;
// Receive request // Receive request
@ -1090,7 +1102,7 @@ Object CallRPC(const string& strMethod, const Array& params)
GetConfigFile().c_str())); GetConfigFile().c_str()));
// Connect to localhost // Connect to localhost
tcp::iostream stream("127.0.0.1", "8332"); tcp::iostream stream(GetArg("-rpcconnect", "127.0.0.1"), "8332");
if (stream.fail()) if (stream.fail())
throw runtime_error("couldn't connect to server"); throw runtime_error("couldn't connect to server");

View File

@ -23,7 +23,7 @@ class CAutoFile;
static const unsigned int MAX_SIZE = 0x02000000; static const unsigned int MAX_SIZE = 0x02000000;
static const int VERSION = 312; static const int VERSION = 312;
static const char* pszSubVer = ".5"; static const char* pszSubVer = ".6";

View File

@ -127,6 +127,11 @@ uint64 GetRand(uint64 nMax)
return (nRand % nMax); return (nRand % nMax);
} }
int GetRandInt(int nMax)
{
return GetRand(nMax);
}
@ -405,7 +410,7 @@ vector<unsigned char> ParseHex(const char* psz)
return vch; return vch;
} }
vector<unsigned char> ParseHex(const std::string& str) vector<unsigned char> ParseHex(const string& str)
{ {
return ParseHex(str.c_str()); return ParseHex(str.c_str());
} }
@ -473,6 +478,34 @@ const char* wxGetTranslation(const char* pszEnglish)
} }
bool WildcardMatch(const char* psz, const char* mask)
{
loop
{
switch (*mask)
{
case '\0':
return (*psz == '\0');
case '*':
return WildcardMatch(psz, mask+1) || (*psz && WildcardMatch(psz+1, mask));
case '?':
if (*psz == '\0')
return false;
break;
default:
if (*psz != *mask)
return false;
break;
}
psz++;
mask++;
}
}
bool WildcardMatch(const string& str, const string& mask)
{
return WildcardMatch(str.c_str(), mask.c_str());
}
@ -650,7 +683,7 @@ string GetDataDir()
string GetConfigFile() string GetConfigFile()
{ {
namespace fs = boost::filesystem; namespace fs = boost::filesystem;
fs::path pathConfig(mapArgs.count("-conf") ? mapArgs["-conf"] : string("bitcoin.conf")); fs::path pathConfig(GetArg("-conf", "bitcoin.conf"));
if (!pathConfig.is_complete()) if (!pathConfig.is_complete())
pathConfig = fs::path(GetDataDir()) / pathConfig; pathConfig = fs::path(GetDataDir()) / pathConfig;
return pathConfig.string(); return pathConfig.string();
@ -718,13 +751,10 @@ void ShrinkDebugFile()
// //
// "Never go to sea with two chronometers; take one or three." // "Never go to sea with two chronometers; take one or three."
// Our three chronometers are: // Our three time sources are:
// - System clock // - System clock
// - Median of other server's clocks // - Median of other nodes's clocks
// - NTP servers // - The user (asking the user to fix the system clock if the first two disagree)
//
// note: NTP isn't implemented yet, so until then we just use the median
// of other nodes clocks to correct ours.
// //
int64 GetTime() int64 GetTime()
{ {
@ -768,7 +798,7 @@ void AddTimeData(unsigned int ip, int64 nTime)
// If nobody else has the same time as us, give a warning // If nobody else has the same time as us, give a warning
bool fMatch = false; bool fMatch = false;
foreach(int64 nOffset, vTimeOffsets) foreach(int64 nOffset, vTimeOffsets)
if (nOffset != 0 && abs64(nOffset) < 10 * 60) if (nOffset != 0 && abs64(nOffset) < 5 * 60)
fMatch = true; fMatch = true;
static bool fDone; static bool fDone;
if (!fMatch && !fDone) if (!fMatch && !fDone)

19
util.h
View File

@ -160,9 +160,11 @@ string FormatMoney(int64 n, bool fPlus=false);
bool ParseMoney(const string& str, int64& nRet); bool ParseMoney(const string& str, int64& nRet);
bool ParseMoney(const char* pszIn, int64& nRet); bool ParseMoney(const char* pszIn, int64& nRet);
vector<unsigned char> ParseHex(const char* psz); vector<unsigned char> ParseHex(const char* psz);
vector<unsigned char> ParseHex(const std::string& str); vector<unsigned char> ParseHex(const string& str);
void ParseParameters(int argc, char* argv[]); void ParseParameters(int argc, char* argv[]);
const char* wxGetTranslation(const char* psz); const char* wxGetTranslation(const char* psz);
bool WildcardMatch(const char* psz, const char* mask);
bool WildcardMatch(const string& str, const string& mask);
int GetFilesize(FILE* file); int GetFilesize(FILE* file);
void GetDataDir(char* pszDirRet); void GetDataDir(char* pszDirRet);
string GetConfigFile(); string GetConfigFile();
@ -173,6 +175,7 @@ string MyGetSpecialFolderPath(int nFolder, bool fCreate);
string GetDefaultDataDir(); string GetDefaultDataDir();
string GetDataDir(); string GetDataDir();
void ShrinkDebugFile(); void ShrinkDebugFile();
int GetRandInt(int nMax);
uint64 GetRand(uint64 nMax); uint64 GetRand(uint64 nMax);
int64 GetTime(); int64 GetTime();
int64 GetAdjustedTime(); int64 GetAdjustedTime();
@ -399,6 +402,20 @@ inline bool IsSwitchChar(char c)
#endif #endif
} }
inline string GetArg(const string& strArg, const string& strDefault)
{
if (mapArgs.count(strArg))
return mapArgs[strArg];
return strDefault;
}
inline int64 GetArg(const string& strArg, int64 nDefault)
{
if (mapArgs.count(strArg))
return atoi64(mapArgs[strArg]);
return nDefault;
}