solid_queue_monitor 0.1.0 → 0.1.2

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 CHANGED
@@ -1,7 +1,7 @@
1
1
  ---
2
2
  SHA256:
3
- metadata.gz: 21024cd4e55a7bc73fde9161c68186a293976fe9fe6149dde1e0c48c0638f1b0
4
- data.tar.gz: b0bfa87b894001e9a14e8666a26073334302b34e89f9e723ed0266c9ced4cfd9
3
+ metadata.gz: 1cadc5702b1a5de9167c64a48abfb6639362a3d4e765a3fb1bdf250f5ed4fa81
4
+ data.tar.gz: 9f76e05408872948070221edb908146e134bbf67d6be1bc5fe8e8cbbd6b181ea
5
5
  SHA512:
6
- metadata.gz: 82e972e2209c95dcc4693f7a1bc0ac0794743f1ccd02405708730005164c3a652a648bb816f0eaf15ccae92da2ca0ca7d1498877502428fc9e6d6c78698f98fc
7
- data.tar.gz: c695e22c0957ba4b3449c9a0fe46e503f6d32d4e2282bd52ee2d7271a13880e16829cae7b8c4ee74d2fb2422f44f4d4a50ba39e79b3c20d46d9907458d2e79ab
6
+ metadata.gz: c27275ffbfc1c18e3c790abe1c6776f417af6c8684ffd33e392ae94dccb022b6465fe4754154aec86e6c2690f77e5ec674cf77893ce71c34e40a71498d9364e7
7
+ data.tar.gz: ebb05f940720c5f10aab733f2632584d274af3c786300ba56d9f208535610d5034bbd3b0e8554d55e9f2244e4c6a460f646ef68ad533f03edab038264d7c2c12
data/README.md CHANGED
@@ -1,6 +1,9 @@
1
1
  # SolidQueueMonitor
2
2
 
3
- A lightweight, zero-dependency web interface for monitoring Solid Queue jobs in Rails applications.
3
+ [![Gem Version](https://badge.fury.io/rb/solid_queue_monitor.svg)](https://badge.fury.io/rb/solid_queue_monitor)
4
+ [![MIT License](https://img.shields.io/badge/license-MIT-blue.svg)](LICENSE)
5
+
6
+ A lightweight, zero-dependency web interface for monitoring Solid Queue background jobs in Ruby on Rails applications. Perfect for Rails API-only applications and traditional Rails apps.
4
7
 
5
8
  ## Key Advantages
6
9
 
@@ -8,6 +11,7 @@ A lightweight, zero-dependency web interface for monitoring Solid Queue jobs in
8
11
  - **No External Dependencies**: No JavaScript frameworks, no CSS libraries, no additional gems required - just pure Rails.
9
12
  - **Self-contained UI**: All HTML, CSS, and JavaScript are generated server-side, making deployment simple and reliable.
10
13
  - **Minimal Footprint**: Adds minimal overhead to your application while providing powerful monitoring capabilities.
14
+ - **Rails 7 Compatible**: Fully compatible with Rails 7.1+ and the latest Solid Queue versions.
11
15
 
12
16
  ## Features
13
17
 
@@ -16,6 +20,7 @@ A lightweight, zero-dependency web interface for monitoring Solid Queue jobs in
16
20
  - **Job Management**: Execute scheduled jobs on demand
17
21
  - **Failed Job Inspection**: View detailed error information for failed jobs
18
22
  - **Queue Monitoring**: Track job distribution across different queues
23
+ - **Recurring Jobs**: Monitor and manage recurring background tasks
19
24
  - **Pagination**: Navigate through large job lists with ease
20
25
  - **Optional Authentication**: Secure your dashboard with HTTP Basic Authentication
21
26
  - **Responsive Design**: Works on desktop and mobile devices
@@ -36,7 +41,7 @@ A lightweight, zero-dependency web interface for monitoring Solid Queue jobs in
36
41
  Add this line to your application's Gemfile:
37
42
 
38
43
  ```ruby
39
- gem 'solid_queue_monitor'
44
+ gem 'solid_queue_monitor', '~> 0.1.2'
40
45
  ```
41
46
 
42
47
  Then execute:
@@ -96,6 +101,7 @@ The dashboard provides several views:
96
101
  - **Overview**: Shows statistics and recent jobs
97
102
  - **Ready Jobs**: Jobs that are ready to be executed
98
103
  - **Scheduled Jobs**: Jobs scheduled for future execution
104
+ - **Recurring Jobs**: Jobs that run on a recurring schedule
99
105
  - **Failed Jobs**: Jobs that have failed with error details
100
106
  - **Queues**: Distribution of jobs across different queues
101
107
 
@@ -103,6 +109,20 @@ The dashboard provides several views:
103
109
 
104
110
  For API-only Rails applications, SolidQueueMonitor works out of the box without requiring you to enable the asset pipeline or webpacker. This makes it an ideal choice for monitoring background jobs in modern API-based architectures.
105
111
 
112
+ ## Use Cases
113
+
114
+ - **Production Monitoring**: Keep an eye on your background job processing in production environments
115
+ - **Debugging**: Quickly identify and troubleshoot failed jobs
116
+ - **Job Management**: Execute scheduled jobs on demand when needed
117
+ - **Performance Analysis**: Track job distribution and identify bottlenecks
118
+ - **DevOps Integration**: Easily integrate with your monitoring stack
119
+
120
+ ## Compatibility
121
+
122
+ - **Ruby**: 3.1.6 or higher
123
+ - **Rails**: 7.1 or higher
124
+ - **Solid Queue**: 0.1.0 or higher
125
+
106
126
  ## Contributing
107
127
 
108
128
  Contributions are welcome! Here's how you can contribute:
@@ -126,3 +146,9 @@ The gem is available as open source under the terms of the [MIT License](https:/
126
146
  ## Code of Conduct
127
147
 
128
148
  Everyone interacting in the SolidQueueMonitor project's codebases, issue trackers, chat rooms and mailing lists is expected to follow the [code of conduct](https://github.com/yourusername/solid_queue_monitor/blob/main/CODE_OF_CONDUCT.md).
149
+
150
+ ## Related Projects
151
+
152
+ - [Solid Queue](https://github.com/rails/solid_queue) - The official Rails background job framework
153
+ - [Rails](https://github.com/rails/rails) - The web application framework
154
+ - [ActiveJob](https://github.com/rails/rails/tree/main/activejob) - Rails job framework
@@ -1,10 +1,17 @@
1
1
  module SolidQueueMonitor
2
2
  class MonitorController < ActionController::Base
3
3
  include ActionController::HttpAuthentication::Basic::ControllerMethods
4
+ include ActionController::Flash
4
5
 
5
6
  before_action :authenticate, if: -> { SolidQueueMonitor::AuthenticationService.authentication_required? }
6
7
  layout false
7
- skip_before_action :verify_authenticity_token, only: [:execute_jobs]
8
+ skip_before_action :verify_authenticity_token, only: [:execute_jobs, :retry_failed_job, :discard_failed_job, :retry_failed_jobs, :discard_failed_jobs]
9
+
10
+ # Define a helper method for setting flash messages
11
+ def set_flash_message(message, type)
12
+ session[:flash_message] = message
13
+ session[:flash_type] = type
14
+ end
8
15
 
9
16
  def index
10
17
  @stats = SolidQueueMonitor::StatsCalculator.calculate
@@ -68,11 +75,57 @@ module SolidQueueMonitor
68
75
  def execute_jobs
69
76
  if params[:job_ids].present?
70
77
  SolidQueueMonitor::ExecuteJobService.new.execute_many(params[:job_ids])
71
- redirect_url = "#{scheduled_jobs_path}?message=Selected jobs moved to ready queue&message_type=success"
78
+ set_flash_message('Selected jobs moved to ready queue', 'success')
72
79
  else
73
- redirect_url = "#{scheduled_jobs_path}?message=No jobs selected&message_type=error"
80
+ set_flash_message('No jobs selected', 'error')
74
81
  end
75
- redirect_to redirect_url
82
+ redirect_to scheduled_jobs_path
83
+ end
84
+
85
+ def retry_failed_job
86
+ id = params[:id]
87
+ service = SolidQueueMonitor::FailedJobService.new
88
+
89
+ if service.retry_job(id)
90
+ set_flash_message("Job #{id} has been queued for retry.", 'success')
91
+ else
92
+ set_flash_message("Failed to retry job #{id}.", 'error')
93
+ end
94
+ redirect_to failed_jobs_path
95
+ end
96
+
97
+ def discard_failed_job
98
+ id = params[:id]
99
+ service = SolidQueueMonitor::FailedJobService.new
100
+
101
+ if service.discard_job(id)
102
+ set_flash_message("Job #{id} has been discarded.", 'success')
103
+ else
104
+ set_flash_message("Failed to discard job #{id}.", 'error')
105
+ end
106
+ redirect_to failed_jobs_path
107
+ end
108
+
109
+ def retry_failed_jobs
110
+ result = SolidQueueMonitor::FailedJobService.new.retry_all(params[:job_ids])
111
+
112
+ if result[:success]
113
+ set_flash_message(result[:message], 'success')
114
+ else
115
+ set_flash_message(result[:message], 'error')
116
+ end
117
+ redirect_to failed_jobs_path
118
+ end
119
+
120
+ def discard_failed_jobs
121
+ result = SolidQueueMonitor::FailedJobService.new.discard_all(params[:job_ids])
122
+
123
+ if result[:success]
124
+ set_flash_message(result[:message], 'success')
125
+ else
126
+ set_flash_message(result[:message], 'error')
127
+ end
128
+ redirect_to failed_jobs_path
76
129
  end
77
130
 
78
131
  private
@@ -88,11 +141,19 @@ module SolidQueueMonitor
88
141
  end
89
142
 
90
143
  def render_page(title, content)
144
+ # Get flash message from session
145
+ message = session[:flash_message]
146
+ message_type = session[:flash_type]
147
+
148
+ # Clear the flash message from session after using it
149
+ session.delete(:flash_message)
150
+ session.delete(:flash_type)
151
+
91
152
  html = SolidQueueMonitor::HtmlGenerator.new(
92
153
  title: title,
93
154
  content: content,
94
- message: params[:notice] || params[:alert],
95
- message_type: params[:notice] ? 'success' : 'error'
155
+ message: message,
156
+ message_type: message_type
96
157
  ).generate
97
158
 
98
159
  render html: html.html_safe
@@ -233,7 +294,14 @@ module SolidQueueMonitor
233
294
  end
234
295
 
235
296
  if params[:queue_name].present?
236
- relation = relation.where("queue_name LIKE ?", "%#{params[:queue_name]}%")
297
+ # Check if FailedExecution has queue_name column
298
+ if relation.column_names.include?('queue_name')
299
+ relation = relation.where("queue_name LIKE ?", "%#{params[:queue_name]}%")
300
+ else
301
+ # If not, filter by job's queue_name
302
+ job_ids = SolidQueue::Job.where("queue_name LIKE ?", "%#{params[:queue_name]}%").pluck(:id)
303
+ relation = relation.where(job_id: job_ids)
304
+ end
237
305
  end
238
306
 
239
307
  relation
@@ -22,55 +22,33 @@ module SolidQueueMonitor
22
22
  def generate_pagination(current_page, total_pages)
23
23
  return '' if total_pages <= 1
24
24
 
25
- links = []
25
+ html = '<div class="pagination">'
26
26
 
27
27
  # Previous page link
28
28
  if current_page > 1
29
- links << "<a href='?page=#{current_page - 1}#{query_params}' class='pagination-link'>&laquo; Previous</a>"
29
+ html += "<a href=\"?page=#{current_page - 1}#{query_params}\" class=\"pagination-link pagination-nav\">Previous</a>"
30
30
  else
31
- links << "<span class='pagination-link disabled'>&laquo; Previous</span>"
31
+ html += '<span class="pagination-link pagination-nav disabled">Previous</span>'
32
32
  end
33
33
 
34
- # Page number links
35
- if total_pages <= 7
36
- # Show all pages if there are 7 or fewer
37
- (1..total_pages).each do |page|
38
- links << page_link(page, current_page)
39
- end
40
- else
41
- # Show first page, last page, and pages around current
42
- links << page_link(1, current_page)
43
-
44
- if current_page > 3
45
- links << "<span class='pagination-gap'>...</span>"
46
- end
47
-
48
- start_page = [current_page - 1, 2].max
49
- end_page = [current_page + 1, total_pages - 1].min
50
-
51
- (start_page..end_page).each do |page|
52
- links << page_link(page, current_page)
53
- end
54
-
55
- if current_page < total_pages - 2
56
- links << "<span class='pagination-gap'>...</span>"
34
+ # Page links
35
+ (1..total_pages).each do |page|
36
+ if page == current_page
37
+ html += "<span class=\"pagination-current\">#{page}</span>"
38
+ else
39
+ html += "<a href=\"?page=#{page}#{query_params}\" class=\"pagination-link\">#{page}</a>"
57
40
  end
58
-
59
- links << page_link(total_pages, current_page)
60
41
  end
61
42
 
62
43
  # Next page link
63
44
  if current_page < total_pages
64
- links << "<a href='?page=#{current_page + 1}#{query_params}' class='pagination-link'>Next &raquo;</a>"
45
+ html += "<a href=\"?page=#{current_page + 1}#{query_params}\" class=\"pagination-link pagination-nav\">Next</a>"
65
46
  else
66
- links << "<span class='pagination-link disabled'>Next &raquo;</span>"
47
+ html += '<span class="pagination-link pagination-nav disabled">Next</span>'
67
48
  end
68
49
 
69
- <<-HTML
70
- <div class="pagination">
71
- #{links.join}
72
- </div>
73
- HTML
50
+ html += '</div>'
51
+ html
74
52
  end
75
53
 
76
54
  def calculate_visible_pages(current_page, total_pages)
@@ -96,12 +74,15 @@ module SolidQueueMonitor
96
74
  def format_arguments(arguments)
97
75
  return '-' unless arguments.present?
98
76
 
99
- if arguments.is_a?(Array) && arguments.length == 1 && arguments[0].is_a?(Hash)
100
- # Handle ActiveJob-style arguments
101
- format_hash(arguments[0])
102
- else
103
- "<code>#{arguments.to_json}</code>"
77
+ # For ActiveJob format
78
+ if arguments.is_a?(Hash) && arguments['arguments'].present?
79
+ return "<code>#{arguments['arguments'].inspect}</code>"
80
+ elsif arguments.is_a?(Array) && arguments.length == 1 && arguments[0].is_a?(Hash) && arguments[0]['arguments'].present?
81
+ return "<code>#{arguments[0]['arguments'].inspect}</code>"
104
82
  end
83
+
84
+ # For regular arguments format
85
+ "<code>#{arguments.inspect}</code>"
105
86
  end
106
87
 
107
88
  def format_hash(hash)
@@ -114,23 +95,47 @@ module SolidQueueMonitor
114
95
  "<code>#{formatted}</code>"
115
96
  end
116
97
 
117
- private
98
+ # Helper method to get the current request path
99
+ def request_path
100
+ # Try to get the current path from the controller's request
101
+ if defined?(controller) && controller.respond_to?(:request)
102
+ controller.request.path
103
+ else
104
+ # Fallback to a default path if we can't get the current path
105
+ "/solid_queue"
106
+ end
107
+ end
118
108
 
119
- def page_link(page, current_page)
120
- if page == current_page
121
- "<span class='pagination-current'>#{page}</span>"
109
+ # Helper method to get the mount point of the engine
110
+ def engine_mount_point
111
+ path_parts = request_path.split('/')
112
+ if path_parts.length >= 3
113
+ "/#{path_parts[1]}/#{path_parts[2]}"
122
114
  else
123
- "<a href='?page=#{page}#{query_params}' class='pagination-link'>#{page}</a>"
115
+ "/solid_queue"
124
116
  end
125
117
  end
126
118
 
119
+ private
120
+
127
121
  def query_params
128
122
  params = []
129
- params << "class_name=#{CGI.escape(@filters[:class_name])}" if @filters && @filters[:class_name].present?
130
- params << "queue_name=#{CGI.escape(@filters[:queue_name])}" if @filters && @filters[:queue_name].present?
131
- params << "status=#{CGI.escape(@filters[:status])}" if @filters && @filters[:status].present?
123
+ params << "class_name=#{@filters[:class_name]}" if @filters && @filters[:class_name].present?
124
+ params << "queue_name=#{@filters[:queue_name]}" if @filters && @filters[:queue_name].present?
125
+ params << "status=#{@filters[:status]}" if @filters && @filters[:status].present?
132
126
 
133
127
  params.empty? ? '' : "&#{params.join('&')}"
134
128
  end
129
+
130
+ # Helper method to get the full path for a route
131
+ def full_path(route_name, *args)
132
+ begin
133
+ # Try to use the engine routes first
134
+ SolidQueueMonitor::Engine.routes.url_helpers.send(route_name, *args)
135
+ rescue NoMethodError
136
+ # Fall back to main app routes
137
+ Rails.application.routes.url_helpers.send("solid_queue_#{route_name}", *args)
138
+ end
139
+ end
135
140
  end
136
141
  end
@@ -1,5 +1,8 @@
1
1
  module SolidQueueMonitor
2
2
  class FailedJobsPresenter < BasePresenter
3
+ include Rails.application.routes.url_helpers
4
+ include SolidQueueMonitor::Engine.routes.url_helpers
5
+
3
6
  def initialize(jobs, current_page: 1, total_pages: 1, filters: {})
4
7
  @jobs = jobs
5
8
  @current_page = current_page
@@ -16,7 +19,7 @@ module SolidQueueMonitor
16
19
  def generate_filter_form
17
20
  <<-HTML
18
21
  <div class="filter-form-container">
19
- <form method="get" action="" class="filter-form">
22
+ <form method="get" action="#{failed_jobs_path}" class="filter-form">
20
23
  <div class="filter-group">
21
24
  <label for="class_name">Job Class:</label>
22
25
  <input type="text" name="class_name" id="class_name" value="#{@filters[:class_name]}" placeholder="Filter by class name">
@@ -33,38 +36,275 @@ module SolidQueueMonitor
33
36
  </div>
34
37
  </form>
35
38
  </div>
39
+
40
+ <div class="bulk-actions-bar">
41
+ <button type="button" class="action-button retry-button" id="retry-selected-top" disabled>Retry Selected</button>
42
+ <button type="button" class="action-button discard-button" id="discard-selected-top" disabled>Discard Selected</button>
43
+ </div>
36
44
  HTML
37
45
  end
38
46
 
39
47
  def generate_table
40
48
  <<-HTML
41
- <div class="table-container">
42
- <table>
43
- <thead>
44
- <tr>
45
- <th>Job</th>
46
- <th>Error</th>
47
- <th>Failed At</th>
48
- <th>Arguments</th>
49
- </tr>
50
- </thead>
51
- <tbody>
52
- #{@jobs.map { |execution| generate_row(execution) }.join}
53
- </tbody>
54
- </table>
55
- </div>
49
+ <form method="post" id="failed-jobs-form">
50
+ <div class="table-container">
51
+ <table>
52
+ <thead>
53
+ <tr>
54
+ <th><input type="checkbox" id="select-all" class="select-all-checkbox"></th>
55
+ <th>Job</th>
56
+ <th>Queue</th>
57
+ <th>Error</th>
58
+ <th>Actions</th>
59
+ </tr>
60
+ </thead>
61
+ <tbody>
62
+ #{@jobs.map { |failed_execution| generate_row(failed_execution) }.join}
63
+ </tbody>
64
+ </table>
65
+ </div>
66
+ </form>
67
+
68
+ <script>
69
+ document.addEventListener('DOMContentLoaded', function() {
70
+ // Handle select all checkboxes
71
+ const selectAllHeader = document.getElementById('select-all');
72
+ const checkboxes = document.querySelectorAll('.job-checkbox');
73
+ const retrySelectedBtn = document.getElementById('retry-selected-top');
74
+ const discardSelectedBtn = document.getElementById('discard-selected-top');
75
+ const form = document.getElementById('failed-jobs-form');
76
+
77
+ function updateButtonState() {
78
+ const checkedBoxes = document.querySelectorAll('.job-checkbox:checked');
79
+ retrySelectedBtn.disabled = checkedBoxes.length === 0;
80
+ discardSelectedBtn.disabled = checkedBoxes.length === 0;
81
+ }
82
+
83
+ function toggleAll(checked) {
84
+ checkboxes.forEach(checkbox => {
85
+ checkbox.checked = checked;
86
+ });
87
+ selectAllHeader.checked = checked;
88
+ updateButtonState();
89
+ }
90
+
91
+ selectAllHeader.addEventListener('change', function() {
92
+ toggleAll(this.checked);
93
+ });
94
+
95
+ checkboxes.forEach(checkbox => {
96
+ checkbox.addEventListener('change', function() {
97
+ updateButtonState();
98
+
99
+ // Update select all checkboxes if needed
100
+ const allChecked = document.querySelectorAll('.job-checkbox:checked').length === checkboxes.length;
101
+ selectAllHeader.checked = allChecked;
102
+ });
103
+ });
104
+
105
+ // Handle bulk actions
106
+ retrySelectedBtn.addEventListener('click', function() {
107
+ const selectedIds = Array.from(document.querySelectorAll('.job-checkbox:checked')).map(cb => cb.value);
108
+ if (selectedIds.length === 0) return;
109
+
110
+ if (confirm('Are you sure you want to retry the selected jobs?')) {
111
+ form.action = '#{retry_failed_jobs_path}';
112
+
113
+ // Add a special flag to indicate this should redirect properly
114
+ const redirectInput = document.createElement('input');
115
+ redirectInput.type = 'hidden';
116
+ redirectInput.name = 'redirect_cleanly';
117
+ redirectInput.value = 'true';
118
+ form.appendChild(redirectInput);
119
+
120
+ // Add selected IDs as hidden inputs
121
+ selectedIds.forEach(id => {
122
+ const input = document.createElement('input');
123
+ input.type = 'hidden';
124
+ input.name = 'job_ids[]';
125
+ input.value = id;
126
+ form.appendChild(input);
127
+ });
128
+
129
+ // Submit the form and then replace the URL location immediately after
130
+ form.submit();
131
+
132
+ // Delay the redirect to give the form time to submit
133
+ setTimeout(function() {
134
+ // Reset to the clean URL without query parameters
135
+ window.history.replaceState({}, '', window.location.pathname);
136
+ }, 100);
137
+ }
138
+ });
139
+
140
+ discardSelectedBtn.addEventListener('click', function() {
141
+ const selectedIds = Array.from(document.querySelectorAll('.job-checkbox:checked')).map(cb => cb.value);
142
+ if (selectedIds.length === 0) return;
143
+
144
+ if (confirm('Are you sure you want to discard the selected jobs?')) {
145
+ form.action = '#{discard_failed_jobs_path}';
146
+
147
+ // Add a special flag to indicate this should redirect properly
148
+ const redirectInput = document.createElement('input');
149
+ redirectInput.type = 'hidden';
150
+ redirectInput.name = 'redirect_cleanly';
151
+ redirectInput.value = 'true';
152
+ form.appendChild(redirectInput);
153
+
154
+ // Add selected IDs as hidden inputs
155
+ selectedIds.forEach(id => {
156
+ const input = document.createElement('input');
157
+ input.type = 'hidden';
158
+ input.name = 'job_ids[]';
159
+ input.value = id;
160
+ form.appendChild(input);
161
+ });
162
+
163
+ // Submit the form and then replace the URL location immediately after
164
+ form.submit();
165
+
166
+ // Delay the redirect to give the form time to submit
167
+ setTimeout(function() {
168
+ // Reset to the clean URL without query parameters
169
+ window.history.replaceState({}, '', window.location.pathname);
170
+ }, 100);
171
+ }
172
+ });
173
+
174
+ // Initialize button state
175
+ updateButtonState();
176
+
177
+ // Global function for retry action
178
+ window.submitRetryForm = function(id) {
179
+ const form = document.createElement('form');
180
+ form.method = 'post';
181
+ form.action = '#{retry_failed_job_path(id: "PLACEHOLDER")}';
182
+ form.action = form.action.replace('PLACEHOLDER', id);
183
+ form.style.display = 'none';
184
+
185
+ // Add a special flag to indicate this should redirect properly
186
+ const redirectInput = document.createElement('input');
187
+ redirectInput.type = 'hidden';
188
+ redirectInput.name = 'redirect_cleanly';
189
+ redirectInput.value = 'true';
190
+ form.appendChild(redirectInput);
191
+
192
+ document.body.appendChild(form);
193
+
194
+ // Submit the form and then replace the URL location immediately after
195
+ form.submit();
196
+
197
+ // Delay the redirect to give the form time to submit
198
+ setTimeout(function() {
199
+ // Reset to the clean URL without query parameters
200
+ window.history.replaceState({}, '', window.location.pathname);
201
+ }, 100);
202
+ };
203
+
204
+ // Global function for discard action
205
+ window.submitDiscardForm = function(id) {
206
+ if (confirm('Are you sure you want to discard this job?')) {
207
+ const form = document.createElement('form');
208
+ form.method = 'post';
209
+ form.action = '#{discard_failed_job_path(id: "PLACEHOLDER")}';
210
+ form.action = form.action.replace('PLACEHOLDER', id);
211
+ form.style.display = 'none';
212
+
213
+ // Add a special flag to indicate this should redirect properly
214
+ const redirectInput = document.createElement('input');
215
+ redirectInput.type = 'hidden';
216
+ redirectInput.name = 'redirect_cleanly';
217
+ redirectInput.value = 'true';
218
+ form.appendChild(redirectInput);
219
+
220
+ document.body.appendChild(form);
221
+
222
+ // Submit the form and then replace the URL location immediately after
223
+ form.submit();
224
+
225
+ // Delay the redirect to give the form time to submit
226
+ setTimeout(function() {
227
+ // Reset to the clean URL without query parameters
228
+ window.history.replaceState({}, '', window.location.pathname);
229
+ }, 100);
230
+ }
231
+ };
232
+ });
233
+ </script>
56
234
  HTML
57
235
  end
58
236
 
59
- def generate_row(execution)
237
+ def generate_row(failed_execution)
238
+ job = failed_execution.job
239
+ error = parse_error(failed_execution.error)
240
+
60
241
  <<-HTML
61
242
  <tr>
62
- <td>#{execution.job.class_name}</td>
63
- <td class="error-message">#{execution.error['message']}</td>
64
- <td>#{format_datetime(execution.created_at)}</td>
65
- <td>#{format_arguments(execution.job.arguments)}</td>
243
+ <td><input type="checkbox" class="job-checkbox" value="#{failed_execution.id}"></td>
244
+ <td>
245
+ <div class="job-class">#{job.class_name}</div>
246
+ <div class="job-meta">
247
+ <span class="job-timestamp">Queued at: #{format_datetime(job.created_at)}</span>
248
+ </div>
249
+ </td>
250
+ <td>
251
+ <div class="job-queue">#{job.queue_name}</div>
252
+ </td>
253
+ <td>
254
+ <div class="error-message">#{error[:message]}</div>
255
+ <div class="job-meta">
256
+ <span class="job-timestamp">Failed at: #{format_datetime(failed_execution.created_at)}</span>
257
+ </div>
258
+ <details>
259
+ <summary>Backtrace</summary>
260
+ <pre class="error-backtrace">#{error[:backtrace]}</pre>
261
+ </details>
262
+ </td>
263
+ <td class="actions-cell">
264
+ <div class="job-actions">
265
+ <a href="javascript:void(0)"
266
+ onclick="submitRetryForm(#{failed_execution.id})"
267
+ class="action-button retry-button">Retry</a>
268
+
269
+ <a href="javascript:void(0)"
270
+ onclick="submitDiscardForm(#{failed_execution.id})"
271
+ class="action-button discard-button">Discard</a>
272
+ </div>
273
+ </td>
66
274
  </tr>
67
275
  HTML
68
276
  end
277
+
278
+ def parse_error(error)
279
+ return { message: 'Unknown error', backtrace: '' } unless error
280
+
281
+ if error.is_a?(String)
282
+ { message: error, backtrace: '' }
283
+ elsif error.is_a?(Hash)
284
+ message = error['message'] || error[:message] || 'Unknown error'
285
+ backtrace = error['backtrace'] || error[:backtrace] || []
286
+ backtrace = backtrace.join("\n") if backtrace.is_a?(Array)
287
+ { message: message, backtrace: backtrace }
288
+ else
289
+ { message: 'Unknown error format', backtrace: error.to_s }
290
+ end
291
+ end
292
+
293
+ def get_queue_name(failed_execution, job)
294
+ # Try to get queue_name from failed_execution if the method exists
295
+ if failed_execution.respond_to?(:queue_name) && !failed_execution.queue_name.nil?
296
+ failed_execution.queue_name
297
+ else
298
+ # Fall back to job's queue_name
299
+ job.queue_name
300
+ end
301
+ rescue NoMethodError
302
+ # If there's an error accessing queue_name, fall back to job's queue_name
303
+ job.queue_name
304
+ end
305
+
306
+ def default_url_options
307
+ { only_path: true }
308
+ end
69
309
  end
70
310
  end