#include "main.h"
#include "bitcoinrpc.h"
+#include <boost/filesystem.hpp>
+#include <boost/iostreams/device/file.hpp>
+#include <boost/iostreams/stream.hpp>
+#include <ostream>
using namespace json_spirit;
using namespace std;
-extern void TxToJSON(const CTransaction& tx, const uint256 hashBlock, json_spirit::Object& entry);
+extern void TxToJSON(const CTransaction& tx, const uint256& hashBlock, json_spirit::Object& entry);
extern enum Checkpoints::CPMode CheckpointsMode;
double GetDifficulty(const CBlockIndex* blockindex)
Object blockToJSON(const CBlock& block, const CBlockIndex* blockindex, bool fPrintTransactionDetail)
{
Object result;
- result.push_back(Pair("hash", block.GetHash().GetHex()));
+ result.push_back(Pair("hash", blockindex->GetBlockHash().GetHex()));
CMerkleTx txGen(block.vtx[0]);
txGen.SetMerkleBranch(&block);
result.push_back(Pair("confirmations", (int)txGen.GetDepthInMainChain()));
result.push_back(Pair("version", block.nVersion));
result.push_back(Pair("merkleroot", block.hashMerkleRoot.GetHex()));
result.push_back(Pair("mint", ValueFromAmount(blockindex->nMint)));
- result.push_back(Pair("time", (boost::int64_t)block.GetBlockTime()));
- result.push_back(Pair("nonce", (boost::uint64_t)block.nNonce));
+ result.push_back(Pair("time", (int64_t)block.GetBlockTime()));
+ result.push_back(Pair("nonce", (uint64_t)block.nNonce));
result.push_back(Pair("bits", HexBits(block.nBits)));
result.push_back(Pair("difficulty", GetDifficulty(blockindex)));
result.push_back(Pair("blocktrust", leftTrim(blockindex->GetBlockTrust().GetHex(), '0')));
result.push_back(Pair("modifier", strprintf("%016" PRIx64, blockindex->nStakeModifier)));
result.push_back(Pair("modifierchecksum", strprintf("%08x", blockindex->nStakeModifierChecksum)));
Array txinfo;
- BOOST_FOREACH (const CTransaction& tx, block.vtx)
+ for (const CTransaction& tx : block.vtx)
{
if (fPrintTransactionDetail)
{
result.push_back(Pair("tx", txinfo));
- if ( block.IsProofOfStake() || (!fTestNet && block.GetBlockTime() < ENTROPY_SWITCH_TIME) )
+ if ( block.IsProofOfStake() )
result.push_back(Pair("signature", HexStr(block.vchBlockSig.begin(), block.vchBlockSig.end())));
return result;
mempool.queryHashes(vtxid);
Array a;
- BOOST_FOREACH(const uint256& hash, vtxid)
+ for (const uint256& hash : vtxid)
a.push_back(hash.ToString());
return a;
{
if (fHelp || params.size() < 1 || params.size() > 2)
throw runtime_error(
- "getblock <number> [txinfo]\n"
+ "getblockbynumber <number> [txinfo]\n"
"txinfo optional to print more detailed tx info\n"
"Returns details of a block with given block-number.");
while (pblockindex->nHeight > nHeight)
pblockindex = pblockindex->pprev;
- uint256 hash = *pblockindex->phashBlock;
-
- pblockindex = mapBlockIndex[hash];
+ pblockindex = mapBlockIndex[*pblockindex->phashBlock];
block.ReadFromDisk(pblockindex, true);
return blockToJSON(block, pblockindex, params.size() > 1 ? params[1].get_bool() : false);
}
+bool ExportBlock(const string& strBlockHash, const CDataStream& ssBlock)
+{
+ boost::filesystem::path pathDest = GetDataDir() / strBlockHash;
+ if (boost::filesystem::is_directory(pathDest))
+ pathDest /= strBlockHash;
+
+ try {
+ boost::iostreams::stream_buffer<boost::iostreams::file_sink> buf(pathDest.string());
+ ostream exportStream(&buf);
+ exportStream << HexStr(ssBlock.begin(), ssBlock.end());
+ exportStream.flush();
+
+ printf("Successfully exported block to %s\n", pathDest.string().c_str());
+ return true;
+ } catch(const boost::filesystem::filesystem_error &e) {
+ printf("error exporting the block data %s (%s)\n", pathDest.string().c_str(), e.what());
+ return false;
+ }
+}
+
+
+Value dumpblock(const Array& params, bool fHelp)
+{
+ if (fHelp || params.size() < 1 || params.size() > 2)
+ throw runtime_error(
+ "dumpblock <hash> [destination]\n"
+ "Returns serialized contents of a block with given block-hash.");
+
+ std::string strHash = params[0].get_str();
+ uint256 hash(strHash);
+
+ if (mapBlockIndex.count(hash) == 0)
+ throw JSONRPCError(RPC_INVALID_ADDRESS_OR_KEY, "Block not found");
+
+ CBlock block;
+ CBlockIndex* pblockindex = mapBlockIndex[hash];
+ block.ReadFromDisk(pblockindex, true);
+
+ CDataStream ssBlock(SER_NETWORK, PROTOCOL_VERSION);
+ ssBlock << block;
+
+ if (params.size() > 1)
+ {
+ return ExportBlock(params[1].get_str(), ssBlock);
+ }
+
+ return HexStr(ssBlock.begin(), ssBlock.end());
+}
+
+
+Value dumpblockbynumber(const Array& params, bool fHelp)
+{
+ if (fHelp || params.size() < 1 || params.size() > 2)
+ throw runtime_error(
+ "dumpblockbynumber <number> [destination]\n"
+ "Returns serialized contents of a block with given block-number.");
+
+ int nHeight = params[0].get_int();
+ if (nHeight < 0 || nHeight > nBestHeight)
+ throw runtime_error("Block number out of range.");
+
+ CBlock block;
+ CBlockIndex* pblockindex = mapBlockIndex[hashBestChain];
+ while (pblockindex->nHeight > nHeight)
+ pblockindex = pblockindex->pprev;
+
+ pblockindex = mapBlockIndex[*pblockindex->phashBlock];
+ block.ReadFromDisk(pblockindex, true);
+
+ CDataStream ssBlock(SER_NETWORK, PROTOCOL_VERSION);
+ ssBlock << block;
+
+ if (params.size() > 1)
+ {
+ return ExportBlock(params[1].get_str(), ssBlock);
+ }
+
+ return HexStr(ssBlock.begin(), ssBlock.end());
+}
+
+
// get information of sync-checkpoint
Value getcheckpoint(const Array& params, bool fHelp)
{