A Clojure wrapper for Elastic APM Java Agent's Public API.
First you need to download Elastic APM Java Agent and configure it. Follow the instructions in APM Java Agent documentation on how to do so.
The easiest way to enable the Java Agent on development is to add it to :jvm-opts
in Leiningen
along with the agent's recommended configuration options:
:jvm-opts ["-javaagent:path/to/elastic-apm-agent-AGENT_VERSION.jar"
"-Delastic.apm.service_name=my-app-production"
"-Delastic.apm.application_packages=my_app"
"-Delastic.apm.server_urls=http://localhost:8200"]
Now with the agent setup, you need to add two dependencies to your project: one for the Elastic APM Public API and one for clojure-elastic-apm. clojure-elastic-apm doesn't provide this dependency automatically, because it should match exactly with the agent version.
So, add these to your project.clj dependencies:
[co.elastic.apm/apm-agent-api "AGENT_VERSION"]
[clojure-elastic-apm "0.13.1"]
Note, in the agent configuration, the elastic.apm.application_packages
option should be the top level namespace in your
Clojure application, but with hyphens replaced with underscores.
Wrap any code block into APM transaction and track spans within the transaction:
(require '[clojure-elastic-apm.core :as apm])
(apm/with-apm-transaction [tx {:name "MyTransaction" :type "Background Job"}]
(do-something)
(apm/with-apm-span [span {:name "Operation"}]
(do-something-in-a-span))
(apm/with-apm-exit-span [span {:name "Call third party" :type "ext" :subtype "Third party service"}]
(call-third-party-service)))
The options hash for with-apm-transaction
accepts the following options:
:name
-String
- the transaction name:type
-String
- the transaction type (the special type"request"
, or viaapm/type-request
, should be used for transactions that track requests):labels
-{String any}
- map or sequence of label names and values to add to the transaction:tags
-{String String}
-map or sequence of label names and values to add to the transactionDeprecated as of 0.5.0:activate?
-Boolean
- whether to make the transaction active in the context of the executing thread (defaults to true). When activated, callingapm/current-apm-transaction
returns this transaction.:headers
-{String String}
- a map of trace context headers (i.e.traceparent
andtracestate
) for distributed tracing.:traceparent
-String
-the trace id when using APM's Distributed tracing. Usually value is passed within HTTP request headers.Deprecated as of 0.13.0
The options hash for with-apm-span
accepts the following options:
:name
-String
- the span name:parent
-Span
- the parent span, the new span will be created as child of this span (defaults to current active span or transaction):activate?
-Boolean
- whether to make the span active in the context of the executing thread (defaults to true). When activated, callingapm/current-apm-span
returns this span.:type
-String
- the span type:labels
-{String any}
- map or sequence of label names and values to add to the transaction:tags
-{String String}
-map or sequence of label names and values to add to the transactionDeprecated as of 0.5.0
In both cases, all options are optional and the options hash can be omitted completely. However, it's good idea to at least provide the name. At the time of writing, the default transaction type is "custom"
.
The with-apm-exit-span
should be used when calling a third party service that should be shown as a dependency in analytics.
The options hash for with-apm-exit-span
accepts the following options:
:name
-String
- the span name:parent
-Span
- the parent span, the new span will be created as child of this span (defaults to current active span or transaction):activate?
-Boolean
- whether to make the span active in the context of the executing thread (defaults to true). When activated, callingapm/current-apm-span
returns this span. Optional.:type
-String
- The general type of the new span. Though there are no naming restrictions for the general types, the following are standardized across all Elastic APM agents:app
,db
,cache
,template
, andext
. Defaults toext
.:subtype
-String
- Set this as the third party service name. Defaults toundefined subtype
.:action
-String
- Describes the action eg.query
.:labels
-{String any}
- map or sequence of label names and values to add to the transaction
All options are optional. However, it's a good idea to provide at least a name and a subtype.
The with-apm-transaction
and with-apm-span
macros are useful if the transaction or span starts and ends in the same thread, which is often
the case. However, if you need to track transactions that start in a different thread than where they end, you can also manually start a transaction
in one thread, hand it off to another thread and activate it there.
For example, consider a scenario where your HTTP API accepts a request to build reports. You queue each request to be processed by a ExecutorService
and then send email when things finish. If you wanted to measure the time between HTTP API getting the request and it being processed, you could
do something like this:
(defn process-report-request [tx request]
(try
(with-open [scope (apm/activate tx)]
(build-and-send-report request))
(catch Exception e
(send-failure-notice request)
(apm/capture-exception e))
(finally
(apm/end tx))))
(defn handle-report-request [request]
(let [tx (apm/start-transaction {:name "" :type apm/type-request})]
(.submit executor #(process-report-request tx request))
{:status 201 :body "Accepted"}))
The process is similar for spans, but instead of apm/start-transaction
you would use apm/create-span
.
All the same options, except for the activation, are supported here.
Remember to always close any resources, or otherwise you might cause memory leaks:
- Always wrap any code in span/transaction to try..catch and end the transaction in
finally
block usingapm/end
- Use
with-open
when activating a span/transaction - The scope opened by
apm/activate
needs to be closed in the same thread
Current active transaction can be retrieved by calling apm/current-apm-transaction
. The active span can be retrieved with apm/current-apm-span
.
Note that these functions might return a "noop" span or transaction in case there's no active span/transaction. This means you never need to check for null values. You can still capture exceptions on noop spans/transactions - they just will be reported at the application level and will not be associated with any particular span.
Any exceptions thrown out of with-apm-transaction
and with-apm-span
macro bodies will automatically be captured and reported on the APM
server and then rethrown.
You can also manually report exceptions that you don't wish to propagate further by calling apm/capture-exception
:
(with-apm-transaction [tx {:name "BackgroundJob"}]
(try
(do-something)
(catch Exception e
(apm/capture-exception tx e)
(do-something-else))))
A function specifically designed for doing this, clojure-elastic-apm.core/catch-error-to-apm
is included, which takes a thunk function, attempts
to evaluate it, and captures the exception if one occurs.
You can set the result
of a transaction by calling apm/set-result
, which takes a custom string that is visible in your trace samples.
(with-apm-transaction [tx {:name "BackgroundJob"}]
(try
(do-something)
(set-result tx "success")
(catch PSQLException e
(set-result tx "database-error"))
(catch Throwable t
(set-result tx "unexpected-error"))))
You can set the outcome
of a transaction or a span by calling apm/set-outcome
. There are 3 available outcomes, apm/outcome-success
,
apm/outcome-failure
and apm/outcome-unknown
, out of which apm/outcome-success
and apm/outcome-failure
determine
the error rate displayed in APM.
(with-apm-transaction [tx {:name "BackgroundJob"}]
(try
(do-something)
(set-outcome tx apm/outcome-success)
(catch Exception e
(set-outcome tx apm/outcome-failure))))
You can also use the wrappers apm/set-outcome-success
, apm/set-outcome-failure
and apm/set-outcome-unknown
instead.
You can add labels to any transaction or span by using apm/set-label
:
(apm/with-apm-transaction [tx {:name "CreatePayment"}]
(let [payment (create-payment ...)]
(apm/set-label tx "payment_id" (:id payment)
(store-payment payment)
...)))
APM supports labels with string keys and string, number or boolean values. Any other types set-label
will convert
to a string using .toString
.
If you wish to override the transaction name (for example, because you didn't know it when starting the transaction), you can do so
any time by calling apm/set-name
on the transaction:
(apm/with-apm-transaction [tx]
(do-something)
(apm/set-name tx "BackgroundJob"))
See the section below on using with Ring for a more concrete example where this might be useful.:
For convenience, the clojure-elastic-apm.ring
namespace provides wrap-apm-transaction
Ring middleware:
(require '[clojure-elastic-apm.ring :as apm-ring])
(def app (-> app-routes
wrap-params
wrap-json-params
apm-ring/wrap-apm-transaction))
With this setup, all requests will be tracked as APM transactions with "request"
type. The transaction will be named METHOD /path
.
The wrap-apm-transaction
middleware takes an optional argument, that allows you to pass a list of URI patterns, which
will be matched against any requests, and only log a transaction for those that match the pattern:
(require '[clojure-elastic-apm.ring :as apm-ring])
(def app (-> app-routes
wrap-params
wrap-json-params
(apm-ring/wrap-apm-transaction ["/*/*/_"])))
The pattern format breaks up the URI into segments on the slashes ("/") and matches the segments with the following syntax:
*
: matches any data in the segment, and keeps it in the resulting transaction name_
: matches any data in the segment, but ignores it in the resulting tx name, instead inserting an underscore('_') character- any other string: matches only if the segment in the pattern and the URI match exactly, and retains the string in the tx name
In the example above, the URI /v1/foo/124
would match and yield /v1/foo/*
as the tx name, while /v1/foo
would not. This allows you to both ignore all but a specific set of URLs, while also grouping URLs in the APM tx logs by leaving out
unnecessary extra detail like URL parameters.
If any segments fail to match one of the given patterns, then no transaction is logged. The matcher will attempt to match
the patterns in descending order as given in the vector. Matches are "eager": a short pattern like "/*"
will match any URI, but ignore segments after it (so /*
would match against /v1/foo/bar
but only return /v1
).
You can access the APM transaction created by the middleware from the request map under :clojure-elastic-apm/transaction
key.
No other request information will be added to the transaction. The APM Java Agent's Public API, at the time of writing this library, doesn't allow setting transaction context fields. However, if you want to include extra context as labels, it's easy to do so by adding custom middleware:
(defn wrap-apm-transaction-context [handler]
(fn [request]
(when-let [tx (:clojure-elastic-apm/transaction request)]
(apm/set-label tx "user_agent" (get-in request [:headers "user-agent"]))
(apm/set-label tx "request_query_string" (:query-string request)))
(handler request)))
(def app (-> app-routes
wrap-apm-transaction-context
apm-ring/wrap-apm-transaction))
If you use Compojure and would like to name the transactions after the Compojure route, it's possible, albeit some extra work is needed.
Compojure includes the matched route information under :compojure/route
request key, but unfortunately wrap-apm-transaction
doesn't have
access to this info - only the routes can see it.
Thus, in each route, you need to manually override the transaction name by invoking apm/set-transaction-name
with the transaction reference
from request and the transaction name derived from compojure route information:
(defroutes app-routes
(GET "/exchange_rates/:base" request
(when-let [tx (:clojure-elastic-apm/transaction request)]
(apm/set-name tx (str (:compojure/route request))))
{:status 200 :body ...}))
If you have a lot of routes or find this to be tedious, it might be worth it creating a custom routes macro that does this automatically. This is not included here because integration with compojure and other routing libraries is out of this project's scope.
To define a function whose body is wrapped in a span, use clojure-elastic-apm.core/defnspan
. defnspan
is like clojure.core/defn
, except that it wraps the function body with clojure-elastic-apm.core/with-apm-span
.
By default, defnspan
uses the fully-qualified name of the function as the span name.
Thus, for example, this:
(ns my.ns
(:require [clojure-elastic-apm.core :refer [defnspan]]))
(defnspan my-sum
[a b]
(+ a b))
Expands (roughly) to this:
(defn my-sum
[a b]
(apm/with-apm-span [_ {:name "my.ns/my-sum"}]
(+ a b)))
To pass options to with-apm-span
, use var metadata. For example, this:
(defnspan my-sum
;; Every metadata entry in the "apm" namespace become options to
;; with-apm-span.
{:apm/name "Add number a to number b"
:apm/type "algo"}
[a b]
(+ a b))
Expands (roughly) to this:
(defn my-sum
[a b]
(apm/with-apm-span [_ {:name "Add number a to number b" :type "algo"}]
(+ a b)))
There are some tests for the functionality. To run them, you need to have ElasticSearch, Kibana and APM server running at ports 9200, 5601 and 8200 respectively. You can either manually install the necessary components, or use the provided docker-compose file to launch all three as containers:
$ docker compose up -d
Then, to run tests:
$ lein test
Note that the tests are extremely slow. The APM Agent's Public API doesn't provide a way to retrieve the information we set in clojure-elastic-apm. The only way to access all the details is to fetch the transaction info from ElasticSearch. This takes time, because the agent can sync only every 1 second and the APM server doesn't flush to ElasticSearch all the time either.
Also note that the ElasticSearch docker requires a minimum of 4 gigabytes of memory, so ensure that you have allocated enough memory for your docker containers: from Docker Dashboard Preferences -> Resources -> Advanced -> Memory.
- Update version in project.clj.
- Run
lein deploy clojars
- If you need to login to Clojars the username is
yleisradio
. Password is known at least by Sitouttamistiimi. - Publish username is
yleisradio
and passwod is a token. This token is known at least by Sitouttamistiimi.
- If you need to login to Clojars the username is
- Update release notes in GitHub.