diff --git a/coding/traffic.cpp b/coding/traffic.cpp index 2883a5f1f3..aa34a40711 100644 --- a/coding/traffic.cpp +++ b/coding/traffic.cpp @@ -5,7 +5,7 @@ namespace coding { // static -uint32_t const TrafficGPSEncoder::kLatestVersion = 0; +uint32_t const TrafficGPSEncoder::kLatestVersion = 1; uint32_t const TrafficGPSEncoder::kCoordBits = 30; double const TrafficGPSEncoder::kMinDeltaLat = ms::LatLon::kMinLat - ms::LatLon::kMaxLat; double const TrafficGPSEncoder::kMaxDeltaLat = ms::LatLon::kMaxLat - ms::LatLon::kMinLat; diff --git a/coding/traffic.hpp b/coding/traffic.hpp index 7b237082d5..f4e3c0b0db 100644 --- a/coding/traffic.hpp +++ b/coding/traffic.hpp @@ -25,11 +25,16 @@ public: { DataPoint() = default; - DataPoint(uint64_t timestamp, ms::LatLon latLon) : m_timestamp(timestamp), m_latLon(latLon) {} + DataPoint(uint64_t timestamp, ms::LatLon latLon, uint8_t traffic) + : m_timestamp(timestamp), m_latLon(latLon), m_traffic(traffic) + { + } // Uint64 should be enough for all our use cases. // It is expected that |m_timestamp| stores time since epoch in seconds. uint64_t m_timestamp = 0; ms::LatLon m_latLon = ms::LatLon::Zero(); + // TODO: comment the decision + uint8_t m_traffic = 0; bool operator==(DataPoint const & p) const { @@ -43,20 +48,53 @@ public: // Coordinates are truncated and stored as integers. All integers // are written as varints. template - static size_t SerializeDataPoints(uint32_t version, Writer & writer, - Collection const & points) + static size_t SerializeDataPoints(uint32_t version, Writer & writer, Collection const & points) { - ASSERT_LESS_OR_EQUAL(version, kLatestVersion, ()); + switch (version) + { + case 0: return SerializeDataPointsV0(writer, points); + case 1: return SerializeDataPointsV1(writer, points); + default: + ASSERT(false, ("Unexpected serializer version:", version)); + break; + } + return 0; + } + + + // Deserializes the points from |source| and appends them to |result|. + template + static void DeserializeDataPoints(uint32_t version, Source & src, Collection & result) + { + switch (version) + { + case 0: return DeserializeDataPointsV0(src, result); + case 1: return DeserializeDataPointsV1(src, result); + + default: + ASSERT(false, ("Unexpected serializer version:", version)); + break; + } + } + +private: + static uint32_t DoubleToUint32(double x, double min, double max); + + static double Uint32ToDouble(uint32_t x, double min, double max); + + template + static size_t SerializeDataPointsV0(Writer & writer, Collection const & points) + { auto const startPos = writer.Pos(); if (!points.empty()) { uint64_t const firstTimestamp = points[0].m_timestamp; uint32_t const firstLat = - DoubleToUint32(points[0].m_latLon.lat, ms::LatLon::kMinLat, ms::LatLon::kMaxLat); + DoubleToUint32(points[0].m_latLon.lat, ms::LatLon::kMinLat, ms::LatLon::kMaxLat); uint32_t const firstLon = - DoubleToUint32(points[0].m_latLon.lon, ms::LatLon::kMinLon, ms::LatLon::kMaxLon); + DoubleToUint32(points[0].m_latLon.lon, ms::LatLon::kMinLon, ms::LatLon::kMaxLon); WriteVarUint(writer, firstTimestamp); WriteVarUint(writer, firstLat); WriteVarUint(writer, firstLon); @@ -82,16 +120,55 @@ public: return static_cast(writer.Pos() - startPos); } - // Deserializes the points from |source| and appends them to |result|. - template - static void DeserializeDataPoints(uint32_t version, Source & src, Collection & result) + template + static size_t SerializeDataPointsV1(Writer & writer, Collection const & points) { - ASSERT_LESS_OR_EQUAL(version, kLatestVersion, ()); + auto const startPos = writer.Pos(); + if (!points.empty()) + { + uint64_t const firstTimestamp = points[0].m_timestamp; + uint32_t const firstLat = + DoubleToUint32(points[0].m_latLon.lat, ms::LatLon::kMinLat, ms::LatLon::kMaxLat); + uint32_t const firstLon = + DoubleToUint32(points[0].m_latLon.lon, ms::LatLon::kMinLon, ms::LatLon::kMaxLon); + uint32_t const traffic = points[0].m_traffic; + WriteVarUint(writer, firstTimestamp); + WriteVarUint(writer, firstLat); + WriteVarUint(writer, firstLon); + WriteVarUint(writer, traffic); + } + + for (size_t i = 1; i < points.size(); ++i) + { + ASSERT_LESS_OR_EQUAL(points[i - 1].m_timestamp, points[i].m_timestamp, ()); + + uint64_t const deltaTimestamp = points[i].m_timestamp - points[i - 1].m_timestamp; + uint32_t deltaLat = DoubleToUint32(points[i].m_latLon.lat - points[i - 1].m_latLon.lat, + kMinDeltaLat, kMaxDeltaLat); + uint32_t deltaLon = DoubleToUint32(points[i].m_latLon.lon - points[i - 1].m_latLon.lon, + kMinDeltaLon, kMaxDeltaLon); + + uint32_t const traffic = points[i - 1].m_traffic; + WriteVarUint(writer, deltaTimestamp); + WriteVarUint(writer, deltaLat); + WriteVarUint(writer, deltaLon); + WriteVarUint(writer, traffic); + } + + ASSERT_LESS_OR_EQUAL(writer.Pos() - startPos, numeric_limits::max(), + ("Too much data.")); + return static_cast(writer.Pos() - startPos); + } + + template + static void DeserializeDataPointsV0(Source & src, Collection & result) + { bool first = true; uint64_t lastTimestamp = 0; double lastLat = 0.0; double lastLon = 0.0; + uint8_t traffic = 0; while (src.Size() > 0) { @@ -99,10 +176,10 @@ public: { lastTimestamp = ReadVarUint(src); lastLat = - Uint32ToDouble(ReadVarUint(src), ms::LatLon::kMinLat, ms::LatLon::kMaxLat); + Uint32ToDouble(ReadVarUint(src), ms::LatLon::kMinLat, ms::LatLon::kMaxLat); lastLon = - Uint32ToDouble(ReadVarUint(src), ms::LatLon::kMinLon, ms::LatLon::kMaxLon); - result.push_back(DataPoint(lastTimestamp, ms::LatLon(lastLat, lastLon))); + Uint32ToDouble(ReadVarUint(src), ms::LatLon::kMinLon, ms::LatLon::kMaxLon); + result.push_back(DataPoint(lastTimestamp, ms::LatLon(lastLat, lastLon), traffic)); first = false; } else @@ -110,14 +187,43 @@ public: lastTimestamp += ReadVarUint(src); lastLat += Uint32ToDouble(ReadVarUint(src), kMinDeltaLat, kMaxDeltaLat); lastLon += Uint32ToDouble(ReadVarUint(src), kMinDeltaLon, kMaxDeltaLon); - result.push_back(DataPoint(lastTimestamp, ms::LatLon(lastLat, lastLon))); + result.push_back(DataPoint(lastTimestamp, ms::LatLon(lastLat, lastLon), traffic)); } } } -private: - static uint32_t DoubleToUint32(double x, double min, double max); + template + static void DeserializeDataPointsV1(Source & src, Collection & result) + { + bool first = true; + uint64_t lastTimestamp = 0; + double lastLat = 0.0; + double lastLon = 0.0; + uint8_t traffic = 0; + + while (src.Size() > 0) + { + if (first) + { + lastTimestamp = ReadVarUint(src); + lastLat = + Uint32ToDouble(ReadVarUint(src), ms::LatLon::kMinLat, ms::LatLon::kMaxLat); + lastLon = + Uint32ToDouble(ReadVarUint(src), ms::LatLon::kMinLon, ms::LatLon::kMaxLon); + traffic = static_cast(ReadVarUint(src)); + result.push_back(DataPoint(lastTimestamp, ms::LatLon(lastLat, lastLon), traffic)); + first = false; + } + else + { + lastTimestamp += ReadVarUint(src); + lastLat += Uint32ToDouble(ReadVarUint(src), kMinDeltaLat, kMaxDeltaLat); + lastLon += Uint32ToDouble(ReadVarUint(src), kMinDeltaLon, kMaxDeltaLon); + traffic = static_cast(ReadVarUint(src)); + result.push_back(DataPoint(lastTimestamp, ms::LatLon(lastLat, lastLon), traffic)); + } + } + } - static double Uint32ToDouble(uint32_t x, double min, double max); }; } // namespace coding diff --git a/tracking/protocol.cpp b/tracking/protocol.cpp index 8a2fa90374..3d97400a87 100644 --- a/tracking/protocol.cpp +++ b/tracking/protocol.cpp @@ -12,15 +12,23 @@ namespace { template -vector CreateDataPacketImpl(Container const & points) +vector CreateDataPacketImpl(Container const & points, + tracking::Protocol::PacketType const type) { vector buffer; MemWriter writer(buffer); - tracking::Protocol::Encoder::SerializeDataPoints(tracking::Protocol::Encoder::kLatestVersion, - writer, points); - auto packet = tracking::Protocol::CreateHeader(tracking::Protocol::PacketType::CurrentData, - static_cast(buffer.size())); + uint32_t serializer_version = tracking::Protocol::Encoder::kLatestVersion; + switch (type) + { + case tracking::Protocol::PacketType::DataV0: serializer_version = 0; break; + case tracking::Protocol::PacketType::DataV1: serializer_version = 1; break; + case tracking::Protocol::PacketType::AuthV0: ASSERT(false, ("Not a DATA type.")); break; + } + + tracking::Protocol::Encoder::SerializeDataPoints(serializer_version, writer, points); + + auto packet = tracking::Protocol::CreateHeader(type, static_cast(buffer.size())); packet.insert(packet.end(), begin(buffer), end(buffer)); return packet; @@ -54,15 +62,15 @@ vector Protocol::CreateAuthPacket(string const & clientId) } // static -vector Protocol::CreateDataPacket(DataElementsCirc const & points) +vector Protocol::CreateDataPacket(DataElementsCirc const & points, PacketType type) { - return CreateDataPacketImpl(points); + return CreateDataPacketImpl(points, type); } // static -vector Protocol::CreateDataPacket(DataElementsVec const & points) +vector Protocol::CreateDataPacket(DataElementsVec const & points, PacketType type) { - return CreateDataPacketImpl(points); + return CreateDataPacketImpl(points, type); } // static @@ -82,9 +90,9 @@ string Protocol::DecodeAuthPacket(Protocol::PacketType type, vector con { switch (type) { - case Protocol::PacketType::AuthV0: - return string(begin(data), end(data)); - case Protocol::PacketType::DataV0: break; + case Protocol::PacketType::AuthV0: return string(begin(data), end(data)); + case Protocol::PacketType::DataV0: + case Protocol::PacketType::DataV1: ASSERT(false, ("Not AUTH packet.")); break; } return string(); } @@ -98,9 +106,12 @@ Protocol::DataElementsVec Protocol::DecodeDataPacket(PacketType type, vector(type) << ")"; diff --git a/tracking/protocol.hpp b/tracking/protocol.hpp index f351dad98c..03efd3dac9 100644 --- a/tracking/protocol.hpp +++ b/tracking/protocol.hpp @@ -23,15 +23,18 @@ public: { AuthV0 = 0x81, DataV0 = 0x82, + DataV1 = 0x92, CurrentAuth = AuthV0, - CurrentData = DataV0 + CurrentData = DataV1 }; static vector CreateHeader(PacketType type, uint32_t payloadSize); static vector CreateAuthPacket(string const & clientId); - static vector CreateDataPacket(DataElementsCirc const & points); - static vector CreateDataPacket(DataElementsVec const & points); + static vector CreateDataPacket(DataElementsCirc const & points, + PacketType type = PacketType::CurrentData); + static vector CreateDataPacket(DataElementsVec const & points, + PacketType type = PacketType::CurrentData); static std::pair DecodeHeader(vector const & data); static string DecodeAuthPacket(PacketType type, vector const & data); diff --git a/tracking/pytracking/bindings.cpp b/tracking/pytracking/bindings.cpp index 3bd9bdd70b..528351e614 100644 --- a/tracking/pytracking/bindings.cpp +++ b/tracking/pytracking/bindings.cpp @@ -34,6 +34,7 @@ BOOST_PYTHON_MODULE(pytracking) enum_("PacketType") .value("AuthV0", Protocol::PacketType::AuthV0) .value("DataV0", Protocol::PacketType::DataV0) + .value("DataV1", Protocol::PacketType::DataV1) .value("CurrentAuth", Protocol::PacketType::CurrentAuth) .value("CurrentData", Protocol::PacketType::CurrentData); diff --git a/tracking/reporter.cpp b/tracking/reporter.cpp index 269131e0f2..d125d5b4ce 100644 --- a/tracking/reporter.cpp +++ b/tracking/reporter.cpp @@ -43,7 +43,7 @@ Reporter::~Reporter() m_thread.join(); } -void Reporter::AddLocation(location::GpsInfo const & info, traffic::SpeedGroup /* speedGroup */) +void Reporter::AddLocation(location::GpsInfo const & info, traffic::SpeedGroup traffic) { lock_guard lg(m_mutex); @@ -54,7 +54,9 @@ void Reporter::AddLocation(location::GpsInfo const & info, traffic::SpeedGroup / return; m_lastGpsTime = info.m_timestamp; - m_input.push_back(DataPoint(info.m_timestamp, ms::LatLon(info.m_latitude, info.m_longitude))); + m_input.push_back(DataPoint(info.m_timestamp, + ms::LatLon(info.m_latitude, info.m_longitude), + static_cast::type>(traffic))); } void Reporter::Run() diff --git a/tracking/reporter.hpp b/tracking/reporter.hpp index 84f42f74e6..a8f63b935b 100644 --- a/tracking/reporter.hpp +++ b/tracking/reporter.hpp @@ -39,7 +39,7 @@ public: milliseconds pushDelay); ~Reporter(); - void AddLocation(location::GpsInfo const & info, traffic::SpeedGroup speedGroup); + void AddLocation(location::GpsInfo const & info, traffic::SpeedGroup traffic); void SetAllowSendingPoints(bool allow) { m_allowSendingPoints = allow; }