Merge pull request #3018 from greatroar/background-check

Fix IsProcessBackground on Linux with stdin redirection
This commit is contained in:
MichaelEischer 2020-10-16 22:50:19 +02:00 committed by GitHub
commit 39fe1e96fe
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
4 changed files with 36 additions and 15 deletions

View file

@ -4,6 +4,6 @@ package termstatus
// IsProcessBackground reports whether the current process is running in the // IsProcessBackground reports whether the current process is running in the
// background. Not implemented for this platform. // background. Not implemented for this platform.
func IsProcessBackground() bool { func IsProcessBackground(uintptr) bool {
return false return false
} }

View file

@ -1,21 +1,23 @@
package termstatus package termstatus
import ( import (
"syscall"
"unsafe"
"github.com/restic/restic/internal/debug" "github.com/restic/restic/internal/debug"
"golang.org/x/sys/unix"
) )
// IsProcessBackground reports whether the current process is running in the background. // IsProcessBackground reports whether the current process is running in the
func IsProcessBackground() bool { // background. fd must be a file descriptor for the terminal.
var pid int func IsProcessBackground(fd uintptr) bool {
_, _, err := syscall.Syscall(syscall.SYS_IOCTL, uintptr(syscall.Stdin), syscall.TIOCGPGRP, uintptr(unsafe.Pointer(&pid))) bg, err := isProcessBackground(fd)
if err != nil {
if err != 0 {
debug.Log("Can't check if we are in the background. Using default behaviour. Error: %s\n", err.Error()) debug.Log("Can't check if we are in the background. Using default behaviour. Error: %s\n", err.Error())
return false return false
} }
return bg
return pid != syscall.Getpgrp() }
func isProcessBackground(fd uintptr) (bool, error) {
pid, err := unix.IoctlGetInt(int(fd), unix.TIOCGPGRP)
return pid != unix.Getpgrp(), err
} }

View file

@ -0,0 +1,19 @@
package termstatus
import (
"os"
"testing"
rtest "github.com/restic/restic/internal/test"
)
func TestIsProcessBackground(t *testing.T) {
tty, err := os.Open("/dev/tty")
if err != nil {
t.Skipf("can't open terminal: %v", err)
}
defer tty.Close()
_, err = isProcessBackground(tty.Fd())
rtest.OK(t, err)
}

View file

@ -95,7 +95,7 @@ func (t *Terminal) run(ctx context.Context) {
for { for {
select { select {
case <-ctx.Done(): case <-ctx.Done():
if IsProcessBackground() { if IsProcessBackground(t.fd) {
// ignore all messages, do nothing, we are in the background process group // ignore all messages, do nothing, we are in the background process group
continue continue
} }
@ -104,7 +104,7 @@ func (t *Terminal) run(ctx context.Context) {
return return
case msg := <-t.msg: case msg := <-t.msg:
if IsProcessBackground() { if IsProcessBackground(t.fd) {
// ignore all messages, do nothing, we are in the background process group // ignore all messages, do nothing, we are in the background process group
continue continue
} }
@ -136,7 +136,7 @@ func (t *Terminal) run(ctx context.Context) {
} }
case stat := <-t.status: case stat := <-t.status:
if IsProcessBackground() { if IsProcessBackground(t.fd) {
// ignore all messages, do nothing, we are in the background process group // ignore all messages, do nothing, we are in the background process group
continue continue
} }