Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat-logger-update #1173

Merged
merged 9 commits into from
Sep 10, 2024
2 changes: 2 additions & 0 deletions engine/CMakeLists.txt
Original file line number Diff line number Diff line change
@@ -115,6 +115,7 @@ if(DEFINED CMAKE_JS_INC)

add_library(${PROJECT_NAME} SHARED addon.cc
${CMAKE_CURRENT_SOURCE_DIR}/utils/cpuid/cpu_info.cc
${CMAKE_CURRENT_SOURCE_DIR}/utils/file_logger.cc
${CMAKE_JS_SRC}
)

@@ -131,6 +132,7 @@ if(DEFINED CMAKE_JS_INC)
else() # Official build
add_executable(${PROJECT_NAME} main.cc
${CMAKE_CURRENT_SOURCE_DIR}/utils/cpuid/cpu_info.cc
${CMAKE_CURRENT_SOURCE_DIR}/utils/file_logger.cc
)
endif()

26 changes: 21 additions & 5 deletions engine/main.cc
Original file line number Diff line number Diff line change
@@ -6,6 +6,7 @@
#include "utils/archive_utils.h"
#include "utils/cortex_utils.h"
#include "utils/dylib.h"
#include "utils/file_logger.h"
#include "utils/file_manager_utils.h"
#include "utils/logging_utils.h"

@@ -29,16 +30,19 @@ void RunServer() {
LOG_INFO << "Host: " << config.host << " Port: " << config.port << "\n";

// Create logs/ folder and setup log to file
std::filesystem::create_directory(cortex_utils::logs_folder);
trantor::AsyncFileLogger asyncFileLogger;
asyncFileLogger.setFileName(cortex_utils::logs_base_name);
std::filesystem::create_directory(config.logFolderPath + "/" +
cortex_utils::logs_folder);
trantor::FileLogger asyncFileLogger;
asyncFileLogger.setFileName(config.logFolderPath + "/" +
nguyenhoangthuan99 marked this conversation as resolved.
Show resolved Hide resolved
cortex_utils::logs_base_name);
asyncFileLogger.setMaxLines(
cortex_utils::log_file_max_lines); // Keep last 100000 lines
asyncFileLogger.startLogging();
trantor::Logger::setOutputFunction(
[&](const char* msg, const uint64_t len) {
asyncFileLogger.output(msg, len);
asyncFileLogger.output_(msg, len);
},
[&]() { asyncFileLogger.flush(); });
asyncFileLogger.setFileSizeLimit(cortex_utils::log_file_size_limit);
// Number of cortex.cpp threads
// if (argc > 1) {
// thread_num = std::atoi(argv[1]);
@@ -154,6 +158,18 @@ int main(int argc, char* argv[]) {
RunServer();
return 0;
} else {
auto config = file_manager_utils::GetCortexConfig();
trantor::FileLogger asyncFileLogger;
asyncFileLogger.setFileName(config.logFolderPath + "/" +
cortex_utils::logs_cli_base_name);
asyncFileLogger.setMaxLines(
cortex_utils::log_file_max_lines); // Keep last 1 million lines
asyncFileLogger.startLogging();
trantor::Logger::setOutputFunction(
[&](const char* msg, const uint64_t len) {
asyncFileLogger.output_(msg, len);
},
[&]() { asyncFileLogger.flush(); });
CommandLineParser clp;
clp.SetupCommand(argc, argv);
return 0;
4 changes: 3 additions & 1 deletion engine/utils/config_yaml_utils.h
Original file line number Diff line number Diff line change
@@ -8,6 +8,7 @@

namespace config_yaml_utils {
struct CortexConfig {
std::string logFolderPath;
std::string dataFolderPath;
std::string host;
std::string port;
@@ -27,6 +28,7 @@ inline void DumpYamlConfig(const CortexConfig& config,
throw std::runtime_error("Failed to open output file.");
}
YAML::Node node;
node["logFolderPath"] = config.logFolderPath;
node["dataFolderPath"] = config.dataFolderPath;
node["host"] = config.host;
node["port"] = config.port;
@@ -49,6 +51,7 @@ inline CortexConfig FromYaml(const std::string& path,
try {
auto node = YAML::LoadFile(config_file_path.string());
CortexConfig config = {
.logFolderPath = node["logFolderPath"].as<std::string>(),
.dataFolderPath = node["dataFolderPath"].as<std::string>(),
.host = node["host"].as<std::string>(),
.port = node["port"].as<std::string>(),
@@ -60,5 +63,4 @@ inline CortexConfig FromYaml(const std::string& path,
}
}


} // namespace config_yaml_utils
4 changes: 3 additions & 1 deletion engine/utils/cortex_utils.h
Original file line number Diff line number Diff line change
@@ -34,8 +34,10 @@ constexpr static auto kTensorrtLlmPath = "/engines/cortex.tensorrt-llm";

inline std::string models_folder = "./models";
inline std::string logs_folder = "./logs";
inline std::string logs_base_name = "./logs/cortex";
inline std::string logs_base_name = "./logs/cortex.log";
inline std::string logs_cli_base_name = "./logs/cortex-cli.log";
inline size_t log_file_size_limit = 20000000; // ~20 mb
nguyenhoangthuan99 marked this conversation as resolved.
Show resolved Hide resolved
inline size_t log_file_max_lines = 100000; // 100,000 lines
nguyenhoangthuan99 marked this conversation as resolved.
Show resolved Hide resolved

inline std::string extractBase64(const std::string& input) {
std::regex pattern("base64,(.*)");
175 changes: 175 additions & 0 deletions engine/utils/file_logger.cc
Original file line number Diff line number Diff line change
@@ -0,0 +1,175 @@
#include "file_logger.h"
#include <algorithm>
#include <iostream>
#include <sstream>

#ifdef _WIN32
#include <io.h>
#define ftruncate _chsize
#else
#include <unistd.h>
#endif
#include <string.h>

using namespace trantor;

FileLogger::FileLogger() : AsyncFileLogger() {}

FileLogger::~FileLogger() = default;

void FileLogger::output_(const char* msg, const uint64_t len) {
if (!circular_log_file_ptr_) {
circular_log_file_ptr_ =
std::make_unique<CircularLogFile>(fileBaseName_, max_lines_);
}
circular_log_file_ptr_->writeLog(msg, len);
}

FileLogger::CircularLogFile::CircularLogFile(const std::string& fileName,
uint64_t maxLines)
: max_lines_(maxLines), file_name_(fileName) {
OpenFile();
LoadExistingLines();
TruncateFileIfNeeded();
}

FileLogger::CircularLogFile::~CircularLogFile() {
CloseFile();
}
void FileLogger::CircularLogFile::writeLog(const char* logLine,
const uint64_t len) {
nguyenhoangthuan99 marked this conversation as resolved.
Show resolved Hide resolved
if (!fp_)
return;

std::string logString(logLine, len);
std::istringstream iss(logString);
std::string line;
while (std::getline(iss, line)) {
if (totalLines_ >= max_lines_) {
lineBuffer_.pop_front();
--totalLines_;
}
lineBuffer_.push_back(line);
++totalLines_;
AppendToFile(line + "\n");
++linesWrittenSinceLastTruncate_;
if (linesWrittenSinceLastTruncate_ >= TRUNCATE_CHECK_INTERVAL) {

TruncateFileIfNeeded();
}
}
}
void FileLogger::CircularLogFile::flush() {
if (fp_) {
fflush(fp_);
}
}

void FileLogger::CircularLogFile::TruncateFileIfNeeded() {
// std::cout<<"Truncating file "<< totalLines_ <<std::endl;
if (!fp_ || totalLines_ < max_lines_)
return;

// Close the current file
fclose(fp_);
fp_ = nullptr;

// Open a temporary file for writing
std::string tempFileName = file_name_ + ".temp";
FILE* tempFile = fopen(tempFileName.c_str(), "w");
if (!tempFile) {

std::cout << "Error opening temporary file for truncation: "
<< strerror(errno) << std::endl;
OpenFile(); // Reopen the original file
return;
}

// Write only the last max_lines_ lines to the temporary file
size_t startIndex =
lineBuffer_.size() > max_lines_ ? lineBuffer_.size() - max_lines_ : 0;

for (size_t i = startIndex; i < lineBuffer_.size(); ++i) {
fprintf(tempFile, "%s\n", lineBuffer_[i].c_str());
}

fclose(tempFile);

// Replace the original file with the temporary file
if (std::rename(tempFileName.c_str(), file_name_.c_str()) != 0) {
std::cout << "Error replacing original file with truncated file: "
<< strerror(errno) << std::endl;
std::remove(tempFileName.c_str()); // Clean up the temporary file
} else {
// std::cout<<"Truncating file" <<std::endl;
totalLines_ =
lineBuffer_.size() > max_lines_ ? max_lines_ : lineBuffer_.size();
}

// Reopen the file
OpenFile();
// LoadExistingLines();
linesWrittenSinceLastTruncate_ = 0;
}

void FileLogger::CircularLogFile::OpenFile() {
#ifdef _WIN32
auto wFileName = utils::toNativePath(file_name_);
fp_ = _wfopen(wFileName.c_str(), L"r+");
#else
fp_ = fopen(file_name_.c_str(), "r+");
#endif

if (!fp_) {
// If file doesn't exist, create it
#ifdef _WIN32
fp_ = _wfopen(wFileName.c_str(), L"w+");
#else
fp_ = fopen(file_name_.c_str(), "w+");
#endif

if (!fp_) {
std::cerr << "Error opening file: " << strerror(errno) << std::endl;
}
}
}
void FileLogger::CircularLogFile::LoadExistingLines() {
if (!fp_)
return;

// Move to the beginning of the file
fseek(fp_, 0, SEEK_SET);

lineBuffer_.clear();
totalLines_ = 0;

std::string line;
char buffer[4096];
while (fgets(buffer, sizeof(buffer), fp_) != nullptr) {
line = buffer;
if (!line.empty() && line.back() == '\n') {
line.pop_back(); // Remove trailing newline
}
if (totalLines_ >= max_lines_) {
lineBuffer_.pop_front();
}
lineBuffer_.push_back(line);
++totalLines_;
}

// Move back to the end of the file for appending
fseek(fp_, 0, SEEK_END);
}
void FileLogger::CircularLogFile::AppendToFile(const std::string& line) {
if (fp_) {
fwrite(line.c_str(), 1, line.length(), fp_);
fflush(fp_);
}
}

void FileLogger::CircularLogFile::CloseFile() {
if (fp_) {
fclose(fp_);
fp_ = nullptr;
}
}
68 changes: 68 additions & 0 deletions engine/utils/file_logger.h
Original file line number Diff line number Diff line change
@@ -0,0 +1,68 @@
#pragma once

#include <trantor/utils/AsyncFileLogger.h>
#include <trantor/utils/Utilities.h>
#include <deque>

#ifdef _WIN32
#include <windows.h>
#else
#include <unistd.h>
#endif

namespace trantor {

class TRANTOR_EXPORT FileLogger : public AsyncFileLogger {
nguyenhoangthuan99 marked this conversation as resolved.
Show resolved Hide resolved
public:
FileLogger();
~FileLogger();

/**
* @brief Set the maximum number of lines to keep in the log file.
*
* @param maxLines
*/
void setMaxLines(uint64_t maxLines) { max_lines_ = maxLines; }

/**
* @brief Set the log file name.
*
* @param fileName The full name of the log file.
*/
void setFileName(const std::string& fileName) {
filePath_ = "./";
fileBaseName_ = fileName;
fileExtName_ = "";
}
void output_(const char* msg, const uint64_t len);

protected:
class CircularLogFile {
public:
CircularLogFile(const std::string& fileName, uint64_t maxLines);
~CircularLogFile();

void writeLog(const char* logLine, const uint64_t len);
void flush();
uint64_t getLength() const { return totalLines_; }

private:
FILE* fp_{nullptr};
uint64_t max_lines_;
uint64_t totalLines_{0};
nguyenhoangthuan99 marked this conversation as resolved.
Show resolved Hide resolved
std::string file_name_;
std::deque<std::string> lineBuffer_;
uint64_t linesWrittenSinceLastTruncate_{0};
static const uint64_t TRUNCATE_CHECK_INTERVAL = 1000;

void LoadExistingLines();
void TruncateFileIfNeeded();
void AppendToFile(const std::string& line);
void OpenFile();
void CloseFile();
};
std::unique_ptr<CircularLogFile> circular_log_file_ptr_;
uint64_t max_lines_{100000}; // Default to 100000 lines
};

} // namespace trantor
22 changes: 22 additions & 0 deletions engine/utils/file_manager_utils.h
Original file line number Diff line number Diff line change
@@ -135,6 +135,7 @@ inline void CreateConfigFileIfNotExist() {
CTL_INF("Default data folder path: " + defaultDataFolderPath.string());

auto config = config_yaml_utils::CortexConfig{
.logFolderPath = defaultDataFolderPath.string(),
.dataFolderPath = defaultDataFolderPath.string(),
.host = config_yaml_utils::kDefaultHost,
.port = config_yaml_utils::kDefaultPort,
@@ -169,6 +170,27 @@ inline std::filesystem::path GetCortexDataPath() {
return data_folder_path;
}

inline std::filesystem::path GetCortexLogPath() {
// TODO: We will need to support user to move the data folder to other place.
// TODO: get the variant of cortex. As discussed, we will have: prod, beta, nightly
// currently we will store cortex data at ~/cortexcpp
auto config = GetCortexConfig();
std::filesystem::path log_folder_path;
if (!config.logFolderPath.empty()) {
log_folder_path = std::filesystem::path(config.logFolderPath);
} else {
auto home_path = GetHomeDirectoryPath();
log_folder_path = home_path / config_yaml_utils::kCortexFolderName;
}

if (!std::filesystem::exists(log_folder_path)) {
CTL_INF("Cortex log folder not found. Create one: " +
log_folder_path.string());
std::filesystem::create_directory(log_folder_path);
}
return log_folder_path;
}

inline std::filesystem::path GetModelsContainerPath() {
auto cortex_path = GetCortexDataPath();
auto models_container_path = cortex_path / "models";
1 change: 0 additions & 1 deletion engine/utils/logging_utils.h
Original file line number Diff line number Diff line change
@@ -30,4 +30,3 @@ inline bool log_verbose = false;
} else { \
std::cout << msg << std::endl; \
}