forked from hoylabs/OpenDTU-OnBattery
-
Notifications
You must be signed in to change notification settings - Fork 0
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
This implements RFC5424 version of the protocol. I didn't use https://github.com/arcao/Syslog since the protocol itself is trivial and most of the libraries functionality is not needed here. The library also doesn't support setting the PROCID field, which is set to a random id to indicate a reboot here.
- Loading branch information
Showing
5 changed files
with
181 additions
and
1 deletion.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,34 @@ | ||
// SPDX-License-Identifier: GPL-2.0-or-later | ||
#pragma once | ||
#include <WiFiUdp.h> | ||
#include <TaskSchedulerDeclarations.h> | ||
#include <mutex> | ||
|
||
class SyslogLogger { | ||
public: | ||
SyslogLogger(); | ||
void init(Scheduler& scheduler); | ||
void updateSettings(const String&& hostname); | ||
void write(const uint8_t *buffer, size_t size); | ||
|
||
private: | ||
void loop(); | ||
void disable(); | ||
void enable(); | ||
bool resolveAndStart(); | ||
bool isResolved() const { | ||
return _address != INADDR_NONE; | ||
} | ||
|
||
Task _loopTask; | ||
std::mutex _mutex; | ||
WiFiUDP _udp; | ||
IPAddress _address; | ||
String _syslog_hostname; | ||
String _proc_id; | ||
String _header; | ||
uint16_t _port; | ||
bool _enabled; | ||
}; | ||
|
||
extern SyslogLogger Syslog; |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,138 @@ | ||
// SPDX-License-Identifier: GPL-2.0-or-later | ||
/* | ||
* Copyright (C) 2022-2024 Thomas Basler and others | ||
*/ | ||
#include <HardwareSerial.h> | ||
#include <ESPmDNS.h> | ||
#include "defaults.h" | ||
#include "SyslogLogger.h" | ||
#include "Configuration.h" | ||
#include "MessageOutput.h" | ||
#include "NetworkSettings.h" | ||
|
||
SyslogLogger::SyslogLogger() | ||
: _loopTask(TASK_IMMEDIATE, TASK_FOREVER, std::bind(&SyslogLogger::loop, this)) | ||
{ | ||
} | ||
|
||
void SyslogLogger::init(Scheduler& scheduler) | ||
{ | ||
// PROCID change indicates a restart. | ||
_proc_id = String(esp_random(), HEX); | ||
|
||
scheduler.addTask(_loopTask); | ||
_loopTask.enable(); | ||
} | ||
|
||
void SyslogLogger::updateSettings(const String&& hostname) | ||
{ | ||
auto& config = Configuration.get().Syslog; | ||
|
||
// Disable logger while it is reconfigured. | ||
disable(); | ||
|
||
if (!config.Enabled) { | ||
MessageOutput.println("[SyslogLogger] Syslog not enabled"); | ||
return; | ||
} | ||
|
||
_port = config.Port; | ||
_syslog_hostname = config.Hostname; | ||
if (_syslog_hostname.isEmpty()) { | ||
MessageOutput.println("[SyslogLogger] Hostname not configured"); | ||
return; | ||
} | ||
|
||
MessageOutput.printf("[SyslogLogger] Logging to %s!\r\n", _syslog_hostname.c_str()); | ||
|
||
_header = "<7>1 - "; // RFC5424: Facility KERNEL, severity DEBUG, version 1, NIL timestamp. | ||
_header += hostname; | ||
_header += " OpenDTU "; | ||
_header += _proc_id; | ||
// NIL values for message id and structured // data; utf-8 BOM. | ||
_header += " - - \xEF\xBB\xBF"; | ||
|
||
// Enable logger. | ||
enable(); | ||
} | ||
|
||
void SyslogLogger::write(const uint8_t *buffer, size_t size) | ||
{ | ||
std::lock_guard<std::mutex> lock(_mutex); | ||
if (!_enabled || !isResolved()) { | ||
return; | ||
} | ||
for (int i = 0; i < size; i++) { | ||
uint8_t c = buffer[i]; | ||
bool overflow = false; | ||
if (c != '\r' && c != '\n') { | ||
// Replace control and non-ASCII characters with '?'. | ||
overflow = !_udp.write(c >= 0x20 && c < 0x7f ? c : '?'); | ||
} | ||
if (c == '\n' || overflow) { | ||
_udp.endPacket(); | ||
_udp.beginPacket(_address, _port); | ||
_udp.print(_header); | ||
} | ||
} | ||
} | ||
|
||
void SyslogLogger::disable() | ||
{ | ||
MessageOutput.println("[SyslogLogger] Disable"); | ||
std::lock_guard<std::mutex> lock(_mutex); | ||
if (_enabled) { | ||
_enabled = false; | ||
_address = INADDR_NONE; | ||
_udp.stop(); | ||
} | ||
} | ||
|
||
void SyslogLogger::enable() | ||
{ | ||
// Bind random source port. | ||
if (!_udp.begin(0)) { | ||
MessageOutput.println("[SyslogLogger] No sockets available"); | ||
return; | ||
} | ||
|
||
std::lock_guard<std::mutex> lock(_mutex); | ||
_enabled = true; | ||
} | ||
|
||
bool SyslogLogger::resolveAndStart() | ||
{ | ||
if (Configuration.get().Mdns.Enabled) { | ||
_address = MDNS.queryHost(_syslog_hostname); // INADDR_NONE if failed | ||
} | ||
if (_address != INADDR_NONE) { | ||
if (!_udp.beginPacket(_address, _port)) { | ||
return false; | ||
} | ||
} else { | ||
if (!_udp.beginPacket(_syslog_hostname.c_str(), _port)) { | ||
return false; | ||
} | ||
_address = _udp.remoteIP(); // Store resolved address. | ||
} | ||
_udp.print(_header); | ||
_udp.print("[SyslogLogger] Logging to "); | ||
_udp.print(_syslog_hostname); | ||
_udp.endPacket(); | ||
_udp.beginPacket(_address, _port); | ||
_udp.print(_header); | ||
return true; | ||
} | ||
|
||
void SyslogLogger::loop() | ||
{ | ||
std::lock_guard<std::mutex> lock(_mutex); | ||
if (!_enabled || !NetworkSettings.isConnected() || isResolved()) { | ||
return; | ||
} | ||
if (!resolveAndStart()) { | ||
_enabled = false; | ||
} | ||
} | ||
|
||
SyslogLogger Syslog; |