-
Notifications
You must be signed in to change notification settings - Fork 29
/
Copy pathgow_cmd.go
145 lines (122 loc) · 3.24 KB
/
gow_cmd.go
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
package main
import (
"errors"
"io"
"os"
"os/exec"
"sync"
"syscall"
"github.com/mitranim/gg"
)
type Cmd struct {
Mained
sync.Mutex
Cmd *exec.Cmd
Stdin io.WriteCloser // Writer half of `self.Cmd.Stdin` for forwarding.
}
func (self *Cmd) Deinit() {
defer gg.Lock(self).Unlock()
self.DeinitUnsync()
}
func (self *Cmd) DeinitUnsync() {
// Should kill the entire sub-process group.
self.BroadcastUnsync(syscall.SIGTERM)
self.Cmd = nil
self.Stdin = nil
}
/*
Note: because the sub-process state changes concurrently, it may change
from "running" to "stopped" immediately after this function returns, while the
caller is taking actions based on the old "running" state. Decision making at
the callsite must account for this.
*/
func (self *Cmd) IsRunning() bool {
defer gg.Lock(self).Unlock()
return self.IsRunningUnsync()
}
func (self *Cmd) IsRunningUnsync() bool {
cmd := self.Cmd
return cmd != nil && cmd.ProcessState == nil
}
func (self *Cmd) Restart() {
defer gg.Lock(self).Unlock()
self.DeinitUnsync()
cmd, stdin := self.MakeCmd()
if cmd == nil {
return
}
// Starting the sub-process populates its `.Process`,
// which allows us to kill the sub-process group on demand.
err := cmd.Start()
if err != nil {
log.Println(`unable to start subcommand:`, err)
return
}
self.Cmd = cmd
self.Stdin = stdin
go self.Main().CmdWait(cmd)
}
func (self *Cmd) MakeCmd() (*exec.Cmd, io.WriteCloser) {
main := self.Main()
cmd := main.Opt.MakeCmd()
// Causes the OS to assign process group ID = `cmd.Process.Pid`.
// We use this to broadcast signals to the entire sub-process group.
cmd.SysProcAttr = &syscall.SysProcAttr{Setpgid: true}
cmd.Stdout = os.Stdout
cmd.Stderr = os.Stderr
/**
TODO: ideally, this should be done only when using terminal raw mode. In
non-raw mode, we should simply connect the stdio as-is:
cmd.Stdin = os.Stdin
cmd.Stdout = os.Stdout
cmd.Stderr = os.Stderr
However, this ideal approach does not seem to work properly for subprocesses
that actually use stdin, such as a simple "echo" implementation. For the
time being, we always pipe stdin "manually". This needs additional
investigation.
*/
stdin, err := cmd.StdinPipe()
if err != nil {
log.Println(`unable to initialize subcommand stdin:`, err)
return nil, nil
}
return cmd, stdin
}
func (self *Cmd) Broadcast(sig syscall.Signal) {
defer gg.Lock(self).Unlock()
self.BroadcastUnsync(sig)
}
/*
Sends the signal to the sub-process group, denoted by the negative sign on the
PID. Requires `syscall.SysProcAttr{Setpgid: true}`.
*/
func (self *Cmd) BroadcastUnsync(sig syscall.Signal) {
proc := self.ProcUnsync()
if proc != nil {
gg.Nop1(syscall.Kill(-proc.Pid, sig))
}
}
func (self *Cmd) WriteChar(char byte) {
// On the author's system at the time of writing, locking takes under a
// microsecond and unlocking takes just over two microseconds. Because this
// is intended for manual user input, this amount of delay per keystroke
// seems acceptable.
defer gg.Lock(self).Unlock()
stdin := self.Stdin
if stdin == nil {
return
}
_, err := writeByte(stdin, char)
if errors.Is(err, os.ErrClosed) {
self.Stdin = nil
return
}
gg.Try(err)
}
func (self *Cmd) ProcUnsync() *os.Process {
cmd := self.Cmd
if cmd != nil {
return cmd.Process
}
return nil
}