rails_exception_logger 0.2.1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- checksums.yaml +7 -0
- data/MIT-LICENSE +20 -0
- data/README.rdoc +135 -0
- data/Rakefile +40 -0
- data/app/assets/javascripts/rails_exception_logger/application.js +15 -0
- data/app/assets/javascripts/rails_exception_logger/logged_exceptions.js +2 -0
- data/app/assets/javascripts/rails_exception_logger/rails_exception_logger.js +69 -0
- data/app/assets/stylesheets/rails_exception_logger/application.css +13 -0
- data/app/assets/stylesheets/rails_exception_logger/rails_exception_logger.css +320 -0
- data/app/assets/stylesheets/rails_exception_logger/rails_logged_exceptions.css +4 -0
- data/app/controllers/rails_exception_logger/application_controller.rb +4 -0
- data/app/controllers/rails_exception_logger/logged_exceptions_controller.rb +105 -0
- data/app/helpers/exception_logger/application_helper.rb +4 -0
- data/app/helpers/exception_logger/logged_exceptions_helper.rb +42 -0
- data/app/models/rails_exception_logger/logged_exception.rb +83 -0
- data/app/views/layouts/rails_exception_logger/application.html.erb +20 -0
- data/app/views/rails_exception_logger/logged_exceptions/_exceptions.html.erb +32 -0
- data/app/views/rails_exception_logger/logged_exceptions/_feed.html.erb +3 -0
- data/app/views/rails_exception_logger/logged_exceptions/_show.html.erb +23 -0
- data/app/views/rails_exception_logger/logged_exceptions/destroy.js.erb +2 -0
- data/app/views/rails_exception_logger/logged_exceptions/destroy_all.js.erb +2 -0
- data/app/views/rails_exception_logger/logged_exceptions/feed.rss.builder +20 -0
- data/app/views/rails_exception_logger/logged_exceptions/index.html.erb +46 -0
- data/app/views/rails_exception_logger/logged_exceptions/index.js.erb +2 -0
- data/app/views/rails_exception_logger/logged_exceptions/query.js.erb +2 -0
- data/app/views/rails_exception_logger/logged_exceptions/show.html.erb +8 -0
- data/app/views/rails_exception_logger/logged_exceptions/show.js.erb +2 -0
- data/config/initializers/date_formats.rb +5 -0
- data/config/locales/en.yml +17 -0
- data/config/routes.rb +13 -0
- data/db/migrate/20120507081835_create_rails_exception_logger_logged_exceptions.rb +14 -0
- data/lib/rails_exception_logger/engine.rb +5 -0
- data/lib/rails_exception_logger/version.rb +3 -0
- data/lib/rails_exception_logger.rb +68 -0
- data/lib/tasks/rails_exception_logger_tasks.rake +4 -0
- data/test/dummy/README.rdoc +261 -0
- data/test/dummy/Rakefile +7 -0
- data/test/dummy/app/assets/javascripts/application.js +15 -0
- data/test/dummy/app/assets/stylesheets/application.css +13 -0
- data/test/dummy/app/controllers/application_controller.rb +3 -0
- data/test/dummy/app/controllers/simulate_controller.rb +10 -0
- data/test/dummy/app/helpers/application_helper.rb +2 -0
- data/test/dummy/app/helpers/simulate_helper.rb +2 -0
- data/test/dummy/app/views/layouts/application.html.erb +14 -0
- data/test/dummy/app/views/simulate/failure.html.erb +2 -0
- data/test/dummy/config/application.rb +50 -0
- data/test/dummy/config/boot.rb +10 -0
- data/test/dummy/config/database.yml +25 -0
- data/test/dummy/config/environment.rb +5 -0
- data/test/dummy/config/environments/development.rb +29 -0
- data/test/dummy/config/environments/production.rb +63 -0
- data/test/dummy/config/environments/test.rb +33 -0
- data/test/dummy/config/initializers/backtrace_silencers.rb +7 -0
- data/test/dummy/config/initializers/inflections.rb +15 -0
- data/test/dummy/config/initializers/mime_types.rb +5 -0
- data/test/dummy/config/initializers/secret_token.rb +7 -0
- data/test/dummy/config/initializers/session_store.rb +8 -0
- data/test/dummy/config/initializers/wrap_parameters.rb +14 -0
- data/test/dummy/config/locales/en.yml +5 -0
- data/test/dummy/config/routes.rb +6 -0
- data/test/dummy/config.ru +4 -0
- data/test/dummy/db/migrate/20120507083836_create_exception_logger_logged_exceptions.exception_logger.rb +15 -0
- data/test/dummy/db/schema.rb +27 -0
- data/test/dummy/public/404.html +26 -0
- data/test/dummy/public/422.html +26 -0
- data/test/dummy/public/500.html +25 -0
- data/test/dummy/public/favicon.ico +0 -0
- data/test/dummy/script/rails +6 -0
- data/test/dummy/test/functional/simulate_controller_test.rb +9 -0
- data/test/dummy/test/unit/helpers/simulate_helper_test.rb +4 -0
- data/test/exception_logger_test.rb +7 -0
- data/test/fixtures/exception_logger/logged_exceptions.yml +11 -0
- data/test/functional/exception_logger/logged_exceptions_controller_test.rb +9 -0
- data/test/integration/navigation_test.rb +10 -0
- data/test/test_helper.rb +10 -0
- data/test/unit/exception_logger/logged_exception_test.rb +9 -0
- data/test/unit/helpers/exception_logger/logged_exceptions_helper_test.rb +6 -0
- metadata +204 -0
| @@ -0,0 +1,105 @@ | |
| 1 | 
            +
            module RailsExceptionLogger
         | 
| 2 | 
            +
              class LoggedExceptionsController < ApplicationController
         | 
| 3 | 
            +
                layout 'rails_exception_logger/application'
         | 
| 4 | 
            +
             | 
| 5 | 
            +
                cattr_accessor :application_name
         | 
| 6 | 
            +
             | 
| 7 | 
            +
                helper_method :params_filters
         | 
| 8 | 
            +
             | 
| 9 | 
            +
                #ApplicationController.class_eval do
         | 
| 10 | 
            +
                #  rescue_from Exception, :with => :log_exception_handler
         | 
| 11 | 
            +
                #end
         | 
| 12 | 
            +
             | 
| 13 | 
            +
                def index
         | 
| 14 | 
            +
                  @exception_names    = LoggedException.class_names
         | 
| 15 | 
            +
                  @controller_actions = LoggedException.controller_actions
         | 
| 16 | 
            +
                  query
         | 
| 17 | 
            +
                end
         | 
| 18 | 
            +
             | 
| 19 | 
            +
                def query
         | 
| 20 | 
            +
                  exceptions = LoggedException.sorted
         | 
| 21 | 
            +
                  unless params[:id].blank?
         | 
| 22 | 
            +
                    exceptions = exceptions.where(:id => params[:id])
         | 
| 23 | 
            +
                  end
         | 
| 24 | 
            +
                  unless params[:query].blank?
         | 
| 25 | 
            +
                    exceptions = exceptions.message_like(params[:query])
         | 
| 26 | 
            +
                  end
         | 
| 27 | 
            +
                  unless params[:date_ranges_filter].blank?
         | 
| 28 | 
            +
                    exceptions = exceptions.days_old(params[:date_ranges_filter])
         | 
| 29 | 
            +
                  end
         | 
| 30 | 
            +
                  unless params[:exception_names_filter].blank?
         | 
| 31 | 
            +
                    exceptions = exceptions.by_exception_class(params[:exception_names_filter])
         | 
| 32 | 
            +
                  end
         | 
| 33 | 
            +
                  unless params[:controller_actions_filter].blank?
         | 
| 34 | 
            +
                    c_a_params = params[:controller_actions_filter].split('/')
         | 
| 35 | 
            +
                    controller_filter = c_a_params.first.underscore
         | 
| 36 | 
            +
                    action_filter = c_a_params.last.downcase
         | 
| 37 | 
            +
                    exceptions = exceptions.by_controller(controller_filter)
         | 
| 38 | 
            +
                    exceptions = exceptions.by_action(action_filter)
         | 
| 39 | 
            +
                  end
         | 
| 40 | 
            +
                  @exceptions = exceptions.paginate(:page => params[:page], :per_page => 30)
         | 
| 41 | 
            +
             | 
| 42 | 
            +
                  respond_to do |format|
         | 
| 43 | 
            +
                    format.html { redirect_to :action => 'index' unless action_name == 'index' }
         | 
| 44 | 
            +
                    format.js
         | 
| 45 | 
            +
                  end
         | 
| 46 | 
            +
                end
         | 
| 47 | 
            +
             | 
| 48 | 
            +
                def feed
         | 
| 49 | 
            +
                  @exceptions = LoggedException.all
         | 
| 50 | 
            +
             | 
| 51 | 
            +
                  respond_to do |format|
         | 
| 52 | 
            +
                    format.rss { render :layout => false }
         | 
| 53 | 
            +
                  end
         | 
| 54 | 
            +
                end
         | 
| 55 | 
            +
             | 
| 56 | 
            +
                def show
         | 
| 57 | 
            +
                  @exception = LoggedException.where(:id => params[:id]).first
         | 
| 58 | 
            +
             | 
| 59 | 
            +
                  respond_to do |format|
         | 
| 60 | 
            +
                    format.js
         | 
| 61 | 
            +
                    format.html
         | 
| 62 | 
            +
                  end
         | 
| 63 | 
            +
                end
         | 
| 64 | 
            +
             | 
| 65 | 
            +
                def destroy
         | 
| 66 | 
            +
                  @exception = LoggedException.where(:id => params[:id]).first
         | 
| 67 | 
            +
                  @exception.destroy
         | 
| 68 | 
            +
                end
         | 
| 69 | 
            +
             | 
| 70 | 
            +
                def destroy_all
         | 
| 71 | 
            +
                  LoggedException.delete_all(:id => params[:ids]) unless params[:ids].blank?
         | 
| 72 | 
            +
                  query
         | 
| 73 | 
            +
                end
         | 
| 74 | 
            +
             | 
| 75 | 
            +
                def clear
         | 
| 76 | 
            +
                  LoggedException.delete_all
         | 
| 77 | 
            +
                  redirect_to :back
         | 
| 78 | 
            +
                end
         | 
| 79 | 
            +
             | 
| 80 | 
            +
                private
         | 
| 81 | 
            +
             | 
| 82 | 
            +
                def params_filters
         | 
| 83 | 
            +
                  {
         | 
| 84 | 
            +
                    :query => params[:query],
         | 
| 85 | 
            +
                    :date_ranges_filter => params[:date_ranges_filter],
         | 
| 86 | 
            +
                    :exception_names_filter => params[:exception_names_filter],
         | 
| 87 | 
            +
                    :controller_actions_filter => params[:controller_actions_filter],
         | 
| 88 | 
            +
                  }
         | 
| 89 | 
            +
                end
         | 
| 90 | 
            +
             | 
| 91 | 
            +
                def access_denied_with_basic_auth
         | 
| 92 | 
            +
                  headers["Status"]           = "Unauthorized"
         | 
| 93 | 
            +
                  headers["WWW-Authenticate"] = %(Basic realm="Web Password")
         | 
| 94 | 
            +
                  render :text => "Could't authenticate you", :status => '401 Unauthorized'
         | 
| 95 | 
            +
                end
         | 
| 96 | 
            +
             | 
| 97 | 
            +
                @@http_auth_headers = %w(X-HTTP_AUTHORIZATION HTTP_AUTHORIZATION Authorization)
         | 
| 98 | 
            +
                # gets BASIC auth info
         | 
| 99 | 
            +
                def get_auth_data
         | 
| 100 | 
            +
                  auth_key  = @@http_auth_headers.detect { |h| request.env.has_key?(h) }
         | 
| 101 | 
            +
                  auth_data = request.env[auth_key].to_s.split unless auth_key.blank?
         | 
| 102 | 
            +
                  return auth_data && auth_data[0] == 'Basic' ? Base64.decode64(auth_data[1]).split(':')[0..1] : [nil, nil]
         | 
| 103 | 
            +
                end
         | 
| 104 | 
            +
              end
         | 
| 105 | 
            +
            end
         | 
| @@ -0,0 +1,42 @@ | |
| 1 | 
            +
            module RailsExceptionLogger
         | 
| 2 | 
            +
              module LoggedExceptionsHelper
         | 
| 3 | 
            +
                def pretty_exception_date(exception)
         | 
| 4 | 
            +
                  if Date.today == exception.created_at.to_date
         | 
| 5 | 
            +
                    if false # exception.created_at > Time.now - 4.hours
         | 
| 6 | 
            +
                      "#{time_ago_in_words(exception.created_at).gsub(/about /,"~ ")} agox"
         | 
| 7 | 
            +
                    else
         | 
| 8 | 
            +
                      "Today, #{exception.created_at.strftime(Time::DATE_FORMATS[:exc_time])}"
         | 
| 9 | 
            +
                    end
         | 
| 10 | 
            +
                  else
         | 
| 11 | 
            +
                    exception.created_at.strftime(Time::DATE_FORMATS[:exc_date])
         | 
| 12 | 
            +
                  end
         | 
| 13 | 
            +
                end
         | 
| 14 | 
            +
             | 
| 15 | 
            +
                def filtered?
         | 
| 16 | 
            +
                  [:query, :date_ranges_filter, :exception_names_filter, :controller_actions_filter].any? { |p| params[p] }
         | 
| 17 | 
            +
                end
         | 
| 18 | 
            +
             | 
| 19 | 
            +
                def listify(text)
         | 
| 20 | 
            +
                  list_items = text.scan(/^\s*\* (.+)/).map {|match| content_tag(:li, match.first) }
         | 
| 21 | 
            +
                  content_tag(:ul, list_items)
         | 
| 22 | 
            +
                end
         | 
| 23 | 
            +
             | 
| 24 | 
            +
                def page_title(text)
         | 
| 25 | 
            +
                  title = ""
         | 
| 26 | 
            +
                  unless controller.application_name.blank?
         | 
| 27 | 
            +
                    title << "#{controller.application_name} :: "
         | 
| 28 | 
            +
                  end
         | 
| 29 | 
            +
                  title << text.to_s
         | 
| 30 | 
            +
                  content_for(:title, title.to_s)
         | 
| 31 | 
            +
                end
         | 
| 32 | 
            +
             | 
| 33 | 
            +
                # Rescue textilize call if RedCloth is not available.
         | 
| 34 | 
            +
                def pretty_format(text)
         | 
| 35 | 
            +
                  begin
         | 
| 36 | 
            +
                    textilize(text).html_safe
         | 
| 37 | 
            +
                  rescue
         | 
| 38 | 
            +
                    simple_format(text).html_safe
         | 
| 39 | 
            +
                  end
         | 
| 40 | 
            +
                end
         | 
| 41 | 
            +
              end
         | 
| 42 | 
            +
            end
         | 
| @@ -0,0 +1,83 @@ | |
| 1 | 
            +
            module RailsExceptionLogger
         | 
| 2 | 
            +
              class LoggedException < ActiveRecord::Base
         | 
| 3 | 
            +
                self.table_name = "logged_exceptions"
         | 
| 4 | 
            +
                HOSTNAME = `hostname -s`.chomp
         | 
| 5 | 
            +
                class << self
         | 
| 6 | 
            +
                  def create_from_exception(controller, exception, data)
         | 
| 7 | 
            +
                    message = exception.message.inspect
         | 
| 8 | 
            +
                    message << "\n* Extra Data\n\n#{data}" unless data.blank?
         | 
| 9 | 
            +
                    e = create! \
         | 
| 10 | 
            +
                      :exception_class => exception.class.name,
         | 
| 11 | 
            +
                      :controller_name => controller.controller_path,
         | 
| 12 | 
            +
                      :action_name     => controller.action_name,
         | 
| 13 | 
            +
                      :message         => message,
         | 
| 14 | 
            +
                      :backtrace       => exception.backtrace,
         | 
| 15 | 
            +
                      :request         => controller.request
         | 
| 16 | 
            +
                  end
         | 
| 17 | 
            +
             | 
| 18 | 
            +
                  def host_name
         | 
| 19 | 
            +
                    HOSTNAME
         | 
| 20 | 
            +
                  end
         | 
| 21 | 
            +
                end
         | 
| 22 | 
            +
             | 
| 23 | 
            +
                scope :by_exception_class, lambda {|exception_class| where(:exception_class => exception_class)}
         | 
| 24 | 
            +
                scope :by_controller_and_action, lambda {|controller_name, action_name| where(:controller_name => controller_name, :action_name => action_name)}
         | 
| 25 | 
            +
                scope :by_controller, lambda {|controller_name| where(:controller_name => controller_name)}
         | 
| 26 | 
            +
                scope :by_action, lambda {|action_name| where(:action_name => action_name)}
         | 
| 27 | 
            +
                scope :message_like, lambda {|query|  where('message like ?', "%#{query}%")}
         | 
| 28 | 
            +
                scope :days_old, lambda {|day_number| where('created_at >= ?', day_number.to_f.days.ago.utc)}
         | 
| 29 | 
            +
                scope :sorted, lambda { order('created_at DESC') }
         | 
| 30 | 
            +
             | 
| 31 | 
            +
                def name
         | 
| 32 | 
            +
                  "#{self.exception_class} in #{self.controller_action}"
         | 
| 33 | 
            +
                end
         | 
| 34 | 
            +
             | 
| 35 | 
            +
                def backtrace=(trace)
         | 
| 36 | 
            +
                  trace = sanitize_backtrace(trace) * "\n" unless trace.is_a?(String)
         | 
| 37 | 
            +
                  write_attribute :backtrace, trace
         | 
| 38 | 
            +
                end
         | 
| 39 | 
            +
             | 
| 40 | 
            +
                def request=(request)
         | 
| 41 | 
            +
                  if request.is_a?(String)
         | 
| 42 | 
            +
                    write_attribute :request, request
         | 
| 43 | 
            +
                  else
         | 
| 44 | 
            +
                    max = request.env.keys.max { |a,b| a.length <=> b.length }
         | 
| 45 | 
            +
                    env = request.env.keys.sort.inject [] do |env, key|
         | 
| 46 | 
            +
                      env << '* ' + ("%-*s: %s" % [max.length, key, request.env[key].to_s.strip])
         | 
| 47 | 
            +
                    end
         | 
| 48 | 
            +
                    write_attribute(:environment, (env << "* Process: #{$$}" << "* Server : #{self.class.host_name}") * "\n")
         | 
| 49 | 
            +
             | 
| 50 | 
            +
                    write_attribute(:request, [
         | 
| 51 | 
            +
                        "* URL:#{" #{request.method.to_s.upcase}" unless request.get?} #{request.protocol}#{request.env["HTTP_HOST"]}#{request.fullpath}",
         | 
| 52 | 
            +
                        "* Format: #{request.format.to_s}",
         | 
| 53 | 
            +
                        "* Parameters: #{request.parameters.inspect}",
         | 
| 54 | 
            +
                        "* Rails Root: #{rails_root}"
         | 
| 55 | 
            +
                      ] * "\n")
         | 
| 56 | 
            +
                  end
         | 
| 57 | 
            +
                end
         | 
| 58 | 
            +
             | 
| 59 | 
            +
                def controller_action
         | 
| 60 | 
            +
                  @controller_action ||= "#{controller_name.camelcase}/#{action_name}"
         | 
| 61 | 
            +
                end
         | 
| 62 | 
            +
             | 
| 63 | 
            +
                def self.class_names
         | 
| 64 | 
            +
                  select("DISTINCT exception_class").order(:exception_class).collect(&:exception_class)
         | 
| 65 | 
            +
                end
         | 
| 66 | 
            +
             | 
| 67 | 
            +
                def self.controller_actions
         | 
| 68 | 
            +
                  select("DISTINCT controller_name, action_name").order(:controller_name,:action_name).collect(&:controller_action)
         | 
| 69 | 
            +
                end
         | 
| 70 | 
            +
             | 
| 71 | 
            +
                private
         | 
| 72 | 
            +
                @@rails_root      = Pathname.new(Rails.root).cleanpath.to_s
         | 
| 73 | 
            +
                @@backtrace_regex = /^#{Regexp.escape(@@rails_root)}/
         | 
| 74 | 
            +
             | 
| 75 | 
            +
                def sanitize_backtrace(trace)
         | 
| 76 | 
            +
                  trace.collect { |line| Pathname.new(line.gsub(@@backtrace_regex, "[RAILS_ROOT]")).cleanpath.to_s }
         | 
| 77 | 
            +
                end
         | 
| 78 | 
            +
             | 
| 79 | 
            +
                def rails_root
         | 
| 80 | 
            +
                  @@rails_root
         | 
| 81 | 
            +
                end
         | 
| 82 | 
            +
              end
         | 
| 83 | 
            +
            end
         | 
| @@ -0,0 +1,20 @@ | |
| 1 | 
            +
            <!DOCTYPE html>
         | 
| 2 | 
            +
            <html>
         | 
| 3 | 
            +
              <head>
         | 
| 4 | 
            +
                <meta charset="UTF-8" />
         | 
| 5 | 
            +
                <%= csrf_meta_tag %>
         | 
| 6 | 
            +
                <title><%= yield(:title) || "Exception Logger" %></title>
         | 
| 7 | 
            +
             | 
| 8 | 
            +
                <%= stylesheet_link_tag "rails_exception_logger/application" %>
         | 
| 9 | 
            +
             | 
| 10 | 
            +
                <%= auto_discovery_link_tag(:rss, {:action => "feed"}, {:title => "RSS Feed"}) %>
         | 
| 11 | 
            +
              </head>
         | 
| 12 | 
            +
              <body>
         | 
| 13 | 
            +
                <div id="container">
         | 
| 14 | 
            +
                  <%= yield %>
         | 
| 15 | 
            +
                </div>
         | 
| 16 | 
            +
                <br style="clear:both" />
         | 
| 17 | 
            +
             | 
| 18 | 
            +
                <%= javascript_include_tag "rails_exception_logger/application" %>
         | 
| 19 | 
            +
              </body>
         | 
| 20 | 
            +
            </html>
         | 
| @@ -0,0 +1,32 @@ | |
| 1 | 
            +
            <div id="exceptions">
         | 
| 2 | 
            +
              <div class="pages">
         | 
| 3 | 
            +
                <%= link_to('Clear History', {:action => :clear}, :confirm => "are you sure?", :method => :post) %>
         | 
| 4 | 
            +
                <%= link_to 'Delete Visible', { :action => 'destroy_all' }.merge(params_filters), :class => 'delete_visible_link' %>
         | 
| 5 | 
            +
                <strong><%= will_paginate @exceptions, :params => { :controller => "logged_exceptions", :action => "index" }.merge(params_filters), :style => 'display:inline' %></strong>
         | 
| 6 | 
            +
              </div>
         | 
| 7 | 
            +
              <h1>Exceptions <%= raw "<span>(filtered)</span>" if filtered? %> </h1>
         | 
| 8 | 
            +
              <table>
         | 
| 9 | 
            +
                <thead>
         | 
| 10 | 
            +
                  <tr>
         | 
| 11 | 
            +
                    <th>Exception</th>
         | 
| 12 | 
            +
                    <th>Date</th>
         | 
| 13 | 
            +
                    <th></th>
         | 
| 14 | 
            +
                  </tr>
         | 
| 15 | 
            +
                </thead>
         | 
| 16 | 
            +
                <tbody>
         | 
| 17 | 
            +
                  <% @exceptions.each do |exception| %>
         | 
| 18 | 
            +
                    <tr id="<%= dom_id(exception) %>" class="<%= cycle("eor", "") %> exception">
         | 
| 19 | 
            +
                      <td>
         | 
| 20 | 
            +
                        <div class="expclass"><%= link_to exception.name, {:action => "show", :id => exception}, :class => 'show_link' %></div>
         | 
| 21 | 
            +
                        <span class="message"><%= h exception.message %></span>
         | 
| 22 | 
            +
                      </td>
         | 
| 23 | 
            +
                      <td class="time"><%= pretty_exception_date(exception) %></td>
         | 
| 24 | 
            +
                      <td><%= link_to 'Delete', { :action => 'destroy', :id => exception }, :class => "util delete_link" %></td>
         | 
| 25 | 
            +
                    </tr>
         | 
| 26 | 
            +
                  <% end %>
         | 
| 27 | 
            +
                </tbody>
         | 
| 28 | 
            +
              </table>
         | 
| 29 | 
            +
               <div class="pages pages-bottom">
         | 
| 30 | 
            +
                 <strong><%= will_paginate @exceptions, :params => { :controller => "logged_exceptions", :action => "index" }.merge(params_filters), :style => 'display:inline' %></strong>
         | 
| 31 | 
            +
               </div>
         | 
| 32 | 
            +
            </div> <!-- #exceptions -->
         | 
| @@ -0,0 +1,23 @@ | |
| 1 | 
            +
            <div class="tools">
         | 
| 2 | 
            +
              <%= link_to 'Delete', { :action => 'destroy', :id => @exception }, :class => "util delete_link" %>
         | 
| 3 | 
            +
              <span class="pipe">|</span>
         | 
| 4 | 
            +
              <%= link_to "Close", "#", :class => "util close_link" %>
         | 
| 5 | 
            +
            </div>
         | 
| 6 | 
            +
             | 
| 7 | 
            +
            <div class="date">
         | 
| 8 | 
            +
              <%= @exception.created_at.strftime(Time::DATE_FORMATS[:exc_full]) %>
         | 
| 9 | 
            +
            </div>
         | 
| 10 | 
            +
            <h1><%= @exception.name %></h1>
         | 
| 11 | 
            +
             | 
| 12 | 
            +
            <h2>Request</h2>
         | 
| 13 | 
            +
            <%= pretty_format(@exception.request) %>
         | 
| 14 | 
            +
             | 
| 15 | 
            +
            <h2>Backtrace</h2>
         | 
| 16 | 
            +
            <%= simple_format @exception.message %>
         | 
| 17 | 
            +
             | 
| 18 | 
            +
            <div id="backtrace">
         | 
| 19 | 
            +
              <%= raw @exception.backtrace.gsub(/\n/,"<br />") %>
         | 
| 20 | 
            +
            </div>
         | 
| 21 | 
            +
             | 
| 22 | 
            +
            <h2>Environment</h2>
         | 
| 23 | 
            +
            <%= pretty_format(@exception.environment) %>
         | 
| @@ -0,0 +1,20 @@ | |
| 1 | 
            +
            xml.instruct! :xml, :version=>"1.0", :encoding=>"UTF-8"
         | 
| 2 | 
            +
             | 
| 3 | 
            +
            xml.rss "version" => "2.0" do
         | 
| 4 | 
            +
              xml.channel do
         | 
| 5 | 
            +
                xml.title "#{RailsExceptionLogger::LoggedExceptionsController.application_name}"
         | 
| 6 | 
            +
                xml.link url_for(:only_path => false, :skip_relative_url_root => false)
         | 
| 7 | 
            +
                xml.language "en-us"
         | 
| 8 | 
            +
                xml.ttl "60"
         | 
| 9 | 
            +
             | 
| 10 | 
            +
                @exceptions.each do |exc|
         | 
| 11 | 
            +
                  xml.item do
         | 
| 12 | 
            +
                    xml.title "#{exc.name} @ #{exc.created_at.rfc822}"
         | 
| 13 | 
            +
                    xml.description exc.message
         | 
| 14 | 
            +
                    xml.pubDate exc.created_at.rfc822
         | 
| 15 | 
            +
                    xml.guid [request.host_with_port, 'exceptions', exc.id.to_s].join(":"), "isPermaLink" => "false"
         | 
| 16 | 
            +
                    xml.link url_for(:action => 'index', :anchor => "e#{exc.id}", :only_path => false, :skip_relative_url_root => false)
         | 
| 17 | 
            +
                  end
         | 
| 18 | 
            +
                end
         | 
| 19 | 
            +
              end
         | 
| 20 | 
            +
            end
         | 
| @@ -0,0 +1,46 @@ | |
| 1 | 
            +
            <% page_title t(".title") %>
         | 
| 2 | 
            +
            <div id="right">
         | 
| 3 | 
            +
              <h3><%=t(".filters")%></h3>
         | 
| 4 | 
            +
              <ul id="all_exceptions" class="filters">
         | 
| 5 | 
            +
                <li><%= link_to t(".latest_exceptions"), :action => 'index', :id => nil %></li>
         | 
| 6 | 
            +
              </ul>
         | 
| 7 | 
            +
              <h4><%=t(".exception")%></h4>
         | 
| 8 | 
            +
              <ul id="exception_names" class="filters">
         | 
| 9 | 
            +
                <% @exception_names.each do |name| %>
         | 
| 10 | 
            +
                  <li><%= link_to name, {:action => 'query', :exception_names_filter => "#{name}"}, :class => 'filter_link' %></li>
         | 
| 11 | 
            +
                <% end %>
         | 
| 12 | 
            +
              </ul>
         | 
| 13 | 
            +
              <h4><%=t(".controller_action")%></h4>
         | 
| 14 | 
            +
              <ul id="controller_actions" class="filters">
         | 
| 15 | 
            +
                <% @controller_actions.each do |action| %>
         | 
| 16 | 
            +
                  <li><%= link_to action, {:action => 'query', :controller_actions_filter => "#{action}"}, :class => 'filter_link' %></li>
         | 
| 17 | 
            +
                <% end %>
         | 
| 18 | 
            +
              </ul>
         | 
| 19 | 
            +
              <h4><%=t(".dates")%></h4>
         | 
| 20 | 
            +
              <ul id="date_ranges" class="filters">
         | 
| 21 | 
            +
                <li><%= link_to t(".today"), {:action => 'query', :date_ranges_filter => "1"}, :title => "1", :class => 'filter_link' %></li>
         | 
| 22 | 
            +
                <li><%= link_to t(".last_few_days"), {:action => 'query', :date_ranges_filter => "3"}, :title => "3", :class => 'filter_link' %></li>
         | 
| 23 | 
            +
                <li><%= link_to t(".last_7_days"), {:action => 'query', :date_ranges_filter => "7"}, :title => "7", :class => 'filter_link' %></li>
         | 
| 24 | 
            +
                <li><%= link_to t(".last_30_days"), {:action => 'query', :date_ranges_filter => "30"}, :title => "30", :class => 'filter_link' %></li>
         | 
| 25 | 
            +
              </ul>
         | 
| 26 | 
            +
              <div id="search">
         | 
| 27 | 
            +
                <%= form_tag "logged_exceptions/query", :id => 'query-form' %>
         | 
| 28 | 
            +
                <div>
         | 
| 29 | 
            +
                  <%= text_field_tag :query, "", :size => 17 %>
         | 
| 30 | 
            +
                  <%= submit_tag :Find %>
         | 
| 31 | 
            +
                  <%= hidden_field_tag :exception_names_filter %>
         | 
| 32 | 
            +
                  <%= hidden_field_tag :date_ranges_filter %>
         | 
| 33 | 
            +
                  <%= hidden_field_tag :controller_actions_filter %>
         | 
| 34 | 
            +
                  <%= hidden_field_tag :page, (params[:page] || 1) %>
         | 
| 35 | 
            +
                </div>
         | 
| 36 | 
            +
                </form>
         | 
| 37 | 
            +
              </div>
         | 
| 38 | 
            +
              <%= render :partial => 'feed' %>
         | 
| 39 | 
            +
            </div> <!-- right -->
         | 
| 40 | 
            +
            <div id="left">
         | 
| 41 | 
            +
              <div class="page" id="showpage" style="display:none;"></div>
         | 
| 42 | 
            +
              <div class="page">
         | 
| 43 | 
            +
                <%= render :partial => "exceptions" %>
         | 
| 44 | 
            +
              </div>
         | 
| 45 | 
            +
            </div>
         | 
| 46 | 
            +
            <div id="activity" style="display:none;">Busy...</div>
         | 
| @@ -0,0 +1,17 @@ | |
| 1 | 
            +
            en:
         | 
| 2 | 
            +
              rails_exception_logger:
         | 
| 3 | 
            +
                logged_exceptions:
         | 
| 4 | 
            +
                  index:
         | 
| 5 | 
            +
                    title: Logged Exceptions
         | 
| 6 | 
            +
                    filters: Filters
         | 
| 7 | 
            +
                    latest_exceptions: Latest Exceptions
         | 
| 8 | 
            +
                    exception: Exception
         | 
| 9 | 
            +
                    controller_action: Controller / Action
         | 
| 10 | 
            +
                    dates: Dates
         | 
| 11 | 
            +
                    today: Today
         | 
| 12 | 
            +
                    last_few_days: Last few days
         | 
| 13 | 
            +
                    last_7_days: Last 7 days
         | 
| 14 | 
            +
                    last_30_days: Last 30 days
         | 
| 15 | 
            +
                  show:
         | 
| 16 | 
            +
                    title: Logged Exception
         | 
| 17 | 
            +
             | 
    
        data/config/routes.rb
    ADDED
    
    
| @@ -0,0 +1,14 @@ | |
| 1 | 
            +
            class CreateRailsExceptionLoggerLoggedExceptions < ActiveRecord::Migration
         | 
| 2 | 
            +
              def change
         | 
| 3 | 
            +
                create_table :rails_logged_exceptions, :force => true do |t|
         | 
| 4 | 
            +
                  t.string :exception_class
         | 
| 5 | 
            +
                  t.string :controller_name
         | 
| 6 | 
            +
                  t.string :action_name
         | 
| 7 | 
            +
                  t.text :message
         | 
| 8 | 
            +
                  t.text :backtrace
         | 
| 9 | 
            +
                  t.text :environment
         | 
| 10 | 
            +
                  t.text :request
         | 
| 11 | 
            +
                  t.datetime :created_at
         | 
| 12 | 
            +
                end
         | 
| 13 | 
            +
              end
         | 
| 14 | 
            +
            end
         | 
| @@ -0,0 +1,68 @@ | |
| 1 | 
            +
            require "rails_exception_logger/engine"
         | 
| 2 | 
            +
            require "will_paginate"
         | 
| 3 | 
            +
            require 'ipaddr'
         | 
| 4 | 
            +
             | 
| 5 | 
            +
            module RailsExceptionLogger
         | 
| 6 | 
            +
              # Copyright (c) 2005 Jamis Buck
         | 
| 7 | 
            +
              #
         | 
| 8 | 
            +
              # Permission is hereby granted, free of charge, to any person obtaining
         | 
| 9 | 
            +
              # a copy of this software and associated documentation files (the
         | 
| 10 | 
            +
              # "Software"), to deal in the Software without restriction, including
         | 
| 11 | 
            +
              # without limitation the rights to use, copy, modify, merge, publish,
         | 
| 12 | 
            +
              # distribute, sublicense, and/or sell copies of the Software, and to
         | 
| 13 | 
            +
              # permit persons to whom the Software is furnished to do so, subject to
         | 
| 14 | 
            +
              # the following conditions:
         | 
| 15 | 
            +
              #
         | 
| 16 | 
            +
              # The above copyright notice and this permission notice shall be
         | 
| 17 | 
            +
              # included in all copies or substantial portions of the Software.
         | 
| 18 | 
            +
              #
         | 
| 19 | 
            +
              # THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
         | 
| 20 | 
            +
              # EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
         | 
| 21 | 
            +
              # MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
         | 
| 22 | 
            +
              # NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
         | 
| 23 | 
            +
              # LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
         | 
| 24 | 
            +
              # OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
         | 
| 25 | 
            +
              # WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
         | 
| 26 | 
            +
              module ExceptionLoggable
         | 
| 27 | 
            +
                def self.included(target)
         | 
| 28 | 
            +
                  target.extend(ClassMethods)
         | 
| 29 | 
            +
                  target.class_attribute :local_addresses, :exception_data
         | 
| 30 | 
            +
             | 
| 31 | 
            +
                  target.local_addresses = [IPAddr.new("127.0.0.1")]
         | 
| 32 | 
            +
                end
         | 
| 33 | 
            +
             | 
| 34 | 
            +
                module ClassMethods
         | 
| 35 | 
            +
                  def consider_local(*args)
         | 
| 36 | 
            +
                    local_addresses.concat(args.flatten.map { |a| IPAddr.new(a) })
         | 
| 37 | 
            +
                  end
         | 
| 38 | 
            +
                end
         | 
| 39 | 
            +
             | 
| 40 | 
            +
                def local_request?
         | 
| 41 | 
            +
                  remote = IPAddr.new(request.remote_ip)
         | 
| 42 | 
            +
                  !self.class.local_addresses.detect { |addr| addr.include?(remote) }.nil?
         | 
| 43 | 
            +
                end
         | 
| 44 | 
            +
             | 
| 45 | 
            +
                # we log the exception and raise it again, for the normal handling.
         | 
| 46 | 
            +
                def log_exception_handler(exception)
         | 
| 47 | 
            +
                  log_exception(exception)
         | 
| 48 | 
            +
                  raise exception
         | 
| 49 | 
            +
                end
         | 
| 50 | 
            +
             | 
| 51 | 
            +
                def rescue_action(exception)
         | 
| 52 | 
            +
                  status = response_code_for_rescue(exception)
         | 
| 53 | 
            +
                  log_exception(exception) if status != :not_found
         | 
| 54 | 
            +
                  super
         | 
| 55 | 
            +
                end
         | 
| 56 | 
            +
             | 
| 57 | 
            +
                def log_exception(exception)
         | 
| 58 | 
            +
                  deliverer = self.class.exception_data
         | 
| 59 | 
            +
                  data = case deliverer
         | 
| 60 | 
            +
                  when nil    then {}
         | 
| 61 | 
            +
                  when Symbol then send(deliverer)
         | 
| 62 | 
            +
                  when Proc   then deliverer.call(self)
         | 
| 63 | 
            +
                  end
         | 
| 64 | 
            +
             | 
| 65 | 
            +
                  LoggedException.create_from_exception(self, exception, data)
         | 
| 66 | 
            +
                end
         | 
| 67 | 
            +
              end
         | 
| 68 | 
            +
            end
         |