apex / log

Structured logging package for Go.
MIT License
1.36k stars 110 forks source link

fatal error: concurrent map iteration and map write #74

Open dpup opened 5 years ago

dpup commented 5 years ago

We experienced a crash today that I'm having trouble diagnosing. The error occurred when calling logger.Info inside a goroutine. There were several parallel tasks, all which log when they complete. This code has been running, unchanged for months and I don't believe we've had any issues before.

Are logger.WithField and logger.WithError intended to be threadsafe?

Thanks in advance.


The error occurs at line 155 in (e *Entry) mergedFields

151 func (e *Entry) mergedFields() Fields {
152     f := Fields{}
153 
154     for _, fields := range e.fields {
155         for k, v := range fields {
156             f[k] = v
157         }
158     }
159
160     return f
161 }
/usr/lib/golang/src/runtime/panic.go:617 +0x72 fp=0xc001167d48 sp=0xc001167d18 pc=0x43a112
runtime.mapiternext(0xc001167e28)
/usr/lib/golang/src/runtime/map.go:860 +0x597 fp=0xc001167dd0 sp=0xc001167d48 pc=0x41ccd7
vendor/github.com/apex/log.(*Entry).mergedFields(0xc00017d260, 0x14)
/go/src/vendor/github.com/apex/log/entry.go:155 +0x6c fp=0xc001167e98 sp=0xc001167dd0 pc=0x77aeac
vendor/github.com/apex/log.(*Entry).finalize(0xc00017d260, 0x1, 0x1779ed3, 0x1c, 0x0)
/go/src/vendor/github.com/apex/log/entry.go:167 +0x2f fp=0xc001167ee0 sp=0xc001167e98 pc=0x77afef
vendor/github.com/apex/log.(*Logger).log(0x2978830, 0x1, 0xc00017d260, 0x1779ed3, 0x1c)
/go/src/vendor/github.com/apex/log/logger.go:146 +0x5c fp=0xc001167f28 sp=0xc001167ee0 pc=0x77c31c
vendor/github.com/apex/log.(*Entry).Info(...)
tj commented 5 years ago

hmm yea definitely should be, sounds like a bug to me, thanks for reporting!

dpup commented 4 years ago

Just an FYI that we're seeing this regularly now. A refinement on the original data, the code is essentially calling logger.Info("...") from multiple goroutines.

seanami commented 4 years ago

I have some more info about how this bug occurs:

We can address this in our own code by ensuring that fields is never written to after it's used to make a logger (by copying it).

But I wonder if it would also make sense to prevent this type of mistake at the apex/log level?

Either of those options would ensure that no further writes to a fields map can occur that would be concurrent with a call to log something.

bertzzie commented 1 year ago

Hi, opening a PR to fix the issue, please let me know if there's anything to improve there :)