2025-09-07 12:57:01 +02:00
|
|
|
//go:build unix
|
2018-01-17 23:02:47 +01:00
|
|
|
|
2025-09-07 11:58:02 +02:00
|
|
|
package terminal
|
2018-01-17 23:02:47 +01:00
|
|
|
|
|
|
|
import (
|
|
|
|
"os"
|
|
|
|
"os/exec"
|
|
|
|
"os/signal"
|
|
|
|
|
2018-01-26 19:21:14 +01:00
|
|
|
"github.com/restic/restic/internal/debug"
|
2018-01-17 23:02:47 +01:00
|
|
|
"github.com/restic/restic/internal/errors"
|
2023-05-25 17:31:51 +02:00
|
|
|
|
|
|
|
"golang.org/x/sys/unix"
|
2018-01-17 23:02:47 +01:00
|
|
|
)
|
|
|
|
|
2020-10-03 13:27:23 +02:00
|
|
|
func startForeground(cmd *exec.Cmd) (bg func() error, err error) {
|
2025-04-14 20:18:58 +02:00
|
|
|
// run the command in its own process group
|
|
|
|
// this ensures that sending ctrl-c to restic will not immediately stop the backend process.
|
|
|
|
cmd.SysProcAttr = &unix.SysProcAttr{
|
|
|
|
Setpgid: true,
|
|
|
|
}
|
|
|
|
|
2018-01-17 23:02:47 +01:00
|
|
|
// open the TTY, we need the file descriptor
|
|
|
|
tty, err := os.OpenFile("/dev/tty", os.O_RDWR, 0)
|
|
|
|
if err != nil {
|
2018-01-26 19:21:14 +01:00
|
|
|
debug.Log("unable to open tty: %v", err)
|
2025-04-11 23:54:02 +02:00
|
|
|
return startFallback(cmd)
|
|
|
|
}
|
|
|
|
|
|
|
|
// only move child process to foreground if restic is in the foreground
|
2025-09-08 10:55:58 +02:00
|
|
|
prev, err := tcgetpgrp(int(tty.Fd()))
|
2025-04-11 23:54:02 +02:00
|
|
|
if err != nil {
|
|
|
|
_ = tty.Close()
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2025-09-08 10:55:58 +02:00
|
|
|
self := getpgrp()
|
2025-04-11 23:54:02 +02:00
|
|
|
if prev != self {
|
2025-09-07 12:57:01 +02:00
|
|
|
debug.Log("restic is not controlling the tty; err = %v", err)
|
2025-04-11 23:54:02 +02:00
|
|
|
if err := tty.Close(); err != nil {
|
|
|
|
return nil, err
|
2018-01-26 19:21:14 +01:00
|
|
|
}
|
2025-04-11 23:54:02 +02:00
|
|
|
return startFallback(cmd)
|
2018-01-17 23:02:47 +01:00
|
|
|
}
|
|
|
|
|
2025-04-11 23:54:02 +02:00
|
|
|
// Prevent getting suspended when interacting with the tty
|
2023-05-25 17:31:51 +02:00
|
|
|
signal.Ignore(unix.SIGTTIN)
|
|
|
|
signal.Ignore(unix.SIGTTOU)
|
2018-01-17 23:02:47 +01:00
|
|
|
|
|
|
|
// start the process
|
|
|
|
err = cmd.Start()
|
|
|
|
if err != nil {
|
|
|
|
_ = tty.Close()
|
|
|
|
return nil, errors.Wrap(err, "cmd.Start")
|
|
|
|
}
|
|
|
|
|
|
|
|
// move the command's process group into the foreground
|
2025-09-08 10:55:58 +02:00
|
|
|
err = tcsetpgrp(int(tty.Fd()), cmd.Process.Pid)
|
2018-01-17 23:02:47 +01:00
|
|
|
if err != nil {
|
|
|
|
_ = tty.Close()
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
bg = func() error {
|
2023-05-25 17:31:51 +02:00
|
|
|
signal.Reset(unix.SIGTTIN)
|
|
|
|
signal.Reset(unix.SIGTTOU)
|
2018-01-17 23:02:47 +01:00
|
|
|
|
|
|
|
// reset the foreground process group
|
2025-09-08 10:55:58 +02:00
|
|
|
err = tcsetpgrp(int(tty.Fd()), prev)
|
2018-01-17 23:02:47 +01:00
|
|
|
if err != nil {
|
|
|
|
_ = tty.Close()
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
return tty.Close()
|
|
|
|
}
|
|
|
|
|
|
|
|
return bg, nil
|
|
|
|
}
|
2025-04-11 23:54:02 +02:00
|
|
|
|
|
|
|
func startFallback(cmd *exec.Cmd) (bg func() error, err error) {
|
|
|
|
bg = func() error {
|
|
|
|
return nil
|
|
|
|
}
|
2025-04-14 20:18:58 +02:00
|
|
|
|
2025-04-11 23:54:02 +02:00
|
|
|
return bg, cmd.Start()
|
|
|
|
}
|