mirror of
https://github.com/Xahau/xahaud.git
synced 2025-11-20 10:35:50 +00:00
Allow Json parser understand TER strings where appropriate
This commit is contained in:
committed by
Scott Schurr
parent
7e9ac16c22
commit
46004158a2
@@ -4849,6 +4849,10 @@
|
|||||||
<ExcludedFromBuild Condition="'$(Configuration)|$(Platform)'=='debug|x64'">True</ExcludedFromBuild>
|
<ExcludedFromBuild Condition="'$(Configuration)|$(Platform)'=='debug|x64'">True</ExcludedFromBuild>
|
||||||
<ExcludedFromBuild Condition="'$(Configuration)|$(Platform)'=='release|x64'">True</ExcludedFromBuild>
|
<ExcludedFromBuild Condition="'$(Configuration)|$(Platform)'=='release|x64'">True</ExcludedFromBuild>
|
||||||
</ClCompile>
|
</ClCompile>
|
||||||
|
<ClCompile Include="..\..\src\test\protocol\TER_test.cpp">
|
||||||
|
<ExcludedFromBuild Condition="'$(Configuration)|$(Platform)'=='debug|x64'">True</ExcludedFromBuild>
|
||||||
|
<ExcludedFromBuild Condition="'$(Configuration)|$(Platform)'=='release|x64'">True</ExcludedFromBuild>
|
||||||
|
</ClCompile>
|
||||||
<ClCompile Include="..\..\src\test\protocol\types_test.cpp">
|
<ClCompile Include="..\..\src\test\protocol\types_test.cpp">
|
||||||
<ExcludedFromBuild Condition="'$(Configuration)|$(Platform)'=='debug|x64'">True</ExcludedFromBuild>
|
<ExcludedFromBuild Condition="'$(Configuration)|$(Platform)'=='debug|x64'">True</ExcludedFromBuild>
|
||||||
<ExcludedFromBuild Condition="'$(Configuration)|$(Platform)'=='release|x64'">True</ExcludedFromBuild>
|
<ExcludedFromBuild Condition="'$(Configuration)|$(Platform)'=='release|x64'">True</ExcludedFromBuild>
|
||||||
|
|||||||
@@ -5556,6 +5556,9 @@
|
|||||||
<ClCompile Include="..\..\src\test\protocol\STTx_test.cpp">
|
<ClCompile Include="..\..\src\test\protocol\STTx_test.cpp">
|
||||||
<Filter>test\protocol</Filter>
|
<Filter>test\protocol</Filter>
|
||||||
</ClCompile>
|
</ClCompile>
|
||||||
|
<ClCompile Include="..\..\src\test\protocol\TER_test.cpp">
|
||||||
|
<Filter>test\protocol</Filter>
|
||||||
|
</ClCompile>
|
||||||
<ClCompile Include="..\..\src\test\protocol\types_test.cpp">
|
<ClCompile Include="..\..\src\test\protocol\types_test.cpp">
|
||||||
<Filter>test\protocol</Filter>
|
<Filter>test\protocol</Filter>
|
||||||
</ClCompile>
|
</ClCompile>
|
||||||
|
|||||||
@@ -20,6 +20,7 @@
|
|||||||
#ifndef RIPPLE_PROTOCOL_TER_H_INCLUDED
|
#ifndef RIPPLE_PROTOCOL_TER_H_INCLUDED
|
||||||
#define RIPPLE_PROTOCOL_TER_H_INCLUDED
|
#define RIPPLE_PROTOCOL_TER_H_INCLUDED
|
||||||
|
|
||||||
|
#include <boost/optional.hpp>
|
||||||
#include <string>
|
#include <string>
|
||||||
|
|
||||||
namespace ripple {
|
namespace ripple {
|
||||||
@@ -254,6 +255,10 @@ extern
|
|||||||
std::string
|
std::string
|
||||||
transHuman (TER code);
|
transHuman (TER code);
|
||||||
|
|
||||||
|
extern
|
||||||
|
boost::optional<TER>
|
||||||
|
transCode(std::string const& token);
|
||||||
|
|
||||||
} // ripple
|
} // ripple
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|||||||
@@ -31,6 +31,7 @@
|
|||||||
#include <ripple/protocol/STInteger.h>
|
#include <ripple/protocol/STInteger.h>
|
||||||
#include <ripple/protocol/STParsedJSON.h>
|
#include <ripple/protocol/STParsedJSON.h>
|
||||||
#include <ripple/protocol/STPathSet.h>
|
#include <ripple/protocol/STPathSet.h>
|
||||||
|
#include <ripple/protocol/TER.h>
|
||||||
#include <ripple/protocol/TxFormats.h>
|
#include <ripple/protocol/TxFormats.h>
|
||||||
#include <ripple/protocol/types.h>
|
#include <ripple/protocol/types.h>
|
||||||
#include <ripple/protocol/impl/STVar.h>
|
#include <ripple/protocol/impl/STVar.h>
|
||||||
@@ -189,31 +190,61 @@ static boost::optional<detail::STVar> parseLeaf (
|
|||||||
case STI_UINT8:
|
case STI_UINT8:
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
|
constexpr auto minValue = std::numeric_limits<std::uint8_t>::min();
|
||||||
|
constexpr auto maxValue = std::numeric_limits<std::uint8_t>::max();
|
||||||
if (value.isString ())
|
if (value.isString ())
|
||||||
{
|
{
|
||||||
// VFALCO TODO wtf?
|
std::string const strValue = value.asString();
|
||||||
|
|
||||||
|
if (!strValue.empty() &&
|
||||||
|
((strValue[0] < '0') || (strValue[0] > '9')))
|
||||||
|
{
|
||||||
|
if (field == sfTransactionResult)
|
||||||
|
{
|
||||||
|
auto ter = transCode(strValue);
|
||||||
|
|
||||||
|
if (!ter || *ter < minValue || *ter > maxValue)
|
||||||
|
{
|
||||||
|
error = out_of_range(json_name, fieldName);
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
|
ret = detail::make_stvar<STUInt8>(field,
|
||||||
|
static_cast<std::uint8_t>(*ter));
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
error = bad_type(json_name, fieldName);
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
ret = detail::make_stvar <STUInt8>(field,
|
||||||
|
beast::lexicalCastThrow <std::uint8_t>(strValue));
|
||||||
|
}
|
||||||
}
|
}
|
||||||
else if (value.isInt ())
|
else if (value.isInt ())
|
||||||
{
|
{
|
||||||
if (value.asInt () < 0 || value.asInt () > 255)
|
if (value.asInt () < minValue || value.asInt () > maxValue)
|
||||||
{
|
{
|
||||||
error = out_of_range (json_name, fieldName);
|
error = out_of_range (json_name, fieldName);
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
ret = detail::make_stvar <STUInt8> (field,
|
ret = detail::make_stvar <STUInt8> (field,
|
||||||
static_cast <unsigned char> (value.asInt ()));
|
static_cast <std::uint8_t> (value.asInt ()));
|
||||||
}
|
}
|
||||||
else if (value.isUInt ())
|
else if (value.isUInt ())
|
||||||
{
|
{
|
||||||
if (value.asUInt () > 255)
|
if (value.asUInt () > maxValue)
|
||||||
{
|
{
|
||||||
error = out_of_range (json_name, fieldName);
|
error = out_of_range (json_name, fieldName);
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
ret = detail::make_stvar <STUInt8> (field,
|
ret = detail::make_stvar <STUInt8> (field,
|
||||||
static_cast <unsigned char> (value.asUInt ()));
|
static_cast <std::uint8_t> (value.asUInt ()));
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
|
|||||||
@@ -19,12 +19,19 @@
|
|||||||
|
|
||||||
#include <BeastConfig.h>
|
#include <BeastConfig.h>
|
||||||
#include <ripple/protocol/TER.h>
|
#include <ripple/protocol/TER.h>
|
||||||
|
#include <boost/range/adaptor/transformed.hpp>
|
||||||
#include <unordered_map>
|
#include <unordered_map>
|
||||||
#include <type_traits>
|
#include <type_traits>
|
||||||
|
|
||||||
namespace ripple {
|
namespace ripple {
|
||||||
|
|
||||||
bool transResultInfo (TER code, std::string& token, std::string& text)
|
namespace detail {
|
||||||
|
|
||||||
|
static
|
||||||
|
std::unordered_map<
|
||||||
|
std::underlying_type_t<TER>,
|
||||||
|
std::pair<char const* const, char const* const>> const&
|
||||||
|
transResults()
|
||||||
{
|
{
|
||||||
static
|
static
|
||||||
std::unordered_map<
|
std::unordered_map<
|
||||||
@@ -141,6 +148,14 @@ bool transResultInfo (TER code, std::string& token, std::string& text)
|
|||||||
|
|
||||||
{ tesSUCCESS, { "tesSUCCESS", "The transaction was applied. Only final in a validated ledger." } },
|
{ tesSUCCESS, { "tesSUCCESS", "The transaction was applied. Only final in a validated ledger." } },
|
||||||
};
|
};
|
||||||
|
return results;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
bool transResultInfo (TER code, std::string& token, std::string& text)
|
||||||
|
{
|
||||||
|
auto& results = detail::transResults();
|
||||||
|
|
||||||
auto const r = results.find (
|
auto const r = results.find (
|
||||||
static_cast<std::underlying_type_t<TER>> (code));
|
static_cast<std::underlying_type_t<TER>> (code));
|
||||||
@@ -169,4 +184,35 @@ std::string transHuman (TER code)
|
|||||||
return transResultInfo (code, token, text) ? text : "-";
|
return transResultInfo (code, token, text) ? text : "-";
|
||||||
}
|
}
|
||||||
|
|
||||||
|
boost::optional<TER>
|
||||||
|
transCode(std::string const& token)
|
||||||
|
{
|
||||||
|
static
|
||||||
|
auto const results = []
|
||||||
|
{
|
||||||
|
auto& byTer = detail::transResults();
|
||||||
|
auto range = boost::make_iterator_range(byTer.begin(),
|
||||||
|
byTer.end());
|
||||||
|
auto tRange = boost::adaptors::transform(
|
||||||
|
range,
|
||||||
|
[](auto const& r)
|
||||||
|
{
|
||||||
|
return std::make_pair(r.second.first, r.first);
|
||||||
|
}
|
||||||
|
);
|
||||||
|
std::unordered_map<
|
||||||
|
std::string,
|
||||||
|
std::underlying_type_t<TER>> const
|
||||||
|
byToken(tRange.begin(), tRange.end());
|
||||||
|
return byToken;
|
||||||
|
}();
|
||||||
|
|
||||||
|
auto const r = results.find(token);
|
||||||
|
|
||||||
|
if (r == results.end())
|
||||||
|
return boost::none;
|
||||||
|
|
||||||
|
return static_cast<TER>(r->second);
|
||||||
|
}
|
||||||
|
|
||||||
} // ripple
|
} // ripple
|
||||||
|
|||||||
@@ -19,6 +19,7 @@
|
|||||||
|
|
||||||
#include <BeastConfig.h>
|
#include <BeastConfig.h>
|
||||||
#include <ripple/basics/Log.h>
|
#include <ripple/basics/Log.h>
|
||||||
|
#include <ripple/protocol/JsonFields.h>
|
||||||
#include <ripple/protocol/SecretKey.h>
|
#include <ripple/protocol/SecretKey.h>
|
||||||
#include <ripple/protocol/st.h>
|
#include <ripple/protocol/st.h>
|
||||||
#include <ripple/json/json_reader.h>
|
#include <ripple/json/json_reader.h>
|
||||||
@@ -88,6 +89,7 @@ public:
|
|||||||
if (parsedOK)
|
if (parsedOK)
|
||||||
{
|
{
|
||||||
STParsedJSONObject parsed ("test", jsonObject);
|
STParsedJSONObject parsed ("test", jsonObject);
|
||||||
|
BEAST_EXPECT(parsed.object);
|
||||||
std::string const& serialized (
|
std::string const& serialized (
|
||||||
to_string (parsed.object->getJson(0)));
|
to_string (parsed.object->getJson(0)));
|
||||||
BEAST_EXPECT(serialized == json);
|
BEAST_EXPECT(serialized == json);
|
||||||
@@ -98,6 +100,138 @@ public:
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void testParseJSONEdgeCases()
|
||||||
|
{
|
||||||
|
testcase("parse json object");
|
||||||
|
|
||||||
|
{
|
||||||
|
std::string const goodJson(
|
||||||
|
R"({"CloseResolution":19,"Method":250,)"
|
||||||
|
R"("TransactionResult":"tecFROZEN"})");
|
||||||
|
|
||||||
|
Json::Value jv;
|
||||||
|
if (BEAST_EXPECT(parseJSONString(goodJson, jv)))
|
||||||
|
{
|
||||||
|
STParsedJSONObject parsed("test", jv);
|
||||||
|
if (BEAST_EXPECT(parsed.object))
|
||||||
|
{
|
||||||
|
std::string const& serialized(
|
||||||
|
to_string(parsed.object->getJson(0)));
|
||||||
|
BEAST_EXPECT(serialized == goodJson);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
{
|
||||||
|
std::string const goodJson(
|
||||||
|
R"({"CloseResolution":19,"Method":"250",)"
|
||||||
|
R"("TransactionResult":"tecFROZEN"})");
|
||||||
|
std::string const expectedJson(
|
||||||
|
R"({"CloseResolution":19,"Method":250,)"
|
||||||
|
R"("TransactionResult":"tecFROZEN"})");
|
||||||
|
|
||||||
|
Json::Value jv;
|
||||||
|
if (BEAST_EXPECT(parseJSONString(goodJson, jv)))
|
||||||
|
{
|
||||||
|
// Integer values are always parsed as int,
|
||||||
|
// unless they're too big. We want a small uint.
|
||||||
|
jv["CloseResolution"] = Json::UInt(19);
|
||||||
|
STParsedJSONObject parsed("test", jv);
|
||||||
|
if (BEAST_EXPECT(parsed.object))
|
||||||
|
{
|
||||||
|
std::string const& serialized(
|
||||||
|
to_string(parsed.object->getJson(0)));
|
||||||
|
BEAST_EXPECT(serialized == expectedJson);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
{
|
||||||
|
std::string const json(
|
||||||
|
R"({"CloseResolution":19,"Method":250,)"
|
||||||
|
R"("TransactionResult":"terQUEUED"})");
|
||||||
|
|
||||||
|
Json::Value jv;
|
||||||
|
if (BEAST_EXPECT(parseJSONString(json, jv)))
|
||||||
|
{
|
||||||
|
STParsedJSONObject parsed("test", jv);
|
||||||
|
BEAST_EXPECT(!parsed.object);
|
||||||
|
BEAST_EXPECT(parsed.error);
|
||||||
|
BEAST_EXPECT(parsed.error[jss::error] == "invalidParams");
|
||||||
|
BEAST_EXPECT(parsed.error[jss::error_message] ==
|
||||||
|
"Field 'test.TransactionResult' is out of range.");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
{
|
||||||
|
std::string const json(
|
||||||
|
R"({"CloseResolution":19,"Method":"pony",)"
|
||||||
|
R"("TransactionResult":"tesSUCCESS"})");
|
||||||
|
|
||||||
|
Json::Value jv;
|
||||||
|
if (BEAST_EXPECT(parseJSONString(json, jv)))
|
||||||
|
{
|
||||||
|
STParsedJSONObject parsed("test", jv);
|
||||||
|
BEAST_EXPECT(!parsed.object);
|
||||||
|
BEAST_EXPECT(parsed.error);
|
||||||
|
BEAST_EXPECT(parsed.error[jss::error] == "invalidParams");
|
||||||
|
BEAST_EXPECT(parsed.error[jss::error_message] ==
|
||||||
|
"Field 'test.Method' has bad type.");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
{
|
||||||
|
std::string const json(
|
||||||
|
R"({"CloseResolution":19,"Method":3294967296,)"
|
||||||
|
R"("TransactionResult":"tesSUCCESS"})");
|
||||||
|
|
||||||
|
Json::Value jv;
|
||||||
|
if (BEAST_EXPECT(parseJSONString(json, jv)))
|
||||||
|
{
|
||||||
|
STParsedJSONObject parsed("test", jv);
|
||||||
|
BEAST_EXPECT(!parsed.object);
|
||||||
|
BEAST_EXPECT(parsed.error);
|
||||||
|
BEAST_EXPECT(parsed.error[jss::error] == "invalidParams");
|
||||||
|
BEAST_EXPECT(parsed.error[jss::error_message] ==
|
||||||
|
"Field 'test.Method' is out of range.");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
{
|
||||||
|
std::string const json(
|
||||||
|
R"({"CloseResolution":-10,"Method":42,)"
|
||||||
|
R"("TransactionResult":"tesSUCCESS"})");
|
||||||
|
|
||||||
|
Json::Value jv;
|
||||||
|
if (BEAST_EXPECT(parseJSONString(json, jv)))
|
||||||
|
{
|
||||||
|
STParsedJSONObject parsed("test", jv);
|
||||||
|
BEAST_EXPECT(!parsed.object);
|
||||||
|
BEAST_EXPECT(parsed.error);
|
||||||
|
BEAST_EXPECT(parsed.error[jss::error] == "invalidParams");
|
||||||
|
BEAST_EXPECT(parsed.error[jss::error_message] ==
|
||||||
|
"Field 'test.CloseResolution' is out of range.");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
{
|
||||||
|
std::string const json(
|
||||||
|
R"({"CloseResolution":19,"Method":3.141592653,)"
|
||||||
|
R"("TransactionResult":"tesSUCCESS"})");
|
||||||
|
|
||||||
|
Json::Value jv;
|
||||||
|
if (BEAST_EXPECT(parseJSONString(json, jv)))
|
||||||
|
{
|
||||||
|
STParsedJSONObject parsed("test", jv);
|
||||||
|
BEAST_EXPECT(!parsed.object);
|
||||||
|
BEAST_EXPECT(parsed.error);
|
||||||
|
BEAST_EXPECT(parsed.error[jss::error] == "invalidParams");
|
||||||
|
BEAST_EXPECT(parsed.error[jss::error_message] ==
|
||||||
|
"Field 'test.Method' has bad type.");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
void testSerialization ()
|
void testSerialization ()
|
||||||
{
|
{
|
||||||
testcase ("serialization");
|
testcase ("serialization");
|
||||||
@@ -510,6 +644,7 @@ public:
|
|||||||
testSerialization();
|
testSerialization();
|
||||||
testParseJSONArray();
|
testParseJSONArray();
|
||||||
testParseJSONArrayWithInvalidChildrenObjects();
|
testParseJSONArrayWithInvalidChildrenObjects();
|
||||||
|
testParseJSONEdgeCases();
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|||||||
62
src/test/protocol/TER_test.cpp
Normal file
62
src/test/protocol/TER_test.cpp
Normal file
@@ -0,0 +1,62 @@
|
|||||||
|
//------------------------------------------------------------------------------
|
||||||
|
/*
|
||||||
|
This file is part of rippled: https://github.com/ripple/rippled
|
||||||
|
Copyright (c) 2012, 2013 Ripple Labs Inc.
|
||||||
|
|
||||||
|
Permission to use, copy, modify, and/or distribute this software for any
|
||||||
|
purpose with or without fee is hereby granted, provided that the above
|
||||||
|
copyright notice and this permission notice appear in all copies.
|
||||||
|
|
||||||
|
THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
|
||||||
|
WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
|
||||||
|
MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
|
||||||
|
ANY SPECIAL , DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
|
||||||
|
WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
|
||||||
|
ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
|
||||||
|
OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
|
||||||
|
*/
|
||||||
|
//==============================================================================
|
||||||
|
|
||||||
|
#include <BeastConfig.h>
|
||||||
|
#include <ripple/protocol/TER.h>
|
||||||
|
#include <ripple/beast/unit_test.h>
|
||||||
|
|
||||||
|
namespace ripple {
|
||||||
|
|
||||||
|
struct TER_test : public beast::unit_test::suite
|
||||||
|
{
|
||||||
|
void
|
||||||
|
testTransResultInfo()
|
||||||
|
{
|
||||||
|
for (auto i = -400; i < 400; ++i)
|
||||||
|
{
|
||||||
|
TER t = static_cast<TER>(i);
|
||||||
|
auto inRange = isTelLocal(t) ||
|
||||||
|
isTemMalformed(t) ||
|
||||||
|
isTefFailure(t) ||
|
||||||
|
isTerRetry(t) ||
|
||||||
|
isTesSuccess(t) ||
|
||||||
|
isTecClaim(t);
|
||||||
|
|
||||||
|
std::string token, text;
|
||||||
|
auto good = transResultInfo(t, token, text);
|
||||||
|
BEAST_EXPECT(inRange || !good);
|
||||||
|
BEAST_EXPECT(transToken(t) == (good ? token : "-"));
|
||||||
|
BEAST_EXPECT(transHuman(t) == (good ? text : "-"));
|
||||||
|
|
||||||
|
auto code = transCode(token);
|
||||||
|
BEAST_EXPECT(good == !!code);
|
||||||
|
BEAST_EXPECT(!code || *code == t);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
run() override
|
||||||
|
{
|
||||||
|
testTransResultInfo();
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
BEAST_DEFINE_TESTSUITE(TER,protocol,ripple);
|
||||||
|
|
||||||
|
}
|
||||||
@@ -31,5 +31,6 @@
|
|||||||
#include <test/protocol/STAmount_test.cpp>
|
#include <test/protocol/STAmount_test.cpp>
|
||||||
#include <test/protocol/STObject_test.cpp>
|
#include <test/protocol/STObject_test.cpp>
|
||||||
#include <test/protocol/STTx_test.cpp>
|
#include <test/protocol/STTx_test.cpp>
|
||||||
|
#include <test/protocol/TER_test.cpp>
|
||||||
#include <test/protocol/types_test.cpp>
|
#include <test/protocol/types_test.cpp>
|
||||||
#include <test/protocol/XRPAmount_test.cpp>
|
#include <test/protocol/XRPAmount_test.cpp>
|
||||||
Reference in New Issue
Block a user