daemon_controller 1.2.0 → 2.0.0
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 +5 -13
- data/LICENSE.txt +1 -2
- data/{README.markdown → README.md} +40 -76
- data/Rakefile +16 -574
- data/daemon_controller.gemspec +5 -4
- data/lib/daemon_controller/lock_file.rb +106 -105
- data/lib/daemon_controller/packaging.rb +10 -16
- data/lib/daemon_controller/spawn.rb +8 -7
- data/lib/daemon_controller/version.rb +10 -8
- data/lib/daemon_controller.rb +941 -851
- data/spec/daemon_controller_spec.rb +572 -409
- data/spec/echo_server.rb +139 -115
- data/spec/test_helper.rb +132 -105
- data/spec/unresponsive_daemon.rb +13 -7
- metadata +7 -20
- checksums.yaml.gz.asc +0 -12
- data/debian.template/changelog +0 -5
- data/debian.template/compat +0 -1
- data/debian.template/control.template +0 -15
- data/debian.template/copyright +0 -24
- data/debian.template/ruby-daemon-controller.install +0 -2
- data/debian.template/rules +0 -17
- data/debian.template/source/format +0 -1
- data/rpm/get_distro_id.py +0 -4
- data/rpm/rubygem-daemon_controller.spec.template +0 -102
- data.tar.gz.asc +0 -12
- metadata.gz.asc +0 -12
@@ -1,424 +1,587 @@
|
|
1
|
-
|
2
|
-
|
3
|
-
|
4
|
-
require
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
require_relative "test_helper"
|
4
|
+
require "daemon_controller"
|
5
|
+
require "benchmark"
|
6
|
+
require "socket"
|
7
|
+
require "tmpdir"
|
8
|
+
require "shellwords"
|
5
9
|
|
6
10
|
describe DaemonController, "#start" do
|
7
|
-
|
8
|
-
|
9
|
-
|
10
|
-
|
11
|
-
|
12
|
-
|
13
|
-
|
14
|
-
|
15
|
-
|
16
|
-
|
17
|
-
|
18
|
-
|
19
|
-
|
20
|
-
|
21
|
-
|
22
|
-
|
23
|
-
|
24
|
-
|
25
|
-
|
26
|
-
|
27
|
-
|
28
|
-
|
29
|
-
|
30
|
-
|
31
|
-
|
32
|
-
|
33
|
-
|
34
|
-
|
35
|
-
|
36
|
-
|
37
|
-
|
38
|
-
|
39
|
-
|
40
|
-
|
41
|
-
|
42
|
-
|
43
|
-
|
44
|
-
|
45
|
-
|
46
|
-
|
47
|
-
|
48
|
-
|
49
|
-
|
50
|
-
|
51
|
-
|
52
|
-
|
53
|
-
|
54
|
-
|
55
|
-
|
56
|
-
|
57
|
-
|
58
|
-
|
59
|
-
|
60
|
-
|
61
|
-
|
62
|
-
|
63
|
-
|
64
|
-
|
65
|
-
|
66
|
-
|
67
|
-
|
68
|
-
|
69
|
-
|
70
|
-
|
71
|
-
|
72
|
-
|
73
|
-
|
74
|
-
|
75
|
-
|
76
|
-
|
77
|
-
|
78
|
-
|
79
|
-
|
80
|
-
|
81
|
-
|
82
|
-
|
83
|
-
|
84
|
-
|
85
|
-
|
86
|
-
|
87
|
-
|
88
|
-
|
89
|
-
|
90
|
-
|
91
|
-
|
92
|
-
|
93
|
-
|
94
|
-
|
95
|
-
|
96
|
-
|
97
|
-
|
98
|
-
|
99
|
-
|
100
|
-
|
101
|
-
|
102
|
-
|
103
|
-
|
104
|
-
|
105
|
-
|
106
|
-
|
107
|
-
|
108
|
-
|
109
|
-
|
110
|
-
|
111
|
-
|
112
|
-
|
113
|
-
|
114
|
-
|
115
|
-
|
116
|
-
|
117
|
-
|
118
|
-
|
119
|
-
|
120
|
-
|
121
|
-
|
122
|
-
|
123
|
-
|
124
|
-
|
125
|
-
|
126
|
-
|
127
|
-
|
128
|
-
|
129
|
-
|
130
|
-
|
131
|
-
|
132
|
-
|
133
|
-
|
134
|
-
|
135
|
-
|
136
|
-
|
137
|
-
|
138
|
-
|
139
|
-
|
140
|
-
|
141
|
-
|
142
|
-
|
143
|
-
|
144
|
-
|
145
|
-
|
146
|
-
|
147
|
-
|
148
|
-
|
149
|
-
|
150
|
-
|
151
|
-
|
152
|
-
|
153
|
-
|
154
|
-
|
155
|
-
|
156
|
-
|
157
|
-
|
158
|
-
|
159
|
-
|
160
|
-
|
161
|
-
|
162
|
-
|
163
|
-
|
164
|
-
|
165
|
-
|
166
|
-
|
167
|
-
|
168
|
-
|
169
|
-
|
170
|
-
|
171
|
-
|
172
|
-
|
173
|
-
|
174
|
-
|
175
|
-
|
176
|
-
|
177
|
-
|
178
|
-
|
179
|
-
|
180
|
-
|
181
|
-
|
182
|
-
|
183
|
-
|
184
|
-
|
185
|
-
|
186
|
-
|
187
|
-
|
188
|
-
|
189
|
-
|
190
|
-
|
191
|
-
|
192
|
-
|
193
|
-
|
194
|
-
|
195
|
-
|
196
|
-
|
197
|
-
|
198
|
-
|
199
|
-
|
200
|
-
|
201
|
-
|
202
|
-
|
203
|
-
|
204
|
-
|
205
|
-
|
206
|
-
|
207
|
-
|
208
|
-
|
209
|
-
|
210
|
-
|
211
|
-
|
212
|
-
|
213
|
-
|
214
|
-
|
215
|
-
|
216
|
-
|
217
|
-
|
218
|
-
|
219
|
-
|
220
|
-
|
221
|
-
|
222
|
-
|
223
|
-
|
224
|
-
|
225
|
-
|
226
|
-
|
227
|
-
|
228
|
-
|
229
|
-
|
230
|
-
|
231
|
-
|
232
|
-
|
233
|
-
|
234
|
-
|
235
|
-
|
236
|
-
|
237
|
-
|
238
|
-
|
239
|
-
|
240
|
-
|
241
|
-
|
242
|
-
|
243
|
-
|
244
|
-
|
11
|
+
include TestHelper
|
12
|
+
|
13
|
+
after :each do
|
14
|
+
@controller.stop if @controller
|
15
|
+
end
|
16
|
+
|
17
|
+
it "works" do
|
18
|
+
new_controller
|
19
|
+
@controller.start
|
20
|
+
expect(ping_echo_server).to be(true)
|
21
|
+
end
|
22
|
+
|
23
|
+
it "raises AlreadyStarted if the daemon is already running" do
|
24
|
+
new_controller
|
25
|
+
begin
|
26
|
+
expect(@controller).to receive(:daemon_is_running?).and_return(true)
|
27
|
+
expect { @controller.start }.to raise_error(DaemonController::AlreadyStarted)
|
28
|
+
ensure
|
29
|
+
@controller = nil # Don't invoke @controller.stop in after hook
|
30
|
+
end
|
31
|
+
end
|
32
|
+
|
33
|
+
it "deletes existing PID file before starting the daemon" do
|
34
|
+
write_file("spec/echo_server.pid", "1234")
|
35
|
+
new_controller
|
36
|
+
expect(@controller).to receive(:daemon_is_running?).and_return(false)
|
37
|
+
expect(@controller).to receive(:spawn_daemon)
|
38
|
+
expect(@controller).to receive(:pid_file_available?).and_return(true)
|
39
|
+
expect(@controller).to receive(:run_ping_command).at_least(:once).and_return(true)
|
40
|
+
@controller.start
|
41
|
+
expect(File.exist?("spec/echo_server.pid")).to be false
|
42
|
+
ensure
|
43
|
+
@controller = nil # Don't invoke @controller.stop in after hook
|
44
|
+
end
|
45
|
+
|
46
|
+
it "blocks until the daemon has written to its PID file" do
|
47
|
+
thread = WaitingThread.new do
|
48
|
+
sleep 0.15
|
49
|
+
write_file("spec/echo_server.pid", "1234")
|
50
|
+
end
|
51
|
+
new_controller
|
52
|
+
expect(@controller).to receive(:daemon_is_running?) { false }
|
53
|
+
expect(@controller).to receive(:spawn_daemon) { thread.go! }
|
54
|
+
expect(@controller).to receive(:run_ping_command).at_least(:once).and_return(true)
|
55
|
+
begin
|
56
|
+
result = Benchmark.measure do
|
57
|
+
@controller.start
|
58
|
+
end
|
59
|
+
expect(result.real).to be_between(0.15, 0.30)
|
60
|
+
ensure
|
61
|
+
thread.join
|
62
|
+
end
|
63
|
+
ensure
|
64
|
+
@controller = nil # Don't invoke @controller.stop in after hook
|
65
|
+
end
|
66
|
+
|
67
|
+
it "blocks until the daemon can be pinged" do
|
68
|
+
ping_ok = false
|
69
|
+
running = false
|
70
|
+
thread = WaitingThread.new do
|
71
|
+
sleep 0.15
|
72
|
+
ping_ok = true
|
73
|
+
end
|
74
|
+
new_controller
|
75
|
+
expect(@controller).to receive(:daemon_is_running?).at_least(:once) { running }
|
76
|
+
expect(@controller).to receive(:spawn_daemon) {
|
77
|
+
thread.go!
|
78
|
+
running = true
|
79
|
+
}
|
80
|
+
expect(@controller).to receive(:pid_file_available?).and_return(true)
|
81
|
+
expect(@controller).to receive(:run_ping_command).at_least(:once) { ping_ok }
|
82
|
+
begin
|
83
|
+
result = Benchmark.measure do
|
84
|
+
@controller.start
|
85
|
+
end
|
86
|
+
expect(result.real).to be_between(0.15, 0.30)
|
87
|
+
ensure
|
88
|
+
thread.join
|
89
|
+
end
|
90
|
+
ensure
|
91
|
+
@controller = nil # Don't invoke @controller.stop in after hook
|
92
|
+
end
|
93
|
+
|
94
|
+
it "works when the log file is not a regular file" do
|
95
|
+
new_controller(log_file: "/dev/stderr")
|
96
|
+
@controller.start
|
97
|
+
expect(ping_echo_server).to be(true)
|
98
|
+
end
|
99
|
+
|
100
|
+
context "if the daemon doesn't start in time" do
|
101
|
+
let(:start_timeout) { exec_is_slow? ? 4 : 0.5 }
|
102
|
+
let(:min_start_timeout) { 0.5 }
|
103
|
+
let(:max_start_timeout) { exec_is_slow? ? 6 : 1 }
|
104
|
+
|
105
|
+
it "raises StartTimeout" do
|
106
|
+
new_controller(start_command: "sleep 2", start_timeout: start_timeout)
|
107
|
+
start_time = Time.now
|
108
|
+
end_time = nil
|
109
|
+
expect(@controller).to receive(:start_timed_out) { end_time = Time.now }
|
110
|
+
expect { @controller.start }.to raise_error(DaemonController::StartTimeout)
|
111
|
+
expect(end_time - start_time).to be_between(min_start_timeout, max_start_timeout)
|
112
|
+
end
|
113
|
+
|
114
|
+
it "reports logs written to stderr if the timeout happened before forking" do
|
115
|
+
new_controller(start_command: "echo hello world; sleep 10", start_timeout: start_timeout)
|
116
|
+
expect(@controller).to receive(:daemonization_timed_out)
|
117
|
+
begin
|
118
|
+
@controller.start
|
119
|
+
fail
|
120
|
+
rescue DaemonController::StartTimeout => e
|
121
|
+
expect(e.message).to include("hello world")
|
122
|
+
end
|
123
|
+
end
|
124
|
+
|
125
|
+
it "reports logs written to the log file if the timeout happened before forking" do
|
126
|
+
new_controller(log_message2: "hello world",
|
127
|
+
wait2: 10,
|
128
|
+
start_timeout: start_timeout,
|
129
|
+
no_daemonize: true)
|
130
|
+
expect(@controller).to receive(:daemonization_timed_out)
|
131
|
+
begin
|
132
|
+
@controller.start
|
133
|
+
fail
|
134
|
+
rescue DaemonController::StartTimeout => e
|
135
|
+
expect(e.message).to include("hello world")
|
136
|
+
end
|
137
|
+
end
|
138
|
+
|
139
|
+
it "reports logs if the timeout happened after forking" do
|
140
|
+
new_controller(log_message2: "hello world",
|
141
|
+
wait2: 10,
|
142
|
+
start_timeout: start_timeout)
|
143
|
+
expect(@controller).not_to receive(:daemonization_timed_out)
|
144
|
+
begin
|
145
|
+
@controller.start
|
146
|
+
fail
|
147
|
+
rescue DaemonController::StartTimeout => e
|
148
|
+
expect(e.message).to include("hello world")
|
149
|
+
end
|
150
|
+
end
|
151
|
+
|
152
|
+
specify "if there are no logs, then the error says so" do
|
153
|
+
new_controller(start_command: "sleep 10", start_timeout: start_timeout)
|
154
|
+
expect(@controller).to receive(:daemonization_timed_out)
|
155
|
+
begin
|
156
|
+
@controller.start
|
157
|
+
fail
|
158
|
+
rescue DaemonController::StartTimeout => e
|
159
|
+
expect(e.message).to eq("(logs empty; timed out)")
|
160
|
+
end
|
161
|
+
end
|
162
|
+
|
163
|
+
specify "if logs cannot be captured, then the error says so" do
|
164
|
+
new_controller(start_command: "sleep 10", start_timeout: start_timeout, log_file: "/dev/stderr")
|
165
|
+
expect(@controller).to receive(:daemonization_timed_out)
|
166
|
+
begin
|
167
|
+
@controller.start
|
168
|
+
fail
|
169
|
+
rescue DaemonController::StartTimeout => e
|
170
|
+
expect(e.message).to eq("(logs not available; timed out)")
|
171
|
+
end
|
172
|
+
end
|
173
|
+
end
|
174
|
+
|
175
|
+
it "kills the daemon forcefully if the daemon has forked but doesn't " \
|
176
|
+
"become pingable in time, and there's a PID file" do
|
177
|
+
new_controller(wait2: 3, start_timeout: 1)
|
178
|
+
pid = nil
|
179
|
+
expect(@controller).to receive(:start_timed_out) {
|
180
|
+
@controller.send(:wait_until) do
|
181
|
+
@controller.send(:pid_file_available?)
|
182
|
+
end
|
183
|
+
pid = @controller.send(:read_pid_file)
|
184
|
+
}
|
185
|
+
begin
|
186
|
+
block = lambda { @controller.start }
|
187
|
+
expect(&block).to raise_error(DaemonController::StartTimeout)
|
188
|
+
eventually(1) do
|
189
|
+
!process_is_alive?(pid)
|
190
|
+
end
|
191
|
+
|
192
|
+
# The daemon should not be able to clean up its PID file since
|
193
|
+
# it's killed with SIGKILL.
|
194
|
+
expect(File.exist?("spec/echo_server.pid")).to be true
|
195
|
+
ensure
|
196
|
+
begin
|
197
|
+
File.unlink("spec/echo_server.pid")
|
198
|
+
rescue
|
199
|
+
nil
|
200
|
+
end
|
201
|
+
end
|
202
|
+
end
|
203
|
+
|
204
|
+
it "kills the daemon if it doesn't start in time and hasn't forked yet" do
|
205
|
+
new_controller(start_command: "./spec/unresponsive_daemon.rb",
|
206
|
+
start_timeout: 0.2)
|
207
|
+
pid = nil
|
208
|
+
expect(@controller).to receive(:daemonization_timed_out) {
|
209
|
+
@controller.send(:wait_until) do
|
210
|
+
@controller.send(:pid_file_available?)
|
211
|
+
end
|
212
|
+
pid = @controller.send(:read_pid_file)
|
213
|
+
}
|
214
|
+
block = lambda { @controller.start }
|
215
|
+
expect(&block).to raise_error(DaemonController::StartTimeout, /logs empty/)
|
216
|
+
eventually(1) do
|
217
|
+
!process_is_alive?(pid)
|
218
|
+
end
|
219
|
+
ensure
|
220
|
+
begin
|
221
|
+
File.unlink("spec/echo_server.pid")
|
222
|
+
rescue
|
223
|
+
nil
|
224
|
+
end
|
225
|
+
end
|
226
|
+
|
227
|
+
it "raises an error if the daemon exits with an error before forking" do
|
228
|
+
new_controller(start_command: "false")
|
229
|
+
expect { @controller.start }.to raise_error(DaemonController::StartError)
|
230
|
+
end
|
231
|
+
|
232
|
+
it "raises an error if the daemon exits with an error after forking" do
|
233
|
+
new_controller(crash_before_bind: true)
|
234
|
+
expect { @controller.start }.to raise_error(DaemonController::StartError)
|
235
|
+
end
|
236
|
+
|
237
|
+
specify "if the daemon exits after forking without writing a PID file, then this is detected through log file inactivity" do
|
238
|
+
new_controller(log_message1: "hello",
|
239
|
+
log_message2: "world",
|
240
|
+
no_write_pid_file: true,
|
241
|
+
log_file_activity_timeout: 0.5)
|
242
|
+
begin
|
243
|
+
@controller.start
|
244
|
+
fail
|
245
|
+
rescue DaemonController::StartTimeout => e
|
246
|
+
expect(e.message).to include("hello")
|
247
|
+
expect(e.message).to include("world")
|
248
|
+
expect(e.message).to include("(timed out)")
|
249
|
+
ensure
|
250
|
+
# Kill echo_server without PID file
|
251
|
+
kill_and_wait_echo_server
|
252
|
+
end
|
253
|
+
end
|
254
|
+
|
255
|
+
def find_echo_server_pid
|
256
|
+
process_line = `ps aux`.lines.grep(/echo_server\.rb/).first
|
257
|
+
process_line.split[1].to_i if process_line
|
258
|
+
end
|
259
|
+
|
260
|
+
def kill_and_wait_echo_server
|
261
|
+
pid = find_echo_server_pid
|
262
|
+
if pid
|
263
|
+
Process.kill("SIGTERM", pid)
|
264
|
+
Timeout.timeout(5) do
|
265
|
+
while find_echo_server_pid
|
266
|
+
sleep(0.1)
|
267
|
+
end
|
268
|
+
end
|
269
|
+
end
|
270
|
+
end
|
271
|
+
|
272
|
+
specify "the daemon's logs before forking is made available in the exception" do
|
273
|
+
new_controller(start_command: "(echo hello world; false)")
|
274
|
+
begin
|
275
|
+
@controller.start
|
276
|
+
rescue DaemonController::StartError => e
|
277
|
+
expect(e.message).to eq("hello world\n(exited with status 1)")
|
278
|
+
end
|
279
|
+
end
|
280
|
+
|
281
|
+
specify "the daemon's logs after forking is made available in the exception" do
|
282
|
+
new_controller(crash_before_bind: true)
|
283
|
+
begin
|
284
|
+
@controller.start
|
285
|
+
fail
|
286
|
+
rescue DaemonController::StartError => e
|
287
|
+
expect(e.message).to include("crashing, as instructed")
|
288
|
+
end
|
289
|
+
end
|
290
|
+
|
291
|
+
specify "the daemon's exit signal is made available in the exception if the log file is a regular file" do
|
292
|
+
new_controller(crash_before_bind: true, crash_signal: "SIGXCPU", no_daemonize: true)
|
293
|
+
begin
|
294
|
+
@controller.start
|
295
|
+
fail
|
296
|
+
rescue DaemonController::StartError => e
|
297
|
+
expect(e.message).to include("crashing, as instructed")
|
298
|
+
expect(e.message).to include("SIGXCPU")
|
299
|
+
end
|
300
|
+
end
|
301
|
+
|
302
|
+
specify "the daemon's exit signal is made available in the exception if the log file is not a regular file" do
|
303
|
+
new_controller(crash_before_bind: true,
|
304
|
+
crash_signal: "SIGXCPU",
|
305
|
+
no_daemonize: true,
|
306
|
+
log_file: "/dev/stderr")
|
307
|
+
begin
|
308
|
+
@controller.start
|
309
|
+
fail
|
310
|
+
rescue DaemonController::StartError => e
|
311
|
+
expect(e.message).to include("logs not available")
|
312
|
+
expect(e.message).to include("SIGXCPU")
|
313
|
+
end
|
314
|
+
end
|
315
|
+
|
316
|
+
specify "the daemon's logs are not available if the log file is not a regular file" do
|
317
|
+
new_controller(crash_before_bind: true, log_file: "/dev/stderr")
|
318
|
+
begin
|
319
|
+
@controller.start
|
320
|
+
fail
|
321
|
+
rescue DaemonController::StartError => e
|
322
|
+
expect(e.message).to eq("(logs not available)")
|
323
|
+
end
|
324
|
+
end
|
325
|
+
|
326
|
+
specify "if the logs are available but empty, then the error exception says so" do
|
327
|
+
new_controller(start_command: "exit 1")
|
328
|
+
begin
|
329
|
+
@controller.start
|
330
|
+
fail
|
331
|
+
rescue DaemonController::StartError => e
|
332
|
+
expect(e.message).to eq("(logs empty; exited with status 1)")
|
333
|
+
end
|
334
|
+
end
|
335
|
+
|
336
|
+
specify "the start command may be a Proc" do
|
337
|
+
called = true
|
338
|
+
new_controller(start_command: lambda {
|
339
|
+
called = true
|
340
|
+
@start_command
|
341
|
+
})
|
342
|
+
@controller.start
|
343
|
+
expect(called).to be true
|
344
|
+
end
|
345
|
+
|
346
|
+
specify "if the start command is a Proc then it is called after before_start" do
|
347
|
+
log = []
|
348
|
+
new_controller(
|
349
|
+
start_command: lambda {
|
350
|
+
log << "start_command"
|
351
|
+
@start_command
|
352
|
+
},
|
353
|
+
before_start: lambda { log << "before_start" }
|
354
|
+
)
|
355
|
+
@controller.start
|
356
|
+
expect(log).to eq(["before_start", "start_command"])
|
357
|
+
end
|
358
|
+
|
359
|
+
it "keeps the file descriptors in 'keep_ios' open" do
|
360
|
+
a, b = IO.pipe
|
361
|
+
begin
|
362
|
+
new_controller(keep_ios: [b])
|
363
|
+
@controller.start
|
364
|
+
b.close
|
365
|
+
expect(select([a], nil, nil, 0)).to be_nil
|
366
|
+
ensure
|
367
|
+
a.close if !a.closed?
|
368
|
+
b.close if !b.closed?
|
369
|
+
end
|
370
|
+
end
|
371
|
+
|
372
|
+
it "performs the daemonization on behalf of the daemon if 'daemonize_for_me' is set" do
|
373
|
+
new_controller(no_daemonize: true, daemonize_for_me: true)
|
374
|
+
@controller.start
|
375
|
+
expect(ping_echo_server).to be true
|
376
|
+
end
|
377
|
+
|
378
|
+
it "receives environment variables" do
|
379
|
+
new_controller(env: {"ENV_FILE" => "spec/env_file.tmp"})
|
380
|
+
File.unlink("spec/env_file.tmp") if File.exist?("spec/env_file.tmp")
|
381
|
+
@controller.start
|
382
|
+
expect(File.exist?("spec/env_file.tmp")).to be true
|
383
|
+
end
|
245
384
|
end
|
246
385
|
|
247
386
|
describe DaemonController, "#stop" do
|
248
|
-
|
249
|
-
|
250
|
-
|
251
|
-
|
252
|
-
|
253
|
-
|
254
|
-
|
255
|
-
|
256
|
-
|
257
|
-
|
258
|
-
|
259
|
-
|
260
|
-
|
261
|
-
|
262
|
-
|
263
|
-
|
264
|
-
|
265
|
-
|
266
|
-
|
267
|
-
|
268
|
-
|
269
|
-
|
270
|
-
|
271
|
-
|
272
|
-
|
273
|
-
|
274
|
-
|
275
|
-
|
276
|
-
|
277
|
-
|
278
|
-
|
279
|
-
|
280
|
-
|
281
|
-
|
282
|
-
|
283
|
-
|
284
|
-
|
285
|
-
|
286
|
-
|
287
|
-
|
288
|
-
|
289
|
-
|
290
|
-
|
291
|
-
|
292
|
-
|
293
|
-
|
294
|
-
|
295
|
-
|
296
|
-
|
297
|
-
|
298
|
-
|
299
|
-
|
387
|
+
include TestHelper
|
388
|
+
|
389
|
+
before :each do
|
390
|
+
new_controller
|
391
|
+
end
|
392
|
+
|
393
|
+
after :each do
|
394
|
+
@controller.stop if @controller
|
395
|
+
end
|
396
|
+
|
397
|
+
it "raises no exception if the daemon is not running" do
|
398
|
+
@controller.stop
|
399
|
+
end
|
400
|
+
|
401
|
+
it "waits until the daemon is no longer running" do
|
402
|
+
new_controller(stop_time: 0.3)
|
403
|
+
@controller.start
|
404
|
+
result = Benchmark.measure do
|
405
|
+
@controller.stop
|
406
|
+
end
|
407
|
+
expect(@controller).not_to be_running
|
408
|
+
expect(result.real).to be_between(0.3, 0.6)
|
409
|
+
end
|
410
|
+
|
411
|
+
it "raises StopTimeout if the daemon does not stop in time" do
|
412
|
+
new_controller(stop_time: 0.3, stop_timeout: 0.1)
|
413
|
+
@controller.start
|
414
|
+
begin
|
415
|
+
expect { @controller.stop }.to raise_error(DaemonController::StopTimeout)
|
416
|
+
ensure
|
417
|
+
new_controller.stop
|
418
|
+
end
|
419
|
+
end
|
420
|
+
|
421
|
+
describe "if stop command was given" do
|
422
|
+
it "raises StopError if the stop command exits with an error" do
|
423
|
+
new_controller(stop_command: "(echo hello world; false)")
|
424
|
+
begin
|
425
|
+
begin
|
426
|
+
@controller.stop
|
427
|
+
fail
|
428
|
+
rescue DaemonController::StopError => e
|
429
|
+
expect(e.message).to eq("hello world\n(exited with status 1)")
|
430
|
+
end
|
431
|
+
ensure
|
432
|
+
new_controller.stop
|
433
|
+
end
|
434
|
+
end
|
435
|
+
|
436
|
+
it "makes the stop command's error message available in the exception" do
|
437
|
+
end
|
438
|
+
|
439
|
+
it "calls the stop command if the PID file is invalid and :dont_stop_if_pid_file_invalid is not set" do
|
440
|
+
Dir.mktmpdir do |tmpdir|
|
441
|
+
File.open("spec/echo_server.pid", "w").close
|
442
|
+
new_controller(stop_command: "touch #{Shellwords.escape tmpdir}/stopped")
|
443
|
+
@controller.stop
|
444
|
+
expect(File.exist?("#{tmpdir}/stopped")).to be_truthy
|
445
|
+
end
|
446
|
+
ensure
|
447
|
+
@controller = nil
|
448
|
+
end
|
449
|
+
|
450
|
+
it "does not call the stop command if the PID file is invalid and :dont_stop_if_pid_file_invalid is set" do
|
451
|
+
Dir.mktmpdir do |tmpdir|
|
452
|
+
File.open("spec/echo_server.pid", "w").close
|
453
|
+
new_controller(stop_command: "touch #{Shellwords.escape tmpdir}/stopped",
|
454
|
+
dont_stop_if_pid_file_invalid: true)
|
455
|
+
@controller.stop
|
456
|
+
expect(File.exist?("#{tmpdir}/stopped")).to be_falsey
|
457
|
+
end
|
458
|
+
end
|
459
|
+
end
|
300
460
|
end
|
301
461
|
|
302
462
|
describe DaemonController, "#restart" do
|
303
|
-
|
304
|
-
|
305
|
-
|
306
|
-
|
307
|
-
|
308
|
-
|
309
|
-
|
310
|
-
|
311
|
-
|
312
|
-
|
313
|
-
|
314
|
-
|
315
|
-
|
316
|
-
|
317
|
-
|
318
|
-
|
319
|
-
|
320
|
-
|
321
|
-
|
322
|
-
|
323
|
-
|
324
|
-
|
325
|
-
|
326
|
-
|
327
|
-
|
328
|
-
|
329
|
-
|
463
|
+
include TestHelper
|
464
|
+
|
465
|
+
before :each do
|
466
|
+
new_controller
|
467
|
+
end
|
468
|
+
|
469
|
+
it "raises no exception if the daemon is not running" do
|
470
|
+
@controller.restart
|
471
|
+
end
|
472
|
+
|
473
|
+
describe "with no restart command" do
|
474
|
+
it "restart the daemon using stop and start" do
|
475
|
+
expect(@controller).to receive(:stop)
|
476
|
+
expect(@controller).to receive(:start)
|
477
|
+
@controller.restart
|
478
|
+
end
|
479
|
+
end
|
480
|
+
|
481
|
+
describe "with a restart_command" do
|
482
|
+
it "restarts the daemon using the restart_command" do
|
483
|
+
stop_cmd = "echo 'hello world'"
|
484
|
+
new_controller restart_command: stop_cmd
|
485
|
+
|
486
|
+
expect(@controller).to receive(:run_command).with(stop_cmd)
|
487
|
+
@controller.restart
|
488
|
+
end
|
489
|
+
end
|
330
490
|
end
|
331
491
|
|
332
492
|
describe DaemonController, "#connect" do
|
333
|
-
|
334
|
-
|
335
|
-
|
336
|
-
|
337
|
-
|
338
|
-
|
339
|
-
|
340
|
-
|
341
|
-
|
342
|
-
|
343
|
-
|
344
|
-
|
345
|
-
|
346
|
-
|
347
|
-
|
348
|
-
|
349
|
-
|
350
|
-
|
351
|
-
|
352
|
-
|
353
|
-
|
354
|
-
|
355
|
-
|
356
|
-
|
357
|
-
|
493
|
+
include TestHelper
|
494
|
+
|
495
|
+
before :each do
|
496
|
+
new_controller
|
497
|
+
end
|
498
|
+
|
499
|
+
it "starts the daemon if it isn't already running" do
|
500
|
+
socket = @controller.connect do
|
501
|
+
TCPSocket.new("localhost", 3230)
|
502
|
+
end
|
503
|
+
socket.close
|
504
|
+
@controller.stop
|
505
|
+
end
|
506
|
+
|
507
|
+
it "connects to the existing daemon if it's already running" do
|
508
|
+
@controller.start
|
509
|
+
begin
|
510
|
+
socket = @controller.connect do
|
511
|
+
TCPSocket.new("localhost", 3230)
|
512
|
+
end
|
513
|
+
socket.close
|
514
|
+
ensure
|
515
|
+
@controller.stop
|
516
|
+
end
|
517
|
+
end
|
358
518
|
end
|
359
519
|
|
360
520
|
describe DaemonController do
|
361
|
-
|
362
|
-
|
363
|
-
|
364
|
-
|
365
|
-
|
366
|
-
|
367
|
-
|
368
|
-
|
369
|
-
|
370
|
-
|
371
|
-
|
372
|
-
|
373
|
-
|
374
|
-
|
375
|
-
|
376
|
-
|
377
|
-
|
378
|
-
|
379
|
-
|
380
|
-
|
381
|
-
|
382
|
-
|
383
|
-
|
384
|
-
|
385
|
-
|
386
|
-
|
387
|
-
|
388
|
-
|
389
|
-
|
390
|
-
|
391
|
-
|
392
|
-
|
393
|
-
|
394
|
-
|
395
|
-
|
396
|
-
|
397
|
-
|
398
|
-
|
399
|
-
|
400
|
-
|
401
|
-
|
402
|
-
|
403
|
-
|
404
|
-
|
405
|
-
|
406
|
-
|
407
|
-
|
408
|
-
|
409
|
-
|
410
|
-
|
411
|
-
|
412
|
-
|
413
|
-
|
414
|
-
|
415
|
-
|
416
|
-
|
417
|
-
|
418
|
-
|
419
|
-
|
420
|
-
|
421
|
-
|
422
|
-
|
423
|
-
|
521
|
+
include TestHelper
|
522
|
+
|
523
|
+
after :each do
|
524
|
+
@server.close if @server && !@server.closed?
|
525
|
+
begin
|
526
|
+
File.unlink("spec/foo.sock")
|
527
|
+
rescue
|
528
|
+
nil
|
529
|
+
end
|
530
|
+
end
|
531
|
+
|
532
|
+
specify "if the ping command is a block that raises Errno::ECONNREFUSED, then that's " \
|
533
|
+
"an indication that the daemon cannot be connected to" do
|
534
|
+
new_controller(ping_command: lambda do
|
535
|
+
raise Errno::ECONNREFUSED, "dummy"
|
536
|
+
end)
|
537
|
+
expect(@controller.send(:run_ping_command)).to be false
|
538
|
+
end
|
539
|
+
|
540
|
+
specify "if the ping command is a block that returns an object that responds to #close, " \
|
541
|
+
"then the close method will be called on that object" do
|
542
|
+
@server = TCPServer.new("localhost", 8278)
|
543
|
+
socket = nil
|
544
|
+
new_controller(ping_command: lambda do
|
545
|
+
socket = TCPSocket.new("localhost", 8278)
|
546
|
+
end)
|
547
|
+
@controller.send(:run_ping_command)
|
548
|
+
expect(socket).to be_closed
|
549
|
+
end
|
550
|
+
|
551
|
+
specify "if the ping command is a block that returns an object that responds to #close, " \
|
552
|
+
"and #close raises an exception, then that exception is ignored" do
|
553
|
+
@server = TCPServer.new("localhost", 8278)
|
554
|
+
o = Object.new
|
555
|
+
expect(o).to receive(:close) { raise StandardError, "foo" }
|
556
|
+
new_controller(ping_command: lambda do
|
557
|
+
o
|
558
|
+
end)
|
559
|
+
expect { @controller.send(:run_ping_command) }.not_to raise_error
|
560
|
+
end
|
561
|
+
|
562
|
+
specify "the ping command may be [:tcp, hostname, port]" do
|
563
|
+
new_controller(ping_command: [:tcp, "127.0.0.1", 8278])
|
564
|
+
expect(@controller.send(:run_ping_command)).to be false
|
565
|
+
|
566
|
+
@server = TCPServer.new("127.0.0.1", 8278)
|
567
|
+
expect(@controller.send(:run_ping_command)).to be true
|
568
|
+
end
|
569
|
+
|
570
|
+
if DaemonController.can_ping_unix_sockets?
|
571
|
+
specify "the ping command may be [:unix, filename]" do
|
572
|
+
new_controller(ping_command: [:unix, "spec/foo.sock"])
|
573
|
+
expect(@controller.send(:run_ping_command)).to be false
|
574
|
+
|
575
|
+
@server = UNIXServer.new("spec/foo.sock")
|
576
|
+
expect(@controller.send(:run_ping_command)).to be true
|
577
|
+
end
|
578
|
+
else
|
579
|
+
specify "a ping command of type [:unix, filename] is not supported on this Ruby implementation" do
|
580
|
+
new_controller(ping_command: [:unix, "spec/foo.sock"])
|
581
|
+
@server = UNIXServer.new("spec/foo.sock")
|
582
|
+
expect { @controller.send(:run_ping_command) }.to raise_error(
|
583
|
+
"Pinging Unix domain sockets is not supported on this Ruby implementation"
|
584
|
+
)
|
585
|
+
end
|
586
|
+
end
|
424
587
|
end
|