Skip to content

Yleisradio/clojure-elastic-apm

Repository files navigation

clojure-elastic-apm

A Clojure wrapper for Elastic APM Java Agent's Public API.

Installation

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.

Usage

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 via apm/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 transaction Deprecated 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, calling apm/current-apm-transaction returns this transaction.
  • :headers - {String String} - a map of trace context headers (i.e. traceparent and tracestate) 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, calling apm/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 transaction Deprecated 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, calling apm/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, and ext. Defaults to ext.
  • :subtype - String - Set this as the third party service name. Defaults to undefined 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.

Manually starting, activating and ending transactions and spans

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:

  1. Always wrap any code in span/transaction to try..catch and end the transaction in finally block using apm/end
  2. Use with-open when activating a span/transaction
  3. The scope opened by apm/activate needs to be closed in the same thread

Getting current active transaction or span

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.

Capture and report exceptions

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.

Set the result of a transaction

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"))))

Set the outcome of a transaction or a span

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.

Adding labels

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.

Overriding transaction name

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.:

Using with Ring

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.

Filtering and matching URIs

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).

Accessing the current transaction

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.

Defining span-emitting functions

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)))

Development

Running tests

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.

Release

  • 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.
  • Update release notes in GitHub.