peerplays-fc/src/error_report.cpp

142 lines
4 KiB
C++
Raw Normal View History

#include <fc/error_report.hpp>
2012-12-12 18:41:33 +00:00
#include <fc/filesystem.hpp>
#include <fc/sstream.hpp>
#include <fc/value.hpp>
#include <fc/json.hpp>
namespace fc {
error_frame::error_frame( const fc::string& f, uint64_t l, const fc::string& m, const fc::string& d, fc::value met )
2012-12-12 18:41:33 +00:00
:desc(d),file(fc::path(f).filename().generic_string()),line(l),method(m),meta(fc::move(met)){}
error_report::error_report()
{
}
error_frame::error_frame(const fc::error_frame& e)
:desc(e.desc),file(e.file),line(e.line),method(e.method),time(e.time),meta(e.meta){}
error_frame::error_frame(fc::error_frame&& e)
:desc(fc::move(e.desc)),
file(fc::move(e.file)),
line(e.line),
method(fc::move(e.method)),
time(e.time),
meta(fc::move(e.meta))
{}
fc::error_frame& fc::error_frame::operator=(const fc::error_frame& f ) {
auto tmp = f;
fc_swap( tmp, *this );
return *this;
}
fc::error_frame& fc::error_frame::operator=(fc::error_frame&& e )
{
desc=fc::move(e.desc);
file=fc::move(e.file);
line=fc::move(e.line);
method=fc::move(e.method);
time=e.time;
meta=fc::move(e.meta);
return *this;
}
error_report::error_report( const fc::string& file, uint64_t line, const fc::string& method, const fc::string& desc, fc::value meta )
{
push_frame( file, line, method, desc, meta );
}
fc::error_frame& error_report::current()
{
if( !stack.size() ) stack.resize(1);
return stack.back();
}
fc::error_report& error_report::pop_frame()
{
stack.pop_back();
return *this;
}
fc::error_report& error_report::push_frame( const fc::string& file, uint64_t line, const fc::string& method, const fc::string& desc, fc::value meta )
{
stack.push_back( fc::error_frame( file, line, method, desc, meta ) );
return *this;
}
fc::error_report& error_report::append( const error_report& e )
{
// TODO: what to do about the 'failure...?'
stack.reserve( stack.size()+e.stack.size());
for( uint32_t i = 0; i < e.stack.size(); ++i ) {
stack.push_back( e.stack[i] );
}
return *this;
}
fc::string error_frame::to_detail_string()const {
fc::stringstream ss;
ss << to_string() << "\n\t";
ss << file << ":" << line << "\t"<<method;
if( meta ) ss << "\t" <<fc::json::to_string(*meta);
return ss.str();
}
fc::string error_frame::to_string()const {
fc::stringstream ss;
int64_t prev = 0;
auto next = desc.find( '$' );
while( prev != int64_t(fc::string::npos) && prev < int64_t(desc.size()) ) {
// print everything from the last pos until the first '$'
ss << desc.substr( prev, next );
// if we got to the end, return it.
if( next == string::npos ) { return ss.str(); }
// if we are not at the end, then update the start
prev = next + 1;
if( desc[prev] == '{' ) {
// if the next char is a open, then find close
next = desc.find( '}', prev );
// if we found close...
if( next != fc::string::npos ) {
// the key is between prev and next
fc::string key = desc.substr( prev+1, (next-prev-1) );
if( meta ) {
auto itr = meta->find( key.c_str() );
if( itr != meta->end() ) {
ss << fc::json::to_string( itr->val );
} else {
ss << "???";
}
}
prev = next + 1;
// find the next $
next = desc.find( '$', prev );
} else {
// we didn't find it.. continue to while...
}
} else {
ss << desc[prev];
++prev;
next = desc.find( '$', prev );
}
}
return ss.str();
}
fc::string error_report::to_string()const {
fc::stringstream ss;
for( int i = 0; i < stack.size(); ++i ) {
ss << stack[i].to_string() << "\n";
}
return ss.str();
}
fc::string error_report::to_detail_string()const {
fc::stringstream ss;
for( int i = 0; i < stack.size(); ++i ) {
ss << stack[i].to_detail_string() << "\n";
}
return ss.str();
}
} // namespace fc