2023-07-12 13:44:15 -04:00
|
|
|
package xray
|
2022-06-01 13:35:41 -04:00
|
|
|
|
|
|
|
import (
|
2022-12-18 10:31:06 -05:00
|
|
|
"crypto/rand"
|
|
|
|
"encoding/base64"
|
|
|
|
"encoding/hex"
|
2023-06-02 10:55:53 -04:00
|
|
|
"errors"
|
2022-06-01 13:35:41 -04:00
|
|
|
"fmt"
|
2022-08-16 01:04:33 -04:00
|
|
|
"github.com/Yuzuki616/V2bX/api/panel"
|
2022-07-28 11:00:05 -04:00
|
|
|
"github.com/Yuzuki616/V2bX/conf"
|
2022-10-11 04:05:19 -04:00
|
|
|
"github.com/goccy/go-json"
|
2022-06-01 13:35:41 -04:00
|
|
|
"github.com/xtls/xray-core/common/net"
|
|
|
|
"github.com/xtls/xray-core/core"
|
2022-07-28 11:00:05 -04:00
|
|
|
coreConf "github.com/xtls/xray-core/infra/conf"
|
2022-06-01 13:35:41 -04:00
|
|
|
)
|
|
|
|
|
2023-05-21 22:36:10 -04:00
|
|
|
// BuildInbound build Inbound config for different protocol
|
2023-07-12 13:44:15 -04:00
|
|
|
func buildInbound(config *conf.ControllerConfig, nodeInfo *panel.NodeInfo, tag string) (*core.InboundHandlerConfig, error) {
|
2023-05-22 09:01:31 -04:00
|
|
|
in := &coreConf.InboundDetourConfig{}
|
2022-12-18 10:31:06 -05:00
|
|
|
// Set network protocol
|
|
|
|
t := coreConf.TransportProtocol(nodeInfo.Network)
|
2023-05-22 09:01:31 -04:00
|
|
|
in.StreamSetting = &coreConf.StreamConfig{Network: &t}
|
2022-12-18 10:31:06 -05:00
|
|
|
var err error
|
2023-06-08 10:46:33 -04:00
|
|
|
switch nodeInfo.Type {
|
2023-05-18 23:27:08 -04:00
|
|
|
case "v2ray":
|
2023-05-22 09:01:31 -04:00
|
|
|
err = buildV2ray(config, nodeInfo, in)
|
2023-05-18 23:27:08 -04:00
|
|
|
case "trojan":
|
2023-06-09 00:36:49 -04:00
|
|
|
err = buildTrojan(config, in)
|
2023-05-18 23:27:08 -04:00
|
|
|
case "shadowsocks":
|
2023-05-22 09:01:31 -04:00
|
|
|
err = buildShadowsocks(config, nodeInfo, in)
|
2022-12-18 10:31:06 -05:00
|
|
|
default:
|
2023-06-08 10:46:33 -04:00
|
|
|
return nil, fmt.Errorf("unsupported node type: %s, Only support: V2ray, Trojan, Shadowsocks", nodeInfo.Type)
|
2022-10-09 20:59:00 -04:00
|
|
|
}
|
2022-12-18 10:31:06 -05:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
// Set server port
|
2023-05-22 09:01:31 -04:00
|
|
|
in.PortList = &coreConf.PortList{
|
2023-06-08 10:46:33 -04:00
|
|
|
Range: []coreConf.PortRange{{From: uint32(nodeInfo.Port), To: uint32(nodeInfo.Port)}},
|
2022-12-18 10:31:06 -05:00
|
|
|
}
|
|
|
|
// Set Listen IP address
|
2022-06-01 13:35:41 -04:00
|
|
|
ipAddress := net.ParseAddress(config.ListenIP)
|
2023-05-22 09:01:31 -04:00
|
|
|
in.ListenOn = &coreConf.Address{Address: ipAddress}
|
2022-12-18 10:31:06 -05:00
|
|
|
// Set SniffingConfig
|
2022-07-28 11:00:05 -04:00
|
|
|
sniffingConfig := &coreConf.SniffingConfig{
|
2022-06-01 13:35:41 -04:00
|
|
|
Enabled: true,
|
2022-07-28 11:00:05 -04:00
|
|
|
DestOverride: &coreConf.StringList{"http", "tls"},
|
2022-06-01 13:35:41 -04:00
|
|
|
}
|
2023-06-08 10:46:33 -04:00
|
|
|
if config.XrayOptions.DisableSniffing {
|
2022-06-01 13:35:41 -04:00
|
|
|
sniffingConfig.Enabled = false
|
|
|
|
}
|
2023-05-22 09:01:31 -04:00
|
|
|
in.SniffingConfig = sniffingConfig
|
|
|
|
if *in.StreamSetting.Network == "tcp" {
|
|
|
|
if in.StreamSetting.TCPSettings != nil {
|
2023-07-08 10:44:49 -04:00
|
|
|
in.StreamSetting.TCPSettings.AcceptProxyProtocol = config.XrayOptions.EnableProxyProtocol
|
2022-06-02 13:22:56 -04:00
|
|
|
} else {
|
2022-07-28 11:00:05 -04:00
|
|
|
tcpSetting := &coreConf.TCPConfig{
|
2023-07-08 10:44:49 -04:00
|
|
|
AcceptProxyProtocol: config.XrayOptions.EnableProxyProtocol,
|
2022-10-09 20:59:00 -04:00
|
|
|
} //Enable proxy protocol
|
2023-05-22 09:01:31 -04:00
|
|
|
in.StreamSetting.TCPSettings = tcpSetting
|
2022-06-01 13:35:41 -04:00
|
|
|
}
|
2023-05-22 09:01:31 -04:00
|
|
|
} else if *in.StreamSetting.Network == "ws" {
|
|
|
|
in.StreamSetting.WSSettings = &coreConf.WebSocketConfig{
|
2023-07-08 10:44:49 -04:00
|
|
|
AcceptProxyProtocol: config.XrayOptions.EnableProxyProtocol} //Enable proxy protocol
|
2022-06-01 13:35:41 -04:00
|
|
|
}
|
2023-06-02 10:55:53 -04:00
|
|
|
// Set TLS or Reality settings
|
2023-06-08 10:46:33 -04:00
|
|
|
if nodeInfo.Tls {
|
2023-06-02 10:55:53 -04:00
|
|
|
if config.CertConfig == nil {
|
|
|
|
return nil, errors.New("the CertConfig is not vail")
|
|
|
|
}
|
|
|
|
switch config.CertConfig.CertMode {
|
|
|
|
case "none", "": // disable
|
|
|
|
case "reality":
|
|
|
|
// Reality
|
|
|
|
in.StreamSetting.Security = "reality"
|
|
|
|
d, err := json.Marshal(config.CertConfig.RealityConfig.Dest)
|
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("marshal reality dest error: %s", err)
|
|
|
|
}
|
|
|
|
in.StreamSetting.REALITYSettings = &coreConf.REALITYConfig{
|
|
|
|
Dest: d,
|
|
|
|
Xver: config.CertConfig.RealityConfig.Xver,
|
|
|
|
ServerNames: config.CertConfig.RealityConfig.ServerNames,
|
|
|
|
PrivateKey: config.CertConfig.RealityConfig.PrivateKey,
|
|
|
|
MinClientVer: config.CertConfig.RealityConfig.MinClientVer,
|
|
|
|
MaxClientVer: config.CertConfig.RealityConfig.MaxClientVer,
|
|
|
|
MaxTimeDiff: config.CertConfig.RealityConfig.MaxTimeDiff,
|
|
|
|
ShortIds: config.CertConfig.RealityConfig.ShortIds,
|
|
|
|
}
|
|
|
|
default:
|
2023-05-21 22:36:10 -04:00
|
|
|
// Normal tls
|
2023-06-20 05:08:36 -04:00
|
|
|
in.StreamSetting.Security = "tls"
|
2023-05-22 09:01:31 -04:00
|
|
|
in.StreamSetting.TLSSettings = &coreConf.TLSConfig{
|
2023-05-21 22:36:10 -04:00
|
|
|
Certs: []*coreConf.TLSCertConfig{
|
|
|
|
{
|
2023-06-08 10:46:33 -04:00
|
|
|
CertFile: config.CertConfig.CertFile,
|
|
|
|
KeyFile: config.CertConfig.KeyFile,
|
2023-05-21 22:36:10 -04:00
|
|
|
OcspStapling: 3600,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
RejectUnknownSNI: config.CertConfig.RejectUnknownSni,
|
|
|
|
}
|
2022-06-01 13:35:41 -04:00
|
|
|
}
|
2023-07-13 02:16:21 -04:00
|
|
|
// use remote reality replace local config
|
|
|
|
if nodeInfo.ExtraConfig.EnableReality {
|
|
|
|
rc := nodeInfo.ExtraConfig.RealityConfig
|
|
|
|
in.StreamSetting.Security = "reality"
|
|
|
|
d, err := json.Marshal(rc.Dest)
|
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("marshal reality dest error: %s", err)
|
|
|
|
}
|
|
|
|
in.StreamSetting.REALITYSettings = &coreConf.REALITYConfig{
|
|
|
|
Dest: d,
|
|
|
|
Xver: rc.Xver,
|
|
|
|
ServerNames: rc.ServerNames,
|
|
|
|
PrivateKey: rc.PrivateKey,
|
|
|
|
MinClientVer: rc.MinClientVer,
|
|
|
|
MaxClientVer: rc.MaxClientVer,
|
|
|
|
MaxTimeDiff: rc.MaxTimeDiff,
|
|
|
|
ShortIds: rc.ShortIds,
|
|
|
|
}
|
2023-06-24 20:22:05 -04:00
|
|
|
}
|
|
|
|
}
|
2022-06-01 13:35:41 -04:00
|
|
|
// Support ProxyProtocol for any transport protocol
|
2023-05-22 09:01:31 -04:00
|
|
|
if *in.StreamSetting.Network != "tcp" &&
|
|
|
|
*in.StreamSetting.Network != "ws" &&
|
2023-07-08 10:44:49 -04:00
|
|
|
config.XrayOptions.EnableProxyProtocol {
|
2023-06-08 10:46:33 -04:00
|
|
|
socketConfig := &coreConf.SocketConfig{
|
2023-07-08 10:44:49 -04:00
|
|
|
AcceptProxyProtocol: config.XrayOptions.EnableProxyProtocol,
|
2023-06-08 10:46:33 -04:00
|
|
|
TFO: config.XrayOptions.EnableTFO,
|
2022-10-09 20:59:00 -04:00
|
|
|
} //Enable proxy protocol
|
2023-06-08 10:46:33 -04:00
|
|
|
in.StreamSetting.SocketSettings = socketConfig
|
2022-12-18 10:31:06 -05:00
|
|
|
}
|
2023-05-22 09:01:31 -04:00
|
|
|
in.Tag = tag
|
|
|
|
return in.Build()
|
2022-12-18 10:31:06 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
func buildV2ray(config *conf.ControllerConfig, nodeInfo *panel.NodeInfo, inbound *coreConf.InboundDetourConfig) error {
|
2023-06-24 20:22:05 -04:00
|
|
|
if config.XrayOptions.EnableVless ||
|
|
|
|
nodeInfo.ExtraConfig.EnableVless {
|
2022-12-18 10:31:06 -05:00
|
|
|
//Set vless
|
|
|
|
inbound.Protocol = "vless"
|
2023-06-08 10:46:33 -04:00
|
|
|
if config.XrayOptions.EnableFallback {
|
2022-12-18 10:31:06 -05:00
|
|
|
// Set fallback
|
2023-06-08 10:46:33 -04:00
|
|
|
fallbackConfigs, err := buildVlessFallbacks(config.XrayOptions.FallBackConfigs)
|
2022-12-18 10:31:06 -05:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
s, err := json.Marshal(&coreConf.VLessInboundConfig{
|
|
|
|
Decryption: "none",
|
|
|
|
Fallbacks: fallbackConfigs,
|
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("marshal vless fallback config error: %s", err)
|
|
|
|
}
|
|
|
|
inbound.Settings = (*json.RawMessage)(&s)
|
|
|
|
} else {
|
|
|
|
var err error
|
|
|
|
s, err := json.Marshal(&coreConf.VLessInboundConfig{
|
|
|
|
Decryption: "none",
|
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("marshal vless config error: %s", err)
|
|
|
|
}
|
|
|
|
inbound.Settings = (*json.RawMessage)(&s)
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
// Set vmess
|
|
|
|
inbound.Protocol = "vmess"
|
|
|
|
var err error
|
|
|
|
s, err := json.Marshal(&coreConf.VMessInboundConfig{})
|
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("marshal vmess settings error: %s", err)
|
|
|
|
}
|
|
|
|
inbound.Settings = (*json.RawMessage)(&s)
|
|
|
|
}
|
2023-05-21 22:36:10 -04:00
|
|
|
if len(nodeInfo.NetworkSettings) == 0 {
|
|
|
|
return nil
|
|
|
|
}
|
2022-12-18 10:31:06 -05:00
|
|
|
switch nodeInfo.Network {
|
|
|
|
case "tcp":
|
|
|
|
err := json.Unmarshal(nodeInfo.NetworkSettings, &inbound.StreamSetting.TCPSettings)
|
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("unmarshal tcp settings error: %s", err)
|
|
|
|
}
|
|
|
|
case "ws":
|
|
|
|
err := json.Unmarshal(nodeInfo.NetworkSettings, &inbound.StreamSetting.WSSettings)
|
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("unmarshal ws settings error: %s", err)
|
|
|
|
}
|
|
|
|
case "grpc":
|
|
|
|
err := json.Unmarshal(nodeInfo.NetworkSettings, &inbound.StreamSetting.GRPCConfig)
|
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("unmarshal grpc settings error: %s", err)
|
|
|
|
}
|
2023-07-13 02:16:21 -04:00
|
|
|
default:
|
|
|
|
return errors.New("the network type is not vail")
|
2022-12-18 10:31:06 -05:00
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2023-06-09 00:36:49 -04:00
|
|
|
func buildTrojan(config *conf.ControllerConfig, inbound *coreConf.InboundDetourConfig) error {
|
2022-12-18 10:31:06 -05:00
|
|
|
inbound.Protocol = "trojan"
|
2023-06-08 10:46:33 -04:00
|
|
|
if config.XrayOptions.EnableFallback {
|
2022-12-18 10:31:06 -05:00
|
|
|
// Set fallback
|
2023-06-08 10:46:33 -04:00
|
|
|
fallbackConfigs, err := buildTrojanFallbacks(config.XrayOptions.FallBackConfigs)
|
2022-12-18 10:31:06 -05:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
s, err := json.Marshal(&coreConf.TrojanServerConfig{
|
|
|
|
Fallbacks: fallbackConfigs,
|
|
|
|
})
|
|
|
|
inbound.Settings = (*json.RawMessage)(&s)
|
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("marshal trojan fallback config error: %s", err)
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
s := []byte("{}")
|
|
|
|
inbound.Settings = (*json.RawMessage)(&s)
|
2022-06-01 13:35:41 -04:00
|
|
|
}
|
2023-06-09 00:36:49 -04:00
|
|
|
t := coreConf.TransportProtocol("tcp")
|
2022-12-18 10:31:06 -05:00
|
|
|
inbound.StreamSetting = &coreConf.StreamConfig{Network: &t}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func buildShadowsocks(config *conf.ControllerConfig, nodeInfo *panel.NodeInfo, inbound *coreConf.InboundDetourConfig) error {
|
|
|
|
inbound.Protocol = "shadowsocks"
|
|
|
|
settings := &coreConf.ShadowsocksServerConfig{
|
|
|
|
Cipher: nodeInfo.Cipher,
|
|
|
|
}
|
|
|
|
p := make([]byte, 32)
|
|
|
|
_, err := rand.Read(p)
|
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("generate random password error: %s", err)
|
|
|
|
}
|
|
|
|
randomPasswd := hex.EncodeToString(p)
|
|
|
|
cipher := nodeInfo.Cipher
|
|
|
|
if nodeInfo.ServerKey != "" {
|
|
|
|
settings.Password = nodeInfo.ServerKey
|
|
|
|
randomPasswd = base64.StdEncoding.EncodeToString([]byte(randomPasswd))
|
|
|
|
cipher = ""
|
|
|
|
}
|
|
|
|
defaultSSuser := &coreConf.ShadowsocksUserConfig{
|
|
|
|
Cipher: cipher,
|
|
|
|
Password: randomPasswd,
|
|
|
|
}
|
|
|
|
settings.Users = append(settings.Users, defaultSSuser)
|
|
|
|
settings.NetworkList = &coreConf.NetworkList{"tcp", "udp"}
|
|
|
|
settings.IVCheck = true
|
2023-06-08 10:46:33 -04:00
|
|
|
if config.XrayOptions.DisableIVCheck {
|
2022-12-18 10:31:06 -05:00
|
|
|
settings.IVCheck = false
|
|
|
|
}
|
|
|
|
t := coreConf.TransportProtocol("tcp")
|
|
|
|
inbound.StreamSetting = &coreConf.StreamConfig{Network: &t}
|
|
|
|
s, err := json.Marshal(settings)
|
|
|
|
inbound.Settings = (*json.RawMessage)(&s)
|
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("marshal shadowsocks settings error: %s", err)
|
|
|
|
}
|
|
|
|
return nil
|
2022-06-01 13:35:41 -04:00
|
|
|
}
|
|
|
|
|
2023-05-18 22:06:31 -04:00
|
|
|
func buildVlessFallbacks(fallbackConfigs []conf.FallBackConfig) ([]*coreConf.VLessInboundFallback, error) {
|
2022-06-01 13:35:41 -04:00
|
|
|
if fallbackConfigs == nil {
|
|
|
|
return nil, fmt.Errorf("you must provide FallBackConfigs")
|
|
|
|
}
|
2022-07-28 11:00:05 -04:00
|
|
|
vlessFallBacks := make([]*coreConf.VLessInboundFallback, len(fallbackConfigs))
|
2022-06-01 13:35:41 -04:00
|
|
|
for i, c := range fallbackConfigs {
|
|
|
|
if c.Dest == "" {
|
|
|
|
return nil, fmt.Errorf("dest is required for fallback fialed")
|
|
|
|
}
|
|
|
|
var dest json.RawMessage
|
|
|
|
dest, err := json.Marshal(c.Dest)
|
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("marshal dest %s config fialed: %s", dest, err)
|
|
|
|
}
|
2022-07-28 11:00:05 -04:00
|
|
|
vlessFallBacks[i] = &coreConf.VLessInboundFallback{
|
2022-06-01 13:35:41 -04:00
|
|
|
Name: c.SNI,
|
|
|
|
Alpn: c.Alpn,
|
|
|
|
Path: c.Path,
|
|
|
|
Dest: dest,
|
|
|
|
Xver: c.ProxyProtocolVer,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return vlessFallBacks, nil
|
|
|
|
}
|
|
|
|
|
2023-05-18 22:06:31 -04:00
|
|
|
func buildTrojanFallbacks(fallbackConfigs []conf.FallBackConfig) ([]*coreConf.TrojanInboundFallback, error) {
|
2022-06-01 13:35:41 -04:00
|
|
|
if fallbackConfigs == nil {
|
|
|
|
return nil, fmt.Errorf("you must provide FallBackConfigs")
|
|
|
|
}
|
|
|
|
|
2022-07-28 11:00:05 -04:00
|
|
|
trojanFallBacks := make([]*coreConf.TrojanInboundFallback, len(fallbackConfigs))
|
2022-06-01 13:35:41 -04:00
|
|
|
for i, c := range fallbackConfigs {
|
|
|
|
|
|
|
|
if c.Dest == "" {
|
|
|
|
return nil, fmt.Errorf("dest is required for fallback fialed")
|
|
|
|
}
|
|
|
|
|
|
|
|
var dest json.RawMessage
|
|
|
|
dest, err := json.Marshal(c.Dest)
|
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("marshal dest %s config fialed: %s", dest, err)
|
|
|
|
}
|
2022-07-28 11:00:05 -04:00
|
|
|
trojanFallBacks[i] = &coreConf.TrojanInboundFallback{
|
2022-06-01 13:35:41 -04:00
|
|
|
Name: c.SNI,
|
|
|
|
Alpn: c.Alpn,
|
|
|
|
Path: c.Path,
|
|
|
|
Dest: dest,
|
|
|
|
Xver: c.ProxyProtocolVer,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return trojanFallBacks, nil
|
|
|
|
}
|