2018-12-05 14:03:41 +03:00
package dnsforward
import (
"bytes"
"compress/gzip"
"encoding/json"
"fmt"
"os"
"sync"
"time"
"github.com/go-test/deep"
2018-12-29 19:12:22 +03:00
"github.com/hmage/golibs/log"
2018-12-05 14:03:41 +03:00
)
var (
fileWriteLock sync . Mutex
)
const enableGzip = false
2019-02-10 20:47:43 +03:00
// clearLogBuffer flushes the current buffer to file and resets the current buffer
func ( l * queryLog ) clearLogBuffer ( ) error {
// flush remainder to file
l . logBufferLock . Lock ( )
flushBuffer := l . logBuffer
l . logBuffer = nil
l . logBufferLock . Unlock ( )
err := l . flushToFile ( flushBuffer )
if err != nil {
log . Printf ( "Saving querylog to file failed: %s" , err )
return err
}
return nil
}
// flushToFile saves the specified log entries to the query log file
func ( l * queryLog ) flushToFile ( buffer [ ] * logEntry ) error {
2018-12-05 14:03:41 +03:00
if len ( buffer ) == 0 {
return nil
}
start := time . Now ( )
var b bytes . Buffer
e := json . NewEncoder ( & b )
for _ , entry := range buffer {
err := e . Encode ( entry )
if err != nil {
log . Printf ( "Failed to marshal entry: %s" , err )
return err
}
}
elapsed := time . Since ( start )
log . Printf ( "%d elements serialized via json in %v: %d kB, %v/entry, %v/entry" , len ( buffer ) , elapsed , b . Len ( ) / 1024 , float64 ( b . Len ( ) ) / float64 ( len ( buffer ) ) , elapsed / time . Duration ( len ( buffer ) ) )
err := checkBuffer ( buffer , b )
if err != nil {
log . Printf ( "failed to check buffer: %s" , err )
return err
}
var zb bytes . Buffer
2019-02-10 20:47:43 +03:00
filename := l . logFile
2018-12-05 14:03:41 +03:00
// gzip enabled?
if enableGzip {
filename += ".gz"
zw := gzip . NewWriter ( & zb )
2019-02-10 20:47:43 +03:00
zw . Name = l . logFile
2018-12-05 14:03:41 +03:00
zw . ModTime = time . Now ( )
_ , err = zw . Write ( b . Bytes ( ) )
if err != nil {
log . Printf ( "Couldn't compress to gzip: %s" , err )
zw . Close ( )
return err
}
if err = zw . Close ( ) ; err != nil {
log . Printf ( "Couldn't close gzip writer: %s" , err )
return err
}
} else {
zb = b
}
fileWriteLock . Lock ( )
defer fileWriteLock . Unlock ( )
f , err := os . OpenFile ( filename , os . O_WRONLY | os . O_CREATE | os . O_APPEND , 0644 )
if err != nil {
log . Printf ( "failed to create file \"%s\": %s" , filename , err )
return err
}
defer f . Close ( )
n , err := f . Write ( zb . Bytes ( ) )
if err != nil {
log . Printf ( "Couldn't write to file: %s" , err )
return err
}
log . Printf ( "ok \"%s\": %v bytes written" , filename , n )
return nil
}
func checkBuffer ( buffer [ ] * logEntry , b bytes . Buffer ) error {
l := len ( buffer )
d := json . NewDecoder ( & b )
i := 0
for d . More ( ) {
entry := & logEntry { }
err := d . Decode ( entry )
if err != nil {
log . Printf ( "Failed to decode: %s" , err )
return err
}
if diff := deep . Equal ( entry , buffer [ i ] ) ; diff != nil {
log . Printf ( "decoded buffer differs: %s" , diff )
return fmt . Errorf ( "decoded buffer differs: %s" , diff )
}
i ++
}
if i != l {
err := fmt . Errorf ( "check fail: %d vs %d entries" , l , i )
log . Print ( err )
return err
}
log . Printf ( "check ok: %d entries" , i )
return nil
}
2019-02-10 20:47:43 +03:00
func ( l * queryLog ) rotateQueryLog ( ) error {
from := l . logFile
to := l . logFile + ".1"
2018-12-05 14:03:41 +03:00
if enableGzip {
2019-02-10 20:47:43 +03:00
from = l . logFile + ".gz"
to = l . logFile + ".gz.1"
2018-12-05 14:03:41 +03:00
}
if _ , err := os . Stat ( from ) ; os . IsNotExist ( err ) {
// do nothing, file doesn't exist
return nil
}
err := os . Rename ( from , to )
if err != nil {
log . Printf ( "Failed to rename querylog: %s" , err )
return err
}
log . Printf ( "Rotated from %s to %s successfully" , from , to )
return nil
}
2019-02-10 20:47:43 +03:00
func ( l * queryLog ) periodicQueryLogRotate ( ) {
2018-12-05 14:03:41 +03:00
for range time . Tick ( queryLogRotationPeriod ) {
2019-02-10 20:47:43 +03:00
err := l . rotateQueryLog ( )
2018-12-05 14:03:41 +03:00
if err != nil {
log . Printf ( "Failed to rotate querylog: %s" , err )
// do nothing, continue rotating
}
}
}
2019-02-10 20:47:43 +03:00
func ( l * queryLog ) genericLoader ( onEntry func ( entry * logEntry ) error , needMore func ( ) bool , timeWindow time . Duration ) error {
2018-12-05 14:03:41 +03:00
now := time . Now ( )
// read from querylog files, try newest file first
2019-01-24 20:11:01 +03:00
var files [ ] string
2018-12-05 14:03:41 +03:00
if enableGzip {
files = [ ] string {
2019-02-10 20:47:43 +03:00
l . logFile + ".gz" ,
l . logFile + ".gz.1" ,
2018-12-05 14:03:41 +03:00
}
} else {
files = [ ] string {
2019-02-10 20:47:43 +03:00
l . logFile ,
l . logFile + ".1" ,
2018-12-05 14:03:41 +03:00
}
}
// read from all files
for _ , file := range files {
if ! needMore ( ) {
break
}
if _ , err := os . Stat ( file ) ; os . IsNotExist ( err ) {
// do nothing, file doesn't exist
continue
}
f , err := os . Open ( file )
if err != nil {
log . Printf ( "Failed to open file \"%s\": %s" , file , err )
// try next file
continue
}
defer f . Close ( )
var d * json . Decoder
if enableGzip {
zr , err := gzip . NewReader ( f )
if err != nil {
log . Printf ( "Failed to create gzip reader: %s" , err )
continue
}
defer zr . Close ( )
d = json . NewDecoder ( zr )
} else {
d = json . NewDecoder ( f )
}
i := 0
over := 0
max := 10000 * time . Second
var sum time . Duration
// entries on file are in oldest->newest order
// we want maxLen newest
for d . More ( ) {
if ! needMore ( ) {
break
}
var entry logEntry
err := d . Decode ( & entry )
if err != nil {
log . Printf ( "Failed to decode: %s" , err )
// next entry can be fine, try more
continue
}
if now . Sub ( entry . Time ) > timeWindow {
2018-12-29 17:23:42 +03:00
// log.Tracef("skipping entry") // debug logging
2018-12-05 14:03:41 +03:00
continue
}
if entry . Elapsed > max {
over ++
} else {
sum += entry . Elapsed
}
i ++
err = onEntry ( & entry )
if err != nil {
return err
}
}
elapsed := time . Since ( now )
var perunit time . Duration
var avg time . Duration
if i > 0 {
perunit = elapsed / time . Duration ( i )
avg = sum / time . Duration ( i )
}
log . Printf ( "file \"%s\": read %d entries in %v, %v/entry, %v over %v, %v avg" , file , i , elapsed , perunit , over , max , avg )
}
return nil
}