Add json logger to map_matching
This adds additional data to the json response, when OSRM is compiled in debug mode.
This commit is contained in:
parent
34d5d353af
commit
a760aec791
@ -40,6 +40,7 @@ SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|||||||
#include "../util/compute_angle.hpp"
|
#include "../util/compute_angle.hpp"
|
||||||
#include "../util/integer_range.hpp"
|
#include "../util/integer_range.hpp"
|
||||||
#include "../util/simple_logger.hpp"
|
#include "../util/simple_logger.hpp"
|
||||||
|
#include "../util/json_logger.hpp"
|
||||||
#include "../util/string_util.hpp"
|
#include "../util/string_util.hpp"
|
||||||
|
|
||||||
#include <cstdlib>
|
#include <cstdlib>
|
||||||
@ -237,9 +238,10 @@ template <class DataFacadeT> class MapMatchingPlugin : public BasePlugin
|
|||||||
}
|
}
|
||||||
|
|
||||||
// call the actual map matching
|
// call the actual map matching
|
||||||
osrm::json::Object debug_info;
|
if (osrm::json::Logger::get())
|
||||||
|
osrm::json::Logger::get()->initialize("matching");
|
||||||
Matching::SubMatchingList sub_matchings;
|
Matching::SubMatchingList sub_matchings;
|
||||||
search_engine_ptr->map_matching(candidates_lists, input_coords, input_timestamps, sub_matchings, debug_info);
|
search_engine_ptr->map_matching(candidates_lists, input_coords, input_timestamps, sub_matchings);
|
||||||
|
|
||||||
if (1 > sub_matchings.size())
|
if (1 > sub_matchings.size())
|
||||||
{
|
{
|
||||||
@ -283,7 +285,8 @@ template <class DataFacadeT> class MapMatchingPlugin : public BasePlugin
|
|||||||
matchings.values.emplace_back(submatchingToJSON(sub, route_parameters, raw_route));
|
matchings.values.emplace_back(submatchingToJSON(sub, route_parameters, raw_route));
|
||||||
}
|
}
|
||||||
|
|
||||||
json_result.values["debug"] = debug_info;
|
if (osrm::json::Logger::get())
|
||||||
|
osrm::json::Logger::get()->render("matching", json_result);
|
||||||
json_result.values["matchings"] = matchings;
|
json_result.values["matchings"] = matchings;
|
||||||
|
|
||||||
return 200;
|
return 200;
|
||||||
|
@ -33,6 +33,7 @@ SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|||||||
#include "../data_structures/coordinate_calculation.hpp"
|
#include "../data_structures/coordinate_calculation.hpp"
|
||||||
#include "../util/simple_logger.hpp"
|
#include "../util/simple_logger.hpp"
|
||||||
#include "../util/json_util.hpp"
|
#include "../util/json_util.hpp"
|
||||||
|
#include "../util/json_logger.hpp"
|
||||||
|
|
||||||
#include <osrm/json_container.hpp>
|
#include <osrm/json_container.hpp>
|
||||||
#include <variant/variant.hpp>
|
#include <variant/variant.hpp>
|
||||||
@ -43,9 +44,6 @@ SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|||||||
#include <iomanip>
|
#include <iomanip>
|
||||||
#include <numeric>
|
#include <numeric>
|
||||||
|
|
||||||
using JSONVariantArray = mapbox::util::recursive_wrapper<osrm::json::Array>;
|
|
||||||
using JSONVariantObject = mapbox::util::recursive_wrapper<osrm::json::Object>;
|
|
||||||
|
|
||||||
namespace Matching
|
namespace Matching
|
||||||
{
|
{
|
||||||
|
|
||||||
@ -299,6 +297,119 @@ template <class DataFacadeT> class MapMatching final
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
struct DebugInfo
|
||||||
|
{
|
||||||
|
DebugInfo(const osrm::json::Logger* logger)
|
||||||
|
: logger(logger)
|
||||||
|
{
|
||||||
|
if (logger)
|
||||||
|
{
|
||||||
|
object = &logger->map->at("matching");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void initialize(const HiddenMarkovModel& model,
|
||||||
|
unsigned initial_timestamp,
|
||||||
|
const Matching::CandidateLists& candidates_list)
|
||||||
|
{
|
||||||
|
// json logger not enabled
|
||||||
|
if (!logger)
|
||||||
|
return;
|
||||||
|
|
||||||
|
osrm::json::Array states;
|
||||||
|
for (unsigned t = 0; t < candidates_list.size(); t++)
|
||||||
|
{
|
||||||
|
osrm::json::Array timestamps;
|
||||||
|
for (unsigned s = 0; s < candidates_list[t].size(); s++)
|
||||||
|
{
|
||||||
|
osrm::json::Object state;
|
||||||
|
state.values["transitions"] = osrm::json::Array();
|
||||||
|
state.values["coordinate"] = osrm::json::make_array(
|
||||||
|
candidates_list[t][s].first.location.lat / COORDINATE_PRECISION,
|
||||||
|
candidates_list[t][s].first.location.lon / COORDINATE_PRECISION);
|
||||||
|
if (t < initial_timestamp)
|
||||||
|
{
|
||||||
|
state.values["viterbi"] = osrm::json::clamp_float(Matching::IMPOSSIBLE_LOG_PROB);
|
||||||
|
state.values["pruned"] = 0u;
|
||||||
|
}
|
||||||
|
else if (t == initial_timestamp)
|
||||||
|
{
|
||||||
|
state.values["viterbi"] = osrm::json::clamp_float(model.viterbi[t][s]);
|
||||||
|
state.values["pruned"] = static_cast<unsigned>(model.pruned[t][s]);
|
||||||
|
}
|
||||||
|
timestamps.values.push_back(state);
|
||||||
|
}
|
||||||
|
states.values.push_back(timestamps);
|
||||||
|
}
|
||||||
|
osrm::json::get(*object, "states") = states;
|
||||||
|
}
|
||||||
|
|
||||||
|
void add_transition_info(const unsigned prev_t,
|
||||||
|
const unsigned current_t,
|
||||||
|
const unsigned prev_state,
|
||||||
|
const unsigned current_state,
|
||||||
|
const double prev_viterbi,
|
||||||
|
const double emission_pr,
|
||||||
|
const double transition_pr,
|
||||||
|
const double network_distance,
|
||||||
|
const double great_circle_distance)
|
||||||
|
{
|
||||||
|
// json logger not enabled
|
||||||
|
if (!logger)
|
||||||
|
return;
|
||||||
|
|
||||||
|
osrm::json::Object transistion;
|
||||||
|
transistion.values["to"] = osrm::json::make_array(current_t, current_state);
|
||||||
|
transistion.values["properties"] =
|
||||||
|
osrm::json::make_array(osrm::json::clamp_float(prev_viterbi),
|
||||||
|
osrm::json::clamp_float(emission_pr),
|
||||||
|
osrm::json::clamp_float(transition_pr),
|
||||||
|
network_distance,
|
||||||
|
great_circle_distance);
|
||||||
|
|
||||||
|
osrm::json::get(*object, "states", prev_t, prev_state, "transitions")
|
||||||
|
.get<mapbox::util::recursive_wrapper<osrm::json::Array>>()
|
||||||
|
.get().values.push_back(transistion);
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
void add_viterbi(const unsigned t,
|
||||||
|
const std::vector<double>& current_viterbi,
|
||||||
|
const std::vector<bool>& current_pruned)
|
||||||
|
{
|
||||||
|
// json logger not enabled
|
||||||
|
if (!logger)
|
||||||
|
return;
|
||||||
|
|
||||||
|
for (auto s_prime = 0u; s_prime < current_viterbi.size(); ++s_prime)
|
||||||
|
{
|
||||||
|
osrm::json::get(*object, "states", t, s_prime, "viterbi") = osrm::json::clamp_float(current_viterbi[s_prime]);
|
||||||
|
osrm::json::get(*object, "states", t, s_prime, "pruned") = static_cast<unsigned>(current_pruned[s_prime]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void add_chosen(const unsigned t, const unsigned s)
|
||||||
|
{
|
||||||
|
// json logger not enabled
|
||||||
|
if (!logger)
|
||||||
|
return;
|
||||||
|
|
||||||
|
osrm::json::get(*object, "states", t, s, "chosen") = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
void add_breakage(const std::vector<bool>& breakage)
|
||||||
|
{
|
||||||
|
// json logger not enabled
|
||||||
|
if (!logger)
|
||||||
|
return;
|
||||||
|
|
||||||
|
osrm::json::get(*object, "breakage") = osrm::json::make_array(breakage);
|
||||||
|
}
|
||||||
|
|
||||||
|
const osrm::json::Logger* logger;
|
||||||
|
osrm::json::Value* object;
|
||||||
|
};
|
||||||
|
|
||||||
public:
|
public:
|
||||||
MapMatching(DataFacadeT *facade, SearchEngineData &engine_working_data)
|
MapMatching(DataFacadeT *facade, SearchEngineData &engine_working_data)
|
||||||
: super(facade), engine_working_data(engine_working_data)
|
: super(facade), engine_working_data(engine_working_data)
|
||||||
@ -308,8 +419,7 @@ template <class DataFacadeT> class MapMatching final
|
|||||||
void operator()(const Matching::CandidateLists &candidates_list,
|
void operator()(const Matching::CandidateLists &candidates_list,
|
||||||
const std::vector<FixedPointCoordinate> &trace_coordinates,
|
const std::vector<FixedPointCoordinate> &trace_coordinates,
|
||||||
const std::vector<unsigned> &trace_timestamps,
|
const std::vector<unsigned> &trace_timestamps,
|
||||||
Matching::SubMatchingList &sub_matchings,
|
Matching::SubMatchingList &sub_matchings) const
|
||||||
osrm::json::Object &_debug_info) const
|
|
||||||
{
|
{
|
||||||
BOOST_ASSERT(candidates_list.size() > 0);
|
BOOST_ASSERT(candidates_list.size() > 0);
|
||||||
|
|
||||||
@ -321,32 +431,8 @@ template <class DataFacadeT> class MapMatching final
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
osrm::json::Array _debug_states;
|
DebugInfo debug(osrm::json::Logger::get());
|
||||||
for (unsigned t = 0; t < candidates_list.size(); t++)
|
debug.initialize(model, initial_timestamp, candidates_list);
|
||||||
{
|
|
||||||
osrm::json::Array _debug_timestamps;
|
|
||||||
for (unsigned s = 0; s < candidates_list[t].size(); s++)
|
|
||||||
{
|
|
||||||
osrm::json::Object _debug_state;
|
|
||||||
_debug_state.values["transitions"] = osrm::json::Array();
|
|
||||||
_debug_state.values["coordinate"] = osrm::json::makeArray(
|
|
||||||
candidates_list[t][s].first.location.lat / COORDINATE_PRECISION,
|
|
||||||
candidates_list[t][s].first.location.lon / COORDINATE_PRECISION);
|
|
||||||
if (t < initial_timestamp)
|
|
||||||
{
|
|
||||||
_debug_state.values["viterbi"] =
|
|
||||||
osrm::json::clampFloat(Matching::IMPOSSIBLE_LOG_PROB);
|
|
||||||
_debug_state.values["pruned"] = 0u;
|
|
||||||
}
|
|
||||||
else if (t == initial_timestamp)
|
|
||||||
{
|
|
||||||
_debug_state.values["viterbi"] = osrm::json::clampFloat(model.viterbi[t][s]);
|
|
||||||
_debug_state.values["pruned"] = static_cast<unsigned>(model.pruned[t][s]);
|
|
||||||
}
|
|
||||||
_debug_timestamps.values.push_back(_debug_state);
|
|
||||||
}
|
|
||||||
_debug_states.values.push_back(_debug_timestamps);
|
|
||||||
}
|
|
||||||
|
|
||||||
unsigned breakage_begin = std::numeric_limits<unsigned>::max();
|
unsigned breakage_begin = std::numeric_limits<unsigned>::max();
|
||||||
std::vector<unsigned> split_points;
|
std::vector<unsigned> split_points;
|
||||||
@ -400,24 +486,12 @@ template <class DataFacadeT> class MapMatching final
|
|||||||
const double transition_pr = log_transition_probability(d_t, beta);
|
const double transition_pr = log_transition_probability(d_t, beta);
|
||||||
new_value += transition_pr;
|
new_value += transition_pr;
|
||||||
|
|
||||||
osrm::json::Object _debug_transistion;
|
debug.add_transition_info(prev_unbroken_timestamp, t, s, s_prime,
|
||||||
_debug_transistion.values["to"] = osrm::json::makeArray(t, s_prime);
|
prev_viterbi[s],
|
||||||
_debug_transistion.values["properties"] =
|
emission_pr,
|
||||||
osrm::json::makeArray(osrm::json::clampFloat(prev_viterbi[s]),
|
transition_pr,
|
||||||
osrm::json::clampFloat(emission_pr),
|
|
||||||
osrm::json::clampFloat(transition_pr),
|
|
||||||
network_distance,
|
network_distance,
|
||||||
great_circle_distance);
|
great_circle_distance);
|
||||||
_debug_states.values[prev_unbroken_timestamp]
|
|
||||||
.get<JSONVariantArray>()
|
|
||||||
.get()
|
|
||||||
.values[s]
|
|
||||||
.get<JSONVariantObject>()
|
|
||||||
.get()
|
|
||||||
.values["transitions"]
|
|
||||||
.get<JSONVariantArray>()
|
|
||||||
.get()
|
|
||||||
.values.push_back(_debug_transistion);
|
|
||||||
|
|
||||||
if (new_value > current_viterbi[s_prime])
|
if (new_value > current_viterbi[s_prime])
|
||||||
{
|
{
|
||||||
@ -430,23 +504,7 @@ template <class DataFacadeT> class MapMatching final
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
for (auto s_prime = 0u; s_prime < current_viterbi.size(); ++s_prime)
|
debug.add_viterbi(t, current_viterbi, current_pruned);
|
||||||
{
|
|
||||||
_debug_states.values[t]
|
|
||||||
.get<JSONVariantArray>()
|
|
||||||
.get()
|
|
||||||
.values[s_prime]
|
|
||||||
.get<JSONVariantObject>()
|
|
||||||
.get()
|
|
||||||
.values["viterbi"] = osrm::json::clampFloat(current_viterbi[s_prime]);
|
|
||||||
_debug_states.values[t]
|
|
||||||
.get<JSONVariantArray>()
|
|
||||||
.get()
|
|
||||||
.values[s_prime]
|
|
||||||
.get<JSONVariantObject>()
|
|
||||||
.get()
|
|
||||||
.values["pruned"] = static_cast<unsigned>(current_pruned[s_prime]);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (model.breakage[t])
|
if (model.breakage[t])
|
||||||
{
|
{
|
||||||
@ -570,13 +628,7 @@ template <class DataFacadeT> class MapMatching final
|
|||||||
matching.nodes[i] = candidates_list[timestamp_index][location_index].first;
|
matching.nodes[i] = candidates_list[timestamp_index][location_index].first;
|
||||||
matching.length += model.path_lengths[timestamp_index][location_index];
|
matching.length += model.path_lengths[timestamp_index][location_index];
|
||||||
|
|
||||||
_debug_states.values[timestamp_index]
|
debug.add_chosen(timestamp_index, location_index);
|
||||||
.get<JSONVariantArray>()
|
|
||||||
.get()
|
|
||||||
.values[location_index]
|
|
||||||
.get<JSONVariantObject>()
|
|
||||||
.get()
|
|
||||||
.values["chosen"] = true;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
sub_matchings.push_back(matching);
|
sub_matchings.push_back(matching);
|
||||||
@ -584,14 +636,7 @@ template <class DataFacadeT> class MapMatching final
|
|||||||
sub_matching_begin = sub_matching_end;
|
sub_matching_begin = sub_matching_end;
|
||||||
}
|
}
|
||||||
|
|
||||||
osrm::json::Array _debug_breakage;
|
debug.add_breakage(model.breakage);
|
||||||
for (auto b : model.breakage)
|
|
||||||
{
|
|
||||||
_debug_breakage.values.push_back(static_cast<unsigned>(b));
|
|
||||||
}
|
|
||||||
|
|
||||||
_debug_info.values["breakage"] = _debug_breakage;
|
|
||||||
_debug_info.values["states"] = _debug_states;
|
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
77
util/json_logger.hpp
Normal file
77
util/json_logger.hpp
Normal file
@ -0,0 +1,77 @@
|
|||||||
|
/*
|
||||||
|
|
||||||
|
Copyright (c) 2015, Project OSRM, Dennis Luxen, others
|
||||||
|
All rights reserved.
|
||||||
|
|
||||||
|
Redistribution and use in source and binary forms, with or without modification,
|
||||||
|
are permitted provided that the following conditions are met:
|
||||||
|
|
||||||
|
Redistributions of source code must retain the above copyright notice, this list
|
||||||
|
of conditions and the following disclaimer.
|
||||||
|
Redistributions in binary form must reproduce the above copyright notice, this
|
||||||
|
list of conditions and the following disclaimer in the documentation and/or
|
||||||
|
other materials provided with the distribution.
|
||||||
|
|
||||||
|
THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND
|
||||||
|
ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
|
||||||
|
WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
||||||
|
DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR
|
||||||
|
ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
|
||||||
|
(INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
|
||||||
|
LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON
|
||||||
|
ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
||||||
|
(INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
|
||||||
|
SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||||
|
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef JSON_LOGGER_HPP
|
||||||
|
#define JSON_LOGGER_HPP
|
||||||
|
|
||||||
|
#include <osrm/json_container.hpp>
|
||||||
|
|
||||||
|
#include <boost/thread.hpp>
|
||||||
|
|
||||||
|
namespace osrm
|
||||||
|
{
|
||||||
|
namespace json
|
||||||
|
{
|
||||||
|
|
||||||
|
class Logger
|
||||||
|
{
|
||||||
|
using MapT = std::unordered_map<std::string, osrm::json::Value>;
|
||||||
|
|
||||||
|
public:
|
||||||
|
static Logger* get()
|
||||||
|
{
|
||||||
|
static Logger logger;
|
||||||
|
|
||||||
|
#ifdef NDEBUG
|
||||||
|
return nullptr;
|
||||||
|
#else
|
||||||
|
return &logger;
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
|
||||||
|
void initialize(const std::string& name)
|
||||||
|
{
|
||||||
|
if (!map.get())
|
||||||
|
{
|
||||||
|
map.reset(new MapT());
|
||||||
|
}
|
||||||
|
(*map)[name] = Object();
|
||||||
|
}
|
||||||
|
|
||||||
|
void render(const std::string& name, Object& obj) const
|
||||||
|
{
|
||||||
|
obj.values["debug"] = map->at(name);
|
||||||
|
}
|
||||||
|
|
||||||
|
boost::thread_specific_ptr<MapT> map;
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#endif /* SIMPLE_LOGGER_HPP */
|
@ -41,7 +41,8 @@ namespace osrm
|
|||||||
namespace json
|
namespace json
|
||||||
{
|
{
|
||||||
|
|
||||||
template <typename T> T clampFloat(T d)
|
// Make sure we don't have inf and NaN values
|
||||||
|
template <typename T> T clamp_float(T d)
|
||||||
{
|
{
|
||||||
if (std::isnan(d) || std::numeric_limits<T>::infinity() == d)
|
if (std::isnan(d) || std::numeric_limits<T>::infinity() == d)
|
||||||
{
|
{
|
||||||
@ -55,21 +56,46 @@ template <typename T> T clampFloat(T d)
|
|||||||
return d;
|
return d;
|
||||||
}
|
}
|
||||||
|
|
||||||
void appendToArray(osrm::json::Array &a) {}
|
void append_to_array(osrm::json::Array &a) {}
|
||||||
template <typename T, typename... Args>
|
template <typename T, typename... Args>
|
||||||
void appendToArray(osrm::json::Array &a, T value, Args... args)
|
void append_to_array(osrm::json::Array &a, T value, Args... args)
|
||||||
{
|
{
|
||||||
a.values.emplace_back(value);
|
a.values.emplace_back(value);
|
||||||
appendToJSONArray(a, args...);
|
append_to_array(a, args...);
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename... Args> osrm::json::Array makeArray(Args... args)
|
template <typename... Args> osrm::json::Array make_array(Args... args)
|
||||||
{
|
{
|
||||||
osrm::json::Array a;
|
osrm::json::Array a;
|
||||||
appendToJSONArray(a, args...);
|
append_to_array(a, args...);
|
||||||
return a;
|
return a;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
template <typename T> osrm::json::Array make_array(const std::vector<T>& vector)
|
||||||
|
{
|
||||||
|
osrm::json::Array a;
|
||||||
|
for (const auto& v : vector)
|
||||||
|
a.values.emplace_back(v);
|
||||||
|
return a;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Easy acces to object hierachies
|
||||||
|
osrm::json::Value& get(osrm::json::Value& value) { return value; }
|
||||||
|
|
||||||
|
template<typename... Keys>
|
||||||
|
osrm::json::Value& get(osrm::json::Value& value, const char* key, Keys... keys)
|
||||||
|
{
|
||||||
|
using recursive_object_t = mapbox::util::recursive_wrapper<osrm::json::Object>;
|
||||||
|
return get(value.get<recursive_object_t>().get().values[key], keys...);
|
||||||
|
}
|
||||||
|
|
||||||
|
template<typename... Keys>
|
||||||
|
osrm::json::Value& get(osrm::json::Value& value, unsigned key, Keys... keys)
|
||||||
|
{
|
||||||
|
using recursive_array_t = mapbox::util::recursive_wrapper<osrm::json::Array>;
|
||||||
|
return get(value.get<recursive_array_t>().get().values[key], keys...);
|
||||||
|
}
|
||||||
|
|
||||||
} // namespace json
|
} // namespace json
|
||||||
} // namespace osrm
|
} // namespace osrm
|
||||||
#endif // JSON_RENDERER_HPP
|
#endif // JSON_RENDERER_HPP
|
||||||
|
Loading…
Reference in New Issue
Block a user