frp/pkg/sdk/client/client.go

141 lines
3.2 KiB
Go
Raw Normal View History

2021-06-18 15:48:36 +07:00
package client
import (
"context"
2021-06-18 15:48:36 +07:00
"encoding/json"
"fmt"
"io"
2021-06-18 15:48:36 +07:00
"net"
"net/http"
2023-11-16 20:03:36 +07:00
"net/url"
2021-06-18 15:48:36 +07:00
"strconv"
"strings"
"github.com/fatedier/frp/client"
2023-11-27 14:47:49 +07:00
httppkg "github.com/fatedier/frp/pkg/util/http"
2021-06-18 15:48:36 +07:00
)
type Client struct {
address string
authUser string
authPwd string
}
func New(host string, port int) *Client {
return &Client{
address: net.JoinHostPort(host, strconv.Itoa(port)),
}
}
func (c *Client) SetAuth(user, pwd string) {
c.authUser = user
c.authPwd = pwd
}
func (c *Client) GetProxyStatus(ctx context.Context, name string) (*client.ProxyStatusResp, error) {
req, err := http.NewRequestWithContext(ctx, "GET", "http://"+c.address+"/api/status", nil)
2021-06-18 15:48:36 +07:00
if err != nil {
return nil, err
}
content, err := c.do(req)
if err != nil {
return nil, err
}
2023-02-26 01:54:53 +07:00
allStatus := make(client.StatusResp)
2021-06-18 15:48:36 +07:00
if err = json.Unmarshal([]byte(content), &allStatus); err != nil {
return nil, fmt.Errorf("unmarshal http response error: %s", strings.TrimSpace(content))
}
2023-02-26 01:54:53 +07:00
for _, pss := range allStatus {
for _, ps := range pss {
if ps.Name == name {
return &ps, nil
}
2021-06-18 15:48:36 +07:00
}
}
return nil, fmt.Errorf("no proxy status found")
}
func (c *Client) GetAllProxyStatus(ctx context.Context) (client.StatusResp, error) {
req, err := http.NewRequestWithContext(ctx, "GET", "http://"+c.address+"/api/status", nil)
if err != nil {
return nil, err
}
content, err := c.do(req)
if err != nil {
return nil, err
}
allStatus := make(client.StatusResp)
if err = json.Unmarshal([]byte(content), &allStatus); err != nil {
return nil, fmt.Errorf("unmarshal http response error: %s", strings.TrimSpace(content))
}
return allStatus, nil
}
func (c *Client) Reload(ctx context.Context, strictMode bool) error {
2023-11-16 20:03:36 +07:00
v := url.Values{}
if strictMode {
v.Set("strictConfig", "true")
}
queryStr := ""
if len(v) > 0 {
queryStr = "?" + v.Encode()
}
req, err := http.NewRequestWithContext(ctx, "GET", "http://"+c.address+"/api/reload"+queryStr, nil)
2021-06-18 15:48:36 +07:00
if err != nil {
2023-06-30 16:35:37 +07:00
return err
}
_, err = c.do(req)
return err
}
func (c *Client) Stop(ctx context.Context) error {
req, err := http.NewRequestWithContext(ctx, "POST", "http://"+c.address+"/api/stop", nil)
2023-06-30 16:35:37 +07:00
if err != nil {
2021-06-18 15:48:36 +07:00
return err
}
_, err = c.do(req)
return err
}
func (c *Client) GetConfig(ctx context.Context) (string, error) {
req, err := http.NewRequestWithContext(ctx, "GET", "http://"+c.address+"/api/config", nil)
2021-06-18 15:48:36 +07:00
if err != nil {
return "", err
}
return c.do(req)
}
func (c *Client) UpdateConfig(ctx context.Context, content string) error {
req, err := http.NewRequestWithContext(ctx, "PUT", "http://"+c.address+"/api/config", strings.NewReader(content))
2021-06-18 15:48:36 +07:00
if err != nil {
return err
}
_, err = c.do(req)
return err
}
func (c *Client) setAuthHeader(req *http.Request) {
if c.authUser != "" || c.authPwd != "" {
2023-11-27 14:47:49 +07:00
req.Header.Set("Authorization", httppkg.BasicAuth(c.authUser, c.authPwd))
2021-06-18 15:48:36 +07:00
}
}
func (c *Client) do(req *http.Request) (string, error) {
c.setAuthHeader(req)
resp, err := http.DefaultClient.Do(req)
if err != nil {
return "", err
}
defer resp.Body.Close()
if resp.StatusCode != 200 {
return "", fmt.Errorf("api status code [%d]", resp.StatusCode)
}
buf, err := io.ReadAll(resp.Body)
2021-06-18 15:48:36 +07:00
if err != nil {
return "", err
}
return string(buf), nil
}