/* This file is part of MAUS: http://micewww.pp.rl.ac.uk:8080/projects/maus
*
* MAUS is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* MAUS is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with MAUS. If not, see .
*
*/
#ifndef _SRC_COMMON_CPP_API_MAPBASE_INL_
#define _SRC_COMMON_CPP_API_MAPBASE_INL_
#include
#include "Utils/Squeak.hh"
#include "src/common_cpp/API/APIExceptions.hh"
#include "src/common_cpp/Utils/PyObjectWrapper.hh"
#include "src/common_cpp/Utils/Exception.hh"
#include "src/common_cpp/Utils/CppErrorHandler.hh"
#include "src/common_cpp/Converter/ConverterFactory.hh"
namespace MAUS {
template
MapBase::MapBase(const std::string& s) :
IMap(), ModuleBase(s) {}
template
MapBase::MapBase(const MapBase& mb) :
IMap(), ModuleBase(mb._classname) {}
template
MapBase::~MapBase() {}
template
PyObject* MapBase::process_pyobj(PyObject* py_input) const {
// this function owns cpp_data; py_input is still owned by caller
TYPE* cpp_data = NULL;
try {
cpp_data = PyObjectWrapper::unwrap(py_input);
_process(cpp_data);
} catch (MAUS::Exceptions::Exception& exc) {
Squeak::mout(Squeak::debug) << "Stack trace:" << exc.GetStackTrace()
<< std::endl;
HandleException(&cpp_data, &exc, _classname);
} catch (std::exception& exc) {
HandleException(&cpp_data, &exc, _classname);
} catch (...) {
throw Exceptions::Exception(Exceptions::recoverable,
_classname+" threw an unhandled exception",
"MapBase::process_pyobj");
}
PyObject* py_output = PyObjectWrapper::wrap(cpp_data);
// py_output now owns cpp_data
return py_output;
}
// In case evaluation of the spill fails, we can make an empty spill so that
// error information propagates okay
template
Json::Value* MapBase::EmptyJsonSpill() const {
Json::Value* val;
MAUS::Data data_temp;
data_temp.SetSpill(new Spill());
val = ConverterFactory().convert(&data_temp);
return val;
}
// Need some wrapper code for the exception handler as the interface is
// assumed to be json. As conversion is error prone, need to be a bit careful
// and do the conversion here.
template
void MapBase::HandleException(TYPE** data,
std::exception* exc,
std::string class_name) const {
if (!data) {
throw *exc;
}
Json::Value* val = NULL;
try {
try {
val = ConverterFactory().convert(*data);
} catch (...) {
// do nothing; catch data == NULL or failed conversion to json
}
delete *data; // we don't need it any more
if (val == NULL) { // conversion failed, try to build from scratch
val = EmptyJsonSpill();
}
try {
*val = CppErrorHandler::getInstance()->HandleStdExc(*val, *exc, class_name);
// Rogers: found a weird case where json failed to parse inf at
// this point; so fall back to an empty spill in that case
*data = ConverterFactory().convert(val);
} catch (...) {
val = EmptyJsonSpill();
*val = CppErrorHandler::getInstance()->HandleStdExc(*val, *exc, class_name);
*data = ConverterFactory().convert(val);
}
delete val;
} catch (...) {
std::cerr << "Unhandled exception" << std::endl;
throw;
}
}
}// end of namespace
#endif