Skip to content

Commit

Permalink
Deterministic calc of POSIX return code for Fatal
Browse files Browse the repository at this point in the history
  • Loading branch information
cardil committed Apr 25, 2022
1 parent 75dacb4 commit 5fe5ca9
Show file tree
Hide file tree
Showing 6 changed files with 102 additions and 34 deletions.
23 changes: 16 additions & 7 deletions internal/exit/exit.go
Original file line number Diff line number Diff line change
Expand Up @@ -24,24 +24,32 @@ package exit

import "os"

var real = func() { os.Exit(1) }
var realFn = func(code int) { os.Exit(code) }

// Exit normally terminates the process by calling os.Exit(1). If the package
// is stubbed, it instead records a call in the testing spy.
// Deprecated: use With() instead.
func Exit() {
real()
With(1)
}

// With terminates the process by calling os.Exit(code). If the package is
// stubbed, it instead records a call in the testing spy.
func With(code int) {
realFn(code)
}

// A StubbedExit is a testing fake for os.Exit.
type StubbedExit struct {
Exited bool
prev func()
Code int
prev func(code int)
}

// Stub substitutes a fake for the call to os.Exit(1).
func Stub() *StubbedExit {
s := &StubbedExit{prev: real}
real = s.exit
s := &StubbedExit{prev: realFn}
realFn = s.exit
return s
}

Expand All @@ -56,9 +64,10 @@ func WithStub(f func()) *StubbedExit {

// Unstub restores the previous exit function.
func (se *StubbedExit) Unstub() {
real = se.prev
realFn = se.prev
}

func (se *StubbedExit) exit() {
func (se *StubbedExit) exit(code int) {
se.Exited = true
se.Code = code
}
23 changes: 16 additions & 7 deletions internal/exit/exit_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -18,25 +18,34 @@
// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
// THE SOFTWARE.

package exit
package exit_test

import (
"testing"

"github.com/stretchr/testify/assert"
"go.uber.org/zap/internal/exit"
)

func TestStub(t *testing.T) {
type want struct {
exit bool
code int
}
tests := []struct {
f func()
want bool
f func()
want
}{
{Exit, true},
{func() {}, false},
{func() {
exit.With(42)
}, want{exit: true, code: 42}},
{exit.Exit, want{exit: true, code: 1}},
{func() {}, want{}},
}

for _, tt := range tests {
s := WithStub(tt.f)
assert.Equal(t, tt.want, s.Exited, "Stub captured unexpected exit value.")
s := exit.WithStub(tt.f)
assert.Equal(t, tt.want.exit, s.Exited, "Stub captured unexpected exit value.")
assert.Equal(t, tt.want.code, s.Code, "Stub captured unexpected exit value.")
}
}
4 changes: 2 additions & 2 deletions logger.go
Original file line number Diff line number Diff line change
Expand Up @@ -288,9 +288,9 @@ func (log *Logger) check(lvl zapcore.Level, msg string) *zapcore.CheckedEntry {
ce = ce.Should(ent, zapcore.WriteThenPanic)
case zapcore.FatalLevel:
onFatal := log.onFatal
// Noop is the default value for CheckWriteAction, and it leads to
// The nil is the default value for CheckWriteAction, and it leads to
// continued execution after a Fatal which is unexpected.
if onFatal == zapcore.WriteThenNoop {
if onFatal == nil {
onFatal = zapcore.WriteThenFatal
}
ce = ce.Should(ent, onFatal)
Expand Down
10 changes: 9 additions & 1 deletion logger_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -213,15 +213,23 @@ func TestLoggerAlwaysFatals(t *testing.T) {
// Users can disable writing out fatal-level logs, but calls to logger.Fatal()
// should still terminate the process.
withLogger(t, FatalLevel+1, nil, func(logger *Logger, logs *observer.ObservedLogs) {
stub := exit.WithStub(func() { logger.Fatal("") })
stub := exit.WithStub(func() { logger.Fatal("foo") })
assert.True(t, stub.Exited, "Expected calls to logger.Fatal to terminate process.")
assert.Equal(t, 1, stub.Code, "Expected calls to logger.Fatal to terminate process with predictable retcode.")

logger = logger.WithOptions(OnFatal(zapcore.WriteThenPosixExitCode))
err := errors.New("bar")
stub = exit.WithStub(func() { logger.Fatal("foo", Error(err)) })
assert.True(t, stub.Exited, "Expected calls to logger.Fatal to terminate process.")
assert.Equal(t, 129, stub.Code, "Expected calls to logger.Fatal to terminate process with predictable retcode.")

stub = exit.WithStub(func() {
if ce := logger.Check(FatalLevel, ""); ce != nil {
ce.Write()
}
})
assert.True(t, stub.Exited, "Expected calls to logger.Check(FatalLevel, ...) to terminate process.")
assert.Equal(t, 1, stub.Code, "Expected calls to logger.Check(FatalLevel, ...) to terminate process with predictable retcode.")

assert.Equal(t, 0, logs.Len(), "Shouldn't write out logs when fatal-level logging is disabled.")
})
Expand Down
53 changes: 37 additions & 16 deletions zapcore/entry.go
Original file line number Diff line number Diff line change
Expand Up @@ -22,6 +22,7 @@ package zapcore

import (
"fmt"
"hash/crc32"
"runtime"
"strings"
"sync"
Expand Down Expand Up @@ -154,18 +155,30 @@ type Entry struct {

// CheckWriteAction indicates what action to take after a log entry is
// processed. Actions are ordered in increasing severity.
type CheckWriteAction uint8
type CheckWriteAction func(*CheckedEntry, []Field)

const (
var (
// WriteThenNoop indicates that nothing special needs to be done. It's the
// default behavior.
WriteThenNoop CheckWriteAction = iota
WriteThenNoop CheckWriteAction
// WriteThenGoexit runs runtime.Goexit after Write.
WriteThenGoexit
WriteThenGoexit CheckWriteAction = func(ce *CheckedEntry, fields []Field) {
runtime.Goexit()
}
// WriteThenPanic causes a panic after Write.
WriteThenPanic
// WriteThenFatal causes a fatal os.Exit after Write.
WriteThenFatal
WriteThenPanic CheckWriteAction = func(ce *CheckedEntry, fields []Field) {
panic(ce.Message)
}
// WriteThenFatal causes a fatal os.Exit(1) after Write.
WriteThenFatal CheckWriteAction = func(ce *CheckedEntry, fields []Field) {
exit.With(1)
}
// WriteThenPosixExitCode causes an os.Exit(code) after Write. The code is
// calculated deterministically from the message, or from attached error
// Field.
WriteThenPosixExitCode CheckWriteAction = func(ce *CheckedEntry, fields []Field) {
exit.With(retcode(ce, fields))
}
)

// CheckedEntry is an Entry together with a collection of Cores that have
Expand All @@ -186,7 +199,9 @@ func (ce *CheckedEntry) reset() {
ce.Entry = Entry{}
ce.ErrorOutput = nil
ce.dirty = false
ce.should = WriteThenNoop
if ce.should != nil {
ce.should = WriteThenNoop
}
for i := range ce.cores {
// don't keep references to cores
ce.cores[i] = nil
Expand Down Expand Up @@ -224,16 +239,11 @@ func (ce *CheckedEntry) Write(fields ...Field) {
ce.ErrorOutput.Sync()
}

should, msg := ce.should, ce.Message
putCheckedEntry(ce)

switch should {
case WriteThenPanic:
panic(msg)
case WriteThenFatal:
exit.Exit()
case WriteThenGoexit:
runtime.Goexit()
// Terminal operation
if ce.should != nil {
ce.should(ce, fields)
}
}

Expand All @@ -260,3 +270,14 @@ func (ce *CheckedEntry) Should(ent Entry, should CheckWriteAction) *CheckedEntry
ce.should = should
return ce
}

func retcode(ce *CheckedEntry, fields []Field) int {
msg := ce.Message
for _, field := range fields {
if field.Type == ErrorType {
msg = field.Interface.(error).Error()
break
}
}
return int(crc32.ChecksumIEEE([]byte(msg)))%254 + 1
}
23 changes: 22 additions & 1 deletion zapcore/entry_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -64,7 +64,7 @@ func TestPutNilEntry(t *testing.T) {
assert.NotNil(t, ce, "Expected only non-nil CheckedEntries in pool.")
assert.False(t, ce.dirty, "Unexpected dirty bit set.")
assert.Nil(t, ce.ErrorOutput, "Non-nil ErrorOutput.")
assert.Equal(t, WriteThenNoop, ce.should, "Unexpected terminal behavior.")
assert.Nil(t, ce.should, "Unexpected terminal behavior.")
assert.Equal(t, 0, len(ce.cores), "Expected empty slice of cores.")
assert.True(t, cap(ce.cores) > 0, "Expected pooled CheckedEntries to pre-allocate slice of Cores.")
}
Expand Down Expand Up @@ -128,5 +128,26 @@ func TestCheckedEntryWrite(t *testing.T) {
ce.Write()
})
assert.True(t, stub.Exited, "Expected to exit when WriteThenFatal is set.")
assert.Equal(t, 1, stub.Code, "Expected to exit when WriteThenFatal is set.")
})

t.Run("WriteThenPosixExitCode", func(t *testing.T) {
var ce *CheckedEntry
ce = ce.Should(Entry{Message: "foo"}, WriteThenPosixExitCode)
stub := exit.WithStub(func() {
ce.Write()
})
assert.True(t, stub.Exited, "Expected to exit when WriteThenPosixExitCode is set.")
assert.Equal(t, 38, stub.Code, "Expected to exit with specific code when WriteThenPosixExitCode is set.")
})

t.Run("Custom", func(t *testing.T) {
var called bool
var ce *CheckedEntry
ce = ce.Should(Entry{}, func(entry *CheckedEntry, fields []Field) {
called = true
})
ce.Write()
assert.True(t, called, "Expected to call custom action after Write.")
})
}

0 comments on commit 5fe5ca9

Please sign in to comment.