Merge #9268: Fix rounding privacy leak introduced in #9260
[bitcoinplatinum.git] / src / core_read.cpp
blobb6d0285459ed8ee6249da7a630044690153fc48d
1 // Copyright (c) 2009-2015 The Bitcoin Core developers
2 // Distributed under the MIT software license, see the accompanying
3 // file COPYING or http://www.opensource.org/licenses/mit-license.php.
5 #include "core_io.h"
7 #include "primitives/block.h"
8 #include "primitives/transaction.h"
9 #include "script/script.h"
10 #include "serialize.h"
11 #include "streams.h"
12 #include <univalue.h>
13 #include "util.h"
14 #include "utilstrencodings.h"
15 #include "version.h"
17 #include <boost/algorithm/string/classification.hpp>
18 #include <boost/algorithm/string/predicate.hpp>
19 #include <boost/algorithm/string/replace.hpp>
20 #include <boost/algorithm/string/split.hpp>
21 #include <boost/assign/list_of.hpp>
23 using namespace std;
25 CScript ParseScript(const std::string& s)
27 CScript result;
29 static map<string, opcodetype> mapOpNames;
31 if (mapOpNames.empty())
33 for (int op = 0; op <= OP_NOP10; op++)
35 // Allow OP_RESERVED to get into mapOpNames
36 if (op < OP_NOP && op != OP_RESERVED)
37 continue;
39 const char* name = GetOpName((opcodetype)op);
40 if (strcmp(name, "OP_UNKNOWN") == 0)
41 continue;
42 string strName(name);
43 mapOpNames[strName] = (opcodetype)op;
44 // Convenience: OP_ADD and just ADD are both recognized:
45 boost::algorithm::replace_first(strName, "OP_", "");
46 mapOpNames[strName] = (opcodetype)op;
50 vector<string> words;
51 boost::algorithm::split(words, s, boost::algorithm::is_any_of(" \t\n"), boost::algorithm::token_compress_on);
53 for (std::vector<std::string>::const_iterator w = words.begin(); w != words.end(); ++w)
55 if (w->empty())
57 // Empty string, ignore. (boost::split given '' will return one word)
59 else if (all(*w, boost::algorithm::is_digit()) ||
60 (boost::algorithm::starts_with(*w, "-") && all(string(w->begin()+1, w->end()), boost::algorithm::is_digit())))
62 // Number
63 int64_t n = atoi64(*w);
64 result << n;
66 else if (boost::algorithm::starts_with(*w, "0x") && (w->begin()+2 != w->end()) && IsHex(string(w->begin()+2, w->end())))
68 // Raw hex data, inserted NOT pushed onto stack:
69 std::vector<unsigned char> raw = ParseHex(string(w->begin()+2, w->end()));
70 result.insert(result.end(), raw.begin(), raw.end());
72 else if (w->size() >= 2 && boost::algorithm::starts_with(*w, "'") && boost::algorithm::ends_with(*w, "'"))
74 // Single-quoted string, pushed as data. NOTE: this is poor-man's
75 // parsing, spaces/tabs/newlines in single-quoted strings won't work.
76 std::vector<unsigned char> value(w->begin()+1, w->end()-1);
77 result << value;
79 else if (mapOpNames.count(*w))
81 // opcode, e.g. OP_ADD or ADD:
82 result << mapOpNames[*w];
84 else
86 throw runtime_error("script parse error");
90 return result;
93 bool DecodeHexTx(CMutableTransaction& tx, const std::string& strHexTx, bool fTryNoWitness)
95 if (!IsHex(strHexTx))
96 return false;
98 vector<unsigned char> txData(ParseHex(strHexTx));
100 if (fTryNoWitness) {
101 CDataStream ssData(txData, SER_NETWORK, PROTOCOL_VERSION | SERIALIZE_TRANSACTION_NO_WITNESS);
102 try {
103 ssData >> tx;
104 if (ssData.eof()) {
105 return true;
108 catch (const std::exception&) {
109 // Fall through.
113 CDataStream ssData(txData, SER_NETWORK, PROTOCOL_VERSION);
114 try {
115 ssData >> tx;
117 catch (const std::exception&) {
118 return false;
121 return true;
124 bool DecodeHexBlk(CBlock& block, const std::string& strHexBlk)
126 if (!IsHex(strHexBlk))
127 return false;
129 std::vector<unsigned char> blockData(ParseHex(strHexBlk));
130 CDataStream ssBlock(blockData, SER_NETWORK, PROTOCOL_VERSION);
131 try {
132 ssBlock >> block;
134 catch (const std::exception&) {
135 return false;
138 return true;
141 uint256 ParseHashUV(const UniValue& v, const string& strName)
143 string strHex;
144 if (v.isStr())
145 strHex = v.getValStr();
146 return ParseHashStr(strHex, strName); // Note: ParseHashStr("") throws a runtime_error
149 uint256 ParseHashStr(const std::string& strHex, const std::string& strName)
151 if (!IsHex(strHex)) // Note: IsHex("") is false
152 throw runtime_error(strName+" must be hexadecimal string (not '"+strHex+"')");
154 uint256 result;
155 result.SetHex(strHex);
156 return result;
159 vector<unsigned char> ParseHexUV(const UniValue& v, const string& strName)
161 string strHex;
162 if (v.isStr())
163 strHex = v.getValStr();
164 if (!IsHex(strHex))
165 throw runtime_error(strName+" must be hexadecimal string (not '"+strHex+"')");
166 return ParseHex(strHex);