aboutsummaryrefslogtreecommitdiff
path: root/src/utils/logger.cc
blob: 92d26d917dea4686e82a5eba5354991370befb71 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
#include "logger.h"
#include <iostream>
#include <fstream>
#include <chrono>
#include <iomanip>
#include <ctime>

// Global logger instance
Logger g_logger;

Logger::Logger() 
    : current_level(LogLevel::INFO)
    , console_enabled(true)
    , file_enabled(false) {
}

Logger::~Logger() {
    if (file_stream) {
        file_stream->flush();
    }
}

void Logger::set_level(LogLevel level) {
    current_level = level;
}

void Logger::debug(const std::string& message) {
    log(LogLevel::DEBUG, message);
}

void Logger::info(const std::string& message) {
    log(LogLevel::INFO, message);
}

void Logger::warning(const std::string& message) {
    log(LogLevel::WARNING, message);
}

void Logger::error(const std::string& message) {
    log(LogLevel::ERROR, message);
}

void Logger::fatal(const std::string& message) {
    log(LogLevel::FATAL, message);
}

void Logger::log(LogLevel level, const std::string& message) {
    if (level >= current_level) {
        write_log(level, message);
    }
}

void Logger::set_output_file(const std::string& filename) {
    output_filename = filename;
    if (file_enabled && !filename.empty()) {
        file_stream = std::make_unique<std::ofstream>(filename, std::ios::app);
        if (!file_stream->good()) {
            std::cerr << "Failed to open log file: " << filename << std::endl;
            file_stream.reset();
        }
    }
}

void Logger::enable_console(bool enable) {
    console_enabled = enable;
}

void Logger::enable_file(bool enable) {
    file_enabled = enable;
    if (enable && !output_filename.empty()) {
        set_output_file(output_filename);
    } else if (!enable) {
        file_stream.reset();
    }
}

std::string Logger::level_to_string(LogLevel level) {
    switch (level) {
        case LogLevel::DEBUG:   return "DEBUG";
        case LogLevel::INFO:    return "INFO";
        case LogLevel::WARNING: return "WARNING";
        case LogLevel::ERROR:   return "ERROR";
        case LogLevel::FATAL:   return "FATAL";
        default:                return "UNKNOWN";
    }
}

std::string Logger::get_timestamp() {
    auto now = std::chrono::system_clock::now();
    auto time_t = std::chrono::system_clock::to_time_t(now);
    auto ms = std::chrono::duration_cast<std::chrono::milliseconds>(
        now.time_since_epoch()) % 1000;
    
    std::stringstream ss;
    ss << std::put_time(std::localtime(&time_t), "%Y-%m-%d %H:%M:%S");
    ss << '.' << std::setfill('0') << std::setw(3) << ms.count();
    return ss.str();
}

void Logger::write_log(LogLevel level, const std::string& message) {
    std::string timestamp = get_timestamp();
    std::string level_str = level_to_string(level);
    std::string log_entry = "[" + timestamp + "] [" + level_str + "] " + message + "\n";
    
    if (console_enabled) {
        if (level == LogLevel::ERROR || level == LogLevel::FATAL) {
            std::cerr << log_entry;
        } else {
            std::cout << log_entry;
        }
    }
    
    if (file_enabled && file_stream && file_stream->good()) {
        *file_stream << log_entry;
        file_stream->flush();
    }
}

// LogStream implementation
LogStream::LogStream(Logger& logger, LogLevel level) 
    : logger(logger), level(level) {
}

LogStream::~LogStream() {
    logger.log(level, stream.str());
}