mirror of
https://github.com/SagerNet/sing.git
synced 2025-04-04 20:37:40 +03:00
287 lines
6.7 KiB
Go
287 lines
6.7 KiB
Go
package main
|
|
|
|
import (
|
|
"context"
|
|
"encoding/json"
|
|
|
|
"github.com/go-resty/resty/v2"
|
|
E "github.com/sagernet/sing/common/exceptions"
|
|
)
|
|
|
|
type NodeClient struct {
|
|
client *resty.Client
|
|
}
|
|
|
|
func NewNodeClient(baseURL string, token string, node string) *NodeClient {
|
|
r := resty.New()
|
|
r.SetBaseURL(baseURL)
|
|
r.SetQueryParams(map[string]string{
|
|
"node_id": node,
|
|
"token": token,
|
|
})
|
|
// r.SetDebug(true)
|
|
return &NodeClient{r}
|
|
}
|
|
|
|
type ShadowsocksUserList struct {
|
|
Port uint16
|
|
Method string
|
|
Users map[int]string // id password
|
|
}
|
|
|
|
type RawShadowsocksUserList struct {
|
|
Data []RawShadowsocksUser `json:"data"`
|
|
}
|
|
|
|
type RawShadowsocksUser struct {
|
|
Id int `json:"id"`
|
|
Port uint16 `json:"port"`
|
|
Cipher string `json:"cipher"`
|
|
Secret string `json:"secret"`
|
|
}
|
|
|
|
func (c *NodeClient) GetShadowsocksUserList(ctx context.Context) (*ShadowsocksUserList, error) {
|
|
resp, err := c.client.R().
|
|
SetContext(ctx).
|
|
SetResult(new(RawShadowsocksUserList)).
|
|
Get("/api/v1/server/ShadowsocksTidalab/user")
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
if !resp.IsSuccess() {
|
|
return nil, E.New("HTTP ", resp.StatusCode(), " ", resp.Body())
|
|
}
|
|
|
|
rawUserList := resp.Result().(*RawShadowsocksUserList)
|
|
|
|
userList := &ShadowsocksUserList{
|
|
Method: rawUserList.Data[0].Cipher,
|
|
Port: rawUserList.Data[0].Port,
|
|
Users: make(map[int]string),
|
|
}
|
|
|
|
for _, item := range rawUserList.Data {
|
|
if item.Cipher != userList.Method {
|
|
return nil, E.New("not unique method in item ", item.Id)
|
|
}
|
|
if item.Port != userList.Port {
|
|
return nil, E.New("not unique port in item ", item.Id)
|
|
}
|
|
userList.Users[item.Id] = item.Secret
|
|
}
|
|
|
|
return userList, nil
|
|
}
|
|
|
|
type TrojanUserList struct {
|
|
Users map[int]string // id password
|
|
}
|
|
|
|
type RawTrojanUserList struct {
|
|
Msg string `json:"msg"`
|
|
Data []RawTrojanUser `json:"data"`
|
|
}
|
|
|
|
type RawTrojanUser struct {
|
|
ID int `json:"id"`
|
|
T int `json:"t"`
|
|
U int64 `json:"u"`
|
|
D int64 `json:"d"`
|
|
TransferEnable int64 `json:"transfer_enable"`
|
|
TrojanUser struct {
|
|
Password string `json:"password"`
|
|
} `json:"trojan_user"`
|
|
}
|
|
|
|
func (c *NodeClient) GetTrojanUserList(ctx context.Context) (*TrojanUserList, error) {
|
|
resp, err := c.client.R().
|
|
SetContext(ctx).
|
|
SetResult(new(RawTrojanUserList)).
|
|
Get("/api/v1/server/TrojanTidalab/user")
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
if !resp.IsSuccess() {
|
|
return nil, E.New("HTTP ", resp.StatusCode(), " ", resp.String())
|
|
}
|
|
|
|
rawUserList := resp.Result().(*RawTrojanUserList)
|
|
|
|
userList := &TrojanUserList{
|
|
Users: make(map[int]string),
|
|
}
|
|
|
|
for _, item := range rawUserList.Data {
|
|
userList.Users[item.ID] = item.TrojanUser.Password
|
|
}
|
|
|
|
return userList, nil
|
|
}
|
|
|
|
type TrojanConfig struct {
|
|
LocalPort uint16
|
|
SNI string
|
|
}
|
|
|
|
type RawTrojanConfig struct {
|
|
LocalPort uint16 `json:"local_port"`
|
|
Ssl struct {
|
|
Sni string `json:"sni"`
|
|
} `json:"ssl"`
|
|
}
|
|
|
|
func (c *NodeClient) GetTrojanConfig(ctx context.Context) (*TrojanConfig, error) {
|
|
resp, err := c.client.R().
|
|
SetContext(ctx).
|
|
SetQueryParam("local_port", "1").
|
|
Get("/api/v1/server/TrojanTidalab/config")
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
if !resp.IsSuccess() {
|
|
return nil, E.New("HTTP ", resp.StatusCode(), " ", resp.String())
|
|
}
|
|
|
|
rawConfig := new(RawTrojanConfig)
|
|
err = json.Unmarshal(resp.Body(), rawConfig)
|
|
if err != nil {
|
|
return nil, E.Cause(err, "parse raw trojan config")
|
|
}
|
|
|
|
trojanConfig := new(TrojanConfig)
|
|
trojanConfig.LocalPort = rawConfig.LocalPort
|
|
trojanConfig.SNI = rawConfig.Ssl.Sni
|
|
|
|
return trojanConfig, nil
|
|
}
|
|
|
|
type VMessUserList struct {
|
|
AlterID int
|
|
Users map[int]string // id uuid
|
|
}
|
|
|
|
type RawVMessUserList struct {
|
|
Msg string `json:"msg"`
|
|
Data []RawVMessUser `json:"data"`
|
|
}
|
|
|
|
type RawVMessUser struct {
|
|
Id int `json:"id"`
|
|
T int `json:"t"`
|
|
U int64 `json:"u"`
|
|
D int64 `json:"d"`
|
|
TransferEnable int64 `json:"transfer_enable"`
|
|
V2RayUser struct {
|
|
Uuid string `json:"uuid"`
|
|
Email string `json:"email"`
|
|
AlterId int `json:"alter_id"`
|
|
Level int `json:"level"`
|
|
} `json:"v2ray_user"`
|
|
}
|
|
|
|
func (c *NodeClient) GetVMessUserList(ctx context.Context) (*VMessUserList, error) {
|
|
resp, err := c.client.R().
|
|
SetContext(ctx).
|
|
SetResult(new(RawVMessUserList)).
|
|
Get("/api/v1/server/Deepbwork/user")
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
if !resp.IsSuccess() {
|
|
return nil, E.New("HTTP ", resp.StatusCode(), " ", resp.String())
|
|
}
|
|
|
|
rawUserList := resp.Result().(*RawVMessUserList)
|
|
|
|
userList := &VMessUserList{
|
|
AlterID: rawUserList.Data[0].V2RayUser.AlterId,
|
|
Users: make(map[int]string),
|
|
}
|
|
|
|
for _, user := range rawUserList.Data {
|
|
userList.Users[user.Id] = user.V2RayUser.Uuid
|
|
}
|
|
|
|
return userList, nil
|
|
}
|
|
|
|
type VMessConfig struct {
|
|
Port uint16
|
|
Network string
|
|
Security string
|
|
}
|
|
|
|
type RawV2RayConfig struct {
|
|
Inbounds []struct {
|
|
Protocol string `json:"protocol"`
|
|
Port uint16 `json:"port"`
|
|
StreamSettings struct {
|
|
Network string `json:"network"`
|
|
Security string `json:"security,omitempty"`
|
|
} `json:"streamSettings,omitempty"`
|
|
} `json:"inbounds"`
|
|
}
|
|
|
|
func (c *NodeClient) GetVMessConfig(ctx context.Context) (*VMessConfig, error) {
|
|
resp, err := c.client.R().
|
|
SetContext(ctx).
|
|
SetQueryParam("local_port", "1").
|
|
Get("/api/v1/server/Deepbwork/config")
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
if !resp.IsSuccess() {
|
|
return nil, E.New("HTTP ", resp.StatusCode(), " ", resp.String())
|
|
}
|
|
|
|
rawConfig := new(RawV2RayConfig)
|
|
err = json.Unmarshal(resp.Body(), rawConfig)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
vmessConfig := new(VMessConfig)
|
|
vmessConfig.Port = rawConfig.Inbounds[0].Port
|
|
vmessConfig.Network = rawConfig.Inbounds[0].StreamSettings.Network
|
|
vmessConfig.Security = rawConfig.Inbounds[0].StreamSettings.Security
|
|
|
|
return vmessConfig, nil
|
|
}
|
|
|
|
type UserTraffic struct {
|
|
UID int `json:"user_id"`
|
|
Upload int64 `json:"u"`
|
|
Download int64 `json:"d"`
|
|
}
|
|
|
|
func (c *NodeClient) ReportShadowsocksTraffic(ctx context.Context, userTraffic []UserTraffic) error {
|
|
return c.reportTraffic(ctx, "/api/v1/server/ShadowsocksTidalab/submit", userTraffic)
|
|
}
|
|
|
|
func (c *NodeClient) ReportVMessTraffic(ctx context.Context, userTraffic []UserTraffic) error {
|
|
return c.reportTraffic(ctx, "/api/v1/server/Deepbwork/submit", userTraffic)
|
|
}
|
|
|
|
func (c *NodeClient) ReportTrojanTraffic(ctx context.Context, userTraffic []UserTraffic) error {
|
|
return c.reportTraffic(ctx, "/api/v1/server/TrojanTidalab/submit", userTraffic)
|
|
}
|
|
|
|
func (c *NodeClient) reportTraffic(ctx context.Context, path string, userTraffic []UserTraffic) error {
|
|
resp, err := c.client.R().
|
|
SetContext(ctx).
|
|
SetBody(userTraffic).
|
|
Post(path)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
if !resp.IsSuccess() {
|
|
return E.New("HTTP ", resp.StatusCode(), " ", resp.String())
|
|
}
|
|
return nil
|
|
}
|