Merge pull request #495 from fkautz/pr_out_quotas_are_more_accurate_occur_on_read_
commit
a3af3514ca
@ -0,0 +1,83 @@ |
|||||||
|
/* |
||||||
|
* 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 ( |
||||||
|
"errors" |
||||||
|
"io" |
||||||
|
"net" |
||||||
|
"net/http" |
||||||
|
"time" |
||||||
|
) |
||||||
|
|
||||||
|
// bandwidthQuotaHandler
|
||||||
|
type bandwidthQuotaHandler struct { |
||||||
|
handler http.Handler |
||||||
|
quotas *quotaMap |
||||||
|
} |
||||||
|
|
||||||
|
// ServeHTTP is an http.Handler ServeHTTP method
|
||||||
|
func (h *bandwidthQuotaHandler) ServeHTTP(w http.ResponseWriter, req *http.Request) { |
||||||
|
host, _, _ := net.SplitHostPort(req.RemoteAddr) |
||||||
|
longIP := longIP{net.ParseIP(host)}.IptoUint32() |
||||||
|
req.Body = quotaReader{ |
||||||
|
ReadCloser: req.Body, |
||||||
|
quotas: h.quotas, |
||||||
|
ip: longIP, |
||||||
|
} |
||||||
|
h.handler.ServeHTTP(w, req) |
||||||
|
} |
||||||
|
|
||||||
|
// BandwidthCap sets a quote based upon bandwidth used
|
||||||
|
func BandwidthCap(h http.Handler, limit int64, duration time.Duration) http.Handler { |
||||||
|
return &bandwidthQuotaHandler{ |
||||||
|
handler: h, |
||||||
|
quotas: "aMap{ |
||||||
|
data: make(map[int64]map[uint32]int64), |
||||||
|
limit: int64(limit), |
||||||
|
duration: duration, |
||||||
|
segmentSize: segmentSize(duration), |
||||||
|
}, |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
type quotaReader struct { |
||||||
|
io.ReadCloser |
||||||
|
quotas *quotaMap |
||||||
|
ip uint32 |
||||||
|
} |
||||||
|
|
||||||
|
func (q quotaReader) Read(b []byte) (int, error) { |
||||||
|
if q.quotas.IsQuotaMet(q.ip) { |
||||||
|
return 0, errors.New("Quota Met") |
||||||
|
} |
||||||
|
n, err := q.ReadCloser.Read(b) |
||||||
|
q.quotas.Add(q.ip, int64(n)) |
||||||
|
return n, err |
||||||
|
} |
||||||
|
|
||||||
|
func (q quotaReader) Close() error { |
||||||
|
return q.ReadCloser.Close() |
||||||
|
} |
||||||
|
|
||||||
|
func segmentSize(duration time.Duration) time.Duration { |
||||||
|
var segmentSize time.Duration |
||||||
|
for i := int64(1); i < duration.Nanoseconds(); i = i * 10 { |
||||||
|
segmentSize = time.Duration(i) |
||||||
|
} |
||||||
|
return segmentSize |
||||||
|
} |
@ -0,0 +1,53 @@ |
|||||||
|
/* |
||||||
|
* 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 ( |
||||||
|
"net" |
||||||
|
"net/http" |
||||||
|
"time" |
||||||
|
) |
||||||
|
|
||||||
|
// requestLimitHandler
|
||||||
|
type requestLimitHandler struct { |
||||||
|
handler http.Handler |
||||||
|
quotas *quotaMap |
||||||
|
} |
||||||
|
|
||||||
|
// ServeHTTP is an http.Handler ServeHTTP method
|
||||||
|
func (h *requestLimitHandler) ServeHTTP(w http.ResponseWriter, req *http.Request) { |
||||||
|
host, _, _ := net.SplitHostPort(req.RemoteAddr) |
||||||
|
longIP := longIP{net.ParseIP(host)}.IptoUint32() |
||||||
|
if h.quotas.IsQuotaMet(longIP) { |
||||||
|
return |
||||||
|
} |
||||||
|
h.quotas.Add(longIP, 1) |
||||||
|
h.handler.ServeHTTP(w, req) |
||||||
|
} |
||||||
|
|
||||||
|
// RequestLimit sets a quote based upon number of requests allowed over a time period
|
||||||
|
func RequestLimit(h http.Handler, limit int64, duration time.Duration) http.Handler { |
||||||
|
return &requestLimitHandler{ |
||||||
|
handler: h, |
||||||
|
quotas: "aMap{ |
||||||
|
data: make(map[int64]map[uint32]int64), |
||||||
|
limit: int64(limit), |
||||||
|
duration: duration, |
||||||
|
segmentSize: segmentSize(duration), |
||||||
|
}, |
||||||
|
} |
||||||
|
} |
Loading…
Reference in new issue