A Syslog-based logging framework for Erlang. This project is inspired by the
great work put in the two projects
sasl_syslog and
lager. In fact syslog
tries to combine both
approaches. In a nutshell syslog
can be seen as a lightweight version of the
lager
logging framework supporting only a fully compliant, Erlang-only Syslog
backend allowing remote logging.
The main difference between sasl_syslog
and syslog
is that sasl_syslog
does only provide logging of error_logger
reports. However, the error_logger
is known for its bad memory consumption behaviour under heavy load (due to its
asynchronous logging mechanism). Additionally, syslog
provides an optional
RFC 3164 (BSD Syslog) compliant protocol backend which is the only standard
supported by old versions of syslog-ng
and rsyslog
.
Compared to lager
, syslog
has a very limited set of backends. As its name
implies syslog
is specialized in delivering its messages using Syslog only,
there is no file or console backend, no custom-written and configurable log
rotation, no line formatting and no tracing support. However, syslog
does not
rely on port drivers or NIFs to implement the Syslog protocol and it includes
measures to enhance the overall robustness of a node, e.g. load distribution,
back-pressure mechanisms, throughput optimization, etc.
- Log messages and standard
error_logger
reports formatted according to RFC 3164 (BSD Syslog) or RFC 5424 (Syslog Protocol) without the need for drivers, ports or NIFs. - System independent logging to local or remote facilities using one of the
following transports:
- UDP (RFC 3164 and RFC 5426)
- TCP (Octet Counting according to RFC 6587)
- TCP/TLS (RFC 5425)
- Robust event handlers - using supervised event handler subscription.
- Optionally independent error messages using a separate facility.
- Get the well-known SASL event format for
supervisor
andcrash
reports. - Configurable verbosity of SASL printing format (printing depth is also configurable).
- Load distribution between all concurrently logging processes by moving the message formatting into the calling process(es).
- Built-in
lager
backend to bridge between both frameworks.
- Configurable maximum packet size.
- Utilize the RFC 5424 STRUCTURED-DATA field for
info_report
,warning_report
orerror_report
withproplists
.
The syslog
application already comes with sensible defaults (except for
the facilities used and the destination host). However, many things can be
customized if desired. For this purpose the following configuration options
are available and can be configured in the application environment:
-
{msg_queue_limit, Limit :: pos_integer() | infinity}
Specifies a limit for the number of entries allowed in the
error_logger
message queue. If the message queue exceeds this limitsyslog
will drop the events exceeding the limit. Default isinfinity
. -
{drop_percentage, Percentage :: 1..100}
Specifies the number of messages that will be dropped (additionally), if the
error_logger
message queue exceeds the configuredmsg_queue_limit
. Thedrop_percentage
is given as percentage (of themsg_queue_limit
). E.g. ifdrop_percentage
is10
(the default),msg_queue_limit
is100
and the current length of theerror_logger
message queue is 120, then20 + 10
messages will be dropped to give thesyslog_error_h
handler some air to catch up. -
{protocol, rfc3164 | rfc5424 | {rfc3164 | rfc5424, tcp | udp} | {rfc3164 | rfc5424, udp, [gen_udp:option()]} | {rfc3164 | rfc5424, tcp, [gen_tcp:option()]} | {rfc5424, tls, [ssl:connect_option()]}}
Specifies which protocol/transport standard should be used to format and send outgoing Syslog packets. Please note that empty/default TLS/SSL options are currently not supported. Default formatting is
rfc3164
and the default transport isudp
. -
{use_rfc5424_bom, boolean()}
Specifies whether the RFC5424 protocol backend should include the UTF-8 BOM in the message part of a Syslog packet. Default is
false
. -
{dest_host, inet:ip_address() | inet:hostname()}
Specifies the host to which Syslog packets will be sent. Default is
{127, 0, 0, 1}
. -
{dest_port, inet:port_number()}
Specifies the port to which Syslog packets will be sent. Default is
514
. -
{facility, syslog:facility()}
Specifies the facility Syslog packets will be sent with. Default is
daemon
. -
{crash_facility, syslog:facility()}
Specifies the facility Syslog packets with severity
crash
, will be sent with. It replaces the previouserror_facility
property. This accompanies a change in behaviour. Starting with release2.0.0
error messages will also be sent withfacility
. Only crash and supervisor reports will be sent to this (maybe) separate facility. If the values of the propertiesfacility
andcrash_facility
differ a short one-line summary will additionally be sent tofacility
. Default isdaemon
. -
{verbose, true | {false, Depth :: pos_integer()}}
Configures which pretty printing mode to use when formatting
error_logger
reports (that is progress reports, not format messages). If verbose istrue
the~p
format character will be used when formatting terms. This will likely result in a lot of multiline strings. If set to{false, Depth}
the~P
format character is used along with the specified printing depth. Default istrue
. -
{no_progress, boolean()}
This flag can be used to completely omit progress reports from the log output. So if you you don't care when a new process is started, set this flag to
true
. Default isfalse
. -
{app_name, atom() | string() | binary()}
Configured the value reported in the
APP-NAME
field of Syslog messages. If not set (the default), the name part of the node name will be used. If the node is not alive (not running in distributed mode) the stringbeam
will be used. -
{log_level, syslog:severity()}
Configures the minimal log level. All messages with a severity value smaller then the configured level will be discarded. Default is
debug
(discard nothing). -
{async, boolean()}
Specifies whether log message offloading into the
syslog_logger
server is done synchronously or asynchronously. It is highly recommended to leave this at its default valuefalse
because asynchronous delivery is really dangerous. A simple log burst of a few thousand processes may be enough to take your node down (due to out-of-memory). -
{timeout, pos_integer()}
Specifies an upper bound in milliseconds a single process gets blocked in a log call (only if
{async, false}
). This is useful if your application has performance restrictions (under high load) and you are willing to sacrifice safety for this reason. It's still safer to set this to a lower value than completely switching to{async, true}
. Default is1000
ms.
If your application really needs fast asynchronous logging and you like to live
dangerously, logging can be done either with the error_logger
or the syslog
API and the syslog
application should be configured with {async, true}
and
{msg_queue_limit, infinity}
. This sets syslog
into asynchronous delivery
mode and all message queues are allowed to grow indefinitely.
The syslog
application will disable the standard error_logger
TTY output on
application startup. This has nothing to do with the standard SASL logging. It
only disables non-SASL logging via, for example error_logger:info_msg/1,2
.
This kind of standard logging can be re-enabled at any time using the following:
error_logger:tty(true).
The syslog
application will not touch the standard SASL report handlers
attached to the error_logger
when SASL starts. However, having SASL progress
reports on TTY can be quite annoying when trying to use the shell. The correct
way to disable this output is to configure the SASL application in the
sys.config
of a release, for example the following line will instruct SASL
not to attach any TTY handlers to the error_logger
:
{sasl, [{sasl_error_logger, false}]}
The syslog
application will log everything that is logged using the standard
error_logger
API. However, this should not be used for ordinary application
logging.
The proper way to add logging to your application is to use the API functions
provided by the syslog
module. These functions are similar to the ones
provided by the error_logger
module and should feel familiar (see the
*msg/1,2
functions).
The syslog
application comes with a built-in, optional backend for lager
.
This is especially useful if your release has dependencies that require lager
although you wish to forward logging using syslog
only. To forward lager
logging into syslog
you can use something like the following in your
sys.config
:
{lager, [{handlers, [{syslog_lager_backend, []}]}]}
Performance profiling has been made with a the benchmark script located in the
benchmark
subdirectory. The figures below show the results of best-of-five
runs on an Intel(R) Core(TM) i7-4790 CPU running OTP 17.4.
The benchmark simulates a hard burst by spawning N
processes that each send
log messages, using a specific logging framework, in a tight loop for 10000ms.
All log messages will be delivered over UDP (faked remote Syslog) to a socket
opened by the benchmark itself. The total duration is the time it took to spawn
the processes, send the messages and the time it took to receive all sent
messages at the socket that the benchmark process listens on. The benchmark has
two profiles. The large
profiles uses log messages with a size of ~840bytes
and the small
profile uses log messages with a size of ~40bytes.
All frameworks are used in their default configuration with their native logging
function (not error_logger
).
To be able to compare lager
with the other frameworks, the benchmark contains
a very simple backend, that forwards log messages formatted with the
lager_default_formatter
using gen_udp
, e.g. like syslog
does it. The
sasl_syslog
application is left out of scope, simply because it crashes the
Erlang VM on every run (due to its purely asynchronous logging).
Now let's see the numbers.
For a relatively small number of senders, all frameworks have a pretty good
throughput and the completion time is almost equal. While memory consumption is
generally low, the lager
application uses significant more memory. This is due
to the dynamic switching between synchronous and asynchronous message delivery:
When delivery starts in asynchronous mode many log messages pile up in the
internal gen_event
's message queue. When the framework finally switches to
synchronous mode the event manager is busy sending out these messages while
senders get blocked.
Adding more processes to the party makes the above mentioned effect more
dramatic. While log4erl
and syslog
perform quite well by evening the
load on the internal event manager or server using synchronous logging, lager
piles up a hugh amount of messages in its internal gen_event
message queue. It
takes the framework almost two minutes to process these messages.
When looking at the memory usage it can be observed that lager
's backend
throttling (which was also used in syslog
up to this version) is effectively
useless to protect from bursts with small log messages. Switching is simply too
slow and once messages begin to pile up, the node keeps being busy sending these
messages. Additionally, senders may get blocked an indefinite amount of time
(remember that gen_event:sync_notify/2
uses infinity
for timeouts). During
the test it could be observed that some senders were blocked over 50 seconds!
Numbers get interesting when large strings need to be formatted and copied
around. log4erl
as well as lager
do the actual message formatting inside
the internal event managers. Additionally, these frameworks pass around the
unprocessed format string and its argument list.
syslog
on the other hand, is now able to score by offloading the formatting
work into the logging processes and by exchanging binaries with the internal
logging server.
Interestingly, the total duration is excellent for all frameworks. Most
probably, copying the large terms makes the logging processes slow enough that
lager
's backend throttling can kick in and prevent the internal message queue
from growing too much. However, it could be observed again that some senders
were blocked over 10 seconds.
TODO more details?
- No difference to latest tag
syslog
is now available on hex.pm under the package namesyslog_app
(the namesyslog
is already assigned to another project)- Make project compatible to rebar3
- Remove dynamic switching of log message delivery mode. Make mode explicitly
configurable with the new
async
directive (which replaces theasync_limit
directive) and thesyslog:set_log_mode/1
API. - Add
app_name
configuration directive to allow configuration of theAPP-NAME
field value (thanks to @comtihon). - Change severity of messages sent by
error_logger:info_[msg|report]/1,2
andsyslog:info_msg/1,2
fromnotice
toinformational
(thanks to @comtihon). - Add
log_level
configuration directive. With this configuration it is possible to discard messages with undesired severity values (thanks to @comtihon). - Add optional
lager
backend to forward messages fromlager
tosyslog
. - Further improvement of robustness, especially when many processes try to log concurrently by moving the message formatting away from the main event manager into the logging processes.
- Fix event handler supervision. Due to a defective match pattern in
syslog_monitor
crashed handlers did not get re-added as expected/supposed.
- Replace the property
error_facility
withcrash_facility
. Refer to the explanation ofcrash_facility
above to learn more about this behaviour change. - Performance improvements (e.g. binary as internal message format)
- Provide discrete API for robust
error_logger
independent logging - Automatic toggling of sync/async logging for better load protection (default)
- Various performance improvements (e.g. timestamps, process name resolution)
- Configurable verbosity of progress report logging
- Support for release upgrades
- Supervised
error_logger
integration - Message queue length based load protection
- RFC 3164 compliant backend (default)
- RFC 5424 compliant backend
- Support for local and remote facilities using UDP
- Separate facility for error messages (default off)
- Standard SASL event format for
supervisor
andcrash
reports
For the curious; the above illustration shows the very simple supervision
hierarchy used by the syslog
application.