153 lines
		
	
	
		
			4.6 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
		
		
			
		
	
	
			153 lines
		
	
	
		
			4.6 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| 
								 | 
							
								// Copyright(c) 2015-present, Gabi Melman & spdlog contributors.
							 | 
						||
| 
								 | 
							
								// Distributed under the MIT License (http://opensource.org/licenses/MIT)
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								#pragma once
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								#ifndef SPDLOG_HEADER_ONLY
							 | 
						||
| 
								 | 
							
								    #include <spdlog/details/file_helper.h>
							 | 
						||
| 
								 | 
							
								#endif
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								#include <spdlog/common.h>
							 | 
						||
| 
								 | 
							
								#include <spdlog/details/os.h>
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								#include <cerrno>
							 | 
						||
| 
								 | 
							
								#include <chrono>
							 | 
						||
| 
								 | 
							
								#include <cstdio>
							 | 
						||
| 
								 | 
							
								#include <string>
							 | 
						||
| 
								 | 
							
								#include <thread>
							 | 
						||
| 
								 | 
							
								#include <tuple>
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								namespace spdlog {
							 | 
						||
| 
								 | 
							
								namespace details {
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								SPDLOG_INLINE file_helper::file_helper(const file_event_handlers &event_handlers)
							 | 
						||
| 
								 | 
							
								    : event_handlers_(event_handlers) {}
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								SPDLOG_INLINE file_helper::~file_helper() { close(); }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								SPDLOG_INLINE void file_helper::open(const filename_t &fname, bool truncate) {
							 | 
						||
| 
								 | 
							
								    close();
							 | 
						||
| 
								 | 
							
								    filename_ = fname;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    auto *mode = SPDLOG_FILENAME_T("ab");
							 | 
						||
| 
								 | 
							
								    auto *trunc_mode = SPDLOG_FILENAME_T("wb");
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    if (event_handlers_.before_open) {
							 | 
						||
| 
								 | 
							
								        event_handlers_.before_open(filename_);
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								    for (int tries = 0; tries < open_tries_; ++tries) {
							 | 
						||
| 
								 | 
							
								        // create containing folder if not exists already.
							 | 
						||
| 
								 | 
							
								        os::create_dir(os::dir_name(fname));
							 | 
						||
| 
								 | 
							
								        if (truncate) {
							 | 
						||
| 
								 | 
							
								            // Truncate by opening-and-closing a tmp file in "wb" mode, always
							 | 
						||
| 
								 | 
							
								            // opening the actual log-we-write-to in "ab" mode, since that
							 | 
						||
| 
								 | 
							
								            // interacts more politely with eternal processes that might
							 | 
						||
| 
								 | 
							
								            // rotate/truncate the file underneath us.
							 | 
						||
| 
								 | 
							
								            std::FILE *tmp;
							 | 
						||
| 
								 | 
							
								            if (os::fopen_s(&tmp, fname, trunc_mode)) {
							 | 
						||
| 
								 | 
							
								                continue;
							 | 
						||
| 
								 | 
							
								            }
							 | 
						||
| 
								 | 
							
								            std::fclose(tmp);
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								        if (!os::fopen_s(&fd_, fname, mode)) {
							 | 
						||
| 
								 | 
							
								            if (event_handlers_.after_open) {
							 | 
						||
| 
								 | 
							
								                event_handlers_.after_open(filename_, fd_);
							 | 
						||
| 
								 | 
							
								            }
							 | 
						||
| 
								 | 
							
								            return;
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        details::os::sleep_for_millis(open_interval_);
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    throw_spdlog_ex("Failed opening file " + os::filename_to_str(filename_) + " for writing",
							 | 
						||
| 
								 | 
							
								                    errno);
							 | 
						||
| 
								 | 
							
								}
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								SPDLOG_INLINE void file_helper::reopen(bool truncate) {
							 | 
						||
| 
								 | 
							
								    if (filename_.empty()) {
							 | 
						||
| 
								 | 
							
								        throw_spdlog_ex("Failed re opening file - was not opened before");
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								    this->open(filename_, truncate);
							 | 
						||
| 
								 | 
							
								}
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								SPDLOG_INLINE void file_helper::flush() {
							 | 
						||
| 
								 | 
							
								    if (std::fflush(fd_) != 0) {
							 | 
						||
| 
								 | 
							
								        throw_spdlog_ex("Failed flush to file " + os::filename_to_str(filename_), errno);
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								}
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								SPDLOG_INLINE void file_helper::sync() {
							 | 
						||
| 
								 | 
							
								    if (!os::fsync(fd_)) {
							 | 
						||
| 
								 | 
							
								        throw_spdlog_ex("Failed to fsync file " + os::filename_to_str(filename_), errno);
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								}
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								SPDLOG_INLINE void file_helper::close() {
							 | 
						||
| 
								 | 
							
								    if (fd_ != nullptr) {
							 | 
						||
| 
								 | 
							
								        if (event_handlers_.before_close) {
							 | 
						||
| 
								 | 
							
								            event_handlers_.before_close(filename_, fd_);
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        std::fclose(fd_);
							 | 
						||
| 
								 | 
							
								        fd_ = nullptr;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        if (event_handlers_.after_close) {
							 | 
						||
| 
								 | 
							
								            event_handlers_.after_close(filename_);
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								}
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								SPDLOG_INLINE void file_helper::write(const memory_buf_t &buf) {
							 | 
						||
| 
								 | 
							
								    if (fd_ == nullptr) return;
							 | 
						||
| 
								 | 
							
								    size_t msg_size = buf.size();
							 | 
						||
| 
								 | 
							
								    auto data = buf.data();
							 | 
						||
| 
								 | 
							
								    if (std::fwrite(data, 1, msg_size, fd_) != msg_size) {
							 | 
						||
| 
								 | 
							
								        throw_spdlog_ex("Failed writing to file " + os::filename_to_str(filename_), errno);
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								}
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								SPDLOG_INLINE size_t file_helper::size() const {
							 | 
						||
| 
								 | 
							
								    if (fd_ == nullptr) {
							 | 
						||
| 
								 | 
							
								        throw_spdlog_ex("Cannot use size() on closed file " + os::filename_to_str(filename_));
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								    return os::filesize(fd_);
							 | 
						||
| 
								 | 
							
								}
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								SPDLOG_INLINE const filename_t &file_helper::filename() const { return filename_; }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								//
							 | 
						||
| 
								 | 
							
								// return file path and its extension:
							 | 
						||
| 
								 | 
							
								//
							 | 
						||
| 
								 | 
							
								// "mylog.txt" => ("mylog", ".txt")
							 | 
						||
| 
								 | 
							
								// "mylog" => ("mylog", "")
							 | 
						||
| 
								 | 
							
								// "mylog." => ("mylog.", "")
							 | 
						||
| 
								 | 
							
								// "/dir1/dir2/mylog.txt" => ("/dir1/dir2/mylog", ".txt")
							 | 
						||
| 
								 | 
							
								//
							 | 
						||
| 
								 | 
							
								// the starting dot in filenames is ignored (hidden files):
							 | 
						||
| 
								 | 
							
								//
							 | 
						||
| 
								 | 
							
								// ".mylog" => (".mylog". "")
							 | 
						||
| 
								 | 
							
								// "my_folder/.mylog" => ("my_folder/.mylog", "")
							 | 
						||
| 
								 | 
							
								// "my_folder/.mylog.txt" => ("my_folder/.mylog", ".txt")
							 | 
						||
| 
								 | 
							
								SPDLOG_INLINE std::tuple<filename_t, filename_t> file_helper::split_by_extension(
							 | 
						||
| 
								 | 
							
								    const filename_t &fname) {
							 | 
						||
| 
								 | 
							
								    auto ext_index = fname.rfind('.');
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    // no valid extension found - return whole path and empty string as
							 | 
						||
| 
								 | 
							
								    // extension
							 | 
						||
| 
								 | 
							
								    if (ext_index == filename_t::npos || ext_index == 0 || ext_index == fname.size() - 1) {
							 | 
						||
| 
								 | 
							
								        return std::make_tuple(fname, filename_t());
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    // treat cases like "/etc/rc.d/somelogfile or "/abc/.hiddenfile"
							 | 
						||
| 
								 | 
							
								    auto folder_index = fname.find_last_of(details::os::folder_seps_filename);
							 | 
						||
| 
								 | 
							
								    if (folder_index != filename_t::npos && folder_index >= ext_index - 1) {
							 | 
						||
| 
								 | 
							
								        return std::make_tuple(fname, filename_t());
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    // finally - return a valid base and extension tuple
							 | 
						||
| 
								 | 
							
								    return std::make_tuple(fname.substr(0, ext_index), fname.substr(ext_index));
							 | 
						||
| 
								 | 
							
								}
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								}  // namespace details
							 | 
						||
| 
								 | 
							
								}  // namespace spdlog
							 |