Skip to content

Commit 3c1a0ee

Browse files
committed
[RAILS] Added ActiveSupport instrumentation integration
Require the component in your `application.rb` file: require 'elasticsearch/rails/instrumentation' You should see an output like this in your application log in development environment: Article Search (321.3ms) { index: "articles", type: "article", body: { query: ... } } Also, the total duration of the request to Elasticsearch is displayed in the Rails request breakdown: Completed 200 OK in 615ms (Views: 230.9ms | ActiveRecord: 0.0ms | Elasticsearch: 321.3ms) A special component for the Lograge logger implementation is provided as well. See: * http://edgeguides.rubyonrails.org/active_support_instrumentation.html * http://api.rubyonrails.org/classes/ActiveSupport/Notifications.html * https://github.com/roidrage/lograge
1 parent 45eba52 commit 3c1a0ee

File tree

10 files changed

+329
-0
lines changed

10 files changed

+329
-0
lines changed

elasticsearch-rails/README.md

+28
Original file line numberDiff line numberDiff line change
@@ -46,6 +46,34 @@ Run this command to display usage instructions:
4646
$ bundle exec rake -D elasticsearch
4747
```
4848

49+
### ActiveSupport Instrumentation
50+
51+
To display information about the search request (duration, search definition) during development,
52+
and to include the information in the Rails log file, require the component in your `application.rb` file:
53+
54+
```ruby
55+
require 'elasticsearch/rails/instrumentation'
56+
```
57+
58+
You should see an output like this in your application log in development environment:
59+
60+
Article Search (321.3ms) { index: "articles", type: "article", body: { query: ... } }
61+
62+
Also, the total duration of the request to Elasticsearch is displayed in the Rails request breakdown:
63+
64+
Completed 200 OK in 615ms (Views: 230.9ms | ActiveRecord: 0.0ms | Elasticsearch: 321.3ms)
65+
66+
There's a special component for the [Lograge](https://github.com/roidrage/lograge) logger.
67+
Require the component in your `application.rb` file (and set `config.lograge.enabled`):
68+
69+
```ruby
70+
require 'elasticsearch/rails/lograge'
71+
```
72+
73+
You should see the duration of the request to Elasticsearch as part of each log event:
74+
75+
method=GET path=/search ... status=200 duration=380.89 view=99.64 db=0.00 es=279.37
76+
4977
### Rails Application Templates
5078

5179
You can generate a fully working example Ruby on Rails application, with an `Article` model and a search form,

elasticsearch-rails/elasticsearch-rails.gemspec

+3
Original file line numberDiff line numberDiff line change
@@ -28,6 +28,9 @@ Gem::Specification.new do |s|
2828

2929
s.add_development_dependency "oj"
3030
s.add_development_dependency "rails", "> 3.0"
31+
32+
s.add_development_dependency "lograge"
33+
3134
s.add_development_dependency "shoulda-context"
3235
s.add_development_dependency "mocha"
3336
s.add_development_dependency "turn"
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,36 @@
1+
require 'elasticsearch/rails/instrumentation/railtie'
2+
require 'elasticsearch/rails/instrumentation/publishers'
3+
4+
module Elasticsearch
5+
module Rails
6+
7+
# This module adds support for displaying statistics about search duration in the Rails application log
8+
# by integrating with the `ActiveSupport::Notifications` framework and `ActionController` logger.
9+
#
10+
# == Usage
11+
#
12+
# Require the component in your `application.rb` file:
13+
#
14+
# require 'elasticsearch/rails/instrumentation'
15+
#
16+
# You should see an output like this in your application log in development environment:
17+
#
18+
# Article Search (321.3ms) { index: "articles", type: "article", body: { query: ... } }
19+
#
20+
# Also, the total duration of the request to Elasticsearch is displayed in the Rails request breakdown:
21+
#
22+
# Completed 200 OK in 615ms (Views: 230.9ms | ActiveRecord: 0.0ms | Elasticsearch: 321.3ms)
23+
#
24+
# @note The displayed duration includes the HTTP transfer -- the time it took Elasticsearch
25+
# to process your request is available in the `response.took` property.
26+
#
27+
# @see Elasticsearch::Rails::Instrumentation::Publishers
28+
# @see Elasticsearch::Rails::Instrumentation::Railtie
29+
#
30+
# @see http://api.rubyonrails.org/classes/ActiveSupport/Notifications.html
31+
#
32+
#
33+
module Instrumentation
34+
end
35+
end
36+
end
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,41 @@
1+
require 'active_support/core_ext/module/attr_internal'
2+
3+
module Elasticsearch
4+
module Rails
5+
module Instrumentation
6+
7+
# Hooks into ActionController to display Elasticsearch runtime
8+
#
9+
# @see https://github.com/rails/rails/blob/master/activerecord/lib/active_record/railties/controller_runtime.rb
10+
#
11+
module ControllerRuntime
12+
extend ActiveSupport::Concern
13+
14+
protected
15+
16+
attr_internal :elasticsearch_runtime
17+
18+
def cleanup_view_runtime
19+
elasticsearch_rt_before_render = Elasticsearch::Rails::Instrumentation::LogSubscriber.reset_runtime
20+
runtime = super
21+
elasticsearch_rt_after_render = Elasticsearch::Rails::Instrumentation::LogSubscriber.reset_runtime
22+
self.elasticsearch_runtime = elasticsearch_rt_before_render + elasticsearch_rt_after_render
23+
runtime - elasticsearch_rt_after_render
24+
end
25+
26+
def append_info_to_payload(payload)
27+
super
28+
payload[:elasticsearch_runtime] = (elasticsearch_runtime || 0) + Elasticsearch::Rails::Instrumentation::LogSubscriber.reset_runtime
29+
end
30+
31+
module ClassMethods
32+
def log_process_action(payload)
33+
messages, elasticsearch_runtime = super, payload[:elasticsearch_runtime]
34+
messages << ("Elasticsearch: %.1fms" % elasticsearch_runtime.to_f) if elasticsearch_runtime
35+
messages
36+
end
37+
end
38+
end
39+
end
40+
end
41+
end
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,41 @@
1+
module Elasticsearch
2+
module Rails
3+
module Instrumentation
4+
5+
# A log subscriber to attach to Elasticsearch related events
6+
#
7+
# @see https://github.com/rails/rails/blob/master/activerecord/lib/active_record/log_subscriber.rb
8+
#
9+
class LogSubscriber < ActiveSupport::LogSubscriber
10+
def self.runtime=(value)
11+
Thread.current["elasticsearch_runtime"] = value
12+
end
13+
14+
def self.runtime
15+
Thread.current["elasticsearch_runtime"] ||= 0
16+
end
17+
18+
def self.reset_runtime
19+
rt, self.runtime = runtime, 0
20+
rt
21+
end
22+
23+
# Intercept `search.elasticsearch` events, and display them in the Rails log
24+
#
25+
def search(event)
26+
self.class.runtime += event.duration
27+
return unless logger.debug?
28+
29+
payload = event.payload
30+
name = "#{payload[:klass]} #{payload[:name]} (#{event.duration.round(1)}ms)"
31+
search = payload[:search].inspect.gsub(/:(\w+)=>/, '\1: ')
32+
33+
debug %Q| #{color(name, GREEN, true)} #{colorize_logging ? "\e[2m#{search}\e[0m" : search}|
34+
end
35+
end
36+
37+
end
38+
end
39+
end
40+
41+
Elasticsearch::Rails::Instrumentation::LogSubscriber.attach_to :elasticsearch
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,32 @@
1+
module Elasticsearch
2+
module Rails
3+
module Instrumentation
4+
module Publishers
5+
6+
# Wraps the `SearchRequest` methods to perform the instrumentation
7+
#
8+
# @see SearchRequest#execute_with_instrumentation!
9+
# @see http://api.rubyonrails.org/classes/ActiveSupport/Notifications.html
10+
#
11+
module SearchRequest
12+
extend ActiveSupport::Concern
13+
14+
included do
15+
alias_method_chain :execute!, :instrumentation
16+
end
17+
18+
# Wrap `Search#execute!` and perform instrumentation
19+
#
20+
def execute_with_instrumentation!
21+
ActiveSupport::Notifications.instrument "search.elasticsearch",
22+
name: 'Search',
23+
klass: (self.klass.is_a?(Elasticsearch::Model::Proxy::ClassMethodsProxy) ? self.klass.target.to_s : self.klass.to_s),
24+
search: self.definition do
25+
execute_without_instrumentation!
26+
end
27+
end
28+
end
29+
end
30+
end
31+
end
32+
end
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,27 @@
1+
module Elasticsearch
2+
module Rails
3+
module Instrumentation
4+
5+
# Rails initializer class to require Elasticsearch::Rails::Instrumentation files,
6+
# set up Elasticsearch::Model and hook into ActionController to display Elasticsearch-related duration
7+
#
8+
# @see http://edgeguides.rubyonrails.org/active_support_instrumentation.html
9+
#
10+
class Railtie < ::Rails::Railtie
11+
initializer "elasticsearch.instrumentation" do |app|
12+
require 'elasticsearch/rails/instrumentation/log_subscriber'
13+
require 'elasticsearch/rails/instrumentation/controller_runtime'
14+
15+
Elasticsearch::Model::Searching::SearchRequest.class_eval do
16+
include Elasticsearch::Rails::Instrumentation::Publishers::SearchRequest
17+
end
18+
19+
ActiveSupport.on_load(:action_controller) do
20+
include Elasticsearch::Rails::Instrumentation::ControllerRuntime
21+
end
22+
end
23+
end
24+
25+
end
26+
end
27+
end
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,40 @@
1+
module Elasticsearch
2+
module Rails
3+
module Lograge
4+
5+
# Rails initializer class to require Elasticsearch::Rails::Instrumentation files,
6+
# set up Elasticsearch::Model and add Lograge configuration to display Elasticsearch-related duration
7+
#
8+
# Require the component in your `application.rb` file and enable Lograge:
9+
#
10+
# require 'elasticsearch/rails/lograge'
11+
#
12+
# You should see the full duration of the request to Elasticsearch as part of each log event:
13+
#
14+
# method=GET path=/search ... status=200 duration=380.89 view=99.64 db=0.00 es=279.37
15+
#
16+
# @see https://github.com/roidrage/lograge
17+
#
18+
class Railtie < ::Rails::Railtie
19+
initializer "elasticsearch.lograge" do |app|
20+
require 'elasticsearch/rails/instrumentation/publishers'
21+
require 'elasticsearch/rails/instrumentation/log_subscriber'
22+
require 'elasticsearch/rails/instrumentation/controller_runtime'
23+
24+
config.lograge.custom_options = lambda do |event|
25+
{ es: event.payload[:elasticsearch_runtime].to_f.round(2) }
26+
end
27+
28+
Elasticsearch::Model::Searching::SearchRequest.class_eval do
29+
include Elasticsearch::Rails::Instrumentation::Publishers::SearchRequest
30+
end
31+
32+
ActiveSupport.on_load(:action_controller) do
33+
include Elasticsearch::Rails::Instrumentation::ControllerRuntime
34+
end
35+
end
36+
end
37+
38+
end
39+
end
40+
end
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,61 @@
1+
require 'test_helper'
2+
3+
require 'rails/railtie'
4+
require 'active_support/log_subscriber/test_helper'
5+
6+
require 'elasticsearch/rails/instrumentation'
7+
8+
class Elasticsearch::Rails::InstrumentationTest < Test::Unit::TestCase
9+
include ActiveSupport::LogSubscriber::TestHelper
10+
11+
context "ActiveSupport::Instrumentation integration" do
12+
class ::DummyInstrumentationModel
13+
extend Elasticsearch::Model::Searching::ClassMethods
14+
15+
def self.index_name; 'foo'; end
16+
def self.document_type; 'bar'; end
17+
end
18+
19+
RESPONSE = { 'took' => '5ms', 'hits' => { 'total' => 123, 'max_score' => 456, 'hits' => [] } }
20+
21+
setup do
22+
@search = Elasticsearch::Model::Searching::SearchRequest.new ::DummyInstrumentationModel, '*'
23+
@response = Elasticsearch::Model::Response::Response.new ::DummyInstrumentationModel, @search
24+
25+
@client = stub('client', search: @response)
26+
DummyInstrumentationModel.stubs(:client).returns(@client)
27+
28+
Elasticsearch::Rails::Instrumentation::Railtie.run_initializers
29+
end
30+
31+
should "wrap SearchRequest#execute! with instrumentation" do
32+
s = Elasticsearch::Model::Searching::SearchRequest.new ::DummyInstrumentationModel, 'foo'
33+
assert_respond_to s, :execute_without_instrumentation!
34+
assert_respond_to s, :execute_with_instrumentation!
35+
end
36+
37+
should "publish the notification" do
38+
@query = { query: { match: { foo: 'bar' } } }
39+
40+
ActiveSupport::Notifications.expects(:instrument).with do |name, payload|
41+
assert_equal "search.elasticsearch", name
42+
assert_equal 'DummyInstrumentationModel', payload[:klass]
43+
assert_equal @query, payload[:search][:body]
44+
end
45+
46+
s = ::DummyInstrumentationModel.search @query
47+
s.response
48+
end
49+
50+
should "print the debug information to the Rails log" do
51+
s = ::DummyInstrumentationModel.search query: { match: { moo: 'bam' } }
52+
s.response
53+
54+
logged = @logger.logged(:debug).first
55+
56+
assert_not_nil logged
57+
assert_match /DummyInstrumentationModel Search \(\d+\.\d+ms\)/, logged
58+
assert_match /body\: \{query\: \{match\: \{moo\: "bam"\}\}\}\}/, logged
59+
end
60+
end
61+
end
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,20 @@
1+
require 'test_helper'
2+
3+
require 'rails/railtie'
4+
require 'lograge'
5+
6+
require 'elasticsearch/rails/lograge'
7+
8+
class Elasticsearch::Rails::LogrageTest < Test::Unit::TestCase
9+
context "Lograge integration" do
10+
setup do
11+
Elasticsearch::Rails::Lograge::Railtie.run_initializers
12+
end
13+
14+
should "customize the Lograge configuration" do
15+
assert_not_nil Elasticsearch::Rails::Lograge::Railtie.initializers
16+
.select { |i| i.name == 'elasticsearch.lograge' }
17+
.first
18+
end
19+
end
20+
end

0 commit comments

Comments
 (0)