2021-06-10 16:17:03 +01:00
|
|
|
|
// Copyright (c) 2015-2021 MinIO, Inc.
|
|
|
|
|
//
|
|
|
|
|
// This file is part of MinIO Object Storage stack
|
|
|
|
|
//
|
|
|
|
|
// This program is free software: you can redistribute it and/or modify
|
|
|
|
|
// it under the terms of the GNU Affero General Public License as published by
|
|
|
|
|
// the Free Software Foundation, either version 3 of the License, or
|
|
|
|
|
// (at your option) any later version.
|
|
|
|
|
//
|
|
|
|
|
// This program is distributed in the hope that it will be useful
|
|
|
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
|
// GNU Affero General Public License for more details.
|
|
|
|
|
//
|
|
|
|
|
// You should have received a copy of the GNU Affero General Public License
|
|
|
|
|
// along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
|
|
|
|
|
|
package cmd
|
|
|
|
|
|
|
|
|
|
import (
|
|
|
|
|
"bytes"
|
|
|
|
|
"context"
|
2021-11-25 09:11:25 -08:00
|
|
|
|
"encoding/base64"
|
2021-06-10 16:17:03 +01:00
|
|
|
|
"errors"
|
|
|
|
|
"fmt"
|
|
|
|
|
"io"
|
2021-11-02 08:11:50 -07:00
|
|
|
|
"io/ioutil"
|
2021-06-10 16:17:03 +01:00
|
|
|
|
"net/http"
|
|
|
|
|
"sort"
|
|
|
|
|
"strings"
|
|
|
|
|
|
|
|
|
|
"github.com/minio/minio/internal/crypto"
|
2021-11-02 08:11:50 -07:00
|
|
|
|
xioutil "github.com/minio/minio/internal/ioutil"
|
2021-06-10 16:17:03 +01:00
|
|
|
|
"github.com/minio/minio/internal/logger"
|
|
|
|
|
"github.com/minio/pkg/bucket/policy"
|
2021-06-14 14:54:37 -07:00
|
|
|
|
xnet "github.com/minio/pkg/net"
|
2021-06-10 16:17:03 +01:00
|
|
|
|
"github.com/minio/zipindex"
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
const (
|
|
|
|
|
archiveType = "zip"
|
|
|
|
|
archiveExt = "." + archiveType // ".zip"
|
|
|
|
|
archiveSeparator = "/"
|
|
|
|
|
archivePattern = archiveExt + archiveSeparator // ".zip/"
|
|
|
|
|
archiveTypeMetadataKey = ReservedMetadataPrefixLower + "archive-type" // "x-minio-internal-archive-type"
|
|
|
|
|
archiveInfoMetadataKey = ReservedMetadataPrefixLower + "archive-info" // "x-minio-internal-archive-info"
|
|
|
|
|
|
|
|
|
|
// Peek into a zip archive
|
|
|
|
|
xMinIOExtract = "x-minio-extract"
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
// splitZipExtensionPath splits the S3 path to the zip file and the path inside the zip:
|
|
|
|
|
// e.g /path/to/archive.zip/backup-2021/myimage.png => /path/to/archive.zip, backup/myimage.png
|
|
|
|
|
func splitZipExtensionPath(input string) (zipPath, object string, err error) {
|
|
|
|
|
idx := strings.Index(input, archivePattern)
|
|
|
|
|
if idx < 0 {
|
|
|
|
|
// Should never happen
|
|
|
|
|
return "", "", errors.New("unable to parse zip path")
|
|
|
|
|
}
|
|
|
|
|
return input[:idx+len(archivePattern)-1], input[idx+len(archivePattern):], nil
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// getObjectInArchiveFileHandler - GET Object in the archive file
|
|
|
|
|
func (api objectAPIHandlers) getObjectInArchiveFileHandler(ctx context.Context, objectAPI ObjectLayer, bucket, object string, w http.ResponseWriter, r *http.Request) {
|
|
|
|
|
if crypto.S3.IsRequested(r.Header) || crypto.S3KMS.IsRequested(r.Header) { // If SSE-S3 or SSE-KMS present -> AWS fails with undefined error
|
2021-06-17 20:27:04 -07:00
|
|
|
|
writeErrorResponse(ctx, w, errorCodes.ToAPIErr(ErrBadRequest), r.URL)
|
2021-06-10 16:17:03 +01:00
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
if _, ok := crypto.IsRequested(r.Header); !objectAPI.IsEncryptionSupported() && ok {
|
2021-06-17 20:27:04 -07:00
|
|
|
|
writeErrorResponse(ctx, w, errorCodes.ToAPIErr(ErrBadRequest), r.URL)
|
2021-06-10 16:17:03 +01:00
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
zipPath, object, err := splitZipExtensionPath(object)
|
|
|
|
|
if err != nil {
|
2021-06-17 20:27:04 -07:00
|
|
|
|
writeErrorResponse(ctx, w, toAPIError(ctx, err), r.URL)
|
2021-06-10 16:17:03 +01:00
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// get gateway encryption options
|
|
|
|
|
opts, err := getOpts(ctx, r, bucket, zipPath)
|
|
|
|
|
if err != nil {
|
2021-06-17 20:27:04 -07:00
|
|
|
|
writeErrorResponse(ctx, w, toAPIError(ctx, err), r.URL)
|
2021-06-10 16:17:03 +01:00
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
getObjectInfo := objectAPI.GetObjectInfo
|
|
|
|
|
if api.CacheAPI() != nil {
|
|
|
|
|
getObjectInfo = api.CacheAPI().GetObjectInfo
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Check for auth type to return S3 compatible error.
|
|
|
|
|
// type to return the correct error (NoSuchKey vs AccessDenied)
|
|
|
|
|
if s3Error := checkRequestAuthType(ctx, r, policy.GetObjectAction, bucket, zipPath); s3Error != ErrNone {
|
|
|
|
|
if getRequestAuthType(r) == authTypeAnonymous {
|
|
|
|
|
// As per "Permission" section in
|
|
|
|
|
// https://docs.aws.amazon.com/AmazonS3/latest/API/RESTObjectGET.html
|
|
|
|
|
// If the object you request does not exist,
|
|
|
|
|
// the error Amazon S3 returns depends on
|
|
|
|
|
// whether you also have the s3:ListBucket
|
|
|
|
|
// permission.
|
|
|
|
|
// * If you have the s3:ListBucket permission
|
|
|
|
|
// on the bucket, Amazon S3 will return an
|
|
|
|
|
// HTTP status code 404 ("no such key")
|
|
|
|
|
// error.
|
|
|
|
|
// * if you don’t have the s3:ListBucket
|
|
|
|
|
// permission, Amazon S3 will return an HTTP
|
|
|
|
|
// status code 403 ("access denied") error.`
|
|
|
|
|
if globalPolicySys.IsAllowed(policy.Args{
|
|
|
|
|
Action: policy.ListBucketAction,
|
|
|
|
|
BucketName: bucket,
|
|
|
|
|
ConditionValues: getConditionValues(r, "", "", nil),
|
|
|
|
|
IsOwner: false,
|
|
|
|
|
}) {
|
|
|
|
|
_, err = getObjectInfo(ctx, bucket, zipPath, opts)
|
|
|
|
|
if toAPIError(ctx, err).Code == "NoSuchKey" {
|
|
|
|
|
s3Error = ErrNoSuchKey
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
2021-06-17 20:27:04 -07:00
|
|
|
|
writeErrorResponse(ctx, w, errorCodes.ToAPIErr(s3Error), r.URL)
|
2021-06-10 16:17:03 +01:00
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Validate pre-conditions if any.
|
|
|
|
|
opts.CheckPrecondFn = func(oi ObjectInfo) bool {
|
|
|
|
|
if objectAPI.IsEncryptionSupported() {
|
|
|
|
|
if _, err := DecryptObjectInfo(&oi, r); err != nil {
|
2021-06-17 20:27:04 -07:00
|
|
|
|
writeErrorResponse(ctx, w, toAPIError(ctx, err), r.URL)
|
2021-06-10 16:17:03 +01:00
|
|
|
|
return true
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return checkPreconditions(ctx, w, r, oi, opts)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
zipObjInfo, err := getObjectInfo(ctx, bucket, zipPath, opts)
|
|
|
|
|
if err != nil {
|
2021-06-17 20:27:04 -07:00
|
|
|
|
writeErrorResponse(ctx, w, toAPIError(ctx, err), r.URL)
|
2021-06-10 16:17:03 +01:00
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
var zipInfo []byte
|
|
|
|
|
|
|
|
|
|
if z, ok := zipObjInfo.UserDefined[archiveInfoMetadataKey]; ok {
|
2021-11-25 09:11:25 -08:00
|
|
|
|
if globalIsErasure {
|
|
|
|
|
zipInfo = []byte(z)
|
|
|
|
|
} else {
|
|
|
|
|
zipInfo, err = base64.StdEncoding.DecodeString(z)
|
|
|
|
|
logger.LogIf(ctx, err)
|
|
|
|
|
// Will attempt to re-read...
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if len(zipInfo) == 0 {
|
2021-06-10 16:17:03 +01:00
|
|
|
|
zipInfo, err = updateObjectMetadataWithZipInfo(ctx, objectAPI, bucket, zipPath, opts)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if err != nil {
|
2021-06-17 20:27:04 -07:00
|
|
|
|
writeErrorResponse(ctx, w, toAPIError(ctx, err), r.URL)
|
2021-06-10 16:17:03 +01:00
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
file, err := zipindex.FindSerialized(zipInfo, object)
|
|
|
|
|
if err != nil {
|
|
|
|
|
if err == io.EOF {
|
2021-06-17 20:27:04 -07:00
|
|
|
|
writeErrorResponse(ctx, w, errorCodes.ToAPIErr(ErrNoSuchKey), r.URL)
|
2021-06-10 16:17:03 +01:00
|
|
|
|
} else {
|
2021-06-17 20:27:04 -07:00
|
|
|
|
writeErrorResponse(ctx, w, toAPIError(ctx, err), r.URL)
|
2021-06-10 16:17:03 +01:00
|
|
|
|
}
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// New object info
|
|
|
|
|
fileObjInfo := ObjectInfo{
|
|
|
|
|
Bucket: bucket,
|
|
|
|
|
Name: object,
|
|
|
|
|
Size: int64(file.UncompressedSize64),
|
|
|
|
|
ModTime: zipObjInfo.ModTime,
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
var rc io.ReadCloser
|
|
|
|
|
|
|
|
|
|
if file.UncompressedSize64 > 0 {
|
2021-10-14 12:37:30 -07:00
|
|
|
|
// We do not know where the file ends, but the returned reader only returns UncompressedSize.
|
|
|
|
|
rs := &HTTPRangeSpec{Start: file.Offset, End: -1}
|
2021-06-10 16:17:03 +01:00
|
|
|
|
gr, err := objectAPI.GetObjectNInfo(ctx, bucket, zipPath, rs, nil, readLock, opts)
|
|
|
|
|
if err != nil {
|
2021-06-17 20:27:04 -07:00
|
|
|
|
writeErrorResponse(ctx, w, toAPIError(ctx, err), r.URL)
|
2021-06-10 16:17:03 +01:00
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
defer gr.Close()
|
|
|
|
|
rc, err = file.Open(gr)
|
|
|
|
|
if err != nil {
|
2021-06-17 20:27:04 -07:00
|
|
|
|
writeErrorResponse(ctx, w, toAPIError(ctx, err), r.URL)
|
2021-06-10 16:17:03 +01:00
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
} else {
|
2021-11-02 08:11:50 -07:00
|
|
|
|
rc = ioutil.NopCloser(bytes.NewReader([]byte{}))
|
2021-06-10 16:17:03 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
defer rc.Close()
|
|
|
|
|
|
|
|
|
|
if err = setObjectHeaders(w, fileObjInfo, nil, opts); err != nil {
|
2021-06-17 20:27:04 -07:00
|
|
|
|
writeErrorResponse(ctx, w, toAPIError(ctx, err), r.URL)
|
2021-06-10 16:17:03 +01:00
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2021-08-07 22:43:01 -07:00
|
|
|
|
setHeadGetRespHeaders(w, r.Form)
|
2021-06-10 16:17:03 +01:00
|
|
|
|
|
2021-11-02 08:11:50 -07:00
|
|
|
|
httpWriter := xioutil.WriteOnClose(w)
|
2021-06-10 16:17:03 +01:00
|
|
|
|
|
|
|
|
|
// Write object content to response body
|
2021-11-02 08:11:50 -07:00
|
|
|
|
if _, err = xioutil.Copy(httpWriter, rc); err != nil {
|
2021-06-10 16:17:03 +01:00
|
|
|
|
if !httpWriter.HasWritten() {
|
|
|
|
|
// write error response only if no data or headers has been written to client yet
|
2021-06-17 20:27:04 -07:00
|
|
|
|
writeErrorResponse(ctx, w, toAPIError(ctx, err), r.URL)
|
2021-06-10 16:17:03 +01:00
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
if !xnet.IsNetworkOrHostDown(err, true) { // do not need to log disconnected clients
|
|
|
|
|
logger.LogIf(ctx, fmt.Errorf("Unable to write all the data to client %w", err))
|
|
|
|
|
}
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if err = httpWriter.Close(); err != nil {
|
|
|
|
|
if !httpWriter.HasWritten() { // write error response only if no data or headers has been written to client yet
|
2021-06-17 20:27:04 -07:00
|
|
|
|
writeErrorResponse(ctx, w, toAPIError(ctx, err), r.URL)
|
2021-06-10 16:17:03 +01:00
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
if !xnet.IsNetworkOrHostDown(err, true) { // do not need to log disconnected clients
|
|
|
|
|
logger.LogIf(ctx, fmt.Errorf("Unable to write all the data to client %w", err))
|
|
|
|
|
}
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// listObjectsV2InArchive generates S3 listing result ListObjectsV2Info from zip file, all parameters are already validated by the caller.
|
|
|
|
|
func listObjectsV2InArchive(ctx context.Context, objectAPI ObjectLayer, bucket, prefix, token, delimiter string, maxKeys int, fetchOwner bool, startAfter string) (ListObjectsV2Info, error) {
|
|
|
|
|
zipPath, _, err := splitZipExtensionPath(prefix)
|
|
|
|
|
if err != nil {
|
|
|
|
|
// Return empty listing
|
|
|
|
|
return ListObjectsV2Info{}, nil
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
zipObjInfo, err := objectAPI.GetObjectInfo(ctx, bucket, zipPath, ObjectOptions{})
|
|
|
|
|
if err != nil {
|
|
|
|
|
// Return empty listing
|
|
|
|
|
return ListObjectsV2Info{}, nil
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
var zipInfo []byte
|
|
|
|
|
|
|
|
|
|
if z, ok := zipObjInfo.UserDefined[archiveInfoMetadataKey]; ok {
|
|
|
|
|
zipInfo = []byte(z)
|
|
|
|
|
} else {
|
|
|
|
|
// Always update the latest version
|
|
|
|
|
zipInfo, err = updateObjectMetadataWithZipInfo(ctx, objectAPI, bucket, zipPath, ObjectOptions{})
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
|
return ListObjectsV2Info{}, err
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
files, err := zipindex.DeserializeFiles(zipInfo)
|
|
|
|
|
if err != nil {
|
|
|
|
|
return ListObjectsV2Info{}, err
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
sort.Slice(files, func(i, j int) bool {
|
|
|
|
|
return files[i].Name < files[j].Name
|
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
var (
|
|
|
|
|
count int
|
|
|
|
|
isTruncated bool
|
|
|
|
|
nextToken string
|
|
|
|
|
listObjectsInfo ListObjectsV2Info
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
// Always set this
|
|
|
|
|
listObjectsInfo.ContinuationToken = token
|
|
|
|
|
|
|
|
|
|
// Open and iterate through the files in the archive.
|
|
|
|
|
for _, file := range files {
|
|
|
|
|
objName := zipObjInfo.Name + archiveSeparator + file.Name
|
|
|
|
|
if objName <= startAfter || objName <= token {
|
|
|
|
|
continue
|
|
|
|
|
}
|
|
|
|
|
if strings.HasPrefix(objName, prefix) {
|
|
|
|
|
if count == maxKeys {
|
|
|
|
|
isTruncated = true
|
|
|
|
|
break
|
|
|
|
|
}
|
|
|
|
|
if delimiter != "" {
|
|
|
|
|
i := strings.Index(objName[len(prefix):], delimiter)
|
|
|
|
|
if i >= 0 {
|
|
|
|
|
commonPrefix := objName[:len(prefix)+i+1]
|
|
|
|
|
if len(listObjectsInfo.Prefixes) == 0 || commonPrefix != listObjectsInfo.Prefixes[len(listObjectsInfo.Prefixes)-1] {
|
|
|
|
|
listObjectsInfo.Prefixes = append(listObjectsInfo.Prefixes, commonPrefix)
|
|
|
|
|
count++
|
|
|
|
|
}
|
|
|
|
|
goto next
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
listObjectsInfo.Objects = append(listObjectsInfo.Objects, ObjectInfo{
|
|
|
|
|
Bucket: bucket,
|
|
|
|
|
Name: objName,
|
|
|
|
|
Size: int64(file.UncompressedSize64),
|
|
|
|
|
ModTime: zipObjInfo.ModTime,
|
|
|
|
|
})
|
|
|
|
|
count++
|
|
|
|
|
}
|
|
|
|
|
next:
|
|
|
|
|
nextToken = objName
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if isTruncated {
|
|
|
|
|
listObjectsInfo.IsTruncated = true
|
|
|
|
|
listObjectsInfo.NextContinuationToken = nextToken
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return listObjectsInfo, nil
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// getFilesFromZIPObject reads a partial stream of a zip file to build the zipindex.Files index
|
|
|
|
|
func getFilesListFromZIPObject(ctx context.Context, objectAPI ObjectLayer, bucket, object string, opts ObjectOptions) (zipindex.Files, ObjectInfo, error) {
|
2022-01-02 09:15:06 -08:00
|
|
|
|
size := 1 << 20
|
2021-06-10 16:17:03 +01:00
|
|
|
|
var objSize int64
|
|
|
|
|
for {
|
|
|
|
|
rs := &HTTPRangeSpec{IsSuffixLength: true, Start: int64(-size)}
|
|
|
|
|
gr, err := objectAPI.GetObjectNInfo(ctx, bucket, object, rs, nil, readLock, opts)
|
|
|
|
|
if err != nil {
|
|
|
|
|
return nil, ObjectInfo{}, err
|
|
|
|
|
}
|
2021-11-02 08:11:50 -07:00
|
|
|
|
b, err := ioutil.ReadAll(gr)
|
2021-06-10 16:17:03 +01:00
|
|
|
|
if err != nil {
|
|
|
|
|
gr.Close()
|
|
|
|
|
return nil, ObjectInfo{}, err
|
|
|
|
|
}
|
|
|
|
|
gr.Close()
|
|
|
|
|
if size > len(b) {
|
|
|
|
|
size = len(b)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Calculate the object real size if encrypted
|
|
|
|
|
if _, ok := crypto.IsEncrypted(gr.ObjInfo.UserDefined); ok {
|
|
|
|
|
objSize, err = gr.ObjInfo.DecryptedSize()
|
|
|
|
|
if err != nil {
|
|
|
|
|
return nil, ObjectInfo{}, err
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
objSize = gr.ObjInfo.Size
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
files, err := zipindex.ReadDir(b[len(b)-size:], objSize, nil)
|
|
|
|
|
if err == nil {
|
|
|
|
|
return files, gr.ObjInfo, nil
|
|
|
|
|
}
|
|
|
|
|
var terr zipindex.ErrNeedMoreData
|
|
|
|
|
if errors.As(err, &terr) {
|
|
|
|
|
size = int(terr.FromEnd)
|
|
|
|
|
if size <= 0 || size > 100<<20 {
|
|
|
|
|
return nil, ObjectInfo{}, errors.New("zip directory too large")
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
return nil, ObjectInfo{}, err
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// headObjectInArchiveFileHandler - HEAD Object in an archive file
|
|
|
|
|
func (api objectAPIHandlers) headObjectInArchiveFileHandler(ctx context.Context, objectAPI ObjectLayer, bucket, object string, w http.ResponseWriter, r *http.Request) {
|
|
|
|
|
if crypto.S3.IsRequested(r.Header) || crypto.S3KMS.IsRequested(r.Header) { // If SSE-S3 or SSE-KMS present -> AWS fails with undefined error
|
|
|
|
|
writeErrorResponseHeadersOnly(w, errorCodes.ToAPIErr(ErrBadRequest))
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
if _, ok := crypto.IsRequested(r.Header); !objectAPI.IsEncryptionSupported() && ok {
|
2021-06-17 20:27:04 -07:00
|
|
|
|
writeErrorResponse(ctx, w, errorCodes.ToAPIErr(ErrBadRequest), r.URL)
|
2021-06-10 16:17:03 +01:00
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
zipPath, object, err := splitZipExtensionPath(object)
|
|
|
|
|
if err != nil {
|
|
|
|
|
writeErrorResponseHeadersOnly(w, toAPIError(ctx, err))
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
getObjectInfo := objectAPI.GetObjectInfo
|
|
|
|
|
if api.CacheAPI() != nil {
|
|
|
|
|
getObjectInfo = api.CacheAPI().GetObjectInfo
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
opts, err := getOpts(ctx, r, bucket, zipPath)
|
|
|
|
|
if err != nil {
|
|
|
|
|
writeErrorResponseHeadersOnly(w, toAPIError(ctx, err))
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if s3Error := checkRequestAuthType(ctx, r, policy.GetObjectAction, bucket, zipPath); s3Error != ErrNone {
|
|
|
|
|
if getRequestAuthType(r) == authTypeAnonymous {
|
|
|
|
|
// As per "Permission" section in
|
|
|
|
|
// https://docs.aws.amazon.com/AmazonS3/latest/API/RESTObjectHEAD.html
|
|
|
|
|
// If the object you request does not exist,
|
|
|
|
|
// the error Amazon S3 returns depends on
|
|
|
|
|
// whether you also have the s3:ListBucket
|
|
|
|
|
// permission.
|
|
|
|
|
// * If you have the s3:ListBucket permission
|
|
|
|
|
// on the bucket, Amazon S3 will return an
|
|
|
|
|
// HTTP status code 404 ("no such key")
|
|
|
|
|
// error.
|
|
|
|
|
// * if you don’t have the s3:ListBucket
|
|
|
|
|
// permission, Amazon S3 will return an HTTP
|
|
|
|
|
// status code 403 ("access denied") error.`
|
|
|
|
|
if globalPolicySys.IsAllowed(policy.Args{
|
|
|
|
|
Action: policy.ListBucketAction,
|
|
|
|
|
BucketName: bucket,
|
|
|
|
|
ConditionValues: getConditionValues(r, "", "", nil),
|
|
|
|
|
IsOwner: false,
|
|
|
|
|
}) {
|
|
|
|
|
_, err = getObjectInfo(ctx, bucket, zipPath, opts)
|
|
|
|
|
if toAPIError(ctx, err).Code == "NoSuchKey" {
|
|
|
|
|
s3Error = ErrNoSuchKey
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
writeErrorResponseHeadersOnly(w, errorCodes.ToAPIErr(s3Error))
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
var rs *HTTPRangeSpec
|
|
|
|
|
|
|
|
|
|
// Validate pre-conditions if any.
|
|
|
|
|
opts.CheckPrecondFn = func(oi ObjectInfo) bool {
|
|
|
|
|
return checkPreconditions(ctx, w, r, oi, opts)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
zipObjInfo, err := getObjectInfo(ctx, bucket, zipPath, opts)
|
|
|
|
|
if err != nil {
|
2021-06-17 20:27:04 -07:00
|
|
|
|
writeErrorResponse(ctx, w, toAPIError(ctx, err), r.URL)
|
2021-06-10 16:17:03 +01:00
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
var zipInfo []byte
|
|
|
|
|
|
|
|
|
|
if z, ok := zipObjInfo.UserDefined[archiveInfoMetadataKey]; ok {
|
|
|
|
|
zipInfo = []byte(z)
|
|
|
|
|
} else {
|
|
|
|
|
zipInfo, err = updateObjectMetadataWithZipInfo(ctx, objectAPI, bucket, zipPath, opts)
|
|
|
|
|
}
|
|
|
|
|
if err != nil {
|
2021-06-17 20:27:04 -07:00
|
|
|
|
writeErrorResponse(ctx, w, toAPIError(ctx, err), r.URL)
|
2021-06-10 16:17:03 +01:00
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
file, err := zipindex.FindSerialized(zipInfo, object)
|
|
|
|
|
if err != nil {
|
|
|
|
|
if err == io.EOF {
|
2021-06-17 20:27:04 -07:00
|
|
|
|
writeErrorResponse(ctx, w, errorCodes.ToAPIErr(ErrNoSuchKey), r.URL)
|
2021-06-10 16:17:03 +01:00
|
|
|
|
} else {
|
2021-06-17 20:27:04 -07:00
|
|
|
|
writeErrorResponse(ctx, w, toAPIError(ctx, err), r.URL)
|
2021-06-10 16:17:03 +01:00
|
|
|
|
}
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
objInfo := ObjectInfo{
|
|
|
|
|
Bucket: bucket,
|
|
|
|
|
Name: file.Name,
|
|
|
|
|
Size: int64(file.UncompressedSize64),
|
|
|
|
|
ModTime: zipObjInfo.ModTime,
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Set standard object headers.
|
|
|
|
|
if err = setObjectHeaders(w, objInfo, nil, opts); err != nil {
|
|
|
|
|
writeErrorResponseHeadersOnly(w, toAPIError(ctx, err))
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Set any additional requested response headers.
|
2021-08-07 22:43:01 -07:00
|
|
|
|
setHeadGetRespHeaders(w, r.Form)
|
2021-06-10 16:17:03 +01:00
|
|
|
|
|
|
|
|
|
// Successful response.
|
|
|
|
|
if rs != nil {
|
|
|
|
|
w.WriteHeader(http.StatusPartialContent)
|
|
|
|
|
} else {
|
|
|
|
|
w.WriteHeader(http.StatusOK)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Update the passed zip object metadata with the zip contents info, file name, modtime, size, etc..
|
|
|
|
|
func updateObjectMetadataWithZipInfo(ctx context.Context, objectAPI ObjectLayer, bucket, object string, opts ObjectOptions) ([]byte, error) {
|
|
|
|
|
files, srcInfo, err := getFilesListFromZIPObject(ctx, objectAPI, bucket, object, opts)
|
|
|
|
|
if err != nil {
|
|
|
|
|
return nil, err
|
|
|
|
|
}
|
|
|
|
|
files.OptimizeSize()
|
|
|
|
|
zipInfo, err := files.Serialize()
|
|
|
|
|
if err != nil {
|
|
|
|
|
return nil, err
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
srcInfo.UserDefined[archiveTypeMetadataKey] = archiveType
|
2021-11-25 09:11:25 -08:00
|
|
|
|
if globalIsErasure {
|
|
|
|
|
srcInfo.UserDefined[archiveInfoMetadataKey] = string(zipInfo)
|
|
|
|
|
} else {
|
|
|
|
|
srcInfo.UserDefined[archiveInfoMetadataKey] = base64.StdEncoding.EncodeToString(zipInfo)
|
|
|
|
|
}
|
2021-06-10 16:17:03 +01:00
|
|
|
|
srcInfo.metadataOnly = true
|
|
|
|
|
|
|
|
|
|
// Always update the same version id & modtime
|
|
|
|
|
|
|
|
|
|
// Passing opts twice as source & destination options will update the metadata
|
|
|
|
|
// of the same object version to avoid creating a new version.
|
|
|
|
|
_, err = objectAPI.CopyObject(ctx, bucket, object, bucket, object, srcInfo, opts, opts)
|
|
|
|
|
if err != nil {
|
|
|
|
|
return nil, err
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return zipInfo, nil
|
|
|
|
|
}
|