[tracking] A wasteful version of the traffic data receiver.

This commit is contained in:
Maxim Pimenov 2016-10-28 20:14:18 +03:00
parent 77269ecfb0
commit c796ec6f57
9 changed files with 378 additions and 1 deletions

View file

@ -64,7 +64,7 @@ SUBDIRS = 3party base coding geometry editor indexer routing search
}
!CONFIG(gtool):!CONFIG(osrm) {
SUBDIRS *= drape drape_frontend partners_api tracking map
SUBDIRS *= drape drape_frontend partners_api tracking traffic map
CONFIG(map_designer):CONFIG(desktop) {
SUBDIRS *= skin_generator
@ -253,5 +253,8 @@ SUBDIRS = 3party base coding geometry editor indexer routing search
tracking_tests.depends = 3party base routing tracking platform_tests_support platform coding geometry
SUBDIRS *= tracking_tests
traffic_tests.subdir = traffic/traffic_tests
traffic_tests.depends = 3party base routing traffic platform_tests_support platform coding geometry
SUBDIRS *= traffic_tests
} # !no-tests
} # !gtool

13
traffic/CMakeLists.txt Normal file
View file

@ -0,0 +1,13 @@
project(traffic)
set(
SRC
speed_groups.cpp
speed_groups.hpp
traffic_info.cpp
traffic_info.hpp
)
add_library(${PROJECT_NAME} ${SRC})
add_subdirectory(traffic_tests)

22
traffic/speed_groups.cpp Normal file
View file

@ -0,0 +1,22 @@
#include "traffic/speed_groups.hpp"
namespace traffic
{
uint32_t const kSpeedGroupThresholdPercentage[] = {8, 16, 33, 58, 83, 100, 100, 100};
string DebugPrint(SpeedGroup const & group)
{
switch (group)
{
case SpeedGroup::G0: return "G0";
case SpeedGroup::G1: return "G1";
case SpeedGroup::G2: return "G2";
case SpeedGroup::G3: return "G3";
case SpeedGroup::G4: return "G4";
case SpeedGroup::G5: return "G5";
case SpeedGroup::TempBlock: return "TempBlock";
case SpeedGroup::Unknown: return "Unknown";
case SpeedGroup::Count: return "Count";
}
}
} // namespace traffic

41
traffic/speed_groups.hpp Normal file
View file

@ -0,0 +1,41 @@
#pragma once
#include "std/cstdint.hpp"
#include "std/string.hpp"
namespace traffic
{
enum class SpeedGroup : uint8_t
{
G0 = 0,
G1,
G2,
G3,
G4,
G5,
TempBlock,
Unknown,
Count
};
static_assert(static_cast<uint8_t>(SpeedGroup::Count) <= 8, "");
// Let M be the maximal speed that is possible on a free road
// and let V be the maximal speed that is possible on this road when
// taking the traffic data into account.
// We group all possible ratios (V/M) into a small number of
// buckets and only use the number of a bucket everywhere.
// That is, we forget the specific values of V when transmitting and
// displaying traffic information. The value M of a road is known at the
// stage of building the mwm containing this road.
//
// kSpeedGroupThresholdPercentage[g] denotes the maximal value of (V/M)
// that is possible for group |g|. Values falling on a border of two groups
// may belong to either group.
//
// The threshold percentage is defined to be 100 for the
// special groups where V is unknown or not defined.
extern uint32_t const kSpeedGroupThresholdPercentage[static_cast<size_t>(SpeedGroup::Count)];
string DebugPrint(SpeedGroup const & group);
} // namespace traffic

15
traffic/traffic.pro Normal file
View file

@ -0,0 +1,15 @@
TARGET = traffic
TEMPLATE = lib
CONFIG += staticlib warn_on
ROOT_DIR = ..
include($$ROOT_DIR/common.pri)
SOURCES += \
speed_groups.cpp \
traffic_info.cpp \
HEADERS += \
speed_groups.hpp \
traffic_info.hpp \

132
traffic/traffic_info.cpp Normal file
View file

@ -0,0 +1,132 @@
#include "traffic/traffic_info.hpp"
#include "platform/http_client.hpp"
#include "coding/bit_streams.hpp"
#include "coding/reader.hpp"
#include "coding/varint.hpp"
#include "coding/write_to_sink.hpp"
#include "coding/writer.hpp"
#include "base/assert.hpp"
#include "base/logging.hpp"
#include "std/algorithm.hpp"
#include "std/string.hpp"
#include "private.h"
namespace traffic
{
namespace
{
char const kTestFileName[] = "traffic_data";
bool ReadRemoteFile(string const & url, vector<uint8_t> & result)
{
platform::HttpClient request(url);
if (!request.RunHttpRequest() || request.ErrorCode() != 200)
return false;
string const & response = request.ServerResponse();
result.resize(response.size());
for (size_t i = 0; i < response.size(); ++i)
result[i] = static_cast<uint8_t>(response[i]);
return true;
}
} // namespace
// TrafficInfo::RoadSegmentId -----------------------------------------------------------------
TrafficInfo::RoadSegmentId::RoadSegmentId() : m_fid(0), m_idx(0), m_dir(0) {}
TrafficInfo::RoadSegmentId::RoadSegmentId(uint32_t fid, uint16_t idx, uint8_t dir)
: m_fid(fid), m_idx(idx), m_dir(dir)
{
}
// TrafficInfo --------------------------------------------------------------------------------
TrafficInfo::TrafficInfo(MwmSet::MwmId const & mwmId) : m_mwmId(mwmId) {}
bool TrafficInfo::ReceiveTrafficData()
{
if (strlen(TRAFFIC_DATA_BASE_URL) == 0)
return false;
string const url = string(TRAFFIC_DATA_BASE_URL) + string(kTestFileName);
vector<uint8_t> contents;
if (!ReadRemoteFile(url, contents))
return false;
Coloring coloring;
try
{
DeserializeTrafficData(contents, coloring);
}
catch (Reader::Exception const & e)
{
LOG(LINFO, ("Could not read traffic data received from server."));
return false;
}
m_coloring.swap(coloring);
return true;
}
SpeedGroup TrafficInfo::GetSpeedGroup(RoadSegmentId const & id) const
{
auto const it = m_coloring.find(id);
if (it == m_coloring.cend())
return SpeedGroup::Unknown;
return it->second;
}
// static
void TrafficInfo::SerializeTrafficData(Coloring const & coloring, vector<uint8_t> & result)
{
MemWriter<vector<uint8_t>> memWriter(result);
WriteToSink(memWriter, static_cast<uint32_t>(coloring.size()));
for (auto const & p : coloring)
{
WriteVarUint(memWriter, p.first.m_fid);
uint16_t const x = (p.first.m_idx << 1) | p.first.m_dir;
WriteVarUint(memWriter, x);
}
{
BitWriter<decltype(memWriter)> bitWriter(memWriter);
for (auto const & p : coloring)
{
// SpeedGroup's values fit into 3 bits.
bitWriter.Write(static_cast<uint8_t>(p.second), 3);
}
}
}
// static
void TrafficInfo::DeserializeTrafficData(vector<uint8_t> const & data, Coloring & coloring)
{
MemReader memReader(data.data(), data.size());
ReaderSource<decltype(memReader)> src(memReader);
auto const n = ReadPrimitiveFromSource<uint32_t>(src);
vector<RoadSegmentId> keys(n);
for (size_t i = 0; i < static_cast<size_t>(n); ++i)
{
keys[i].m_fid = ReadVarUint<uint32_t>(src);
auto const x = ReadVarUint<uint32_t>(src);
keys[i].m_idx = static_cast<uint16_t>(x >> 1);
keys[i].m_dir = static_cast<uint8_t>(x & 1);
}
vector<SpeedGroup> values(n);
BitReader<decltype(src)> bitReader(src);
for (size_t i = 0; i < static_cast<size_t>(n); ++i)
{
// SpeedGroup's values fit into 3 bits.
values[i] = static_cast<SpeedGroup>(bitReader.Read(3));
}
coloring.clear();
for (size_t i = 0; i < static_cast<size_t>(n); ++i)
coloring[keys[i]] = values[i];
ASSERT_EQUAL(src.Size(), 0, ());
}
} // namespace traffic

70
traffic/traffic_info.hpp Normal file
View file

@ -0,0 +1,70 @@
#pragma once
#include "traffic/speed_groups.hpp"
#include "indexer/mwm_set.hpp"
#include "std/cstdint.hpp"
#include "std/map.hpp"
#include "std/vector.hpp"
namespace traffic
{
// This class is responsible for providing the real-time
// information about road traffic for one mwm file.
class TrafficInfo
{
public:
struct RoadSegmentId
{
RoadSegmentId();
RoadSegmentId(uint32_t fid, uint16_t idx, uint8_t dir);
bool operator<(RoadSegmentId const & o) const
{
if (m_fid != o.m_fid)
return m_fid < o.m_fid;
if (m_idx != o.m_idx)
return m_idx < o.m_idx;
return m_dir < o.m_dir;
}
// The ordinal number of feature this segment belongs to.
uint32_t m_fid;
// The ordinal number of this segment in the list of
// its feature's segments.
uint16_t m_idx : 15;
// The direction of the segment.
// todo(@m) Write up what the values 0 and 1 mean specifically.
uint8_t m_dir : 1;
};
// todo(@m) unordered_map?
using Coloring = map<RoadSegmentId, SpeedGroup>;
TrafficInfo() = default;
TrafficInfo(MwmSet::MwmId const & mwmId);
// Fetches the latest traffic data from the server and updates the coloring.
// todo(@m, @syershov) Currently a hardcoded path is used.
// Construct the url by passing an MwmId.
// *NOTE* This method must not be called on the UI thread.
bool ReceiveTrafficData();
// Returns the latest known speed group by a feature segment's id.
SpeedGroup GetSpeedGroup(RoadSegmentId const & id) const;
static void SerializeTrafficData(Coloring const & coloring, vector<uint8_t> & result);
static void DeserializeTrafficData(vector<uint8_t> const & data, Coloring & coloring);
private:
// The mapping from feature segments to speed groups (see speed_groups.hpp).
Coloring m_coloring;
MwmSet::MwmId m_mwmId;
};
} // namespace traffic

View file

@ -0,0 +1,52 @@
#include "testing/testing.hpp"
#include "traffic/speed_groups.hpp"
#include "traffic/traffic_info.hpp"
#include "std/algorithm.hpp"
namespace traffic
{
namespace
{
SpeedGroup GetSpeedGroup(TrafficInfo::Coloring const & coloring,
TrafficInfo::RoadSegmentId const & fid)
{
auto const it = coloring.find(fid);
if (it == coloring.cend())
return SpeedGroup::Unknown;
return it->second;
}
} // namespace
UNIT_TEST(TrafficInfo_RemoteFile)
{
TrafficInfo r;
TEST(r.ReceiveTrafficData(), ());
}
UNIT_TEST(TrafficInfo_Serialization)
{
TrafficInfo::Coloring coloring = {
{TrafficInfo::RoadSegmentId(0, 0, 0), SpeedGroup::G0},
{TrafficInfo::RoadSegmentId(1000, 1, 1), SpeedGroup::G1},
{TrafficInfo::RoadSegmentId(1000000, 0, 0), SpeedGroup::G5},
{TrafficInfo::RoadSegmentId(4294967295, 32767, 1), SpeedGroup::TempBlock},
};
vector<uint8_t> buf;
TrafficInfo::SerializeTrafficData(coloring, buf);
TrafficInfo::Coloring deserializedColoring;
TrafficInfo::DeserializeTrafficData(buf, deserializedColoring);
TEST_EQUAL(coloring.size(), deserializedColoring.size(), ());
for (auto const & p : coloring)
{
auto const g1 = p.second;
auto const g2 = GetSpeedGroup(deserializedColoring, p.first);
TEST_EQUAL(g1, g2, ());
}
}
} // namespace traffic

View file

@ -0,0 +1,29 @@
TARGET = traffic_tests
CONFIG += console warn_on
CONFIG -= app_bundle
TEMPLATE = app
ROOT_DIR = ../..
INCLUDEPATH *= $$ROOT_DIR/3party/jansson/src
DEPENDENCIES = routing traffic indexer platform_tests_support platform coding geometry base stats_client tomcrypt
include($$ROOT_DIR/common.pri)
DEFINES *= OMIM_UNIT_TEST_WITH_QT_EVENT_LOOP
QT *= core
macx-* {
QT *= widgets # needed for QApplication with event loop, to test async events
LIBS *= "-framework IOKit" "-framework SystemConfiguration"
}
win*|linux* {
QT *= network
}
SOURCES += \
$$ROOT_DIR/testing/testingmain.cpp \
traffic_info_test.cpp \