Skip to content

Commit

Permalink
fix: Only register signal handlers if user intends to use them (#1215)
Browse files Browse the repository at this point in the history
closes #1212, and fixes a regression from #989. Previously we would only
register signal handlers if the user intended to use them. #989 changed
this behavior
[here](https://github.com/uber-go/fx/pull/989/files#diff-6c4b6ed7dc8834cef100f50dae61c30ffe7775a3f3f6f5a557517cb740c44a2dR649).

This regression meant that if you only used app.Start()/app.Stop(), fx
would register signal handlers for no reason as the user didn't use
app.Done/app.Wait.
  • Loading branch information
MarcoPolo authored Jun 25, 2024
1 parent d0d12e9 commit 38e64ec
Show file tree
Hide file tree
Showing 6 changed files with 66 additions and 9 deletions.
3 changes: 2 additions & 1 deletion app.go
Original file line number Diff line number Diff line change
Expand Up @@ -704,7 +704,6 @@ func (app *App) start(ctx context.Context) error {
if err := app.lifecycle.Start(ctx); err != nil {
return err
}
app.receivers.Start(ctx)
return nil
})
}
Expand Down Expand Up @@ -742,6 +741,7 @@ func (app *App) Stop(ctx context.Context) (err error) {
// Alternatively, a signal can be broadcast to all done channels manually by
// using the Shutdown functionality (see the [Shutdowner] documentation for details).
func (app *App) Done() <-chan os.Signal {
app.receivers.Start() // No-op if running
return app.receivers.Done()
}

Expand All @@ -752,6 +752,7 @@ func (app *App) Done() <-chan os.Signal {
// in the [ShutdownSignal] struct.
// Otherwise, the signal that was received will be set.
func (app *App) Wait() <-chan ShutdownSignal {
app.receivers.Start() // No-op if running
return app.receivers.Wait()
}

Expand Down
23 changes: 23 additions & 0 deletions app_internal_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -21,8 +21,10 @@
package fx

import (
"context"
"errors"
"fmt"
"os"
"sync"
"testing"

Expand Down Expand Up @@ -115,3 +117,24 @@ func TestAnnotationError(t *testing.T) {
assert.ErrorIs(t, err, wantErr)
assert.Contains(t, err.Error(), wantErr.Error())
}

// TestStartDoesNotRegisterSignals verifies that signal.Notify is not called
// when a user starts an app. signal.Notify should only be called when the
// .Wait/.Done are called. Note that app.Run calls .Wait() implicitly.
func TestStartDoesNotRegisterSignals(t *testing.T) {
app := New()
calledNotify := false

// Mock notify function to spy when this is called.
app.receivers.notify = func(c chan<- os.Signal, sig ...os.Signal) {
calledNotify = true
}
app.receivers.stopNotify = func(c chan<- os.Signal) {}

app.Start(context.Background())
defer app.Stop(context.Background())
assert.False(t, calledNotify, "notify should not be called when app starts")

_ = app.Wait() // User signals intent have fx listen for signals. This should call notify
assert.True(t, calledNotify, "notify should be called after Wait")
}
36 changes: 35 additions & 1 deletion app_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -2331,7 +2331,9 @@ func TestHookConstructors(t *testing.T) {
func TestDone(t *testing.T) {
t.Parallel()

done := fxtest.New(t).Done()
app := fxtest.New(t)
defer app.RequireStop()
done := app.Done()
require.NotNil(t, done, "Got a nil channel.")
select {
case sig := <-done:
Expand All @@ -2340,6 +2342,38 @@ func TestDone(t *testing.T) {
}
}

// TestShutdownThenWait tests that if we call .Shutdown before waiting, the wait
// will still return the last shutdown signal.
func TestShutdownThenWait(t *testing.T) {
t.Parallel()

var (
s Shutdowner
stopped bool
)
app := fxtest.New(
t,
Populate(&s),
Invoke(func(lc Lifecycle) {
lc.Append(StopHook(func() {
stopped = true
}))
}),
).RequireStart()
require.NotNil(t, s)

err := s.Shutdown(ExitCode(1337))
assert.NoError(t, err)
assert.False(t, stopped)

shutdownSig := <-app.Wait()
assert.Equal(t, 1337, shutdownSig.ExitCode)
assert.False(t, stopped)

app.RequireStop()
assert.True(t, stopped)
}

func TestReplaceLogger(t *testing.T) {
t.Parallel()

Expand Down
1 change: 1 addition & 0 deletions shutdown_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -116,6 +116,7 @@ func TestShutdown(t *testing.T) {
)

require.NoError(t, app.Start(context.Background()), "error starting app")
t.Cleanup(func() { app.Stop(context.Background()) }) // in t.Cleanup so this happens after all subtests return (not just this function)
defer require.NoError(t, app.Stop(context.Background()))

for i := 0; i < 10; i++ {
Expand Down
2 changes: 1 addition & 1 deletion signal.go
Original file line number Diff line number Diff line change
Expand Up @@ -102,7 +102,7 @@ func (recv *signalReceivers) running() bool {
return recv.shutdown != nil && recv.finished != nil
}

func (recv *signalReceivers) Start(ctx context.Context) {
func (recv *signalReceivers) Start() {
recv.m.Lock()
defer recv.m.Unlock()

Expand Down
10 changes: 4 additions & 6 deletions signal_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -74,9 +74,7 @@ func TestSignal(t *testing.T) {
t.Parallel()
t.Run("timeout", func(t *testing.T) {
recv := newSignalReceivers()
ctx, cancel := context.WithCancel(context.Background())
defer cancel()
recv.Start(ctx)
recv.Start()
timeoutCtx, cancel := context.WithTimeout(context.Background(), 0)
defer cancel()
err := recv.Stop(timeoutCtx)
Expand All @@ -86,8 +84,8 @@ func TestSignal(t *testing.T) {
recv := newSignalReceivers()
ctx, cancel := context.WithCancel(context.Background())
defer cancel()
recv.Start(ctx)
recv.Start(ctx) // should be a no-op if already running
recv.Start()
recv.Start() // should be a no-op if already running
require.NoError(t, recv.Stop(ctx))
})
t.Run("notify", func(t *testing.T) {
Expand All @@ -106,7 +104,7 @@ func TestSignal(t *testing.T) {
}
ctx, cancel := context.WithCancel(context.Background())
defer cancel()
recv.Start(ctx)
recv.Start()
stub <- syscall.SIGTERM
stub <- syscall.SIGTERM
require.Equal(t, syscall.SIGTERM, <-recv.Done())
Expand Down

0 comments on commit 38e64ec

Please sign in to comment.