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,16 +1,18 @@
1
+ # frozen_string_literal: true
2
+
1
3
  # daemon_controller, library for robust daemon management
2
- # Copyright (c) 2010 Phusion
3
- #
4
+ # Copyright (c) 2010-2025 Asynchronous B.V.
5
+ #
4
6
  # Permission is hereby granted, free of charge, to any person obtaining a copy
5
7
  # of this software and associated documentation files (the "Software"), to deal
6
8
  # in the Software without restriction, including without limitation the rights
7
9
  # to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
8
10
  # copies of the Software, and to permit persons to whom the Software is
9
11
  # furnished to do so, subject to the following conditions:
10
- #
12
+ #
11
13
  # The above copyright notice and this permission notice shall be included in
12
14
  # all copies or substantial portions of the Software.
13
- #
15
+ #
14
16
  # THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
15
17
  # IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
16
18
  # FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
@@ -19,108 +21,107 @@
19
21
  # OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
20
22
  # THE SOFTWARE.
21
23
 
22
- require 'fcntl'
24
+ require "fcntl"
23
25
 
24
26
  class DaemonController
27
+ # A lock file is a synchronization mechanism, like a Mutex, but it also allows
28
+ # inter-process synchronization (as opposed to only inter-thread synchronization
29
+ # within a single process).
30
+ #
31
+ # Processes can obtain either a shared lock or an exclusive lock. It's possible
32
+ # for multiple processes to obtain a shared lock on a file as long as no
33
+ # exclusive lock has been obtained by a process. If a process has obtained an
34
+ # exclusive lock, then no other processes can lock the file, whether they're
35
+ # trying to obtain a shared lock or an exclusive lock.
36
+ #
37
+ # Note that on JRuby, LockFile can only guarantee synchronization between
38
+ # threads if the different threads use the same LockFile object. Specifying the
39
+ # same filename is not enough.
40
+ class LockFile
41
+ class AlreadyLocked < StandardError
42
+ end
25
43
 
26
- # A lock file is a synchronization mechanism, like a Mutex, but it also allows
27
- # inter-process synchronization (as opposed to only inter-thread synchronization
28
- # within a single process).
29
- #
30
- # Processes can obtain either a shared lock or an exclusive lock. It's possible
31
- # for multiple processes to obtain a shared lock on a file as long as no
32
- # exclusive lock has been obtained by a process. If a process has obtained an
33
- # exclusive lock, then no other processes can lock the file, whether they're
34
- # trying to obtain a shared lock or an exclusive lock.
35
- #
36
- # Note that on JRuby, LockFile can only guarantee synchronization between
37
- # threads if the different threads use the same LockFile object. Specifying the
38
- # same filename is not enough.
39
- class LockFile
40
- class AlreadyLocked < StandardError
41
- end
42
-
43
- # Create a LockFile object. The lock file is initially not locked.
44
- #
45
- # +filename+ may point to a nonexistant file. In that case, the lock
46
- # file will not be created until one's trying to obtain a lock.
47
- #
48
- # Note that LockFile will use this exact filename. So if +filename+
49
- # is a relative filename, then the actual lock file that will be used
50
- # depends on the current working directory.
51
- def initialize(filename)
52
- @filename = filename
53
- end
54
-
55
- # Obtain an exclusive lock on the lock file, yield the given block,
56
- # then unlock the lockfile. If the lock file was already locked (whether
57
- # shared or exclusively) by another process/thread then this method will
58
- # block until the lock file has been unlocked.
59
- #
60
- # The lock file *must* be writable, otherwise an Errno::EACCESS
61
- # exception will be raised.
62
- def exclusive_lock
63
- File.open(@filename, 'w') do |f|
64
- if Fcntl.const_defined? :F_SETFD
65
- f.fcntl(Fcntl::F_SETFD, Fcntl::FD_CLOEXEC)
66
- end
67
- f.flock(File::LOCK_EX)
68
- yield
69
- end
70
- end
71
-
72
- # Obtain an exclusive lock on the lock file, yield the given block,
73
- # then unlock the lockfile. If the lock file was already exclusively
74
- # locked by another process/thread then this method will
75
- # block until the exclusive lock has been released. This method will not
76
- # block if only shared locks have been obtained.
77
- #
78
- # The lock file *must* be writable, otherwise an Errno::EACCESS
79
- # exception will be raised.
80
- def shared_lock
81
- File.open(@filename, 'w+') do |f|
82
- if Fcntl.const_defined? :F_SETFD
83
- f.fcntl(Fcntl::F_SETFD, Fcntl::FD_CLOEXEC)
84
- end
85
- f.flock(File::LOCK_SH)
86
- yield
87
- end
88
- end
89
-
90
- # Try to obtain a shared lock on the lock file, similar to #shared_lock.
91
- # But unlike #shared_lock, this method will raise AlreadyLocked if
92
- # no lock can be obtained, instead of blocking.
93
- #
94
- # If a lock can be obtained, then the given block will be yielded.
95
- def try_shared_lock
96
- File.open(@filename, 'w+') do |f|
97
- if Fcntl.const_defined? :F_SETFD
98
- f.fcntl(Fcntl::F_SETFD, Fcntl::FD_CLOEXEC)
99
- end
100
- if f.flock(File::LOCK_SH | File::LOCK_NB)
101
- yield
102
- else
103
- raise AlreadyLocked
104
- end
105
- end
106
- end
107
-
108
- # Try to obtain an exclusive lock on the lock file, similar to #exclusive_lock.
109
- # But unlike #exclusive_lock, this method will raise AlreadyLocked if
110
- # no lock can be obtained, instead of blocking.
111
- #
112
- # If a lock can be obtained, then the given block will be yielded.
113
- def try_exclusive_lock
114
- File.open(@filename, 'w') do |f|
115
- if Fcntl.const_defined? :F_SETFD
116
- f.fcntl(Fcntl::F_SETFD, Fcntl::FD_CLOEXEC)
117
- end
118
- if f.flock(File::LOCK_EX | File::LOCK_NB)
119
- yield
120
- else
121
- raise AlreadyLocked
122
- end
123
- end
124
- end
125
- end # class LockFile
44
+ # Create a LockFile object. The lock file is initially not locked.
45
+ #
46
+ # +filename+ may point to a nonexistant file. In that case, the lock
47
+ # file will not be created until one's trying to obtain a lock.
48
+ #
49
+ # Note that LockFile will use this exact filename. So if +filename+
50
+ # is a relative filename, then the actual lock file that will be used
51
+ # depends on the current working directory.
52
+ def initialize(filename)
53
+ @filename = filename
54
+ end
55
+
56
+ # Obtain an exclusive lock on the lock file, yield the given block,
57
+ # then unlock the lockfile. If the lock file was already locked (whether
58
+ # shared or exclusively) by another process/thread then this method will
59
+ # block until the lock file has been unlocked.
60
+ #
61
+ # The lock file *must* be writable, otherwise an Errno::EACCESS
62
+ # exception will be raised.
63
+ def exclusive_lock
64
+ File.open(@filename, "w") do |f|
65
+ if Fcntl.const_defined? :F_SETFD
66
+ f.fcntl(Fcntl::F_SETFD, Fcntl::FD_CLOEXEC)
67
+ end
68
+ f.flock(File::LOCK_EX)
69
+ yield
70
+ end
71
+ end
72
+
73
+ # Obtain an exclusive lock on the lock file, yield the given block,
74
+ # then unlock the lockfile. If the lock file was already exclusively
75
+ # locked by another process/thread then this method will
76
+ # block until the exclusive lock has been released. This method will not
77
+ # block if only shared locks have been obtained.
78
+ #
79
+ # The lock file *must* be writable, otherwise an Errno::EACCESS
80
+ # exception will be raised.
81
+ def shared_lock
82
+ File.open(@filename, "w+") do |f|
83
+ if Fcntl.const_defined? :F_SETFD
84
+ f.fcntl(Fcntl::F_SETFD, Fcntl::FD_CLOEXEC)
85
+ end
86
+ f.flock(File::LOCK_SH)
87
+ yield
88
+ end
89
+ end
90
+
91
+ # Try to obtain a shared lock on the lock file, similar to #shared_lock.
92
+ # But unlike #shared_lock, this method will raise AlreadyLocked if
93
+ # no lock can be obtained, instead of blocking.
94
+ #
95
+ # If a lock can be obtained, then the given block will be yielded.
96
+ def try_shared_lock
97
+ File.open(@filename, "w+") do |f|
98
+ if Fcntl.const_defined? :F_SETFD
99
+ f.fcntl(Fcntl::F_SETFD, Fcntl::FD_CLOEXEC)
100
+ end
101
+ if f.flock(File::LOCK_SH | File::LOCK_NB)
102
+ yield
103
+ else
104
+ raise AlreadyLocked
105
+ end
106
+ end
107
+ end
108
+
109
+ # Try to obtain an exclusive lock on the lock file, similar to #exclusive_lock.
110
+ # But unlike #exclusive_lock, this method will raise AlreadyLocked if
111
+ # no lock can be obtained, instead of blocking.
112
+ #
113
+ # If a lock can be obtained, then the given block will be yielded.
114
+ def try_exclusive_lock
115
+ File.open(@filename, "w") do |f|
116
+ if Fcntl.const_defined? :F_SETFD
117
+ f.fcntl(Fcntl::F_SETFD, Fcntl::FD_CLOEXEC)
118
+ end
119
+ if f.flock(File::LOCK_EX | File::LOCK_NB)
120
+ yield
121
+ else
122
+ raise AlreadyLocked
123
+ end
124
+ end
125
+ end
126
+ end # class LockFile
126
127
  end # class DaemonController
@@ -1,16 +1,18 @@
1
+ # frozen_string_literal: true
2
+
1
3
  # daemon_controller, library for robust daemon management
2
- # Copyright (c) 2013 Phusion
3
- #
4
+ # Copyright (c) 2013-2025 Asynchronous B.V.
5
+ #
4
6
  # Permission is hereby granted, free of charge, to any person obtaining a copy
5
7
  # of this software and associated documentation files (the "Software"), to deal
6
8
  # in the Software without restriction, including without limitation the rights
7
9
  # to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
8
10
  # copies of the Software, and to permit persons to whom the Software is
9
11
  # furnished to do so, subject to the following conditions:
10
- #
12
+ #
11
13
  # The above copyright notice and this permission notice shall be included in
12
14
  # all copies or substantial portions of the Software.
13
- #
15
+ #
14
16
  # THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
15
17
  # IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
16
18
  # FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
@@ -20,16 +22,8 @@
20
22
  # THE SOFTWARE.
21
23
 
22
24
  DAEMON_CONTROLLER_FILES = [
23
- "README.markdown", "LICENSE.txt", "Rakefile", "daemon_controller.gemspec",
24
- "lib/**/*.rb",
25
- "debian.template/**/*",
26
- "rpm/**/*",
27
- "spec/*.rb",
28
- "spec/run_echo_server"
29
- ]
30
-
31
- DAEMON_CONTROLLER_DEBIAN_EXCLUDE_FILES = [
32
- "Rakefile",
33
- "debian.template/**/*",
34
- "rpm/**/*"
25
+ "README.md", "LICENSE.txt", "Rakefile", "daemon_controller.gemspec",
26
+ "lib/**/*.rb",
27
+ "spec/*.rb",
28
+ "spec/run_echo_server"
35
29
  ]
@@ -1,16 +1,18 @@
1
+ # frozen_string_literal: true
2
+
1
3
  # daemon_controller, library for robust daemon management
2
- # Copyright (c) 2010 Phusion
3
- #
4
+ # Copyright (c) 2010-2025 Asynchronous B.V.
5
+ #
4
6
  # Permission is hereby granted, free of charge, to any person obtaining a copy
5
7
  # of this software and associated documentation files (the "Software"), to deal
6
8
  # in the Software without restriction, including without limitation the rights
7
9
  # to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
8
10
  # copies of the Software, and to permit persons to whom the Software is
9
11
  # furnished to do so, subject to the following conditions:
10
- #
12
+ #
11
13
  # The above copyright notice and this permission notice shall be included in
12
14
  # all copies or substantial portions of the Software.
13
- #
15
+ #
14
16
  # THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
15
17
  # IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
16
18
  # FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
@@ -20,7 +22,6 @@
20
22
  # THE SOFTWARE.
21
23
 
22
24
  # This helper script is used for daemonizing a command by executing it and
23
- # then exiting ourselves. Used on Ruby 1.9 and JRuby because forking may not
24
- # be safe/supported on all platforms.
25
+ # then exiting ourselves. Used because Process.spawn has no setsid support.
25
26
  Process.setsid
26
- Process.spawn(ARGV[0])
27
+ Process.spawn(ARGV[0])
@@ -1,16 +1,18 @@
1
+ # frozen_string_literal: true
2
+
1
3
  # daemon_controller, library for robust daemon management
2
- # Copyright (c) 2010-2014 Phusion
3
- #
4
+ # Copyright (c) 2010-2025 Asynchronous B.V.
5
+ #
4
6
  # Permission is hereby granted, free of charge, to any person obtaining a copy
5
7
  # of this software and associated documentation files (the "Software"), to deal
6
8
  # in the Software without restriction, including without limitation the rights
7
9
  # to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
8
10
  # copies of the Software, and to permit persons to whom the Software is
9
11
  # furnished to do so, subject to the following conditions:
10
- #
12
+ #
11
13
  # The above copyright notice and this permission notice shall be included in
12
14
  # all copies or substantial portions of the Software.
13
- #
15
+ #
14
16
  # THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
15
17
  # IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
16
18
  # FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
@@ -20,8 +22,8 @@
20
22
  # THE SOFTWARE.
21
23
 
22
24
  class DaemonController
23
- MAJOR = 1
24
- MINOR = 2
25
- TINY = 0
26
- VERSION_STRING = "#{MAJOR}.#{MINOR}.#{TINY}"
25
+ MAJOR = 2
26
+ MINOR = 0
27
+ TINY = 0
28
+ VERSION_STRING = "#{MAJOR}.#{MINOR}.#{TINY}"
27
29
  end # class DaemonController