2020-07-29 18:00:04 +08:00
|
|
|
package handler
|
2020-07-26 17:09:05 +08:00
|
|
|
|
|
|
|
import (
|
|
|
|
"net/http"
|
|
|
|
|
2022-01-04 15:51:32 +08:00
|
|
|
"github.com/zeromicro/go-zero/core/logx"
|
|
|
|
"github.com/zeromicro/go-zero/core/syncx"
|
|
|
|
"github.com/zeromicro/go-zero/rest/internal"
|
2020-07-26 17:09:05 +08:00
|
|
|
)
|
|
|
|
|
2023-01-08 16:41:53 +08:00
|
|
|
// MaxConnsHandler returns a middleware that limit the concurrent connections.
|
|
|
|
func MaxConnsHandler(n int) func(http.Handler) http.Handler {
|
2020-07-26 17:09:05 +08:00
|
|
|
if n <= 0 {
|
|
|
|
return func(next http.Handler) http.Handler {
|
|
|
|
return next
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return func(next http.Handler) http.Handler {
|
2020-10-11 19:42:40 +08:00
|
|
|
latch := syncx.NewLimit(n)
|
2020-07-26 17:09:05 +08:00
|
|
|
|
|
|
|
return http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
2020-10-11 19:42:40 +08:00
|
|
|
if latch.TryBorrow() {
|
2020-07-26 17:09:05 +08:00
|
|
|
defer func() {
|
2020-10-11 19:42:40 +08:00
|
|
|
if err := latch.Return(); err != nil {
|
2020-07-26 17:09:05 +08:00
|
|
|
logx.Error(err)
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
|
|
|
|
next.ServeHTTP(w, r)
|
|
|
|
} else {
|
2020-10-10 11:53:49 +08:00
|
|
|
internal.Errorf(r, "concurrent connections over %d, rejected with code %d",
|
2020-07-26 17:09:05 +08:00
|
|
|
n, http.StatusServiceUnavailable)
|
|
|
|
w.WriteHeader(http.StatusServiceUnavailable)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|