Both the Java client and the Clojure client have Dropwizard
Metrics support - they both accept as an
option a MetricRegistry
to which they will register http metrics for each
request, as well as metrics for any metric-ids specified in the request
options. This support is experimental - names of metrics and the exact API may
change.
For using metrics with either the Java client or the Clojure client you must
already have created a Dropwizard MetricRegistry
.
By default, http metrics are prefixed with the namespace
puppetlabs.http-client.experimental
. This namespace can be customized with two
client options, server-id
and metric-prefix
.
When server-id
is set, the metric namespace becomes
puppetlabs.<server-id>.http-client.experimental
.
When metric-prefix
is set, the metric namespace becomes
<metric-prefix>.http-client.experimental
.
If both server-id
and metric-prefix
are set, metric-prefix
wins out and
a warning message is logged.
For a Clojure client, the get-client-metric-namespace
protocol method can
be used to get back the metric namespace set for the client.
For a Java client, the getMetricNamespace
method can be used to get back the
configured metric namespace.
There are two types of metrics: full response and initial response. Full
response metrics stop when all bytes of the response have been read by the
client. Initial response metrics stop when the first byte of the response has
been received by the client. Full response metrics are suffixed with
full-response
. Initial response metric support has not yet been
implementedare suffixed with full-response
. Initial response metric support
has not yet been implemented.
There are three categories of metrics: url
metrics, url-and-method
metrics, and metric-id
metrics. url
and url-and-method
metrics are
created automatically for each request; metric-id
metrics are only created
for requests that have a metric-id
request option specified.
Each http request will have a metric name created for its url (stripped of
query strings and url fragments), as well as for the url + method name. url
metrics have with-url
after the prefix, followed by the url.
url-and-method
metrics have with-url-and-method
after the prefix, followed
by the url and then the capitalized HTTP method.
So, for example, a GET
request to http://foobar.com
would create a metric
puppetlabs.http-client.experimental.with-url.http://foobar.com.full-response
and a metric
puppetlabs.http-client.experimental.with-url-and-method.http://foobar.com.GET.full-response
.
It is also possible to give a request a metric-id
. The metric-id
is an
array of strings. For each element in the array, a metric name will be
created, appending to the previous elements. metric-id
metrics have
with-metric-id
after the metric prefix, followed by the metric-id.
So, for example, for a metric-id ["foo", "bar", "baz"]
, the metrics
puppetlabs.http-client.experimental.with-metric-id.foo.full-response
,
puppetlabs.http-client.experimental.with-metric-id.foo.bar.full-response
,
and
puppetlabs.http-client.experimental.with-metric-id.foo.bar.baz.full-response
would be created.
Both the Clojure API and the Java API have functions to get back from a
MetricRegistry
either the Timer
objects or a selection of metric data, and
to filter this information based on url, url and method, or metric-id.
There are three different Timer
objects: UrlClientTimer
(which includes a
field for the url and getUrl
method), UrlAndMethodClientTimer
(which
includes a field for the url and a field for the method and accompanying
getUrl
and getMethod
methods), and MetricIdClientTimer
(which includes a
field for the metric-id and accompanying getMetricId
method). Each of these
timers also has an isCategory
method that returns whether the timer is or is
not the provided MetricCategory
.
Both the Clojure and Java APIs also include functions that return a list of
metrics data. This metrics data includes the metric-name
, count
, mean
(in ms), and aggregate
(computed as count * mean
) for each metric. In
addition, for url
metrics the accompanying metrics data includes the url
,
for url-and-method
metrics it includes the url
and method
, and for
metric-id
metrics it includes the metric-id
.
The Java API returns ClientMetricData
objects - of which there are three
types - UrlClientMetricData
, UrlAndMethodClientMetricData
, and
MetricIdClientMetricData
. The Clojure API returns maps with keys for this
information.
Both APIs have functions for returning all metrics/metric data, for returning all metrics/metrics data from a specific category, and for filtering metrics/metrics data within a category.
To use metrics with the Clojure client, pass a MetricRegistry
to it as part
of the options map:
(async/create-client {:metric-registry (MetricRegistry.)})
(the same works with the sync
client).
In Trapperkeeper applications,
creating and managing a MetricRegistry
can be done easily with
trapperkeeper-metrics:
(defservice my-trapperkeeper-service
MyService
[[:MetricsService get-metrics-registry]]
(init [this context]
(let [registry (get-metrics-registry)
client (async/create-client {:metric-registry registry})]
...)))
Any client that is created with a MetricRegistry
will automatically have
url
and url-and-method
metrics registered.
get-client-metric-registry
protocol function can be called on a client to
get the MetricRegistry
from it.
In addition to the url
and url-and-method
metrics, it is possible to set a
metric-id
for a request that will create additional metrics.
For the Clojure API, a metric-id
is a vector of keywords or strings. Either
is supported by the API, however, if special characters are needed, use
strings. Note than even when specified as a vector of keywords in the request
option, the metric-id will be returned as a vector of strings in the metrics
data.
To set a metric-id
for a request, include it as an option in the request
options map.
(common/get client "https://foobar.com" {:metric-id [:foo :bar :baz]})
(common/get client "https://foobar.com" {:metric-id ["f/o/o"]})
To get all Timer
objects registered for a MetricRegistry
, use the
get-client-metrics
function in the metrics
namespace. This takes the
MetricRegistry
as an argument and returns a map with three keys: :url
:url-and-method
, and :metric-id
. Under each of these keys is a sequence of
Timer
objects of the corresponding type. If there are no timers of a certain
type, the sequence will be empty. The output of this function conforms to the
common/AllTimers
schema.
Example:
(common/get client "http://test.com" {:metric-id [:foo :bar :baz]})
...
(metrics/get-client-metrics metric-registry)
=>
{:url [#object[com.puppetlabs.http.client.metrics.UrlClientTimer
0x66cf1f05
"com.puppetlabs.http.client.metrics.UrlClientTimer@66cf1f05"]]
:url-and-method [#object[com.puppetlabs.http.client.metrics.UrlAndMethodClientTimer
0x6fe5444c
"com.puppetlabs.http.client.metrics.UrlAndMethodClientTimer@6fe5444c"]]
:metric-id [#object[com.puppetlabs.http.client.metrics.MetricIdClientTimer
0x690c10c5
"com.puppetlabs.http.client.metrics.MetricIdClientTimer@690c10c5"]
#object[com.puppetlabs.http.client.metrics.MetricIdClientTimer
0xb7aca2e
"com.puppetlabs.http.client.metrics.MetricIdClientTimer@b7aca2e"]
#object[com.puppetlabs.http.client.metrics.MetricIdClientTimer
0x4ef82829
"com.puppetlabs.http.client.metrics.MetricIdClientTimer@4ef82829"]]}
To get metric data for all Timer
s registered on a MetricRegistry
, use the
get-client-metrics-data
function. This takes the MetricRegistry
and returns
a map with :url
, :url-and-method
, and :metric-id
as keys. Under each of
these keys is a sequence of maps, each map containing metrics data (see
Getting back metrics above, conforming to the
common/AllMetricsData
schema.
Example:
(common/get client "http://test.com" {:metric-id [:foo :bar :baz]})
...
(metrics/get-client-metrics-data metric-registry)
=>
{:url ({:count 1
:mean 553
:aggregate 553
:metric-name "puppetlabs.http-client.experimental.with-url.http://test.com.full-response"
:url "http://test.com"})
:url-and-method ({:count 1
:mean 554
:aggregate 554
:metric-name "puppetlabs.http-client.experimental.with-url-and-method.http://test.com.GET.full-response"
:url "http://test.com"
:method "GET"})
:metric-id ({:count 1
:mean 554
:aggregate 554
:metric-name "puppetlabs.http-client.experimental.with-metric-id.foo.bar.baz.full-response"
:metric-id ["foo" "bar" "baz"]}
{:count 1
:mean 554
:aggregate 554
:metric-name "puppetlabs.http-client.experimental.with-metric-id.foo.bar.full-response"
:metric-id ["foo" "bar"]}
{:count 1
:mean 554
:aggregate 554
:metric-name "puppetlabs.http-client.experimental.with-metric-id.foo.full-response"
:metric-id ["foo"]})}
To get URL metrics and metrics data, use the get-client-metrics-by-url
function and get-client-metrics-data-by-url
in the metrics
namespace.
Both of these take as arguments the MetricRegistry
and a string url. If no
url is provided, return all url metrics/metrics data in a sequence. If no
metrics are registered for that url, return an empty sequence.
Example:
(common/get client "http://test.com" {:metric-id [:foo :bar :baz]})
...
(metrics/get-client-metrics-data-by-url metric-registry "http://test.com")
=>
({:count 1
:mean 553
:aggregate 553
:metric-name "puppetlabs.http-client.experimental.with-url.http://test.com.full-response"
:url "http://test.com"})
(metrics/get-client-metrics-data-by-url metric-registry "http://not-a-matching-url.com")
=> ()
To get URL and method metrics and metrics data, use the
get-client-metrics-by-url-and-method
and
get-client-metrics-data-by-url-and-method
functions in the metrics
namespace.
Both of these take as arguments the MetricRegistry
, a string url, and a
keyword HTTP method. If no url and method is provided, return all url
metrics/metrics data in a sequence. If no metrics are registered for that url
and method, return an empty sequence.
Example:
(common/get client "http://test.com" {:metric-id [:foo :bar :baz]})
...
(metrics/get-client-metrics-data-by-url-and-method metric-registry "http://test.com" :get)
=>
({:count 1
:mean 554
:aggregate 554
:metric-name "puppetlabs.http-client.experimental.with-url-and-method.http://test.com.GET.full-response"
:url "http://test.com"
:method "GET"})
:method "GET"})
(metrics/get-client-metrics-data-by-url-and-method metric-registry "http://test.com" :post)
=> ()
To get metric-id metrics and metrics data, use the
get-client-metrics-by-metric-id
and get-client-metrics-data-by-metric-id
functions in the metrics
namespace.
Both of these take as arguments the MetricRegistry
and a metric-id - as a
vector of keywords or strings (if special characters are needed, use strings).
If no metric-id is provided, will return all metric-id metrics. If no metrics
are registered for that metric-id, returns an empty list.
Example:
(common/get client "http://test.com" {:metric-id [:foo :bar :baz]})
...
(metrics/get-client-metrics-data-by-metric-id metric-registry [:foo])
=>
({:count 1
:mean 554
:aggregate 554
:metric-name "puppetlabs.http-client.experimental.with-metric-id.foo.full-response"
:metric-id ["foo"]})
(metrics/get-client-metrics-data-by-metric-id metric-registry [:foo :bar])
=>
({:count 1
:mean 554
:aggregate 554
:metric-name "puppetlabs.http-client.experimental.with-metric-id.foo.bar.full-response"
:metric-id ["foo" "bar"]})
(metrics/get-client-metrics-data-by-metric-id metric-registry [:foo :nope])
=> ()
To use metrics with the Java client, call setMetricRegistry()
on the
ClientOptions
before supplying them to create the client.
MetricRegistry registry = new MetricRegistry();
ClientOptions options = new ClientOptions();
AsyncHttpClient client = Async.createClient(options);
(the same works with the SyncHttpClient
).
Any client that is created with a MetricRegistry
will automatically have
url
and url-and-method
metrics registered.
getMetricRegistry()
can be called on a client to get the MetricRegistry
from it.
In addition to the url
and url-and-method
metrics, it is possible to set a
metric-id
for a request that will create additional metrics.
A metric-id
is an array of strings.
To set a metric-id
for a request, use the .setMetricId
method on the
RequestOptions
class.
RequestOptions options = new RequestOptions("https://foobar.com");
options.setMetricId(["foo", "bar", "baz"]);
client.get(options);
getMetricId()
can be called on RequestOptions
to get back the metric-id
.
To get all Timer
objects registered for a MetricRegistry
, use the
getClientMetrics()
method in the Metrics
class. This takes the
MetricRegistry
as an argument and returns a ClientTimerContainer
object.
The ClientTimerContainer
object has three fields - urlTimers
,
urlAndMethodTimers
, and metricIdTimers
. The list of URLClientTimers
can
be retrieved from the ClientTimerContainer
with the getUrlTimers()
method.
A list of URlAndMethodClientTimers
can be retrieved with the
getUrlAndMethod()
method. A list of MetricIdClientTimer
s can be retrieved
with the getMetricIdTimers()
method.
To get all MetricData
objects, representing data for each Metric
registered for a MetricRegistry
, use the getClientMetricsData()
methods in
the Metric
class. This takes the MetricRegistry
as an argument and returns
a ClientMetricDataContainer
object. This object has a field for each type of
metric data - urlData
, urlAndMethodData
, and metricIdData
. Each field
has an associated getter returning a list of the appropriate data object -
UrlClientMetricData
, UrlAndMethodClientMetricData
, and
MetricIdClientMetricData
.
To get URL metrics and metrics data, use the getClientMetricsByUrl()
and
getClientMetricsDataByUrl()
methods in the Metric
class.
Both of these take as arguments the MetricRegistry
and a string url. If no
url is provided, return all url metrics. If no metrics are registered for that
url, return an empty list.
getClientMetricsByUrl()
returns a list of UrlClientTimer
objects.
getClientMetricsDataByUrl
returns a list of UrlClientMetricData
objects.
To get URL and method metrics and metrics data, use the
getClientMetricsByUrlAndMethod()
and getClientMetricsDataByUrlAndMethod()
methods in the Metric
class.
Both of these take as arguments the MetricRegistry
, a string url, and a
string HTTP method. If no url or method is provided, will return all
url-and-method metrics. If no metrics are registered for that url and method,
returns an empty list.
getClientMetricsByUrlAndMethod()
returns a list of UrlAndMethodClientTimer
objects. getClientMetricsDataByUrlAndMethod
returns a list of
UrlAndMethodClientMetricData
objects.
To get metric-id metrics and metrics data, use the
getClientMetricsByMetricId()
and getClientMetricsDataByMetricId()
methods
in the Metric
class.
Both of these take as arguments the MetricRegistry
and a metric-id - as an
array of strings. If no metric-id is provided, will return all metric-id
metrics. If no metrics are registered for that metric-id, returns an empty
list.
getClientMetricsByMetricId()
returns a list of MetricIdClientTimer
objects. getClientMetricsDataByMetricId()
returns a list of
MetricIdClientMetricData
objects.