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.
data/spec/echo_server.rb CHANGED
@@ -1,138 +1,162 @@
1
1
  #!/usr/bin/env ruby
2
+ # frozen_string_literal: true
3
+
2
4
  # A simple echo server, used by the unit test.
3
- require 'socket'
4
- require 'optparse'
5
+ require "socket"
6
+ require "optparse"
5
7
 
6
8
  options = {
7
- :port => 3230,
8
- :chdir => "/",
9
- :log_file => "/dev/null",
10
- :wait1 => 0,
11
- :wait2 => 0,
12
- :stop_time => 0,
13
- :daemonize => true
9
+ port: 3230,
10
+ chdir: "/",
11
+ log_file: "/dev/null",
12
+ wait1: 0,
13
+ wait2: 0,
14
+ stop_time: 0,
15
+ daemonize: true
14
16
  }
15
17
  parser = OptionParser.new do |opts|
16
- opts.banner = "Usage: echo_server.rb [options]"
17
- opts.separator ""
18
-
19
- opts.separator "Options:"
20
- opts.on("-p", "--port PORT", Integer, "Port to use. Default: 3230") do |value|
21
- options[:port] = value
22
- end
23
- opts.on("-C", "--change-dir DIR", String, "Change working directory. Default: /") do |value|
24
- options[:chdir] = value
25
- end
26
- opts.on("-l", "--log-file FILENAME", String, "Log file to use. Default: /dev/null") do |value|
27
- options[:log_file] = value
28
- end
29
- opts.on("-P", "--pid-file FILENAME", String, "Pid file to use.") do |value|
30
- options[:pid_file] = File.expand_path(value)
31
- end
32
- opts.on("--wait1 SECONDS", Float, "Wait a few seconds before writing pid file.") do |value|
33
- options[:wait1] = value
34
- end
35
- opts.on("--wait2 SECONDS", Float, "Wait a few seconds before binding server socket.") do |value|
36
- options[:wait2] = value
37
- end
38
- opts.on("--stop-time SECONDS", Float, "Wait a few seconds before exiting.") do |value|
39
- options[:stop_time] = value
40
- end
41
- opts.on("--crash-before-bind", "Whether the daemon should crash before binding the server socket.") do
42
- options[:crash_before_bind] = true
43
- end
44
- opts.on("--no-daemonize", "Don't daemonize.") do
45
- options[:daemonize] = false
46
- end
18
+ opts.banner = "Usage: echo_server.rb [options]"
19
+ opts.separator ""
20
+
21
+ opts.separator "Options:"
22
+ opts.on("-p", "--port PORT", Integer, "Port to use. Default: 3230") do |value|
23
+ options[:port] = value
24
+ end
25
+ opts.on("-C", "--change-dir DIR", String, "Change working directory. Default: /") do |value|
26
+ options[:chdir] = value
27
+ end
28
+ opts.on("-l", "--log-file FILENAME", String, "Log file to use. Default: /dev/null") do |value|
29
+ options[:log_file] = value
30
+ end
31
+ opts.on("-P", "--pid-file FILENAME", String, "Pid file to use.") do |value|
32
+ options[:pid_file] = File.absolute_path(value)
33
+ end
34
+ opts.on("--log-message1 MESSAGE", String, "Log message before opening log file.") do |value|
35
+ options[:log_message1] = value
36
+ end
37
+ opts.on("--log-message2 MESSAGE", String, "Log message after opening log file.") do |value|
38
+ options[:log_message2] = value
39
+ end
40
+ opts.on("--wait1 SECONDS", Float, "Wait a few seconds before writing pid file.") do |value|
41
+ options[:wait1] = value
42
+ end
43
+ opts.on("--wait2 SECONDS", Float, "Wait a few seconds before binding server socket.") do |value|
44
+ options[:wait2] = value
45
+ end
46
+ opts.on("--stop-time SECONDS", Float, "Wait a few seconds before exiting.") do |value|
47
+ options[:stop_time] = value
48
+ end
49
+ opts.on("--crash-before-bind", "Whether the daemon should crash before binding the server socket.") do
50
+ options[:crash_before_bind] = true
51
+ end
52
+ opts.on("--crash-signal SIGNAL", "Signal to send to the daemon when crashing.") do |value|
53
+ options[:crash_signal] = value
54
+ end
55
+ opts.on("--no-daemonize", "Don't daemonize.") do
56
+ options[:daemonize] = false
57
+ end
47
58
  end
48
59
  begin
49
- parser.parse!
60
+ parser.parse!
50
61
  rescue OptionParser::ParseError => e
51
- puts e
52
- puts
53
- puts "Please see '--help' for valid options."
54
- exit 1
62
+ puts e
63
+ puts
64
+ puts "Please see '--help' for valid options."
65
+ exit 1
55
66
  end
56
67
 
57
68
  if options[:pid_file]
58
- if File.exist?(options[:pid_file])
59
- STDERR.puts "*** ERROR: pid file #{options[:pid_file]} exists."
60
- exit 1
61
- end
69
+ if File.exist?(options[:pid_file])
70
+ warn "*** ERROR: pid file #{options[:pid_file]} exists."
71
+ exit 1
72
+ end
73
+ end
74
+
75
+ if options[:log_message1]
76
+ puts options[:log_message1]
77
+ $stdout.flush
62
78
  end
63
79
 
64
- if ENV['ENV_FILE']
65
- options[:env_file] = File.expand_path(ENV['ENV_FILE'])
80
+ if ENV["ENV_FILE"]
81
+ options[:env_file] = File.absolute_path(ENV["ENV_FILE"])
66
82
  end
67
83
 
68
84
  def main(options)
69
- STDIN.reopen("/dev/null", 'r')
70
- STDOUT.reopen(options[:log_file], 'a')
71
- STDERR.reopen(options[:log_file], 'a')
72
- STDOUT.sync = true
73
- STDERR.sync = true
74
- Dir.chdir(options[:chdir])
75
- File.umask(0)
85
+ $stdin.reopen("/dev/null", "r")
86
+ $stdout.reopen(options[:log_file], "a")
87
+ $stderr.reopen(options[:log_file], "a")
88
+ $stdout.sync = true
89
+ $stderr.sync = true
90
+ Dir.chdir(options[:chdir])
91
+ File.umask(0)
92
+
93
+ if options[:log_message2]
94
+ puts options[:log_message2]
95
+ end
96
+
97
+ if options[:env_file]
98
+ File.write(options[:env_file], "\0")
99
+ at_exit do
100
+ File.unlink(options[:env_file])
101
+ rescue
102
+ nil
103
+ end
104
+ end
105
+
106
+ if options[:pid_file]
107
+ sleep(options[:wait1])
108
+ File.open(options[:pid_file], "w") do |f|
109
+ f.puts(Process.pid)
110
+ end
111
+ end
112
+
113
+ sleep(options[:wait2])
114
+ if options[:crash_before_bind]
115
+ puts "#{Time.now}: crashing, as instructed."
116
+ if options[:crash_signal]
117
+ Process.kill(options[:crash_signal], Process.pid)
118
+ end
119
+ exit 2
120
+ end
76
121
 
77
- if options[:env_file]
78
- File.open(options[:env_file], 'w') do |f|
79
- f.write("\0")
80
- end
81
- at_exit do
82
- File.unlink(options[:env_file]) rescue nil
83
- end
84
- end
85
-
86
- if options[:pid_file]
87
- sleep(options[:wait1])
88
- File.open(options[:pid_file], 'w') do |f|
89
- f.puts(Process.pid)
90
- end
91
- at_exit do
92
- File.unlink(options[:pid_file]) rescue nil
93
- end
94
- end
95
-
96
- sleep(options[:wait2])
97
- if options[:crash_before_bind]
98
- puts "#{Time.now}: crashing, as instructed."
99
- exit 2
100
- end
101
-
102
- server = TCPServer.new('127.0.0.1', options[:port])
103
- begin
104
- puts "*** #{Time.now}: echo server started"
105
- while (client = server.accept)
106
- puts "#{Time.now}: new client"
107
- begin
108
- while (line = client.readline)
109
- puts "#{Time.now}: client sent: #{line.strip}"
110
- client.puts(line)
111
- end
112
- rescue EOFError
113
- ensure
114
- puts "#{Time.now}: connection closed"
115
- client.close rescue nil
116
- end
117
- end
118
- rescue SignalException
119
- exit 2
120
- rescue => e
121
- puts e.to_s
122
- puts " " << e.backtrace.join("\n ")
123
- exit 3
124
- ensure
125
- puts "*** #{Time.now}: echo server exiting..."
126
- sleep(options[:stop_time])
127
- puts "*** #{Time.now}: echo server exited"
128
- end
122
+ server = TCPServer.new("127.0.0.1", options[:port])
123
+ begin
124
+ puts "*** #{Time.now}: echo server started"
125
+ while (client = server.accept)
126
+ puts "#{Time.now}: new client"
127
+ begin
128
+ while (line = client.readline)
129
+ puts "#{Time.now}: client sent: #{line.strip}"
130
+ client.puts(line)
131
+ end
132
+ rescue EOFError
133
+ ensure
134
+ puts "#{Time.now}: connection closed"
135
+ begin
136
+ client.close
137
+ rescue
138
+ nil
139
+ end
140
+ end
141
+ end
142
+ rescue SignalException
143
+ exit 2
144
+ rescue => e
145
+ puts e
146
+ puts " " + e.backtrace.join("\n ")
147
+ exit 3
148
+ ensure
149
+ puts "*** #{Time.now}: echo server exiting..."
150
+ sleep(options[:stop_time])
151
+ puts "*** #{Time.now}: echo server exited"
152
+ end
129
153
  end
130
154
 
131
155
  if options[:daemonize]
132
- fork do
133
- Process.setsid
134
- main(options)
135
- end
156
+ fork do
157
+ Process.setsid
158
+ main(options)
159
+ end
136
160
  else
137
- main(options)
161
+ main(options)
138
162
  end
data/spec/test_helper.rb CHANGED
@@ -1,116 +1,143 @@
1
- root = File.expand_path(File.join(File.dirname(__FILE__), ".."))
2
- $LOAD_PATH.unshift(File.join(root, "lib"))
1
+ # frozen_string_literal: true
2
+
3
+ require "shellwords"
4
+
5
+ root = File.absolute_path(File.join(File.dirname(__FILE__), ".."))
3
6
  Dir.chdir(root)
4
7
 
5
- if !ENV['MRI_RUBY']
6
- if RUBY_PLATFORM =~ /java/
7
- # We need a Ruby implementation that starts fast and supports forking.
8
- # JRuby is neither.
9
- abort "In order to run these tests in JRuby, you must set " +
10
- "the environment variable $MRI_RUBY to an MRI Ruby interpeter."
11
- else
12
- require 'rbconfig'
13
- rb_config = defined?(RbConfig) ? RbConfig::CONFIG : Config::CONFIG
14
- ENV['MRI_RUBY'] = rb_config['bindir'] + '/' + rb_config['RUBY_INSTALL_NAME'] +
15
- rb_config['EXEEXT']
16
- puts ENV['MRI_RUBY']
17
- end
8
+ # Ensure subprocesses (with could be a different Ruby) are
9
+ # started without Bundler environment variables.
10
+ ENV.replace(Bundler.with_unbundled_env { ENV.to_h.dup })
11
+
12
+ if !ENV["MRI_RUBY"]
13
+ if RUBY_PLATFORM.match?(/java/)
14
+ # We need a Ruby implementation that starts fast and supports forking.
15
+ # JRuby is neither.
16
+ abort "In order to run these tests in JRuby, you must set " \
17
+ "the environment variable $MRI_RUBY to an MRI Ruby interpeter."
18
+ else
19
+ require "rbconfig"
20
+ rb_config = defined?(RbConfig) ? RbConfig::CONFIG : Config::CONFIG
21
+ ENV["MRI_RUBY"] = rb_config["bindir"] + "/" + rb_config["RUBY_INSTALL_NAME"] +
22
+ rb_config["EXEEXT"]
23
+ puts ENV["MRI_RUBY"]
24
+ end
25
+ end
26
+
27
+ trap("SIGQUIT") do
28
+ if Thread.respond_to?(:list)
29
+ output = String.new("----- #{Time.now} -----\n")
30
+ Thread.list.each do |thread|
31
+ output << "##### #{thread}\n"
32
+ output << thread.backtrace.join("\n")
33
+ output << "\n\n"
34
+ end
35
+ output << "--------------------"
36
+ warn(output)
37
+ $stderr.flush
38
+ end
18
39
  end
19
40
 
20
41
  module TestHelper
21
- def new_controller(options = {})
22
- @start_command = './spec/run_echo_server -l spec/echo_server.log -P spec/echo_server.pid'
23
- if options[:wait1]
24
- @start_command << " --wait1 #{options[:wait1]}"
25
- end
26
- if options[:wait2]
27
- @start_command << " --wait2 #{options[:wait2]}"
28
- end
29
- if options[:stop_time]
30
- @start_command << " --stop-time #{options[:stop_time]}"
31
- end
32
- if options[:crash_before_bind]
33
- @start_command << " --crash-before-bind"
34
- end
35
- if options[:no_daemonize]
36
- @start_command << " --no-daemonize"
37
- end
38
- new_options = {
39
- :identifier => 'My Test Daemon',
40
- :start_command => @start_command,
41
- :ping_command => method(:ping_echo_server),
42
- :pid_file => 'spec/echo_server.pid',
43
- :log_file => 'spec/echo_server.log',
44
- :start_timeout => 3,
45
- :stop_timeout => 3
46
- }.merge(options)
47
- @controller = DaemonController.new(new_options)
48
- end
49
-
50
- def ping_echo_server
51
- begin
52
- TCPSocket.new('127.0.0.1', 3230)
53
- true
54
- rescue SystemCallError
55
- false
56
- end
57
- end
58
-
59
- def write_file(filename, contents)
60
- File.open(filename, 'w') do |f|
61
- f.write(contents)
62
- end
63
- end
64
-
65
- def exec_is_slow?
66
- return RUBY_PLATFORM == "java"
67
- end
68
-
69
- def process_is_alive?(pid)
70
- begin
71
- Process.kill(0, pid)
72
- return true
73
- rescue Errno::ESRCH
74
- return false
75
- rescue SystemCallError => e
76
- return true
77
- end
78
- end
79
-
80
- def eventually(deadline_duration = 1, check_interval = 0.05)
81
- deadline = Time.now + deadline_duration
82
- while Time.now < deadline
83
- if yield
84
- return
85
- else
86
- sleep(check_interval)
87
- end
88
- end
89
- raise "Time limit exceeded"
90
- end
42
+ def new_controller(options = {})
43
+ @start_command = String.new("./spec/run_echo_server -l spec/echo_server.log")
44
+ if options[:log_message1]
45
+ @start_command << " --log-message1 #{Shellwords.escape options[:log_message1]}"
46
+ end
47
+ if options[:log_message2]
48
+ @start_command << " --log-message2 #{Shellwords.escape options[:log_message2]}"
49
+ end
50
+ if options[:wait1]
51
+ @start_command << " --wait1 #{options[:wait1]}"
52
+ end
53
+ if options[:wait2]
54
+ @start_command << " --wait2 #{options[:wait2]}"
55
+ end
56
+ if options[:stop_time]
57
+ @start_command << " --stop-time #{options[:stop_time]}"
58
+ end
59
+ if options[:crash_before_bind]
60
+ @start_command << " --crash-before-bind"
61
+ end
62
+ if options[:crash_signal]
63
+ @start_command << " --crash-signal #{options[:crash_signal]}"
64
+ end
65
+ if options[:no_daemonize]
66
+ @start_command << " --no-daemonize"
67
+ end
68
+ if !options[:no_write_pid_file]
69
+ @start_command << " -P spec/echo_server.pid"
70
+ end
71
+ new_options = {
72
+ identifier: "My Test Daemon",
73
+ start_command: @start_command,
74
+ ping_command: method(:ping_echo_server),
75
+ pid_file: "spec/echo_server.pid",
76
+ log_file: "spec/echo_server.log",
77
+ start_timeout: 30,
78
+ stop_timeout: 30
79
+ }.merge(options)
80
+ @controller = DaemonController.new(new_options)
81
+ end
82
+
83
+ def ping_echo_server
84
+ TCPSocket.new("127.0.0.1", 3230)
85
+ true
86
+ rescue SystemCallError
87
+ false
88
+ end
89
+
90
+ def write_file(filename, contents)
91
+ File.write(filename, contents)
92
+ end
93
+
94
+ def exec_is_slow?
95
+ RUBY_PLATFORM == "java"
96
+ end
97
+
98
+ def process_is_alive?(pid)
99
+ Process.kill(0, pid)
100
+ true
101
+ rescue Errno::ESRCH
102
+ false
103
+ rescue SystemCallError
104
+ true
105
+ end
106
+
107
+ def eventually(deadline_duration = 1, check_interval = 0.05)
108
+ deadline = Time.now + deadline_duration
109
+ while Time.now < deadline
110
+ if yield
111
+ return
112
+ else
113
+ sleep(check_interval)
114
+ end
115
+ end
116
+ raise "Time limit exceeded"
117
+ end
91
118
  end
92
119
 
93
120
  # A thread which doesn't execute its block until the
94
121
  # 'go!' method has been called.
95
122
  class WaitingThread < Thread
96
- def initialize
97
- @mutex = Mutex.new
98
- @cond = ConditionVariable.new
99
- @go = false
100
- super do
101
- @mutex.synchronize do
102
- while !@go
103
- @cond.wait(@mutex)
104
- end
105
- end
106
- yield
107
- end
108
- end
109
-
110
- def go!
111
- @mutex.synchronize do
112
- @go = true
113
- @cond.broadcast
114
- end
115
- end
123
+ def initialize
124
+ @mutex = Mutex.new
125
+ @cond = ConditionVariable.new
126
+ @go = false
127
+ super do
128
+ @mutex.synchronize do
129
+ until @go
130
+ @cond.wait(@mutex)
131
+ end
132
+ end
133
+ yield
134
+ end
135
+ end
136
+
137
+ def go!
138
+ @mutex.synchronize do
139
+ @go = true
140
+ @cond.broadcast
141
+ end
142
+ end
116
143
  end
@@ -1,12 +1,18 @@
1
1
  #!/usr/bin/env ruby
2
+ # frozen_string_literal: true
3
+
2
4
  dir = File.dirname(__FILE__)
3
5
  Dir.chdir(File.dirname(dir))
4
6
  begin
5
- File.open("spec/echo_server.pid", "w") do |f|
6
- f.puts(Process.pid)
7
- end
8
- sleep 30
7
+ File.open("spec/echo_server.pid", "w") do |f|
8
+ f.puts(Process.pid)
9
+ end
10
+ sleep 30
9
11
  rescue SignalException
10
- File.unlink("spec/echo_server.pid") rescue nil
11
- raise
12
- end
12
+ begin
13
+ File.unlink("spec/echo_server.pid")
14
+ rescue
15
+ nil
16
+ end
17
+ raise
18
+ end
metadata CHANGED
@@ -1,14 +1,13 @@
1
1
  --- !ruby/object:Gem::Specification
2
2
  name: daemon_controller
3
3
  version: !ruby/object:Gem::Version
4
- version: 1.2.0
4
+ version: 2.0.0
5
5
  platform: ruby
6
6
  authors:
7
7
  - Hongli Lai
8
- autorequire:
9
8
  bindir: bin
10
9
  cert_chain: []
11
- date: 2014-03-03 00:00:00.000000000 Z
10
+ date: 2025-03-24 00:00:00.000000000 Z
12
11
  dependencies: []
13
12
  description: A library for robust daemon management.
14
13
  email: software-signing@phusion.nl
@@ -17,23 +16,14 @@ extensions: []
17
16
  extra_rdoc_files: []
18
17
  files:
19
18
  - LICENSE.txt
20
- - README.markdown
19
+ - README.md
21
20
  - Rakefile
22
21
  - daemon_controller.gemspec
23
- - debian.template/changelog
24
- - debian.template/compat
25
- - debian.template/control.template
26
- - debian.template/copyright
27
- - debian.template/ruby-daemon-controller.install
28
- - debian.template/rules
29
- - debian.template/source/format
30
22
  - lib/daemon_controller.rb
31
23
  - lib/daemon_controller/lock_file.rb
32
24
  - lib/daemon_controller/packaging.rb
33
25
  - lib/daemon_controller/spawn.rb
34
26
  - lib/daemon_controller/version.rb
35
- - rpm/get_distro_id.py
36
- - rpm/rubygem-daemon_controller.spec.template
37
27
  - spec/daemon_controller_spec.rb
38
28
  - spec/echo_server.rb
39
29
  - spec/run_echo_server
@@ -43,24 +33,21 @@ homepage: https://github.com/FooBarWidget/daemon_controller
43
33
  licenses:
44
34
  - MIT
45
35
  metadata: {}
46
- post_install_message:
47
36
  rdoc_options: []
48
37
  require_paths:
49
38
  - lib
50
39
  required_ruby_version: !ruby/object:Gem::Requirement
51
40
  requirements:
52
- - - ! '>='
41
+ - - ">="
53
42
  - !ruby/object:Gem::Version
54
- version: '0'
43
+ version: 2.0.0
55
44
  required_rubygems_version: !ruby/object:Gem::Requirement
56
45
  requirements:
57
- - - ! '>='
46
+ - - ">="
58
47
  - !ruby/object:Gem::Version
59
48
  version: '0'
60
49
  requirements: []
61
- rubyforge_project:
62
- rubygems_version: 2.2.0
63
- signing_key:
50
+ rubygems_version: 3.6.2
64
51
  specification_version: 4
65
52
  summary: A library for implementing daemon management capabilities
66
53
  test_files: []
checksums.yaml.gz.asc DELETED
@@ -1,12 +0,0 @@
1
- -----BEGIN PGP SIGNATURE-----
2
- Version: GnuPG/MacGPG2 v2.0.17 (Darwin)
3
- Comment: GPGTools - http://gpgtools.org
4
-
5
- iQEcBAABAgAGBQJTFNXmAAoJECrHRaUKISqMzVcH/3xePU605BycvtYvhZDEDmIk
6
- O27O70O3hDW6+ljjprz7aAniJMGkF3ewn34BJ35kb5xmhxi5gFdVzu/Su9un1KWw
7
- RKy0b/5rR6SJvuFxaWAOBUtTk/YO2FyG43dTvKb12OtAha9V2F5etoTPMzPyzllH
8
- GfIpaW0EuHqPfq/+6+rgtNGyrfXagx4udvh394RJlIsys/zrgH/gP37x/1XsHTAn
9
- Hbe6AeHuaCLPm8PMLTsETQleeioA/Dkgju1Ag9ihvG4qlVqRFM5EA2zJvR9yWb33
10
- 7YfgSHUSU4TYdPER4FrKQfUTbGCuJiInrxF2M1awpHSgMKC3hgnOAG9Koh9hiz8=
11
- =wUWc
12
- -----END PGP SIGNATURE-----
@@ -1,5 +0,0 @@
1
- daemon-controller-ruby (1.1.2-1) precise; urgency=low
2
-
3
- * Initial packaging
4
-
5
- -- Hongli Lai <hongli@phusion.nl> Wed, 01 May 2013 16:04:07 +0200
@@ -1 +0,0 @@
1
- 7
@@ -1,15 +0,0 @@
1
- Source: ruby-daemon-controller
2
- Section: ruby
3
- Priority: extra
4
- Maintainer: Hongli Lai <hongli@phusion.nl>
5
- Build-Depends: debhelper (>= 7.0.50~)
6
- Standards-Version: 3.9.3
7
- Homepage: https://github.com/FooBarWidget/daemon_controller
8
- Vcs-Git: git://github.com/FooBarWidget/daemon_controller.git
9
- Vcs-Browser: https://github.com/FooBarWidget/daemon_controller
10
-
11
- Package: ruby-daemon-controller
12
- Architecture: all
13
- Depends: ${shlibs:Depends}, ${misc:Depends}, ruby | ruby-interpreter
14
- Description: Library for robust daemon management
15
- Library which implements daemon management capabilities.