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.
@@ -1,424 +1,587 @@
1
- require File.expand_path(File.join(File.dirname(__FILE__), "test_helper"))
2
- require 'daemon_controller'
3
- require 'benchmark'
4
- require 'socket'
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
- before :each do
8
- new_controller
9
- end
10
-
11
- include TestHelper
12
-
13
- it "works" do
14
- @controller.start
15
- ping_echo_server.should be_true
16
- @controller.stop
17
- end
18
-
19
- it "raises AlreadyStarted if the daemon is already running" do
20
- @controller.should_receive(:daemon_is_running?).and_return(true)
21
- lambda { @controller.start }.should raise_error(DaemonController::AlreadyStarted)
22
- end
23
-
24
- it "deletes existing PID file before starting the daemon" do
25
- write_file('spec/echo_server.pid', '1234')
26
- @controller.should_receive(:daemon_is_running?).and_return(false)
27
- @controller.should_receive(:spawn_daemon)
28
- @controller.should_receive(:pid_file_available?).and_return(true)
29
- @controller.should_receive(:run_ping_command).at_least(:once).and_return(true)
30
- @controller.start
31
- File.exist?('spec/echo_server.pid').should be_false
32
- end
33
-
34
- it "blocks until the daemon has written to its PID file" do
35
- thread = WaitingThread.new do
36
- sleep 0.15
37
- write_file('spec/echo_server.pid', '1234')
38
- end
39
- @controller.should_receive(:daemon_is_running?).and_return(false)
40
- @controller.should_receive(:spawn_daemon).and_return do
41
- thread.go!
42
- end
43
- @controller.should_receive(:run_ping_command).at_least(:once).and_return(true)
44
- begin
45
- result = Benchmark.measure do
46
- @controller.start
47
- end
48
- (0.15 .. 0.30).should === result.real
49
- ensure
50
- thread.join
51
- end
52
- end
53
-
54
- it "blocks until the daemon can be pinged" do
55
- ping_ok = false
56
- running = false
57
- thread = WaitingThread.new do
58
- sleep 0.15
59
- ping_ok = true
60
- end
61
- @controller.should_receive(:daemon_is_running?).at_least(:once).and_return do
62
- running
63
- end
64
- @controller.should_receive(:spawn_daemon).and_return do
65
- thread.go!
66
- running = true
67
- end
68
- @controller.should_receive(:pid_file_available?).and_return(true)
69
- @controller.should_receive(:run_ping_command).at_least(:once).and_return do
70
- ping_ok
71
- end
72
- begin
73
- result = Benchmark.measure do
74
- @controller.start
75
- end
76
- (0.15 .. 0.30).should === result.real
77
- ensure
78
- thread.join
79
- end
80
- end
81
-
82
- it "raises StartTimeout if the daemon doesn't start in time" do
83
- if exec_is_slow?
84
- start_timeout = 4
85
- min_start_timeout = 0
86
- max_start_timeout = 6
87
- else
88
- start_timeout = 0.15
89
- min_start_timeout = 0.15
90
- max_start_timeout = 0.30
91
- end
92
- new_controller(:start_command => 'sleep 2', :start_timeout => start_timeout)
93
- start_time = Time.now
94
- end_time = nil
95
- @controller.should_receive(:start_timed_out).and_return do
96
- end_time = Time.now
97
- end
98
- begin
99
- lambda { @controller.start }.should raise_error(DaemonController::StartTimeout)
100
- (min_start_timeout .. max_start_timeout).should === end_time - start_time
101
- ensure
102
- @controller.stop
103
- end
104
- end
105
-
106
- it "kills the daemon forcefully if the daemon has forked but doesn't " <<
107
- "become pingable in time, and there's a PID file" do
108
- new_controller(:wait2 => 3, :start_timeout => 1)
109
- pid = nil
110
- @controller.should_receive(:start_timed_out).and_return do
111
- @controller.send(:wait_until) do
112
- @controller.send(:pid_file_available?)
113
- end
114
- pid = @controller.send(:read_pid_file)
115
- end
116
- begin
117
- block = lambda { @controller.start }
118
- block.should raise_error(DaemonController::StartTimeout, /failed to start in time/)
119
- eventually(1) do
120
- !process_is_alive?(pid)
121
- end
122
-
123
- # The daemon should not be able to clean up its PID file since
124
- # it's killed with SIGKILL.
125
- File.exist?("spec/echo_server.pid").should be_true
126
- ensure
127
- File.unlink("spec/echo_server.pid") rescue nil
128
- end
129
- end
130
-
131
- if DaemonController.send(:fork_supported?) || DaemonController.send(:spawn_supported?)
132
- it "kills the daemon if it doesn't start in time and hasn't forked " <<
133
- "yet, on platforms where Ruby supports fork() or Process.spawn" do
134
- begin
135
- new_controller(:start_command => "./spec/unresponsive_daemon.rb",
136
- :start_timeout => 0.2)
137
- pid = nil
138
- @controller.should_receive(:daemonization_timed_out).and_return do
139
- @controller.send(:wait_until) do
140
- @controller.send(:pid_file_available?)
141
- end
142
- pid = @controller.send(:read_pid_file)
143
- end
144
- block = lambda { @controller.start }
145
- block.should raise_error(DaemonController::StartTimeout, /didn't daemonize in time/)
146
- eventually(1) do
147
- !process_is_alive?(pid)
148
- end
149
- ensure
150
- File.unlink("spec/echo_server.pid") rescue nil
151
- end
152
- end
153
- end
154
-
155
- it "raises an error if the daemon exits with an error before forking" do
156
- new_controller(:start_command => 'false')
157
- lambda { @controller.start }.should raise_error(DaemonController::Error)
158
- end
159
-
160
- it "raises an error if the daemon exits with an error after forking" do
161
- new_controller(:crash_before_bind => true, :log_file_activity_timeout => 0.2)
162
- lambda { @controller.start }.should raise_error(DaemonController::Error)
163
- end
164
-
165
- specify "the daemon's error output before forking is made available in the exception" do
166
- new_controller(:start_command => '(echo hello world; false)')
167
- begin
168
- @controller.start
169
- rescue DaemonController::Error => e
170
- e.message.should == "hello world"
171
- end
172
- end
173
-
174
- specify "the daemon's error output after forking is made available in the exception" do
175
- new_controller(:crash_before_bind => true, :log_file_activity_timeout => 0.1)
176
- begin
177
- @controller.start
178
- violated
179
- rescue DaemonController::StartTimeout => e
180
- e.message.should =~ /crashing, as instructed/
181
- end
182
- end
183
-
184
- specify "the start command may be a Proc" do
185
- called = true
186
- new_controller(:start_command => lambda { called = true; @start_command })
187
- begin
188
- @controller.start
189
- ensure
190
- @controller.stop
191
- end
192
- called.should be_true
193
- end
194
-
195
- specify "if the start command is a Proc then it is called after before_start" do
196
- log = []
197
- new_controller(
198
- :start_command => lambda {
199
- log << "start_command"
200
- @start_command
201
- },
202
- :before_start => lambda { log << "before_start" }
203
- )
204
- begin
205
- @controller.start
206
- ensure
207
- @controller.stop
208
- end
209
- log.should == ["before_start", "start_command"]
210
- end
211
-
212
- if DaemonController.send(:fork_supported?) || DaemonController.send(:spawn_supported?)
213
- it "keeps the file descriptors in 'keep_ios' open" do
214
- a, b = IO.pipe
215
- begin
216
- new_controller(:keep_ios => [b])
217
- begin
218
- @controller.start
219
- b.close
220
- select([a], nil, nil, 0).should be_nil
221
- ensure
222
- @controller.stop
223
- end
224
- ensure
225
- a.close if !a.closed?
226
- b.close if !b.closed?
227
- end
228
- end
229
-
230
- it "performs the daemonization on behalf of the daemon if 'daemonize_for_me' is set" do
231
- new_controller(:no_daemonize => true, :daemonize_for_me => true)
232
- @controller.start
233
- ping_echo_server.should be_true
234
- @controller.stop
235
- end
236
- end
237
-
238
- it "receives environment variables" do
239
- new_controller(:env => {'ENV_FILE' => 'spec/env_file.tmp'})
240
- File.unlink('spec/env_file.tmp') if File.exist?('spec/env_file.tmp')
241
- @controller.start
242
- File.exist?('spec/env_file.tmp').should be_true
243
- @controller.stop
244
- end
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
- include TestHelper
249
-
250
- before :each do
251
- new_controller
252
- end
253
-
254
- after :each do
255
- @controller.stop
256
- end
257
-
258
- it "raises no exception if the daemon is not running" do
259
- @controller.stop
260
- end
261
-
262
- it "waits until the daemon is no longer running" do
263
- new_controller(:stop_time => 0.3)
264
- @controller.start
265
- result = Benchmark.measure do
266
- @controller.stop
267
- end
268
- @controller.should_not be_running
269
- result.real.should be_between(0.3, 0.6)
270
- end
271
-
272
- it "raises StopTimeout if the daemon does not stop in time" do
273
- new_controller(:stop_time => 0.3, :stop_timeout => 0.1)
274
- @controller.start
275
- begin
276
- lambda { @controller.stop }.should raise_error(DaemonController::StopTimeout)
277
- ensure
278
- new_controller.stop
279
- end
280
- end
281
-
282
- describe "if stop command was given" do
283
- it "raises StopError if the stop command exits with an error" do
284
- new_controller(:stop_command => '(echo hello world; false)')
285
- begin
286
- begin
287
- @controller.stop
288
- violated
289
- rescue DaemonController::StopError => e
290
- e.message.should == 'hello world'
291
- end
292
- ensure
293
- new_controller.stop
294
- end
295
- end
296
-
297
- it "makes the stop command's error message available in the exception" do
298
- end
299
- end
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
- include TestHelper
304
-
305
- before :each do
306
- new_controller
307
- end
308
-
309
- it "raises no exception if the daemon is not running" do
310
- @controller.restart
311
- end
312
-
313
- describe 'with no restart command' do
314
- it "restart the daemon using stop and start" do
315
- @controller.should_receive(:stop)
316
- @controller.should_receive(:start)
317
- @controller.restart
318
- end
319
- end
320
-
321
- describe 'with a restart_command' do
322
- it 'restarts the daemon using the restart_command' do
323
- stop_cmd = "echo 'hello world'"
324
- new_controller :restart_command => stop_cmd
325
-
326
- @controller.should_receive(:run_command).with(stop_cmd)
327
- @controller.restart
328
- end
329
- end
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
- include TestHelper
334
-
335
- before :each do
336
- new_controller
337
- end
338
-
339
- it "starts the daemon if it isn't already running" do
340
- socket = @controller.connect do
341
- TCPSocket.new('localhost', 3230)
342
- end
343
- socket.close
344
- @controller.stop
345
- end
346
-
347
- it "connects to the existing daemon if it's already running" do
348
- @controller.start
349
- begin
350
- socket = @controller.connect do
351
- TCPSocket.new('localhost', 3230)
352
- end
353
- socket.close
354
- ensure
355
- @controller.stop
356
- end
357
- end
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
- include TestHelper
362
-
363
- after :each do
364
- @server.close if @server && !@server.closed?
365
- File.unlink('spec/foo.sock') rescue nil
366
- end
367
-
368
- specify "if the ping command is a block that raises Errno::ECONNREFUSED, then that's " <<
369
- "an indication that the daemon cannot be connected to" do
370
- new_controller(:ping_command => lambda do
371
- raise Errno::ECONNREFUSED, "dummy"
372
- end)
373
- @controller.send(:run_ping_command).should be_false
374
- end
375
-
376
- specify "if the ping command is a block that returns an object that responds to #close, " <<
377
- "then the close method will be called on that object" do
378
- @server = TCPServer.new('localhost', 8278)
379
- socket = nil
380
- new_controller(:ping_command => lambda do
381
- socket = TCPSocket.new('localhost', 8278)
382
- end)
383
- @controller.send(:run_ping_command)
384
- socket.should be_closed
385
- end
386
-
387
- specify "if the ping command is a block that returns an object that responds to #close, " <<
388
- "and #close raises an exception, then that exception is ignored" do
389
- @server = TCPServer.new('localhost', 8278)
390
- o = Object.new
391
- o.should_receive(:close).and_return do
392
- raise StandardError, "foo"
393
- end
394
- new_controller(:ping_command => lambda do
395
- o
396
- end)
397
- lambda { @controller.send(:run_ping_command) }.should_not raise_error
398
- end
399
-
400
- specify "the ping command may be [:tcp, hostname, port]" do
401
- new_controller(:ping_command => [:tcp, "127.0.0.1", 8278])
402
- @controller.send(:run_ping_command).should be_false
403
-
404
- @server = TCPServer.new('127.0.0.1', 8278)
405
- @controller.send(:run_ping_command).should be_true
406
- end
407
-
408
- if DaemonController.can_ping_unix_sockets?
409
- specify "the ping command may be [:unix, filename]" do
410
- new_controller(:ping_command => [:unix, "spec/foo.sock"])
411
- @controller.send(:run_ping_command).should be_false
412
-
413
- @server = UNIXServer.new('spec/foo.sock')
414
- @controller.send(:run_ping_command).should be_true
415
- end
416
- else
417
- specify "a ping command of type [:unix, filename] is not supported on this Ruby implementation" do
418
- new_controller(:ping_command => [:unix, "spec/foo.sock"])
419
- @server = UNIXServer.new('spec/foo.sock')
420
- lambda { @controller.send(:run_ping_command) }.should raise_error(
421
- "Pinging Unix domain sockets is not supported on this Ruby implementation")
422
- end
423
- end
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