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
 | |
| }
 |