solid_queue_monitor 0.1.1 → 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 +4 -4
- data/README.md +28 -2
- data/app/controllers/solid_queue_monitor/monitor_controller.rb +75 -7
- data/app/presenters/solid_queue_monitor/base_presenter.rb +53 -48
- data/app/presenters/solid_queue_monitor/failed_jobs_presenter.rb +261 -21
- data/app/presenters/solid_queue_monitor/scheduled_jobs_presenter.rb +33 -10
- data/app/services/solid_queue_monitor/failed_job_service.rb +103 -0
- data/app/services/solid_queue_monitor/html_generator.rb +26 -1
- data/app/services/solid_queue_monitor/stylesheet_generator.rb +150 -1
- data/config/routes.rb +12 -6
- data/lib/solid_queue_monitor/version.rb +1 -1
- metadata +3 -2
checksums.yaml
CHANGED
@@ -1,7 +1,7 @@
|
|
1
1
|
---
|
2
2
|
SHA256:
|
3
|
-
metadata.gz:
|
4
|
-
data.tar.gz:
|
3
|
+
metadata.gz: 1cadc5702b1a5de9167c64a48abfb6639362a3d4e765a3fb1bdf250f5ed4fa81
|
4
|
+
data.tar.gz: 9f76e05408872948070221edb908146e134bbf67d6be1bc5fe8e8cbbd6b181ea
|
5
5
|
SHA512:
|
6
|
-
metadata.gz:
|
7
|
-
data.tar.gz:
|
6
|
+
metadata.gz: c27275ffbfc1c18e3c790abe1c6776f417af6c8684ffd33e392ae94dccb022b6465fe4754154aec86e6c2690f77e5ec674cf77893ce71c34e40a71498d9364e7
|
7
|
+
data.tar.gz: ebb05f940720c5f10aab733f2632584d274af3c786300ba56d9f208535610d5034bbd3b0e8554d55e9f2244e4c6a460f646ef68ad533f03edab038264d7c2c12
|
data/README.md
CHANGED
@@ -1,6 +1,9 @@
|
|
1
1
|
# SolidQueueMonitor
|
2
2
|
|
3
|
-
|
3
|
+
[](https://badge.fury.io/rb/solid_queue_monitor)
|
4
|
+
[](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
|
-
|
78
|
+
set_flash_message('Selected jobs moved to ready queue', 'success')
|
72
79
|
else
|
73
|
-
|
80
|
+
set_flash_message('No jobs selected', 'error')
|
74
81
|
end
|
75
|
-
redirect_to
|
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:
|
95
|
-
message_type:
|
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
|
-
|
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
|
-
|
25
|
+
html = '<div class="pagination">'
|
26
26
|
|
27
27
|
# Previous page link
|
28
28
|
if current_page > 1
|
29
|
-
|
29
|
+
html += "<a href=\"?page=#{current_page - 1}#{query_params}\" class=\"pagination-link pagination-nav\">Previous</a>"
|
30
30
|
else
|
31
|
-
|
31
|
+
html += '<span class="pagination-link pagination-nav disabled">Previous</span>'
|
32
32
|
end
|
33
33
|
|
34
|
-
# Page
|
35
|
-
|
36
|
-
|
37
|
-
|
38
|
-
|
39
|
-
|
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
|
-
|
45
|
+
html += "<a href=\"?page=#{current_page + 1}#{query_params}\" class=\"pagination-link pagination-nav\">Next</a>"
|
65
46
|
else
|
66
|
-
|
47
|
+
html += '<span class="pagination-link pagination-nav disabled">Next</span>'
|
67
48
|
end
|
68
49
|
|
69
|
-
|
70
|
-
|
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
|
-
|
100
|
-
|
101
|
-
|
102
|
-
|
103
|
-
"<code>#{arguments.
|
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
|
-
|
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
|
-
|
120
|
-
|
121
|
-
|
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
|
-
"
|
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=#{
|
130
|
-
params << "queue_name=#{
|
131
|
-
params << "status=#{
|
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
|
-
<
|
42
|
-
<table>
|
43
|
-
<
|
44
|
-
<
|
45
|
-
<
|
46
|
-
|
47
|
-
|
48
|
-
|
49
|
-
|
50
|
-
|
51
|
-
|
52
|
-
|
53
|
-
|
54
|
-
|
55
|
-
|
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(
|
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
|
63
|
-
<td
|
64
|
-
|
65
|
-
|
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
|
@@ -37,21 +37,24 @@ module SolidQueueMonitor
|
|
37
37
|
</div>
|
38
38
|
</form>
|
39
39
|
</div>
|
40
|
+
|
41
|
+
<div class="bulk-actions-bar">
|
42
|
+
<button type="button" class="action-button execute-button" id="execute-selected-top" disabled>Execute Selected</button>
|
43
|
+
</div>
|
40
44
|
HTML
|
41
45
|
end
|
42
46
|
|
43
47
|
def generate_table_with_actions
|
44
48
|
<<-HTML
|
45
|
-
<form action="#{execute_jobs_path}" method="POST">
|
49
|
+
<form id="scheduled-jobs-form" action="#{execute_jobs_path}" method="POST">
|
46
50
|
#{generate_table}
|
47
|
-
<div class="table-actions">
|
48
|
-
<button type="submit" class="execute-btn" id="bulk-execute" disabled>Execute Selected</button>
|
49
|
-
</div>
|
50
51
|
</form>
|
51
52
|
<script>
|
52
53
|
document.addEventListener('DOMContentLoaded', function() {
|
53
54
|
const selectAllCheckbox = document.querySelector('th input[type="checkbox"]');
|
54
55
|
const jobCheckboxes = document.getElementsByName('job_ids[]');
|
56
|
+
const executeButton = document.getElementById('execute-selected-top');
|
57
|
+
const form = document.getElementById('scheduled-jobs-form');
|
55
58
|
|
56
59
|
selectAllCheckbox.addEventListener('change', function() {
|
57
60
|
jobCheckboxes.forEach(checkbox => checkbox.checked = this.checked);
|
@@ -64,13 +67,33 @@ module SolidQueueMonitor
|
|
64
67
|
updateExecuteButton();
|
65
68
|
});
|
66
69
|
});
|
70
|
+
|
71
|
+
// Add event listener for the execute button
|
72
|
+
executeButton.addEventListener('click', function() {
|
73
|
+
const selectedIds = Array.from(document.querySelectorAll('input[name="job_ids[]"]:checked')).map(cb => cb.value);
|
74
|
+
if (selectedIds.length === 0) return;
|
75
|
+
|
76
|
+
// Add selected IDs as hidden inputs
|
77
|
+
selectedIds.forEach(id => {
|
78
|
+
const input = document.createElement('input');
|
79
|
+
input.type = 'hidden';
|
80
|
+
input.name = 'job_ids[]';
|
81
|
+
input.value = id;
|
82
|
+
form.appendChild(input);
|
83
|
+
});
|
84
|
+
|
85
|
+
form.submit();
|
86
|
+
});
|
87
|
+
|
88
|
+
function updateExecuteButton() {
|
89
|
+
const checkboxes = document.getElementsByName('job_ids[]');
|
90
|
+
const checked = Array.from(checkboxes).some(cb => cb.checked);
|
91
|
+
executeButton.disabled = !checked;
|
92
|
+
}
|
93
|
+
|
94
|
+
// Initialize button state
|
95
|
+
updateExecuteButton();
|
67
96
|
});
|
68
|
-
|
69
|
-
function updateExecuteButton() {
|
70
|
-
const checkboxes = document.getElementsByName('job_ids[]');
|
71
|
-
const checked = Array.from(checkboxes).some(cb => cb.checked);
|
72
|
-
document.getElementById('bulk-execute').disabled = !checked;
|
73
|
-
}
|
74
97
|
</script>
|
75
98
|
HTML
|
76
99
|
end
|
@@ -0,0 +1,103 @@
|
|
1
|
+
module SolidQueueMonitor
|
2
|
+
class FailedJobService
|
3
|
+
def retry_job(failed_execution_id)
|
4
|
+
failed_execution = SolidQueue::FailedExecution.find_by(id: failed_execution_id)
|
5
|
+
return { success: false, message: "Failed job not found" } unless failed_execution
|
6
|
+
|
7
|
+
job = failed_execution.job
|
8
|
+
return { success: false, message: "Associated job not found" } unless job
|
9
|
+
|
10
|
+
ActiveRecord::Base.transaction do
|
11
|
+
# Create a ready execution for the job
|
12
|
+
SolidQueue::ReadyExecution.create!(
|
13
|
+
job_id: job.id,
|
14
|
+
queue_name: get_queue_name(failed_execution, job),
|
15
|
+
priority: job.priority
|
16
|
+
)
|
17
|
+
|
18
|
+
# Delete the failed execution
|
19
|
+
failed_execution.destroy!
|
20
|
+
end
|
21
|
+
|
22
|
+
{ success: true, message: "Job moved to ready queue for retry" }
|
23
|
+
end
|
24
|
+
|
25
|
+
def discard_job(failed_execution_id)
|
26
|
+
failed_execution = SolidQueue::FailedExecution.find_by(id: failed_execution_id)
|
27
|
+
return { success: false, message: "Failed job not found" } unless failed_execution
|
28
|
+
|
29
|
+
job = failed_execution.job
|
30
|
+
return { success: false, message: "Associated job not found" } unless job
|
31
|
+
|
32
|
+
ActiveRecord::Base.transaction do
|
33
|
+
# Mark the job as finished
|
34
|
+
job.update!(finished_at: Time.current)
|
35
|
+
|
36
|
+
# Delete the failed execution
|
37
|
+
failed_execution.destroy!
|
38
|
+
end
|
39
|
+
|
40
|
+
{ success: true, message: "Job has been discarded" }
|
41
|
+
end
|
42
|
+
|
43
|
+
def retry_all(job_ids)
|
44
|
+
return { success: false, message: "No jobs selected" } if job_ids.blank?
|
45
|
+
|
46
|
+
success_count = 0
|
47
|
+
failed_count = 0
|
48
|
+
|
49
|
+
job_ids.each do |id|
|
50
|
+
result = retry_job(id)
|
51
|
+
if result[:success]
|
52
|
+
success_count += 1
|
53
|
+
else
|
54
|
+
failed_count += 1
|
55
|
+
end
|
56
|
+
end
|
57
|
+
|
58
|
+
if success_count > 0 && failed_count == 0
|
59
|
+
{ success: true, message: "All selected jobs have been queued for retry" }
|
60
|
+
elsif success_count > 0 && failed_count > 0
|
61
|
+
{ success: true, message: "#{success_count} jobs queued for retry, #{failed_count} failed" }
|
62
|
+
else
|
63
|
+
{ success: false, message: "Failed to retry jobs" }
|
64
|
+
end
|
65
|
+
end
|
66
|
+
|
67
|
+
def discard_all(job_ids)
|
68
|
+
return { success: false, message: "No jobs selected" } if job_ids.blank?
|
69
|
+
|
70
|
+
success_count = 0
|
71
|
+
failed_count = 0
|
72
|
+
|
73
|
+
job_ids.each do |id|
|
74
|
+
result = discard_job(id)
|
75
|
+
if result[:success]
|
76
|
+
success_count += 1
|
77
|
+
else
|
78
|
+
failed_count += 1
|
79
|
+
end
|
80
|
+
end
|
81
|
+
|
82
|
+
if success_count > 0 && failed_count == 0
|
83
|
+
{ success: true, message: "All selected jobs have been discarded" }
|
84
|
+
elsif success_count > 0 && failed_count > 0
|
85
|
+
{ success: true, message: "#{success_count} jobs discarded, #{failed_count} failed" }
|
86
|
+
else
|
87
|
+
{ success: false, message: "Failed to discard jobs" }
|
88
|
+
end
|
89
|
+
end
|
90
|
+
|
91
|
+
private
|
92
|
+
|
93
|
+
def get_queue_name(failed_execution, job)
|
94
|
+
# Try to get queue_name from failed_execution if the method exists
|
95
|
+
if failed_execution.respond_to?(:queue_name) && failed_execution.queue_name.present?
|
96
|
+
failed_execution.queue_name
|
97
|
+
else
|
98
|
+
# Fall back to job's queue_name
|
99
|
+
job.queue_name
|
100
|
+
end
|
101
|
+
end
|
102
|
+
end
|
103
|
+
end
|
@@ -54,7 +54,32 @@ module SolidQueueMonitor
|
|
54
54
|
def render_message
|
55
55
|
return '' unless @message
|
56
56
|
class_name = @message_type == 'success' ? 'message-success' : 'message-error'
|
57
|
-
|
57
|
+
<<-HTML
|
58
|
+
<div id="flash-message" class="message #{class_name}">#{@message}</div>
|
59
|
+
<script>
|
60
|
+
// Automatically hide the flash message after 5 seconds
|
61
|
+
document.addEventListener('DOMContentLoaded', function() {
|
62
|
+
var flashMessage = document.getElementById('flash-message');
|
63
|
+
if (flashMessage) {
|
64
|
+
setTimeout(function() {
|
65
|
+
flashMessage.style.opacity = '1';
|
66
|
+
// Fade out animation
|
67
|
+
var fadeEffect = setInterval(function() {
|
68
|
+
if (!flashMessage.style.opacity) {
|
69
|
+
flashMessage.style.opacity = 1;
|
70
|
+
}
|
71
|
+
if (flashMessage.style.opacity > 0) {
|
72
|
+
flashMessage.style.opacity -= 0.1;
|
73
|
+
} else {
|
74
|
+
clearInterval(fadeEffect);
|
75
|
+
flashMessage.style.display = 'none';
|
76
|
+
}
|
77
|
+
}, 50);
|
78
|
+
}, 5000); // 5 seconds
|
79
|
+
}
|
80
|
+
});
|
81
|
+
</script>
|
82
|
+
HTML
|
58
83
|
end
|
59
84
|
|
60
85
|
def generate_header
|
@@ -200,6 +200,7 @@ module SolidQueueMonitor
|
|
200
200
|
padding: 1rem;
|
201
201
|
margin-bottom: 1rem;
|
202
202
|
border-radius: 0.375rem;
|
203
|
+
transition: opacity 0.5s ease-in-out;
|
203
204
|
}
|
204
205
|
|
205
206
|
.solid_queue_monitor .message-success {
|
@@ -318,12 +319,50 @@ module SolidQueueMonitor
|
|
318
319
|
}
|
319
320
|
}
|
320
321
|
|
322
|
+
.solid_queue_monitor .filter-and-actions-container {
|
323
|
+
display: flex;
|
324
|
+
justify-content: space-between;
|
325
|
+
align-items: flex-start;
|
326
|
+
gap: 1rem;
|
327
|
+
margin-bottom: 1rem;
|
328
|
+
}
|
329
|
+
|
321
330
|
.solid_queue_monitor .filter-form-container {
|
322
331
|
background: white;
|
323
332
|
padding: 1rem;
|
324
333
|
border-radius: 0.5rem;
|
325
|
-
margin-bottom: 1rem;
|
326
334
|
box-shadow: 0 1px 3px rgba(0,0,0,0.1);
|
335
|
+
flex: 3;
|
336
|
+
}
|
337
|
+
|
338
|
+
.solid_queue_monitor .bulk-actions-container {
|
339
|
+
display: flex;
|
340
|
+
flex-direction: row;
|
341
|
+
gap: 0.75rem;
|
342
|
+
padding: 1rem;
|
343
|
+
background: white;
|
344
|
+
border-radius: 0.5rem;
|
345
|
+
box-shadow: 0 1px 3px rgba(0,0,0,0.1);
|
346
|
+
flex: 2;
|
347
|
+
align-items: center;
|
348
|
+
justify-content: center;
|
349
|
+
}
|
350
|
+
|
351
|
+
.solid_queue_monitor .large-button {
|
352
|
+
padding: 0.75rem 1.25rem;
|
353
|
+
font-size: 0.9rem;
|
354
|
+
text-align: center;
|
355
|
+
flex: 1;
|
356
|
+
}
|
357
|
+
|
358
|
+
@media (max-width: 992px) {
|
359
|
+
.solid_queue_monitor .filter-and-actions-container {
|
360
|
+
flex-direction: column;
|
361
|
+
}
|
362
|
+
|
363
|
+
.solid_queue_monitor .bulk-actions-container {
|
364
|
+
width: 100%;
|
365
|
+
}
|
327
366
|
}
|
328
367
|
|
329
368
|
.solid_queue_monitor .filter-form {
|
@@ -390,6 +429,116 @@ module SolidQueueMonitor
|
|
390
429
|
.solid_queue_monitor .reset-button:hover {
|
391
430
|
background: #e5e7eb;
|
392
431
|
}
|
432
|
+
|
433
|
+
/* Action buttons for retry/discard */
|
434
|
+
.solid_queue_monitor .action-button {
|
435
|
+
padding: 0.5rem 1rem;
|
436
|
+
border-radius: 0.375rem;
|
437
|
+
font-size: 0.75rem;
|
438
|
+
font-weight: 500;
|
439
|
+
cursor: pointer;
|
440
|
+
transition: background-color 0.2s;
|
441
|
+
border: none;
|
442
|
+
text-decoration: none;
|
443
|
+
}
|
444
|
+
|
445
|
+
.solid_queue_monitor .retry-button {
|
446
|
+
background: #3b82f6;
|
447
|
+
color: white;
|
448
|
+
}
|
449
|
+
|
450
|
+
.solid_queue_monitor .retry-button:hover {
|
451
|
+
background: #2563eb;
|
452
|
+
}
|
453
|
+
|
454
|
+
.solid_queue_monitor .discard-button {
|
455
|
+
background: #ef4444;
|
456
|
+
color: white;
|
457
|
+
}
|
458
|
+
|
459
|
+
.solid_queue_monitor .discard-button:hover {
|
460
|
+
background: #dc2626;
|
461
|
+
}
|
462
|
+
|
463
|
+
.solid_queue_monitor .action-button:disabled {
|
464
|
+
opacity: 0.5;
|
465
|
+
cursor: not-allowed;
|
466
|
+
}
|
467
|
+
|
468
|
+
.solid_queue_monitor .inline-form {
|
469
|
+
display: inline-block;
|
470
|
+
margin-right: 0.5rem;
|
471
|
+
}
|
472
|
+
|
473
|
+
.solid_queue_monitor .actions-cell {
|
474
|
+
white-space: nowrap;
|
475
|
+
}
|
476
|
+
|
477
|
+
.solid_queue_monitor .bulk-actions {
|
478
|
+
display: flex;
|
479
|
+
gap: 0.5rem;
|
480
|
+
}
|
481
|
+
|
482
|
+
.solid_queue_monitor .error-message {
|
483
|
+
color: #dc2626;
|
484
|
+
font-weight: 500;
|
485
|
+
margin-bottom: 0.25rem;
|
486
|
+
}
|
487
|
+
|
488
|
+
.solid_queue_monitor .error-backtrace {
|
489
|
+
font-size: 0.75rem;
|
490
|
+
white-space: pre-wrap;
|
491
|
+
max-height: 200px;
|
492
|
+
overflow-y: auto;
|
493
|
+
background: #f3f4f6;
|
494
|
+
padding: 0.5rem;
|
495
|
+
border-radius: 0.25rem;
|
496
|
+
margin-top: 0.5rem;
|
497
|
+
}
|
498
|
+
|
499
|
+
.solid_queue_monitor details {
|
500
|
+
margin-top: 0.25rem;
|
501
|
+
}
|
502
|
+
|
503
|
+
.solid_queue_monitor summary {
|
504
|
+
cursor: pointer;
|
505
|
+
color: #6b7280;
|
506
|
+
font-size: 0.75rem;
|
507
|
+
}
|
508
|
+
|
509
|
+
.solid_queue_monitor summary:hover {
|
510
|
+
color: #4b5563;
|
511
|
+
}
|
512
|
+
|
513
|
+
.solid_queue_monitor .job-checkbox,
|
514
|
+
.solid_queue_monitor .select-all-checkbox {
|
515
|
+
width: 1rem;
|
516
|
+
height: 1rem;
|
517
|
+
}
|
518
|
+
|
519
|
+
.solid_queue_monitor .bulk-actions-bar {
|
520
|
+
display: flex;
|
521
|
+
gap: 0.75rem;
|
522
|
+
margin: 1rem 0;
|
523
|
+
background: white;
|
524
|
+
padding: 0.75rem;
|
525
|
+
border-radius: 0.5rem;
|
526
|
+
box-shadow: 0 1px 3px rgba(0,0,0,0.1);
|
527
|
+
}
|
528
|
+
|
529
|
+
.solid_queue_monitor .bulk-actions-bar .action-button {
|
530
|
+
padding: 0.6rem 1rem;
|
531
|
+
font-size: 0.875rem;
|
532
|
+
}
|
533
|
+
|
534
|
+
.solid_queue_monitor .execute-button {
|
535
|
+
background: var(--primary-color);
|
536
|
+
color: white;
|
537
|
+
}
|
538
|
+
|
539
|
+
.solid_queue_monitor .execute-button:hover {
|
540
|
+
background: #2563eb;
|
541
|
+
}
|
393
542
|
CSS
|
394
543
|
end
|
395
544
|
end
|
data/config/routes.rb
CHANGED
@@ -1,11 +1,17 @@
|
|
1
1
|
SolidQueueMonitor::Engine.routes.draw do
|
2
2
|
root to: 'monitor#index'
|
3
3
|
|
4
|
-
get 'ready_jobs', to: 'monitor#ready_jobs'
|
5
|
-
get 'scheduled_jobs', to: 'monitor#scheduled_jobs'
|
6
|
-
get 'failed_jobs', to: 'monitor#failed_jobs'
|
7
|
-
get 'recurring_jobs', to: 'monitor#recurring_jobs'
|
8
|
-
get 'queues', to: 'monitor#queues'
|
4
|
+
get 'ready_jobs', to: 'monitor#ready_jobs', as: 'ready_jobs'
|
5
|
+
get 'scheduled_jobs', to: 'monitor#scheduled_jobs', as: 'scheduled_jobs'
|
6
|
+
get 'failed_jobs', to: 'monitor#failed_jobs', as: 'failed_jobs'
|
7
|
+
get 'recurring_jobs', to: 'monitor#recurring_jobs', as: 'recurring_jobs'
|
8
|
+
get 'queues', to: 'monitor#queues', as: 'queues'
|
9
9
|
|
10
|
-
post 'execute_jobs', to: 'monitor#execute_jobs'
|
10
|
+
post 'execute_jobs', to: 'monitor#execute_jobs', as: 'execute_jobs'
|
11
|
+
|
12
|
+
# Failed job actions
|
13
|
+
post 'retry_failed_job/:id', to: 'monitor#retry_failed_job', as: 'retry_failed_job'
|
14
|
+
post 'discard_failed_job/:id', to: 'monitor#discard_failed_job', as: 'discard_failed_job'
|
15
|
+
post 'retry_failed_jobs', to: 'monitor#retry_failed_jobs', as: 'retry_failed_jobs'
|
16
|
+
post 'discard_failed_jobs', to: 'monitor#discard_failed_jobs', as: 'discard_failed_jobs'
|
11
17
|
end
|
metadata
CHANGED
@@ -1,13 +1,13 @@
|
|
1
1
|
--- !ruby/object:Gem::Specification
|
2
2
|
name: solid_queue_monitor
|
3
3
|
version: !ruby/object:Gem::Version
|
4
|
-
version: 0.1.
|
4
|
+
version: 0.1.2
|
5
5
|
platform: ruby
|
6
6
|
authors:
|
7
7
|
- Vishal Sadriya
|
8
8
|
bindir: exe
|
9
9
|
cert_chain: []
|
10
|
-
date: 2025-03-
|
10
|
+
date: 2025-03-18 00:00:00.000000000 Z
|
11
11
|
dependencies:
|
12
12
|
- !ruby/object:Gem::Dependency
|
13
13
|
name: rails
|
@@ -198,6 +198,7 @@ files:
|
|
198
198
|
- app/presenters/solid_queue_monitor/stats_presenter.rb
|
199
199
|
- app/services/solid_queue_monitor/authentication_service.rb
|
200
200
|
- app/services/solid_queue_monitor/execute_job_service.rb
|
201
|
+
- app/services/solid_queue_monitor/failed_job_service.rb
|
201
202
|
- app/services/solid_queue_monitor/html_generator.rb
|
202
203
|
- app/services/solid_queue_monitor/pagination_service.rb
|
203
204
|
- app/services/solid_queue_monitor/stats_calculator.rb
|