Error reporting: report Traffic errors
Also add logging to Sentry.io
This commit is contained in:
parent
251b3aeb2b
commit
fbbcc75229
6 changed files with 94 additions and 50 deletions
|
@ -638,6 +638,9 @@ bool FGAIBase::init(ModelSearchOrder searchOrder)
|
|||
_modeldata->addErrorContext("ai", _name);
|
||||
_modeldata->captureErrorContext("scenario-path");
|
||||
|
||||
// set by FGAISchedule::createAIAircraft
|
||||
_modeldata->captureErrorContext("traffic-aircraft-callsign");
|
||||
|
||||
if (_otype == otMultiplayer) {
|
||||
_modeldata->addErrorContext("multiplayer", getCallSign());
|
||||
}
|
||||
|
|
|
@ -38,6 +38,7 @@
|
|||
#include <Main/fg_props.hxx>
|
||||
#include <Main/globals.hxx>
|
||||
#include <Main/locale.hxx>
|
||||
#include <Main/sentryIntegration.hxx>
|
||||
#include <Scripting/NasalClipboard.hxx> // clipboard access
|
||||
|
||||
using std::string;
|
||||
|
@ -69,8 +70,9 @@ enum class Aggregation {
|
|||
InputDevice,
|
||||
FGData,
|
||||
MultiPlayer,
|
||||
Unknown, ///< error coudln't be attributed more specifcially
|
||||
OutOfMemory ///< seperate category to give it a custom message
|
||||
Unknown, ///< error coudln't be attributed more specifcially
|
||||
OutOfMemory, ///< seperate category to give it a custom message
|
||||
Traffic
|
||||
};
|
||||
|
||||
// these should correspond to simgear::ErrorCode enum
|
||||
|
@ -111,7 +113,8 @@ string_list static_categoryIds = {
|
|||
"error-category-fgdata",
|
||||
"error-category-multiplayer",
|
||||
"error-category-unknown",
|
||||
"error-category-out-of-memory"};
|
||||
"error-category-out-of-memory",
|
||||
"error-category-traffic"};
|
||||
|
||||
class RecentLogCallback : public simgear::LogCallback
|
||||
{
|
||||
|
@ -234,6 +237,7 @@ public:
|
|||
|
||||
void collectError(simgear::LoadFailure type, simgear::ErrorCode code, const std::string& details, const sg_location& location)
|
||||
{
|
||||
SG_LOG(SG_GENERAL, SG_WARN, "Error:" << static_errorTypeIds.at(static_cast<int>(type)) << " from " << static_errorIds.at(static_cast<int>(code)) << "::" << details << "\n\t" << location.asString());
|
||||
ErrorOcurrence occurrence{code, type, details, location, time(nullptr)};
|
||||
|
||||
// snapshot the top of the context stacks into our occurence data
|
||||
|
@ -322,6 +326,8 @@ public:
|
|||
});
|
||||
assert(it != _aggregated.end());
|
||||
_activeReportIndex = std::distance(_aggregated.begin(), it);
|
||||
|
||||
flightgear::sentryReportUserError(static_categoryIds.at(catId), detailsTextStream.str());
|
||||
}
|
||||
|
||||
void writeReportToStream(const AggregateReport& report, std::ostream& os) const;
|
||||
|
@ -355,6 +361,16 @@ auto ErrorReporter::ErrorReporterPrivate::getAggregateForOccurence(const ErrorRe
|
|||
return getAggregate(Aggregation::MultiPlayer, {});
|
||||
}
|
||||
|
||||
// traffic cases: need to handle errors in the traffic files (schedule, rwyuse)
|
||||
// but also errors loading aircraft models associated with traffic
|
||||
if (oc.code == simgear::ErrorCode::AITrafficSchedule) {
|
||||
return getAggregate(Aggregation::Traffic, {});
|
||||
}
|
||||
|
||||
if (oc.hasContextKey("traffic-aircraft-callsign")) {
|
||||
return getAggregate(Aggregation::Traffic, {});
|
||||
}
|
||||
|
||||
// all TerraSync coded errors go there: this is errors for the
|
||||
// actual download process (eg, failed to write to disk)
|
||||
if (oc.code == simgear::ErrorCode::TerraSync) {
|
||||
|
|
|
@ -485,6 +485,26 @@ void sentryThreadReportXMLErrors(bool report)
|
|||
perThread_reportXMLParseErrors = report;
|
||||
}
|
||||
|
||||
void sentryReportUserError(const std::string& aggregate, const std::string& details)
|
||||
{
|
||||
if (!static_sentryEnabled)
|
||||
return;
|
||||
|
||||
sentry_value_t sentryMessage = sentry_value_new_object();
|
||||
sentry_value_set_by_key(sentryMessage, "type", sentry_value_new_string("Error"));
|
||||
|
||||
sentry_value_t info = sentry_value_new_object();
|
||||
sentry_value_set_by_key(info, "details", sentry_value_new_string(details.c_str()));
|
||||
|
||||
sentry_set_context("what", info);
|
||||
|
||||
sentry_value_set_by_key(sentryMessage, "value", sentry_value_new_string(aggregate.c_str()));
|
||||
sentry_value_t event = sentry_value_new_event();
|
||||
sentry_value_set_by_key(event, "message", sentryMessage);
|
||||
|
||||
sentry_capture_event(event);
|
||||
}
|
||||
|
||||
} // of namespace
|
||||
|
||||
#else
|
||||
|
@ -535,6 +555,10 @@ void sentryThreadReportXMLErrors(bool)
|
|||
{
|
||||
}
|
||||
|
||||
void sentryReportUserError(const std::string&, const std::string&)
|
||||
{
|
||||
}
|
||||
|
||||
} // of namespace
|
||||
|
||||
#endif
|
||||
|
|
|
@ -43,6 +43,7 @@ void sentryReportException(const std::string& msg, const std::string& location =
|
|||
|
||||
void sentryReportFatalError(const std::string& msg, const std::string& more = {});
|
||||
|
||||
void sentryReportUserError(const std::string& aggregate, const std::string& details);
|
||||
/**
|
||||
* @brief helper to allow per-thread supression of
|
||||
* error reports based on XML parse/include errors.
|
||||
|
|
|
@ -38,12 +38,13 @@
|
|||
#include <algorithm>
|
||||
|
||||
#include <simgear/compiler.h>
|
||||
#include <simgear/sg_inlines.h>
|
||||
#include <simgear/debug/ErrorReportingCallback.hxx>
|
||||
#include <simgear/math/sg_geodesy.hxx>
|
||||
#include <simgear/props/props.hxx>
|
||||
#include <simgear/sg_inlines.h>
|
||||
#include <simgear/structure/subsystem_mgr.hxx>
|
||||
#include <simgear/xml/easyxml.hxx>
|
||||
#include <simgear/timing/sg_time.hxx>
|
||||
#include <simgear/xml/easyxml.hxx>
|
||||
|
||||
#include <AIModel/AIFlightPlan.hxx>
|
||||
#include <AIModel/AIManager.hxx>
|
||||
|
@ -380,18 +381,21 @@ bool FGAISchedule::createAIAircraft(FGScheduledFlight* flight, double speedKnots
|
|||
speedKnots, flightType, acType,
|
||||
airline));
|
||||
if (fp->isValidPlan()) {
|
||||
aiAircraft->FGAIBase::setFlightPlan(std::move(fp));
|
||||
globals->get_subsystem<FGAIManager>()->attach(aiAircraft);
|
||||
if (aiAircraft->_getProps()) {
|
||||
SGPropertyNode *nodeForAircraft = aiAircraft->_getProps();
|
||||
if (dep) {
|
||||
nodeForAircraft->getChild("departure-airport-id", 0, true)->setStringValue(dep->getId());
|
||||
nodeForAircraft->getChild("departure-time-sec", 0, true)->setIntValue(deptime);
|
||||
}
|
||||
if (arr) {
|
||||
nodeForAircraft->getChild("arrival-airport-id", 0, true)->setStringValue(arr->getId());
|
||||
// arrival time not known here
|
||||
}
|
||||
// set this here so it's available inside attach, which calls AIBase::init
|
||||
simgear::ErrorReportContext ec{"traffic-aircraft-callsign", flight->getCallSign()};
|
||||
|
||||
aiAircraft->FGAIBase::setFlightPlan(std::move(fp));
|
||||
globals->get_subsystem<FGAIManager>()->attach(aiAircraft);
|
||||
if (aiAircraft->_getProps()) {
|
||||
SGPropertyNode* nodeForAircraft = aiAircraft->_getProps();
|
||||
if (dep) {
|
||||
nodeForAircraft->getChild("departure-airport-id", 0, true)->setStringValue(dep->getId());
|
||||
nodeForAircraft->getChild("departure-time-sec", 0, true)->setIntValue(deptime);
|
||||
}
|
||||
if (arr) {
|
||||
nodeForAircraft->getChild("arrival-airport-id", 0, true)->setStringValue(arr->getId());
|
||||
// arrival time not known here
|
||||
}
|
||||
}
|
||||
return true;
|
||||
} else {
|
||||
|
|
|
@ -323,7 +323,7 @@ private:
|
|||
|
||||
if (!FGAISchedule::validModelPath(mdl)) {
|
||||
missingModels.insert(mdl);
|
||||
SG_LOG(SG_AI, SG_DEV_WARN, "TrafficMgr: Missing model path:" << mdl);
|
||||
simgear::reportFailure(simgear::LoadFailure::NotFound, simgear::ErrorCode::AITrafficSchedule, "Missing traffic model path:" + mdl, _currentFile);
|
||||
requiredAircraft = homePort = "";
|
||||
return;
|
||||
}
|
||||
|
@ -387,6 +387,7 @@ private:
|
|||
SG_LOG(SG_AI, SG_DEBUG, "parsing traffic in:" << p);
|
||||
simgear::PathList trafficFiles = d2.children(simgear::Dir::TYPE_FILE, ".xml");
|
||||
for (const auto& xml : trafficFiles) {
|
||||
_currentFile = xml;
|
||||
try {
|
||||
readXML(xml, *this);
|
||||
if (_cancelThread) {
|
||||
|
@ -395,7 +396,6 @@ private:
|
|||
} catch (sg_exception& e) {
|
||||
simgear::reportFailure(simgear::LoadFailure::BadData, simgear::ErrorCode::AITrafficSchedule,
|
||||
"XML errors parsinng traffic:" + e.getFormattedMessage(), xml);
|
||||
SG_LOG(SG_AI, SG_WARN, "XML error parsing traffic file:" << xml << "\n\t" << e.getFormattedMessage());
|
||||
}
|
||||
}
|
||||
} // of sub-directories iteration
|
||||
|
@ -408,21 +408,22 @@ private:
|
|||
bool _isFinished;
|
||||
bool _cancelThread;
|
||||
simgear::PathList _trafficDirPaths;
|
||||
SGPath _currentFile;
|
||||
|
||||
// parser state
|
||||
// parser state
|
||||
|
||||
string_list elementValueStack;
|
||||
// record model paths which are missing, to avoid duplicate
|
||||
// warnings when parsing traffic schedules.
|
||||
std::set<std::string> missingModels;
|
||||
string_list elementValueStack;
|
||||
// record model paths which are missing, to avoid duplicate
|
||||
// warnings when parsing traffic schedules.
|
||||
std::set<std::string> missingModels;
|
||||
|
||||
std::string mdl, livery, registration, callsign, fltrules,
|
||||
port, timeString, departurePort, departureTime, arrivalPort, arrivalTime,
|
||||
repeat, acType, airline, m_class, flighttype, requiredAircraft, homePort;
|
||||
int cruiseAlt;
|
||||
int score, acCounter;
|
||||
double radius, offset;
|
||||
bool heavy;
|
||||
std::string mdl, livery, registration, callsign, fltrules,
|
||||
port, timeString, departurePort, departureTime, arrivalPort, arrivalTime,
|
||||
repeat, acType, airline, m_class, flighttype, requiredAircraft, homePort;
|
||||
int cruiseAlt;
|
||||
int score, acCounter;
|
||||
double radius, offset;
|
||||
bool heavy;
|
||||
|
||||
};
|
||||
|
||||
|
@ -828,26 +829,21 @@ void FGTrafficManager::readTimeTableFromFile(SGPath infileName)
|
|||
offset = atof(tokens[7].c_str());;
|
||||
|
||||
if (!FGAISchedule::validModelPath(model)) {
|
||||
SG_LOG(SG_AI, SG_WARN, "TrafficMgr: Missing model path:" <<
|
||||
model << " from " << infileName);
|
||||
simgear::reportFailure(simgear::LoadFailure::NotFound, simgear::ErrorCode::AITrafficSchedule, "Missing traffic model path:" + model, infileName);
|
||||
} else {
|
||||
|
||||
SG_LOG(SG_AI, SG_INFO, "Adding Aircraft" << model << " " << livery << " " << homePort << " "
|
||||
<< registration << " " << flightReq << " " << isHeavy
|
||||
<< " " << acType << " " << airline << " " << m_class
|
||||
<< " " << FlightType << " " << radius << " " << offset);
|
||||
scheduledAircraft.push_back(new FGAISchedule(model,
|
||||
livery,
|
||||
homePort,
|
||||
registration,
|
||||
flightReq,
|
||||
isHeavy,
|
||||
acType,
|
||||
airline,
|
||||
m_class,
|
||||
FlightType,
|
||||
radius,
|
||||
offset));
|
||||
SG_LOG(SG_AI, SG_DEBUG, "Adding Aircraft" << model << " " << livery << " " << homePort << " " << registration << " " << flightReq << " " << isHeavy << " " << acType << " " << airline << " " << m_class << " " << FlightType << " " << radius << " " << offset);
|
||||
scheduledAircraft.push_back(new FGAISchedule(model,
|
||||
livery,
|
||||
homePort,
|
||||
registration,
|
||||
flightReq,
|
||||
isHeavy,
|
||||
acType,
|
||||
airline,
|
||||
m_class,
|
||||
FlightType,
|
||||
radius,
|
||||
offset));
|
||||
} // of valid model path
|
||||
}
|
||||
if (tokens[0] == string("FLIGHT")) {
|
||||
|
|
Loading…
Add table
Reference in a new issue