exception_notification_server 0.0.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.
Files changed (87) hide show
  1. checksums.yaml +7 -0
  2. data/.document +5 -0
  3. data/.gitignore +51 -0
  4. data/.ruby-gemset +1 -0
  5. data/.ruby-version +1 -0
  6. data/Gemfile +23 -0
  7. data/Gemfile.lock +162 -0
  8. data/LICENSE.txt +20 -0
  9. data/README.rdoc +19 -0
  10. data/Rakefile +59 -0
  11. data/VERSION +1 -0
  12. data/app/assets/images/exception_notification_server/.keep +0 -0
  13. data/app/assets/javascripts/exception_notification_server/application.js.coffee +8 -0
  14. data/app/assets/javascripts/exception_notification_server/flot/excanvas.js +1428 -0
  15. data/app/assets/javascripts/exception_notification_server/flot/jquery.flot.js +3168 -0
  16. data/app/assets/javascripts/exception_notification_server/flot/jquery.flot.resize.js +59 -0
  17. data/app/assets/javascripts/exception_notification_server/flot/jquery.flot.time.js +432 -0
  18. data/app/assets/javascripts/exception_notification_server/jquery.sparkline.js +3054 -0
  19. data/app/assets/javascripts/exception_notification_server/main.js.coffee +6 -0
  20. data/app/assets/javascripts/exception_notification_server/pages/notifications.js.coffee +15 -0
  21. data/app/assets/stylesheets/exception_notification_server/application.css.sass +4 -0
  22. data/app/assets/stylesheets/exception_notification_server/layout.css.sass +97 -0
  23. data/app/assets/stylesheets/exception_notification_server/notifications.css.sass +19 -0
  24. data/app/controllers/exception_notification_server/application_controller.rb +20 -0
  25. data/app/controllers/exception_notification_server/notifications_controller.rb +97 -0
  26. data/app/helpers/exception_notification_server/application_helper.rb +46 -0
  27. data/app/models/exception_notification_server/notification.rb +69 -0
  28. data/app/views/exception_notification_server/notifications/_notifications.html.haml +37 -0
  29. data/app/views/exception_notification_server/notifications/index.html.haml +7 -0
  30. data/app/views/exception_notification_server/notifications/index.js.haml +2 -0
  31. data/app/views/exception_notification_server/notifications/show.html.haml +76 -0
  32. data/app/views/layouts/exception_notification_server/application.html.haml +17 -0
  33. data/config/routes.rb +10 -0
  34. data/exception_notification_server.gemspec +162 -0
  35. data/lib/exception_notification_server/engine.rb +10 -0
  36. data/lib/exception_notification_server/version.rb +3 -0
  37. data/lib/exception_notification_server.rb +21 -0
  38. data/lib/generators/exception_notification_server/install_generator.rb +33 -0
  39. data/lib/generators/exception_notification_server/templates/exception_notification_server.rb +5 -0
  40. data/lib/generators/exception_notification_server/templates/migration.rb +26 -0
  41. data/lib/tasks/exception_notification_server_tasks.rake +4 -0
  42. data/readme.md +5 -0
  43. data/test/dummy/README.rdoc +28 -0
  44. data/test/dummy/Rakefile +6 -0
  45. data/test/dummy/app/assets/images/.keep +0 -0
  46. data/test/dummy/app/assets/javascripts/application.js +13 -0
  47. data/test/dummy/app/assets/stylesheets/application.css +15 -0
  48. data/test/dummy/app/controllers/application_controller.rb +5 -0
  49. data/test/dummy/app/controllers/concerns/.keep +0 -0
  50. data/test/dummy/app/helpers/application_helper.rb +2 -0
  51. data/test/dummy/app/mailers/.keep +0 -0
  52. data/test/dummy/app/models/.keep +0 -0
  53. data/test/dummy/app/models/concerns/.keep +0 -0
  54. data/test/dummy/app/views/layouts/application.html.erb +14 -0
  55. data/test/dummy/bin/bundle +3 -0
  56. data/test/dummy/bin/rails +4 -0
  57. data/test/dummy/bin/rake +4 -0
  58. data/test/dummy/config/application.rb +22 -0
  59. data/test/dummy/config/boot.rb +5 -0
  60. data/test/dummy/config/database.yml +25 -0
  61. data/test/dummy/config/environment.rb +5 -0
  62. data/test/dummy/config/environments/development.rb +37 -0
  63. data/test/dummy/config/environments/production.rb +82 -0
  64. data/test/dummy/config/environments/test.rb +39 -0
  65. data/test/dummy/config/initializers/assets.rb +8 -0
  66. data/test/dummy/config/initializers/backtrace_silencers.rb +7 -0
  67. data/test/dummy/config/initializers/cookies_serializer.rb +3 -0
  68. data/test/dummy/config/initializers/filter_parameter_logging.rb +4 -0
  69. data/test/dummy/config/initializers/inflections.rb +16 -0
  70. data/test/dummy/config/initializers/mime_types.rb +4 -0
  71. data/test/dummy/config/initializers/session_store.rb +3 -0
  72. data/test/dummy/config/initializers/wrap_parameters.rb +14 -0
  73. data/test/dummy/config/locales/en.yml +23 -0
  74. data/test/dummy/config/routes.rb +3 -0
  75. data/test/dummy/config/secrets.yml +22 -0
  76. data/test/dummy/config.ru +4 -0
  77. data/test/dummy/db/test.sqlite3 +0 -0
  78. data/test/dummy/lib/assets/.keep +0 -0
  79. data/test/dummy/log/.keep +0 -0
  80. data/test/dummy/public/404.html +67 -0
  81. data/test/dummy/public/422.html +67 -0
  82. data/test/dummy/public/500.html +66 -0
  83. data/test/dummy/public/favicon.ico +0 -0
  84. data/test/exception_notification_server_test.rb +7 -0
  85. data/test/integration/navigation_test.rb +9 -0
  86. data/test/test_helper.rb +15 -0
  87. metadata +357 -0
@@ -0,0 +1,6 @@
1
+ @initialize_sparkline = ->
2
+ $('.sparkline').each ->
3
+ $(this).sparkline $(this).data('values'),
4
+ {type: $(this).data('type') || 'line'}
5
+ $ ->
6
+ initialize_sparkline()
@@ -0,0 +1,15 @@
1
+ $ ->
2
+ $('body.notifications_show').each ->
3
+ options =
4
+ series:
5
+ bars:
6
+ show: true
7
+ barWidth: 86400000
8
+ align: 'center'
9
+ yaxes:
10
+ min: 0
11
+ xaxis:
12
+ mode: 'time',
13
+ timeformat: "%m/%d/%y",
14
+ tickSize: [7, "day"],
15
+ $('.graph').plot([$('.graph').data('values')], options)
@@ -0,0 +1,4 @@
1
+ /*
2
+ *= require_tree
3
+ *= require_self
4
+ */
@@ -0,0 +1,97 @@
1
+ body
2
+ background: #f3f3f3 url()
3
+ header
4
+ background-color: #f8f8f8
5
+ border-color: #e7e7e7
6
+ top: 0
7
+ z-index: 1000
8
+ position: fixed
9
+ right: 0
10
+ left: 0
11
+ border-width: 0 0 1px
12
+ min-height: 50px
13
+ max-height: 50px
14
+ .logo
15
+ display: block
16
+ float: left
17
+ width: 168px
18
+ margin: 0px 10px
19
+ font-size: 18px
20
+ font-family: Armata
21
+ font-weight: 400
22
+ line-height: 50px
23
+ text-decoration: none
24
+ color: #b1003e
25
+ &:focus, &:visited
26
+ color: #b1003e
27
+ &:hover
28
+ color: #5e5e5e
29
+ @media (max-width: 800px)
30
+ margin: 0px auto
31
+ float: none
32
+ & > div.content
33
+ margin: 60px auto 0px
34
+ width: 1280px
35
+ @media (max-width: 1280px)
36
+ width: 100%
37
+ a
38
+ color: #b1003e
39
+ &:hover, &:focus
40
+ color: #2a6496
41
+ &:visited
42
+ color: #4b001a
43
+ table
44
+ width: 100%
45
+ background-color: #fff
46
+ border: 1px solid #ddd
47
+ border-collapse: collapse
48
+ border-spacing: 0
49
+ tr
50
+ th, td
51
+ padding: 8px
52
+ border: 1px solid #ddd
53
+ line-height: 1.428571429
54
+ text-align: left
55
+ th
56
+ border-bottom: 2px solid #ddd
57
+ td
58
+ background-color: #f9f9f9
59
+ vertical-align: top
60
+ border-top: 1px solid #ddd
61
+ &:hover td
62
+ background-color: #f5f5f5
63
+ .pagination
64
+ margin-top: 10px
65
+ text-align: center
66
+ float: left
67
+ width: 100%
68
+ ul.pagination
69
+ float: left
70
+ margin: 0px
71
+ padding: 0px
72
+ text-align: center
73
+ li
74
+ float: left
75
+ list-style: none
76
+ margin: 0px 4px
77
+ a
78
+ text-decoration: none
79
+ &.disabled a
80
+ cursor: default
81
+ &.active a
82
+ cursor: default
83
+ text-decoration: underline
84
+ pre
85
+ word-wrap: break-word
86
+ @media (max-width: 300px)
87
+ .hide-xs
88
+ display: none
89
+ @media (max-width: 500px)
90
+ .hide-sm
91
+ display: none
92
+ @media (max-width: 800px)
93
+ .hide-md
94
+ display: none
95
+ @media (max-width: 1100px)
96
+ .hide-lg
97
+ display: none
@@ -0,0 +1,19 @@
1
+ .notifications_controller
2
+ .filter
3
+ float: left
4
+ width: 100%
5
+ margin-bottom: 10px
6
+ select
7
+ float: left
8
+ width: 30%
9
+ margin-right: 3%
10
+ &:last-child
11
+ float: right
12
+ margin-right: 0
13
+ .inline-value
14
+ font-weight: normal
15
+ font-size: 16px
16
+ display: inline
17
+ .graph
18
+ width: 100%
19
+ height: 200px
@@ -0,0 +1,20 @@
1
+ module ExceptionNotificationServer
2
+ class ApplicationController < ActionController::Base
3
+ before_action :redirect_to_root, unless: :admin?
4
+
5
+ add_flash_types :error, :success, :info
6
+
7
+ def redirect_to_root
8
+ respond_to do |format|
9
+ format.js { render js: "window.location = '#{Rails.application.routes.url_helpers.root_url}';" }
10
+ format.all { redirect_to Rails.application.routes.url_helpers.root_url }
11
+ end
12
+ end
13
+
14
+ protected
15
+
16
+ def admin?
17
+ ExceptionNotificationServer.configuration.access_callback.present? ? ExceptionNotificationServer.configuration.access_callback.try(:call, self) : true
18
+ end
19
+ end
20
+ end
@@ -0,0 +1,97 @@
1
+ module ExceptionNotificationServer
2
+ class NotificationsController < ExceptionNotificationServer::ApplicationController
3
+ http_basic_authenticate_with name: ExceptionNotificationServer.configuration.name,
4
+ password: ExceptionNotificationServer.configuration.password, only: :create
5
+ respond_to :html, :json, :js
6
+
7
+ before_filter :load_notification, only: [:show, :update, :investigate, :fix, :renew]
8
+ skip_before_filter :redirect_to_root, only: [:create]
9
+
10
+ def index
11
+ params[:env] ||= Rails.env
12
+ params[:status] ||= :new
13
+ @notifications = Notification.base_notifications(params[:status])
14
+ .application(params[:application])
15
+ .env(params[:env])
16
+ .includes(:childrens)
17
+ .joins('LEFT JOIN "exception_notification_server_notifications" "ensn" on "ensn"."parent_id" = "exception_notification_server_notifications"."id"')
18
+ .group('"exception_notification_server_notifications"."id"')
19
+ .order('count(ensn.id) DESC')
20
+ .paginate(page: params[:page], per_page: 10)
21
+ # .where('last month')
22
+ respond_with @notifications
23
+ end
24
+
25
+ def create
26
+ @notification = Notification.create(notification_params)
27
+ respond_with @notification
28
+ rescue
29
+ render nothing: true
30
+ end
31
+
32
+ def show
33
+ require 'coderay'
34
+ respond_with @notification
35
+ end
36
+
37
+ def update
38
+ @notification.send("update#{'_recursive' if params[:recursive]}", notification_params_update)
39
+ respond_with @notification
40
+ end
41
+
42
+ def investigate
43
+ params[:recursive] = true
44
+ params[:notification] ||= {}
45
+ params[:notification][:status] = :investigating
46
+ update
47
+ end
48
+
49
+ def fix
50
+ params[:recursive] = true
51
+ params[:notification] ||= {}
52
+ params[:notification][:status] = :fixed
53
+ update
54
+ end
55
+
56
+ def renew
57
+ params[:recursive] = true
58
+ params[:notification] ||= {}
59
+ params[:notification][:status] = :new
60
+ update
61
+ end
62
+
63
+ protected
64
+
65
+ def load_notification
66
+ @notification = Notification.find(params[:id])
67
+ rescue ActiveRecord::RecordNotFound
68
+ flash[:alert] = "Can't find notification with id #{params[:id]}."
69
+ respond_to do |format|
70
+ format.js { render js: "window.location = '#{notifications_url}';" }
71
+ format.html { redirect_to notifications_path }
72
+ end
73
+ end
74
+
75
+ def notification_params
76
+ {
77
+ application: params[:application],
78
+ env: params[:env],
79
+ status: params[:status],
80
+ server: params[:server],
81
+ process: params[:process],
82
+ rails_root: params[:rails_root],
83
+ exception_class: params[:exception][:error_class],
84
+ exception_message: params[:exception][:message],
85
+ backtrace: params[:exception][:backtrace],
86
+ data: params[:data],
87
+ request: params[:request],
88
+ session: params[:session],
89
+ environment: params[:environment]
90
+ }.delete_if { |_, value| value.blank? }
91
+ end
92
+
93
+ def notification_params_update
94
+ params.require(:notification).permit!
95
+ end
96
+ end
97
+ end
@@ -0,0 +1,46 @@
1
+ module ExceptionNotificationServer
2
+ module ApplicationHelper
3
+ def pretty_json(data, lines = nil)
4
+ json = JSON.pretty_generate(data).html_safe
5
+ json_colored = CodeRay.scan(json, :json).span(css: :style, line_numbers: :inline, line_number_anchors: false)
6
+ return "<pre>\n#{json_colored}</pre>".html_safe if lines.nil? || json.count("\n") < lines
7
+ "<pre>\n#{json_colored.match(/([^\n]*\n){#{lines}}/m)}</span></pre>".html_safe
8
+ rescue
9
+ data
10
+ end
11
+
12
+ def status_options(options)
13
+ options_for_select(Notification::STATUSES.map { |status| [status.to_s.humanize, status] }, options[:selected])
14
+ end
15
+
16
+ def application_options(options)
17
+ options_for_select(Notification.group(:application).pluck(:application).map { |application| [application.humanize, application] }, options[:selected])
18
+ end
19
+
20
+ def environment_options(options)
21
+ default_environment = %w(production staging development test)
22
+ options_for_select((default_environment + Notification.group(:env).pluck(:env)).uniq.map { |env| [env.humanize, env] }, options[:selected])
23
+ end
24
+
25
+ def time_format
26
+ ExceptionNotificationServer.configuration.time_format
27
+ end
28
+
29
+ def logo_string
30
+ <<-Text
31
+ ______ _ _ _ _ _ _ __ _ _ _ _____
32
+ | ____| | | (_) | \\ | | | | (_)/ _(_) | | (_) / ____|
33
+ | |__ __ _____ ___ _ __ | |_ _ ___ _ __ | \\| | ___ | |_ _| |_ _ ___ __ _| |_ _ ___ _ __ | (___ ___ _ ____ _____ _ __
34
+ | __| \\ \\/ / __/ _ \\ \'_ \\| __| |/ _ \\| \'_ \\| . ` |/ _ \\| __| | _| |/ __/ _\` | __| |/ _ \\| \'_ \\ \\___ \\ / _ \\ \'__\\ \\ / / _ \\ \'__|
35
+ | |____ > < (_| __/ |_) | |_| | (_) | | | | |\\ | (_) | |_| | | | | (_| (_| | |_| | (_) | | | |____) | __/ | \\ V / __/ |
36
+ |______/_/\\_\\___\\___| .__/ \\__|_|\\___/|_| |_|_| \\_|\\___/ \\__|_|_| |_|\\___\\__,_|\\__|_|\\___/|_| |_|_____/ \\___|_| \\_/ \\___|_|
37
+ | |
38
+ |_|
39
+ Text
40
+ end
41
+
42
+ def author
43
+ 'Author: Anatoliy Varanitsa'
44
+ end
45
+ end
46
+ end
@@ -0,0 +1,69 @@
1
+ module ExceptionNotificationServer
2
+ class Notification < ActiveRecord::Base
3
+ serialize :backtrace, Array
4
+ serialize :data, Hash
5
+ serialize :request, Hash
6
+ serialize :session, Hash
7
+ serialize :environment, Hash
8
+
9
+ belongs_to :parent, class: ExceptionNotificationServer::Notification
10
+ has_many :childrens, class: ExceptionNotificationServer::Notification, foreign_key: :parent_id
11
+
12
+ scope :base_notifications, ->(status = nil) { status.present? ? where(parent: nil, status: status) : where(parent: nil) }
13
+ STATUSES = [:new, :investigating, :fixed].freeze
14
+ STATUSES.each do |status|
15
+ scope "#{status}_notifications", -> { where(status: status) }
16
+ end
17
+ scope :application, ->(application = nil) { application.present? ? where(application: application) : all }
18
+ scope :env, ->(env = nil) { where(env: env || Rails.env) }
19
+
20
+ before_create do
21
+ self.status = :new
22
+ self.exception_hash = gen_exception_hash
23
+ self.parent_id = Notification.where(exception_hash: exception_hash, status: [:new, :investigating]).first.try(:id)
24
+ end
25
+
26
+ def similar
27
+ parent_id.nil? ? childrens : Notification.where(arel_table[:id].eq(parent_id).or(arel_table[:parent_id].eq(parent_id).and(arel_table[:id].eq(id).not)))
28
+ end
29
+
30
+ def similar_count(from = nil)
31
+ return childrens.length + 1 if from.nil?
32
+ childrens.count { |notification| notification.created_at >= from } + (created_at >= from ? 1 : 0)
33
+ end
34
+
35
+ def similar_count_sparkline(from = 1.month.ago.beginning_of_day)
36
+ graph_data(from).values
37
+ end
38
+
39
+ def similar_count_flot(from = 3.month.ago.beginning_of_day)
40
+ graph_data(from).to_a
41
+ end
42
+
43
+ def last_time
44
+ [self, *childrens].sort_by(&:created_at).last.created_at
45
+ end
46
+
47
+ def update_recursive(updates)
48
+ base_id = parent_id || id
49
+ Notification.where(arel_table[:id].eq(base_id).or(arel_table[:parent_id].eq(base_id))).update_all(updates) if base_id.present?
50
+ end
51
+
52
+ protected
53
+
54
+ def graph_data(from)
55
+ result = from.to_datetime.step(Time.zone.now.to_datetime, 1).map { |time| [time.beginning_of_day.to_i * 1000, 0] }.to_h
56
+ childrens.each { |notification| result[notification.created_at.beginning_of_day.to_i * 1000] += 1 if notification.created_at >= from }
57
+ result[created_at.beginning_of_day.to_i * 1000] += 1 if created_at >= from
58
+ result
59
+ end
60
+
61
+ def arel_table
62
+ self.class.arel_table
63
+ end
64
+
65
+ def gen_exception_hash
66
+ Digest::SHA1.hexdigest("#{application}#{exception_class}#{exception_message}#{backtrace.to_s.gsub(rails_root, '')}")
67
+ end
68
+ end
69
+ end
@@ -0,0 +1,37 @@
1
+ :ruby
2
+ count = local_assigns.fetch :count, true
3
+ last_time = local_assigns.fetch :last_time, true
4
+ env = local_assigns.fetch :env, false
5
+ %table
6
+ %thead
7
+ %tr
8
+ %th
9
+ %th.hide-md Class
10
+ %th Message
11
+ %th.hide-md.hide-lg{style: 'width: 100px'} Server
12
+ %th.hide-md.hide-lg Process
13
+ - if env
14
+ %th.hide-md.hide-lg Environment
15
+ %th.hide-md Created at
16
+ - if last_time
17
+ %th.hide-md Last time
18
+ - if count
19
+ %th Last 30 days
20
+ %th Total count
21
+ %tbody
22
+ - notifications.each do |notification|
23
+ %tr
24
+ %td= link_to notification.id, notification_path(notification)
25
+ %td.hide-md= link_to notification.exception_class, notification_path(notification)
26
+ %td= link_to notification.exception_message, notification_path(notification)
27
+ %td.hide-md.hide-lg= link_to notification.server, notification_path(notification)
28
+ %td.hide-md.hide-lg= link_to notification.process, notification_path(notification)
29
+ - if env
30
+ %td.hide-md.hide-lg= link_to notification.env, notification_path(notification)
31
+ %td.hide-md= notification.created_at.strftime(time_format)
32
+ - if last_time
33
+ %td.hide-md= notification.last_time.strftime(time_format)
34
+ - if count
35
+ %td
36
+ %div.sparkline{data: {values: notification.similar_count_sparkline.to_json}}
37
+ %td= notification.similar_count
@@ -0,0 +1,7 @@
1
+ .filter
2
+ = form_tag '', remote: true, method: :get do
3
+ = select_tag :status, status_options(selected: params[:status]), onchange: '$(this.form).trigger(\'submit.rails\')'
4
+ = select_tag :application, application_options(selected: params[:application]), prompt: 'All', onchange: '$(this.form).trigger(\'submit.rails\')'
5
+ = select_tag :env, environment_options(selected: params[:env]), onchange: '$(this.form).trigger(\'submit.rails\')'
6
+ = render partial: 'exception_notification_server/notifications/notifications', object: @notifications, as: :notifications
7
+ = will_paginate @notifications, remote: true
@@ -0,0 +1,2 @@
1
+ $('body > .content').html("#{escape_javascript( render file: 'exception_notification_server/notifications/index', layout: false, formats: [:html] )}");
2
+ initialize_sparkline();
@@ -0,0 +1,76 @@
1
+ .buttons
2
+ = link_to 'Investigate', investigate_notification_path, data: {method: :put} if @notification.status.to_sym == :new
3
+ = link_to 'Fix', fix_notification_path, data: {method: :put} if @notification.status.to_sym == :investigating
4
+ = link_to 'Renew', renew_notification_path, data: {method: :put} if @notification.status.to_sym == :fixed
5
+ .graphic
6
+ %h3 Last 3 month count:
7
+ .graph{data: {values: @notification.similar_count_flot.to_json}}
8
+ .id
9
+ %h3
10
+ Id:
11
+ .inline-value= @notification.id
12
+ .status
13
+ %h3
14
+ Status:
15
+ .inline-value= @notification.status
16
+ .exception-hash
17
+ %h3
18
+ Exception hash:
19
+ .inline-value= @notification.exception_hash
20
+ .exception-class
21
+ %h3
22
+ Exception class:
23
+ .inline-value= @notification.exception_class
24
+ .exception-message
25
+ %h3
26
+ Exception message:
27
+ .inline-value= @notification.exception_message
28
+ - if @notification.application.present?
29
+ .environment
30
+ %h3
31
+ Application:
32
+ .inline-value= @notification.application
33
+ - if @notification.env.present?
34
+ .environment
35
+ %h3
36
+ Environment:
37
+ .inline-value= @notification.env
38
+ - if @notification.server.present?
39
+ .server
40
+ %h3
41
+ Server:
42
+ .inline-value= @notification.server
43
+ - if @notification.process.present?
44
+ .process
45
+ %h3
46
+ Process:
47
+ .inline-value= @notification.process
48
+ - if @notification.rails_root.present?
49
+ .rails-root
50
+ %h3
51
+ Rails root:
52
+ .inline-value= @notification.rails_root
53
+ - if @notification.backtrace.present?
54
+ .backtrace
55
+ %h3 Backtrace
56
+ = pretty_json @notification.backtrace
57
+ - if @notification.data.present?
58
+ .data
59
+ %h3 Data
60
+ = pretty_json @notification.data
61
+ - if @notification.request.present?
62
+ .request
63
+ %h3 Request
64
+ = pretty_json @notification.request
65
+ - if @notification.session.present?
66
+ .session
67
+ %h3 Session
68
+ = pretty_json @notification.session
69
+ - if @notification.environment.present?
70
+ .environment
71
+ %h3 Environment
72
+ = pretty_json @notification.environment
73
+ - if @notification.similar.to_a.present?
74
+ %br
75
+ = render partial: 'exception_notification_server/notifications/notifications', object: @notification.similar, as: :notifications, locals: {count: false, last_time: false, env: true}
76
+ %br