| /* |
| * Copyright 2015 The WebRTC Project Authors. All rights reserved. |
| * |
| * Use of this source code is governed by a BSD-style license |
| * that can be found in the LICENSE file in the root of the source |
| * tree. An additional intellectual property rights grant can be found |
| * in the file PATENTS. All contributing project authors may |
| * be found in the AUTHORS file in the root of the source tree. |
| */ |
| |
| #include "rtc_base/file_rotating_stream.h" |
| |
| #include <cstdio> |
| #include <string> |
| #include <utility> |
| |
| #include "absl/strings/string_view.h" |
| |
| #if defined(WEBRTC_WIN) |
| #include <windows.h> |
| |
| #include "rtc_base/string_utils.h" |
| #else |
| #include <dirent.h> |
| #include <sys/stat.h> |
| #include <unistd.h> |
| #endif // WEBRTC_WIN |
| |
| #include "absl/algorithm/container.h" |
| #include "absl/strings/match.h" |
| #include "absl/types/optional.h" |
| #include "rtc_base/checks.h" |
| #include "rtc_base/logging.h" |
| #include "rtc_base/strings/string_builder.h" |
| |
| // Note: We use fprintf for logging in the write paths of this stream to avoid |
| // infinite loops when logging. |
| |
| namespace rtc { |
| |
| namespace { |
| |
| const char kCallSessionLogPrefix[] = "webrtc_log"; |
| |
| std::string AddTrailingPathDelimiterIfNeeded(absl::string_view directory); |
| |
| // `dir` must have a trailing delimiter. `prefix` must not include wild card |
| // characters. |
| std::vector<std::string> GetFilesWithPrefix(absl::string_view directory, |
| absl::string_view prefix); |
| bool DeleteFile(absl::string_view file); |
| bool MoveFile(absl::string_view old_file, absl::string_view new_file); |
| bool IsFile(absl::string_view file); |
| bool IsFolder(absl::string_view file); |
| absl::optional<size_t> GetFileSize(absl::string_view file); |
| |
| #if defined(WEBRTC_WIN) |
| |
| std::string AddTrailingPathDelimiterIfNeeded(absl::string_view directory) { |
| if (absl::EndsWith(directory, "\\")) { |
| return std::string(directory); |
| } |
| return std::string(directory) + "\\"; |
| } |
| |
| std::vector<std::string> GetFilesWithPrefix(absl::string_view directory, |
| absl::string_view prefix) { |
| RTC_DCHECK(absl::EndsWith(directory, "\\")); |
| WIN32_FIND_DATAW data; |
| HANDLE handle; |
| StringBuilder pattern_builder{directory}; |
| pattern_builder << prefix << "*"; |
| handle = ::FindFirstFileW(ToUtf16(pattern_builder.str()).c_str(), &data); |
| if (handle == INVALID_HANDLE_VALUE) |
| return {}; |
| |
| std::vector<std::string> file_list; |
| do { |
| StringBuilder file_builder{directory}; |
| file_builder << ToUtf8(data.cFileName); |
| file_list.emplace_back(file_builder.Release()); |
| } while (::FindNextFileW(handle, &data) == TRUE); |
| |
| ::FindClose(handle); |
| return file_list; |
| } |
| |
| bool DeleteFile(absl::string_view file) { |
| return ::DeleteFileW(ToUtf16(file).c_str()) != 0; |
| } |
| |
| bool MoveFile(absl::string_view old_file, absl::string_view new_file) { |
| return ::MoveFileW(ToUtf16(old_file).c_str(), ToUtf16(new_file).c_str()) != 0; |
| } |
| |
| bool IsFile(absl::string_view file) { |
| WIN32_FILE_ATTRIBUTE_DATA data = {0}; |
| if (0 == ::GetFileAttributesExW(ToUtf16(file).c_str(), GetFileExInfoStandard, |
| &data)) |
| return false; |
| return (data.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY) == 0; |
| } |
| |
| bool IsFolder(absl::string_view file) { |
| WIN32_FILE_ATTRIBUTE_DATA data = {0}; |
| if (0 == ::GetFileAttributesExW(ToUtf16(file).c_str(), GetFileExInfoStandard, |
| &data)) |
| return false; |
| return (data.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY) == |
| FILE_ATTRIBUTE_DIRECTORY; |
| } |
| |
| absl::optional<size_t> GetFileSize(absl::string_view file) { |
| WIN32_FILE_ATTRIBUTE_DATA data = {0}; |
| if (::GetFileAttributesExW(ToUtf16(file).c_str(), GetFileExInfoStandard, |
| &data) == 0) |
| return absl::nullopt; |
| return data.nFileSizeLow; |
| } |
| |
| #else // defined(WEBRTC_WIN) |
| |
| std::string AddTrailingPathDelimiterIfNeeded(absl::string_view directory) { |
| if (absl::EndsWith(directory, "/")) { |
| return std::string(directory); |
| } |
| return std::string(directory) + "/"; |
| } |
| |
| std::vector<std::string> GetFilesWithPrefix(absl::string_view directory, |
| absl::string_view prefix) { |
| RTC_DCHECK(absl::EndsWith(directory, "/")); |
| std::string directory_str(directory); |
| DIR* dir = ::opendir(directory_str.c_str()); |
| if (dir == nullptr) |
| return {}; |
| std::vector<std::string> file_list; |
| for (struct dirent* dirent = ::readdir(dir); dirent; |
| dirent = ::readdir(dir)) { |
| std::string name = dirent->d_name; |
| if (name.compare(0, prefix.size(), prefix.data(), prefix.size()) == 0) { |
| file_list.emplace_back(directory_str + name); |
| } |
| } |
| ::closedir(dir); |
| return file_list; |
| } |
| |
| bool DeleteFile(absl::string_view file) { |
| return ::unlink(std::string(file).c_str()) == 0; |
| } |
| |
| bool MoveFile(absl::string_view old_file, absl::string_view new_file) { |
| return ::rename(std::string(old_file).c_str(), |
| std::string(new_file).c_str()) == 0; |
| } |
| |
| bool IsFile(absl::string_view file) { |
| struct stat st; |
| int res = ::stat(std::string(file).c_str(), &st); |
| // Treat symlinks, named pipes, etc. all as files. |
| return res == 0 && !S_ISDIR(st.st_mode); |
| } |
| |
| bool IsFolder(absl::string_view file) { |
| struct stat st; |
| int res = ::stat(std::string(file).c_str(), &st); |
| return res == 0 && S_ISDIR(st.st_mode); |
| } |
| |
| absl::optional<size_t> GetFileSize(absl::string_view file) { |
| struct stat st; |
| if (::stat(std::string(file).c_str(), &st) != 0) |
| return absl::nullopt; |
| return st.st_size; |
| } |
| |
| #endif |
| |
| } // namespace |
| |
| FileRotatingStream::FileRotatingStream(absl::string_view dir_path, |
| absl::string_view file_prefix, |
| size_t max_file_size, |
| size_t num_files) |
| : dir_path_(AddTrailingPathDelimiterIfNeeded(dir_path)), |
| file_prefix_(file_prefix), |
| max_file_size_(max_file_size), |
| current_file_index_(0), |
| rotation_index_(0), |
| current_bytes_written_(0), |
| disable_buffering_(false) { |
| RTC_DCHECK_GT(max_file_size, 0); |
| RTC_DCHECK_GT(num_files, 1); |
| RTC_DCHECK(IsFolder(dir_path)); |
| file_names_.clear(); |
| for (size_t i = 0; i < num_files; ++i) { |
| file_names_.push_back(GetFilePath(i, num_files)); |
| } |
| rotation_index_ = num_files - 1; |
| } |
| |
| FileRotatingStream::~FileRotatingStream() {} |
| |
| bool FileRotatingStream::IsOpen() const { |
| return file_.is_open(); |
| } |
| |
| bool FileRotatingStream::Write(const void* data, size_t data_len) { |
| if (!file_.is_open()) { |
| std::fprintf(stderr, "Open() must be called before Write.\n"); |
| return false; |
| } |
| while (data_len > 0) { |
| // Write as much as will fit in to the current file. |
| RTC_DCHECK_LT(current_bytes_written_, max_file_size_); |
| size_t remaining_bytes = max_file_size_ - current_bytes_written_; |
| size_t write_length = std::min(data_len, remaining_bytes); |
| |
| if (!file_.Write(data, write_length)) { |
| return false; |
| } |
| if (disable_buffering_ && !file_.Flush()) { |
| return false; |
| } |
| |
| current_bytes_written_ += write_length; |
| |
| // If we're done with this file, rotate it out. |
| if (current_bytes_written_ >= max_file_size_) { |
| RTC_DCHECK_EQ(current_bytes_written_, max_file_size_); |
| RotateFiles(); |
| } |
| data_len -= write_length; |
| data = |
| static_cast<const void*>(static_cast<const char*>(data) + write_length); |
| } |
| return true; |
| } |
| |
| bool FileRotatingStream::Flush() { |
| if (!file_.is_open()) { |
| return false; |
| } |
| return file_.Flush(); |
| } |
| |
| void FileRotatingStream::Close() { |
| CloseCurrentFile(); |
| } |
| |
| bool FileRotatingStream::Open() { |
| // Delete existing files when opening for write. |
| std::vector<std::string> matching_files = |
| GetFilesWithPrefix(dir_path_, file_prefix_); |
| for (const auto& matching_file : matching_files) { |
| if (!DeleteFile(matching_file)) { |
| std::fprintf(stderr, "Failed to delete: %s\n", matching_file.c_str()); |
| } |
| } |
| return OpenCurrentFile(); |
| } |
| |
| bool FileRotatingStream::DisableBuffering() { |
| disable_buffering_ = true; |
| return true; |
| } |
| |
| std::string FileRotatingStream::GetFilePath(size_t index) const { |
| RTC_DCHECK_LT(index, file_names_.size()); |
| return file_names_[index]; |
| } |
| |
| bool FileRotatingStream::OpenCurrentFile() { |
| CloseCurrentFile(); |
| |
| // Opens the appropriate file in the appropriate mode. |
| RTC_DCHECK_LT(current_file_index_, file_names_.size()); |
| std::string file_path = file_names_[current_file_index_]; |
| |
| // We should always be writing to the zero-th file. |
| RTC_DCHECK_EQ(current_file_index_, 0); |
| int error; |
| file_ = webrtc::FileWrapper::OpenWriteOnly(file_path, &error); |
| if (!file_.is_open()) { |
| std::fprintf(stderr, "Failed to open: %s Error: %d\n", file_path.c_str(), |
| error); |
| return false; |
| } |
| return true; |
| } |
| |
| void FileRotatingStream::CloseCurrentFile() { |
| if (!file_.is_open()) { |
| return; |
| } |
| current_bytes_written_ = 0; |
| file_.Close(); |
| } |
| |
| void FileRotatingStream::RotateFiles() { |
| CloseCurrentFile(); |
| // Rotates the files by deleting the file at `rotation_index_`, which is the |
| // oldest file and then renaming the newer files to have an incremented index. |
| // See header file comments for example. |
| RTC_DCHECK_LT(rotation_index_, file_names_.size()); |
| std::string file_to_delete = file_names_[rotation_index_]; |
| if (IsFile(file_to_delete)) { |
| if (!DeleteFile(file_to_delete)) { |
| std::fprintf(stderr, "Failed to delete: %s\n", file_to_delete.c_str()); |
| } |
| } |
| for (auto i = rotation_index_; i > 0; --i) { |
| std::string rotated_name = file_names_[i]; |
| std::string unrotated_name = file_names_[i - 1]; |
| if (IsFile(unrotated_name)) { |
| if (!MoveFile(unrotated_name, rotated_name)) { |
| std::fprintf(stderr, "Failed to move: %s to %s\n", |
| unrotated_name.c_str(), rotated_name.c_str()); |
| } |
| } |
| } |
| // Create a new file for 0th index. |
| OpenCurrentFile(); |
| OnRotation(); |
| } |
| |
| std::string FileRotatingStream::GetFilePath(size_t index, |
| size_t num_files) const { |
| RTC_DCHECK_LT(index, num_files); |
| |
| const size_t buffer_size = 32; |
| char file_postfix[buffer_size]; |
| // We want to zero pad the index so that it will sort nicely. |
| const int max_digits = std::snprintf(nullptr, 0, "%zu", num_files - 1); |
| RTC_DCHECK_LT(1 + max_digits, buffer_size); |
| std::snprintf(file_postfix, buffer_size, "_%0*zu", max_digits, index); |
| |
| return dir_path_ + file_prefix_ + file_postfix; |
| } |
| |
| CallSessionFileRotatingStream::CallSessionFileRotatingStream( |
| absl::string_view dir_path, |
| size_t max_total_log_size) |
| : FileRotatingStream(dir_path, |
| kCallSessionLogPrefix, |
| max_total_log_size / 2, |
| GetNumRotatingLogFiles(max_total_log_size) + 1), |
| max_total_log_size_(max_total_log_size), |
| num_rotations_(0) { |
| RTC_DCHECK_GE(max_total_log_size, 4); |
| } |
| |
| const size_t CallSessionFileRotatingStream::kRotatingLogFileDefaultSize = |
| 1024 * 1024; |
| |
| void CallSessionFileRotatingStream::OnRotation() { |
| ++num_rotations_; |
| if (num_rotations_ == 1) { |
| // On the first rotation adjust the max file size so subsequent files after |
| // the first are smaller. |
| SetMaxFileSize(GetRotatingLogSize(max_total_log_size_)); |
| } else if (num_rotations_ == (GetNumFiles() - 1)) { |
| // On the next rotation the very first file is going to be deleted. Change |
| // the rotation index so this doesn't happen. |
| SetRotationIndex(GetRotationIndex() - 1); |
| } |
| } |
| |
| size_t CallSessionFileRotatingStream::GetRotatingLogSize( |
| size_t max_total_log_size) { |
| size_t num_rotating_log_files = GetNumRotatingLogFiles(max_total_log_size); |
| size_t rotating_log_size = num_rotating_log_files > 2 |
| ? kRotatingLogFileDefaultSize |
| : max_total_log_size / 4; |
| return rotating_log_size; |
| } |
| |
| size_t CallSessionFileRotatingStream::GetNumRotatingLogFiles( |
| size_t max_total_log_size) { |
| // At minimum have two rotating files. Otherwise split the available log size |
| // evenly across 1MB files. |
| return std::max((size_t)2, |
| (max_total_log_size / 2) / kRotatingLogFileDefaultSize); |
| } |
| |
| FileRotatingStreamReader::FileRotatingStreamReader( |
| absl::string_view dir_path, |
| absl::string_view file_prefix) { |
| file_names_ = GetFilesWithPrefix(AddTrailingPathDelimiterIfNeeded(dir_path), |
| file_prefix); |
| |
| // Plain sort of the file names would sort by age, i.e., oldest last. Using |
| // std::greater gives us the desired chronological older, oldest first. |
| absl::c_sort(file_names_, std::greater<std::string>()); |
| } |
| |
| FileRotatingStreamReader::~FileRotatingStreamReader() = default; |
| |
| size_t FileRotatingStreamReader::GetSize() const { |
| size_t total_size = 0; |
| for (const auto& file_name : file_names_) { |
| total_size += GetFileSize(file_name).value_or(0); |
| } |
| return total_size; |
| } |
| |
| size_t FileRotatingStreamReader::ReadAll(void* buffer, size_t size) const { |
| size_t done = 0; |
| for (const auto& file_name : file_names_) { |
| if (done < size) { |
| webrtc::FileWrapper f = webrtc::FileWrapper::OpenReadOnly(file_name); |
| if (!f.is_open()) { |
| break; |
| } |
| done += f.Read(static_cast<char*>(buffer) + done, size - done); |
| } else { |
| break; |
| } |
| } |
| return done; |
| } |
| |
| CallSessionFileRotatingStreamReader::CallSessionFileRotatingStreamReader( |
| absl::string_view dir_path) |
| : FileRotatingStreamReader(dir_path, kCallSessionLogPrefix) {} |
| |
| } // namespace rtc |