Puma::Plugin::TelemetryToo
NOTE: This is a fork of puma-plugin-telemetry, modified to:
- Support Puma 7
- Add
LogTarget, with customformatter:andtransform:options - Add
OpenTelemetrytarget - Warn about socket telemetry on unsupported platforms
Puma plugin which should be able to handle all your metric needs regarding your webserver:
- ability to publish basic puma statistics (like queue backlog) to both logs and Datadog
- ability to add custom target whenever you need it
- ability to monitor puma socket listen queue (!)
- ability to report requests queue time via custom rack middleware - the time request spent between being accepted by Load Balancer and start of its processing by Puma worker
Install
Add this line to your application's Gemfile:
gem "puma-plugin-telemetry_too"And then execute:
$ bundle installOr install it yourself as:
$ gem install puma-plugin-telemetry_tooUsage
In your puma configuration file (i.e. config/puma.rb or config/puma/<env>.rb):
plugin "telemetry_too"
Puma::Plugin::TelemetryToo.configure do |config|
config.enabled = true
# << here rest of the configuration, examples below
endBasic IO Target
A basic I/O target will emit telemetry data to STDOUT, formatted in JSON.
config.add_target(:io)Options
This target has configurable formatter: and transform: options.
The formatter: options are
-
:json(default) - Print the logs in JSON. -
:logfmt- Print the logs in key/value pairs, as perlogfmt. -
:passthrough- A pass-through formatter which returns the telemetryHashunaltered, passing it directly to theio:instance.
The transform: options are
-
:cloud_watch(default) - Transforms telemetry keys, replacing dots with dashes to support AWS CloudWatch Log Metrics filters. -
:logfmt- Transforms telemetry keys, prependingsample#for L2Met consumption. -
:passthrough- A pass-through transform which returns the telemetryHashunaltered.
Log target
While emitting to STDOUT via the basic IOTarget can work for getting telemetry into logs, we also provide an explicit LogTarget.
This target will defaults to emitting telemetry at the INFO log level via a standard library ::Logger instance.
That default logger will print to STDOUT in the logfmt format.
config.add_target(:log)You can pass an explicit logger: option if you wanted to, for example, use the same logger as Rails.
config.add_target(:log, logger: Rails.logger)This target also has configurable formatter: and transform: options.
The possible options are the same as for the IOTarget, but the defaults are different.
The LogTarget defaults to formatter: :logfmt, and transform: :passthrough.
Datadog StatsD target
A target for the Datadog StatsD client, that uses batch operation to publish metrics.
config.add_target(:dogstatsd, client: Datadog::Statsd.new)You can provide all the tags, namespaces, and other configuration options as always to Datadog::Statsd.new method.
OpenTelemetry target
A target for the OpenTelemetry Metrics API, which uses batch operations to publish metrics.
config.add_target :open_telemetry, meter_provider: OpenTelemetry.meter_providerThis target supports the following options:
| Option | Description | Default | Required |
|---|---|---|---|
| meter_provider | An instance of OpenTelemetry::Metrics::MeterProvider
|
OpenTelemetry.meter_provider |
No |
| force_flush | Force-flush metrics after each call. This can be expensive! | false |
No |
| prefix | Metric name prefix. e.g. prefix: "app" → app.workers.booted
|
puma |
No |
| suffix | Metric name suffix. e.g. suffix: "v1" → puma.workers.booted.v1
|
nil |
No |
| attributes | Attributes to be included with the metric | {} |
No |
All available options
For detailed documentation checkout Puma::Plugin::TelemetryToo::Config class.
Puma::Plugin::TelemetryToo.configure do |config|
config.enabled = true
config.initial_delay = 10
config.frequency = 30
config.puma_telemetry = %w[workers.requests_count queue.backlog queue.capacity]
config.socket_telemetry!
config.socket_parser = :inspect
config.add_target :io, io: StringIO.new, formatter: :json, transform: :passthrough
config.add_target :dogstatsd, client: Datadog::Statsd.new(tags: { env: ENV["RAILS_ENV"] })
endCustom Targets
Target is a simple object that implements call methods that accepts telemetry hash object. This means it can be super simple proc or some sophisticated class calling some external API.
Just be mindful that if the API takes long to call, it will slow down frequency with which telemetry will get reported.
# Example key/value log to `STDOUT` target
config.add_target ->(telemetry) { puts telemetry.map { |k, v| "#{k}=#{v.inspect}" }.join(" ") }Extra middleware
This gems comes together with middleware for measuring request queue time, which will be reported in request.env and published to given StatsD client.
Example configuration:
# in Gemfile add `require` part
gem "puma-plugin-telemetry_too", require: ["rack/request_queue_time_middleware"]
# in initializer, i.e. `request_queue_time.rb`
Rails.application.config.middleware.insert_after(
0,
RequestQueueTimeMiddleware,
statsd: Datadog::Statsd.new(namespace: "ruby.puma", tags: { "app" => "accounts" })
)
Rails.application.config.log_tags ||= {}
Rails.application.config.log_tags[:queue_time] = ->(req) { req.env[::RequestQueueTimeMiddleware::ENV_KEY] }This will provide proper metric in Datadog and in logs as well. Logs can be transformed into log metrics and used for auto scaling purposes.
Development
After checking out the repo, run bin/setup to install dependencies. Then, run bundle exec rake spec to run the tests. You can also run bin/console for an interactive prompt that will allow you to experiment.
To install this gem onto your local machine, run bundle exec rake install.
Release
All gem releases are manual, in order to create a new release follow:
- Create new PR (this could be included in feature PR, if it's meant to be released)
- update
VERSION, we use Semantic Versioning - update
CHANGELOG - merge
- update
- Draft new release via Github Releases
- use
v#{VERSION}as a tag, i.e.v0.1.0 - add release notes based on the Changelog
- create
- use
- Gem will get automatically published to given rubygems server
Contributing
Bug reports and pull requests are welcome on GitHub at https://github.com/stevenharman/puma-plugin-telemetry_too.
License
The gem is available as open source under the terms of the MIT License.
Code of Conduct
Everyone interacting in the puma-plugin-telemetry_too project's codebases, issue trackers, chat rooms and mailing lists is expected to follow the code of conduct.