mirror of
https://github.com/MetaCubeX/mihomo.git
synced 2024-11-16 11:42:43 +08:00
chore: listeners support tuic
This commit is contained in:
parent
62474e0ed6
commit
c7f83d3ff1
|
@ -4,7 +4,8 @@ import (
|
||||||
"crypto/tls"
|
"crypto/tls"
|
||||||
"fmt"
|
"fmt"
|
||||||
)
|
)
|
||||||
func ParseCert(certificate,privateKey string) (tls.Certificate, error) {
|
|
||||||
|
func ParseCert(certificate, privateKey string) (tls.Certificate, error) {
|
||||||
cert, painTextErr := tls.X509KeyPair([]byte(certificate), []byte(privateKey))
|
cert, painTextErr := tls.X509KeyPair([]byte(certificate), []byte(privateKey))
|
||||||
if painTextErr == nil {
|
if painTextErr == nil {
|
||||||
return cert, nil
|
return cert, nil
|
||||||
|
@ -12,7 +13,7 @@ func ParseCert(certificate,privateKey string) (tls.Certificate, error) {
|
||||||
|
|
||||||
cert, loadErr := tls.LoadX509KeyPair(certificate, privateKey)
|
cert, loadErr := tls.LoadX509KeyPair(certificate, privateKey)
|
||||||
if loadErr != nil {
|
if loadErr != nil {
|
||||||
return tls.Certificate{}, fmt.Errorf("parse certificate failed,maybe format error:%s, or path error: %s", painTextErr.Error(), loadErr.Error())
|
return tls.Certificate{}, fmt.Errorf("parse certificate failed, maybe format error:%s, or path error: %s", painTextErr.Error(), loadErr.Error())
|
||||||
}
|
}
|
||||||
return cert, nil
|
return cert, nil
|
||||||
}
|
}
|
||||||
|
|
|
@ -695,3 +695,18 @@ listeners:
|
||||||
listen: 0.0.0.0
|
listen: 0.0.0.0
|
||||||
# udp: false # 默认 true
|
# udp: false # 默认 true
|
||||||
# rule: sub-rule
|
# rule: sub-rule
|
||||||
|
|
||||||
|
- name: tuic-in-1
|
||||||
|
type: tuic
|
||||||
|
port: 10813
|
||||||
|
listen: 0.0.0.0
|
||||||
|
# token:
|
||||||
|
# - TOKEN
|
||||||
|
# certificate: ./server.crt
|
||||||
|
# private-key: ./server.key
|
||||||
|
# congestion-controller: bbr
|
||||||
|
# max-idle-time: 15000
|
||||||
|
# authentication-timeout: 1000
|
||||||
|
# alpn:
|
||||||
|
# - h3
|
||||||
|
# max-udp-relay-packet-size: 1500
|
||||||
|
|
86
listener/inbound/tuic.go
Normal file
86
listener/inbound/tuic.go
Normal file
|
@ -0,0 +1,86 @@
|
||||||
|
package inbound
|
||||||
|
|
||||||
|
import (
|
||||||
|
C "github.com/Dreamacro/clash/constant"
|
||||||
|
LC "github.com/Dreamacro/clash/listener/config"
|
||||||
|
"github.com/Dreamacro/clash/listener/tuic"
|
||||||
|
"github.com/Dreamacro/clash/log"
|
||||||
|
)
|
||||||
|
|
||||||
|
type TuicOption struct {
|
||||||
|
BaseOption
|
||||||
|
Token []string `inbound:"token"`
|
||||||
|
Certificate string `inbound:"certificate"`
|
||||||
|
PrivateKey string `inbound:"private-key"`
|
||||||
|
CongestionController string `inbound:"congestion-controllerr,omitempty"`
|
||||||
|
MaxIdleTime int `inbound:"max-idle-timer,omitempty"`
|
||||||
|
AuthenticationTimeout int `inbound:"authentication-timeoutr,omitempty"`
|
||||||
|
ALPN []string `inbound:"alpnr,omitempty"`
|
||||||
|
MaxUdpRelayPacketSize int `inbound:"max-udp-relay-packet-sizer,omitempty"`
|
||||||
|
}
|
||||||
|
|
||||||
|
func (o TuicOption) Equal(config C.InboundConfig) bool {
|
||||||
|
return optionToString(o) == optionToString(config)
|
||||||
|
}
|
||||||
|
|
||||||
|
type Tuic struct {
|
||||||
|
*Base
|
||||||
|
config *TuicOption
|
||||||
|
l *tuic.Listener
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewTuic(options *TuicOption) (*Tuic, error) {
|
||||||
|
base, err := NewBase(&options.BaseOption)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
return &Tuic{
|
||||||
|
Base: base,
|
||||||
|
config: options,
|
||||||
|
}, nil
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
// Config implements constant.InboundListener
|
||||||
|
func (t *Tuic) Config() C.InboundConfig {
|
||||||
|
return t.config
|
||||||
|
}
|
||||||
|
|
||||||
|
// Address implements constant.InboundListener
|
||||||
|
func (t *Tuic) Address() string {
|
||||||
|
if t.l != nil {
|
||||||
|
for _, addr := range t.l.AddrList() {
|
||||||
|
return addr.String()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return ""
|
||||||
|
}
|
||||||
|
|
||||||
|
// Listen implements constant.InboundListener
|
||||||
|
func (t *Tuic) Listen(tcpIn chan<- C.ConnContext, udpIn chan<- C.PacketAdapter) error {
|
||||||
|
var err error
|
||||||
|
t.l, err = tuic.New(LC.TuicServer{
|
||||||
|
Enable: true,
|
||||||
|
Listen: t.RawAddress(),
|
||||||
|
Token: t.config.Token,
|
||||||
|
Certificate: t.config.Certificate,
|
||||||
|
PrivateKey: t.config.PrivateKey,
|
||||||
|
CongestionController: t.config.CongestionController,
|
||||||
|
MaxIdleTime: t.config.MaxIdleTime,
|
||||||
|
AuthenticationTimeout: t.config.AuthenticationTimeout,
|
||||||
|
ALPN: t.config.ALPN,
|
||||||
|
MaxUdpRelayPacketSize: t.config.MaxUdpRelayPacketSize,
|
||||||
|
}, tcpIn, udpIn)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
log.Infoln("Tuic[%s] proxy listening at: %s", t.Name(), t.Address())
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// Close implements constant.InboundListener
|
||||||
|
func (t *Tuic) Close() error {
|
||||||
|
return t.l.Close()
|
||||||
|
}
|
||||||
|
|
||||||
|
var _ C.InboundListener = (*Tuic)(nil)
|
|
@ -371,6 +371,9 @@ func ReCreateTuic(config LC.TuicServer, tcpIn chan<- C.ConnContext, udpIn chan<-
|
||||||
|
|
||||||
tuicListener = listener
|
tuicListener = listener
|
||||||
|
|
||||||
|
for _, addr := range tuicListener.AddrList() {
|
||||||
|
log.Infoln("Tuic proxy listening at: %s", addr.String())
|
||||||
|
}
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -55,6 +55,18 @@ func ParseListener(mapping map[string]any) (C.InboundListener, error) {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
listener, err = IN.NewMixed(mixedOption)
|
listener, err = IN.NewMixed(mixedOption)
|
||||||
|
case "tuic":
|
||||||
|
tuicOption := &IN.TuicOption{
|
||||||
|
MaxIdleTime: 15000,
|
||||||
|
AuthenticationTimeout: 1000,
|
||||||
|
ALPN: []string{"h3"},
|
||||||
|
MaxUdpRelayPacketSize: 1500,
|
||||||
|
}
|
||||||
|
err = decoder.Decode(mapping, tuicOption)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
listener, err = IN.NewTuic(tuicOption)
|
||||||
default:
|
default:
|
||||||
return nil, fmt.Errorf("unsupport proxy type: %s", proxyType)
|
return nil, fmt.Errorf("unsupport proxy type: %s", proxyType)
|
||||||
}
|
}
|
||||||
|
|
|
@ -9,6 +9,7 @@ import (
|
||||||
"github.com/metacubex/quic-go"
|
"github.com/metacubex/quic-go"
|
||||||
|
|
||||||
"github.com/Dreamacro/clash/adapter/inbound"
|
"github.com/Dreamacro/clash/adapter/inbound"
|
||||||
|
CN "github.com/Dreamacro/clash/common/net"
|
||||||
"github.com/Dreamacro/clash/common/sockopt"
|
"github.com/Dreamacro/clash/common/sockopt"
|
||||||
C "github.com/Dreamacro/clash/constant"
|
C "github.com/Dreamacro/clash/constant"
|
||||||
LC "github.com/Dreamacro/clash/listener/config"
|
LC "github.com/Dreamacro/clash/listener/config"
|
||||||
|
@ -25,7 +26,11 @@ type Listener struct {
|
||||||
}
|
}
|
||||||
|
|
||||||
func New(config LC.TuicServer, tcpIn chan<- C.ConnContext, udpIn chan<- C.PacketAdapter) (*Listener, error) {
|
func New(config LC.TuicServer, tcpIn chan<- C.ConnContext, udpIn chan<- C.PacketAdapter) (*Listener, error) {
|
||||||
cert, err := tls.LoadX509KeyPair(config.Certificate, config.PrivateKey)
|
return NewWithInfos("DEFAULT-TUIC", "", config, tcpIn, udpIn)
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewWithInfos(name, specialRules string, config LC.TuicServer, tcpIn chan<- C.ConnContext, udpIn chan<- C.PacketAdapter) (*Listener, error) {
|
||||||
|
cert, err := CN.ParseCert(config.Certificate, config.PrivateKey)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
|
@ -56,12 +61,12 @@ func New(config LC.TuicServer, tcpIn chan<- C.ConnContext, udpIn chan<- C.Packet
|
||||||
|
|
||||||
option := &tuic.ServerOption{
|
option := &tuic.ServerOption{
|
||||||
HandleTcpFn: func(conn net.Conn, addr socks5.Addr) error {
|
HandleTcpFn: func(conn net.Conn, addr socks5.Addr) error {
|
||||||
tcpIn <- inbound.NewSocket(addr, conn, C.TUIC)
|
tcpIn <- inbound.NewSocketWithInfos(addr, conn, C.TUIC, name, specialRules)
|
||||||
return nil
|
return nil
|
||||||
},
|
},
|
||||||
HandleUdpFn: func(addr socks5.Addr, packet C.UDPPacket) error {
|
HandleUdpFn: func(addr socks5.Addr, packet C.UDPPacket) error {
|
||||||
select {
|
select {
|
||||||
case udpIn <- inbound.NewPacket(addr, packet, C.TUIC):
|
case udpIn <- inbound.NewPacketWithInfos(addr, packet, C.TUIC, name, specialRules):
|
||||||
default:
|
default:
|
||||||
}
|
}
|
||||||
return nil
|
return nil
|
||||||
|
@ -99,7 +104,6 @@ func New(config LC.TuicServer, tcpIn chan<- C.ConnContext, udpIn chan<- C.Packet
|
||||||
sl.servers = append(sl.servers, server)
|
sl.servers = append(sl.servers, server)
|
||||||
|
|
||||||
go func() {
|
go func() {
|
||||||
log.Infoln("Tuic proxy listening at: %s", ul.LocalAddr().String())
|
|
||||||
err := server.Serve()
|
err := server.Serve()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
if sl.closed {
|
if sl.closed {
|
||||||
|
@ -112,16 +116,32 @@ func New(config LC.TuicServer, tcpIn chan<- C.ConnContext, udpIn chan<- C.Packet
|
||||||
return sl, nil
|
return sl, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (l *Listener) Close() {
|
// Close implements C.Listener
|
||||||
|
func (l *Listener) Close() error {
|
||||||
l.closed = true
|
l.closed = true
|
||||||
|
var retErr error
|
||||||
for _, lis := range l.servers {
|
for _, lis := range l.servers {
|
||||||
_ = lis.Close()
|
err := lis.Close()
|
||||||
|
if err != nil {
|
||||||
|
retErr = err
|
||||||
|
}
|
||||||
}
|
}
|
||||||
for _, lis := range l.udpListeners {
|
for _, lis := range l.udpListeners {
|
||||||
_ = lis.Close()
|
err := lis.Close()
|
||||||
|
if err != nil {
|
||||||
|
retErr = err
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
return retErr
|
||||||
}
|
}
|
||||||
|
|
||||||
func (l *Listener) Config() LC.TuicServer {
|
func (l *Listener) Config() LC.TuicServer {
|
||||||
return l.config
|
return l.config
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (l *Listener) AddrList() (addrList []net.Addr) {
|
||||||
|
for _, lis := range l.udpListeners {
|
||||||
|
addrList = append(addrList, lis.LocalAddr())
|
||||||
|
}
|
||||||
|
return
|
||||||
|
}
|
Loading…
Reference in New Issue
Block a user