|
@@ -10,14 +10,11 @@ import (
|
|
|
|
|
|
"github.com/Sirupsen/logrus"
|
|
"github.com/Sirupsen/logrus"
|
|
"github.com/docker/docker/daemon/logger"
|
|
"github.com/docker/docker/daemon/logger"
|
|
- "github.com/docker/docker/pkg/filenotify"
|
|
|
|
"github.com/docker/docker/pkg/ioutils"
|
|
"github.com/docker/docker/pkg/ioutils"
|
|
"github.com/docker/docker/pkg/jsonlog"
|
|
"github.com/docker/docker/pkg/jsonlog"
|
|
"github.com/docker/docker/pkg/tailfile"
|
|
"github.com/docker/docker/pkg/tailfile"
|
|
)
|
|
)
|
|
|
|
|
|
-const maxJSONDecodeRetry = 20000
|
|
|
|
-
|
|
|
|
func decodeLogLine(dec *json.Decoder, l *jsonlog.JSONLog) (*logger.Message, error) {
|
|
func decodeLogLine(dec *json.Decoder, l *jsonlog.JSONLog) (*logger.Message, error) {
|
|
l.Reset()
|
|
l.Reset()
|
|
if err := dec.Decode(l); err != nil {
|
|
if err := dec.Decode(l); err != nil {
|
|
@@ -35,7 +32,6 @@ func decodeLogLine(dec *json.Decoder, l *jsonlog.JSONLog) (*logger.Message, erro
|
|
// created by this driver.
|
|
// created by this driver.
|
|
func (l *JSONFileLogger) ReadLogs(config logger.ReadConfig) *logger.LogWatcher {
|
|
func (l *JSONFileLogger) ReadLogs(config logger.ReadConfig) *logger.LogWatcher {
|
|
logWatcher := logger.NewLogWatcher()
|
|
logWatcher := logger.NewLogWatcher()
|
|
-
|
|
|
|
go l.readLogs(logWatcher, config)
|
|
go l.readLogs(logWatcher, config)
|
|
return logWatcher
|
|
return logWatcher
|
|
}
|
|
}
|
|
@@ -85,7 +81,7 @@ func (l *JSONFileLogger) readLogs(logWatcher *logger.LogWatcher, config logger.R
|
|
l.mu.Unlock()
|
|
l.mu.Unlock()
|
|
|
|
|
|
notifyRotate := l.writer.NotifyRotate()
|
|
notifyRotate := l.writer.NotifyRotate()
|
|
- followLogs(latestFile, logWatcher, notifyRotate, config.Since)
|
|
|
|
|
|
+ l.followLogs(latestFile, logWatcher, notifyRotate, config.Since)
|
|
|
|
|
|
l.mu.Lock()
|
|
l.mu.Lock()
|
|
delete(l.readers, logWatcher)
|
|
delete(l.readers, logWatcher)
|
|
@@ -121,95 +117,81 @@ func tailFile(f io.ReadSeeker, logWatcher *logger.LogWatcher, tail int, since ti
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
-func followLogs(f *os.File, logWatcher *logger.LogWatcher, notifyRotate chan interface{}, since time.Time) {
|
|
|
|
- dec := json.NewDecoder(f)
|
|
|
|
- l := &jsonlog.JSONLog{}
|
|
|
|
|
|
+func (l *JSONFileLogger) followLogs(f *os.File, logWatcher *logger.LogWatcher, notifyRotate chan interface{}, since time.Time) {
|
|
|
|
+ var (
|
|
|
|
+ rotated bool
|
|
|
|
|
|
- fileWatcher, err := filenotify.New()
|
|
|
|
- if err != nil {
|
|
|
|
- logWatcher.Err <- err
|
|
|
|
- }
|
|
|
|
- defer fileWatcher.Close()
|
|
|
|
|
|
+ dec = json.NewDecoder(f)
|
|
|
|
+ log = &jsonlog.JSONLog{}
|
|
|
|
+ writeNotify = l.writeNotifier.Subscribe()
|
|
|
|
+ watchClose = logWatcher.WatchClose()
|
|
|
|
+ )
|
|
|
|
|
|
- var retries int
|
|
|
|
- for {
|
|
|
|
- msg, err := decodeLogLine(dec, l)
|
|
|
|
|
|
+ reopenLogFile := func() error {
|
|
|
|
+ f.Close()
|
|
|
|
+ f, err := os.Open(f.Name())
|
|
if err != nil {
|
|
if err != nil {
|
|
- if err != io.EOF {
|
|
|
|
- // try again because this shouldn't happen
|
|
|
|
- if _, ok := err.(*json.SyntaxError); ok && retries <= maxJSONDecodeRetry {
|
|
|
|
- dec = json.NewDecoder(f)
|
|
|
|
- retries++
|
|
|
|
- continue
|
|
|
|
- }
|
|
|
|
-
|
|
|
|
- // io.ErrUnexpectedEOF is returned from json.Decoder when there is
|
|
|
|
- // remaining data in the parser's buffer while an io.EOF occurs.
|
|
|
|
- // If the json logger writes a partial json log entry to the disk
|
|
|
|
- // while at the same time the decoder tries to decode it, the race condition happens.
|
|
|
|
- if err == io.ErrUnexpectedEOF && retries <= maxJSONDecodeRetry {
|
|
|
|
- reader := io.MultiReader(dec.Buffered(), f)
|
|
|
|
- dec = json.NewDecoder(reader)
|
|
|
|
- retries++
|
|
|
|
- continue
|
|
|
|
- }
|
|
|
|
- logWatcher.Err <- err
|
|
|
|
- return
|
|
|
|
- }
|
|
|
|
|
|
+ return err
|
|
|
|
+ }
|
|
|
|
+ dec = json.NewDecoder(f)
|
|
|
|
+ rotated = true
|
|
|
|
+ return nil
|
|
|
|
+ }
|
|
|
|
|
|
- logrus.WithField("logger", "json-file").Debugf("waiting for events")
|
|
|
|
- if err := fileWatcher.Add(f.Name()); err != nil {
|
|
|
|
- logrus.WithField("logger", "json-file").Warn("falling back to file poller")
|
|
|
|
- fileWatcher.Close()
|
|
|
|
- fileWatcher = filenotify.NewPollingWatcher()
|
|
|
|
- if err := fileWatcher.Add(f.Name()); err != nil {
|
|
|
|
- logrus.Errorf("error watching log file for modifications: %v", err)
|
|
|
|
- logWatcher.Err <- err
|
|
|
|
- }
|
|
|
|
|
|
+ readToEnd := func() error {
|
|
|
|
+ for {
|
|
|
|
+ msg, err := decodeLogLine(dec, log)
|
|
|
|
+ if err != nil {
|
|
|
|
+ return err
|
|
}
|
|
}
|
|
- select {
|
|
|
|
- case <-fileWatcher.Events():
|
|
|
|
- dec = json.NewDecoder(f)
|
|
|
|
- fileWatcher.Remove(f.Name())
|
|
|
|
- continue
|
|
|
|
- case <-fileWatcher.Errors():
|
|
|
|
- fileWatcher.Remove(f.Name())
|
|
|
|
- logWatcher.Err <- err
|
|
|
|
- return
|
|
|
|
- case <-logWatcher.WatchClose():
|
|
|
|
- fileWatcher.Remove(f.Name())
|
|
|
|
- return
|
|
|
|
- case <-notifyRotate:
|
|
|
|
- f, err = os.Open(f.Name())
|
|
|
|
- if err != nil {
|
|
|
|
- logWatcher.Err <- err
|
|
|
|
- return
|
|
|
|
- }
|
|
|
|
-
|
|
|
|
- dec = json.NewDecoder(f)
|
|
|
|
- fileWatcher.Remove(f.Name())
|
|
|
|
- fileWatcher.Add(f.Name())
|
|
|
|
|
|
+ if !since.IsZero() && msg.Timestamp.Before(since) {
|
|
continue
|
|
continue
|
|
}
|
|
}
|
|
|
|
+ logWatcher.Msg <- msg
|
|
}
|
|
}
|
|
|
|
+ }
|
|
|
|
|
|
- retries = 0 // reset retries since we've succeeded
|
|
|
|
- if !since.IsZero() && msg.Timestamp.Before(since) {
|
|
|
|
- continue
|
|
|
|
|
|
+ defer func() {
|
|
|
|
+ l.writeNotifier.Evict(writeNotify)
|
|
|
|
+ if rotated {
|
|
|
|
+ f.Close()
|
|
}
|
|
}
|
|
|
|
+ }()
|
|
|
|
+
|
|
|
|
+ for {
|
|
select {
|
|
select {
|
|
- case logWatcher.Msg <- msg:
|
|
|
|
- case <-logWatcher.WatchClose():
|
|
|
|
- logWatcher.Msg <- msg
|
|
|
|
- for {
|
|
|
|
- msg, err := decodeLogLine(dec, l)
|
|
|
|
- if err != nil {
|
|
|
|
|
|
+ case <-watchClose:
|
|
|
|
+ readToEnd()
|
|
|
|
+ return
|
|
|
|
+ case <-notifyRotate:
|
|
|
|
+ readToEnd()
|
|
|
|
+ if err := reopenLogFile(); err != nil {
|
|
|
|
+ logWatcher.Err <- err
|
|
|
|
+ return
|
|
|
|
+ }
|
|
|
|
+ case _, ok := <-writeNotify:
|
|
|
|
+ if err := readToEnd(); err == io.EOF {
|
|
|
|
+ if !ok {
|
|
|
|
+ // The writer is closed, no new logs will be generated.
|
|
return
|
|
return
|
|
}
|
|
}
|
|
- if !since.IsZero() && msg.Timestamp.Before(since) {
|
|
|
|
- continue
|
|
|
|
|
|
+
|
|
|
|
+ select {
|
|
|
|
+ case <-notifyRotate:
|
|
|
|
+ if err := reopenLogFile(); err != nil {
|
|
|
|
+ logWatcher.Err <- err
|
|
|
|
+ return
|
|
|
|
+ }
|
|
|
|
+ default:
|
|
|
|
+ dec = json.NewDecoder(f)
|
|
}
|
|
}
|
|
- logWatcher.Msg <- msg
|
|
|
|
|
|
+
|
|
|
|
+ } else if err == io.ErrUnexpectedEOF {
|
|
|
|
+ dec = json.NewDecoder(io.MultiReader(dec.Buffered(), f))
|
|
|
|
+ } else {
|
|
|
|
+ logrus.Errorf("Failed to decode json log %s: %v", f.Name(), err)
|
|
|
|
+ logWatcher.Err <- err
|
|
|
|
+ return
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|