Clash.Meta/hub/route/proxies.go

138 lines
3.3 KiB
Go
Raw Normal View History

2018-11-21 13:47:46 +08:00
package route
import (
"context"
"fmt"
"net/http"
"net/url"
"strconv"
"time"
A "github.com/Dreamacro/clash/adapters/outbound"
2019-07-02 19:18:03 +08:00
"github.com/Dreamacro/clash/common/picker"
2018-11-21 13:47:46 +08:00
C "github.com/Dreamacro/clash/constant"
T "github.com/Dreamacro/clash/tunnel"
"github.com/go-chi/chi"
"github.com/go-chi/render"
)
func proxyRouter() http.Handler {
r := chi.NewRouter()
r.Get("/", getProxies)
r.Route("/{name}", func(r chi.Router) {
r.Use(parseProxyName, findProxyByName)
r.Get("/", getProxy)
r.Get("/delay", getProxyDelay)
r.Put("/", updateProxy)
})
return r
}
// When name is composed of a partial escape string, Golang does not unescape it
func parseProxyName(next http.Handler) http.Handler {
return http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
name := chi.URLParam(r, "name")
if newName, err := url.PathUnescape(name); err == nil {
name = newName
}
ctx := context.WithValue(r.Context(), CtxKeyProxyName, name)
next.ServeHTTP(w, r.WithContext(ctx))
})
}
func findProxyByName(next http.Handler) http.Handler {
return http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
name := r.Context().Value(CtxKeyProxyName).(string)
proxies := T.Instance().Proxies()
proxy, exist := proxies[name]
if !exist {
2018-12-10 11:33:37 +08:00
render.Status(r, http.StatusNotFound)
render.JSON(w, r, ErrNotFound)
2018-11-21 13:47:46 +08:00
return
}
ctx := context.WithValue(r.Context(), CtxKeyProxy, proxy)
next.ServeHTTP(w, r.WithContext(ctx))
})
}
func getProxies(w http.ResponseWriter, r *http.Request) {
proxies := T.Instance().Proxies()
2019-02-13 23:45:43 +08:00
render.JSON(w, r, render.M{
2018-11-21 13:47:46 +08:00
"proxies": proxies,
})
}
func getProxy(w http.ResponseWriter, r *http.Request) {
proxy := r.Context().Value(CtxKeyProxy).(C.Proxy)
2018-12-10 11:33:37 +08:00
render.JSON(w, r, proxy)
2018-11-21 13:47:46 +08:00
}
type UpdateProxyRequest struct {
Name string `json:"name"`
}
func updateProxy(w http.ResponseWriter, r *http.Request) {
req := UpdateProxyRequest{}
if err := render.DecodeJSON(r.Body, &req); err != nil {
2018-12-10 11:33:37 +08:00
render.Status(r, http.StatusBadRequest)
render.JSON(w, r, ErrBadRequest)
2018-11-21 13:47:46 +08:00
return
}
proxy := r.Context().Value(CtxKeyProxy).(*A.Proxy)
selector, ok := proxy.ProxyAdapter.(*A.Selector)
2018-11-21 13:47:46 +08:00
if !ok {
2018-12-10 11:33:37 +08:00
render.Status(r, http.StatusBadRequest)
render.JSON(w, r, newError("Must be a Selector"))
2018-11-21 13:47:46 +08:00
return
}
if err := selector.Set(req.Name); err != nil {
2018-12-10 11:33:37 +08:00
render.Status(r, http.StatusBadRequest)
render.JSON(w, r, newError(fmt.Sprintf("Selector update error: %s", err.Error())))
2018-11-21 13:47:46 +08:00
return
}
2018-12-10 11:33:37 +08:00
render.NoContent(w, r)
2018-11-21 13:47:46 +08:00
}
func getProxyDelay(w http.ResponseWriter, r *http.Request) {
query := r.URL.Query()
url := query.Get("url")
timeout, err := strconv.ParseInt(query.Get("timeout"), 10, 16)
if err != nil {
2018-12-10 11:33:37 +08:00
render.Status(r, http.StatusBadRequest)
render.JSON(w, r, ErrBadRequest)
2018-11-21 13:47:46 +08:00
return
}
proxy := r.Context().Value(CtxKeyProxy).(C.Proxy)
picker, ctx, cancel := picker.WithTimeout(context.Background(), time.Millisecond*time.Duration(timeout))
2019-07-02 19:18:03 +08:00
defer cancel()
picker.Go(func() (interface{}, error) {
return proxy.URLTest(ctx, url)
})
2018-11-21 13:47:46 +08:00
2019-07-02 19:18:03 +08:00
elm := picker.Wait()
if elm == nil {
2019-09-08 22:33:52 +08:00
render.Status(r, http.StatusGatewayTimeout)
2018-12-10 11:33:37 +08:00
render.JSON(w, r, ErrRequestTimeout)
2019-07-02 19:18:03 +08:00
return
}
delay := elm.(uint16)
if delay == 0 {
render.Status(r, http.StatusServiceUnavailable)
render.JSON(w, r, newError("An error occurred in the delay test"))
return
2018-11-21 13:47:46 +08:00
}
2019-07-02 19:18:03 +08:00
render.JSON(w, r, render.M{
"delay": delay,
})
2018-11-21 13:47:46 +08:00
}