2012-10-10 06:13:05 +08:00
|
|
|
package tail
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bufio"
|
|
|
|
"fmt"
|
|
|
|
"io"
|
|
|
|
"log"
|
|
|
|
"os"
|
|
|
|
"time"
|
|
|
|
)
|
|
|
|
|
|
|
|
type Line struct {
|
|
|
|
Text string
|
|
|
|
UnixTime int64
|
|
|
|
}
|
|
|
|
|
|
|
|
type Tail struct {
|
|
|
|
Filename string
|
|
|
|
Lines chan *Line
|
|
|
|
|
2012-10-12 12:29:06 +08:00
|
|
|
useinotify bool
|
2012-10-10 06:13:05 +08:00
|
|
|
maxlinesize int
|
|
|
|
file *os.File
|
|
|
|
reader *bufio.Reader
|
2012-10-12 12:29:06 +08:00
|
|
|
watcher FileWatcher
|
2012-10-10 06:13:05 +08:00
|
|
|
|
|
|
|
stop chan bool
|
|
|
|
created chan bool
|
|
|
|
}
|
|
|
|
|
|
|
|
// TailFile channels the lines of a logfile along with timestamp. If
|
|
|
|
// end is true, channel only newly added lines. If retry is true, tail
|
|
|
|
// the file name (not descriptor) and retry on file open/read errors.
|
2012-10-12 12:29:06 +08:00
|
|
|
func TailFile(filename string, maxlinesize int, end bool, retry bool, useinotify bool) (*Tail, error) {
|
2012-10-10 06:13:05 +08:00
|
|
|
t := &Tail{
|
|
|
|
filename,
|
|
|
|
make(chan *Line),
|
2012-10-12 12:29:06 +08:00
|
|
|
useinotify,
|
2012-10-10 06:13:05 +08:00
|
|
|
maxlinesize,
|
|
|
|
nil,
|
|
|
|
nil,
|
2012-10-12 12:29:06 +08:00
|
|
|
nil,
|
2012-10-10 06:13:05 +08:00
|
|
|
make(chan bool),
|
|
|
|
make(chan bool)}
|
|
|
|
|
2012-10-12 12:29:06 +08:00
|
|
|
if !useinotify {
|
|
|
|
log.Println("Warning: not using inotify; will poll ", filename)
|
|
|
|
t.watcher = NewPollingFileWatcher(filename)
|
|
|
|
} else {
|
|
|
|
t.watcher = NewInotifyFileWatcher(filename)
|
|
|
|
}
|
|
|
|
|
2012-10-10 06:13:05 +08:00
|
|
|
go t.tailFileSync(end, retry)
|
|
|
|
|
2012-10-11 06:13:13 +08:00
|
|
|
return t, nil
|
2012-10-10 06:13:05 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
func (tail *Tail) Stop() {
|
|
|
|
tail.stop <- true
|
|
|
|
tail.close()
|
|
|
|
}
|
|
|
|
|
|
|
|
func (tail *Tail) close() {
|
|
|
|
close(tail.Lines)
|
|
|
|
if tail.file != nil {
|
|
|
|
tail.file.Close()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (tail *Tail) reopen(wait bool) {
|
|
|
|
if tail.file != nil {
|
|
|
|
tail.file.Close()
|
|
|
|
}
|
|
|
|
for {
|
|
|
|
var err error
|
|
|
|
tail.file, err = os.Open(tail.Filename)
|
|
|
|
if err != nil {
|
|
|
|
if os.IsNotExist(err) && wait {
|
2012-10-12 12:29:06 +08:00
|
|
|
err := tail.watcher.BlockUntilExists()
|
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
2012-10-10 06:13:05 +08:00
|
|
|
}
|
|
|
|
continue
|
|
|
|
}
|
2012-10-12 12:29:06 +08:00
|
|
|
log.Println(fmt.Sprintf("Unable to reopen file (%s): %s", tail.Filename, err))
|
2012-10-10 06:13:05 +08:00
|
|
|
}
|
|
|
|
return
|
|
|
|
}
|
|
|
|
return // unreachable
|
|
|
|
}
|
|
|
|
|
|
|
|
func (tail *Tail) readLine() ([]byte, error) {
|
|
|
|
line, isPrefix, err := tail.reader.ReadLine()
|
|
|
|
|
|
|
|
if isPrefix && err == nil {
|
|
|
|
// line is longer than what we can accept.
|
|
|
|
// ignore the rest of this line.
|
|
|
|
for {
|
|
|
|
_, isPrefix, err := tail.reader.ReadLine()
|
|
|
|
if !isPrefix || err != nil {
|
|
|
|
return line, err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return line, err
|
|
|
|
}
|
|
|
|
|
|
|
|
func (tail *Tail) tailFileSync(end bool, retry bool) {
|
|
|
|
tail.reopen(retry)
|
|
|
|
|
2012-10-12 12:29:06 +08:00
|
|
|
var changes chan bool
|
|
|
|
|
2012-10-10 06:13:05 +08:00
|
|
|
// Note: seeking to end happens only at the beginning; never
|
|
|
|
// during subsequent re-opens.
|
|
|
|
if end {
|
|
|
|
_, err := tail.file.Seek(0, 2) // seek to end of the file
|
|
|
|
if err != nil {
|
|
|
|
// TODO: don't panic here
|
|
|
|
panic(fmt.Sprintf("seek error: %s", err))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
tail.reader = bufio.NewReaderSize(tail.file, tail.maxlinesize)
|
|
|
|
|
|
|
|
for {
|
|
|
|
line, err := tail.readLine()
|
|
|
|
|
2012-10-12 12:29:06 +08:00
|
|
|
if err == nil {
|
|
|
|
if line != nil {
|
|
|
|
tail.Lines <- &Line{string(line), getCurrentTime()}
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if err != io.EOF {
|
|
|
|
log.Println("Error reading file; skipping this file - ", err)
|
|
|
|
tail.close()
|
|
|
|
return
|
|
|
|
}
|
2012-10-10 06:13:05 +08:00
|
|
|
|
2012-10-12 12:29:06 +08:00
|
|
|
// When end of file is reached, wait for more data to
|
|
|
|
// become available. Wait strategy is based on the
|
|
|
|
// `tail.watcher` implementation (inotify or polling).
|
|
|
|
if err == io.EOF {
|
|
|
|
if changes == nil {
|
|
|
|
changes = tail.watcher.ChangeEvents()
|
|
|
|
}
|
2012-10-10 06:13:05 +08:00
|
|
|
|
2012-10-12 12:47:58 +08:00
|
|
|
select {
|
|
|
|
case _, ok := <-changes:
|
|
|
|
if !ok {
|
|
|
|
// file got deleted/renamed
|
|
|
|
if retry {
|
|
|
|
log.Printf("File %s has been moved (logrotation?); reopening..", tail.Filename)
|
|
|
|
tail.reopen(retry)
|
|
|
|
log.Printf("File %s has been reopened.", tail.Filename)
|
|
|
|
tail.reader = bufio.NewReaderSize(tail.file, tail.maxlinesize)
|
|
|
|
changes = nil
|
|
|
|
continue
|
|
|
|
} else {
|
|
|
|
log.Printf("File %s has gone away; skipping this file.\n", tail.Filename)
|
|
|
|
tail.close()
|
|
|
|
return
|
|
|
|
}
|
2012-10-12 12:29:06 +08:00
|
|
|
}
|
2012-10-12 12:47:58 +08:00
|
|
|
case <-tail.stop:
|
|
|
|
// stop the tailer if requested.
|
|
|
|
// FIXME: respect DRY (see below)
|
|
|
|
return
|
2012-10-12 12:29:06 +08:00
|
|
|
}
|
2012-10-12 12:47:58 +08:00
|
|
|
|
2012-10-12 12:29:06 +08:00
|
|
|
}
|
2012-10-10 06:13:05 +08:00
|
|
|
}
|
|
|
|
|
2012-10-12 12:29:06 +08:00
|
|
|
// stop the tailer if requested.
|
2012-10-10 06:13:05 +08:00
|
|
|
select {
|
2012-10-12 12:29:06 +08:00
|
|
|
case <-tail.stop:
|
2012-10-10 06:13:05 +08:00
|
|
|
return
|
|
|
|
default:
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// get current time in unix timestamp
|
|
|
|
func getCurrentTime() int64 {
|
|
|
|
return time.Now().UTC().Unix()
|
|
|
|
}
|