-
Notifications
You must be signed in to change notification settings - Fork 9
/
Copy pathrun
executable file
·300 lines (270 loc) · 7.02 KB
/
run
1
2
3
4
5
6
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
245
246
247
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
300
#!/usr/bin/env ruby
require File.expand_path(File.dirname(__FILE__) + '/shared')
ENV['PATH'] = OLD_PATH
STDOUT.sync = STDERR.sync = true
require 'rubygems'
require 'optparse'
OPTIONS = {}
def parse_options
parser = OptionParser.new do |opts|
nl = "\n" + ' ' * 37
opts.banner = "Usage: ./run [options] COMMAND..."
opts.separator "Run a command with various options."
opts.separator ""
opts.separator "Options:"
opts.on("--log-file FILE", "Log to file in addition to printing to terminal") do |value|
OPTIONS[:log_file] = value
end
opts.on("--append", "Append to log file instead of overwriting it.") do
OPTIONS[:append] = true
end
opts.on("--syslog", "Log to syslog in additional to printing to terminal") do
OPTIONS[:syslog] = true
end
opts.on("--pv", "Pipe output through pv") do
OPTIONS[:pv] = true
end
opts.on("--program-name NAME", "Run command with the given argv[0]") do |value|
OPTIONS[:program_name] = value
end
opts.on("--status-file FILE") do |value|
OPTIONS[:status_file] = value
end
opts.on("--lock-file FILE") do |value|
OPTIONS[:lock_file] = value
end
opts.on("--email-to ADDRESSES", "Separated by comma") do |value|
OPTIONS[:email_to] = value
end
end
begin
parser.parse!
rescue OptionParser::ParseError => e
STDERR.puts e
STDERR.puts
STDERR.puts "Please see '--help' for valid options."
exit 1
end
if ARGV.size < 1
STDERR.puts parser
exit 1
end
end
def can_exec_directly?
return !OPTIONS[:log_file] && !OPTIONS[:syslog] && !OPTIONS[:pv] && !OPTIONS[:status_file] && !OPTIONS[:lock_file] && !OPTIONS[:email_to]
end
def start
parse_options
begin
lock_file = create_lock_file
create_log_file
write_status_file('')
STDIN.reopen("/dev/null", "r")
if has_sink?
main_process = spawn(main_command,
:out => :pipe,
:err => :pipe)
sink_process = spawn_sink(main_process)
[:in, :out].each do |channel|
main_process[channel].close if main_process[channel]
main_process.delete(channel)
end
elsif can_exec_directly?
exec(*main_command)
else
command = spawn(main_command)
end
while true
begin
Process.waitpid(main_process[:pid])
exit_code = ($?.exitstatus || 2)
main_process.delete(:pid)
break
rescue Errno::ECHILD
exit_code = 1
main_process.delete(:pid)
break
rescue SignalException => e
signame = get_signal_name(e)
Process.kill(signame, main_process[:pid])
end
end
# TODO: are we supposed to wait for the output sink process?
# If we only wait for the command then the output sink process
# may not have finished processing all the output yet.
# But if we wait for both, and the command spawns subprocesses,
# then the output sink process doesn't exit until all those
# subprocesses have also exited. Maybe we should provide a
# command line option for this.
if sink_process
sink_process[:pids].each do |pid|
begin
Process.waitpid(pid)
rescue Errno::ECHILD
# Ignore exception.
end
end
sink_process = nil
end
write_status_file(exit_code)
if OPTIONS[:email_to]
email(
OPTIONS[:email_from],
OPTIONS[:email_to],
"Command finished with exit code #{exit_code}: #{ARGV.join(' ')}",
"Command: #{ARGV.join(' ')}\n" +
"Exit code: #{exit_code}\n" +
"Host: #{`hostname`.strip}\n" +
"Log file: #{OPTIONS[:log_file]}\n"
)
end
exit(exit_code)
rescue SystemExit
raise
rescue Exception => e
if OPTIONS[:log_file]
f = File.open(OPTIONS[:log_file], 'a')
else
f = IO.popen("logger -t '#{program_name}:runner[#{$$}]'", "w")
end
begin
f.puts("#{e.class}: #{e.message || e}\n " +
e.backtrace.join("\n "))
ensure
f.close
end
Process.kill('SIGTERM', main_process[:pid]) if main_process && main_process[:pid]
raise e
ensure
delete_lock_file(lock_file) if lock_file
end
end
def spawn(command, options)
result = {}
if options[:in] == :pipe
stdin_pipe = IO.pipe
result[:in] = stdin_pipe[1]
end
if options[:out] == :pipe
stdout_pipe = IO.pipe
result[:out] = stdout_pipe[0]
end
result[:pid] = fork do
if options[:in] == :pipe
STDIN.reopen(stdin_pipe[0])
elsif options[:in].is_a?(Array)
STDIN.reopen(*options[:in])
elsif options[:in]
STDIN.reopen(options[:in])
end
if options[:out] == :pipe
STDOUT.reopen(stdout_pipe[1])
elsif options[:out].is_a?(Array)
STDOUT.reopen(*options[:out])
elsif options[:out]
STDOUT.reopen(options[:out])
end
if options[:err] == :pipe
STDERR.reopen(stdout_pipe[1])
elsif options[:err].is_a?(Array)
STDERR.reopen(*options[:err])
elsif options[:err]
STDERR.reopen(options[:err])
end
stdin_pipe[1].close if stdin_pipe
stdout_pipe[0].close if stdout_pipe
if options[:setsid]
Process.setsid
end
begin
exec(*command)
rescue SystemCallError => e
STDERR.puts "Cannot execute '#{command.join(' ')}': #{e}"
exit! 127
end
end
stdin_pipe[0].close if stdin_pipe
stdout_pipe[1].close if stdout_pipe
return result
end
def has_sink?
return OPTIONS[:syslog] || OPTIONS[:log_file] || OPTIONS[:pv]
end
def spawn_sink(main_process)
if OPTIONS[:syslog]
command = ["#{TOOLS_DIR}/syslog-tee", "-t", "#{program_name}[#{main_process[:pid]}]"]
elsif OPTIONS[:log_file]
if OPTIONS[:append]
command = ["tee", "-a", OPTIONS[:log_file]]
else
command = ["tee", OPTIONS[:log_file]]
end
elsif OPTIONS[:pv]
command = pv_command
else
raise "Unknown options combination"
end
# We setsid because we don't want to let terminal signals reach any sink processes.
if (OPTIONS[:syslog] || OPTIONS[:log_file]) && OPTIONS[:pv]
# Pipeline: main_process | sink | pv
sink_process = spawn(command, :setsid => true, :in => main_process[:out], :out => :pipe)
pv_process = spawn(pv_command, :setsid => true, :in => sink_process[:out])
sink_process[:out].close
sink_process.delete(:out)
return { :pids => [sink_process[:pid], pv_process[:pid]] }
else
# Pipeline: main_process | sink
sink_process = spawn(command, :setsid => true, :in => main_process[:out])
return { :pids => [sink_process[:pid]] }
end
end
def main_command
if OPTIONS[:program_name]
args = ARGV.dup
argv0 = args.shift
return [[argv0, OPTIONS[:program_name]], *args]
else
return ARGV
end
end
def program_name
return OPTIONS[:program_name] || File.basename(ARGV[0])
end
def pv_command
return ["pv"]
end
def get_signal_name(signal_exception)
if signal_exception.is_a?(Interrupt)
return "SIGINT"
else
return signal_exception.signm
end
end
def create_lock_file
if OPTIONS[:lock_file]
File.open(OPTIONS[:lock_file], File::WRONLY | File::EXCL | File::CREAT) do |f|
f.puts Process.pid
end
return true
else
return nil
end
rescue Errno::EEXIST
raise "Lock file #{OPTIONS[:lock_file]} already exists!"
end
def create_log_file
if OPTIONS[:log_file]
File.open(OPTIONS[:log_file], OPTIONS[:append] ? 'a' : 'w').close
end
end
def delete_lock_file(lock_file)
File.unlink(OPTIONS[:lock_file])
end
def write_status_file(content)
if OPTIONS[:status_file]
File.open(OPTIONS[:status_file], "w") do |f|
f.write(content.to_s)
end
end
end
start