module Xray # This is the main point of integration with Rails. This engine hooks into # Sprockets and monkey patches ActionView in order to augment the app's JS # and HTML templates with filepath information that can be used by xray.js # in the browser. It also hooks in a middleware responsible for injecting # xray.js and the xray bar into the app's response bodies. class Engine < ::Rails::Engine initializer "xray.initialize" do |app| ensure_asset_pipeline_enabled! app app.middleware.use Xray::Middleware # Register as a Sprockets processor to augment JS files, including # compiled coffeescript, with filepath information. See # `Xray.augment_js` for details. app.assets.register_postprocessor 'application/javascript', :xray do |context, data| path = context.pathname.to_s if path =~ /^#{app.root}.+\.(js|coffee)(\.|$)/ Xray.augment_js(data, path) else data end end # Monkey patch ActionView::Template to augment server-side templates # with filepath information. See `Xray.augment_template` for details. ActionView::Template.class_eval do def render_with_xray(*args, &block) path = identifier source = render_without_xray(*args, &block) suitable_template = path =~ /\.(html|slim|haml)(\.|$)/ && !path.match(/\.(js|json)\./) && !path.include?('_xray_bar') options = args.last.kind_of?(Hash) ? args.last : {} if suitable_template && !(options.has_key?(:xray) && (options[:xray] == false)) Xray.augment_template(source, path) else source end end alias_method_chain :render, :xray end # Augment JS templates app.assets.register_preprocessor 'application/javascript', :xray do |context, source| path = context.pathname.to_s if path =~ /^#{app.root}.+\.(jst)(\.|$)/ Xray.augment_template(source, path) else source end end # This event is called near the beginning of a request cycle. We use it to # collect information about the controller and action that is responding, for # display in the Xray bar. ActiveSupport::Notifications.subscribe('start_processing.action_controller') do |*args| event = ActiveSupport::Notifications::Event.new(*args) controller_name = event.payload[:controller] action_name = event.payload[:action] path = ActiveSupport::Dependencies.search_for_file(controller_name.underscore) # Reset the request info hash for this request. # NOTE: Nothing about this is thread-safe. Could this affect anyone in dev mode? Xray.request_info.clear Xray.request_info[:controller] = { :path => path, :name => controller_name, :action => action_name } end # This event is called each time during the request cycle that # ActionView renders a template. The first time it's called will most # likely be the view the controller is rendering, which is what we're # interested in. ActiveSupport::Notifications.subscribe('render_template.action_view') do |*args| event = ActiveSupport::Notifications::Event.new(*args) layout = event.payload[:layout] path = event.payload[:identifier] # We are only interested in the first notification that has a layout. if layout Xray.request_info[:view] ||= { :path => path, :layout => layout } end end end def ensure_asset_pipeline_enabled!(app) unless app.assets raise "xray-rails requires the Rails asset pipeline. The asset pipeline is currently disabled in this application. Either convert your application to use the asset pipeline, or remove xrail-rails from your Gemfile." end end end end