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
| | #!/usr/bin/env ruby
# Copyright (C) 2013-2016 all contributors <dtas-all@nongnu.org>
# License: GPL-3.0+ <https://www.gnu.org/licenses/gpl-3.0.txt>
# frozen_string_literal: true
require 'optparse'
require 'dtas/edit_client'
require 'dtas/sigevent'
require 'dtas/watchable'
include DTAS::EditClient
c = client_socket
sources = c.req('source ls') || "(unknown)"
usage = "Usage: #{DTAS_PROGNAME} [OPTIONS] SOURCENAME\n" \
"available SOURCENAME values: #{sources}\n" \
dry_run = verbose = false
watch = defined?(DTAS::Watchable)
OptionParser.new('', 24, ' ') do |op|
op.banner = usage
watch and
op.on('-N', '--no-watch', 'disable inotify support') { watch = false }
op.on('-n', '--dry-run', 'only print commands, do not run them') {
dry_run = true
}
op.on('-V', '--verbose', 'print out commands sent to change the source') {
verbose = true
}
op.on('-h', '--help') { puts(op.to_s); exit }
op.parse!(ARGV)
end
ARGV.size <= 1 or abort usage
name = ARGV[0] || "sox"
st_in = $stdin.stat
buf = c.req(%W(source cat #{name}))
abort(buf) if buf =~ /\AERR/
orig = YAML.load(buf)
commit_update = lambda do |buf|
source = YAML.load(buf)
cmd = %W(source ed #{name})
update_cmd_env(cmd, orig, source)
# nil OK
%w(tryorder command).each { |field| cmd << "#{field}=#{source[field]}" }
warn(Shellwords.join(cmd)) if verbose || dry_run
c.req_ok(cmd) unless dry_run
orig = source
end
if st_in.file? || st_in.pipe?
buf = $stdin.read
commit_update.call(buf)
else
include DTAS::SpawnFix
tmp = tmpyaml
tmp_path = tmp.path
do_update = lambda { commit_update.call(File.read(tmp_path)) }
tmp.write(buf << DTAS_DISCLAIMER)
cmd = "#{editor} #{tmp_path}"
sev = DTAS::Sigevent.new
rset = [ sev ]
if watch
ino = DTAS::Watchable::InotifyReadableIter.new
ino.watch_files(tmp_path, do_update)
rset << ino
end
trap(:CHLD) { sev.signal }
pid = spawn(cmd)
begin
r = IO.select(rset) or next
r[0].each do |io|
case io
when sev
_, status = Process.waitpid2(pid, Process::WNOHANG)
status or next
status.success? or abort "#{cmd} failed: #{status.inspect}"
do_update.call
exit
when ino
ino.readable_iter # calls do_update
end
end
end while true
end
|