From 4fae907d6f542477282425fb59a8c9f3a28b73d3 Mon Sep 17 00:00:00 2001 From: Ainar Garipov Date: Tue, 10 Nov 2020 19:00:55 +0300 Subject: [PATCH] Pull request: * querylog: fix end of log handling Merge in DNS/adguard-home from 2229-query-log to master Closes #2229. Squashed commit of the following: commit 32508a3f3b1e098869e1649a2774f1f17d14d41f Author: Ainar Garipov Date: Tue Nov 10 16:51:25 2020 +0300 * querylog: add test commit 774159cc313a0284a8bb8327489671e5d7a3e4eb Author: Ainar Garipov Date: Tue Nov 10 15:26:26 2020 +0300 * querylog: better errors commit 27b13a4dcaff9e8f9b08aec81c0c03f62ebd3fa5 Author: Ainar Garipov Date: Tue Nov 10 15:18:51 2020 +0300 * querylog: fix end of log handling --- internal/agherr/agherr.go | 8 ++++++++ internal/querylog/qlog_file.go | 22 ++++++++++++--------- internal/querylog/qlog_file_test.go | 29 ++++++++++++++++++++++++++++ internal/querylog/qlog_reader.go | 12 ++++++++---- internal/querylog/querylog_search.go | 13 ++++++++----- 5 files changed, 66 insertions(+), 18 deletions(-) diff --git a/internal/agherr/agherr.go b/internal/agherr/agherr.go index 4ef3b9c4794..dee2946601f 100644 --- a/internal/agherr/agherr.go +++ b/internal/agherr/agherr.go @@ -7,6 +7,14 @@ import ( "strings" ) +// Error is the constant error type. +type Error string + +// Error implements the error interface for Error. +func (err Error) Error() (msg string) { + return string(err) +} + // manyError is an error containing several wrapped errors. It is created to be // a simpler version of the API provided by github.com/joomcode/errorx. type manyError struct { diff --git a/internal/querylog/qlog_file.go b/internal/querylog/qlog_file.go index fdc202421de..5c5b8fc9dd1 100644 --- a/internal/querylog/qlog_file.go +++ b/internal/querylog/qlog_file.go @@ -1,18 +1,21 @@ package querylog import ( - "errors" "io" "os" "sync" "time" + "github.com/AdguardTeam/AdGuardHome/internal/agherr" "github.com/AdguardTeam/golibs/log" ) -// ErrSeekNotFound is returned from the Seek method -// if we failed to find the desired record -var ErrSeekNotFound = errors.New("Seek not found the record") +// ErrSeekNotFound is returned from Seek if when it fails to find the requested +// record. +const ErrSeekNotFound agherr.Error = "seek: record not found" + +// ErrEndOfLog is returned from Seek when the end of the current log is reached. +const ErrEndOfLog agherr.Error = "seek: end of log" // TODO: Find a way to grow buffer instead of relying on this value when reading strings const maxEntrySize = 16 * 1024 @@ -39,8 +42,7 @@ type QLogFile struct { // NewQLogFile initializes a new instance of the QLogFile func NewQLogFile(path string) (*QLogFile, error) { - f, err := os.OpenFile(path, os.O_RDONLY, 0644) - + f, err := os.OpenFile(path, os.O_RDONLY, 0o644) if err != nil { return nil, err } @@ -106,6 +108,8 @@ func (q *QLogFile) Seek(timestamp int64) (int64, int, error) { // the scope is too narrow and we won't find anything anymore log.Error("querylog: didn't find timestamp:%v", timestamp) return 0, depth, ErrSeekNotFound + } else if lineIdx == end && lineEndIdx == end { + return 0, depth, ErrEndOfLog } // Save the last found idx @@ -227,7 +231,7 @@ func (q *QLogFile) readNextLine(position int64) (string, int64, error) { // Look for the end of the prev line // This is where we'll read from - var startLine = int64(0) + startLine := int64(0) for i := relativePos - 1; i >= 0; i-- { if q.buffer[i] == '\n' { startLine = i + 1 @@ -293,7 +297,7 @@ func (q *QLogFile) readProbeLine(position int64) (string, int64, int64, error) { // Now start looking for the new line character starting // from the relativePos and going left - var startLine = int64(0) + startLine := int64(0) for i := relativePos - 1; i >= 0; i-- { if buffer[i] == '\n' { startLine = i + 1 @@ -301,7 +305,7 @@ func (q *QLogFile) readProbeLine(position int64) (string, int64, int64, error) { } } // Looking for the end of line now - var endLine = int64(bufferLen) + endLine := int64(bufferLen) lineEndIdx := endLine + seekPosition for i := relativePos; i < int64(bufferLen); i++ { if buffer[i] == '\n' { diff --git a/internal/querylog/qlog_file_test.go b/internal/querylog/qlog_file_test.go index 0329ceb5f26..de402e8b0b6 100644 --- a/internal/querylog/qlog_file_test.go +++ b/internal/querylog/qlog_file_test.go @@ -288,3 +288,32 @@ func TestQLogSeek(t *testing.T) { assert.Nil(t, err) assert.Equal(t, 1, depth) } + +func TestQLogSeek_ErrEndOfLog(t *testing.T) { + testDir := prepareTestDir() + t.Cleanup(func() { + _ = os.RemoveAll(testDir) + }) + + d := `{"T":"2020-08-31T18:44:23.911246629+03:00","QH":"wfqvjymurpwegyv","QT":"A","QC":"IN","CP":"","Answer":"","Result":{},"Elapsed":66286385,"Upstream":"tls://dns-unfiltered.adguard.com:853"} +{"T":"2020-08-31T18:44:25.376690873+03:00"} +{"T":"2020-08-31T18:44:25.382540454+03:00"} +` + f, err := ioutil.TempFile(testDir, "*.txt") + assert.Nil(t, err) + defer f.Close() + + _, err = f.WriteString(d) + assert.Nil(t, err) + + q, err := NewQLogFile(f.Name()) + assert.Nil(t, err) + defer q.Close() + + target, err := time.Parse(time.RFC3339, "2020-08-31T18:44:25.382540454+03:00") + assert.Nil(t, err) + + _, depth, err := q.Seek(target.UnixNano() + int64(time.Second)) + assert.Equal(t, ErrEndOfLog, err) + assert.Equal(t, 2, depth) +} diff --git a/internal/querylog/qlog_reader.go b/internal/querylog/qlog_reader.go index d376119d207..41677c3e183 100644 --- a/internal/querylog/qlog_reader.go +++ b/internal/querylog/qlog_reader.go @@ -1,6 +1,7 @@ package querylog import ( + "errors" "io" "github.com/AdguardTeam/AdGuardHome/internal/agherr" @@ -52,11 +53,14 @@ func (r *QLogReader) Seek(timestamp int64) error { for i := len(r.qFiles) - 1; i >= 0; i-- { q := r.qFiles[i] _, _, err := q.Seek(timestamp) - if err == nil { - // Our search is finished, we found the element we were looking for - // Update currentFile only, position is already set properly in the QLogFile + if err == nil || errors.Is(err, ErrEndOfLog) { + // Our search is finished, and we either found the + // element we were looking for or reached the end of the + // log. Update currentFile only, position is already + // set properly in QLogFile. r.currentFile = i - return nil + + return err } } diff --git a/internal/querylog/querylog_search.go b/internal/querylog/querylog_search.go index 024253e22a2..133a1dba13f 100644 --- a/internal/querylog/querylog_search.go +++ b/internal/querylog/querylog_search.go @@ -1,6 +1,7 @@ package querylog import ( + "errors" "io" "time" @@ -45,6 +46,7 @@ func (l *queryLog) search(params *searchParams) ([]*logEntry, time.Time) { // remove extra records entries = entries[:totalLimit] } + if params.offset > 0 { if len(entries) > params.offset { entries = entries[params.offset:] @@ -53,11 +55,9 @@ func (l *queryLog) search(params *searchParams) ([]*logEntry, time.Time) { oldest = time.Time{} } } - if len(entries) == totalLimit { - // change the "oldest" value here. - // we cannot use the "oldest" we got from "searchFiles" anymore - // because after adding in-memory records and removing extra records - // the situation has changed + + if len(entries) > 0 && len(entries) <= totalLimit { + // Update oldest after merging in the memory buffer. oldest = entries[len(entries)-1].Time } @@ -95,6 +95,9 @@ func (l *queryLog) searchFiles(params *searchParams) ([]*logEntry, time.Time, in // The one that was specified in the "oldest" param is not needed, // we need only the one next to it _, err = r.ReadNext() + } else if errors.Is(err, ErrEndOfLog) { + // We've reached the end of the log. + return entries, time.Time{}, 0 } }