Chromium Code Reviews| Index: server/internal/logdog/archivist/storageSource.go |
| diff --git a/server/internal/logdog/archivist/storageSource.go b/server/internal/logdog/archivist/storageSource.go |
| new file mode 100644 |
| index 0000000000000000000000000000000000000000..490658dd1e35b7204cd702be2a8b5a91e6600aed |
| --- /dev/null |
| +++ b/server/internal/logdog/archivist/storageSource.go |
| @@ -0,0 +1,125 @@ |
| +// Copyright 2016 The Chromium Authors. All rights reserved. |
| +// Use of this source code is governed by a BSD-style license that can be |
| +// found in the LICENSE file. |
| + |
| +package archivist |
| + |
| +import ( |
| + "github.com/golang/protobuf/proto" |
| + "github.com/luci/luci-go/common/logdog/types" |
| + log "github.com/luci/luci-go/common/logging" |
| + "github.com/luci/luci-go/common/proto/logdog/logpb" |
| + "github.com/luci/luci-go/server/logdog/archive" |
| + "github.com/luci/luci-go/server/logdog/storage" |
| + "golang.org/x/net/context" |
| +) |
| + |
| +// storageSource is an archive.LogEntrySource that pulls log entries from |
| +// intermediate storage via its storage.Storage instance. |
| +type storageSource struct { |
|
dnj
2016/04/11 17:20:04
This was pulled out of archivist.go. Nothing reall
|
| + context.Context |
| + |
| + st storage.Storage // the storage instance to read from |
| + path types.StreamPath // the path of the log stream |
| + contiguous bool // if true, enforce contiguous entries |
| + terminalIndex types.MessageIndex // if >= 0, discard logs beyond this |
| + |
| + buf []*logpb.LogEntry |
| + lastIndex types.MessageIndex |
| + logEntryCount int64 |
| + hasMissingEntries bool // true if some log entries were missing. |
| +} |
| + |
| +func (s *storageSource) bufferEntries(start types.MessageIndex) error { |
| + bytes := 0 |
| + |
| + req := storage.GetRequest{ |
| + Path: s.path, |
| + Index: start, |
| + } |
| + return s.st.Get(req, func(idx types.MessageIndex, d []byte) bool { |
| + le := logpb.LogEntry{} |
| + if err := proto.Unmarshal(d, &le); err != nil { |
| + log.Fields{ |
| + log.ErrorKey: err, |
| + "streamIndex": idx, |
| + }.Errorf(s, "Failed to unmarshal LogEntry.") |
| + return false |
| + } |
| + s.buf = append(s.buf, &le) |
| + |
| + // Stop loading if we've reached or exceeded our buffer size. |
| + bytes += len(d) |
| + return bytes < storageBufferSize |
| + }) |
| +} |
| + |
| +func (s *storageSource) NextLogEntry() (*logpb.LogEntry, error) { |
| + if len(s.buf) == 0 { |
| + s.buf = s.buf[:0] |
| + if err := s.bufferEntries(s.lastIndex + 1); err != nil { |
| + if err == storage.ErrDoesNotExist { |
| + log.Warningf(s, "Archive target stream does not exist in intermediate storage.") |
| + if s.terminalIndex >= 0 { |
| + s.hasMissingEntries = true |
| + } |
| + return nil, archive.ErrEndOfStream |
| + } |
| + |
| + log.WithError(err).Errorf(s, "Failed to retrieve log stream from storage.") |
| + return nil, err |
| + } |
| + } |
| + |
| + // If we have no more buffered entries, we have exhausted our log stream. |
| + if len(s.buf) == 0 { |
| + // If we have a terminal index, but we didn't actually emit that index, |
| + // mark that we have missing entries. |
| + if s.terminalIndex >= 0 && s.lastIndex != s.terminalIndex { |
| + log.Fields{ |
| + "terminalIndex": s.terminalIndex, |
| + "lastIndex": s.lastIndex, |
| + }.Warningf(s, "Log stream stopped before terminal index.") |
| + s.hasMissingEntries = true |
| + } else { |
| + log.Fields{ |
| + "lastIndex": s.lastIndex, |
| + }.Debugf(s, "Encountered end of stream.") |
| + } |
| + |
| + return nil, archive.ErrEndOfStream |
| + } |
| + |
| + // Pop the next log entry and advance the stream. |
| + var le *logpb.LogEntry |
| + le, s.buf = s.buf[0], s.buf[1:] |
| + |
| + // If we're enforcing a contiguous log stream, error if this LogEntry is not |
| + // contiguous. |
| + sidx := types.MessageIndex(le.StreamIndex) |
| + nidx := (s.lastIndex + 1) |
| + if sidx != nidx { |
| + s.hasMissingEntries = true |
| + } |
| + if s.contiguous && s.hasMissingEntries { |
| + log.Fields{ |
| + "index": sidx, |
| + "nextIndex": nidx, |
| + }.Warningf(s, "Non-contiguous log stream while enforcing.") |
| + return nil, archive.ErrEndOfStream |
| + } |
| + |
| + // If we're enforcing a maximum terminal index, return end of stream if this |
| + // LogEntry exceeds that index. |
| + if s.terminalIndex >= 0 && sidx > s.terminalIndex { |
| + log.Fields{ |
| + "index": sidx, |
| + "terminalIndex": s.terminalIndex, |
| + }.Warningf(s, "Discarding log entries beyond expected terminal index.") |
| + return nil, archive.ErrEndOfStream |
| + } |
| + |
| + s.lastIndex = sidx |
| + s.logEntryCount++ |
| + return le, nil |
| +} |