2018-07-11 17:27:47 +02:00
|
|
|
package util
|
2017-12-16 16:59:46 +01:00
|
|
|
|
|
|
|
import (
|
2017-12-18 12:35:09 +01:00
|
|
|
"encoding/base64"
|
2018-01-18 07:53:44 +01:00
|
|
|
"encoding/json"
|
|
|
|
"errors"
|
2017-12-16 16:59:46 +01:00
|
|
|
"fmt"
|
2018-07-12 09:23:34 +02:00
|
|
|
"io"
|
2017-12-18 12:35:09 +01:00
|
|
|
"io/ioutil"
|
2017-12-16 16:59:46 +01:00
|
|
|
"net"
|
2017-12-18 12:35:09 +01:00
|
|
|
"net/http"
|
2018-01-23 10:11:59 +01:00
|
|
|
"net/url"
|
2018-01-18 07:53:44 +01:00
|
|
|
"strings"
|
2017-12-16 16:59:46 +01:00
|
|
|
"time"
|
|
|
|
|
2018-01-18 07:53:44 +01:00
|
|
|
"github.com/fatedier/frp/client"
|
2017-12-16 16:59:46 +01:00
|
|
|
)
|
|
|
|
|
2018-07-11 17:27:47 +02:00
|
|
|
func GetProxyStatus(statusAddr string, user string, passwd string, name string) (status *client.ProxyStatusResp, err error) {
|
|
|
|
req, err := http.NewRequest("GET", "http://"+statusAddr+"/api/status", nil)
|
2018-01-18 07:53:44 +01:00
|
|
|
if err != nil {
|
|
|
|
return status, err
|
|
|
|
}
|
|
|
|
|
2018-07-11 17:27:47 +02:00
|
|
|
authStr := "Basic " + base64.StdEncoding.EncodeToString([]byte(user+":"+passwd))
|
2018-01-18 07:53:44 +01:00
|
|
|
req.Header.Add("Authorization", authStr)
|
|
|
|
resp, err := http.DefaultClient.Do(req)
|
|
|
|
if err != nil {
|
|
|
|
return status, err
|
2019-05-30 06:11:59 +02:00
|
|
|
}
|
2019-05-31 09:56:05 +02:00
|
|
|
defer resp.Body.Close()
|
2019-05-30 06:11:59 +02:00
|
|
|
if resp.StatusCode != 200 {
|
|
|
|
return status, fmt.Errorf("admin api status code [%d]", resp.StatusCode)
|
|
|
|
}
|
|
|
|
body, err := ioutil.ReadAll(resp.Body)
|
|
|
|
if err != nil {
|
|
|
|
return status, err
|
|
|
|
}
|
|
|
|
allStatus := &client.StatusResp{}
|
|
|
|
err = json.Unmarshal(body, &allStatus)
|
|
|
|
if err != nil {
|
|
|
|
return status, fmt.Errorf("unmarshal http response error: %s", strings.TrimSpace(string(body)))
|
|
|
|
}
|
2020-05-24 11:48:37 +02:00
|
|
|
for _, s := range allStatus.TCP {
|
2019-05-30 06:11:59 +02:00
|
|
|
if s.Name == name {
|
|
|
|
return &s, nil
|
2018-01-18 07:53:44 +01:00
|
|
|
}
|
2019-05-30 06:11:59 +02:00
|
|
|
}
|
2020-05-24 11:48:37 +02:00
|
|
|
for _, s := range allStatus.UDP {
|
2019-05-30 06:11:59 +02:00
|
|
|
if s.Name == name {
|
|
|
|
return &s, nil
|
2018-01-18 07:53:44 +01:00
|
|
|
}
|
2019-05-30 06:11:59 +02:00
|
|
|
}
|
2020-05-24 11:48:37 +02:00
|
|
|
for _, s := range allStatus.HTTP {
|
2019-05-30 06:11:59 +02:00
|
|
|
if s.Name == name {
|
|
|
|
return &s, nil
|
2018-01-18 07:53:44 +01:00
|
|
|
}
|
2019-05-30 06:11:59 +02:00
|
|
|
}
|
2020-05-24 11:48:37 +02:00
|
|
|
for _, s := range allStatus.HTTPS {
|
2019-05-30 06:11:59 +02:00
|
|
|
if s.Name == name {
|
|
|
|
return &s, nil
|
2018-01-18 07:53:44 +01:00
|
|
|
}
|
2019-05-30 06:11:59 +02:00
|
|
|
}
|
2020-05-24 11:48:37 +02:00
|
|
|
for _, s := range allStatus.STCP {
|
2019-05-30 06:11:59 +02:00
|
|
|
if s.Name == name {
|
|
|
|
return &s, nil
|
2018-01-18 07:53:44 +01:00
|
|
|
}
|
2019-05-30 06:11:59 +02:00
|
|
|
}
|
2020-05-24 11:48:37 +02:00
|
|
|
for _, s := range allStatus.XTCP {
|
2019-05-30 06:11:59 +02:00
|
|
|
if s.Name == name {
|
|
|
|
return &s, nil
|
2018-01-18 07:53:44 +01:00
|
|
|
}
|
2020-04-22 15:37:45 +02:00
|
|
|
}
|
2020-05-24 11:48:37 +02:00
|
|
|
for _, s := range allStatus.SUDP {
|
2020-04-22 15:37:45 +02:00
|
|
|
if s.Name == name {
|
|
|
|
return &s, nil
|
|
|
|
}
|
2018-01-18 07:53:44 +01:00
|
|
|
}
|
2019-05-30 06:11:59 +02:00
|
|
|
|
2018-01-18 07:53:44 +01:00
|
|
|
return status, errors.New("no proxy status found")
|
|
|
|
}
|
|
|
|
|
2018-07-12 09:23:34 +02:00
|
|
|
func ReloadConf(reloadAddr string, user string, passwd string) error {
|
|
|
|
req, err := http.NewRequest("GET", "http://"+reloadAddr+"/api/reload", nil)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
authStr := "Basic " + base64.StdEncoding.EncodeToString([]byte(user+":"+passwd))
|
|
|
|
req.Header.Add("Authorization", authStr)
|
|
|
|
resp, err := http.DefaultClient.Do(req)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2019-05-30 06:11:59 +02:00
|
|
|
defer resp.Body.Close()
|
|
|
|
|
|
|
|
if resp.StatusCode != 200 {
|
|
|
|
return fmt.Errorf("admin api status code [%d]", resp.StatusCode)
|
|
|
|
}
|
|
|
|
io.Copy(ioutil.Discard, resp.Body)
|
2018-07-12 09:23:34 +02:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2020-05-24 11:48:37 +02:00
|
|
|
func SendTCPMsg(addr string, msg string) (res string, err error) {
|
2019-10-12 14:13:12 +02:00
|
|
|
c, err := net.Dial("tcp", addr)
|
2017-12-16 16:59:46 +01:00
|
|
|
if err != nil {
|
|
|
|
err = fmt.Errorf("connect to tcp server error: %v", err)
|
|
|
|
return
|
|
|
|
}
|
2017-12-18 12:35:09 +01:00
|
|
|
defer c.Close()
|
2020-05-24 11:48:37 +02:00
|
|
|
return SendTCPMsgByConn(c, msg)
|
2018-01-23 10:11:59 +01:00
|
|
|
}
|
2017-12-16 16:59:46 +01:00
|
|
|
|
2020-05-24 11:48:37 +02:00
|
|
|
func SendTCPMsgByConn(c net.Conn, msg string) (res string, err error) {
|
2017-12-16 16:59:46 +01:00
|
|
|
timer := time.Now().Add(5 * time.Second)
|
|
|
|
c.SetDeadline(timer)
|
|
|
|
c.Write([]byte(msg))
|
|
|
|
|
|
|
|
buf := make([]byte, 2048)
|
|
|
|
n, errRet := c.Read(buf)
|
|
|
|
if errRet != nil {
|
|
|
|
err = fmt.Errorf("read from tcp server error: %v", errRet)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
return string(buf[:n]), nil
|
|
|
|
}
|
|
|
|
|
2020-05-24 11:48:37 +02:00
|
|
|
func SendUDPMsg(addr string, msg string) (res string, err error) {
|
2017-12-16 16:59:46 +01:00
|
|
|
udpAddr, errRet := net.ResolveUDPAddr("udp", addr)
|
|
|
|
if errRet != nil {
|
|
|
|
err = fmt.Errorf("resolve udp addr error: %v", err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
conn, errRet := net.DialUDP("udp", nil, udpAddr)
|
|
|
|
if errRet != nil {
|
|
|
|
err = fmt.Errorf("dial udp server error: %v", err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
defer conn.Close()
|
|
|
|
_, err = conn.Write([]byte(msg))
|
|
|
|
if err != nil {
|
|
|
|
err = fmt.Errorf("write to udp server error: %v", err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2020-05-12 16:09:16 +02:00
|
|
|
conn.SetReadDeadline(time.Now().Add(10 * time.Second))
|
2017-12-16 16:59:46 +01:00
|
|
|
buf := make([]byte, 2048)
|
|
|
|
n, errRet := conn.Read(buf)
|
|
|
|
if errRet != nil {
|
|
|
|
err = fmt.Errorf("read from udp server error: %v", err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
return string(buf[:n]), nil
|
|
|
|
}
|
2017-12-18 12:35:09 +01:00
|
|
|
|
2020-05-24 11:48:37 +02:00
|
|
|
func SendHTTPMsg(method, urlStr string, host string, headers map[string]string, proxy string) (code int, body string, header http.Header, err error) {
|
2018-01-23 10:11:59 +01:00
|
|
|
req, errRet := http.NewRequest(method, urlStr, nil)
|
2017-12-18 12:35:09 +01:00
|
|
|
if errRet != nil {
|
|
|
|
err = errRet
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
if host != "" {
|
|
|
|
req.Host = host
|
|
|
|
}
|
2018-05-20 17:55:22 +02:00
|
|
|
for k, v := range headers {
|
2017-12-18 12:35:09 +01:00
|
|
|
req.Header.Set(k, v)
|
|
|
|
}
|
2018-01-23 10:11:59 +01:00
|
|
|
|
|
|
|
tr := &http.Transport{
|
|
|
|
DialContext: (&net.Dialer{
|
|
|
|
Timeout: 30 * time.Second,
|
|
|
|
KeepAlive: 30 * time.Second,
|
|
|
|
DualStack: true,
|
|
|
|
}).DialContext,
|
|
|
|
MaxIdleConns: 100,
|
|
|
|
IdleConnTimeout: 90 * time.Second,
|
|
|
|
TLSHandshakeTimeout: 10 * time.Second,
|
|
|
|
ExpectContinueTimeout: 1 * time.Second,
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(proxy) != 0 {
|
|
|
|
tr.Proxy = func(req *http.Request) (*url.URL, error) {
|
|
|
|
return url.Parse(proxy)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
client := http.Client{
|
|
|
|
Transport: tr,
|
|
|
|
}
|
|
|
|
|
|
|
|
resp, errRet := client.Do(req)
|
2017-12-18 12:35:09 +01:00
|
|
|
if errRet != nil {
|
|
|
|
err = errRet
|
|
|
|
return
|
|
|
|
}
|
|
|
|
code = resp.StatusCode
|
2018-05-20 17:55:22 +02:00
|
|
|
header = resp.Header
|
2017-12-18 12:35:09 +01:00
|
|
|
buf, errRet := ioutil.ReadAll(resp.Body)
|
|
|
|
if errRet != nil {
|
|
|
|
err = errRet
|
|
|
|
return
|
|
|
|
}
|
|
|
|
body = string(buf)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2018-07-11 17:27:47 +02:00
|
|
|
func BasicAuth(username, passwd string) string {
|
2017-12-18 12:35:09 +01:00
|
|
|
auth := username + ":" + passwd
|
|
|
|
return "Basic " + base64.StdEncoding.EncodeToString([]byte(auth))
|
|
|
|
}
|