Active Support 提供的 instrumentation API 允許開發人員提供掛鉤,其他開發人員可以掛接到這些掛鉤。在 Rails 框架內有數個這類掛鉤 。透過此 API,開發人員可以選擇在應用程式或另一段 Ruby 程式碼中發生特定事件時收到通知。
例如,Active Record 中提供一個掛鉤 ,每次 Active Record 在資料庫上使用 SQL 查詢時都會呼叫此掛鉤。可以訂閱 此掛鉤,並用於追蹤特定動作期間的查詢次數。在處理控制器動作時,還有另一個掛鉤 。例如,這可用於追蹤特定動作所花費的時間。
您甚至可以在應用程式中建立自己的事件 ,以便稍後訂閱。
使用 ActiveSupport::Notifications.subscribe
和區塊來監聽任何通知。根據區塊所接受的引數數量,您將收到不同的資料。
訂閱事件的第一種方式是使用帶有單一引數的區塊。該引數將是 ActiveSupport::Notifications::Event
的執行個體。
ActiveSupport :: Notifications . subscribe "process_action.action_controller" do | event |
event . name # => "process_action.action_controller"
event . duration # => 10 (in milliseconds)
event . allocations # => 1826
event . payload # => {:extra=>information}
Rails . logger . info " #{ event } Received!"
end
複製
如果您不需要 Event 物件記錄的所有資料,您也可以指定一個區塊,該區塊接受以下五個引數
事件名稱
事件開始時間
事件結束時間
觸發事件的 instrumenter 的唯一 ID
事件的有效負載
ActiveSupport :: Notifications . subscribe "process_action.action_controller" do | name , started , finished , unique_id , payload |
# your own custom stuff
Rails . logger . info " #{ name } Received! (started: #{ started } , finished: #{ finished } )" # process_action.action_controller Received! (started: 2019-05-05 13:43:57 -0800, finished: 2019-05-05 13:43:58 -0800)
end
複製
如果您擔心 started
和 finished
的準確性,以計算精確的經過時間,則請使用 ActiveSupport::Notifications.monotonic_subscribe
。給定的區塊會收到與上述相同的引數,但 started
和 finished
的值將具有精確的單調時間,而不是掛鐘時間。
ActiveSupport :: Notifications . monotonic_subscribe "process_action.action_controller" do | name , started , finished , unique_id , payload |
# your own custom stuff
duration = finished - started # 1560979.429234 - 1560978.425334
Rails . logger . info " #{ name } Received! (duration: #{ duration } )" # process_action.action_controller Received! (duration: 1.0039)
end
複製
您也可以訂閱符合正規表示式的事件。這可讓您一次訂閱多個事件。以下是如何訂閱 ActionController
中的所有內容
ActiveSupport :: Notifications . subscribe ( /action_controller/ ) do | event |
# inspect all ActionController events
end
複製
在 Ruby on Rails 框架中,針對常見事件提供了一些掛鉤。這些事件及其有效負載的詳細資訊如下所示。
鍵
值
:controller
控制器名稱
:action
動作
:request
ActionDispatch::Request
物件
:params
未篩選參數的要求參數雜湊
:headers
要求標頭
:format
html/js/json/xml 等
:method
HTTP 要求動詞
:path
要求路徑
{
controller: "PostsController" ,
action: "new" ,
params: { "action" => "new" , "controller" => "posts" },
headers: #<ActionDispatch::Http::Headers:0x0055a67a519b88>,
format: :html ,
method: "GET" ,
path: "/posts/new"
}
複製
鍵
值
:controller
控制器名稱
:action
動作
:params
未篩選參數的要求參數雜湊
:headers
要求標頭
:format
html/js/json/xml 等
:method
HTTP 要求動詞
:path
要求路徑
:request
ActionDispatch::Request
物件
:response
ActionDispatch::Response
物件
:status
HTTP 狀態碼
:view_runtime
在視圖中花費的時間 (毫秒)
:db_runtime
執行資料庫查詢所花費的時間 (毫秒)
{
controller: "PostsController" ,
action: "index" ,
params: { "action" => "index" , "controller" => "posts" },
headers: #<ActionDispatch::Http::Headers:0x0055a67a519b88>,
format: :html ,
method: "GET" ,
path: "/posts" ,
request: #<ActionDispatch::Request:0x00007ff1cb9bd7b8>,
response: #<ActionDispatch::Response:0x00007f8521841ec8>,
status: 200 ,
view_runtime: 46.848 ,
db_runtime: 0.157
}
複製
呼叫者可能會新增其他鍵。
ActionController
不會將任何特定資訊新增至有效負載。所有選項都會傳遞至有效負載。
{
status: 302 ,
location: "https://127.0.0.1:3000/posts/new" ,
request: < ActionDispatch :: Request : 0x00007ff1cb9bd7b8 >
}
複製
{
filter: ":halting_filter"
}
複製
鍵
值
:keys
未允許的鍵
:context
具有下列鍵的雜湊::controller
、:action
、:params
、:request
鍵
值
:filename
檔案名稱
:type
HTTP 內容類型
:disposition
HTTP 內容處置
{
filename: "subscribers.csv" ,
type: "text/csv" ,
disposition: "attachment"
}
複製
{
key: 'posts/1-dashboard-view'
}
複製
{
key: 'posts/1-dashboard-view'
}
複製
{
key: 'posts/1-dashboard-view'
}
複製
{
key: 'posts/1-dashboard-view'
}
複製
鍵
值
:identifier
範本的完整路徑
:layout
適用的版面配置
:locals
傳遞至範本的區域變數
{
identifier: "/Users/adam/projects/notifications/app/views/posts/index.html.erb" ,
layout: "layouts/application" ,
locals: { foo: "bar" }
}
複製
鍵
值
:identifier
範本的完整路徑
:locals
傳遞至範本的區域變數
{
identifier: "/Users/adam/projects/notifications/app/views/posts/_form.html.erb" ,
locals: { foo: "bar" }
}
複製
鍵
值
:identifier
範本的完整路徑
:count
集合的大小
:cache_hits
從快取擷取的部分的數量
只有在使用 cached: true
呈現集合時,才會包含 :cache_hits
鍵。
{
identifier: "/Users/adam/projects/notifications/app/views/posts/_post.html.erb" ,
count: 3 ,
cache_hits: 0
}
複製
{
identifier: "/Users/adam/projects/notifications/app/views/layouts/application.html.erb"
}
複製
鍵
值
:sql
SQL 敘述
:name
操作的名稱
:connection
連線物件
:transaction
目前的交易,如果有的話
:binds
綁定參數
:type_casted_binds
型別轉換後的綁定參數
:statement_name
SQL 敘述名稱
:async
如果查詢是以非同步方式載入,則為 true
:cached
當使用快取查詢時會加入 true
:row_count
查詢返回的列數
适配器也可能會添加它們自己的資料。
{
sql: "SELECT \" posts \" .* FROM \" posts \" " ,
name: "Post Load" ,
connection: < ActiveRecord :: ConnectionAdapters :: SQLite3Adapter : 0x00007f9f7a838850 > ,
transaction: < ActiveRecord :: ConnectionAdapters :: RealTransaction : 0x0000000121b5d3e0 >
binds: [ < ActiveModel :: Attribute :: WithCastValue : 0x00007fe19d15dc00 > ],
type_casted_binds: [ 11 ],
statement_name: nil ,
row_count: 5
}
複製
如果查詢不是在交易的上下文中執行,則 :transaction
為 nil
。
只有當 config.active_record.action_on_strict_loading_violation
設定為 :log
時,才會發出此事件。
鍵
值
:owner
已啟用 strict_loading
的模型
:reflection
嘗試載入的關聯的反射
鍵
值
:record_count
已實例化的記錄數
:class_name
記錄的類別
{
record_count: 1 ,
class_name: "User"
}
複製
當交易開始時,會發出此事件。
鍵
值
:transaction
交易物件
:connection
連線物件
請注意,Active Record 在需要之前不會建立實際的資料庫交易
ActiveRecord :: Base . transaction do
# We are inside the block, but no event has been triggered yet.
# The following line makes Active Record start the transaction.
User . count # Event fired here.
end
複製
請記住,普通的巢狀呼叫不會建立新的交易
ActiveRecord :: Base . transaction do | t1 |
User . count # Fires an event for t1.
ActiveRecord :: Base . transaction do | t2 |
# The next line fires no event for t2, because the only
# real database transaction in this example is t1.
User . first . touch
end
end
複製
但是,如果傳遞 requires_new: true
,您也會獲得巢狀交易的事件。這可能是在底層的儲存點
ActiveRecord :: Base . transaction do | t1 |
User . count # Fires an event for t1.
ActiveRecord :: Base . transaction ( requires_new: true ) do | t2 |
User . first . touch # Fires an event for t2.
end
end
複製
當資料庫交易完成時,會發出此事件。交易的狀態可以在 :outcome
鍵中找到。
鍵
值
:transaction
交易物件
:outcome
:commit
、:rollback
、:restart
或 :incomplete
:connection
連線物件
實際上,您無法對交易物件執行太多操作,但它可能仍然有助於追蹤資料庫活動。例如,透過追蹤 transaction.uuid
。
鍵
值
:mailer
郵件程式類別的名稱
:message_id
郵件 gem 產生的郵件 ID
:subject
郵件的主旨
:to
郵件的收件人地址
:from
郵件的寄件人地址
:bcc
郵件的密件副本地址
:cc
郵件的副本地址
:date
郵件的日期
:mail
郵件的編碼形式
:perform_deliveries
是否執行此郵件的傳送
{
mailer: "Notification" ,
message_id: "4f5b5491f1774_181b23fc3d4434d38138e5@mba.local.mail" ,
subject: "Rails Guides" ,
to: [ "users@rails.com" , "dhh@rails.com" ],
from: [ "me@rails.com" ],
date: Sat , 10 Mar 2012 14 : 18 : 09 + 0100 ,
mail: "..." , # omitted for brevity
perform_deliveries: true
}
複製
鍵
值
:mailer
郵件程式類別的名稱
:action
動作
:args
參數
{
mailer: "Notification" ,
action: "welcome_email" ,
args: []
}
複製
鍵
值
:key
在儲存區中使用的鍵
:store
儲存區類別的名稱
:hit
如果此讀取為命中
:super_operation
如果讀取是使用 fetch
執行,則為 :fetch
鍵
值
:key
在儲存區中使用的鍵
:store
儲存區類別的名稱
:hits
快取命中的鍵
:super_operation
如果讀取是使用 fetch_multi
執行,則為 :fetch_multi
只有當使用區塊呼叫 fetch
時,才會發出此事件。
鍵
值
:key
在儲存區中使用的鍵
:store
儲存區類別的名稱
傳遞給 fetch
的選項將在寫入儲存區時與有效負載合併。
{
key: "name-of-complicated-computation" ,
store: "ActiveSupport::Cache::MemCacheStore"
}
複製
只有當使用區塊呼叫 fetch
時,才會發出此事件。
鍵
值
:key
在儲存區中使用的鍵
:store
儲存區類別的名稱
傳遞給 fetch
的選項將與有效負載合併。
{
key: "name-of-complicated-computation" ,
store: "ActiveSupport::Cache::MemCacheStore"
}
複製
鍵
值
:key
在儲存區中使用的鍵
:store
儲存區類別的名稱
快取儲存區也可能會添加它們自己的資料。
{
key: "name-of-complicated-computation" ,
store: "ActiveSupport::Cache::MemCacheStore"
}
複製
鍵
值
:key
寫入儲存區的鍵和值
:store
儲存區類別的名稱
鍵
值
:key
在儲存區中使用的鍵
:store
儲存區類別的名稱
:amount
遞增量
{
key: "bottles-of-beer" ,
store: "ActiveSupport::Cache::RedisCacheStore" ,
amount: 99
}
複製
鍵
值
:key
在儲存區中使用的鍵
:store
儲存區類別的名稱
:amount
遞減量
{
key: "bottles-of-beer" ,
store: "ActiveSupport::Cache::RedisCacheStore" ,
amount: 1
}
複製
鍵
值
:key
在儲存區中使用的鍵
:store
儲存區類別的名稱
{
key: "name-of-complicated-computation" ,
store: "ActiveSupport::Cache::MemCacheStore"
}
複製
鍵
值
:key
在儲存區中使用的鍵
:store
儲存區類別的名稱
只有在使用 RedisCacheStore
、FileStore
或 MemoryStore
時,才會發出此事件。
鍵
值
:key
使用的鍵模式
:store
儲存區類別的名稱
{
key: "posts/*" ,
store: "ActiveSupport::Cache::RedisCacheStore"
}
複製
只有在使用 MemoryStore
時,才會發出此事件。
鍵
值
:store
儲存區類別的名稱
:size
清理之前快取中的條目數
{
store: "ActiveSupport::Cache::MemoryStore" ,
size: 9001
}
複製
只有在使用 MemoryStore
時,才會發出此事件。
鍵
值
:store
儲存區類別的名稱
:key
快取的目標大小 (以位元組為單位)
:from
修剪之前快取的大小 (以位元組為單位)
{
store: "ActiveSupport::Cache::MemoryStore" ,
key: 5000 ,
from: 9001
}
複製
鍵
值
:key
在儲存區中使用的鍵
:store
儲存區類別的名稱
{
key: "name-of-complicated-computation" ,
store: "ActiveSupport::Cache::MemCacheStore"
}
複製
鍵
值
:serializer
主要 (預期) 的序列化器
:fallback
後備 (實際) 序列化器
:serialized
序列化的字串
:deserialized
反序列化的值
{
serializer: :json_allow_marshal ,
fallback: :marshal ,
serialized: " \x04\b { \x06 I \"\n Hello \x06 : \x06 ETI \"\n World \x06 ; \x00 T" ,
deserialized: { "Hello" => "World" },
}
複製
鍵
值
:adapter
處理工作的 QueueAdapter 物件
:job
工作物件
鍵
值
:adapter
處理工作的 QueueAdapter 物件
:job
工作物件
鍵
值
:job
工作物件
:adapter
處理工作的 QueueAdapter 物件
:error
導致重試的錯誤
:wait
重試的延遲
鍵
值
:adapter
處理工作的 QueueAdapter 物件
:jobs
工作物件的陣列
鍵
值
:adapter
處理工作的 QueueAdapter 物件
:job
工作物件
鍵
值
:adapter
處理工作的 QueueAdapter 物件
:job
工作物件
:db_runtime
執行資料庫查詢所花費的時間 (毫秒)
鍵
值
:adapter
處理工作的 QueueAdapter 物件
:job
工作物件
:error
導致重試的錯誤
鍵
值
:adapter
處理工作的 QueueAdapter 物件
:job
工作物件
:error
導致丟棄的錯誤
鍵
值
:channel_class
頻道類別的名稱
:action
動作
:data
資料的雜湊
鍵
值
:channel_class
頻道類別的名稱
:data
資料的雜湊
:via
透過
鍵
值
:channel_class
頻道類別的名稱
鍵
值
:channel_class
頻道類別的名稱
鍵
值
:broadcasting
具名的廣播
:message
訊息的雜湊
:coder
編碼器
鍵
值
:analyzer
分析器的名稱,例如 ffprobe
鍵
值
:key
安全權杖
:service
服務的名稱
:checksum
用於確保完整性的校驗和
鍵
值
:key
安全權杖
:service
服務的名稱
鍵
值
:key
安全權杖
:service
服務的名稱
:range
嘗試讀取的位元組範圍
鍵
值
:key
安全權杖
:service
服務的名稱
鍵
值
:key
安全權杖
:service
服務的名稱
鍵
值
:prefix
鍵前綴
:service
服務的名稱
鍵
值
:key
安全權杖
:service
服務的名稱
:exist
檔案或 Blob 是否存在
鍵
值
:key
安全權杖
:service
服務的名稱
:url
產生的 URL
只有在使用 Google Cloud Storage 服務時,才會發出此事件。
鍵
值
:key
安全權杖
:service
服務的名稱
:content_type
HTTP Content-Type
欄位
:disposition
HTTP Content-Disposition
欄位
{
mailbox: #<RepliesMailbox:0x00007f9f7a8388>,
inbound_email: {
id: 1 ,
message_id: "0CB459E0-0336-41DA-BC88-E6E28C697DDB@37signals.com" ,
status: "processing"
}
}
複製
鍵
值
:initializer
在 config/initializers
中載入的初始化器的路徑
鍵
值
:message
棄用警告
:callstack
棄用的來源
:gem_name
回報棄用的 gem 名稱
:deprecation_horizon
將移除已棄用行為的版本
如果在任何檢測期間發生例外狀況,有效負載將包含有關它的資訊。
鍵
值
:exception
由兩個元素組成的陣列。例外狀況類別名稱和訊息
:exception_object
例外狀況物件
加入您自己的事件也很容易。Active Support 會為您處理所有繁重的工作。只需使用 name
、payload
和區塊呼叫 ActiveSupport::Notifications.instrument
。通知將在區塊傳回後傳送。Active Support 將產生開始和結束時間,並加入檢測器的唯一 ID。傳遞到 instrument
呼叫的所有資料都將進入有效負載。
這裡有一個範例
ActiveSupport :: Notifications . instrument "my.custom.event" , this: :data do
# do your custom stuff here
end
複製
現在您可以使用以下方式監聽此事件
ActiveSupport :: Notifications . subscribe "my.custom.event" do | name , started , finished , unique_id , data |
puts data . inspect # {:this=>:data}
end
複製
您也可以在不傳遞區塊的情況下呼叫 instrument
。這可讓您利用檢測基礎結構進行其他訊息傳遞用途。
ActiveSupport :: Notifications . instrument "my.custom.event" , this: :data
ActiveSupport :: Notifications . subscribe "my.custom.event" do | name , started , finished , unique_id , data |
puts data . inspect # {:this=>:data}
end
複製
定義您自己的事件時,您應該遵循 Rails 慣例。格式為:event.library
。如果您的應用程式正在傳送推文,您應該建立一個名為 tweet.twitter
的事件。