Sha256: 99f25bc0655f2005da9c55a793c9900117c36fc4800d2edf0def668424273654

Contents?: true

Size: 1.83 KB

Versions: 20

Compression:

Stored size: 1.83 KB

Contents

# frozen_string_literal: true

require "active_support/notifications"

module ActionDispatch
  class ServerTiming
    SERVER_TIMING_HEADER = "Server-Timing"

    class Subscriber # :nodoc:
      include Singleton
      KEY = :action_dispatch_server_timing_events

      def initialize
        @mutex = Mutex.new
      end

      def call(event)
        if events = ActiveSupport::IsolatedExecutionState[KEY]
          events << event
        end
      end

      def collect_events
        events = []
        ActiveSupport::IsolatedExecutionState[KEY] = events
        yield
        events
      ensure
        ActiveSupport::IsolatedExecutionState.delete(KEY)
      end

      def ensure_subscribed
        @mutex.synchronize do
          # Subscribe to all events, except those beginning with "!"
          # Ideally we would be more selective of what is being measured
          @subscriber ||= ActiveSupport::Notifications.subscribe(/\A[^!]/, self)
        end
      end

      def unsubscribe
        @mutex.synchronize do
          ActiveSupport::Notifications.unsubscribe @subscriber
          @subscriber = nil
        end
      end
    end

    def self.unsubscribe # :nodoc:
      Subscriber.instance.unsubscribe
    end

    def initialize(app)
      @app = app
      @subscriber = Subscriber.instance
      @subscriber.ensure_subscribed
    end

    def call(env)
      response = nil
      events = @subscriber.collect_events do
        response = @app.call(env)
      end

      headers = response[1]

      header_info = events.group_by(&:name).map do |event_name, events_collection|
        "%s;dur=%.2f" % [event_name, events_collection.sum(&:duration)]
      end

      header_info.prepend(headers[SERVER_TIMING_HEADER]) if headers[SERVER_TIMING_HEADER].present?
      headers[SERVER_TIMING_HEADER] = header_info.join(", ")

      response
    end
  end
end

Version data entries

20 entries across 20 versions & 3 rubygems

Version Path
actionpack-7.0.8.7 lib/action_dispatch/middleware/server_timing.rb
actionpack-7.0.8.6 lib/action_dispatch/middleware/server_timing.rb
actionpack-7.0.8.5 lib/action_dispatch/middleware/server_timing.rb
blacklight-spotlight-3.6.0.beta8 vendor/bundle/ruby/3.2.0/gems/actionpack-7.0.8.4/lib/action_dispatch/middleware/server_timing.rb
cm-admin-1.5.22 vendor/bundle/ruby/3.3.0/gems/actionpack-7.0.5.1/lib/action_dispatch/middleware/server_timing.rb
cm-admin-1.5.21 vendor/bundle/ruby/3.3.0/gems/actionpack-7.0.5.1/lib/action_dispatch/middleware/server_timing.rb
cm-admin-1.5.20 vendor/bundle/ruby/3.3.0/gems/actionpack-7.0.5.1/lib/action_dispatch/middleware/server_timing.rb
actionpack-7.0.8.4 lib/action_dispatch/middleware/server_timing.rb
actionpack-7.0.8.1 lib/action_dispatch/middleware/server_timing.rb
actionpack-7.0.8 lib/action_dispatch/middleware/server_timing.rb
actionpack-7.0.7.2 lib/action_dispatch/middleware/server_timing.rb
actionpack-7.0.7.1 lib/action_dispatch/middleware/server_timing.rb
actionpack-7.0.7 lib/action_dispatch/middleware/server_timing.rb
actionpack-7.0.6 lib/action_dispatch/middleware/server_timing.rb
actionpack-7.0.5.1 lib/action_dispatch/middleware/server_timing.rb
actionpack-7.0.5 lib/action_dispatch/middleware/server_timing.rb
actionpack-7.0.4.3 lib/action_dispatch/middleware/server_timing.rb
actionpack-7.0.4.2 lib/action_dispatch/middleware/server_timing.rb
actionpack-7.0.4.1 lib/action_dispatch/middleware/server_timing.rb
actionpack-7.0.4 lib/action_dispatch/middleware/server_timing.rb