mirror of https://github.com/minio/minio.git
Merge pull request #506 from harshavardhana/pr_out_enhance_error_responses_for_request_limit_and_bring_some_code_from_api_errors
This commit is contained in:
commit
f20515b4ed
|
@ -18,11 +18,12 @@ package quota
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"errors"
|
"errors"
|
||||||
"github.com/minio-io/minio/pkg/iodine"
|
|
||||||
"io"
|
"io"
|
||||||
"net"
|
"net"
|
||||||
"net/http"
|
"net/http"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
|
"github.com/minio-io/minio/pkg/iodine"
|
||||||
)
|
)
|
||||||
|
|
||||||
// bandwidthQuotaHandler
|
// bandwidthQuotaHandler
|
||||||
|
@ -31,28 +32,12 @@ type bandwidthQuotaHandler struct {
|
||||||
quotas *quotaMap
|
quotas *quotaMap
|
||||||
}
|
}
|
||||||
|
|
||||||
var bandwidthQuotaExceeded = ErrorResponse{
|
|
||||||
Code: "BandwithQuotaExceeded",
|
|
||||||
Message: "Bandwidth Quota Exceeded",
|
|
||||||
Resource: "",
|
|
||||||
RequestID: "",
|
|
||||||
HostID: "",
|
|
||||||
}
|
|
||||||
|
|
||||||
var bandwidthInsufficientToProceed = ErrorResponse{
|
|
||||||
Code: "BandwidthQuotaWillBeExceeded",
|
|
||||||
Message: "Bandwidth quota will be exceeded with this request",
|
|
||||||
Resource: "",
|
|
||||||
RequestID: "",
|
|
||||||
HostID: "",
|
|
||||||
}
|
|
||||||
|
|
||||||
// ServeHTTP is an http.Handler ServeHTTP method
|
// ServeHTTP is an http.Handler ServeHTTP method
|
||||||
func (h *bandwidthQuotaHandler) ServeHTTP(w http.ResponseWriter, req *http.Request) {
|
func (h *bandwidthQuotaHandler) ServeHTTP(w http.ResponseWriter, req *http.Request) {
|
||||||
host, _, _ := net.SplitHostPort(req.RemoteAddr)
|
host, _, _ := net.SplitHostPort(req.RemoteAddr)
|
||||||
longIP := longIP{net.ParseIP(host)}.IptoUint32()
|
longIP := longIP{net.ParseIP(host)}.IptoUint32()
|
||||||
if h.quotas.WillExceedQuota(longIP, req.ContentLength) {
|
if h.quotas.WillExceedQuota(longIP, req.ContentLength) {
|
||||||
writeError(w, req, bandwidthInsufficientToProceed, 429)
|
writeErrorResponse(w, req, BandWidthInsufficientToProceed, req.URL.Path)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
req.Body = "aReader{
|
req.Body = "aReader{
|
||||||
|
@ -98,7 +83,7 @@ func (q *quotaReader) Read(b []byte) (int, error) {
|
||||||
}
|
}
|
||||||
if q.quotas.IsQuotaMet(q.ip) {
|
if q.quotas.IsQuotaMet(q.ip) {
|
||||||
q.err = true
|
q.err = true
|
||||||
writeError(q.w, q.req, bandwidthQuotaExceeded, 429)
|
writeErrorResponse(q.w, q.req, BandWidthQuotaExceeded, q.req.URL.Path)
|
||||||
return 0, iodine.New(errors.New("Quota Met"), nil)
|
return 0, iodine.New(errors.New("Quota Met"), nil)
|
||||||
}
|
}
|
||||||
n, err := q.ReadCloser.Read(b)
|
n, err := q.ReadCloser.Read(b)
|
||||||
|
|
|
@ -0,0 +1,121 @@
|
||||||
|
/*
|
||||||
|
* Minimalist Object Storage, (C) 2015 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 quota
|
||||||
|
|
||||||
|
import (
|
||||||
|
"bytes"
|
||||||
|
"encoding/xml"
|
||||||
|
"net/http"
|
||||||
|
)
|
||||||
|
|
||||||
|
// copied from api, no cyclic deps allowed
|
||||||
|
|
||||||
|
// Error structure
|
||||||
|
type Error struct {
|
||||||
|
Code string
|
||||||
|
Description string
|
||||||
|
HTTPStatusCode int
|
||||||
|
}
|
||||||
|
|
||||||
|
// ErrorResponse - error response format
|
||||||
|
type ErrorResponse struct {
|
||||||
|
XMLName xml.Name `xml:"Error" json:"-"`
|
||||||
|
Code string
|
||||||
|
Message string
|
||||||
|
Resource string
|
||||||
|
RequestID string
|
||||||
|
HostID string
|
||||||
|
}
|
||||||
|
|
||||||
|
// Quota standard errors non exhaustive list
|
||||||
|
const (
|
||||||
|
RequestTimeTooSkewed = iota
|
||||||
|
BandWidthQuotaExceeded
|
||||||
|
BandWidthInsufficientToProceed
|
||||||
|
SlowDown
|
||||||
|
)
|
||||||
|
|
||||||
|
// Golang http doesn't implement these
|
||||||
|
const (
|
||||||
|
StatusTooManyRequests = 429
|
||||||
|
)
|
||||||
|
|
||||||
|
func writeErrorResponse(w http.ResponseWriter, req *http.Request, errorType int, resource string) {
|
||||||
|
error := getErrorCode(errorType)
|
||||||
|
errorResponse := getErrorResponse(error, resource)
|
||||||
|
// set headers
|
||||||
|
writeErrorHeaders(w)
|
||||||
|
w.WriteHeader(error.HTTPStatusCode)
|
||||||
|
// write body
|
||||||
|
encodedErrorResponse := encodeErrorResponse(errorResponse)
|
||||||
|
w.Write(encodedErrorResponse)
|
||||||
|
}
|
||||||
|
|
||||||
|
func writeErrorHeaders(w http.ResponseWriter) {
|
||||||
|
w.Header().Set("Server", "Minio")
|
||||||
|
w.Header().Set("Accept-Ranges", "bytes")
|
||||||
|
w.Header().Set("Content-Type", "application/xml")
|
||||||
|
w.Header().Set("Connection", "close")
|
||||||
|
}
|
||||||
|
|
||||||
|
// Error code to Error structure map
|
||||||
|
var errorCodeResponse = map[int]Error{
|
||||||
|
BandWidthQuotaExceeded: {
|
||||||
|
Code: "BandwidthQuotaExceeded",
|
||||||
|
Description: "Bandwidth Quota Exceeded",
|
||||||
|
HTTPStatusCode: StatusTooManyRequests,
|
||||||
|
},
|
||||||
|
BandWidthInsufficientToProceed: {
|
||||||
|
Code: "BandwidthQuotaWillBeExceeded",
|
||||||
|
Description: "Bandwidth quota will be exceeded with this request",
|
||||||
|
HTTPStatusCode: StatusTooManyRequests,
|
||||||
|
},
|
||||||
|
SlowDown: {
|
||||||
|
Code: "SlowDown",
|
||||||
|
Description: "Reduce your request rate.",
|
||||||
|
HTTPStatusCode: StatusTooManyRequests,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
// Write error response headers
|
||||||
|
func encodeErrorResponse(response interface{}) []byte {
|
||||||
|
var bytesBuffer bytes.Buffer
|
||||||
|
encoder := xml.NewEncoder(&bytesBuffer)
|
||||||
|
encoder.Encode(response)
|
||||||
|
return bytesBuffer.Bytes()
|
||||||
|
}
|
||||||
|
|
||||||
|
// errorCodeError provides errorCode to Error. It returns empty if the code provided is unknown
|
||||||
|
func getErrorCode(code int) Error {
|
||||||
|
return errorCodeResponse[code]
|
||||||
|
}
|
||||||
|
|
||||||
|
// getErrorResponse gets in standard error and resource value and
|
||||||
|
// provides a encodable populated response values
|
||||||
|
func getErrorResponse(err Error, resource string) ErrorResponse {
|
||||||
|
var data = ErrorResponse{}
|
||||||
|
data.Code = err.Code
|
||||||
|
data.Message = err.Description
|
||||||
|
if resource != "" {
|
||||||
|
data.Resource = resource
|
||||||
|
}
|
||||||
|
// TODO implement this in future
|
||||||
|
data.RequestID = "3L137"
|
||||||
|
data.HostID = "3L137"
|
||||||
|
|
||||||
|
return data
|
||||||
|
}
|
|
@ -17,11 +17,8 @@
|
||||||
package quota
|
package quota
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"bytes"
|
|
||||||
"encoding/binary"
|
"encoding/binary"
|
||||||
"encoding/xml"
|
|
||||||
"net"
|
"net"
|
||||||
"net/http"
|
|
||||||
"sync"
|
"sync"
|
||||||
"time"
|
"time"
|
||||||
)
|
)
|
||||||
|
@ -95,24 +92,3 @@ func (p longIP) IptoUint32() (result uint32) {
|
||||||
}
|
}
|
||||||
return binary.BigEndian.Uint32(ip)
|
return binary.BigEndian.Uint32(ip)
|
||||||
}
|
}
|
||||||
|
|
||||||
// copied from api, no cyclic deps allowed
|
|
||||||
|
|
||||||
// ErrorResponse - error response format
|
|
||||||
type ErrorResponse struct {
|
|
||||||
XMLName xml.Name `xml:"Error" json:"-"`
|
|
||||||
Code string
|
|
||||||
Message string
|
|
||||||
Resource string
|
|
||||||
RequestID string
|
|
||||||
HostID string
|
|
||||||
}
|
|
||||||
|
|
||||||
func writeError(w http.ResponseWriter, req *http.Request, errorResponse ErrorResponse, status int) {
|
|
||||||
var buf bytes.Buffer
|
|
||||||
encoder := xml.NewEncoder(&buf)
|
|
||||||
w.WriteHeader(status)
|
|
||||||
encoder.Encode(errorResponse)
|
|
||||||
encoder.Flush()
|
|
||||||
w.Write(buf.Bytes())
|
|
||||||
}
|
|
||||||
|
|
|
@ -33,7 +33,7 @@ func (h *requestLimitHandler) ServeHTTP(w http.ResponseWriter, req *http.Request
|
||||||
host, _, _ := net.SplitHostPort(req.RemoteAddr)
|
host, _, _ := net.SplitHostPort(req.RemoteAddr)
|
||||||
longIP := longIP{net.ParseIP(host)}.IptoUint32()
|
longIP := longIP{net.ParseIP(host)}.IptoUint32()
|
||||||
if h.quotas.IsQuotaMet(longIP) {
|
if h.quotas.IsQuotaMet(longIP) {
|
||||||
return
|
writeErrorResponse(w, req, SlowDown, req.URL.Path)
|
||||||
}
|
}
|
||||||
h.quotas.Add(longIP, 1)
|
h.quotas.Add(longIP, 1)
|
||||||
h.handler.ServeHTTP(w, req)
|
h.handler.ServeHTTP(w, req)
|
||||||
|
|
Loading…
Reference in New Issue