2
2
mirror of https://github.com/octoleo/restic.git synced 2025-01-25 16:18:34 +00:00
restic/vendor/github.com/minio/minio-go/api-error-response.go

283 lines
8.0 KiB
Go
Raw Normal View History

2017-07-23 14:24:45 +02:00
/*
2017-12-08 20:45:59 +01:00
* Minio Go Library for Amazon S3 Compatible Cloud Storage
* Copyright 2015-2017 Minio, Inc.
2017-07-23 14:24:45 +02:00
*
* 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 (
"encoding/xml"
"fmt"
"net/http"
)
/* **** SAMPLE ERROR RESPONSE ****
<?xml version="1.0" encoding="UTF-8"?>
<Error>
<Code>AccessDenied</Code>
<Message>Access Denied</Message>
<BucketName>bucketName</BucketName>
<Key>objectName</Key>
<RequestId>F19772218238A85A</RequestId>
<HostId>GuWkjyviSiGHizehqpmsD1ndz5NClSP19DOT+s2mv7gXGQ8/X1lhbDGiIJEXpGFD</HostId>
</Error>
*/
// ErrorResponse - Is the typed error returned by all API operations.
2019-01-27 21:07:57 +01:00
// ErrorResponse struct should be comparable since it is compared inside
// golang http API (https://github.com/golang/go/issues/29768)
2017-07-23 14:24:45 +02:00
type ErrorResponse struct {
XMLName xml.Name `xml:"Error" json:"-"`
Code string
Message string
BucketName string
Key string
RequestID string `xml:"RequestId"`
HostID string `xml:"HostId"`
// Region where the bucket is located. This header is returned
// only in HEAD bucket and ListObjects response.
Region string
2017-12-08 20:45:59 +01:00
// Underlying HTTP status code for the returned error
StatusCode int `xml:"-" json:"-"`
2017-07-23 14:24:45 +02:00
}
// ToErrorResponse - Returns parsed ErrorResponse struct from body and
// http headers.
//
// For example:
//
// import s3 "github.com/minio/minio-go"
// ...
// ...
// reader, stat, err := s3.GetObject(...)
// if err != nil {
// resp := s3.ToErrorResponse(err)
// }
// ...
func ToErrorResponse(err error) ErrorResponse {
switch err := err.(type) {
case ErrorResponse:
return err
default:
return ErrorResponse{}
}
}
// Error - Returns S3 error string.
func (e ErrorResponse) Error() string {
if e.Message == "" {
msg, ok := s3ErrorResponseMap[e.Code]
if !ok {
msg = fmt.Sprintf("Error response code %s.", e.Code)
}
return msg
}
return e.Message
}
// Common string for errors to report issue location in unexpected
// cases.
const (
reportIssue = "Please report this issue at https://github.com/minio/minio-go/issues."
)
// httpRespToErrorResponse returns a new encoded ErrorResponse
// structure as error.
func httpRespToErrorResponse(resp *http.Response, bucketName, objectName string) error {
if resp == nil {
msg := "Response is empty. " + reportIssue
return ErrInvalidArgument(msg)
}
2017-12-08 20:45:59 +01:00
errResp := ErrorResponse{
StatusCode: resp.StatusCode,
}
2017-07-23 14:24:45 +02:00
err := xmlDecoder(resp.Body, &errResp)
// Xml decoding failed with no body, fall back to HTTP headers.
if err != nil {
switch resp.StatusCode {
case http.StatusNotFound:
if objectName == "" {
errResp = ErrorResponse{
2017-12-08 20:45:59 +01:00
StatusCode: resp.StatusCode,
2017-07-23 14:24:45 +02:00
Code: "NoSuchBucket",
Message: "The specified bucket does not exist.",
BucketName: bucketName,
}
} else {
errResp = ErrorResponse{
2017-12-08 20:45:59 +01:00
StatusCode: resp.StatusCode,
2017-07-23 14:24:45 +02:00
Code: "NoSuchKey",
Message: "The specified key does not exist.",
BucketName: bucketName,
Key: objectName,
}
}
case http.StatusForbidden:
errResp = ErrorResponse{
2017-12-08 20:45:59 +01:00
StatusCode: resp.StatusCode,
2017-07-23 14:24:45 +02:00
Code: "AccessDenied",
Message: "Access Denied.",
BucketName: bucketName,
Key: objectName,
}
case http.StatusConflict:
errResp = ErrorResponse{
2017-12-08 20:45:59 +01:00
StatusCode: resp.StatusCode,
2017-07-23 14:24:45 +02:00
Code: "Conflict",
Message: "Bucket not empty.",
BucketName: bucketName,
}
case http.StatusPreconditionFailed:
errResp = ErrorResponse{
2017-12-08 20:45:59 +01:00
StatusCode: resp.StatusCode,
2017-07-23 14:24:45 +02:00
Code: "PreconditionFailed",
Message: s3ErrorResponseMap["PreconditionFailed"],
BucketName: bucketName,
Key: objectName,
}
default:
errResp = ErrorResponse{
2017-12-08 20:45:59 +01:00
StatusCode: resp.StatusCode,
2017-07-23 14:24:45 +02:00
Code: resp.Status,
Message: resp.Status,
BucketName: bucketName,
}
}
}
2017-12-08 20:45:59 +01:00
// Save hostID, requestID and region information
2017-07-23 14:24:45 +02:00
// from headers if not available through error XML.
if errResp.RequestID == "" {
errResp.RequestID = resp.Header.Get("x-amz-request-id")
}
if errResp.HostID == "" {
errResp.HostID = resp.Header.Get("x-amz-id-2")
}
if errResp.Region == "" {
errResp.Region = resp.Header.Get("x-amz-bucket-region")
}
if errResp.Code == "InvalidRegion" && errResp.Region != "" {
2017-12-08 20:45:59 +01:00
errResp.Message = fmt.Sprintf("Region does not match, expecting region %s.", errResp.Region)
2017-07-23 14:24:45 +02:00
}
return errResp
}
// ErrTransferAccelerationBucket - bucket name is invalid to be used with transfer acceleration.
func ErrTransferAccelerationBucket(bucketName string) error {
return ErrorResponse{
2017-12-08 20:45:59 +01:00
StatusCode: http.StatusBadRequest,
2017-07-23 14:24:45 +02:00
Code: "InvalidArgument",
2017-12-08 20:45:59 +01:00
Message: "The name of the bucket used for Transfer Acceleration must be DNS-compliant and must not contain periods ..",
2017-07-23 14:24:45 +02:00
BucketName: bucketName,
}
}
// ErrEntityTooLarge - Input size is larger than supported maximum.
func ErrEntityTooLarge(totalSize, maxObjectSize int64, bucketName, objectName string) error {
msg := fmt.Sprintf("Your proposed upload size %d exceeds the maximum allowed object size %d for single PUT operation.", totalSize, maxObjectSize)
return ErrorResponse{
2017-12-08 20:45:59 +01:00
StatusCode: http.StatusBadRequest,
2017-07-23 14:24:45 +02:00
Code: "EntityTooLarge",
Message: msg,
BucketName: bucketName,
Key: objectName,
}
}
// ErrEntityTooSmall - Input size is smaller than supported minimum.
func ErrEntityTooSmall(totalSize int64, bucketName, objectName string) error {
2017-12-08 20:45:59 +01:00
msg := fmt.Sprintf("Your proposed upload size %d is below the minimum allowed object size 0B for single PUT operation.", totalSize)
2017-07-23 14:24:45 +02:00
return ErrorResponse{
2017-12-08 20:45:59 +01:00
StatusCode: http.StatusBadRequest,
Code: "EntityTooSmall",
2017-07-23 14:24:45 +02:00
Message: msg,
BucketName: bucketName,
Key: objectName,
}
}
// ErrUnexpectedEOF - Unexpected end of file reached.
func ErrUnexpectedEOF(totalRead, totalSize int64, bucketName, objectName string) error {
2017-12-08 20:45:59 +01:00
msg := fmt.Sprintf("Data read %d is not equal to the size %d of the input Reader.", totalRead, totalSize)
2017-07-23 14:24:45 +02:00
return ErrorResponse{
2017-12-08 20:45:59 +01:00
StatusCode: http.StatusBadRequest,
2017-07-23 14:24:45 +02:00
Code: "UnexpectedEOF",
Message: msg,
BucketName: bucketName,
Key: objectName,
}
}
// ErrInvalidBucketName - Invalid bucket name response.
func ErrInvalidBucketName(message string) error {
return ErrorResponse{
2017-12-08 20:45:59 +01:00
StatusCode: http.StatusBadRequest,
Code: "InvalidBucketName",
Message: message,
RequestID: "minio",
2017-07-23 14:24:45 +02:00
}
}
// ErrInvalidObjectName - Invalid object name response.
func ErrInvalidObjectName(message string) error {
return ErrorResponse{
2017-12-08 20:45:59 +01:00
StatusCode: http.StatusNotFound,
Code: "NoSuchKey",
Message: message,
RequestID: "minio",
2017-07-23 14:24:45 +02:00
}
}
// ErrInvalidObjectPrefix - Invalid object prefix response is
// similar to object name response.
var ErrInvalidObjectPrefix = ErrInvalidObjectName
// ErrInvalidArgument - Invalid argument response.
func ErrInvalidArgument(message string) error {
return ErrorResponse{
2017-12-08 20:45:59 +01:00
StatusCode: http.StatusBadRequest,
Code: "InvalidArgument",
Message: message,
RequestID: "minio",
2017-07-23 14:24:45 +02:00
}
}
// ErrNoSuchBucketPolicy - No Such Bucket Policy response
// The specified bucket does not have a bucket policy.
func ErrNoSuchBucketPolicy(message string) error {
return ErrorResponse{
2017-12-08 20:45:59 +01:00
StatusCode: http.StatusNotFound,
Code: "NoSuchBucketPolicy",
Message: message,
RequestID: "minio",
2017-07-23 14:24:45 +02:00
}
}
// ErrAPINotSupported - API not supported response
// The specified API call is not supported
func ErrAPINotSupported(message string) error {
return ErrorResponse{
2017-12-08 20:45:59 +01:00
StatusCode: http.StatusNotImplemented,
Code: "APINotSupported",
Message: message,
RequestID: "minio",
2017-07-23 14:24:45 +02:00
}
}