更多資訊請參考 rubyonrails.org:

Active Support Instrumentation

Active Support 是 Rails 核心的一部分,提供 Ruby 語言擴充、實用工具和其他功能。其中一項功能是 instrumentation API,可用於應用程式內部測量 Ruby 程式碼中發生的特定動作,例如 Rails 應用程式或框架本身的動作。但它不限於 Rails。如果需要,可以獨立用於其他 Ruby 腳本中。

在本指南中,您將學習如何使用 Active Support 的 instrumentation API 來測量 Rails 和其他 Ruby 程式碼中的事件。

閱讀本指南後,您將了解

  • instrumentation 可以提供什麼。
  • 如何將訂閱者新增至掛鉤。
  • Rails 框架中用於 instrumentation 的掛鉤。
  • 如何建構自訂 instrumentation 實作。

1 Instrumentation 簡介

Active Support 提供的 instrumentation API 允許開發人員提供掛鉤,其他開發人員可以掛接到這些掛鉤。在 Rails 框架內有數個這類掛鉤。透過此 API,開發人員可以選擇在應用程式或另一段 Ruby 程式碼中發生特定事件時收到通知。

例如,Active Record 中提供一個掛鉤,每次 Active Record 在資料庫上使用 SQL 查詢時都會呼叫此掛鉤。可以訂閱此掛鉤,並用於追蹤特定動作期間的查詢次數。在處理控制器動作時,還有另一個掛鉤。例如,這可用於追蹤特定動作所花費的時間。

您甚至可以在應用程式中建立自己的事件,以便稍後訂閱。

2 訂閱事件

使用 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

如果您擔心 startedfinished 的準確性,以計算精確的經過時間,則請使用 ActiveSupport::Notifications.monotonic_subscribe。給定的區塊會收到與上述相同的引數,但 startedfinished 的值將具有精確的單調時間,而不是掛鐘時間。

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

3 Rails 框架掛鉤

在 Ruby on Rails 框架中,針對常見事件提供了一些掛鉤。這些事件及其有效負載的詳細資訊如下所示。

3.1 Action Controller

3.1.1 start_processing.action_controller

: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"
}

3.1.2 process_action.action_controller

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

3.1.3 send_file.action_controller

:path 檔案的完整路徑

呼叫者可能會新增其他鍵。

3.1.4 send_data.action_controller

ActionController 不會將任何特定資訊新增至有效負載。所有選項都會傳遞至有效負載。

3.1.5 redirect_to.action_controller

:status HTTP 回應代碼
:location 重新導向至的 URL
:request ActionDispatch::Request 物件
{
  status: 302,
  location: "https://127.0.0.1:3000/posts/new",
  request: <ActionDispatch::Request:0x00007ff1cb9bd7b8>
}

3.1.6 halted_callback.action_controller

:filter 停止動作的篩選器
{
  filter: ":halting_filter"
}

3.1.7 unpermitted_parameters.action_controller

:keys 未允許的鍵
:context 具有下列鍵的雜湊::controller:action:params:request

3.1.8 send_stream.action_controller

:filename 檔案名稱
:type HTTP 內容類型
:disposition HTTP 內容處置
{
  filename: "subscribers.csv",
  type: "text/csv",
  disposition: "attachment"
}

3.2 Action Controller:快取

3.2.1 write_fragment.action_controller

:key 完整的鍵
{
  key: 'posts/1-dashboard-view'
}

3.2.2 read_fragment.action_controller

:key 完整的鍵
{
  key: 'posts/1-dashboard-view'
}

3.2.3 expire_fragment.action_controller

:key 完整的鍵
{
  key: 'posts/1-dashboard-view'
}

3.2.4 exist_fragment?.action_controller

:key 完整的鍵
{
  key: 'posts/1-dashboard-view'
}

3.3 Action Dispatch

3.3.1 process_middleware.action_dispatch

:middleware 中介軟體的名稱

3.3.2 redirect.action_dispatch

:status HTTP 回應代碼
:location 重新導向至的 URL
:request ActionDispatch::Request 物件

3.3.3 request.action_dispatch

:request ActionDispatch::Request 物件

3.4 Action View

3.4.1 render_template.action_view

:identifier 範本的完整路徑
:layout 適用的版面配置
:locals 傳遞至範本的區域變數
{
  identifier: "/Users/adam/projects/notifications/app/views/posts/index.html.erb",
  layout: "layouts/application",
  locals: { foo: "bar" }
}

3.4.2 render_partial.action_view

:identifier 範本的完整路徑
:locals 傳遞至範本的區域變數
{
  identifier: "/Users/adam/projects/notifications/app/views/posts/_form.html.erb",
  locals: { foo: "bar" }
}

3.4.3 render_collection.action_view

:identifier 範本的完整路徑
:count 集合的大小
:cache_hits 從快取擷取的部分的數量

只有在使用 cached: true 呈現集合時,才會包含 :cache_hits 鍵。

{
  identifier: "/Users/adam/projects/notifications/app/views/posts/_post.html.erb",
  count: 3,
  cache_hits: 0
}

3.4.4 render_layout.action_view

:identifier 範本的完整路徑
{
  identifier: "/Users/adam/projects/notifications/app/views/layouts/application.html.erb"
}

3.5 Active Record

3.5.1 sql.active_record

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

如果查詢不是在交易的上下文中執行,則 :transactionnil

3.5.2 strict_loading_violation.active_record

只有當 config.active_record.action_on_strict_loading_violation 設定為 :log 時,才會發出此事件。

:owner 已啟用 strict_loading 的模型
:reflection 嘗試載入的關聯的反射

3.5.3 instantiation.active_record

:record_count 已實例化的記錄數
:class_name 記錄的類別
{
  record_count: 1,
  class_name: "User"
}

3.5.4 start_transaction.active_record

當交易開始時,會發出此事件。

: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

3.5.5 transaction.active_record

當資料庫交易完成時,會發出此事件。交易的狀態可以在 :outcome 鍵中找到。

:transaction 交易物件
:outcome :commit:rollback:restart:incomplete
:connection 連線物件

實際上,您無法對交易物件執行太多操作,但它可能仍然有助於追蹤資料庫活動。例如,透過追蹤 transaction.uuid

3.6 Action Mailer

3.6.1 deliver.action_mailer

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

3.6.2 process.action_mailer

:mailer 郵件程式類別的名稱
:action 動作
:args 參數
{
  mailer: "Notification",
  action: "welcome_email",
  args: []
}

3.7 Active Support:快取

3.7.1 cache_read.active_support

:key 在儲存區中使用的鍵
:store 儲存區類別的名稱
:hit 如果此讀取為命中
:super_operation 如果讀取是使用 fetch 執行,則為 :fetch

3.7.2 cache_read_multi.active_support

:key 在儲存區中使用的鍵
:store 儲存區類別的名稱
:hits 快取命中的鍵
:super_operation 如果讀取是使用 fetch_multi 執行,則為 :fetch_multi

3.7.3 cache_generate.active_support

只有當使用區塊呼叫 fetch 時,才會發出此事件。

:key 在儲存區中使用的鍵
:store 儲存區類別的名稱

傳遞給 fetch 的選項將在寫入儲存區時與有效負載合併。

{
  key: "name-of-complicated-computation",
  store: "ActiveSupport::Cache::MemCacheStore"
}

3.7.4 cache_fetch_hit.active_support

只有當使用區塊呼叫 fetch 時,才會發出此事件。

:key 在儲存區中使用的鍵
:store 儲存區類別的名稱

傳遞給 fetch 的選項將與有效負載合併。

{
  key: "name-of-complicated-computation",
  store: "ActiveSupport::Cache::MemCacheStore"
}

3.7.5 cache_write.active_support

:key 在儲存區中使用的鍵
:store 儲存區類別的名稱

快取儲存區也可能會添加它們自己的資料。

{
  key: "name-of-complicated-computation",
  store: "ActiveSupport::Cache::MemCacheStore"
}

3.7.6 cache_write_multi.active_support

:key 寫入儲存區的鍵和值
:store 儲存區類別的名稱

3.7.7 cache_increment.active_support

:key 在儲存區中使用的鍵
:store 儲存區類別的名稱
:amount 遞增量
{
  key: "bottles-of-beer",
  store: "ActiveSupport::Cache::RedisCacheStore",
  amount: 99
}

3.7.8 cache_decrement.active_support

:key 在儲存區中使用的鍵
:store 儲存區類別的名稱
:amount 遞減量
{
  key: "bottles-of-beer",
  store: "ActiveSupport::Cache::RedisCacheStore",
  amount: 1
}

3.7.9 cache_delete.active_support

:key 在儲存區中使用的鍵
:store 儲存區類別的名稱
{
  key: "name-of-complicated-computation",
  store: "ActiveSupport::Cache::MemCacheStore"
}

3.7.10 cache_delete_multi.active_support

:key 在儲存區中使用的鍵
:store 儲存區類別的名稱

3.7.11 cache_delete_matched.active_support

只有在使用 RedisCacheStoreFileStoreMemoryStore 時,才會發出此事件。

:key 使用的鍵模式
:store 儲存區類別的名稱
{
  key: "posts/*",
  store: "ActiveSupport::Cache::RedisCacheStore"
}

3.7.12 cache_cleanup.active_support

只有在使用 MemoryStore 時,才會發出此事件。

:store 儲存區類別的名稱
:size 清理之前快取中的條目數
{
  store: "ActiveSupport::Cache::MemoryStore",
  size: 9001
}

3.7.13 cache_prune.active_support

只有在使用 MemoryStore 時,才會發出此事件。

:store 儲存區類別的名稱
:key 快取的目標大小 (以位元組為單位)
:from 修剪之前快取的大小 (以位元組為單位)
{
  store: "ActiveSupport::Cache::MemoryStore",
  key: 5000,
  from: 9001
}

3.7.14 cache_exist?.active_support

:key 在儲存區中使用的鍵
:store 儲存區類別的名稱
{
  key: "name-of-complicated-computation",
  store: "ActiveSupport::Cache::MemCacheStore"
}

3.8 Active Support:訊息

3.8.1 message_serializer_fallback.active_support

:serializer 主要 (預期) 的序列化器
:fallback 後備 (實際) 序列化器
:serialized 序列化的字串
:deserialized 反序列化的值
{
  serializer: :json_allow_marshal,
  fallback: :marshal,
  serialized: "\x04\b{\x06I\"\nHello\x06:\x06ETI\"\nWorld\x06;\x00T",
  deserialized: { "Hello" => "World" },
}

3.9 Active Job

3.9.1 enqueue_at.active_job

:adapter 處理工作的 QueueAdapter 物件
:job 工作物件

3.9.2 enqueue.active_job

:adapter 處理工作的 QueueAdapter 物件
:job 工作物件

3.9.3 enqueue_retry.active_job

:job 工作物件
:adapter 處理工作的 QueueAdapter 物件
:error 導致重試的錯誤
:wait 重試的延遲

3.9.4 enqueue_all.active_job

:adapter 處理工作的 QueueAdapter 物件
:jobs 工作物件的陣列

3.9.5 perform_start.active_job

:adapter 處理工作的 QueueAdapter 物件
:job 工作物件

3.9.6 perform.active_job

:adapter 處理工作的 QueueAdapter 物件
:job 工作物件
:db_runtime 執行資料庫查詢所花費的時間 (毫秒)

3.9.7 retry_stopped.active_job

:adapter 處理工作的 QueueAdapter 物件
:job 工作物件
:error 導致重試的錯誤

3.9.8 discard.active_job

:adapter 處理工作的 QueueAdapter 物件
:job 工作物件
:error 導致丟棄的錯誤

3.10 Action Cable

3.10.1 perform_action.action_cable

:channel_class 頻道類別的名稱
:action 動作
:data 資料的雜湊

3.10.2 transmit.action_cable

:channel_class 頻道類別的名稱
:data 資料的雜湊
:via 透過

3.10.3 transmit_subscription_confirmation.action_cable

:channel_class 頻道類別的名稱

3.10.4 transmit_subscription_rejection.action_cable

:channel_class 頻道類別的名稱

3.10.5 broadcast.action_cable

:broadcasting 具名的廣播
:message 訊息的雜湊
:coder 編碼器

3.11 Active Storage

3.11.1 preview.active_storage

:key 安全權杖

3.11.2 transform.active_storage

3.11.3 analyze.active_storage

:analyzer 分析器的名稱,例如 ffprobe

3.12 Active Storage:儲存服務

3.12.1 service_upload.active_storage

:key 安全權杖
:service 服務的名稱
:checksum 用於確保完整性的校驗和

3.12.2 service_streaming_download.active_storage

:key 安全權杖
:service 服務的名稱

3.12.3 service_download_chunk.active_storage

:key 安全權杖
:service 服務的名稱
:range 嘗試讀取的位元組範圍

3.12.4 service_download.active_storage

:key 安全權杖
:service 服務的名稱

3.12.5 service_delete.active_storage

:key 安全權杖
:service 服務的名稱

3.12.6 service_delete_prefixed.active_storage

:prefix 鍵前綴
:service 服務的名稱

3.12.7 service_exist.active_storage

:key 安全權杖
:service 服務的名稱
:exist 檔案或 Blob 是否存在

3.12.8 service_url.active_storage

:key 安全權杖
:service 服務的名稱
:url 產生的 URL

3.12.9 service_update_metadata.active_storage

只有在使用 Google Cloud Storage 服務時,才會發出此事件。

:key 安全權杖
:service 服務的名稱
:content_type HTTP Content-Type 欄位
:disposition HTTP Content-Disposition 欄位

3.13 Action Mailbox

3.13.1 process.action_mailbox

:mailbox 繼承自 ActionMailbox::Base 的 Mailbox 類別的實例
:inbound_email 包含有關正在處理的入站電子郵件的資料的雜湊
{
  mailbox: #<RepliesMailbox:0x00007f9f7a8388>,
  inbound_email: {
    id: 1,
    message_id: "0CB459E0-0336-41DA-BC88-E6E28C697DDB@37signals.com",
    status: "processing"
  }
}

3.14 Railties

3.14.1 load_config_initializer.railties

:initializer config/initializers 中載入的初始化器的路徑

3.15 Rails

3.15.1 deprecation.rails

:message 棄用警告
:callstack 棄用的來源
:gem_name 回報棄用的 gem 名稱
:deprecation_horizon 將移除已棄用行為的版本

4 例外狀況

如果在任何檢測期間發生例外狀況,有效負載將包含有關它的資訊。

:exception 由兩個元素組成的陣列。例外狀況類別名稱和訊息
:exception_object 例外狀況物件

5 建立自訂事件

加入您自己的事件也很容易。Active Support 會為您處理所有繁重的工作。只需使用 namepayload 和區塊呼叫 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 的事件。



回到頂端