Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(15)

Unified Diff: common/logdog/frame/reader.go

Issue 1253353008: logdog: Add frame read/write library. (Closed) Base URL: https://github.com/luci/luci-go@logdog-review-output
Patch Set: Cleanup :( Created 5 years, 4 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View side-by-side diff with in-line comments
Download patch
Index: common/logdog/frame/reader.go
diff --git a/common/logdog/frame/reader.go b/common/logdog/frame/reader.go
new file mode 100644
index 0000000000000000000000000000000000000000..e38562f938f39f86c4d17f033a651518b63270cf
--- /dev/null
+++ b/common/logdog/frame/reader.go
@@ -0,0 +1,104 @@
+// Copyright 2015 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 frame
+
+import (
+ "encoding/binary"
+ "fmt"
+ "io"
+)
+
+// ErrFrameTooLarge is an error that is returned if a frame that is larger than
+// the maximum allowed size (not including the frame header) is read.
+var ErrFrameTooLarge = fmt.Errorf("frame: frame size exceeds maximum")
+
+// Reader reads individual frames from a frame-formatted input Reader.
+type Reader interface {
+ // ReadFrame reads the next frame, returning the frame's size and an io.Reader
+ // for that frame's data. The io.Reader is restricted such that it cannot read
+ // past the frame.
+ //
+ // The frame must be fully read before another Reader call can be made.
+ // Failure to do so will cause the Reader to become unsynchronized.
+ ReadFrame() (int64, io.Reader, error)
+
+ // ReadFrame returns the contents of the next frame. If there are no more
+ // frames available, ReadFrame will return io.EOF.
+ ReadFrameAll() ([]byte, error)
+}
+
+// reader is an implementation of a Reader that uses an underlying
+// io.Reader and io.ByteReader to read frames.
+//
+// The io.Reader and io.ByteReader must read from the same source.
+type reader struct {
+ io.Reader
+ io.ByteReader
+
+ maxSize int64
+}
+
+// NewReader creates a new Reader which reads frame data from the
+// supplied Reader instance.
+//
+// If the Reader instance is also an io.ByteReader, its ReadByte method will
+// be used directly.
+func NewReader(r io.Reader, maxSize int64) Reader {
+ br, ok := r.(io.ByteReader)
+ if !ok {
+ br = &simpleByteReader{Reader: r}
+ }
+ return &reader{
+ Reader: r,
+ ByteReader: br,
+ maxSize: maxSize,
+ }
+}
+
+func (r *reader) ReadFrame() (int64, io.Reader, error) {
+ // Read the frame size.
+ count, err := binary.ReadUvarint(r)
+ if err != nil {
+ return 0, nil, err
+ }
+
+ if count > uint64(r.maxSize) {
+ return 0, nil, ErrFrameTooLarge
+ }
+
+ lr := &io.LimitedReader{
+ R: r.Reader,
+ N: int64(count),
+ }
+ return int64(count), lr, nil
+}
+
+func (r *reader) ReadFrameAll() ([]byte, error) {
+ count, fr, err := r.ReadFrame()
+ if err != nil {
+ return nil, err
+ }
+ if count == 0 {
+ return nil, nil
+ }
+
+ data := make([]byte, count)
+ if _, err := fr.Read(data); err != nil {
+ return nil, err
+ }
+ return data, nil
iannucci 2015/08/04 17:13:50 I think ioutils.ReadAll is what you want
dnj 2015/08/04 17:51:41 No wai! ioutils.ReadAll has no a priori knowledge
+}
+
+// simpleByteReader implements the io.ByteReader interface for an io.Reader.
+type simpleByteReader struct {
+ io.Reader
+
+ buf [1]byte
+}
+
+func (r *simpleByteReader) ReadByte() (byte, error) {
+ _, err := r.Read(r.buf[:])
+ return r.buf[0], err
+}

Powered by Google App Engine
This is Rietveld 408576698