-
Notifications
You must be signed in to change notification settings - Fork 2
/
process.go
97 lines (82 loc) · 2.15 KB
/
process.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
/*
docker-systemd-shim - Shim to allow easy container control via systemd
Copyright (C) 2018 - 2019, Marc Hoersken <[email protected]>
This program is free software: you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation, either version 3 of the License, or
(at your option) any later version.
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
You should have received a copy of the GNU General Public License
along with this program. If not, see <https://www.gnu.org/licenses/>.
*/
package main
import (
"context"
"os"
"strings"
"syscall"
"time"
"github.com/containerd/cgroups"
)
func checkProcess(pid int) bool {
process, err := os.FindProcess(pid)
if err == nil {
err := process.Signal(syscall.Signal(0))
if err == nil {
return true
}
}
return false
}
func checkCGroup(pid int, cgroup string) bool {
found := false
control, err := cgroups.Load(cgroups.V1, cgroups.StaticPath(cgroup))
if err == nil {
subsystems := control.Subsystems()
for si := 0; si < len(subsystems); si++ {
subsystem := subsystems[si]
processes, err := control.Processes(subsystem.Name(), false)
if err == nil {
for pi := 0; pi < len(processes); pi++ {
process := processes[pi]
if process.Pid == pid {
if !strings.HasSuffix(process.Path, cgroup) {
return false
}
found = true
}
}
}
}
}
return found
}
func watchProcess(ctx context.Context, pid int, cgroup string, interval time.Duration) <-chan bool {
stopped := make(chan bool)
go func() {
ticker := time.NewTicker(interval)
defer ticker.Stop()
loop:
for {
select {
case <-ctx.Done():
// returning not to leak the goroutine
break loop
case <-ticker.C:
if !checkProcess(pid) {
stopped <- true
break loop
}
if len(cgroup) > 0 && !checkCGroup(pid, cgroup) {
stopped <- true
break loop
}
}
}
close(stopped)
}()
return stopped
}