ESPHome 2026.3.0-dev
Loading...
Searching...
No Matches
esphome_syslog.cpp
Go to the documentation of this file.
1#include "esphome_syslog.h"
2
5#include "esphome/core/time.h"
6
7namespace esphome::syslog {
8
9// Map log levels to syslog severity using an array, indexed by ESPHome log level (1-7)
11 3, // NONE
12 3, // ERROR
13 4, // WARN
14 5, // INFO
15 6, // CONFIG
16 7, // DEBUG
17 7, // VERBOSE
18 7 // VERY_VERBOSE
19};
20
23 this, [](void *self, uint8_t level, const char *tag, const char *message, size_t message_len) {
24 static_cast<Syslog *>(self)->on_log(level, tag, message, message_len);
25 });
26}
27
28void Syslog::on_log(uint8_t level, const char *tag, const char *message, size_t message_len) {
29 this->log_(level, tag, message, message_len);
30}
31
32void Syslog::log_(const int level, const char *tag, const char *message, size_t message_len) const {
33 if (level > this->log_level_)
34 return;
35 // Syslog PRI calculation: facility * 8 + severity
36 int severity = 7;
37 if ((unsigned) level <= 7) {
38 severity = LOG_LEVEL_TO_SYSLOG_SEVERITY[level];
39 }
40 int pri = this->facility_ * 8 + severity;
41
42 size_t len = message_len;
43 // remove color formatting
44 if (this->strip_ && message[0] == 0x1B && len > 11) {
45 message += 7;
46 len -= 11;
47 }
48
49 // Build syslog packet on stack (508 bytes chosen as practical limit for syslog over UDP)
50 char packet[508];
51 size_t offset = 0;
52 size_t remaining = sizeof(packet);
53
54 // Write PRI - abort if this fails as packet would be malformed
55 offset = buf_append_printf(packet, sizeof(packet), 0, "<%d>", pri);
56 if (offset == 0) {
57 return; // PRI always produces at least "<0>" (3 chars), so 0 means error
58 }
59 remaining -= offset;
60
61 // Write timestamp directly into packet (RFC 5424: use "-" if time not valid or strftime fails)
62 auto now = this->time_->now();
63 size_t ts_written = now.is_valid() ? now.strftime(packet + offset, remaining, "%b %e %H:%M:%S") : 0;
64 if (ts_written > 0) {
65 offset += ts_written;
66 } else if (remaining > 0) {
67 packet[offset++] = '-';
68 }
69
70 // Write hostname, tag, and message
71 offset = buf_append_printf(packet, sizeof(packet), offset, " %s %s: %.*s", App.get_name().c_str(), tag, (int) len,
72 message);
73 // Clamp to exclude null terminator position if buffer was filled
74 if (offset >= sizeof(packet)) {
75 offset = sizeof(packet) - 1;
76 }
77
78 if (offset > 0) {
79 this->parent_->send_packet(reinterpret_cast<const uint8_t *>(packet), offset);
80 }
81}
82
83} // namespace esphome::syslog
const std::string & get_name() const
Get the name of this Application set by pre_setup().
void add_log_callback(void *instance, void(*fn)(void *, uint8_t, const char *, const char *, size_t))
Register a log callback to receive log messages.
Definition logger.h:190
time::RealTimeClock * time_
void log_(int level, const char *tag, const char *message, size_t message_len) const
void on_log(uint8_t level, const char *tag, const char *message, size_t message_len)
ESPTime now()
Get the time in the currently defined timezone.
void send_packet(const uint8_t *data, size_t size)
const char * message
Definition component.cpp:38
Logger * global_logger
Definition logger.cpp:278
constexpr int LOG_LEVEL_TO_SYSLOG_SEVERITY[]
std::string size_t len
Definition helpers.h:817
Application App
Global storage of Application pointer - only one Application can exist.
bool is_valid() const
Check if this ESPTime is valid (all fields in range and year is greater than or equal to 2019)
Definition time.h:78