frp/client/admin_api.go

263 lines
6.7 KiB
Go
Raw Normal View History

// Copyright 2017 fatedier, fatedier@gmail.com
//
// 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 client
import (
2024-02-20 12:01:41 +08:00
"cmp"
"encoding/json"
2018-01-17 01:09:33 +08:00
"fmt"
"io"
2022-06-14 14:24:34 +08:00
"net"
"net/http"
"os"
2024-02-20 12:01:41 +08:00
"slices"
2022-06-14 14:24:34 +08:00
"strconv"
2023-06-30 17:35:37 +08:00
"time"
2018-12-09 22:06:22 +08:00
"github.com/fatedier/frp/client/proxy"
2020-09-23 13:49:14 +08:00
"github.com/fatedier/frp/pkg/config"
"github.com/fatedier/frp/pkg/config/v1/validation"
2023-11-27 15:47:49 +08:00
httppkg "github.com/fatedier/frp/pkg/util/http"
2020-09-23 13:49:14 +08:00
"github.com/fatedier/frp/pkg/util/log"
2023-11-27 15:47:49 +08:00
netpkg "github.com/fatedier/frp/pkg/util/net"
)
type GeneralResponse struct {
2019-02-11 11:26:06 +08:00
Code int
Msg string
}
2023-11-27 15:47:49 +08:00
func (svr *Service) registerRouteHandlers(helper *httppkg.RouterRegisterHelper) {
helper.Router.HandleFunc("/healthz", svr.healthz)
subRouter := helper.Router.NewRoute().Subrouter()
subRouter.Use(helper.AuthMiddleware.Middleware)
// api, see admin_api.go
subRouter.HandleFunc("/api/reload", svr.apiReload).Methods("GET")
subRouter.HandleFunc("/api/stop", svr.apiStop).Methods("POST")
subRouter.HandleFunc("/api/status", svr.apiStatus).Methods("GET")
subRouter.HandleFunc("/api/config", svr.apiGetConfig).Methods("GET")
subRouter.HandleFunc("/api/config", svr.apiPutConfig).Methods("PUT")
// view
subRouter.Handle("/favicon.ico", http.FileServer(helper.AssetsFS)).Methods("GET")
subRouter.PathPrefix("/static/").Handler(
netpkg.MakeHTTPGzipHandler(http.StripPrefix("/static/", http.FileServer(helper.AssetsFS))),
).Methods("GET")
subRouter.HandleFunc("/", func(w http.ResponseWriter, r *http.Request) {
http.Redirect(w, r, "/static/", http.StatusMovedPermanently)
})
}
2021-08-04 14:33:53 +08:00
// /healthz
2023-07-21 10:30:46 +08:00
func (svr *Service) healthz(w http.ResponseWriter, _ *http.Request) {
2021-08-04 14:33:53 +08:00
w.WriteHeader(200)
}
2023-06-30 17:35:37 +08:00
// GET /api/reload
2023-11-16 21:03:36 +08:00
func (svr *Service) apiReload(w http.ResponseWriter, r *http.Request) {
2019-02-11 11:26:06 +08:00
res := GeneralResponse{Code: 200}
2023-11-16 21:03:36 +08:00
strictConfigMode := false
strictStr := r.URL.Query().Get("strictConfig")
if strictStr != "" {
strictConfigMode, _ = strconv.ParseBool(strictStr)
}
2019-01-31 18:35:44 +08:00
log.Info("api request [/api/reload]")
defer func() {
log.Info("api response [/api/reload], code [%d]", res.Code)
2019-02-11 11:26:06 +08:00
w.WriteHeader(res.Code)
if len(res.Msg) > 0 {
2022-08-29 01:02:53 +08:00
_, _ = w.Write([]byte(res.Msg))
2019-02-11 11:26:06 +08:00
}
}()
2023-11-27 15:47:49 +08:00
cliCfg, proxyCfgs, visitorCfgs, _, err := config.LoadClientConfig(svr.configFilePath, strictConfigMode)
if err != nil {
2019-02-11 11:26:06 +08:00
res.Code = 400
res.Msg = err.Error()
2019-02-11 11:26:06 +08:00
log.Warn("reload frpc proxy config error: %s", res.Msg)
return
}
2023-11-27 15:47:49 +08:00
if _, err := validation.ValidateAllClientConfig(cliCfg, proxyCfgs, visitorCfgs); err != nil {
res.Code = 400
res.Msg = err.Error()
log.Warn("reload frpc proxy config error: %s", res.Msg)
return
}
2023-11-27 15:47:49 +08:00
if err := svr.UpdateAllConfigurer(proxyCfgs, visitorCfgs); err != nil {
2019-02-11 11:26:06 +08:00
res.Code = 500
2018-01-17 01:09:33 +08:00
res.Msg = err.Error()
2019-02-11 11:26:06 +08:00
log.Warn("reload frpc proxy config error: %s", res.Msg)
2018-01-17 01:09:33 +08:00
return
}
log.Info("success reload conf")
}
2018-01-17 01:09:33 +08:00
2023-06-30 17:35:37 +08:00
// POST /api/stop
2023-07-21 10:30:46 +08:00
func (svr *Service) apiStop(w http.ResponseWriter, _ *http.Request) {
2023-06-30 17:35:37 +08:00
res := GeneralResponse{Code: 200}
log.Info("api request [/api/stop]")
defer func() {
log.Info("api response [/api/stop], code [%d]", res.Code)
w.WriteHeader(res.Code)
if len(res.Msg) > 0 {
_, _ = w.Write([]byte(res.Msg))
}
}()
go svr.GracefulClose(100 * time.Millisecond)
}
2023-02-26 02:54:53 +08:00
type StatusResp map[string][]ProxyStatusResp
2018-01-17 01:09:33 +08:00
type ProxyStatusResp struct {
Name string `json:"name"`
Type string `json:"type"`
Status string `json:"status"`
Err string `json:"err"`
LocalAddr string `json:"local_addr"`
Plugin string `json:"plugin"`
RemoteAddr string `json:"remote_addr"`
}
2020-05-24 17:48:37 +08:00
func NewProxyStatusResp(status *proxy.WorkingStatus, serverAddr string) ProxyStatusResp {
2018-01-17 01:09:33 +08:00
psr := ProxyStatusResp{
Name: status.Name,
Type: status.Type,
2020-05-24 17:48:37 +08:00
Status: status.Phase,
2018-01-17 01:09:33 +08:00
Err: status.Err,
}
2023-05-30 20:25:22 +08:00
baseCfg := status.Cfg.GetBaseConfig()
2023-02-26 02:54:53 +08:00
if baseCfg.LocalPort != 0 {
psr.LocalAddr = net.JoinHostPort(baseCfg.LocalIP, strconv.Itoa(baseCfg.LocalPort))
}
psr.Plugin = baseCfg.Plugin.Type
2023-02-26 02:54:53 +08:00
if status.Err == "" {
2018-01-17 14:40:08 +08:00
psr.RemoteAddr = status.RemoteAddr
2024-02-20 12:01:41 +08:00
if slices.Contains([]string{"tcp", "udp"}, status.Type) {
2023-02-26 02:54:53 +08:00
psr.RemoteAddr = serverAddr + psr.RemoteAddr
2018-01-17 01:09:33 +08:00
}
}
return psr
}
2023-06-30 17:35:37 +08:00
// GET /api/status
2023-07-21 10:30:46 +08:00
func (svr *Service) apiStatus(w http.ResponseWriter, _ *http.Request) {
2018-01-17 01:09:33 +08:00
var (
buf []byte
2023-02-26 02:54:53 +08:00
res StatusResp = make(map[string][]ProxyStatusResp)
2018-01-17 01:09:33 +08:00
)
2019-01-31 18:35:44 +08:00
2019-02-11 11:26:06 +08:00
log.Info("Http request [/api/status]")
2018-01-17 01:09:33 +08:00
defer func() {
log.Info("Http response [/api/status]")
buf, _ = json.Marshal(&res)
2022-08-29 01:02:53 +08:00
_, _ = w.Write(buf)
2018-01-17 01:09:33 +08:00
}()
svr.ctlMu.RLock()
ctl := svr.ctl
svr.ctlMu.RUnlock()
if ctl == nil {
return
}
ps := ctl.pm.GetAllProxyStatus()
2018-01-17 01:09:33 +08:00
for _, status := range ps {
2023-11-27 15:47:49 +08:00
res[status.Type] = append(res[status.Type], NewProxyStatusResp(status, svr.common.ServerAddr))
2023-02-26 02:54:53 +08:00
}
for _, arrs := range res {
if len(arrs) <= 1 {
continue
2018-01-17 01:09:33 +08:00
}
2024-02-20 12:01:41 +08:00
slices.SortFunc(arrs, func(a, b ProxyStatusResp) int {
return cmp.Compare(a.Name, b.Name)
2023-02-26 02:54:53 +08:00
})
2018-01-17 01:09:33 +08:00
}
}
2019-01-31 17:17:34 +08:00
2023-06-30 17:35:37 +08:00
// GET /api/config
2023-07-21 10:30:46 +08:00
func (svr *Service) apiGetConfig(w http.ResponseWriter, _ *http.Request) {
2019-02-11 11:26:06 +08:00
res := GeneralResponse{Code: 200}
2019-01-31 18:35:44 +08:00
2019-02-11 11:26:06 +08:00
log.Info("Http get request [/api/config]")
2019-01-31 17:17:34 +08:00
defer func() {
2019-02-11 11:26:06 +08:00
log.Info("Http get response [/api/config], code [%d]", res.Code)
w.WriteHeader(res.Code)
if len(res.Msg) > 0 {
2022-08-29 01:02:53 +08:00
_, _ = w.Write([]byte(res.Msg))
2019-01-31 17:17:34 +08:00
}
}()
2023-11-27 15:47:49 +08:00
if svr.configFilePath == "" {
2019-02-11 11:26:06 +08:00
res.Code = 400
res.Msg = "frpc has no config file path"
2019-01-31 18:35:44 +08:00
log.Warn("%s", res.Msg)
2019-01-31 17:17:34 +08:00
return
}
2023-11-27 15:47:49 +08:00
content, err := os.ReadFile(svr.configFilePath)
2019-01-31 17:17:34 +08:00
if err != nil {
2019-02-11 11:26:06 +08:00
res.Code = 400
2019-01-31 17:17:34 +08:00
res.Msg = err.Error()
2019-02-11 11:26:06 +08:00
log.Warn("load frpc config file error: %s", res.Msg)
2019-01-31 17:17:34 +08:00
return
}
res.Msg = string(content)
2019-01-31 17:17:34 +08:00
}
2019-01-31 18:35:44 +08:00
2023-06-30 17:35:37 +08:00
// PUT /api/config
2019-01-31 18:35:44 +08:00
func (svr *Service) apiPutConfig(w http.ResponseWriter, r *http.Request) {
2019-02-11 11:26:06 +08:00
res := GeneralResponse{Code: 200}
2019-01-31 18:35:44 +08:00
2019-02-11 11:26:06 +08:00
log.Info("Http put request [/api/config]")
2019-01-31 18:35:44 +08:00
defer func() {
2019-02-11 11:26:06 +08:00
log.Info("Http put response [/api/config], code [%d]", res.Code)
w.WriteHeader(res.Code)
if len(res.Msg) > 0 {
2022-08-29 01:02:53 +08:00
_, _ = w.Write([]byte(res.Msg))
2019-02-11 11:26:06 +08:00
}
2019-01-31 18:35:44 +08:00
}()
// get new config content
body, err := io.ReadAll(r.Body)
2019-01-31 18:35:44 +08:00
if err != nil {
2019-02-11 11:26:06 +08:00
res.Code = 400
2019-01-31 18:35:44 +08:00
res.Msg = fmt.Sprintf("read request body error: %v", err)
log.Warn("%s", res.Msg)
return
}
if len(body) == 0 {
2019-02-11 11:26:06 +08:00
res.Code = 400
res.Msg = "body can't be empty"
log.Warn("%s", res.Msg)
return
}
2023-11-27 15:47:49 +08:00
if err := os.WriteFile(svr.configFilePath, body, 0o644); err != nil {
2019-02-11 11:26:06 +08:00
res.Code = 500
2019-01-31 18:35:44 +08:00
res.Msg = fmt.Sprintf("write content to frpc config file error: %v", err)
log.Warn("%s", res.Msg)
return
}
}