diff --git a/map/gps_track_file.cpp b/map/gps_track_file.cpp new file mode 100644 index 0000000000..7b37d5c287 --- /dev/null +++ b/map/gps_track_file.cpp @@ -0,0 +1,425 @@ +#include "map/gps_track_file.hpp" + +#include "base/assert.hpp" +#include "base/logging.hpp" + +#include "std/algorithm.hpp" + +namespace +{ + +size_t constexpr kLinearSearchDistance = 10; +size_t constexpr kLazyWriteHeaderMaxCount = 100; + +} // namespace + +size_t const GpsTrackFile::kInvalidId = numeric_limits::max(); + +GpsTrackFile::GpsTrackFile() + : m_lazyWriteHeaderCount(0) +{ +} + +GpsTrackFile::~GpsTrackFile() +{ + ASSERT(!m_stream.is_open(), ("Don't forget to close file")); + + if (m_stream.is_open()) + { + try + { + Close(); + } + catch (RootException const & e) + { + LOG(LINFO, ("Close caused exception", e.Msg())); + } + } +} + +bool GpsTrackFile::Open(string const & filePath, size_t maxItemCount) +{ + ASSERT(!m_stream.is_open(), ("File must not be open on OpenFile")); + ASSERT(maxItemCount > 0, ()); + + m_stream = fstream(filePath, ios::in|ios::out|ios::binary); + + if (!m_stream) + return false; + + try + { + m_filePath = filePath; + + // Check file integrity: + // - file size if correct; + // - Header fields m_first, m_last, m_lastId and m_timestamp are correct; + // - front and back items are correct, if exist. + + m_stream.seekg(0, ios::end); + size_t const fileSize = m_stream.tellg(); + m_stream.seekg(0, ios::beg); + + if (fileSize < sizeof(Header)) + MYTHROW(CorruptedFileException, ("File:", m_filePath)); + + if ((fileSize - sizeof(Header)) % sizeof(TItem) != 0) + MYTHROW(CorruptedFileException, ("File:", m_filePath)); + + size_t const itemCount = (fileSize - sizeof(Header)) / sizeof(TItem); + + if (!ReadHeader(m_header)) + MYTHROW(CorruptedFileException, ("File:", m_filePath)); + + if (m_header.m_maxItemCount != (1 + maxItemCount)) + MYTHROW(CorruptedFileException, ("File:", m_filePath)); + + if (itemCount > m_header.m_maxItemCount) + MYTHROW(CorruptedFileException, ("File:", m_filePath)); + + if (itemCount == 0) + { + if (m_header.m_first != 0 && m_header.m_last != 0) + MYTHROW(CorruptedFileException, ("File:", m_filePath)); + } + else + { + if (m_header.m_first >= itemCount || m_header.m_last > itemCount) + MYTHROW(CorruptedFileException, ("File:", m_filePath)); + } + + if (m_header.m_lastId < itemCount) + MYTHROW(CorruptedFileException, ("File:", m_filePath)); + + if (m_header.m_first != m_header.m_last) + { + TItem frontItem; + if (!ReadItem(m_header.m_first, frontItem)) + MYTHROW(CorruptedFileException, ("File:", m_filePath)); + + TItem backItem; + size_t backIndex = (m_header.m_first + Distance(m_header.m_first, m_header.m_last, m_header.m_maxItemCount) - 1) % m_header.m_maxItemCount; + if (!ReadItem(backIndex, backItem)) + MYTHROW(CorruptedFileException, ("File:", m_filePath)); + + if (frontItem.m_timestamp > backItem.m_timestamp || m_header.m_timestamp != backItem.m_timestamp) + MYTHROW(CorruptedFileException, ("File:", m_filePath)); + } + } + catch (RootException &) + { + m_header = Header(); + m_filePath.clear(); + m_stream.close(); + throw; + } + + return true; +} + +bool GpsTrackFile::Create(string const & filePath, size_t maxItemCount) +{ + ASSERT(!m_stream.is_open(), ("File must not be open on CreateFile")); + ASSERT(maxItemCount > 0, ()); + + m_stream = fstream(filePath, ios::in|ios::out|ios::binary|ios::trunc); + + if (!m_stream) + return false; + + try + { + m_filePath = filePath; + + m_header = Header(); + m_header.m_maxItemCount = maxItemCount + 1; + + WriteHeader(m_header); + } + catch (RootException &) + { + m_header = Header(); + m_filePath.clear(); + m_stream.close(); + throw; + } + + return true; +} + +bool GpsTrackFile::IsOpen() const +{ + return m_stream.is_open(); +} + +void GpsTrackFile::Close() +{ + ASSERT(m_stream.is_open(), ("File is not open")); + + if (m_lazyWriteHeaderCount != 0) + { + m_lazyWriteHeaderCount = 0; + WriteHeader(m_header); + } + + m_stream.close(); + if (0 != (m_stream.rdstate() & (ios::failbit | ios::badbit))) + MYTHROW(WriteFileException, ("File:", m_filePath)); + + m_header = Header(); + m_filePath.clear(); +} + +void GpsTrackFile::Flush() +{ + ASSERT(m_stream.is_open(), ("File is not open")); + + if (m_lazyWriteHeaderCount != 0) + { + m_lazyWriteHeaderCount = 0; + WriteHeader(m_header); + } + + m_stream.flush(); + if (0 != (m_stream.rdstate() & (ios::failbit | ios::badbit))) + MYTHROW(WriteFileException, ("File:", m_filePath)); +} + +size_t GpsTrackFile::Append(TItem const & item, size_t & evictedId) +{ + ASSERT(m_stream.is_open(), ("File is not open")); + + if (item.m_timestamp < m_header.m_timestamp) + { + evictedId = kInvalidId; // nothing was evicted + return kInvalidId; // nothing was added + } + + size_t const newLast = (m_header.m_last + 1) % m_header.m_maxItemCount; + size_t const newFirst = (newLast == m_header.m_first) ? ((m_header.m_first + 1) % m_header.m_maxItemCount) : m_header.m_first; + + WriteItem(m_header.m_last, item); + + size_t const addedId = m_header.m_lastId; + + if (m_header.m_first == newFirst) + evictedId = kInvalidId; // nothing was evicted + else + evictedId = m_header.m_lastId - GetCount(); // the id of the first item + + m_header.m_first = newFirst; + m_header.m_last = newLast; + m_header.m_timestamp = item.m_timestamp; + m_header.m_lastId += 1; + + LazyWriteHeader(); + + return addedId; +} + +void GpsTrackFile::ForEach(function const & fn) +{ + ASSERT(m_stream.is_open(), ("File is not open")); + + double prevTimestamp = 0; + + size_t id = m_header.m_lastId - GetCount(); // the id of the first item + + for (size_t i = m_header.m_first; i != m_header.m_last; i = (i + 1) % m_header.m_maxItemCount) + { + TItem item; + if (!ReadItem(i, item)) + MYTHROW(CorruptedFileException, ("File:", m_filePath)); + + if (prevTimestamp > item.m_timestamp) + MYTHROW(CorruptedFileException, ("File:", m_filePath)); + + size_t itemId = id; + if (!fn(item, itemId)) + break; + + prevTimestamp = item.m_timestamp; + ++id; + } +} + +pair GpsTrackFile::DropEarlierThan(double timestamp) +{ + ASSERT(m_stream.is_open(), ("File is not open")); + + if (IsEmpty()) + return make_pair(kInvalidId, kInvalidId); // nothing was dropped + + if (m_header.m_timestamp <= timestamp) + return Clear(); + + size_t const n = GetCount(); + + ASSERT_GREATER_OR_EQUAL(m_header.m_lastId, n, ()); + + // Try linear search for short distance + // In common case elements will be removed from the tail by small pieces + size_t const linearSearchCount = min(kLinearSearchDistance, n); + for (size_t i = m_header.m_first, j = 0; i != m_header.m_last; i = (i + 1) % m_header.m_maxItemCount, ++j) + { + if (j >= linearSearchCount) + break; + + TItem item; + if (!ReadItem(i, item)) + MYTHROW(CorruptedFileException, ("File:", m_filePath)); + + if (item.m_timestamp >= timestamp) + { + // Dropped range is + pair const res = make_pair(m_header.m_lastId - n, + m_header.m_lastId - n + j - 1); + + // Update header.first, if need + if (i != m_header.m_first) + { + m_header.m_first = i; + + LazyWriteHeader(); + } + + return res; + } + } + + // By nature items are sorted by timestamp. + // Use lower_bound algorithm to find first element later than timestamp. + size_t len = n, first = m_header.m_first; + while (len > 0) + { + size_t const step = len / 2; + size_t const index = (first + step) % m_header.m_maxItemCount; + + TItem item; + if (!ReadItem(index, item)) + MYTHROW(CorruptedFileException, ("File:", m_filePath)); + + if (item.m_timestamp < timestamp) + { + first = (index + 1) % m_header.m_maxItemCount; + len -= step + 1; + } + else + len = step; + } + + // Dropped range is + pair const res = + make_pair(m_header.m_lastId - n, + m_header.m_lastId - n + Distance(m_header.m_first, first, m_header.m_maxItemCount) - 1); + + // Update header.first, if need + if (first != m_header.m_first) + { + m_header.m_first = first; + + LazyWriteHeader(); + } + + return res; +} + +pair GpsTrackFile::Clear() +{ + ASSERT(m_stream.is_open(), ("File is not open")); + + if (m_header.m_first == 0 && m_header.m_last == 0 && + m_header.m_lastId == 0 && m_header.m_timestamp == 0) + { + return make_pair(kInvalidId, kInvalidId); // nothing was dropped + } + + // Dropped range is + pair const res = make_pair(m_header.m_lastId - GetCount(), + m_header.m_lastId - 1); + + m_header.m_first = 0; + m_header.m_last = 0; + m_header.m_lastId = 0; + m_header.m_timestamp = 0; + + LazyWriteHeader(); + + return res; +} + +size_t GpsTrackFile::GetMaxCount() const +{ + return m_header.m_maxItemCount == 0 ? 0 : m_header.m_maxItemCount - 1; +} + +size_t GpsTrackFile::GetCount() const +{ + return Distance(m_header.m_first, m_header.m_last, m_header.m_maxItemCount); +} + +bool GpsTrackFile::IsEmpty() const +{ + return m_header.m_first == m_header.m_last; +} + +double GpsTrackFile::GetTimestamp() const +{ + return m_header.m_timestamp; +} + +void GpsTrackFile::LazyWriteHeader() +{ + ++m_lazyWriteHeaderCount; + if (m_lazyWriteHeaderCount < kLazyWriteHeaderMaxCount) + return; + + m_lazyWriteHeaderCount = 0; + WriteHeader(m_header); +} + +bool GpsTrackFile::ReadHeader(Header & header) +{ + m_stream.seekg(0, ios::beg); + m_stream.read(reinterpret_cast(&header), sizeof(header)); + if (0 != (m_stream.rdstate() & (ios::failbit | ios::badbit))) + MYTHROW(ReadFileException, ("File:", m_filePath)); + return ((m_stream.rdstate() & ios::eofbit) == 0); +} + +void GpsTrackFile::WriteHeader(Header const & header) +{ + m_stream.seekp(0, ios::beg); + m_stream.write(reinterpret_cast(&header), sizeof(header)); + if (0 != (m_stream.rdstate() & (ios::failbit | ios::badbit))) + MYTHROW(WriteFileException, ("File:", m_filePath)); +} + +bool GpsTrackFile::ReadItem(size_t index, TItem & item) +{ + size_t const offset = ItemOffset(index); + m_stream.seekg(offset, ios::beg); + m_stream.read(reinterpret_cast(&item), sizeof(item)); + if (0 != (m_stream.rdstate() & (ios::failbit | ios::badbit))) + MYTHROW(ReadFileException, ("File:", m_filePath)); + return ((m_stream.rdstate() & ios::eofbit) == 0); +} + +void GpsTrackFile::WriteItem(size_t index, TItem const & item) +{ + size_t const offset = ItemOffset(index); + m_stream.seekp(offset, ios::beg); + m_stream.write(reinterpret_cast(&item), sizeof(item)); + if (0 != (m_stream.rdstate() & (ios::failbit | ios::badbit))) + MYTHROW(WriteFileException, ("File:", m_filePath)); +} + +size_t GpsTrackFile::ItemOffset(size_t index) +{ + return sizeof(Header) + index * sizeof(TItem); +} + +size_t GpsTrackFile::Distance(size_t first, size_t last, size_t maxItemCount) +{ + return (first <= last) ? (last - first) : (last + maxItemCount - first); +} diff --git a/map/gps_track_file.hpp b/map/gps_track_file.hpp new file mode 100644 index 0000000000..16cff84283 --- /dev/null +++ b/map/gps_track_file.hpp @@ -0,0 +1,127 @@ +#pragma once + +#include "platform/location.hpp" + +#include "base/exception.hpp" + +#include "std/fstream.hpp" +#include "std/function.hpp" +#include "std/limits.hpp" +#include "std/string.hpp" + +class GpsTrackFile final +{ +public: + DECLARE_EXCEPTION(WriteFileException, RootException); + DECLARE_EXCEPTION(ReadFileException, RootException); + DECLARE_EXCEPTION(CorruptedFileException, RootException); + + /// Invalid identifier for point + static size_t const kInvalidId; // = numeric_limits::max(); + + using TItem = location::GpsTrackInfo; + + GpsTrackFile(); + ~GpsTrackFile(); + + /// Opens file with track data. + /// @param filePath - path to the file on disk + /// @param maxItemCount - max number of items in recycling file + /// @return If file does not exist then returns false, if everything is ok then returns true. + /// @exception If file corrupted then throws CorruptedFileException file, or ReadFileException if read fails. + bool Open(string const & filePath, size_t maxItemCount); + + /// Creates new file + /// @param filePath - path to the file on disk + /// @param maxItemCount - max number of items in recycling file + /// @return If file cannot be created then false is returned, if everything is ok then returns true. + /// @exceptions WriteFileException if write fails. + bool Create(string const & filePath, size_t maxItemCount); + + /// Returns true if file is open, otherwise returns false + bool IsOpen() const; + + /// Flushes all changes and closes file + /// @exceptions WriteFileException if write fails. + void Close(); + + /// Flushes all changes in file + /// @exceptions WriteFileException if write fails. + void Flush(); + + /// Appends new point in the file + /// @param item - gps track point. + /// @param evictedId - identifier of evicted point due to recycling, kInvalidId if there is no evicted point. + /// @returns identifier of point, kInvalidId if point was not added. + /// @note Timestamp must be not less than GetTimestamp(), otherwise function returns false. + /// @note File is circular, when GetMaxItemCount() limit is reached, old point is evicted from file. + /// @exceptions WriteFileException if write fails. + size_t Append(TItem const & item, size_t & evictedId); + + /// Remove all points from the file + /// @returns range of identifiers of evicted points, or pair(kInvalidId,kInvalidId) if nothing was evicted. + /// @exceptions WriteFileException if write fails. + pair Clear(); + + /// Returns max number of points in recycling file + size_t GetMaxCount() const; + + /// Returns number of items in the file, this values is <= GetMaxItemCount() + size_t GetCount() const; + + /// Returns true if file does not contain points + bool IsEmpty() const; + + /// Returns upper bound timestamp, or zero if there is no points + double GetTimestamp() const; + + /// Enumerates all points from the file in timestamp ascending order + /// @param fn - callable object which receives points. If fn returns false then enumeration is stopped. + /// @exceptions CorruptedFileException if file is corrupted or ReadFileException if read fails + void ForEach(function const & fn); + + /// Drops points earlier than specified date + /// @param timestamp - timestamp of lower bound, number of seconds since 1.1.1970. + /// @returns range of identifiers of evicted points, or pair(kInvalidId,kInvalidId) if nothing was evicted. + /// @exceptions CorruptedFileException if file is corrupted, ReadFileException if read fails + /// or WriteFileException if write fails. + pair DropEarlierThan(double timestamp); + +private: + /// Header, stored in beginning of file + /// @note Due to recycling, indexes of items are reused, but idendifiers are unique + /// until Clear or create new file + struct Header + { + size_t m_maxItemCount; // max number of items in recycling file (n + 1 for end element) + double m_timestamp; // upper bound timestamp + size_t m_first; // index of first item + size_t m_last; // index of last item, items are in range [first,last) + size_t m_lastId; // identifier of the last item + + Header() + : m_maxItemCount(0) + , m_timestamp(0) + , m_first(0) + , m_last(0) + , m_lastId(0) + {} + }; + + void LazyWriteHeader(); + + bool ReadHeader(Header & header); + void WriteHeader(Header const & header); + + bool ReadItem(size_t index, TItem & item); + void WriteItem(size_t index, TItem const & item); + + static size_t ItemOffset(size_t index); + static size_t Distance(size_t first, size_t last, size_t maxItemCount); + + string m_filePath; // file name + fstream m_stream; // buffered file stream + Header m_header; // file header + + uint32_t m_lazyWriteHeaderCount; // request count for write header +}; diff --git a/map/map.pro b/map/map.pro index bfb4b86fa6..678e4a1238 100644 --- a/map/map.pro +++ b/map/map.pro @@ -23,6 +23,7 @@ HEADERS += \ geourl_process.hpp \ gps_track_collection.hpp \ gps_track_container.hpp \ + gps_track_file.hpp \ mwm_url.hpp \ storage_bridge.hpp \ styled_point.hpp \ @@ -44,6 +45,7 @@ SOURCES += \ geourl_process.cpp \ gps_track_collection.cpp \ gps_track_container.cpp \ + gps_track_file.cpp \ mwm_url.cpp \ storage_bridge.cpp \ styled_point.cpp \ diff --git a/map/map_tests/gps_track_file_test.cpp b/map/map_tests/gps_track_file_test.cpp new file mode 100644 index 0000000000..2814bfb1ce --- /dev/null +++ b/map/map_tests/gps_track_file_test.cpp @@ -0,0 +1,411 @@ +#include "testing/testing.hpp" + +#include "map/gps_track_file.hpp" + +#include "platform/platform.hpp" + +#include "coding/file_name_utils.hpp" +#include "coding/file_writer.hpp" + +#include "geometry/latlon.hpp" + +#include "base/logging.hpp" +#include "base/scope_guard.hpp" + +#include "std/chrono.hpp" + +namespace +{ + +location::GpsTrackInfo Make(double timestamp, ms::LatLon const & ll, double speed) +{ + location::GpsTrackInfo info; + info.m_timestamp = timestamp; + info.m_speed = speed; + info.m_latitude = ll.lat; + info.m_longitude = ll.lon; + return info; +} + +inline string GetGpsTrackFilePath() +{ + return my::JoinFoldersToPath(GetPlatform().WritableDir(), "gpstrack.bin"); +} + +} // namespace + +UNIT_TEST(GpsTrackFile_SimpleWriteRead) +{ + string const filePath = GetGpsTrackFilePath(); + MY_SCOPE_GUARD(gpsTestFileDeleter, bind(FileWriter::DeleteFileX, filePath)); + + time_t const t = system_clock::to_time_t(system_clock::now()); + double const timestamp = t; + LOG(LINFO, ("Timestamp", ctime(&t), timestamp)); + + size_t const fileMaxItemCount = 100000; + + // Write GPS tracks. + // (write only half of max items to do not do recycling) + { + GpsTrackFile file; + file.Create(filePath, fileMaxItemCount); + + TEST(file.IsOpen(), ()); + + TEST_EQUAL(fileMaxItemCount, file.GetMaxCount(), ()); + + TEST_EQUAL(0, file.GetCount(), ()); + + for (size_t i = 0; i < fileMaxItemCount / 2; ++i) + { + size_t evictedId; + size_t addedId = file.Append(Make(timestamp + i, ms::LatLon(i + 1000, i + 2000), i + 3000), evictedId); + TEST_EQUAL(i, addedId, ()); + TEST_EQUAL(GpsTrackFile::kInvalidId, evictedId, ()); + } + + TEST_EQUAL(fileMaxItemCount / 2, file.GetCount(), ()); + + file.Close(); + + TEST(!file.IsOpen(), ()); + } + + // Read GPS tracks. + { + GpsTrackFile file; + file.Open(filePath, fileMaxItemCount); + + TEST(file.IsOpen(), ()); + TEST(!file.IsEmpty(), ()); + + TEST_EQUAL(fileMaxItemCount, file.GetMaxCount(), ()); + + TEST_EQUAL(fileMaxItemCount/2, file.GetCount(), ()); + + size_t i = 0; + file.ForEach([&i,timestamp](location::GpsTrackInfo const & info, size_t id)->bool + { + TEST_EQUAL(id, i, ()); + TEST_EQUAL(info.m_timestamp, timestamp + i, ()); + TEST_EQUAL(info.m_latitude, i + 1000, ()); + TEST_EQUAL(info.m_longitude, i + 2000, ()); + TEST_EQUAL(info.m_speed, i + 3000, ()); + ++i; + return true; + }); + + TEST_EQUAL(i, fileMaxItemCount / 2, ()); + + auto res = file.Clear(); + TEST_EQUAL(res.first, 0, ()); + TEST_EQUAL(res.second, fileMaxItemCount / 2 - 1, ()); + + TEST(file.IsEmpty(), ()); + + file.Close(); + + TEST(!file.IsOpen(), ()); + } +} + +UNIT_TEST(GpsTrackFile_WriteReadWithEvicting) +{ + string const filePath = GetGpsTrackFilePath(); + MY_SCOPE_GUARD(gpsTestFileDeleter, bind(FileWriter::DeleteFileX, filePath)); + + time_t const t = system_clock::to_time_t(system_clock::now()); + double const timestamp = t; + LOG(LINFO, ("Timestamp", ctime(&t), timestamp)); + + size_t const fileMaxItemCount = 100000; + + // Write GPS tracks. + // 2 x fileMaxItemCount more items are written in cyclic file, first half will be evicted + { + GpsTrackFile file; + file.Create(filePath, fileMaxItemCount); + + TEST(file.IsOpen(), ()); + + TEST_EQUAL(fileMaxItemCount, file.GetMaxCount(), ()); + + TEST_EQUAL(0, file.GetCount(), ()); + + for (size_t i = 0; i < 2 * fileMaxItemCount; ++i) + { + size_t evictedId; + size_t addedId = file.Append(Make(timestamp + i, ms::LatLon(i + 1000, i + 2000), i + 3000), evictedId); + TEST_EQUAL(i, addedId, ()); + if (i >= fileMaxItemCount) + { + TEST_EQUAL(i - fileMaxItemCount, evictedId, ()); + } + else + { + TEST_EQUAL(GpsTrackFile::kInvalidId, evictedId, ()); + } + } + + TEST_EQUAL(fileMaxItemCount, file.GetCount(), ()); + + file.Close(); + + TEST(!file.IsOpen(), ()); + } + + // Read GPS tracks + // Only last fileMaxItemCount must be in cyclic buffer + { + GpsTrackFile file; + file.Open(filePath, fileMaxItemCount); + + TEST(file.IsOpen(), ()); + TEST(!file.IsEmpty(), ()); + + TEST_EQUAL(fileMaxItemCount, file.GetMaxCount(), ()); + TEST_EQUAL(fileMaxItemCount, file.GetCount(), ()); + + size_t i = 0; + file.ForEach([&i,timestamp](location::GpsTrackInfo const & info, size_t id)->bool + { + TEST_EQUAL(id, i + fileMaxItemCount, ()); + TEST_EQUAL(info.m_timestamp, timestamp + i + fileMaxItemCount, ()); + TEST_EQUAL(info.m_latitude, i + 1000 + fileMaxItemCount, ()); + TEST_EQUAL(info.m_longitude, i + 2000 + fileMaxItemCount, ()); + TEST_EQUAL(info.m_speed, i + 3000 + fileMaxItemCount, ()); + ++i; + return true; + }); + + TEST_EQUAL(i, fileMaxItemCount, ()); + + file.Close(); + + TEST(!file.IsOpen(), ()); + } +} + +UNIT_TEST(GpsTrackFile_DropInTail) +{ + string const filePath = GetGpsTrackFilePath(); + MY_SCOPE_GUARD(gpsTestFileDeleter, bind(FileWriter::DeleteFileX, filePath)); + + time_t const t = system_clock::to_time_t(system_clock::now()); + double const timestamp = t; + LOG(LINFO, ("Timestamp", ctime(&t), timestamp)); + + GpsTrackFile file; + file.Create(filePath, 100); + + TEST(file.IsOpen(), ()); + + TEST_EQUAL(100, file.GetMaxCount(), ()); + + for (size_t i = 0; i < 50; ++i) + { + size_t evictedId; + size_t addedId = file.Append(Make(timestamp + i, ms::LatLon(i + 1000, i + 2000), i + 3000), evictedId); + TEST_EQUAL(i, addedId, ()); + TEST_EQUAL(GpsTrackFile::kInvalidId, evictedId, ()); + } + + TEST_EQUAL(50, file.GetCount(), ()); + + auto res = file.DropEarlierThan(timestamp + 4.5); // drop points 0,1,2,3,4 + TEST_EQUAL(res.first, 0, ()); + TEST_EQUAL(res.second, 4, ()); + + TEST_EQUAL(45, file.GetCount(), ()); + + size_t i = 5; // new first + file.ForEach([&i,timestamp](location::GpsTrackInfo const & info, size_t id)->bool + { + TEST_EQUAL(info.m_timestamp, timestamp + i, ()); + TEST_EQUAL(info.m_latitude, i + 1000, ()); + TEST_EQUAL(info.m_longitude, i + 2000, ()); + TEST_EQUAL(info.m_speed, i + 3000, ()); + ++i; + return true; + }); + + res = file.Clear(); + TEST_EQUAL(res.first, 5, ()); + TEST_EQUAL(res.second, 49, ()); + + TEST(file.IsEmpty(), ()) + + file.Close(); + + TEST(!file.IsOpen(), ()); +} + +UNIT_TEST(GpsTrackFile_DropInMiddle) +{ + string const filePath = GetGpsTrackFilePath(); + MY_SCOPE_GUARD(gpsTestFileDeleter, bind(FileWriter::DeleteFileX, filePath)); + + time_t const t = system_clock::to_time_t(system_clock::now()); + double const timestamp = t; + LOG(LINFO, ("Timestamp", ctime(&t), timestamp)); + + GpsTrackFile file; + file.Create(filePath, 100); + + TEST(file.IsOpen(), ()); + + TEST_EQUAL(100, file.GetMaxCount(), ()); + + for (size_t i = 0; i < 50; ++i) + { + size_t evictedId; + size_t addedId = file.Append(Make(timestamp + i, ms::LatLon(i + 1000,i + 2000), i + 3000), evictedId); + TEST_EQUAL(i, addedId, ()); + TEST_EQUAL(GpsTrackFile::kInvalidId, evictedId, ()); + } + + TEST_EQUAL(50, file.GetCount(), ()); + + auto res = file.DropEarlierThan(timestamp + 48.5); // drop all except last + TEST_EQUAL(res.first, 0, ()); + TEST_EQUAL(res.second, 48, ()); + + TEST_EQUAL(1, file.GetCount(), ()); + + size_t i = 49; // new first + file.ForEach([&i,timestamp](location::GpsTrackInfo const & info, size_t id)->bool + { + TEST_EQUAL(info.m_timestamp, timestamp + i, ()); + TEST_EQUAL(info.m_latitude, i + 1000, ()); + TEST_EQUAL(info.m_longitude, i + 2000, ()); + TEST_EQUAL(info.m_speed, i + 3000, ()); + ++i; + return true; + }); + + file.Close(); + + TEST(!file.IsOpen(), ()); +} + +UNIT_TEST(GpsTrackFile_DropAll) +{ + string const filePath = GetGpsTrackFilePath(); + MY_SCOPE_GUARD(gpsTestFileDeleter, bind(FileWriter::DeleteFileX, filePath)); + + time_t const t = system_clock::to_time_t(system_clock::now()); + double const timestamp = t; + LOG(LINFO, ("Timestamp", ctime(&t), timestamp)); + + GpsTrackFile file; + file.Create(filePath, 100); + + TEST(file.IsOpen(), ()); + + TEST_EQUAL(100, file.GetMaxCount(), ()); + + for (size_t i = 0; i < 50; ++i) + { + size_t evictedId; + size_t addedId = file.Append(Make(timestamp + i, ms::LatLon(i + 1000, i + 2000), i + 3000), evictedId); + TEST_EQUAL(i, addedId, ()); + TEST_EQUAL(GpsTrackFile::kInvalidId, evictedId, ()); + } + + TEST_EQUAL(50, file.GetCount(), ()); + + auto res = file.DropEarlierThan(timestamp + 51); // drop all + TEST_EQUAL(res.first, 0, ()); + TEST_EQUAL(res.second, 49, ()); + + TEST(file.IsEmpty(), ()); + + TEST_EQUAL(0, file.GetCount(), ()); + + file.Close(); + + TEST(!file.IsOpen(), ()); +} + +UNIT_TEST(GpsTrackFile_Clear) +{ + string const filePath = GetGpsTrackFilePath(); + MY_SCOPE_GUARD(gpsTestFileDeleter, bind(FileWriter::DeleteFileX, filePath)); + + time_t const t = system_clock::to_time_t(system_clock::now()); + double const timestamp = t; + LOG(LINFO, ("Timestamp", ctime(&t), timestamp)); + + GpsTrackFile file; + file.Create(filePath, 100); + + TEST(file.IsOpen(), ()); + + TEST_EQUAL(100, file.GetMaxCount(), ()); + + for (size_t i = 0; i < 50; ++i) + { + size_t evictedId; + size_t addedId = file.Append(Make(timestamp + i, ms::LatLon(i + 1000, i + 2000), i + 3000), evictedId); + TEST_EQUAL(i, addedId, ()); + TEST_EQUAL(GpsTrackFile::kInvalidId, evictedId, ()); + } + + TEST_EQUAL(50, file.GetCount(), ()); + + auto res = file.Clear(); + TEST_EQUAL(res.first, 0, ()); + TEST_EQUAL(res.second, 49, ()); + + TEST(file.IsEmpty(), ()); + + TEST_EQUAL(0, file.GetCount(), ()); + + file.Close(); + + TEST(!file.IsOpen(), ()); +} + +UNIT_TEST(GpsTrackFile_CreateOpenClose) +{ + string const filePath = GetGpsTrackFilePath(); + MY_SCOPE_GUARD(gpsTestFileDeleter, bind(FileWriter::DeleteFileX, filePath)); + + time_t const t = system_clock::to_time_t(system_clock::now()); + double const timestamp = t; + LOG(LINFO, ("Timestamp", ctime(&t), timestamp)); + + { + GpsTrackFile file; + file.Create(filePath, 100); + + TEST(file.IsOpen(), ()); + + TEST_EQUAL(100, file.GetMaxCount(), ()); + TEST_EQUAL(0, file.GetCount(), ()); + + file.Close(); + + TEST(!file.IsOpen(), ()); + } + + { + GpsTrackFile file; + file.Open(filePath, 100); + + TEST(file.IsOpen(), ()); + + TEST(file.IsEmpty(), ()); + TEST_EQUAL(100, file.GetMaxCount(), ()); + TEST_EQUAL(0, file.GetCount(), ()); + + auto res = file.Clear(); + TEST_EQUAL(res.first, GpsTrackFile::kInvalidId, ()); + TEST_EQUAL(res.second, GpsTrackFile::kInvalidId, ()); + + file.Close(); + + TEST(!file.IsOpen(), ()); + } +} diff --git a/map/map_tests/map_tests.pro b/map/map_tests/map_tests.pro index 9ddcc3d386..2adb196e82 100644 --- a/map/map_tests/map_tests.pro +++ b/map/map_tests/map_tests.pro @@ -33,6 +33,7 @@ SOURCES += \ geourl_test.cpp \ gps_track_container_test.cpp \ gps_track_collection_test.cpp \ + gps_track_file_test.cpp \ kmz_unarchive_test.cpp \ mwm_url_tests.cpp \ diff --git a/std/fstream.hpp b/std/fstream.hpp index efa32d4463..b013c77b58 100644 --- a/std/fstream.hpp +++ b/std/fstream.hpp @@ -7,6 +7,7 @@ #include using std::ofstream; using std::ifstream; +using std::fstream; using std::ios; #ifdef DEBUG_NEW