-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathlib.go
128 lines (109 loc) · 3.09 KB
/
lib.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
package main
import (
"fmt"
"io"
"log"
"os"
"os/exec"
"sync"
"syscall"
"github.com/creack/pty"
"github.com/fsnotify/fsnotify"
)
func Run(c string) {
fmt.Println("Running base command...")
cmd := exec.Command("bash", "-c", c)
cmd.Stdout = os.Stdout
cmd.Stderr = os.Stderr
fmt.Println("Running command: ", cmd.String())
ptmx, err := pty.Start(cmd)
if err != nil {
log.Fatal(err)
}
defer ptmx.Close()
go io.Copy(os.Stdout, ptmx)
go io.Copy(ptmx, os.Stdin)
done := make(chan error, 1)
// Wait for the command to finish in a goroutine
go func() {
done <- cmd.Wait() // Send the error (if any) when done
}()
for <-done != nil {
fmt.Println("Error running base command")
os.Exit(1)
}
}
func eventListener(restart chan bool, wg *sync.WaitGroup) {
defer wg.Done()
watcher, err := fsnotify.NewWatcher()
if err != nil {
log.Println("Error creating watcher:", err)
return
}
defer watcher.Close()
curdir, _ := os.Getwd()
err = watcher.Add(curdir)
if err != nil {
log.Println("Error adding path to watcher:", err)
return
}
for {
select {
case event, ok := <-watcher.Events:
if !ok {
return
}
// Restart if any file changes occur
if event.Op&fsnotify.Write == fsnotify.Write ||
event.Op&fsnotify.Create == fsnotify.Create ||
event.Op&fsnotify.Remove == fsnotify.Remove ||
event.Op&fsnotify.Rename == fsnotify.Rename {
restart <- true
}
case err, ok := <-watcher.Errors:
if !ok {
return
}
log.Println("Watcher error:", err)
}
}
}
func runCommand(wg *sync.WaitGroup, run string, args []string, restart chan bool) {
defer wg.Done()
for {
cmd := exec.Command(run, args...)
ptmx, err := pty.Start(cmd)
if err != nil {
log.Fatal(err)
}
defer ptmx.Close()
go io.Copy(os.Stdout, ptmx)
go io.Copy(ptmx, os.Stdin)
// Create a channel to signal when the command has finished
done := make(chan error, 1)
// Wait for the command to finish in a goroutine
go func() {
done <- cmd.Wait() // Send the error (if any) when done
}()
select {
case err := <-done:
if err != nil {
log.Println("Command exited with error:", err)
} else {
log.Println("Command exited successfully")
}
restart <- false
case <-restart:
log.Println("Restarting command...")
// Gracefully stop the command
if err := cmd.Process.Signal(syscall.SIGTERM); err != nil {
log.Println("Error sending SIGTERM:", err)
if err := cmd.Process.Kill(); err != nil {
log.Fatal("Error killing process:", err)
}
}
// Wait for the command to actually terminate
cmd.Wait()
}
}
}