blob: c56396f157c4f02a09a76bc9c8774799c9bdb2eb [file] [log] [blame]
tkchin93411912015-07-22 12:12:17 -07001/*
2 * Copyright 2015 The WebRTC Project Authors. All rights reserved.
3 *
4 * Use of this source code is governed by a BSD-style license
5 * that can be found in the LICENSE file in the root of the source
6 * tree. An additional intellectual property rights grant can be found
7 * in the file PATENTS. All contributing project authors may
8 * be found in the AUTHORS file in the root of the source tree.
9 */
10
Steve Anton10542f22019-01-11 09:11:00 -080011#include "rtc_base/file_rotating_stream.h"
tkchin93411912015-07-22 12:12:17 -070012
Jonas Olsson55378f42018-05-25 10:23:10 +020013#include <cstdio>
tkchin93411912015-07-22 12:12:17 -070014#include <string>
Yves Gerey988cc082018-10-23 12:03:01 +020015#include <utility>
tkchin93411912015-07-22 12:12:17 -070016
Ali Tofigh7fa90572022-03-17 15:47:49 +010017#include "absl/strings/string_view.h"
18
Niels Möller7b3c76b2018-11-07 09:54:28 +010019#if defined(WEBRTC_WIN)
20#include <windows.h>
Jonas Olssona4d87372019-07-05 19:08:33 +020021
Steve Anton10542f22019-01-11 09:11:00 -080022#include "rtc_base/string_utils.h"
Niels Möller7b3c76b2018-11-07 09:54:28 +010023#else
Niels Möller260770c2018-11-07 15:08:18 +010024#include <dirent.h>
Niels Möller7b3c76b2018-11-07 09:54:28 +010025#include <sys/stat.h>
Niels Möllerb7396662018-11-13 11:55:19 +010026#include <unistd.h>
Niels Möller7b3c76b2018-11-07 09:54:28 +010027#endif // WEBRTC_WIN
28
Steve Anton2acd1632019-03-25 13:48:30 -070029#include "absl/algorithm/container.h"
Niels Möller7b3c76b2018-11-07 09:54:28 +010030#include "absl/strings/match.h"
Yves Gerey3e707812018-11-28 16:47:49 +010031#include "absl/types/optional.h"
Mirko Bonadei92ea95e2017-09-15 06:47:31 +020032#include "rtc_base/checks.h"
Yves Gerey2e00abc2018-10-05 15:39:24 +020033#include "rtc_base/logging.h"
Ali Tofigh7fa90572022-03-17 15:47:49 +010034#include "rtc_base/strings/string_builder.h"
tkchin93411912015-07-22 12:12:17 -070035
Jonas Olsson55378f42018-05-25 10:23:10 +020036// Note: We use fprintf for logging in the write paths of this stream to avoid
tkchin93411912015-07-22 12:12:17 -070037// infinite loops when logging.
38
39namespace rtc {
40
Niels Möller7b3c76b2018-11-07 09:54:28 +010041namespace {
42
Niels Möllerd9ac0582019-01-03 14:21:38 +010043const char kCallSessionLogPrefix[] = "webrtc_log";
44
Ali Tofigh7fa90572022-03-17 15:47:49 +010045std::string AddTrailingPathDelimiterIfNeeded(absl::string_view directory);
Niels Möller260770c2018-11-07 15:08:18 +010046
Artem Titov96e3b992021-07-26 16:03:14 +020047// `dir` must have a trailing delimiter. `prefix` must not include wild card
Niels Möller260770c2018-11-07 15:08:18 +010048// characters.
Ali Tofigh7fa90572022-03-17 15:47:49 +010049std::vector<std::string> GetFilesWithPrefix(absl::string_view directory,
50 absl::string_view prefix);
51bool DeleteFile(absl::string_view file);
52bool MoveFile(absl::string_view old_file, absl::string_view new_file);
53bool IsFile(absl::string_view file);
54bool IsFolder(absl::string_view file);
55absl::optional<size_t> GetFileSize(absl::string_view file);
Niels Möller7b3c76b2018-11-07 09:54:28 +010056
57#if defined(WEBRTC_WIN)
58
Ali Tofigh7fa90572022-03-17 15:47:49 +010059std::string AddTrailingPathDelimiterIfNeeded(absl::string_view directory) {
Niels Möller7b3c76b2018-11-07 09:54:28 +010060 if (absl::EndsWith(directory, "\\")) {
Ali Tofigh7fa90572022-03-17 15:47:49 +010061 return std::string(directory);
Niels Möller7b3c76b2018-11-07 09:54:28 +010062 }
Ali Tofigh7fa90572022-03-17 15:47:49 +010063 return std::string(directory) + "\\";
Niels Möller7b3c76b2018-11-07 09:54:28 +010064}
65
Ali Tofigh7fa90572022-03-17 15:47:49 +010066std::vector<std::string> GetFilesWithPrefix(absl::string_view directory,
67 absl::string_view prefix) {
Niels Möller260770c2018-11-07 15:08:18 +010068 RTC_DCHECK(absl::EndsWith(directory, "\\"));
Mirko Bonadei673f7e52019-03-25 09:01:02 +010069 WIN32_FIND_DATAW data;
Niels Möller260770c2018-11-07 15:08:18 +010070 HANDLE handle;
Ali Tofigh7fa90572022-03-17 15:47:49 +010071 StringBuilder pattern_builder{directory};
72 pattern_builder << prefix << "*";
73 handle = ::FindFirstFileW(ToUtf16(pattern_builder.str()).c_str(), &data);
Niels Möller260770c2018-11-07 15:08:18 +010074 if (handle == INVALID_HANDLE_VALUE)
75 return {};
76
77 std::vector<std::string> file_list;
78 do {
Ali Tofigh7fa90572022-03-17 15:47:49 +010079 StringBuilder file_builder{directory};
80 file_builder << ToUtf8(data.cFileName);
81 file_list.emplace_back(file_builder.Release());
Mirko Bonadei673f7e52019-03-25 09:01:02 +010082 } while (::FindNextFileW(handle, &data) == TRUE);
Niels Möller260770c2018-11-07 15:08:18 +010083
84 ::FindClose(handle);
85 return file_list;
86}
87
Ali Tofigh7fa90572022-03-17 15:47:49 +010088bool DeleteFile(absl::string_view file) {
Mirko Bonadei673f7e52019-03-25 09:01:02 +010089 return ::DeleteFileW(ToUtf16(file).c_str()) != 0;
Niels Möller260770c2018-11-07 15:08:18 +010090}
91
Ali Tofigh7fa90572022-03-17 15:47:49 +010092bool MoveFile(absl::string_view old_file, absl::string_view new_file) {
Mirko Bonadei673f7e52019-03-25 09:01:02 +010093 return ::MoveFileW(ToUtf16(old_file).c_str(), ToUtf16(new_file).c_str()) != 0;
Niels Möller260770c2018-11-07 15:08:18 +010094}
95
Ali Tofigh7fa90572022-03-17 15:47:49 +010096bool IsFile(absl::string_view file) {
Niels Möller260770c2018-11-07 15:08:18 +010097 WIN32_FILE_ATTRIBUTE_DATA data = {0};
Mirko Bonadei673f7e52019-03-25 09:01:02 +010098 if (0 == ::GetFileAttributesExW(ToUtf16(file).c_str(), GetFileExInfoStandard,
99 &data))
Niels Möller260770c2018-11-07 15:08:18 +0100100 return false;
101 return (data.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY) == 0;
102}
103
Ali Tofigh7fa90572022-03-17 15:47:49 +0100104bool IsFolder(absl::string_view file) {
Niels Möller7b3c76b2018-11-07 09:54:28 +0100105 WIN32_FILE_ATTRIBUTE_DATA data = {0};
Mirko Bonadei673f7e52019-03-25 09:01:02 +0100106 if (0 == ::GetFileAttributesExW(ToUtf16(file).c_str(), GetFileExInfoStandard,
107 &data))
Niels Möller7b3c76b2018-11-07 09:54:28 +0100108 return false;
109 return (data.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY) ==
110 FILE_ATTRIBUTE_DIRECTORY;
111}
112
Ali Tofigh7fa90572022-03-17 15:47:49 +0100113absl::optional<size_t> GetFileSize(absl::string_view file) {
Niels Möller260770c2018-11-07 15:08:18 +0100114 WIN32_FILE_ATTRIBUTE_DATA data = {0};
Mirko Bonadei673f7e52019-03-25 09:01:02 +0100115 if (::GetFileAttributesExW(ToUtf16(file).c_str(), GetFileExInfoStandard,
116 &data) == 0)
Niels Möller260770c2018-11-07 15:08:18 +0100117 return absl::nullopt;
118 return data.nFileSizeLow;
119}
120
Niels Möller7b3c76b2018-11-07 09:54:28 +0100121#else // defined(WEBRTC_WIN)
122
Ali Tofigh7fa90572022-03-17 15:47:49 +0100123std::string AddTrailingPathDelimiterIfNeeded(absl::string_view directory) {
Niels Möller7b3c76b2018-11-07 09:54:28 +0100124 if (absl::EndsWith(directory, "/")) {
Ali Tofigh7fa90572022-03-17 15:47:49 +0100125 return std::string(directory);
Niels Möller7b3c76b2018-11-07 09:54:28 +0100126 }
Ali Tofigh7fa90572022-03-17 15:47:49 +0100127 return std::string(directory) + "/";
Niels Möller7b3c76b2018-11-07 09:54:28 +0100128}
129
Ali Tofigh7fa90572022-03-17 15:47:49 +0100130std::vector<std::string> GetFilesWithPrefix(absl::string_view directory,
131 absl::string_view prefix) {
Niels Möller260770c2018-11-07 15:08:18 +0100132 RTC_DCHECK(absl::EndsWith(directory, "/"));
Ali Tofigh98bfd992022-07-22 22:10:49 +0200133 std::string directory_str(directory);
Ali Tofigh7fa90572022-03-17 15:47:49 +0100134 DIR* dir = ::opendir(directory_str.c_str());
Niels Möller260770c2018-11-07 15:08:18 +0100135 if (dir == nullptr)
136 return {};
137 std::vector<std::string> file_list;
138 for (struct dirent* dirent = ::readdir(dir); dirent;
139 dirent = ::readdir(dir)) {
140 std::string name = dirent->d_name;
Ali Tofigh7fa90572022-03-17 15:47:49 +0100141 if (name.compare(0, prefix.size(), prefix.data(), prefix.size()) == 0) {
142 file_list.emplace_back(directory_str + name);
Niels Möller260770c2018-11-07 15:08:18 +0100143 }
144 }
145 ::closedir(dir);
146 return file_list;
147}
148
Ali Tofigh7fa90572022-03-17 15:47:49 +0100149bool DeleteFile(absl::string_view file) {
150 return ::unlink(std::string(file).c_str()) == 0;
Niels Möller260770c2018-11-07 15:08:18 +0100151}
152
Ali Tofigh7fa90572022-03-17 15:47:49 +0100153bool MoveFile(absl::string_view old_file, absl::string_view new_file) {
154 return ::rename(std::string(old_file).c_str(),
155 std::string(new_file).c_str()) == 0;
Niels Möller260770c2018-11-07 15:08:18 +0100156}
157
Ali Tofigh7fa90572022-03-17 15:47:49 +0100158bool IsFile(absl::string_view file) {
Niels Möller260770c2018-11-07 15:08:18 +0100159 struct stat st;
Ali Tofigh7fa90572022-03-17 15:47:49 +0100160 int res = ::stat(std::string(file).c_str(), &st);
Niels Möller260770c2018-11-07 15:08:18 +0100161 // Treat symlinks, named pipes, etc. all as files.
162 return res == 0 && !S_ISDIR(st.st_mode);
163}
164
Ali Tofigh7fa90572022-03-17 15:47:49 +0100165bool IsFolder(absl::string_view file) {
Niels Möller7b3c76b2018-11-07 09:54:28 +0100166 struct stat st;
Ali Tofigh7fa90572022-03-17 15:47:49 +0100167 int res = ::stat(std::string(file).c_str(), &st);
Niels Möller7b3c76b2018-11-07 09:54:28 +0100168 return res == 0 && S_ISDIR(st.st_mode);
169}
170
Ali Tofigh7fa90572022-03-17 15:47:49 +0100171absl::optional<size_t> GetFileSize(absl::string_view file) {
Niels Möller260770c2018-11-07 15:08:18 +0100172 struct stat st;
Ali Tofigh7fa90572022-03-17 15:47:49 +0100173 if (::stat(std::string(file).c_str(), &st) != 0)
Niels Möller260770c2018-11-07 15:08:18 +0100174 return absl::nullopt;
175 return st.st_size;
176}
177
Niels Möller7b3c76b2018-11-07 09:54:28 +0100178#endif
179
180} // namespace
181
Ali Tofigh7fa90572022-03-17 15:47:49 +0100182FileRotatingStream::FileRotatingStream(absl::string_view dir_path,
183 absl::string_view file_prefix,
tkchin93411912015-07-22 12:12:17 -0700184 size_t max_file_size,
185 size_t num_files)
Niels Möller7b3c76b2018-11-07 09:54:28 +0100186 : dir_path_(AddTrailingPathDelimiterIfNeeded(dir_path)),
tkchin93411912015-07-22 12:12:17 -0700187 file_prefix_(file_prefix),
tkchin93411912015-07-22 12:12:17 -0700188 max_file_size_(max_file_size),
189 current_file_index_(0),
190 rotation_index_(0),
191 current_bytes_written_(0),
192 disable_buffering_(false) {
Niels Möller6ffe62a2019-01-08 13:22:57 +0100193 RTC_DCHECK_GT(max_file_size, 0);
194 RTC_DCHECK_GT(num_files, 1);
Niels Möller7b3c76b2018-11-07 09:54:28 +0100195 RTC_DCHECK(IsFolder(dir_path));
Niels Möller6ffe62a2019-01-08 13:22:57 +0100196 file_names_.clear();
197 for (size_t i = 0; i < num_files; ++i) {
198 file_names_.push_back(GetFilePath(i, num_files));
tkchin93411912015-07-22 12:12:17 -0700199 }
Niels Möller6ffe62a2019-01-08 13:22:57 +0100200 rotation_index_ = num_files - 1;
tkchin93411912015-07-22 12:12:17 -0700201}
202
Yves Gerey665174f2018-06-19 15:03:05 +0200203FileRotatingStream::~FileRotatingStream() {}
tkchin93411912015-07-22 12:12:17 -0700204
Niels Möllera9311b62021-03-25 15:29:02 +0100205bool FileRotatingStream::IsOpen() const {
206 return file_.is_open();
tkchin93411912015-07-22 12:12:17 -0700207}
208
Niels Möllera9311b62021-03-25 15:29:02 +0100209bool FileRotatingStream::Write(const void* data, size_t data_len) {
Niels Möller23213d92019-01-22 11:01:24 +0100210 if (!file_.is_open()) {
Jonas Olsson55378f42018-05-25 10:23:10 +0200211 std::fprintf(stderr, "Open() must be called before Write.\n");
Niels Möllera9311b62021-03-25 15:29:02 +0100212 return false;
tkchin93411912015-07-22 12:12:17 -0700213 }
Niels Möllera9311b62021-03-25 15:29:02 +0100214 while (data_len > 0) {
215 // Write as much as will fit in to the current file.
216 RTC_DCHECK_LT(current_bytes_written_, max_file_size_);
217 size_t remaining_bytes = max_file_size_ - current_bytes_written_;
218 size_t write_length = std::min(data_len, remaining_bytes);
tkchin93411912015-07-22 12:12:17 -0700219
Niels Möllera9311b62021-03-25 15:29:02 +0100220 if (!file_.Write(data, write_length)) {
221 return false;
222 }
223 if (disable_buffering_ && !file_.Flush()) {
224 return false;
225 }
Niels Möller23213d92019-01-22 11:01:24 +0100226
Niels Möllera9311b62021-03-25 15:29:02 +0100227 current_bytes_written_ += write_length;
228
229 // If we're done with this file, rotate it out.
230 if (current_bytes_written_ >= max_file_size_) {
231 RTC_DCHECK_EQ(current_bytes_written_, max_file_size_);
232 RotateFiles();
233 }
234 data_len -= write_length;
235 data =
236 static_cast<const void*>(static_cast<const char*>(data) + write_length);
Niels Möller23213d92019-01-22 11:01:24 +0100237 }
Niels Möllera9311b62021-03-25 15:29:02 +0100238 return true;
tkchin93411912015-07-22 12:12:17 -0700239}
240
241bool FileRotatingStream::Flush() {
Niels Möller23213d92019-01-22 11:01:24 +0100242 if (!file_.is_open()) {
tkchin93411912015-07-22 12:12:17 -0700243 return false;
244 }
Niels Möller23213d92019-01-22 11:01:24 +0100245 return file_.Flush();
tkchin93411912015-07-22 12:12:17 -0700246}
247
tkchin93411912015-07-22 12:12:17 -0700248void FileRotatingStream::Close() {
249 CloseCurrentFile();
250}
251
252bool FileRotatingStream::Open() {
Niels Möller6ffe62a2019-01-08 13:22:57 +0100253 // Delete existing files when opening for write.
254 std::vector<std::string> matching_files =
255 GetFilesWithPrefix(dir_path_, file_prefix_);
256 for (const auto& matching_file : matching_files) {
257 if (!DeleteFile(matching_file)) {
258 std::fprintf(stderr, "Failed to delete: %s\n", matching_file.c_str());
tkchin93411912015-07-22 12:12:17 -0700259 }
260 }
Niels Möller6ffe62a2019-01-08 13:22:57 +0100261 return OpenCurrentFile();
tkchin93411912015-07-22 12:12:17 -0700262}
263
264bool FileRotatingStream::DisableBuffering() {
265 disable_buffering_ = true;
Niels Möller23213d92019-01-22 11:01:24 +0100266 return true;
tkchin93411912015-07-22 12:12:17 -0700267}
268
269std::string FileRotatingStream::GetFilePath(size_t index) const {
henrikg91d6ede2015-09-17 00:24:34 -0700270 RTC_DCHECK_LT(index, file_names_.size());
tkchin93411912015-07-22 12:12:17 -0700271 return file_names_[index];
272}
273
274bool FileRotatingStream::OpenCurrentFile() {
275 CloseCurrentFile();
276
277 // Opens the appropriate file in the appropriate mode.
henrikg91d6ede2015-09-17 00:24:34 -0700278 RTC_DCHECK_LT(current_file_index_, file_names_.size());
tkchin93411912015-07-22 12:12:17 -0700279 std::string file_path = file_names_[current_file_index_];
Niels Möller6ffe62a2019-01-08 13:22:57 +0100280
281 // We should always be writing to the zero-th file.
282 RTC_DCHECK_EQ(current_file_index_, 0);
Niels Möller23213d92019-01-22 11:01:24 +0100283 int error;
284 file_ = webrtc::FileWrapper::OpenWriteOnly(file_path, &error);
285 if (!file_.is_open()) {
286 std::fprintf(stderr, "Failed to open: %s Error: %d\n", file_path.c_str(),
Jonas Olsson55378f42018-05-25 10:23:10 +0200287 error);
tkchin93411912015-07-22 12:12:17 -0700288 return false;
289 }
tkchin93411912015-07-22 12:12:17 -0700290 return true;
291}
292
293void FileRotatingStream::CloseCurrentFile() {
Niels Möller23213d92019-01-22 11:01:24 +0100294 if (!file_.is_open()) {
tkchin93411912015-07-22 12:12:17 -0700295 return;
296 }
297 current_bytes_written_ = 0;
Niels Möller23213d92019-01-22 11:01:24 +0100298 file_.Close();
tkchin93411912015-07-22 12:12:17 -0700299}
300
301void FileRotatingStream::RotateFiles() {
tkchin93411912015-07-22 12:12:17 -0700302 CloseCurrentFile();
Artem Titov96e3b992021-07-26 16:03:14 +0200303 // Rotates the files by deleting the file at `rotation_index_`, which is the
tkchin93411912015-07-22 12:12:17 -0700304 // oldest file and then renaming the newer files to have an incremented index.
305 // See header file comments for example.
hayscd02b0fa2015-12-08 13:59:05 -0800306 RTC_DCHECK_LT(rotation_index_, file_names_.size());
tkchin93411912015-07-22 12:12:17 -0700307 std::string file_to_delete = file_names_[rotation_index_];
Niels Möller260770c2018-11-07 15:08:18 +0100308 if (IsFile(file_to_delete)) {
309 if (!DeleteFile(file_to_delete)) {
Jonas Olsson55378f42018-05-25 10:23:10 +0200310 std::fprintf(stderr, "Failed to delete: %s\n", file_to_delete.c_str());
tkchin93411912015-07-22 12:12:17 -0700311 }
312 }
313 for (auto i = rotation_index_; i > 0; --i) {
314 std::string rotated_name = file_names_[i];
315 std::string unrotated_name = file_names_[i - 1];
Niels Möller260770c2018-11-07 15:08:18 +0100316 if (IsFile(unrotated_name)) {
317 if (!MoveFile(unrotated_name, rotated_name)) {
Jonas Olsson55378f42018-05-25 10:23:10 +0200318 std::fprintf(stderr, "Failed to move: %s to %s\n",
319 unrotated_name.c_str(), rotated_name.c_str());
tkchin93411912015-07-22 12:12:17 -0700320 }
321 }
322 }
323 // Create a new file for 0th index.
324 OpenCurrentFile();
325 OnRotation();
326}
327
tkchin93411912015-07-22 12:12:17 -0700328std::string FileRotatingStream::GetFilePath(size_t index,
329 size_t num_files) const {
henrikg91d6ede2015-09-17 00:24:34 -0700330 RTC_DCHECK_LT(index, num_files);
tkchin93411912015-07-22 12:12:17 -0700331
Jonas Olsson671cae22018-06-14 09:57:39 +0200332 const size_t buffer_size = 32;
333 char file_postfix[buffer_size];
334 // We want to zero pad the index so that it will sort nicely.
335 const int max_digits = std::snprintf(nullptr, 0, "%zu", num_files - 1);
336 RTC_DCHECK_LT(1 + max_digits, buffer_size);
337 std::snprintf(file_postfix, buffer_size, "_%0*zu", max_digits, index);
tkchin93411912015-07-22 12:12:17 -0700338
Niels Möller7b3c76b2018-11-07 09:54:28 +0100339 return dir_path_ + file_prefix_ + file_postfix;
tkchin93411912015-07-22 12:12:17 -0700340}
341
342CallSessionFileRotatingStream::CallSessionFileRotatingStream(
Ali Tofigh7fa90572022-03-17 15:47:49 +0100343 absl::string_view dir_path,
tkchin93411912015-07-22 12:12:17 -0700344 size_t max_total_log_size)
345 : FileRotatingStream(dir_path,
Niels Möllerd9ac0582019-01-03 14:21:38 +0100346 kCallSessionLogPrefix,
tkchin93411912015-07-22 12:12:17 -0700347 max_total_log_size / 2,
348 GetNumRotatingLogFiles(max_total_log_size) + 1),
349 max_total_log_size_(max_total_log_size),
350 num_rotations_(0) {
kwibergaf476c72016-11-28 15:21:39 -0800351 RTC_DCHECK_GE(max_total_log_size, 4);
tkchin93411912015-07-22 12:12:17 -0700352}
353
tkchin93411912015-07-22 12:12:17 -0700354const size_t CallSessionFileRotatingStream::kRotatingLogFileDefaultSize =
355 1024 * 1024;
356
357void CallSessionFileRotatingStream::OnRotation() {
358 ++num_rotations_;
359 if (num_rotations_ == 1) {
360 // On the first rotation adjust the max file size so subsequent files after
361 // the first are smaller.
362 SetMaxFileSize(GetRotatingLogSize(max_total_log_size_));
363 } else if (num_rotations_ == (GetNumFiles() - 1)) {
364 // On the next rotation the very first file is going to be deleted. Change
365 // the rotation index so this doesn't happen.
366 SetRotationIndex(GetRotationIndex() - 1);
367 }
368}
369
370size_t CallSessionFileRotatingStream::GetRotatingLogSize(
371 size_t max_total_log_size) {
372 size_t num_rotating_log_files = GetNumRotatingLogFiles(max_total_log_size);
373 size_t rotating_log_size = num_rotating_log_files > 2
374 ? kRotatingLogFileDefaultSize
375 : max_total_log_size / 4;
376 return rotating_log_size;
377}
378
379size_t CallSessionFileRotatingStream::GetNumRotatingLogFiles(
380 size_t max_total_log_size) {
381 // At minimum have two rotating files. Otherwise split the available log size
382 // evenly across 1MB files.
383 return std::max((size_t)2,
384 (max_total_log_size / 2) / kRotatingLogFileDefaultSize);
385}
386
Niels Möllerd9ac0582019-01-03 14:21:38 +0100387FileRotatingStreamReader::FileRotatingStreamReader(
Ali Tofigh7fa90572022-03-17 15:47:49 +0100388 absl::string_view dir_path,
389 absl::string_view file_prefix) {
Niels Möllerd9ac0582019-01-03 14:21:38 +0100390 file_names_ = GetFilesWithPrefix(AddTrailingPathDelimiterIfNeeded(dir_path),
391 file_prefix);
392
393 // Plain sort of the file names would sort by age, i.e., oldest last. Using
394 // std::greater gives us the desired chronological older, oldest first.
Steve Anton2acd1632019-03-25 13:48:30 -0700395 absl::c_sort(file_names_, std::greater<std::string>());
Niels Möllerd9ac0582019-01-03 14:21:38 +0100396}
397
398FileRotatingStreamReader::~FileRotatingStreamReader() = default;
399
400size_t FileRotatingStreamReader::GetSize() const {
401 size_t total_size = 0;
402 for (const auto& file_name : file_names_) {
403 total_size += GetFileSize(file_name).value_or(0);
404 }
405 return total_size;
406}
407
408size_t FileRotatingStreamReader::ReadAll(void* buffer, size_t size) const {
409 size_t done = 0;
410 for (const auto& file_name : file_names_) {
411 if (done < size) {
Niels Möller23213d92019-01-22 11:01:24 +0100412 webrtc::FileWrapper f = webrtc::FileWrapper::OpenReadOnly(file_name);
413 if (!f.is_open()) {
Niels Möllerd9ac0582019-01-03 14:21:38 +0100414 break;
415 }
Niels Möller23213d92019-01-22 11:01:24 +0100416 done += f.Read(static_cast<char*>(buffer) + done, size - done);
Niels Möllerd9ac0582019-01-03 14:21:38 +0100417 } else {
418 break;
419 }
420 }
421 return done;
422}
423
424CallSessionFileRotatingStreamReader::CallSessionFileRotatingStreamReader(
Ali Tofigh7fa90572022-03-17 15:47:49 +0100425 absl::string_view dir_path)
Niels Möllerd9ac0582019-01-03 14:21:38 +0100426 : FileRotatingStreamReader(dir_path, kCallSessionLogPrefix) {}
427
tkchin93411912015-07-22 12:12:17 -0700428} // namespace rtc