/
process_unix.go
98 lines (81 loc) · 1.43 KB
/
process_unix.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
// +build aix darwin dragonfly freebsd linux netbsd openbsd solaris
/**
* @program: lemo
*
* @description:
*
* @author: lemo
*
* @create: 2020-01-02 19:33
**/
package utils
import (
"os"
"strings"
"sync"
"syscall"
)
type pro int
const Process pro = iota
type proc struct {
Cmd *cmd
}
var pMux sync.Mutex
var worker []*proc
var workerNumber int
var managerPid int
func (p pro) Fork(fn func(), number int) {
switch os.Getenv("FORK_CHILD") {
case "":
managerPid = os.Getpid()
_ = os.Setenv("FORK_CHILD", "TRUE")
workerNumber = number
p.run()
default:
go fn()
Signal.ListenKill().Done(func(sig os.Signal) {
os.Exit(0)
})
}
}
func (p pro) run() {
pMux.Lock()
defer pMux.Unlock()
for i := 0; i < workerNumber; i++ {
var c = Cmd.New(strings.Join(os.Args, " "))
err := c.c.Start()
if err != nil {
panic(err)
}
go func() { _, _ = c.c.Process.Wait() }()
worker = append(worker, &proc{Cmd: c})
}
}
func (p pro) Kill(pid int) {
pMux.Lock()
defer pMux.Unlock()
if managerPid == 0 {
return
}
_ = Signal.KillGroup(pid, syscall.SIGTERM)
for i := 0; i < len(worker); i++ {
if worker[i].Cmd.c.Process.Pid == pid {
worker = append(worker[0:i], worker[i+1:]...)
}
}
}
func (p pro) Reload() {
if managerPid == 0 {
return
}
for _, proc := range worker {
p.Kill(proc.Cmd.c.Process.Pid)
}
p.run()
}
func (p pro) Manager() int {
return managerPid
}
func (p pro) Worker() []*proc {
return worker
}