Ecosyste.ms: Awesome

An open API service indexing awesome lists of open source software.

Awesome Lists | Featured Topics | Projects

https://github.com/avisonovate/logging

Easy setup of clojure.tools.logging w/ SLF4j, plus request correlation
https://github.com/avisonovate/logging

clojure correlation logging

Last synced: 2 months ago
JSON representation

Easy setup of clojure.tools.logging w/ SLF4j, plus request correlation

Awesome Lists containing this project

README

        

## Logging - Clojure logging with Logback and SLF4J plus request correlation across servers

image:https://img.shields.io/clojars/v/io.aviso/logging.svg[Clojars Project, link="http://clojars.org/io.aviso/logging"]

image:https://circleci.com/gh/AvisoNovate/logging/tree/master.svg?style=svg["CircleCI", link="https://circleci.com/gh/AvisoNovate/logging/tree/master"]

Logging is available under the terms of the Apache Software License 2.0.

link:http://avisonovate.github.io/docs/logging/[API Documentation]

## Dependencies

Mostly, we set up the dependencies so that
link:https://github.com/clojure/tools.logging[clojure.tools.logging],
uses link:http://www.slf4j.org/[SLF4J] and
link:http://logback.qos.ch/[Logback].

We also include the link:http://www.slf4j.org/legacy.html#jclOverSLF4J[shim library]
that converts
link:http://commons.apache.org/proper/commons-logging/[Apache Commons Logging] to use SLF4J as well.

## Setup

The namespace io.aviso.logging.setup performs all the necessary setup; it installs
link:https://github.com/AvisoNovate/pretty[pretty exception reporting] for console output
and commons logging output, as well as a default uncaught exception handler (that logs the
uncaught exception).

Simply have some part of your application require this namespace.

## Request Correlation

Ever try to trace a request from a client through a cluster of services?
Even if you can get all the logs in one place, how do you pick out the single thread of
requests and responses from all the other requests and responses?

With request correlation, each request gets assigned a correlation id on entry to the system.
When one server sends a request to another server, it includes the correlation id as a header
on the request.

That, plus a little bit of logic in your Logback configuration, allows every request to be flagged.

### Request Correlation Middleware

The function io.aviso.logging.correlation/wrap-with-request-correlation is a Ring middleware function;
it is passed a Ring request handler and wraps it.

On each request, the request correlation id is extracted, or generated if not present.
By default, the correlation id comes from the Correlation-Id header, and is generated
as a random UUID if not present.

In addition, the io.aviso.logging.correlation/\*correlation-id* Var is bound to the correlation id.
This little bit of semi-global state can be used when making requests to other servers,
to include the correlation id in the request.

### Logging the Correlation Id

The final piece of the puzzle is to get the correlation id logged.
this is accomplished by making the correlation id a
link:http://logback.qos.ch/manual/mdc.html[mapped diagnostic context].

In your logback.xml (or logback-test.xml):

[source,xml]
----



%date{HH:mm:ss.SSS} %mdc{correlation-id} %-5level [%thread] %logger{} - %msg%n





----

The CorrelationIdAppender is used to set up the correlation id into the MDC (the mapped
diagnostic context).
After that, the `%mdc{correlation-id}` form can extract that value and include it in the
logged output.

## Extra MDC

Version 0.2.0 adds a generalization of request correlation; an arbitrary map of keys and values
can be added to the MDC.

This takes the form of a new appender class, io.aviso.logging.ExtraMDCAppender:

[source,xml]
----



%date{HH:mm:ss.SSS} %mdc{authenticated} %-5level [%thread] %logger{} - %msg%n





----

In the above example, authenticated is presumably a value that may be bound into the
`*extra-mdc*` var.

A macro, `with-mdc`, makes it easy to bind new keys and values into the extra MDC.

Logback's message diagnostic map is mutable and designed to be inherited by
sub-threads in a way that overlaps with Clojure's inheriting of dynamic vars.
Because of this, a value stored into the MDC is "sticky" and may leak into
later logging on the thread, even after exiting the `with-mdc` block.

The best solution for this is to *always* set a default for any key
that may be added inside a `with-mdc` block:

[source,clojure]
----
(require '[io.aviso.logging.mdc :refer [set-mdc-default with-mdc]
'[clojure.tools.logging :as l])

(set-mdc-default {:authenticated "none"})

(with-mdc {:authenticated "http"}
...
(l/info "Handling request")
... )
----

Although the "authenticated" key will have value "https"
in the Logback MDC even after exiting from the `with-mdc` block,
any subsequent logging will reset it back to the default, "none".