Skip to content

Commit 226bbf9

Browse files
committed
Implement Sidekiq for background jobs and update rate limiting
- Add DeleteExpiredPayloadsJob for automatic payload expiration - Integrate Sidekiq for background job processing - Update rate limiting implementation in RateLimitable concern - Add tests for rate limiting and background jobs - Update documentation (README.md, kamal.md, progress.md) with Sidekiq details - Include instructions for running and deploying with Sidekiq
1 parent 07f8069 commit 226bbf9

File tree

35 files changed

+17182
-51
lines changed

35 files changed

+17182
-51
lines changed

Gemfile

+5-1
Original file line numberDiff line numberDiff line change
@@ -54,5 +54,9 @@ gem "importmap-rails"
5454
group :test do
5555
gem "simplecov", require: false
5656
gem "simplecov-cobertura"
57-
gem "fakeredis"
57+
gem "fakeredis", "~> 0.8.0"
5858
end
59+
60+
# Add these lines
61+
gem "sidekiq"
62+
gem "sidekiq-scheduler"

Gemfile.lock

+17-4
Original file line numberDiff line numberDiff line change
@@ -102,7 +102,8 @@ GEM
102102
factory_bot_rails (6.4.3)
103103
factory_bot (~> 6.4)
104104
railties (>= 5.0.0)
105-
fakeredis (0.1.4)
105+
fakeredis (0.8.0)
106+
redis (~> 4.1)
106107
fugit (1.11.1)
107108
et-orbi (~> 1, >= 1.2.11)
108109
raabro (~> 1.4)
@@ -198,8 +199,7 @@ GEM
198199
rake (13.2.1)
199200
rdoc (6.7.0)
200201
psych (>= 4.0.0)
201-
redis (5.3.0)
202-
redis-client (>= 0.22.0)
202+
redis (4.8.1)
203203
redis-client (0.22.2)
204204
connection_pool
205205
regexp_parser (2.9.2)
@@ -259,6 +259,16 @@ GEM
259259
securerandom (0.3.1)
260260
shoulda-matchers (6.4.0)
261261
activesupport (>= 5.2.0)
262+
sidekiq (7.3.1)
263+
concurrent-ruby (< 2)
264+
connection_pool (>= 2.3.0)
265+
logger
266+
rack (>= 2.2.4)
267+
redis-client (>= 0.22.2)
268+
sidekiq-scheduler (5.0.6)
269+
rufus-scheduler (~> 3.2)
270+
sidekiq (>= 6, < 8)
271+
tilt (>= 1.4.0, < 3)
262272
simplecov (0.22.0)
263273
docile (~> 1.1)
264274
simplecov-html (~> 0.11)
@@ -280,6 +290,7 @@ GEM
280290
stringio (3.1.1)
281291
strscan (3.1.0)
282292
thor (1.3.1)
293+
tilt (2.4.0)
283294
timeout (0.4.1)
284295
tzinfo (2.0.6)
285296
concurrent-ruby (~> 1.0)
@@ -304,7 +315,7 @@ DEPENDENCIES
304315
debug
305316
dotenv-rails
306317
factory_bot_rails
307-
fakeredis
318+
fakeredis (~> 0.8.0)
308319
importmap-rails
309320
puma
310321
rails (~> 7.2.1)
@@ -314,6 +325,8 @@ DEPENDENCIES
314325
rubocop-rails-omakase
315326
rufus-scheduler
316327
shoulda-matchers
328+
sidekiq
329+
sidekiq-scheduler
317330
simplecov
318331
simplecov-cobertura
319332
sprockets-rails

README.md

+7
Original file line numberDiff line numberDiff line change
@@ -58,6 +58,13 @@ bundle exec rspec # check test coverage
5858
bundle exec rails test # check unit test coverage
5959
rails server # to locally start the server
6060
```
61+
## Running service
62+
For local run you can use below command
63+
```bash
64+
rails server # to start the server
65+
bundle exec sidekiq # to start sidekiq
66+
```
67+
6168
## Deployment
6269

6370
This project uses Kamal for deployment. Follow these steps to deploy:

app/controllers/concerns/rate_limitable.rb

+12-9
Original file line numberDiff line numberDiff line change
@@ -8,19 +8,22 @@ module RateLimitable
88
private
99

1010
def check_rate_limit
11-
key = "rate_limit:#{request.ip}:#{controller_name}:#{action_name}"
12-
count = REDIS.get(key).to_i
11+
current_count = REDIS.get(rate_limit_key).to_i
1312

14-
if count >= rate_limit
15-
render json: { error: "Rate limit exceeded" }, status: :too_many_requests
16-
else
17-
REDIS.multi do
18-
REDIS.incr(key)
19-
REDIS.expire(key, 1.hour.to_i)
20-
end
13+
REDIS.multi do |pipeline|
14+
pipeline.incr(rate_limit_key)
15+
pipeline.expire(rate_limit_key, 1.hour.to_i)
16+
end
17+
18+
if current_count >= rate_limit
19+
render json: { error: "Rate limit exceeded. Try again later." }, status: :too_many_requests
2120
end
2221
end
2322

23+
def rate_limit_key
24+
"#{controller_name}:#{action_name}:#{request.ip}"
25+
end
26+
2427
def rate_limit
2528
case action_name.to_sym
2629
when :create
+7
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,7 @@
1+
class DeleteExpiredPayloadsJob < ApplicationJob
2+
queue_as :default
3+
4+
def perform
5+
Payload.where("expiry_time <= ?", Time.current).destroy_all
6+
end
7+
end

config/application.rb

+3
Original file line numberDiff line numberDiff line change
@@ -35,5 +35,8 @@ class Application < Rails::Application
3535
config.assets.compile = false
3636
config.assets.precompile += %w[ manifest.js ]
3737
config.public_file_server.enabled = true
38+
39+
# Use Sidekiq as the job queue adapter
40+
config.active_job.queue_adapter = :sidekiq
3841
end
3942
end

config/initializers/sidekiq.rb

+26
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,26 @@
1+
require "sidekiq"
2+
require "sidekiq-scheduler"
3+
4+
redis_config = if Rails.env.test?
5+
{ url: "redis://localhost:6379/0" }
6+
else
7+
{ url: ENV["REDIS_URL"] || "redis://localhost:6379/0" }
8+
end
9+
10+
Sidekiq.configure_server do |config|
11+
config.redis = redis_config
12+
13+
if Rails.env.production?
14+
config.on(:startup) do
15+
schedule_file = File.expand_path("../../sidekiq.yml", __FILE__)
16+
if File.exist?(schedule_file)
17+
Sidekiq.schedule = YAML.load_file(schedule_file)
18+
SidekiqScheduler::Scheduler.instance.reload_schedule!
19+
end
20+
end
21+
end
22+
end
23+
24+
Sidekiq.configure_client do |config|
25+
config.redis = redis_config
26+
end

config/routes.rb

+6
Original file line numberDiff line numberDiff line change
@@ -1,3 +1,6 @@
1+
require "sidekiq/web"
2+
require "sidekiq-scheduler/web"
3+
14
Rails.application.routes.draw do
25
# Define your application routes per the DSL in https://guides.rubyonrails.org/routing.html
36

@@ -15,4 +18,7 @@
1518
end
1619

1720
resources :payloads, only: [ :index, :show, :create, :update ]
21+
22+
# Mount Sidekiq web interface
23+
mount Sidekiq::Web => "/sidekiq"
1824
end

config/sidekiq.yml

+4
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,4 @@
1+
:schedule:
2+
delete_expired_payloads:
3+
cron: '0 * * * *' # Runs every hour
4+
class: DeleteExpiredPayloadsJob

docker-compose.yml

+13-2
Original file line numberDiff line numberDiff line change
@@ -9,7 +9,18 @@ services:
99
- REDIS_URL=redis://redis:6379/0
1010
depends_on:
1111
- redis
12+
sidekiq:
13+
build: .
14+
command: bundle exec sidekiq
15+
volumes:
16+
- .:/app
17+
depends_on:
18+
- redis
19+
env_file:
20+
- .env
1221
redis:
1322
image: redis:7-alpine
14-
ports:
15-
- "6379:6379"
23+
volumes:
24+
- redis:/data
25+
volumes:
26+
redis:
Original file line numberDiff line numberDiff line change
@@ -0,0 +1 @@
1+
{"files":{"manifest-e3b0c44298fc1c149afbf4c8996fb92427ae41e4649b934ca495991b7852b855.js":{"logical_path":"manifest.js","mtime":"2024-08-30T15:01:58+05:30","size":0,"digest":"e3b0c44298fc1c149afbf4c8996fb92427ae41e4649b934ca495991b7852b855","integrity":"sha256-47DEQpj8HBSa+/TImW+5JCeuQeRkm5NMpJWZG3hSuFU="},"actiontext-338257afa53cc53f94018abf7a897120d1eb3ee71f39e4641b2058442c9c4dc6.js":{"logical_path":"actiontext.js","mtime":"2024-08-30T15:01:58+05:30","size":30336,"digest":"338257afa53cc53f94018abf7a897120d1eb3ee71f39e4641b2058442c9c4dc6","integrity":"sha256-M4JXr6U8xT+UAYq/eolxINHrPucfOeRkGyBYRCycTcY="},"actiontext.esm-e787c6f528f0a88ee04de7c5ce4e3ab37fcf2691629ed21f0b30ee2db848c8c8.js":{"logical_path":"actiontext.esm.js","mtime":"2024-08-30T15:01:58+05:30","size":28519,"digest":"e787c6f528f0a88ee04de7c5ce4e3ab37fcf2691629ed21f0b30ee2db848c8c8","integrity":"sha256-54fG9SjwqI7gTefFzk46s3/PJpFintIfCzDuLbhIyMg="},"trix-fc94199b9d6e67e0780ad852719d0527834679c98a039ee492dedaae29f902df.js":{"logical_path":"trix.js","mtime":"2024-08-30T15:01:58+05:30","size":439967,"digest":"fc94199b9d6e67e0780ad852719d0527834679c98a039ee492dedaae29f902df","integrity":"sha256-/JQZm51uZ+B4CthScZ0FJ4NGecmKA57kkt7arin5At8="},"trix-4c7f56767699ae6c08f21004c9dc23a1b76b0734b0fca61d4534900ee0fef211.css":{"logical_path":"trix.css","mtime":"2024-08-30T15:01:58+05:30","size":20027,"digest":"4c7f56767699ae6c08f21004c9dc23a1b76b0734b0fca61d4534900ee0fef211","integrity":"sha256-TH9WdnaZrmwI8hAEydwjobdrBzSw/KYdRTSQDuD+8hE="},"activestorage-7dbadea816242c6d03cc2f59c58398ab55bb9a70ee859fa962ad4c7ad4c48594.js":{"logical_path":"activestorage.js","mtime":"2024-08-30T15:01:58+05:30","size":29379,"digest":"7dbadea816242c6d03cc2f59c58398ab55bb9a70ee859fa962ad4c7ad4c48594","integrity":"sha256-fbreqBYkLG0DzC9ZxYOYq1W7mnDuhZ+pYq1MetTEhZQ="},"activestorage.esm-b2d9aa45888a540d7e6969e484dbdb0a0a501ff56542ce737db6599b311e3385.js":{"logical_path":"activestorage.esm.js","mtime":"2024-08-30T15:01:58+05:30","size":27247,"digest":"b2d9aa45888a540d7e6969e484dbdb0a0a501ff56542ce737db6599b311e3385","integrity":"sha256-stmqRYiKVA1+aWnkhNvbCgpQH/VlQs5zfbZZmzEeM4U="},"actioncable-1a239ff96644bc8fd626f87a74175b4ee7eafd201f5034034a433c22d8c4dc3e.js":{"logical_path":"actioncable.js","mtime":"2024-08-30T15:01:58+05:30","size":16474,"digest":"1a239ff96644bc8fd626f87a74175b4ee7eafd201f5034034a433c22d8c4dc3e","integrity":"sha256-GiOf+WZEvI/WJvh6dBdbTufq/SAfUDQDSkM8ItjE3D4="},"actioncable.esm-555679e44f119be7a7e54121fca0beb0a5832f2933c1a1edcd0515bbafad22e0.js":{"logical_path":"actioncable.esm.js","mtime":"2024-08-30T15:01:58+05:30","size":14813,"digest":"555679e44f119be7a7e54121fca0beb0a5832f2933c1a1edcd0515bbafad22e0","integrity":"sha256-VVZ55E8Rm+en5UEh/KC+sKWDLykzwaHtzQUVu6+tIuA="}},"assets":{"manifest.js":"manifest-e3b0c44298fc1c149afbf4c8996fb92427ae41e4649b934ca495991b7852b855.js","actiontext.js":"actiontext-338257afa53cc53f94018abf7a897120d1eb3ee71f39e4641b2058442c9c4dc6.js","actiontext.esm.js":"actiontext.esm-e787c6f528f0a88ee04de7c5ce4e3ab37fcf2691629ed21f0b30ee2db848c8c8.js","trix.js":"trix-fc94199b9d6e67e0780ad852719d0527834679c98a039ee492dedaae29f902df.js","trix.css":"trix-4c7f56767699ae6c08f21004c9dc23a1b76b0734b0fca61d4534900ee0fef211.css","activestorage.js":"activestorage-7dbadea816242c6d03cc2f59c58398ab55bb9a70ee859fa962ad4c7ad4c48594.js","activestorage.esm.js":"activestorage.esm-b2d9aa45888a540d7e6969e484dbdb0a0a501ff56542ce737db6599b311e3385.js","actioncable.js":"actioncable-1a239ff96644bc8fd626f87a74175b4ee7eafd201f5034034a433c22d8c4dc3e.js","actioncable.esm.js":"actioncable.esm-555679e44f119be7a7e54121fca0beb0a5832f2933c1a1edcd0515bbafad22e0.js"}}

0 commit comments

Comments
 (0)