You signed in with another tab or window. Reload to refresh your session.You signed out in another tab or window. Reload to refresh your session.You switched accounts on another tab or window. Reload to refresh your session.Dismiss alert
//erp/dev runs other processes via bazelisk run, notably a Go-wrapper around Postgres.
The resulting process tree:
bazelisk run //erp/dev
# Starts a Go binary# The Go binary in turn calls the Go-wrapper for Postgres
bazelisk run //db/go_postgres
# go_postgres is a Go binary that calls the Postgres binary
postgres
The problem is that if //erp/dev sends SIGINT directly to the //db/go_postgres using the Go stdlib exec.Cmd, bazelisk run swallows the signal.
However, if I type ctrl-c at the terminal, Postgres receives the signal. I think this is because the shell wires up the process attributes differently from Go (maybe https://stackoverflow.com/a/35435038/30900?)
Context
As for why bazelisk run has this behavior:
Previously, when running bazelisk run under a shell, typing ctrl-c would send SIGINT twice to the child process. I think that's the same behavior I saw--processes started with Go propagate signals differently from processes started by the shell.
#307 noted the issue and recommended ignoring SIGINT. The resulting PR #322 fixed the issue by ignoring SIGINT but has a few related problems
Package signal will not block sending to c: the caller must ensure that c
has sufficient buffer space to keep up with the expected signal rate. For a
channel used for notification of just one signal value, a buffer of size 1
is sufficient.
The code calls cmd.Process.Kill(), which sends SIGKILL; it doesn't forward
the caught signal.
Design sketch
For my use case, I'd like to be able to forward whatever signal I want when starting processes via bazelisk run from Go. This is particularly important for Postgres, which has robust signal handling:
SIGTERM: smart shutdown
SIGINT: fast shutdown
SIGQUIT: immediate shutdown
SIGHUP: reload config
As a path forward (not yet validated) that avoids the double signaling:
bazelisk run starts processes so that automatic signal propagation is disabled:
SIGQUIT is specially recognized and causes a Java thread dump to be printed
without aborting the invocation. This behavior should be preserved when running
Bazel through Bazelisk.
Note that this also suppresses printing a Go stack dump on SIGQUIT. This is
intentional, because the Go stack dump is confusing: users tend to report it
instead of the much more valuable Java thread dump we asked them for.
Partially addresses bazelbuild#512. I'm not addressing it fully at this time because I'm
primarily concerned about signal handling by the Bazel client, not by an
arbitrary process executed by `bazel run`.
Reproduction
The situation to reproduce is a bit complex:
//erp/dev
.//erp/dev
runs other processes via bazelisk run, notably a Go-wrapper around Postgres.The resulting process tree:
The problem is that if
//erp/dev
sends SIGINT directly to the//db/go_postgres
using the Go stdlibexec.Cmd
, bazelisk run swallows the signal.However, if I type ctrl-c at the terminal, Postgres receives the signal. I think this is because the shell wires up the process attributes differently from Go (maybe https://stackoverflow.com/a/35435038/30900?)
Context
As for why
bazelisk run
has this behavior:Previously, when running
bazelisk run
under a shell, typing ctrl-c would send SIGINT twice to the child process. I think that's the same behavior I saw--processes started with Go propagate signals differently from processes started by the shell.#307 noted the issue and recommended ignoring SIGINT. The resulting PR #322 fixed the issue by ignoring SIGINT but has a few related problems
The relevant bazelisk code:
bazelisk/core/core.go
Line 579 in 89e6693
The two related problems:
The channel isn't buffered, meaning the signal delivery isn't guaranteed.
From https://pkg.go.dev/os/signal#Notify:
See also: https://stackoverflow.com/a/68593935/30900
The code calls cmd.Process.Kill(), which sends SIGKILL; it doesn't forward
the caught signal.
Design sketch
For my use case, I'd like to be able to forward whatever signal I want when starting processes via bazelisk run from Go. This is particularly important for Postgres, which has robust signal handling:
As a path forward (not yet validated) that avoids the double signaling:
The text was updated successfully, but these errors were encountered: