154 lines
4.3 KiB
Go
154 lines
4.3 KiB
Go
// Copyright (C) MongoDB, Inc. 2023-present.
|
|
//
|
|
// Licensed under the Apache License, Version 2.0 (the "License"); you may
|
|
// not use this file except in compliance with the License. You may obtain
|
|
// a copy of the License at http://www.apache.org/licenses/LICENSE-2.0
|
|
//
|
|
// Based on github.com/aws/aws-sdk-go by Amazon.com, Inc. with code from:
|
|
// - github.com/aws/aws-sdk-go/blob/v1.44.225/aws/types.go
|
|
// See THIRD-PARTY-NOTICES for original license terms
|
|
|
|
package aws
|
|
|
|
import (
|
|
"io"
|
|
)
|
|
|
|
// ReadSeekCloser wraps a io.Reader returning a ReaderSeekerCloser. Allows the
|
|
// SDK to accept an io.Reader that is not also an io.Seeker for unsigned
|
|
// streaming payload API operations.
|
|
//
|
|
// A ReadSeekCloser wrapping an nonseekable io.Reader used in an API
|
|
// operation's input will prevent that operation being retried in the case of
|
|
// network errors, and cause operation requests to fail if the operation
|
|
// requires payload signing.
|
|
//
|
|
// Note: If using With S3 PutObject to stream an object upload The SDK's S3
|
|
// Upload manager (s3manager.Uploader) provides support for streaming with the
|
|
// ability to retry network errors.
|
|
func ReadSeekCloser(r io.Reader) ReaderSeekerCloser {
|
|
return ReaderSeekerCloser{r}
|
|
}
|
|
|
|
// ReaderSeekerCloser represents a reader that can also delegate io.Seeker and
|
|
// io.Closer interfaces to the underlying object if they are available.
|
|
type ReaderSeekerCloser struct {
|
|
r io.Reader
|
|
}
|
|
|
|
// IsReaderSeekable returns if the underlying reader type can be seeked. A
|
|
// io.Reader might not actually be seekable if it is the ReaderSeekerCloser
|
|
// type.
|
|
func IsReaderSeekable(r io.Reader) bool {
|
|
switch v := r.(type) {
|
|
case ReaderSeekerCloser:
|
|
return v.IsSeeker()
|
|
case *ReaderSeekerCloser:
|
|
return v.IsSeeker()
|
|
case io.ReadSeeker:
|
|
return true
|
|
default:
|
|
return false
|
|
}
|
|
}
|
|
|
|
// Read reads from the reader up to size of p. The number of bytes read, and
|
|
// error if it occurred will be returned.
|
|
//
|
|
// If the reader is not an io.Reader zero bytes read, and nil error will be
|
|
// returned.
|
|
//
|
|
// Performs the same functionality as io.Reader Read
|
|
func (r ReaderSeekerCloser) Read(p []byte) (int, error) {
|
|
switch t := r.r.(type) {
|
|
case io.Reader:
|
|
return t.Read(p)
|
|
}
|
|
return 0, nil
|
|
}
|
|
|
|
// Seek sets the offset for the next Read to offset, interpreted according to
|
|
// whence: 0 means relative to the origin of the file, 1 means relative to the
|
|
// current offset, and 2 means relative to the end. Seek returns the new offset
|
|
// and an error, if any.
|
|
//
|
|
// If the ReaderSeekerCloser is not an io.Seeker nothing will be done.
|
|
func (r ReaderSeekerCloser) Seek(offset int64, whence int) (int64, error) {
|
|
switch t := r.r.(type) {
|
|
case io.Seeker:
|
|
return t.Seek(offset, whence)
|
|
}
|
|
return int64(0), nil
|
|
}
|
|
|
|
// IsSeeker returns if the underlying reader is also a seeker.
|
|
func (r ReaderSeekerCloser) IsSeeker() bool {
|
|
_, ok := r.r.(io.Seeker)
|
|
return ok
|
|
}
|
|
|
|
// HasLen returns the length of the underlying reader if the value implements
|
|
// the Len() int method.
|
|
func (r ReaderSeekerCloser) HasLen() (int, bool) {
|
|
type lenner interface {
|
|
Len() int
|
|
}
|
|
|
|
if lr, ok := r.r.(lenner); ok {
|
|
return lr.Len(), true
|
|
}
|
|
|
|
return 0, false
|
|
}
|
|
|
|
// GetLen returns the length of the bytes remaining in the underlying reader.
|
|
// Checks first for Len(), then io.Seeker to determine the size of the
|
|
// underlying reader.
|
|
//
|
|
// Will return -1 if the length cannot be determined.
|
|
func (r ReaderSeekerCloser) GetLen() (int64, error) {
|
|
if l, ok := r.HasLen(); ok {
|
|
return int64(l), nil
|
|
}
|
|
|
|
if s, ok := r.r.(io.Seeker); ok {
|
|
return seekerLen(s)
|
|
}
|
|
|
|
return -1, nil
|
|
}
|
|
|
|
// SeekerLen attempts to get the number of bytes remaining at the seeker's
|
|
// current position. Returns the number of bytes remaining or error.
|
|
func SeekerLen(s io.Seeker) (int64, error) {
|
|
// Determine if the seeker is actually seekable. ReaderSeekerCloser
|
|
// hides the fact that a io.Readers might not actually be seekable.
|
|
switch v := s.(type) {
|
|
case ReaderSeekerCloser:
|
|
return v.GetLen()
|
|
case *ReaderSeekerCloser:
|
|
return v.GetLen()
|
|
}
|
|
|
|
return seekerLen(s)
|
|
}
|
|
|
|
func seekerLen(s io.Seeker) (int64, error) {
|
|
curOffset, err := s.Seek(0, io.SeekCurrent)
|
|
if err != nil {
|
|
return 0, err
|
|
}
|
|
|
|
endOffset, err := s.Seek(0, io.SeekEnd)
|
|
if err != nil {
|
|
return 0, err
|
|
}
|
|
|
|
_, err = s.Seek(curOffset, io.SeekStart)
|
|
if err != nil {
|
|
return 0, err
|
|
}
|
|
|
|
return endOffset - curOffset, nil
|
|
}
|