2023-05-22 17:36:34 +03:00
|
|
|
|
package sub
|
2023-04-09 22:43:18 +03:00
|
|
|
|
|
|
|
|
|
import (
|
|
|
|
|
"encoding/base64"
|
|
|
|
|
"fmt"
|
|
|
|
|
"net/url"
|
|
|
|
|
"strings"
|
2023-07-18 02:49:01 +03:00
|
|
|
|
"time"
|
2024-03-11 00:31:24 +03:00
|
|
|
|
|
2023-04-09 22:43:18 +03:00
|
|
|
|
"x-ui/database"
|
|
|
|
|
"x-ui/database/model"
|
|
|
|
|
"x-ui/logger"
|
2023-07-18 02:49:01 +03:00
|
|
|
|
"x-ui/util/common"
|
2024-03-11 13:04:15 +03:00
|
|
|
|
"x-ui/util/random"
|
2023-05-22 17:36:34 +03:00
|
|
|
|
"x-ui/web/service"
|
2023-04-18 21:04:06 +03:00
|
|
|
|
"x-ui/xray"
|
2023-07-18 02:49:01 +03:00
|
|
|
|
|
2023-04-09 22:43:18 +03:00
|
|
|
|
"github.com/goccy/go-json"
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
type SubService struct {
|
|
|
|
|
address string
|
2023-08-26 14:41:12 +03:00
|
|
|
|
showInfo bool
|
2023-12-08 22:31:17 +03:00
|
|
|
|
remarkModel string
|
2024-01-02 11:32:21 +03:00
|
|
|
|
datepicker string
|
2023-05-22 17:36:34 +03:00
|
|
|
|
inboundService service.InboundService
|
2023-12-06 01:09:08 +03:00
|
|
|
|
settingService service.SettingService
|
2023-04-09 22:43:18 +03:00
|
|
|
|
}
|
|
|
|
|
|
2024-02-21 13:47:52 +03:00
|
|
|
|
func NewSubService(showInfo bool, remarkModel string) *SubService {
|
|
|
|
|
return &SubService{
|
|
|
|
|
showInfo: showInfo,
|
|
|
|
|
remarkModel: remarkModel,
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func (s *SubService) GetSubs(subId string, host string) ([]string, string, error) {
|
2023-04-09 22:43:18 +03:00
|
|
|
|
s.address = host
|
|
|
|
|
var result []string
|
2024-02-21 13:47:52 +03:00
|
|
|
|
var header string
|
2023-04-18 21:04:06 +03:00
|
|
|
|
var traffic xray.ClientTraffic
|
|
|
|
|
var clientTraffics []xray.ClientTraffic
|
2023-04-09 22:43:18 +03:00
|
|
|
|
inbounds, err := s.getInboundsBySubId(subId)
|
|
|
|
|
if err != nil {
|
2024-02-21 13:47:52 +03:00
|
|
|
|
return nil, "", err
|
2023-04-09 22:43:18 +03:00
|
|
|
|
}
|
2024-02-21 13:47:52 +03:00
|
|
|
|
|
2024-03-21 09:51:12 +03:00
|
|
|
|
if len(inbounds) == 0 {
|
|
|
|
|
return nil, "", common.NewError("No inbounds found with ", subId)
|
|
|
|
|
}
|
|
|
|
|
|
2024-02-21 13:47:52 +03:00
|
|
|
|
s.datepicker, err = s.settingService.GetDatepicker()
|
2023-12-08 22:31:17 +03:00
|
|
|
|
if err != nil {
|
2024-02-21 13:47:52 +03:00
|
|
|
|
s.datepicker = "gregorian"
|
2023-12-08 22:31:17 +03:00
|
|
|
|
}
|
2023-04-09 22:43:18 +03:00
|
|
|
|
for _, inbound := range inbounds {
|
2023-05-22 17:36:34 +03:00
|
|
|
|
clients, err := s.inboundService.GetClients(inbound)
|
2023-04-09 22:43:18 +03:00
|
|
|
|
if err != nil {
|
2024-02-21 13:47:52 +03:00
|
|
|
|
logger.Error("SubService - GetClients: Unable to get clients from inbound")
|
2023-04-09 22:43:18 +03:00
|
|
|
|
}
|
|
|
|
|
if clients == nil {
|
|
|
|
|
continue
|
|
|
|
|
}
|
2023-05-23 02:45:34 +03:00
|
|
|
|
if len(inbound.Listen) > 0 && inbound.Listen[0] == '@' {
|
2024-02-21 13:47:52 +03:00
|
|
|
|
listen, port, streamSettings, err := s.getFallbackMaster(inbound.Listen, inbound.StreamSettings)
|
2023-05-23 02:45:34 +03:00
|
|
|
|
if err == nil {
|
2024-02-21 13:47:52 +03:00
|
|
|
|
inbound.Listen = listen
|
|
|
|
|
inbound.Port = port
|
|
|
|
|
inbound.StreamSettings = streamSettings
|
2023-05-23 02:45:34 +03:00
|
|
|
|
}
|
|
|
|
|
}
|
2023-04-09 22:43:18 +03:00
|
|
|
|
for _, client := range clients {
|
2023-05-12 17:59:02 +03:00
|
|
|
|
if client.Enable && client.SubID == subId {
|
2023-08-26 14:41:12 +03:00
|
|
|
|
link := s.getLink(inbound, client.Email)
|
2023-04-09 22:43:18 +03:00
|
|
|
|
result = append(result, link)
|
2023-04-18 21:04:06 +03:00
|
|
|
|
clientTraffics = append(clientTraffics, s.getClientTraffics(inbound.ClientStats, client.Email))
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
2024-02-21 13:47:52 +03:00
|
|
|
|
|
|
|
|
|
// Prepare statistics
|
2023-04-18 21:04:06 +03:00
|
|
|
|
for index, clientTraffic := range clientTraffics {
|
|
|
|
|
if index == 0 {
|
|
|
|
|
traffic.Up = clientTraffic.Up
|
|
|
|
|
traffic.Down = clientTraffic.Down
|
|
|
|
|
traffic.Total = clientTraffic.Total
|
|
|
|
|
if clientTraffic.ExpiryTime > 0 {
|
|
|
|
|
traffic.ExpiryTime = clientTraffic.ExpiryTime
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
traffic.Up += clientTraffic.Up
|
|
|
|
|
traffic.Down += clientTraffic.Down
|
|
|
|
|
if traffic.Total == 0 || clientTraffic.Total == 0 {
|
|
|
|
|
traffic.Total = 0
|
|
|
|
|
} else {
|
|
|
|
|
traffic.Total += clientTraffic.Total
|
|
|
|
|
}
|
|
|
|
|
if clientTraffic.ExpiryTime != traffic.ExpiryTime {
|
|
|
|
|
traffic.ExpiryTime = 0
|
2023-04-09 22:43:18 +03:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
2024-02-21 13:47:52 +03:00
|
|
|
|
header = fmt.Sprintf("upload=%d; download=%d; total=%d; expire=%d", traffic.Up, traffic.Down, traffic.Total, traffic.ExpiryTime/1000)
|
|
|
|
|
return result, header, nil
|
2023-04-09 22:43:18 +03:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func (s *SubService) getInboundsBySubId(subId string) ([]*model.Inbound, error) {
|
|
|
|
|
db := database.GetDB()
|
|
|
|
|
var inbounds []*model.Inbound
|
2023-12-08 20:45:21 +03:00
|
|
|
|
err := db.Model(model.Inbound{}).Preload("ClientStats").Where(`id in (
|
|
|
|
|
SELECT DISTINCT inbounds.id
|
|
|
|
|
FROM inbounds,
|
|
|
|
|
JSON_EACH(JSON_EXTRACT(inbounds.settings, '$.clients')) AS client
|
|
|
|
|
WHERE
|
|
|
|
|
protocol in ('vmess','vless','trojan','shadowsocks')
|
|
|
|
|
AND JSON_EXTRACT(client.value, '$.subId') = ? AND enable = ?
|
|
|
|
|
)`, subId, true).Find(&inbounds).Error
|
2023-05-17 00:37:35 +03:00
|
|
|
|
if err != nil {
|
2023-04-09 22:43:18 +03:00
|
|
|
|
return nil, err
|
|
|
|
|
}
|
|
|
|
|
return inbounds, nil
|
|
|
|
|
}
|
|
|
|
|
|
2023-04-18 21:04:06 +03:00
|
|
|
|
func (s *SubService) getClientTraffics(traffics []xray.ClientTraffic, email string) xray.ClientTraffic {
|
|
|
|
|
for _, traffic := range traffics {
|
|
|
|
|
if traffic.Email == email {
|
|
|
|
|
return traffic
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return xray.ClientTraffic{}
|
|
|
|
|
}
|
|
|
|
|
|
2024-02-21 13:47:52 +03:00
|
|
|
|
func (s *SubService) getFallbackMaster(dest string, streamSettings string) (string, int, string, error) {
|
2023-05-23 02:45:34 +03:00
|
|
|
|
db := database.GetDB()
|
|
|
|
|
var inbound *model.Inbound
|
|
|
|
|
err := db.Model(model.Inbound{}).
|
|
|
|
|
Where("JSON_TYPE(settings, '$.fallbacks') = 'array'").
|
|
|
|
|
Where("EXISTS (SELECT * FROM json_each(settings, '$.fallbacks') WHERE json_extract(value, '$.dest') = ?)", dest).
|
|
|
|
|
Find(&inbound).Error
|
|
|
|
|
if err != nil {
|
2024-02-21 13:47:52 +03:00
|
|
|
|
return "", 0, "", err
|
2023-05-23 02:45:34 +03:00
|
|
|
|
}
|
2024-02-21 13:47:52 +03:00
|
|
|
|
|
|
|
|
|
var stream map[string]interface{}
|
|
|
|
|
json.Unmarshal([]byte(streamSettings), &stream)
|
|
|
|
|
var masterStream map[string]interface{}
|
|
|
|
|
json.Unmarshal([]byte(inbound.StreamSettings), &masterStream)
|
|
|
|
|
stream["security"] = masterStream["security"]
|
|
|
|
|
stream["tlsSettings"] = masterStream["tlsSettings"]
|
|
|
|
|
stream["externalProxy"] = masterStream["externalProxy"]
|
|
|
|
|
modifiedStream, _ := json.MarshalIndent(stream, "", " ")
|
|
|
|
|
|
|
|
|
|
return inbound.Listen, inbound.Port, string(modifiedStream), nil
|
2023-05-23 02:45:34 +03:00
|
|
|
|
}
|
|
|
|
|
|
2023-08-26 14:41:12 +03:00
|
|
|
|
func (s *SubService) getLink(inbound *model.Inbound, email string) string {
|
2023-04-09 22:43:18 +03:00
|
|
|
|
switch inbound.Protocol {
|
|
|
|
|
case "vmess":
|
2023-08-26 14:41:12 +03:00
|
|
|
|
return s.genVmessLink(inbound, email)
|
2023-04-09 22:43:18 +03:00
|
|
|
|
case "vless":
|
2023-08-26 14:41:12 +03:00
|
|
|
|
return s.genVlessLink(inbound, email)
|
2023-04-09 22:43:18 +03:00
|
|
|
|
case "trojan":
|
2023-08-26 14:41:12 +03:00
|
|
|
|
return s.genTrojanLink(inbound, email)
|
2023-05-06 19:51:14 +03:00
|
|
|
|
case "shadowsocks":
|
2023-08-26 14:41:12 +03:00
|
|
|
|
return s.genShadowsocksLink(inbound, email)
|
2023-04-09 22:43:18 +03:00
|
|
|
|
}
|
|
|
|
|
return ""
|
|
|
|
|
}
|
|
|
|
|
|
2023-08-26 14:41:12 +03:00
|
|
|
|
func (s *SubService) genVmessLink(inbound *model.Inbound, email string) string {
|
2024-08-11 01:47:44 +03:00
|
|
|
|
if inbound.Protocol != model.VMESS {
|
2023-04-09 22:43:18 +03:00
|
|
|
|
return ""
|
|
|
|
|
}
|
2023-04-27 23:45:06 +03:00
|
|
|
|
obj := map[string]interface{}{
|
|
|
|
|
"v": "2",
|
|
|
|
|
"add": s.address,
|
|
|
|
|
"port": inbound.Port,
|
|
|
|
|
"type": "none",
|
|
|
|
|
}
|
2023-04-09 22:43:18 +03:00
|
|
|
|
var stream map[string]interface{}
|
|
|
|
|
json.Unmarshal([]byte(inbound.StreamSettings), &stream)
|
|
|
|
|
network, _ := stream["network"].(string)
|
2023-04-27 23:45:06 +03:00
|
|
|
|
obj["net"] = network
|
2023-04-09 22:43:18 +03:00
|
|
|
|
switch network {
|
|
|
|
|
case "tcp":
|
|
|
|
|
tcp, _ := stream["tcpSettings"].(map[string]interface{})
|
|
|
|
|
header, _ := tcp["header"].(map[string]interface{})
|
2023-04-27 23:45:06 +03:00
|
|
|
|
typeStr, _ := header["type"].(string)
|
|
|
|
|
obj["type"] = typeStr
|
2023-04-09 22:43:18 +03:00
|
|
|
|
if typeStr == "http" {
|
|
|
|
|
request := header["request"].(map[string]interface{})
|
|
|
|
|
requestPath, _ := request["path"].([]interface{})
|
2023-04-27 23:45:06 +03:00
|
|
|
|
obj["path"] = requestPath[0].(string)
|
2023-04-09 22:43:18 +03:00
|
|
|
|
headers, _ := request["headers"].(map[string]interface{})
|
2023-04-27 23:45:06 +03:00
|
|
|
|
obj["host"] = searchHost(headers)
|
2023-04-09 22:43:18 +03:00
|
|
|
|
}
|
|
|
|
|
case "kcp":
|
|
|
|
|
kcp, _ := stream["kcpSettings"].(map[string]interface{})
|
|
|
|
|
header, _ := kcp["header"].(map[string]interface{})
|
2023-04-27 23:45:06 +03:00
|
|
|
|
obj["type"], _ = header["type"].(string)
|
|
|
|
|
obj["path"], _ = kcp["seed"].(string)
|
2023-04-09 22:43:18 +03:00
|
|
|
|
case "ws":
|
|
|
|
|
ws, _ := stream["wsSettings"].(map[string]interface{})
|
2023-04-27 23:45:06 +03:00
|
|
|
|
obj["path"] = ws["path"].(string)
|
2024-06-18 13:49:20 +03:00
|
|
|
|
if host, ok := ws["host"].(string); ok && len(host) > 0 {
|
|
|
|
|
obj["host"] = host
|
|
|
|
|
} else {
|
|
|
|
|
headers, _ := ws["headers"].(map[string]interface{})
|
|
|
|
|
obj["host"] = searchHost(headers)
|
2024-04-01 15:32:02 +03:00
|
|
|
|
}
|
2023-04-09 22:43:18 +03:00
|
|
|
|
case "http":
|
2023-04-27 23:45:06 +03:00
|
|
|
|
obj["net"] = "h2"
|
2023-04-09 22:43:18 +03:00
|
|
|
|
http, _ := stream["httpSettings"].(map[string]interface{})
|
2023-04-27 23:45:06 +03:00
|
|
|
|
obj["path"], _ = http["path"].(string)
|
|
|
|
|
obj["host"] = searchHost(http)
|
2023-04-09 22:43:18 +03:00
|
|
|
|
case "grpc":
|
|
|
|
|
grpc, _ := stream["grpcSettings"].(map[string]interface{})
|
2023-04-27 23:45:06 +03:00
|
|
|
|
obj["path"] = grpc["serviceName"].(string)
|
2024-03-11 12:30:00 +03:00
|
|
|
|
obj["authority"] = grpc["authority"].(string)
|
2023-04-27 23:45:06 +03:00
|
|
|
|
if grpc["multiMode"].(bool) {
|
|
|
|
|
obj["type"] = "multi"
|
|
|
|
|
}
|
2024-03-11 10:36:33 +03:00
|
|
|
|
case "httpupgrade":
|
|
|
|
|
httpupgrade, _ := stream["httpupgradeSettings"].(map[string]interface{})
|
|
|
|
|
obj["path"] = httpupgrade["path"].(string)
|
2024-06-18 13:49:20 +03:00
|
|
|
|
if host, ok := httpupgrade["host"].(string); ok && len(host) > 0 {
|
|
|
|
|
obj["host"] = host
|
|
|
|
|
} else {
|
|
|
|
|
headers, _ := httpupgrade["headers"].(map[string]interface{})
|
|
|
|
|
obj["host"] = searchHost(headers)
|
|
|
|
|
}
|
|
|
|
|
case "splithttp":
|
|
|
|
|
splithttp, _ := stream["splithttpSettings"].(map[string]interface{})
|
|
|
|
|
obj["path"] = splithttp["path"].(string)
|
|
|
|
|
if host, ok := splithttp["host"].(string); ok && len(host) > 0 {
|
|
|
|
|
obj["host"] = host
|
|
|
|
|
} else {
|
|
|
|
|
headers, _ := splithttp["headers"].(map[string]interface{})
|
|
|
|
|
obj["host"] = searchHost(headers)
|
2024-04-02 23:11:06 +03:00
|
|
|
|
}
|
2023-04-09 22:43:18 +03:00
|
|
|
|
}
|
|
|
|
|
security, _ := stream["security"].(string)
|
2023-04-27 23:45:06 +03:00
|
|
|
|
obj["tls"] = security
|
2023-04-09 22:43:18 +03:00
|
|
|
|
if security == "tls" {
|
|
|
|
|
tlsSetting, _ := stream["tlsSettings"].(map[string]interface{})
|
|
|
|
|
alpns, _ := tlsSetting["alpn"].([]interface{})
|
2023-04-27 23:45:06 +03:00
|
|
|
|
if len(alpns) > 0 {
|
|
|
|
|
var alpn []string
|
|
|
|
|
for _, a := range alpns {
|
|
|
|
|
alpn = append(alpn, a.(string))
|
|
|
|
|
}
|
|
|
|
|
obj["alpn"] = strings.Join(alpn, ",")
|
2023-04-09 22:43:18 +03:00
|
|
|
|
}
|
2023-12-08 20:45:21 +03:00
|
|
|
|
if sniValue, ok := searchKey(tlsSetting, "serverName"); ok {
|
|
|
|
|
obj["sni"], _ = sniValue.(string)
|
|
|
|
|
}
|
|
|
|
|
|
2023-04-09 22:43:18 +03:00
|
|
|
|
tlsSettings, _ := searchKey(tlsSetting, "settings")
|
|
|
|
|
if tlsSetting != nil {
|
|
|
|
|
if fpValue, ok := searchKey(tlsSettings, "fingerprint"); ok {
|
2023-04-27 23:45:06 +03:00
|
|
|
|
obj["fp"], _ = fpValue.(string)
|
2023-04-09 22:43:18 +03:00
|
|
|
|
}
|
|
|
|
|
if insecure, ok := searchKey(tlsSettings, "allowInsecure"); ok {
|
2023-04-27 23:45:06 +03:00
|
|
|
|
obj["allowInsecure"], _ = insecure.(bool)
|
2023-04-09 22:43:18 +03:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2023-05-22 17:36:34 +03:00
|
|
|
|
clients, _ := s.inboundService.GetClients(inbound)
|
2023-04-09 22:43:18 +03:00
|
|
|
|
clientIndex := -1
|
|
|
|
|
for i, client := range clients {
|
|
|
|
|
if client.Email == email {
|
|
|
|
|
clientIndex = i
|
|
|
|
|
break
|
|
|
|
|
}
|
|
|
|
|
}
|
2023-04-27 23:45:06 +03:00
|
|
|
|
obj["id"] = clients[clientIndex].ID
|
2024-08-11 01:47:44 +03:00
|
|
|
|
obj["scy"] = clients[clientIndex].Security
|
2023-04-09 22:43:18 +03:00
|
|
|
|
|
2023-12-08 20:45:21 +03:00
|
|
|
|
externalProxies, _ := stream["externalProxy"].([]interface{})
|
|
|
|
|
|
|
|
|
|
if len(externalProxies) > 0 {
|
2023-05-22 17:36:34 +03:00
|
|
|
|
links := ""
|
2023-12-08 20:45:21 +03:00
|
|
|
|
for index, externalProxy := range externalProxies {
|
|
|
|
|
ep, _ := externalProxy.(map[string]interface{})
|
|
|
|
|
newSecurity, _ := ep["forceTls"].(string)
|
|
|
|
|
newObj := map[string]interface{}{}
|
|
|
|
|
for key, value := range obj {
|
|
|
|
|
if !(newSecurity == "none" && (key == "alpn" || key == "sni" || key == "fp" || key == "allowInsecure")) {
|
|
|
|
|
newObj[key] = value
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
newObj["ps"] = s.genRemark(inbound, email, ep["remark"].(string))
|
|
|
|
|
newObj["add"] = ep["dest"].(string)
|
|
|
|
|
newObj["port"] = int(ep["port"].(float64))
|
|
|
|
|
|
|
|
|
|
if newSecurity != "same" {
|
|
|
|
|
newObj["tls"] = newSecurity
|
|
|
|
|
}
|
2023-05-22 17:36:34 +03:00
|
|
|
|
if index > 0 {
|
|
|
|
|
links += "\n"
|
|
|
|
|
}
|
2023-12-08 20:45:21 +03:00
|
|
|
|
jsonStr, _ := json.MarshalIndent(newObj, "", " ")
|
2023-05-22 17:36:34 +03:00
|
|
|
|
links += "vmess://" + base64.StdEncoding.EncodeToString(jsonStr)
|
|
|
|
|
}
|
|
|
|
|
return links
|
|
|
|
|
}
|
|
|
|
|
|
2023-08-26 14:41:12 +03:00
|
|
|
|
obj["ps"] = s.genRemark(inbound, email, "")
|
|
|
|
|
|
2023-04-09 22:43:18 +03:00
|
|
|
|
jsonStr, _ := json.MarshalIndent(obj, "", " ")
|
|
|
|
|
return "vmess://" + base64.StdEncoding.EncodeToString(jsonStr)
|
|
|
|
|
}
|
|
|
|
|
|
2023-08-26 14:41:12 +03:00
|
|
|
|
func (s *SubService) genVlessLink(inbound *model.Inbound, email string) string {
|
2023-04-09 22:43:18 +03:00
|
|
|
|
address := s.address
|
|
|
|
|
if inbound.Protocol != model.VLESS {
|
|
|
|
|
return ""
|
|
|
|
|
}
|
|
|
|
|
var stream map[string]interface{}
|
|
|
|
|
json.Unmarshal([]byte(inbound.StreamSettings), &stream)
|
2023-05-22 17:36:34 +03:00
|
|
|
|
clients, _ := s.inboundService.GetClients(inbound)
|
2023-04-09 22:43:18 +03:00
|
|
|
|
clientIndex := -1
|
|
|
|
|
for i, client := range clients {
|
|
|
|
|
if client.Email == email {
|
|
|
|
|
clientIndex = i
|
|
|
|
|
break
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
uuid := clients[clientIndex].ID
|
|
|
|
|
port := inbound.Port
|
|
|
|
|
streamNetwork := stream["network"].(string)
|
|
|
|
|
params := make(map[string]string)
|
|
|
|
|
params["type"] = streamNetwork
|
|
|
|
|
|
|
|
|
|
switch streamNetwork {
|
|
|
|
|
case "tcp":
|
|
|
|
|
tcp, _ := stream["tcpSettings"].(map[string]interface{})
|
|
|
|
|
header, _ := tcp["header"].(map[string]interface{})
|
|
|
|
|
typeStr, _ := header["type"].(string)
|
|
|
|
|
if typeStr == "http" {
|
|
|
|
|
request := header["request"].(map[string]interface{})
|
|
|
|
|
requestPath, _ := request["path"].([]interface{})
|
|
|
|
|
params["path"] = requestPath[0].(string)
|
|
|
|
|
headers, _ := request["headers"].(map[string]interface{})
|
|
|
|
|
params["host"] = searchHost(headers)
|
|
|
|
|
params["headerType"] = "http"
|
|
|
|
|
}
|
|
|
|
|
case "kcp":
|
|
|
|
|
kcp, _ := stream["kcpSettings"].(map[string]interface{})
|
|
|
|
|
header, _ := kcp["header"].(map[string]interface{})
|
|
|
|
|
params["headerType"] = header["type"].(string)
|
|
|
|
|
params["seed"] = kcp["seed"].(string)
|
|
|
|
|
case "ws":
|
|
|
|
|
ws, _ := stream["wsSettings"].(map[string]interface{})
|
|
|
|
|
params["path"] = ws["path"].(string)
|
2024-06-18 13:49:20 +03:00
|
|
|
|
if host, ok := ws["host"].(string); ok && len(host) > 0 {
|
|
|
|
|
params["host"] = host
|
|
|
|
|
} else {
|
|
|
|
|
headers, _ := ws["headers"].(map[string]interface{})
|
|
|
|
|
params["host"] = searchHost(headers)
|
2024-04-01 15:32:02 +03:00
|
|
|
|
}
|
2023-04-09 22:43:18 +03:00
|
|
|
|
case "http":
|
|
|
|
|
http, _ := stream["httpSettings"].(map[string]interface{})
|
|
|
|
|
params["path"] = http["path"].(string)
|
|
|
|
|
params["host"] = searchHost(http)
|
|
|
|
|
case "grpc":
|
|
|
|
|
grpc, _ := stream["grpcSettings"].(map[string]interface{})
|
|
|
|
|
params["serviceName"] = grpc["serviceName"].(string)
|
2024-04-01 10:39:45 +03:00
|
|
|
|
params["authority"], _ = grpc["authority"].(string)
|
2023-04-27 23:45:06 +03:00
|
|
|
|
if grpc["multiMode"].(bool) {
|
|
|
|
|
params["mode"] = "multi"
|
|
|
|
|
}
|
2024-03-11 10:36:33 +03:00
|
|
|
|
case "httpupgrade":
|
|
|
|
|
httpupgrade, _ := stream["httpupgradeSettings"].(map[string]interface{})
|
|
|
|
|
params["path"] = httpupgrade["path"].(string)
|
2024-06-18 13:49:20 +03:00
|
|
|
|
if host, ok := httpupgrade["host"].(string); ok && len(host) > 0 {
|
|
|
|
|
params["host"] = host
|
|
|
|
|
} else {
|
|
|
|
|
headers, _ := httpupgrade["headers"].(map[string]interface{})
|
|
|
|
|
params["host"] = searchHost(headers)
|
|
|
|
|
}
|
|
|
|
|
case "splithttp":
|
|
|
|
|
splithttp, _ := stream["splithttpSettings"].(map[string]interface{})
|
|
|
|
|
params["path"] = splithttp["path"].(string)
|
|
|
|
|
if host, ok := splithttp["host"].(string); ok && len(host) > 0 {
|
|
|
|
|
params["host"] = host
|
|
|
|
|
} else {
|
|
|
|
|
headers, _ := splithttp["headers"].(map[string]interface{})
|
|
|
|
|
params["host"] = searchHost(headers)
|
2024-04-02 23:11:06 +03:00
|
|
|
|
}
|
2023-04-09 22:43:18 +03:00
|
|
|
|
}
|
|
|
|
|
security, _ := stream["security"].(string)
|
|
|
|
|
if security == "tls" {
|
|
|
|
|
params["security"] = "tls"
|
|
|
|
|
tlsSetting, _ := stream["tlsSettings"].(map[string]interface{})
|
|
|
|
|
alpns, _ := tlsSetting["alpn"].([]interface{})
|
|
|
|
|
var alpn []string
|
|
|
|
|
for _, a := range alpns {
|
|
|
|
|
alpn = append(alpn, a.(string))
|
|
|
|
|
}
|
|
|
|
|
if len(alpn) > 0 {
|
|
|
|
|
params["alpn"] = strings.Join(alpn, ",")
|
|
|
|
|
}
|
2023-12-08 20:45:21 +03:00
|
|
|
|
if sniValue, ok := searchKey(tlsSetting, "serverName"); ok {
|
|
|
|
|
params["sni"], _ = sniValue.(string)
|
|
|
|
|
}
|
|
|
|
|
|
2023-04-09 22:43:18 +03:00
|
|
|
|
tlsSettings, _ := searchKey(tlsSetting, "settings")
|
|
|
|
|
if tlsSetting != nil {
|
|
|
|
|
if fpValue, ok := searchKey(tlsSettings, "fingerprint"); ok {
|
|
|
|
|
params["fp"], _ = fpValue.(string)
|
|
|
|
|
}
|
|
|
|
|
if insecure, ok := searchKey(tlsSettings, "allowInsecure"); ok {
|
|
|
|
|
if insecure.(bool) {
|
|
|
|
|
params["allowInsecure"] = "1"
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if streamNetwork == "tcp" && len(clients[clientIndex].Flow) > 0 {
|
|
|
|
|
params["flow"] = clients[clientIndex].Flow
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2023-04-11 15:10:45 +03:00
|
|
|
|
if security == "reality" {
|
|
|
|
|
params["security"] = "reality"
|
2023-04-19 11:25:31 +03:00
|
|
|
|
realitySetting, _ := stream["realitySettings"].(map[string]interface{})
|
|
|
|
|
realitySettings, _ := searchKey(realitySetting, "settings")
|
|
|
|
|
if realitySetting != nil {
|
|
|
|
|
if sniValue, ok := searchKey(realitySetting, "serverNames"); ok {
|
2023-04-12 11:44:07 +03:00
|
|
|
|
sNames, _ := sniValue.([]interface{})
|
2024-03-11 13:04:15 +03:00
|
|
|
|
params["sni"] = sNames[random.Num(len(sNames))].(string)
|
2023-04-11 15:10:45 +03:00
|
|
|
|
}
|
|
|
|
|
if pbkValue, ok := searchKey(realitySettings, "publicKey"); ok {
|
|
|
|
|
params["pbk"], _ = pbkValue.(string)
|
|
|
|
|
}
|
2023-04-19 11:25:31 +03:00
|
|
|
|
if sidValue, ok := searchKey(realitySetting, "shortIds"); ok {
|
2023-04-12 11:44:07 +03:00
|
|
|
|
shortIds, _ := sidValue.([]interface{})
|
2024-03-11 13:04:15 +03:00
|
|
|
|
params["sid"] = shortIds[random.Num(len(shortIds))].(string)
|
2023-04-11 15:10:45 +03:00
|
|
|
|
}
|
|
|
|
|
if fpValue, ok := searchKey(realitySettings, "fingerprint"); ok {
|
2023-04-19 11:25:31 +03:00
|
|
|
|
if fp, ok := fpValue.(string); ok && len(fp) > 0 {
|
|
|
|
|
params["fp"] = fp
|
|
|
|
|
}
|
|
|
|
|
}
|
2024-03-11 13:04:15 +03:00
|
|
|
|
params["spx"] = "/" + random.Seq(15)
|
2023-04-11 15:10:45 +03:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if streamNetwork == "tcp" && len(clients[clientIndex].Flow) > 0 {
|
|
|
|
|
params["flow"] = clients[clientIndex].Flow
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2024-10-29 14:50:25 +03:00
|
|
|
|
if security != "tls" && security != "reality" {
|
2023-06-14 16:36:56 +03:00
|
|
|
|
params["security"] = "none"
|
|
|
|
|
}
|
|
|
|
|
|
2023-12-08 20:45:21 +03:00
|
|
|
|
externalProxies, _ := stream["externalProxy"].([]interface{})
|
2023-04-09 22:43:18 +03:00
|
|
|
|
|
2023-12-08 20:45:21 +03:00
|
|
|
|
if len(externalProxies) > 0 {
|
|
|
|
|
links := ""
|
|
|
|
|
for index, externalProxy := range externalProxies {
|
|
|
|
|
ep, _ := externalProxy.(map[string]interface{})
|
|
|
|
|
newSecurity, _ := ep["forceTls"].(string)
|
|
|
|
|
dest, _ := ep["dest"].(string)
|
|
|
|
|
port := int(ep["port"].(float64))
|
|
|
|
|
link := fmt.Sprintf("vless://%s@%s:%d", uuid, dest, port)
|
|
|
|
|
|
|
|
|
|
if newSecurity != "same" {
|
|
|
|
|
params["security"] = newSecurity
|
|
|
|
|
} else {
|
|
|
|
|
params["security"] = security
|
|
|
|
|
}
|
|
|
|
|
url, _ := url.Parse(link)
|
|
|
|
|
q := url.Query()
|
2023-04-09 22:43:18 +03:00
|
|
|
|
|
2023-12-08 20:45:21 +03:00
|
|
|
|
for k, v := range params {
|
|
|
|
|
if !(newSecurity == "none" && (k == "alpn" || k == "sni" || k == "fp" || k == "allowInsecure")) {
|
|
|
|
|
q.Add(k, v)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Set the new query values on the URL
|
|
|
|
|
url.RawQuery = q.Encode()
|
|
|
|
|
|
|
|
|
|
url.Fragment = s.genRemark(inbound, email, ep["remark"].(string))
|
2023-04-09 22:43:18 +03:00
|
|
|
|
|
2023-05-22 17:36:34 +03:00
|
|
|
|
if index > 0 {
|
|
|
|
|
links += "\n"
|
|
|
|
|
}
|
|
|
|
|
links += url.String()
|
|
|
|
|
}
|
|
|
|
|
return links
|
|
|
|
|
}
|
2023-08-26 14:41:12 +03:00
|
|
|
|
|
2023-12-08 20:45:21 +03:00
|
|
|
|
link := fmt.Sprintf("vless://%s@%s:%d", uuid, address, port)
|
|
|
|
|
url, _ := url.Parse(link)
|
|
|
|
|
q := url.Query()
|
|
|
|
|
|
|
|
|
|
for k, v := range params {
|
|
|
|
|
q.Add(k, v)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Set the new query values on the URL
|
|
|
|
|
url.RawQuery = q.Encode()
|
|
|
|
|
|
2023-08-26 14:41:12 +03:00
|
|
|
|
url.Fragment = s.genRemark(inbound, email, "")
|
2023-04-09 22:43:18 +03:00
|
|
|
|
return url.String()
|
|
|
|
|
}
|
|
|
|
|
|
2023-08-26 14:41:12 +03:00
|
|
|
|
func (s *SubService) genTrojanLink(inbound *model.Inbound, email string) string {
|
2023-04-09 22:43:18 +03:00
|
|
|
|
address := s.address
|
|
|
|
|
if inbound.Protocol != model.Trojan {
|
|
|
|
|
return ""
|
|
|
|
|
}
|
|
|
|
|
var stream map[string]interface{}
|
|
|
|
|
json.Unmarshal([]byte(inbound.StreamSettings), &stream)
|
2023-05-22 17:36:34 +03:00
|
|
|
|
clients, _ := s.inboundService.GetClients(inbound)
|
2023-04-09 22:43:18 +03:00
|
|
|
|
clientIndex := -1
|
|
|
|
|
for i, client := range clients {
|
|
|
|
|
if client.Email == email {
|
|
|
|
|
clientIndex = i
|
|
|
|
|
break
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
password := clients[clientIndex].Password
|
|
|
|
|
port := inbound.Port
|
|
|
|
|
streamNetwork := stream["network"].(string)
|
|
|
|
|
params := make(map[string]string)
|
|
|
|
|
params["type"] = streamNetwork
|
|
|
|
|
|
|
|
|
|
switch streamNetwork {
|
|
|
|
|
case "tcp":
|
|
|
|
|
tcp, _ := stream["tcpSettings"].(map[string]interface{})
|
|
|
|
|
header, _ := tcp["header"].(map[string]interface{})
|
|
|
|
|
typeStr, _ := header["type"].(string)
|
|
|
|
|
if typeStr == "http" {
|
|
|
|
|
request := header["request"].(map[string]interface{})
|
|
|
|
|
requestPath, _ := request["path"].([]interface{})
|
|
|
|
|
params["path"] = requestPath[0].(string)
|
|
|
|
|
headers, _ := request["headers"].(map[string]interface{})
|
|
|
|
|
params["host"] = searchHost(headers)
|
|
|
|
|
params["headerType"] = "http"
|
|
|
|
|
}
|
|
|
|
|
case "kcp":
|
|
|
|
|
kcp, _ := stream["kcpSettings"].(map[string]interface{})
|
|
|
|
|
header, _ := kcp["header"].(map[string]interface{})
|
|
|
|
|
params["headerType"] = header["type"].(string)
|
|
|
|
|
params["seed"] = kcp["seed"].(string)
|
|
|
|
|
case "ws":
|
|
|
|
|
ws, _ := stream["wsSettings"].(map[string]interface{})
|
|
|
|
|
params["path"] = ws["path"].(string)
|
2024-06-18 13:49:20 +03:00
|
|
|
|
if host, ok := ws["host"].(string); ok && len(host) > 0 {
|
|
|
|
|
params["host"] = host
|
|
|
|
|
} else {
|
|
|
|
|
headers, _ := ws["headers"].(map[string]interface{})
|
|
|
|
|
params["host"] = searchHost(headers)
|
2024-04-01 15:32:02 +03:00
|
|
|
|
}
|
2023-04-09 22:43:18 +03:00
|
|
|
|
case "http":
|
|
|
|
|
http, _ := stream["httpSettings"].(map[string]interface{})
|
|
|
|
|
params["path"] = http["path"].(string)
|
|
|
|
|
params["host"] = searchHost(http)
|
|
|
|
|
case "grpc":
|
|
|
|
|
grpc, _ := stream["grpcSettings"].(map[string]interface{})
|
|
|
|
|
params["serviceName"] = grpc["serviceName"].(string)
|
2024-04-01 10:39:45 +03:00
|
|
|
|
params["authority"], _ = grpc["authority"].(string)
|
2023-04-27 23:45:06 +03:00
|
|
|
|
if grpc["multiMode"].(bool) {
|
|
|
|
|
params["mode"] = "multi"
|
|
|
|
|
}
|
2024-03-11 10:36:33 +03:00
|
|
|
|
case "httpupgrade":
|
|
|
|
|
httpupgrade, _ := stream["httpupgradeSettings"].(map[string]interface{})
|
|
|
|
|
params["path"] = httpupgrade["path"].(string)
|
2024-06-18 13:49:20 +03:00
|
|
|
|
if host, ok := httpupgrade["host"].(string); ok && len(host) > 0 {
|
|
|
|
|
params["host"] = host
|
|
|
|
|
} else {
|
|
|
|
|
headers, _ := httpupgrade["headers"].(map[string]interface{})
|
|
|
|
|
params["host"] = searchHost(headers)
|
|
|
|
|
}
|
|
|
|
|
case "splithttp":
|
|
|
|
|
splithttp, _ := stream["splithttpSettings"].(map[string]interface{})
|
|
|
|
|
params["path"] = splithttp["path"].(string)
|
|
|
|
|
if host, ok := splithttp["host"].(string); ok && len(host) > 0 {
|
|
|
|
|
params["host"] = host
|
|
|
|
|
} else {
|
|
|
|
|
headers, _ := splithttp["headers"].(map[string]interface{})
|
|
|
|
|
params["host"] = searchHost(headers)
|
2024-04-02 23:11:06 +03:00
|
|
|
|
}
|
2023-04-09 22:43:18 +03:00
|
|
|
|
}
|
|
|
|
|
security, _ := stream["security"].(string)
|
|
|
|
|
if security == "tls" {
|
|
|
|
|
params["security"] = "tls"
|
|
|
|
|
tlsSetting, _ := stream["tlsSettings"].(map[string]interface{})
|
|
|
|
|
alpns, _ := tlsSetting["alpn"].([]interface{})
|
|
|
|
|
var alpn []string
|
|
|
|
|
for _, a := range alpns {
|
|
|
|
|
alpn = append(alpn, a.(string))
|
|
|
|
|
}
|
|
|
|
|
if len(alpn) > 0 {
|
|
|
|
|
params["alpn"] = strings.Join(alpn, ",")
|
|
|
|
|
}
|
2023-12-08 20:45:21 +03:00
|
|
|
|
if sniValue, ok := searchKey(tlsSetting, "serverName"); ok {
|
|
|
|
|
params["sni"], _ = sniValue.(string)
|
|
|
|
|
}
|
2024-02-21 13:47:52 +03:00
|
|
|
|
|
2023-04-09 22:43:18 +03:00
|
|
|
|
tlsSettings, _ := searchKey(tlsSetting, "settings")
|
|
|
|
|
if tlsSetting != nil {
|
|
|
|
|
if fpValue, ok := searchKey(tlsSettings, "fingerprint"); ok {
|
|
|
|
|
params["fp"], _ = fpValue.(string)
|
|
|
|
|
}
|
|
|
|
|
if insecure, ok := searchKey(tlsSettings, "allowInsecure"); ok {
|
|
|
|
|
if insecure.(bool) {
|
|
|
|
|
params["allowInsecure"] = "1"
|
|
|
|
|
}
|
|
|
|
|
}
|
2023-04-11 22:00:24 +03:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if security == "reality" {
|
|
|
|
|
params["security"] = "reality"
|
2023-04-19 11:25:31 +03:00
|
|
|
|
realitySetting, _ := stream["realitySettings"].(map[string]interface{})
|
|
|
|
|
realitySettings, _ := searchKey(realitySetting, "settings")
|
|
|
|
|
if realitySetting != nil {
|
|
|
|
|
if sniValue, ok := searchKey(realitySetting, "serverNames"); ok {
|
2023-04-12 11:44:07 +03:00
|
|
|
|
sNames, _ := sniValue.([]interface{})
|
2024-03-11 13:04:15 +03:00
|
|
|
|
params["sni"] = sNames[random.Num(len(sNames))].(string)
|
2023-04-11 22:00:24 +03:00
|
|
|
|
}
|
|
|
|
|
if pbkValue, ok := searchKey(realitySettings, "publicKey"); ok {
|
|
|
|
|
params["pbk"], _ = pbkValue.(string)
|
|
|
|
|
}
|
2023-06-07 12:15:58 +03:00
|
|
|
|
if sidValue, ok := searchKey(realitySetting, "shortIds"); ok {
|
2023-04-12 11:44:07 +03:00
|
|
|
|
shortIds, _ := sidValue.([]interface{})
|
2024-03-11 13:04:15 +03:00
|
|
|
|
params["sid"] = shortIds[random.Num(len(shortIds))].(string)
|
2023-04-11 22:00:24 +03:00
|
|
|
|
}
|
|
|
|
|
if fpValue, ok := searchKey(realitySettings, "fingerprint"); ok {
|
2023-04-19 11:25:31 +03:00
|
|
|
|
if fp, ok := fpValue.(string); ok && len(fp) > 0 {
|
|
|
|
|
params["fp"] = fp
|
|
|
|
|
}
|
|
|
|
|
}
|
2024-03-11 13:04:15 +03:00
|
|
|
|
params["spx"] = "/" + random.Seq(15)
|
2023-04-11 22:00:24 +03:00
|
|
|
|
}
|
2023-04-12 11:44:07 +03:00
|
|
|
|
|
|
|
|
|
if streamNetwork == "tcp" && len(clients[clientIndex].Flow) > 0 {
|
|
|
|
|
params["flow"] = clients[clientIndex].Flow
|
2023-04-09 22:43:18 +03:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2024-10-29 14:50:25 +03:00
|
|
|
|
if security != "tls" && security != "reality" {
|
2023-06-14 16:36:56 +03:00
|
|
|
|
params["security"] = "none"
|
|
|
|
|
}
|
|
|
|
|
|
2023-12-08 20:45:21 +03:00
|
|
|
|
externalProxies, _ := stream["externalProxy"].([]interface{})
|
2023-04-09 22:43:18 +03:00
|
|
|
|
|
2023-12-08 20:45:21 +03:00
|
|
|
|
if len(externalProxies) > 0 {
|
|
|
|
|
links := ""
|
|
|
|
|
for index, externalProxy := range externalProxies {
|
|
|
|
|
ep, _ := externalProxy.(map[string]interface{})
|
|
|
|
|
newSecurity, _ := ep["forceTls"].(string)
|
|
|
|
|
dest, _ := ep["dest"].(string)
|
|
|
|
|
port := int(ep["port"].(float64))
|
|
|
|
|
link := fmt.Sprintf("trojan://%s@%s:%d", password, dest, port)
|
|
|
|
|
|
|
|
|
|
if newSecurity != "same" {
|
|
|
|
|
params["security"] = newSecurity
|
|
|
|
|
} else {
|
|
|
|
|
params["security"] = security
|
|
|
|
|
}
|
|
|
|
|
url, _ := url.Parse(link)
|
|
|
|
|
q := url.Query()
|
2023-04-09 22:43:18 +03:00
|
|
|
|
|
2023-12-08 20:45:21 +03:00
|
|
|
|
for k, v := range params {
|
|
|
|
|
if !(newSecurity == "none" && (k == "alpn" || k == "sni" || k == "fp" || k == "allowInsecure")) {
|
|
|
|
|
q.Add(k, v)
|
|
|
|
|
}
|
|
|
|
|
}
|
2023-04-09 22:43:18 +03:00
|
|
|
|
|
2023-12-08 20:45:21 +03:00
|
|
|
|
// Set the new query values on the URL
|
|
|
|
|
url.RawQuery = q.Encode()
|
|
|
|
|
|
|
|
|
|
url.Fragment = s.genRemark(inbound, email, ep["remark"].(string))
|
2023-04-09 22:43:18 +03:00
|
|
|
|
|
2023-05-22 17:36:34 +03:00
|
|
|
|
if index > 0 {
|
|
|
|
|
links += "\n"
|
|
|
|
|
}
|
|
|
|
|
links += url.String()
|
|
|
|
|
}
|
|
|
|
|
return links
|
|
|
|
|
}
|
|
|
|
|
|
2023-12-08 20:45:21 +03:00
|
|
|
|
link := fmt.Sprintf("trojan://%s@%s:%d", password, address, port)
|
|
|
|
|
|
|
|
|
|
url, _ := url.Parse(link)
|
|
|
|
|
q := url.Query()
|
|
|
|
|
|
|
|
|
|
for k, v := range params {
|
|
|
|
|
q.Add(k, v)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Set the new query values on the URL
|
|
|
|
|
url.RawQuery = q.Encode()
|
|
|
|
|
|
2023-08-26 14:41:12 +03:00
|
|
|
|
url.Fragment = s.genRemark(inbound, email, "")
|
2023-04-09 22:43:18 +03:00
|
|
|
|
return url.String()
|
|
|
|
|
}
|
|
|
|
|
|
2023-08-26 14:41:12 +03:00
|
|
|
|
func (s *SubService) genShadowsocksLink(inbound *model.Inbound, email string) string {
|
2023-05-06 19:51:14 +03:00
|
|
|
|
address := s.address
|
|
|
|
|
if inbound.Protocol != model.Shadowsocks {
|
|
|
|
|
return ""
|
|
|
|
|
}
|
2023-07-18 02:49:01 +03:00
|
|
|
|
var stream map[string]interface{}
|
|
|
|
|
json.Unmarshal([]byte(inbound.StreamSettings), &stream)
|
2023-05-22 17:36:34 +03:00
|
|
|
|
clients, _ := s.inboundService.GetClients(inbound)
|
2023-05-06 19:51:14 +03:00
|
|
|
|
|
|
|
|
|
var settings map[string]interface{}
|
|
|
|
|
json.Unmarshal([]byte(inbound.Settings), &settings)
|
|
|
|
|
inboundPassword := settings["password"].(string)
|
|
|
|
|
method := settings["method"].(string)
|
|
|
|
|
clientIndex := -1
|
|
|
|
|
for i, client := range clients {
|
|
|
|
|
if client.Email == email {
|
|
|
|
|
clientIndex = i
|
|
|
|
|
break
|
|
|
|
|
}
|
|
|
|
|
}
|
2023-07-18 02:49:01 +03:00
|
|
|
|
streamNetwork := stream["network"].(string)
|
|
|
|
|
params := make(map[string]string)
|
|
|
|
|
params["type"] = streamNetwork
|
|
|
|
|
|
|
|
|
|
switch streamNetwork {
|
|
|
|
|
case "tcp":
|
|
|
|
|
tcp, _ := stream["tcpSettings"].(map[string]interface{})
|
|
|
|
|
header, _ := tcp["header"].(map[string]interface{})
|
|
|
|
|
typeStr, _ := header["type"].(string)
|
|
|
|
|
if typeStr == "http" {
|
|
|
|
|
request := header["request"].(map[string]interface{})
|
|
|
|
|
requestPath, _ := request["path"].([]interface{})
|
|
|
|
|
params["path"] = requestPath[0].(string)
|
|
|
|
|
headers, _ := request["headers"].(map[string]interface{})
|
|
|
|
|
params["host"] = searchHost(headers)
|
|
|
|
|
params["headerType"] = "http"
|
|
|
|
|
}
|
|
|
|
|
case "kcp":
|
|
|
|
|
kcp, _ := stream["kcpSettings"].(map[string]interface{})
|
|
|
|
|
header, _ := kcp["header"].(map[string]interface{})
|
|
|
|
|
params["headerType"] = header["type"].(string)
|
|
|
|
|
params["seed"] = kcp["seed"].(string)
|
|
|
|
|
case "ws":
|
|
|
|
|
ws, _ := stream["wsSettings"].(map[string]interface{})
|
|
|
|
|
params["path"] = ws["path"].(string)
|
2024-06-18 13:49:20 +03:00
|
|
|
|
if host, ok := ws["host"].(string); ok && len(host) > 0 {
|
|
|
|
|
params["host"] = host
|
|
|
|
|
} else {
|
|
|
|
|
headers, _ := ws["headers"].(map[string]interface{})
|
|
|
|
|
params["host"] = searchHost(headers)
|
2024-04-01 15:32:02 +03:00
|
|
|
|
}
|
2023-07-18 02:49:01 +03:00
|
|
|
|
case "http":
|
|
|
|
|
http, _ := stream["httpSettings"].(map[string]interface{})
|
|
|
|
|
params["path"] = http["path"].(string)
|
|
|
|
|
params["host"] = searchHost(http)
|
|
|
|
|
case "grpc":
|
|
|
|
|
grpc, _ := stream["grpcSettings"].(map[string]interface{})
|
|
|
|
|
params["serviceName"] = grpc["serviceName"].(string)
|
2024-04-01 10:39:45 +03:00
|
|
|
|
params["authority"], _ = grpc["authority"].(string)
|
2023-07-18 02:49:01 +03:00
|
|
|
|
if grpc["multiMode"].(bool) {
|
|
|
|
|
params["mode"] = "multi"
|
|
|
|
|
}
|
2024-03-11 10:36:33 +03:00
|
|
|
|
case "httpupgrade":
|
|
|
|
|
httpupgrade, _ := stream["httpupgradeSettings"].(map[string]interface{})
|
|
|
|
|
params["path"] = httpupgrade["path"].(string)
|
2024-06-18 13:49:20 +03:00
|
|
|
|
if host, ok := httpupgrade["host"].(string); ok && len(host) > 0 {
|
|
|
|
|
params["host"] = host
|
|
|
|
|
} else {
|
|
|
|
|
headers, _ := httpupgrade["headers"].(map[string]interface{})
|
|
|
|
|
params["host"] = searchHost(headers)
|
|
|
|
|
}
|
|
|
|
|
case "splithttp":
|
|
|
|
|
splithttp, _ := stream["splithttpSettings"].(map[string]interface{})
|
|
|
|
|
params["path"] = splithttp["path"].(string)
|
|
|
|
|
if host, ok := splithttp["host"].(string); ok && len(host) > 0 {
|
|
|
|
|
params["host"] = host
|
|
|
|
|
} else {
|
|
|
|
|
headers, _ := splithttp["headers"].(map[string]interface{})
|
|
|
|
|
params["host"] = searchHost(headers)
|
2024-04-02 23:11:06 +03:00
|
|
|
|
}
|
2023-07-18 02:49:01 +03:00
|
|
|
|
}
|
|
|
|
|
|
2023-12-08 20:45:21 +03:00
|
|
|
|
security, _ := stream["security"].(string)
|
|
|
|
|
if security == "tls" {
|
|
|
|
|
params["security"] = "tls"
|
|
|
|
|
tlsSetting, _ := stream["tlsSettings"].(map[string]interface{})
|
|
|
|
|
alpns, _ := tlsSetting["alpn"].([]interface{})
|
|
|
|
|
var alpn []string
|
|
|
|
|
for _, a := range alpns {
|
|
|
|
|
alpn = append(alpn, a.(string))
|
|
|
|
|
}
|
|
|
|
|
if len(alpn) > 0 {
|
|
|
|
|
params["alpn"] = strings.Join(alpn, ",")
|
|
|
|
|
}
|
|
|
|
|
if sniValue, ok := searchKey(tlsSetting, "serverName"); ok {
|
|
|
|
|
params["sni"], _ = sniValue.(string)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
tlsSettings, _ := searchKey(tlsSetting, "settings")
|
|
|
|
|
if tlsSetting != nil {
|
|
|
|
|
if fpValue, ok := searchKey(tlsSettings, "fingerprint"); ok {
|
|
|
|
|
params["fp"], _ = fpValue.(string)
|
|
|
|
|
}
|
|
|
|
|
if insecure, ok := searchKey(tlsSettings, "allowInsecure"); ok {
|
|
|
|
|
if insecure.(bool) {
|
|
|
|
|
params["allowInsecure"] = "1"
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2023-07-27 11:28:12 +03:00
|
|
|
|
encPart := fmt.Sprintf("%s:%s", method, clients[clientIndex].Password)
|
|
|
|
|
if method[0] == '2' {
|
|
|
|
|
encPart = fmt.Sprintf("%s:%s:%s", method, inboundPassword, clients[clientIndex].Password)
|
|
|
|
|
}
|
2023-12-08 20:45:21 +03:00
|
|
|
|
|
|
|
|
|
externalProxies, _ := stream["externalProxy"].([]interface{})
|
|
|
|
|
|
|
|
|
|
if len(externalProxies) > 0 {
|
|
|
|
|
links := ""
|
|
|
|
|
for index, externalProxy := range externalProxies {
|
|
|
|
|
ep, _ := externalProxy.(map[string]interface{})
|
|
|
|
|
newSecurity, _ := ep["forceTls"].(string)
|
|
|
|
|
dest, _ := ep["dest"].(string)
|
|
|
|
|
port := int(ep["port"].(float64))
|
|
|
|
|
link := fmt.Sprintf("ss://%s@%s:%d", base64.StdEncoding.EncodeToString([]byte(encPart)), dest, port)
|
|
|
|
|
|
|
|
|
|
if newSecurity != "same" {
|
|
|
|
|
params["security"] = newSecurity
|
|
|
|
|
} else {
|
|
|
|
|
params["security"] = security
|
|
|
|
|
}
|
|
|
|
|
url, _ := url.Parse(link)
|
|
|
|
|
q := url.Query()
|
|
|
|
|
|
|
|
|
|
for k, v := range params {
|
|
|
|
|
if !(newSecurity == "none" && (k == "alpn" || k == "sni" || k == "fp" || k == "allowInsecure")) {
|
|
|
|
|
q.Add(k, v)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Set the new query values on the URL
|
|
|
|
|
url.RawQuery = q.Encode()
|
|
|
|
|
|
|
|
|
|
url.Fragment = s.genRemark(inbound, email, ep["remark"].(string))
|
|
|
|
|
|
|
|
|
|
if index > 0 {
|
|
|
|
|
links += "\n"
|
|
|
|
|
}
|
|
|
|
|
links += url.String()
|
|
|
|
|
}
|
|
|
|
|
return links
|
|
|
|
|
}
|
|
|
|
|
|
2023-07-18 02:49:01 +03:00
|
|
|
|
link := fmt.Sprintf("ss://%s@%s:%d", base64.StdEncoding.EncodeToString([]byte(encPart)), address, inbound.Port)
|
|
|
|
|
url, _ := url.Parse(link)
|
|
|
|
|
q := url.Query()
|
|
|
|
|
|
|
|
|
|
for k, v := range params {
|
|
|
|
|
q.Add(k, v)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Set the new query values on the URL
|
|
|
|
|
url.RawQuery = q.Encode()
|
2023-12-08 20:45:21 +03:00
|
|
|
|
|
2023-08-26 14:41:12 +03:00
|
|
|
|
url.Fragment = s.genRemark(inbound, email, "")
|
|
|
|
|
return url.String()
|
|
|
|
|
}
|
2023-07-18 02:49:01 +03:00
|
|
|
|
|
2023-08-26 14:41:12 +03:00
|
|
|
|
func (s *SubService) genRemark(inbound *model.Inbound, email string, extra string) string {
|
2023-12-08 22:31:17 +03:00
|
|
|
|
separationChar := string(s.remarkModel[0])
|
|
|
|
|
orderChars := s.remarkModel[1:]
|
|
|
|
|
orders := map[byte]string{
|
|
|
|
|
'i': "",
|
|
|
|
|
'e': "",
|
|
|
|
|
'o': "",
|
|
|
|
|
}
|
2023-08-26 14:41:12 +03:00
|
|
|
|
if len(email) > 0 {
|
2023-12-08 22:31:17 +03:00
|
|
|
|
orders['e'] = email
|
|
|
|
|
}
|
|
|
|
|
if len(inbound.Remark) > 0 {
|
|
|
|
|
orders['i'] = inbound.Remark
|
|
|
|
|
}
|
|
|
|
|
if len(extra) > 0 {
|
2023-12-10 20:13:48 +03:00
|
|
|
|
orders['o'] = extra
|
2023-12-08 22:31:17 +03:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
var remark []string
|
|
|
|
|
for i := 0; i < len(orderChars); i++ {
|
|
|
|
|
char := orderChars[i]
|
|
|
|
|
order, exists := orders[char]
|
|
|
|
|
if exists && order != "" {
|
|
|
|
|
remark = append(remark, order)
|
2023-08-26 14:41:12 +03:00
|
|
|
|
}
|
2023-08-01 23:58:51 +03:00
|
|
|
|
}
|
2023-08-26 14:41:12 +03:00
|
|
|
|
|
|
|
|
|
if s.showInfo {
|
|
|
|
|
statsExist := false
|
|
|
|
|
var stats xray.ClientTraffic
|
|
|
|
|
for _, clientStat := range inbound.ClientStats {
|
|
|
|
|
if clientStat.Email == email {
|
|
|
|
|
stats = clientStat
|
|
|
|
|
statsExist = true
|
|
|
|
|
break
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Get remained days
|
|
|
|
|
if statsExist {
|
|
|
|
|
if !stats.Enable {
|
2023-12-08 22:31:17 +03:00
|
|
|
|
return fmt.Sprintf("⛔️N/A%s%s", separationChar, strings.Join(remark, separationChar))
|
2023-08-26 14:41:12 +03:00
|
|
|
|
}
|
|
|
|
|
if vol := stats.Total - (stats.Up + stats.Down); vol > 0 {
|
|
|
|
|
remark = append(remark, fmt.Sprintf("%s%s", common.FormatTraffic(vol), "📊"))
|
|
|
|
|
}
|
|
|
|
|
now := time.Now().Unix()
|
|
|
|
|
switch exp := stats.ExpiryTime / 1000; {
|
|
|
|
|
case exp > 0:
|
2024-07-01 20:11:40 +03:00
|
|
|
|
remainingSeconds := exp - now
|
|
|
|
|
days := remainingSeconds / 86400
|
|
|
|
|
hours := (remainingSeconds % 86400) / 3600
|
|
|
|
|
minutes := (remainingSeconds % 3600) / 60
|
|
|
|
|
if days > 0 {
|
|
|
|
|
if hours > 0 {
|
|
|
|
|
remark = append(remark, fmt.Sprintf("%dD,%dH⏳", days, hours))
|
|
|
|
|
} else {
|
|
|
|
|
remark = append(remark, fmt.Sprintf("%dD⏳", days))
|
|
|
|
|
}
|
|
|
|
|
} else if hours > 0 {
|
|
|
|
|
remark = append(remark, fmt.Sprintf("%dH⏳", hours))
|
|
|
|
|
} else {
|
|
|
|
|
remark = append(remark, fmt.Sprintf("%dM⏳", minutes))
|
|
|
|
|
}
|
2023-08-26 14:41:12 +03:00
|
|
|
|
case exp < 0:
|
2024-07-17 17:39:53 +03:00
|
|
|
|
days := exp / -86400
|
|
|
|
|
hours := (exp % -86400) / 3600
|
|
|
|
|
minutes := (exp % -3600) / 60
|
2024-07-01 20:11:40 +03:00
|
|
|
|
if days > 0 {
|
|
|
|
|
if hours > 0 {
|
|
|
|
|
remark = append(remark, fmt.Sprintf("%dD,%dH⏳", days, hours))
|
|
|
|
|
} else {
|
|
|
|
|
remark = append(remark, fmt.Sprintf("%dD⏳", days))
|
|
|
|
|
}
|
|
|
|
|
} else if hours > 0 {
|
|
|
|
|
remark = append(remark, fmt.Sprintf("%dH⏳", hours))
|
|
|
|
|
} else {
|
|
|
|
|
remark = append(remark, fmt.Sprintf("%dM⏳", minutes))
|
|
|
|
|
}
|
2023-08-26 14:41:12 +03:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
2023-12-08 22:31:17 +03:00
|
|
|
|
return strings.Join(remark, separationChar)
|
2023-05-06 19:51:14 +03:00
|
|
|
|
}
|
|
|
|
|
|
2023-04-09 22:43:18 +03:00
|
|
|
|
func searchKey(data interface{}, key string) (interface{}, bool) {
|
|
|
|
|
switch val := data.(type) {
|
|
|
|
|
case map[string]interface{}:
|
|
|
|
|
for k, v := range val {
|
|
|
|
|
if k == key {
|
|
|
|
|
return v, true
|
|
|
|
|
}
|
|
|
|
|
if result, ok := searchKey(v, key); ok {
|
|
|
|
|
return result, true
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
case []interface{}:
|
|
|
|
|
for _, v := range val {
|
|
|
|
|
if result, ok := searchKey(v, key); ok {
|
|
|
|
|
return result, true
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return nil, false
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func searchHost(headers interface{}) string {
|
|
|
|
|
data, _ := headers.(map[string]interface{})
|
|
|
|
|
for k, v := range data {
|
|
|
|
|
if strings.EqualFold(k, "host") {
|
|
|
|
|
switch v.(type) {
|
|
|
|
|
case []interface{}:
|
|
|
|
|
hosts, _ := v.([]interface{})
|
2023-04-25 14:09:09 +03:00
|
|
|
|
if len(hosts) > 0 {
|
|
|
|
|
return hosts[0].(string)
|
|
|
|
|
} else {
|
|
|
|
|
return ""
|
|
|
|
|
}
|
2023-04-09 22:43:18 +03:00
|
|
|
|
case interface{}:
|
|
|
|
|
return v.(string)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return ""
|
|
|
|
|
}
|