storj/pkg/ranger/readerat.go
Kaloyan Raev f96cc9c955 HTTP Ranger (#11)
* HTTPRanger

* Migrate eestream library to use ReadCloser instead of Reader

* Use MultiReadCloser instead of MultiReader

* Adapt example cli cmds to latest serverbox changes

* Close LazyReadCloser only if only generated

* Close ReadClosers sequentially

* Close response body on unexpected status code

* Avoid double close of the pipe reader

* Better formatting for imports

* Exit instead of panicking
2018-04-17 07:39:14 -06:00

63 lines
1.2 KiB
Go

// Copyright (C) 2018 Storj Labs, Inc.
// See LICENSE for copying information.
package ranger
import (
"io"
"storj.io/storj/internal/pkg/readcloser"
)
type readerAtRanger struct {
r io.ReaderAt
size int64
}
// ReaderAtRanger converts a ReaderAt with a given size to a Ranger
func ReaderAtRanger(r io.ReaderAt, size int64) Ranger {
return &readerAtRanger{
r: r,
size: size,
}
}
func (r *readerAtRanger) Size() int64 {
return r.size
}
type readerAtReader struct {
r io.ReaderAt
offset, length int64
}
func (r *readerAtRanger) Range(offset, length int64) io.ReadCloser {
if offset < 0 {
return readcloser.FatalReadCloser(Error.New("negative offset"))
}
if length < 0 {
return readcloser.FatalReadCloser(Error.New("negative length"))
}
if offset+length > r.size {
return readcloser.FatalReadCloser(Error.New("buffer runoff"))
}
return &readerAtReader{r: r.r, offset: offset, length: length}
}
func (r *readerAtReader) Read(p []byte) (n int, err error) {
if r.length == 0 {
return 0, io.EOF
}
if int64(len(p)) > r.length {
p = p[:r.length]
}
n, err = r.r.ReadAt(p, r.offset)
r.offset += int64(n)
r.length -= int64(n)
return n, err
}
func (r *readerAtReader) Close() error {
return nil
}