async-background 0.2.4 → 0.2.6
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/lib/async/background/min_heap.rb +5 -0
- data/lib/async/background/runner.rb +40 -8
- data/lib/async/background/version.rb +1 -1
- metadata +3 -3
checksums.yaml
CHANGED
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
---
|
|
2
2
|
SHA256:
|
|
3
|
-
metadata.gz:
|
|
4
|
-
data.tar.gz:
|
|
3
|
+
metadata.gz: 6668497aee969150ccc5e31b7c9ec57a47c2b107aa337686393262ee2b78f429
|
|
4
|
+
data.tar.gz: 3cbf76732b4987bfa5d7c9257b529c716e0adfec11e7d02f6a4105b5c5ce54e1
|
|
5
5
|
SHA512:
|
|
6
|
-
metadata.gz:
|
|
7
|
-
data.tar.gz:
|
|
6
|
+
metadata.gz: 5f75ed0c52dda7d988fbf73fa701c00f3c6ff3b869cd36d6566593c9a8095134721a011ada08c337206d64348f919ca807c77e07dd43c4c14f31c637f58769d4
|
|
7
|
+
data.tar.gz: 0b00eb1e333173929e788103042dcd2c3be4696da0acad65a2f5d6bc0bdc6e5bbeeccf0353ad27af7281d0d62aaf3342c1c9d1f8ce8594caee18dc28435e645a
|
|
@@ -12,13 +12,14 @@ module Async
|
|
|
12
12
|
MAX_JITTER = 5
|
|
13
13
|
|
|
14
14
|
class Runner
|
|
15
|
-
attr_reader :logger, :semaphore, :heap, :worker_index, :total_workers
|
|
15
|
+
attr_reader :logger, :semaphore, :heap, :worker_index, :total_workers, :shutdown
|
|
16
16
|
|
|
17
17
|
def initialize(config_path:, job_count: 2, worker_index:, total_workers:)
|
|
18
18
|
@logger = Console.logger
|
|
19
19
|
@worker_index = worker_index
|
|
20
20
|
@total_workers = total_workers
|
|
21
21
|
@running = true
|
|
22
|
+
@shutdown = ::Async::Condition.new
|
|
22
23
|
|
|
23
24
|
logger.info { "Async::Background worker_index=#{worker_index}/#{total_workers}, job_count=#{job_count}" }
|
|
24
25
|
|
|
@@ -28,26 +29,27 @@ module Async
|
|
|
28
29
|
|
|
29
30
|
def run
|
|
30
31
|
Async do |task|
|
|
32
|
+
setup_signal_handlers
|
|
33
|
+
start_signal_watcher(task)
|
|
34
|
+
|
|
31
35
|
loop do
|
|
32
36
|
entry = heap.peek
|
|
33
37
|
break unless entry
|
|
34
38
|
|
|
35
39
|
now = monotonic_now
|
|
36
40
|
wait = [entry.next_run_at - now, MIN_SLEEP_TIME].max
|
|
37
|
-
task
|
|
41
|
+
wait_with_shutdown(task, wait)
|
|
38
42
|
break unless running?
|
|
39
43
|
|
|
40
44
|
now = monotonic_now
|
|
41
|
-
while (
|
|
45
|
+
while (entry = heap.peek) && entry.next_run_at <= now
|
|
42
46
|
break unless running?
|
|
43
47
|
|
|
44
|
-
entry = heap.pop
|
|
45
|
-
|
|
46
48
|
if entry.running
|
|
47
49
|
logger.warn('Async::Background') { "#{entry.name}: skipped, previous run still active" }
|
|
48
50
|
else
|
|
49
51
|
entry.running = true
|
|
50
|
-
semaphore.async
|
|
52
|
+
semaphore.async do
|
|
51
53
|
run_job(task, entry)
|
|
52
54
|
ensure
|
|
53
55
|
entry.running = false
|
|
@@ -55,17 +57,20 @@ module Async
|
|
|
55
57
|
end
|
|
56
58
|
|
|
57
59
|
entry.reschedule(monotonic_now)
|
|
58
|
-
heap.
|
|
60
|
+
heap.replace_top(entry)
|
|
59
61
|
end
|
|
60
62
|
end
|
|
61
63
|
|
|
62
|
-
semaphore.
|
|
64
|
+
semaphore.acquire {}
|
|
63
65
|
end
|
|
64
66
|
end
|
|
65
67
|
|
|
66
68
|
def stop
|
|
69
|
+
return unless @running
|
|
70
|
+
|
|
67
71
|
@running = false
|
|
68
72
|
logger.info { "Async::Background: stopping gracefully" }
|
|
73
|
+
shutdown.signal
|
|
69
74
|
end
|
|
70
75
|
|
|
71
76
|
def running?
|
|
@@ -74,6 +79,33 @@ module Async
|
|
|
74
79
|
|
|
75
80
|
private
|
|
76
81
|
|
|
82
|
+
def setup_signal_handlers
|
|
83
|
+
@signal_r, @signal_w = IO.pipe
|
|
84
|
+
|
|
85
|
+
%w[INT TERM].each do |signal|
|
|
86
|
+
Signal.trap(signal) do
|
|
87
|
+
@running = false
|
|
88
|
+
@signal_w.write_nonblock('.') rescue nil
|
|
89
|
+
end
|
|
90
|
+
end
|
|
91
|
+
end
|
|
92
|
+
|
|
93
|
+
def start_signal_watcher(task)
|
|
94
|
+
task.async(transient: true) do
|
|
95
|
+
loop do
|
|
96
|
+
@signal_r.wait_readable
|
|
97
|
+
@signal_r.read_nonblock(256) rescue nil
|
|
98
|
+
shutdown.signal
|
|
99
|
+
break unless running?
|
|
100
|
+
end
|
|
101
|
+
end
|
|
102
|
+
end
|
|
103
|
+
|
|
104
|
+
def wait_with_shutdown(task, duration)
|
|
105
|
+
task.with_timeout(duration) { shutdown.wait }
|
|
106
|
+
rescue ::Async::TimeoutError
|
|
107
|
+
end
|
|
108
|
+
|
|
77
109
|
def build_heap(config_path)
|
|
78
110
|
raise ConfigError, "Schedule file not found: #{config_path}" unless File.exist?(config_path)
|
|
79
111
|
|
metadata
CHANGED
|
@@ -1,14 +1,14 @@
|
|
|
1
1
|
--- !ruby/object:Gem::Specification
|
|
2
2
|
name: async-background
|
|
3
3
|
version: !ruby/object:Gem::Version
|
|
4
|
-
version: 0.2.
|
|
4
|
+
version: 0.2.6
|
|
5
5
|
platform: ruby
|
|
6
6
|
authors:
|
|
7
7
|
- Roman Hajdarov
|
|
8
8
|
autorequire:
|
|
9
9
|
bindir: bin
|
|
10
10
|
cert_chain: []
|
|
11
|
-
date: 2026-03-
|
|
11
|
+
date: 2026-03-25 00:00:00.000000000 Z
|
|
12
12
|
dependencies:
|
|
13
13
|
- !ruby/object:Gem::Dependency
|
|
14
14
|
name: async
|
|
@@ -117,7 +117,7 @@ required_rubygems_version: !ruby/object:Gem::Requirement
|
|
|
117
117
|
- !ruby/object:Gem::Version
|
|
118
118
|
version: '0'
|
|
119
119
|
requirements: []
|
|
120
|
-
rubygems_version: 3.
|
|
120
|
+
rubygems_version: 3.3.27
|
|
121
121
|
signing_key:
|
|
122
122
|
specification_version: 4
|
|
123
123
|
summary: Lightweight heap-based cron/interval scheduler for Async.
|