mirror of
https://github.com/filebrowser/filebrowser.git
synced 2024-06-07 23:00:43 +00:00
6d5ceae8b4
This prevents the accumulation of zombie processes when using async (&) event commands. Also log async command failures.
119 lines
2.4 KiB
Go
119 lines
2.4 KiB
Go
package runner
|
|
|
|
import (
|
|
"fmt"
|
|
"log"
|
|
"os"
|
|
"os/exec"
|
|
"strings"
|
|
|
|
"github.com/filebrowser/filebrowser/v2/settings"
|
|
"github.com/filebrowser/filebrowser/v2/users"
|
|
)
|
|
|
|
// Runner is a commands runner.
|
|
type Runner struct {
|
|
Enabled bool
|
|
*settings.Settings
|
|
}
|
|
|
|
// RunHook runs the hooks for the before and after event.
|
|
func (r *Runner) RunHook(fn func() error, evt, path, dst string, user *users.User) error {
|
|
path = user.FullPath(path)
|
|
dst = user.FullPath(dst)
|
|
|
|
if r.Enabled {
|
|
if val, ok := r.Commands["before_"+evt]; ok {
|
|
for _, command := range val {
|
|
err := r.exec(command, "before_"+evt, path, dst, user)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
err := fn()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
if r.Enabled {
|
|
if val, ok := r.Commands["after_"+evt]; ok {
|
|
for _, command := range val {
|
|
err := r.exec(command, "after_"+evt, path, dst, user)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func (r *Runner) exec(raw, evt, path, dst string, user *users.User) error {
|
|
blocking := true
|
|
|
|
if strings.HasSuffix(raw, "&") {
|
|
blocking = false
|
|
raw = strings.TrimSpace(strings.TrimSuffix(raw, "&"))
|
|
}
|
|
|
|
command, err := ParseCommand(r.Settings, raw)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
envMapping := func(key string) string {
|
|
switch key {
|
|
case "FILE":
|
|
return path
|
|
case "SCOPE":
|
|
return user.Scope
|
|
case "TRIGGER":
|
|
return evt
|
|
case "USERNAME":
|
|
return user.Username
|
|
case "DESTINATION":
|
|
return dst
|
|
default:
|
|
return os.Getenv(key)
|
|
}
|
|
}
|
|
for i, arg := range command {
|
|
if i == 0 {
|
|
continue
|
|
}
|
|
|
|
command[i] = os.Expand(arg, envMapping)
|
|
}
|
|
|
|
cmd := exec.Command(command[0], command[1:]...) //nolint:gosec
|
|
cmd.Env = append(os.Environ(), fmt.Sprintf("FILE=%s", path))
|
|
cmd.Env = append(cmd.Env, fmt.Sprintf("SCOPE=%s", user.Scope)) //nolint:gocritic
|
|
cmd.Env = append(cmd.Env, fmt.Sprintf("TRIGGER=%s", evt))
|
|
cmd.Env = append(cmd.Env, fmt.Sprintf("USERNAME=%s", user.Username))
|
|
cmd.Env = append(cmd.Env, fmt.Sprintf("DESTINATION=%s", dst))
|
|
|
|
cmd.Stdin = os.Stdin
|
|
cmd.Stdout = os.Stdout
|
|
cmd.Stderr = os.Stderr
|
|
|
|
if !blocking {
|
|
log.Printf("[INFO] Nonblocking Command: \"%s\"", strings.Join(command, " "))
|
|
defer func() {
|
|
go func() {
|
|
err := cmd.Wait()
|
|
if err != nil {
|
|
log.Printf("[INFO] Nonblocking Command \"%s\" failed: %s", strings.Join(command, " "), err)
|
|
}
|
|
}()
|
|
}()
|
|
return cmd.Start()
|
|
}
|
|
|
|
log.Printf("[INFO] Blocking Command: \"%s\"", strings.Join(command, " "))
|
|
return cmd.Run()
|
|
}
|