mirror of
https://github.com/octoleo/syncthing.git
synced 2024-12-23 11:28:59 +00:00
916ec63af6
This is a new revision of the discovery server. Relevant changes and non-changes: - Protocol towards clients is unchanged. - Recommended large scale design is still to be deployed nehind nginx (I tested, and it's still a lot faster at terminating TLS). - Database backend is leveldb again, only. It scales enough, is easy to setup, and we don't need any backend to take care of. - Server supports replication. This is a simple TCP channel - protect it with a firewall when deploying over the internet. (We deploy this within the same datacenter, and with firewall.) Any incoming client announces are sent over the replication channel(s) to other peer discosrvs. Incoming replication changes are applied to the database as if they came from clients, but without the TLS/certificate overhead. - Metrics are exposed using the prometheus library, when enabled. - The database values and replication protocol is protobuf, because JSON was quite CPU intensive when I tried that and benchmarked it. - The "Retry-After" value for failed lookups gets slowly increased from a default of 120 seconds, by 5 seconds for each failed lookup, independently by each discosrv. This lowers the query load over time for clients that are never seen. The Retry-After maxes out at 3600 after a couple of weeks of this increase. The number of failed lookups is stored in the database, now and then (avoiding making each lookup a database put). All in all this means clients can be pointed towards a cluster using just multiple A / AAAA records to gain both load sharing and redundancy (if one is down, clients will talk to the remaining ones). GitHub-Pull-Request: https://github.com/syncthing/syncthing/pull/4648
677 lines
20 KiB
Go
677 lines
20 KiB
Go
/*
|
|
* Minio Go Library for Amazon S3 Compatible Cloud Storage
|
|
* Copyright 2015-2017 Minio, Inc.
|
|
*
|
|
* 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
|
|
*
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
* See the License for the specific language governing permissions and
|
|
* limitations under the License.
|
|
*/
|
|
|
|
package minio
|
|
|
|
import (
|
|
"context"
|
|
"errors"
|
|
"fmt"
|
|
"io"
|
|
"net/http"
|
|
"strings"
|
|
"sync"
|
|
"time"
|
|
|
|
"github.com/minio/minio-go/pkg/encrypt"
|
|
"github.com/minio/minio-go/pkg/s3utils"
|
|
)
|
|
|
|
// GetEncryptedObject deciphers and streams data stored in the server after applying a specified encryption materials,
|
|
// returned stream should be closed by the caller.
|
|
func (c Client) GetEncryptedObject(bucketName, objectName string, encryptMaterials encrypt.Materials) (io.ReadCloser, error) {
|
|
if encryptMaterials == nil {
|
|
return nil, ErrInvalidArgument("Unable to recognize empty encryption properties")
|
|
}
|
|
|
|
return c.GetObject(bucketName, objectName, GetObjectOptions{Materials: encryptMaterials})
|
|
}
|
|
|
|
// GetObject - returns an seekable, readable object.
|
|
func (c Client) GetObject(bucketName, objectName string, opts GetObjectOptions) (*Object, error) {
|
|
return c.getObjectWithContext(context.Background(), bucketName, objectName, opts)
|
|
}
|
|
|
|
// GetObject wrapper function that accepts a request context
|
|
func (c Client) getObjectWithContext(ctx context.Context, bucketName, objectName string, opts GetObjectOptions) (*Object, error) {
|
|
// Input validation.
|
|
if err := s3utils.CheckValidBucketName(bucketName); err != nil {
|
|
return nil, err
|
|
}
|
|
if err := s3utils.CheckValidObjectName(objectName); err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
var httpReader io.ReadCloser
|
|
var objectInfo ObjectInfo
|
|
var err error
|
|
|
|
// Create request channel.
|
|
reqCh := make(chan getRequest)
|
|
// Create response channel.
|
|
resCh := make(chan getResponse)
|
|
// Create done channel.
|
|
doneCh := make(chan struct{})
|
|
|
|
// This routine feeds partial object data as and when the caller reads.
|
|
go func() {
|
|
defer close(reqCh)
|
|
defer close(resCh)
|
|
|
|
// Used to verify if etag of object has changed since last read.
|
|
var etag string
|
|
|
|
// Loop through the incoming control messages and read data.
|
|
for {
|
|
select {
|
|
// When the done channel is closed exit our routine.
|
|
case <-doneCh:
|
|
// Close the http response body before returning.
|
|
// This ends the connection with the server.
|
|
if httpReader != nil {
|
|
httpReader.Close()
|
|
}
|
|
return
|
|
|
|
// Gather incoming request.
|
|
case req := <-reqCh:
|
|
// If this is the first request we may not need to do a getObject request yet.
|
|
if req.isFirstReq {
|
|
// First request is a Read/ReadAt.
|
|
if req.isReadOp {
|
|
// Differentiate between wanting the whole object and just a range.
|
|
if req.isReadAt {
|
|
// If this is a ReadAt request only get the specified range.
|
|
// Range is set with respect to the offset and length of the buffer requested.
|
|
// Do not set objectInfo from the first readAt request because it will not get
|
|
// the whole object.
|
|
opts.SetRange(req.Offset, req.Offset+int64(len(req.Buffer))-1)
|
|
} else if req.Offset > 0 {
|
|
opts.SetRange(req.Offset, 0)
|
|
}
|
|
httpReader, objectInfo, err = c.getObject(ctx, bucketName, objectName, opts)
|
|
if err != nil {
|
|
resCh <- getResponse{Error: err}
|
|
return
|
|
}
|
|
etag = objectInfo.ETag
|
|
// Read at least firstReq.Buffer bytes, if not we have
|
|
// reached our EOF.
|
|
size, err := io.ReadFull(httpReader, req.Buffer)
|
|
if size > 0 && err == io.ErrUnexpectedEOF {
|
|
// If an EOF happens after reading some but not
|
|
// all the bytes ReadFull returns ErrUnexpectedEOF
|
|
err = io.EOF
|
|
}
|
|
// Send back the first response.
|
|
resCh <- getResponse{
|
|
objectInfo: objectInfo,
|
|
Size: int(size),
|
|
Error: err,
|
|
didRead: true,
|
|
}
|
|
} else {
|
|
// First request is a Stat or Seek call.
|
|
// Only need to run a StatObject until an actual Read or ReadAt request comes through.
|
|
objectInfo, err = c.statObject(ctx, bucketName, objectName, StatObjectOptions{opts})
|
|
if err != nil {
|
|
resCh <- getResponse{
|
|
Error: err,
|
|
}
|
|
// Exit the go-routine.
|
|
return
|
|
}
|
|
etag = objectInfo.ETag
|
|
// Send back the first response.
|
|
resCh <- getResponse{
|
|
objectInfo: objectInfo,
|
|
}
|
|
}
|
|
} else if req.settingObjectInfo { // Request is just to get objectInfo.
|
|
if etag != "" {
|
|
opts.SetMatchETag(etag)
|
|
}
|
|
objectInfo, err := c.statObject(ctx, bucketName, objectName, StatObjectOptions{opts})
|
|
if err != nil {
|
|
resCh <- getResponse{
|
|
Error: err,
|
|
}
|
|
// Exit the goroutine.
|
|
return
|
|
}
|
|
// Send back the objectInfo.
|
|
resCh <- getResponse{
|
|
objectInfo: objectInfo,
|
|
}
|
|
} else {
|
|
// Offset changes fetch the new object at an Offset.
|
|
// Because the httpReader may not be set by the first
|
|
// request if it was a stat or seek it must be checked
|
|
// if the object has been read or not to only initialize
|
|
// new ones when they haven't been already.
|
|
// All readAt requests are new requests.
|
|
if req.DidOffsetChange || !req.beenRead {
|
|
if etag != "" {
|
|
opts.SetMatchETag(etag)
|
|
}
|
|
if httpReader != nil {
|
|
// Close previously opened http reader.
|
|
httpReader.Close()
|
|
}
|
|
// If this request is a readAt only get the specified range.
|
|
if req.isReadAt {
|
|
// Range is set with respect to the offset and length of the buffer requested.
|
|
opts.SetRange(req.Offset, req.Offset+int64(len(req.Buffer))-1)
|
|
} else if req.Offset > 0 { // Range is set with respect to the offset.
|
|
opts.SetRange(req.Offset, 0)
|
|
}
|
|
httpReader, objectInfo, err = c.getObject(ctx, bucketName, objectName, opts)
|
|
if err != nil {
|
|
resCh <- getResponse{
|
|
Error: err,
|
|
}
|
|
return
|
|
}
|
|
}
|
|
|
|
// Read at least req.Buffer bytes, if not we have
|
|
// reached our EOF.
|
|
size, err := io.ReadFull(httpReader, req.Buffer)
|
|
if err == io.ErrUnexpectedEOF {
|
|
// If an EOF happens after reading some but not
|
|
// all the bytes ReadFull returns ErrUnexpectedEOF
|
|
err = io.EOF
|
|
}
|
|
// Reply back how much was read.
|
|
resCh <- getResponse{
|
|
Size: int(size),
|
|
Error: err,
|
|
didRead: true,
|
|
objectInfo: objectInfo,
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}()
|
|
|
|
// Create a newObject through the information sent back by reqCh.
|
|
return newObject(reqCh, resCh, doneCh), nil
|
|
}
|
|
|
|
// get request message container to communicate with internal
|
|
// go-routine.
|
|
type getRequest struct {
|
|
Buffer []byte
|
|
Offset int64 // readAt offset.
|
|
DidOffsetChange bool // Tracks the offset changes for Seek requests.
|
|
beenRead bool // Determines if this is the first time an object is being read.
|
|
isReadAt bool // Determines if this request is a request to a specific range
|
|
isReadOp bool // Determines if this request is a Read or Read/At request.
|
|
isFirstReq bool // Determines if this request is the first time an object is being accessed.
|
|
settingObjectInfo bool // Determines if this request is to set the objectInfo of an object.
|
|
}
|
|
|
|
// get response message container to reply back for the request.
|
|
type getResponse struct {
|
|
Size int
|
|
Error error
|
|
didRead bool // Lets subsequent calls know whether or not httpReader has been initiated.
|
|
objectInfo ObjectInfo // Used for the first request.
|
|
}
|
|
|
|
// Object represents an open object. It implements
|
|
// Reader, ReaderAt, Seeker, Closer for a HTTP stream.
|
|
type Object struct {
|
|
// Mutex.
|
|
mutex *sync.Mutex
|
|
|
|
// User allocated and defined.
|
|
reqCh chan<- getRequest
|
|
resCh <-chan getResponse
|
|
doneCh chan<- struct{}
|
|
currOffset int64
|
|
objectInfo ObjectInfo
|
|
|
|
// Ask lower level to initiate data fetching based on currOffset
|
|
seekData bool
|
|
|
|
// Keeps track of closed call.
|
|
isClosed bool
|
|
|
|
// Keeps track of if this is the first call.
|
|
isStarted bool
|
|
|
|
// Previous error saved for future calls.
|
|
prevErr error
|
|
|
|
// Keeps track of if this object has been read yet.
|
|
beenRead bool
|
|
|
|
// Keeps track of if objectInfo has been set yet.
|
|
objectInfoSet bool
|
|
}
|
|
|
|
// doGetRequest - sends and blocks on the firstReqCh and reqCh of an object.
|
|
// Returns back the size of the buffer read, if anything was read, as well
|
|
// as any error encountered. For all first requests sent on the object
|
|
// it is also responsible for sending back the objectInfo.
|
|
func (o *Object) doGetRequest(request getRequest) (getResponse, error) {
|
|
o.reqCh <- request
|
|
response := <-o.resCh
|
|
|
|
// Return any error to the top level.
|
|
if response.Error != nil {
|
|
return response, response.Error
|
|
}
|
|
|
|
// This was the first request.
|
|
if !o.isStarted {
|
|
// The object has been operated on.
|
|
o.isStarted = true
|
|
}
|
|
// Set the objectInfo if the request was not readAt
|
|
// and it hasn't been set before.
|
|
if !o.objectInfoSet && !request.isReadAt {
|
|
o.objectInfo = response.objectInfo
|
|
o.objectInfoSet = true
|
|
}
|
|
// Set beenRead only if it has not been set before.
|
|
if !o.beenRead {
|
|
o.beenRead = response.didRead
|
|
}
|
|
// Data are ready on the wire, no need to reinitiate connection in lower level
|
|
o.seekData = false
|
|
|
|
return response, nil
|
|
}
|
|
|
|
// setOffset - handles the setting of offsets for
|
|
// Read/ReadAt/Seek requests.
|
|
func (o *Object) setOffset(bytesRead int64) error {
|
|
// Update the currentOffset.
|
|
o.currOffset += bytesRead
|
|
|
|
if o.objectInfo.Size > -1 && o.currOffset >= o.objectInfo.Size {
|
|
return io.EOF
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// Read reads up to len(b) bytes into b. It returns the number of
|
|
// bytes read (0 <= n <= len(b)) and any error encountered. Returns
|
|
// io.EOF upon end of file.
|
|
func (o *Object) Read(b []byte) (n int, err error) {
|
|
if o == nil {
|
|
return 0, ErrInvalidArgument("Object is nil")
|
|
}
|
|
|
|
// Locking.
|
|
o.mutex.Lock()
|
|
defer o.mutex.Unlock()
|
|
|
|
// prevErr is previous error saved from previous operation.
|
|
if o.prevErr != nil || o.isClosed {
|
|
return 0, o.prevErr
|
|
}
|
|
// Create a new request.
|
|
readReq := getRequest{
|
|
isReadOp: true,
|
|
beenRead: o.beenRead,
|
|
Buffer: b,
|
|
}
|
|
|
|
// Alert that this is the first request.
|
|
if !o.isStarted {
|
|
readReq.isFirstReq = true
|
|
}
|
|
|
|
// Ask to establish a new data fetch routine based on seekData flag
|
|
readReq.DidOffsetChange = o.seekData
|
|
readReq.Offset = o.currOffset
|
|
|
|
// Send and receive from the first request.
|
|
response, err := o.doGetRequest(readReq)
|
|
if err != nil && err != io.EOF {
|
|
// Save the error for future calls.
|
|
o.prevErr = err
|
|
return response.Size, err
|
|
}
|
|
|
|
// Bytes read.
|
|
bytesRead := int64(response.Size)
|
|
|
|
// Set the new offset.
|
|
oerr := o.setOffset(bytesRead)
|
|
if oerr != nil {
|
|
// Save the error for future calls.
|
|
o.prevErr = oerr
|
|
return response.Size, oerr
|
|
}
|
|
|
|
// Return the response.
|
|
return response.Size, err
|
|
}
|
|
|
|
// Stat returns the ObjectInfo structure describing Object.
|
|
func (o *Object) Stat() (ObjectInfo, error) {
|
|
if o == nil {
|
|
return ObjectInfo{}, ErrInvalidArgument("Object is nil")
|
|
}
|
|
// Locking.
|
|
o.mutex.Lock()
|
|
defer o.mutex.Unlock()
|
|
|
|
if o.prevErr != nil && o.prevErr != io.EOF || o.isClosed {
|
|
return ObjectInfo{}, o.prevErr
|
|
}
|
|
|
|
// This is the first request.
|
|
if !o.isStarted || !o.objectInfoSet {
|
|
statReq := getRequest{
|
|
isFirstReq: !o.isStarted,
|
|
settingObjectInfo: !o.objectInfoSet,
|
|
}
|
|
|
|
// Send the request and get the response.
|
|
_, err := o.doGetRequest(statReq)
|
|
if err != nil {
|
|
o.prevErr = err
|
|
return ObjectInfo{}, err
|
|
}
|
|
}
|
|
|
|
return o.objectInfo, nil
|
|
}
|
|
|
|
// ReadAt reads len(b) bytes from the File starting at byte offset
|
|
// off. It returns the number of bytes read and the error, if any.
|
|
// ReadAt always returns a non-nil error when n < len(b). At end of
|
|
// file, that error is io.EOF.
|
|
func (o *Object) ReadAt(b []byte, offset int64) (n int, err error) {
|
|
if o == nil {
|
|
return 0, ErrInvalidArgument("Object is nil")
|
|
}
|
|
|
|
// Locking.
|
|
o.mutex.Lock()
|
|
defer o.mutex.Unlock()
|
|
|
|
// prevErr is error which was saved in previous operation.
|
|
if o.prevErr != nil || o.isClosed {
|
|
return 0, o.prevErr
|
|
}
|
|
|
|
// Can only compare offsets to size when size has been set.
|
|
if o.objectInfoSet {
|
|
// If offset is negative than we return io.EOF.
|
|
// If offset is greater than or equal to object size we return io.EOF.
|
|
if (o.objectInfo.Size > -1 && offset >= o.objectInfo.Size) || offset < 0 {
|
|
return 0, io.EOF
|
|
}
|
|
}
|
|
|
|
// Create the new readAt request.
|
|
readAtReq := getRequest{
|
|
isReadOp: true,
|
|
isReadAt: true,
|
|
DidOffsetChange: true, // Offset always changes.
|
|
beenRead: o.beenRead, // Set if this is the first request to try and read.
|
|
Offset: offset, // Set the offset.
|
|
Buffer: b,
|
|
}
|
|
|
|
// Alert that this is the first request.
|
|
if !o.isStarted {
|
|
readAtReq.isFirstReq = true
|
|
}
|
|
|
|
// Send and receive from the first request.
|
|
response, err := o.doGetRequest(readAtReq)
|
|
if err != nil && err != io.EOF {
|
|
// Save the error.
|
|
o.prevErr = err
|
|
return response.Size, err
|
|
}
|
|
// Bytes read.
|
|
bytesRead := int64(response.Size)
|
|
// There is no valid objectInfo yet
|
|
// to compare against for EOF.
|
|
if !o.objectInfoSet {
|
|
// Update the currentOffset.
|
|
o.currOffset += bytesRead
|
|
} else {
|
|
// If this was not the first request update
|
|
// the offsets and compare against objectInfo
|
|
// for EOF.
|
|
oerr := o.setOffset(bytesRead)
|
|
if oerr != nil {
|
|
o.prevErr = oerr
|
|
return response.Size, oerr
|
|
}
|
|
}
|
|
return response.Size, err
|
|
}
|
|
|
|
// Seek sets the offset for the next Read or Write 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.
|
|
//
|
|
// Seeking to a negative offset is an error. Seeking to any positive
|
|
// offset is legal, subsequent io operations succeed until the
|
|
// underlying object is not closed.
|
|
func (o *Object) Seek(offset int64, whence int) (n int64, err error) {
|
|
if o == nil {
|
|
return 0, ErrInvalidArgument("Object is nil")
|
|
}
|
|
|
|
// Locking.
|
|
o.mutex.Lock()
|
|
defer o.mutex.Unlock()
|
|
|
|
if o.prevErr != nil {
|
|
// At EOF seeking is legal allow only io.EOF, for any other errors we return.
|
|
if o.prevErr != io.EOF {
|
|
return 0, o.prevErr
|
|
}
|
|
}
|
|
|
|
// Negative offset is valid for whence of '2'.
|
|
if offset < 0 && whence != 2 {
|
|
return 0, ErrInvalidArgument(fmt.Sprintf("Negative position not allowed for %d.", whence))
|
|
}
|
|
|
|
// This is the first request. So before anything else
|
|
// get the ObjectInfo.
|
|
if !o.isStarted || !o.objectInfoSet {
|
|
// Create the new Seek request.
|
|
seekReq := getRequest{
|
|
isReadOp: false,
|
|
Offset: offset,
|
|
isFirstReq: true,
|
|
}
|
|
// Send and receive from the seek request.
|
|
_, err := o.doGetRequest(seekReq)
|
|
if err != nil {
|
|
// Save the error.
|
|
o.prevErr = err
|
|
return 0, err
|
|
}
|
|
}
|
|
|
|
// Switch through whence.
|
|
switch whence {
|
|
default:
|
|
return 0, ErrInvalidArgument(fmt.Sprintf("Invalid whence %d", whence))
|
|
case 0:
|
|
if o.objectInfo.Size > -1 && offset > o.objectInfo.Size {
|
|
return 0, io.EOF
|
|
}
|
|
o.currOffset = offset
|
|
case 1:
|
|
if o.objectInfo.Size > -1 && o.currOffset+offset > o.objectInfo.Size {
|
|
return 0, io.EOF
|
|
}
|
|
o.currOffset += offset
|
|
case 2:
|
|
// If we don't know the object size return an error for io.SeekEnd
|
|
if o.objectInfo.Size < 0 {
|
|
return 0, ErrInvalidArgument("Whence END is not supported when the object size is unknown")
|
|
}
|
|
// Seeking to positive offset is valid for whence '2', but
|
|
// since we are backing a Reader we have reached 'EOF' if
|
|
// offset is positive.
|
|
if offset > 0 {
|
|
return 0, io.EOF
|
|
}
|
|
// Seeking to negative position not allowed for whence.
|
|
if o.objectInfo.Size+offset < 0 {
|
|
return 0, ErrInvalidArgument(fmt.Sprintf("Seeking at negative offset not allowed for %d", whence))
|
|
}
|
|
o.currOffset = o.objectInfo.Size + offset
|
|
}
|
|
// Reset the saved error since we successfully seeked, let the Read
|
|
// and ReadAt decide.
|
|
if o.prevErr == io.EOF {
|
|
o.prevErr = nil
|
|
}
|
|
|
|
// Ask lower level to fetch again from source
|
|
o.seekData = true
|
|
|
|
// Return the effective offset.
|
|
return o.currOffset, nil
|
|
}
|
|
|
|
// Close - The behavior of Close after the first call returns error
|
|
// for subsequent Close() calls.
|
|
func (o *Object) Close() (err error) {
|
|
if o == nil {
|
|
return ErrInvalidArgument("Object is nil")
|
|
}
|
|
// Locking.
|
|
o.mutex.Lock()
|
|
defer o.mutex.Unlock()
|
|
|
|
// if already closed return an error.
|
|
if o.isClosed {
|
|
return o.prevErr
|
|
}
|
|
|
|
// Close successfully.
|
|
close(o.doneCh)
|
|
|
|
// Save for future operations.
|
|
errMsg := "Object is already closed. Bad file descriptor."
|
|
o.prevErr = errors.New(errMsg)
|
|
// Save here that we closed done channel successfully.
|
|
o.isClosed = true
|
|
return nil
|
|
}
|
|
|
|
// newObject instantiates a new *minio.Object*
|
|
// ObjectInfo will be set by setObjectInfo
|
|
func newObject(reqCh chan<- getRequest, resCh <-chan getResponse, doneCh chan<- struct{}) *Object {
|
|
return &Object{
|
|
mutex: &sync.Mutex{},
|
|
reqCh: reqCh,
|
|
resCh: resCh,
|
|
doneCh: doneCh,
|
|
}
|
|
}
|
|
|
|
// getObject - retrieve object from Object Storage.
|
|
//
|
|
// Additionally this function also takes range arguments to download the specified
|
|
// range bytes of an object. Setting offset and length = 0 will download the full object.
|
|
//
|
|
// For more information about the HTTP Range header.
|
|
// go to http://www.w3.org/Protocols/rfc2616/rfc2616-sec14.html#sec14.35.
|
|
func (c Client) getObject(ctx context.Context, bucketName, objectName string, opts GetObjectOptions) (io.ReadCloser, ObjectInfo, error) {
|
|
// Validate input arguments.
|
|
if err := s3utils.CheckValidBucketName(bucketName); err != nil {
|
|
return nil, ObjectInfo{}, err
|
|
}
|
|
if err := s3utils.CheckValidObjectName(objectName); err != nil {
|
|
return nil, ObjectInfo{}, err
|
|
}
|
|
|
|
// Execute GET on objectName.
|
|
resp, err := c.executeMethod(ctx, "GET", requestMetadata{
|
|
bucketName: bucketName,
|
|
objectName: objectName,
|
|
customHeader: opts.Header(),
|
|
contentSHA256Hex: emptySHA256Hex,
|
|
})
|
|
if err != nil {
|
|
return nil, ObjectInfo{}, err
|
|
}
|
|
if resp != nil {
|
|
if resp.StatusCode != http.StatusOK && resp.StatusCode != http.StatusPartialContent {
|
|
return nil, ObjectInfo{}, httpRespToErrorResponse(resp, bucketName, objectName)
|
|
}
|
|
}
|
|
|
|
// Trim off the odd double quotes from ETag in the beginning and end.
|
|
md5sum := strings.TrimPrefix(resp.Header.Get("ETag"), "\"")
|
|
md5sum = strings.TrimSuffix(md5sum, "\"")
|
|
|
|
// Parse the date.
|
|
date, err := time.Parse(http.TimeFormat, resp.Header.Get("Last-Modified"))
|
|
if err != nil {
|
|
msg := "Last-Modified time format not recognized. " + reportIssue
|
|
return nil, ObjectInfo{}, ErrorResponse{
|
|
Code: "InternalError",
|
|
Message: msg,
|
|
RequestID: resp.Header.Get("x-amz-request-id"),
|
|
HostID: resp.Header.Get("x-amz-id-2"),
|
|
Region: resp.Header.Get("x-amz-bucket-region"),
|
|
}
|
|
}
|
|
|
|
// Get content-type.
|
|
contentType := strings.TrimSpace(resp.Header.Get("Content-Type"))
|
|
if contentType == "" {
|
|
contentType = "application/octet-stream"
|
|
}
|
|
|
|
objectStat := ObjectInfo{
|
|
ETag: md5sum,
|
|
Key: objectName,
|
|
Size: resp.ContentLength,
|
|
LastModified: date,
|
|
ContentType: contentType,
|
|
// Extract only the relevant header keys describing the object.
|
|
// following function filters out a list of standard set of keys
|
|
// which are not part of object metadata.
|
|
Metadata: extractObjMetadata(resp.Header),
|
|
}
|
|
|
|
reader := resp.Body
|
|
if opts.Materials != nil {
|
|
err = opts.Materials.SetupDecryptMode(reader, objectStat.Metadata.Get(amzHeaderIV), objectStat.Metadata.Get(amzHeaderKey))
|
|
if err != nil {
|
|
return nil, ObjectInfo{}, err
|
|
}
|
|
reader = opts.Materials
|
|
}
|
|
|
|
// do not close body here, caller will close
|
|
return reader, objectStat, nil
|
|
}
|