mirror of
https://github.com/XTLS/Xray-core.git
synced 2024-11-23 05:36:06 +03:00
111 lines
2.4 KiB
Go
111 lines
2.4 KiB
Go
|
package tcp
|
||
|
|
||
|
import (
|
||
|
"context"
|
||
|
"fmt"
|
||
|
"io"
|
||
|
|
||
|
"github.com/xtls/xray-core/v1/common/buf"
|
||
|
"github.com/xtls/xray-core/v1/common/net"
|
||
|
"github.com/xtls/xray-core/v1/common/task"
|
||
|
"github.com/xtls/xray-core/v1/transport/internet"
|
||
|
"github.com/xtls/xray-core/v1/transport/pipe"
|
||
|
)
|
||
|
|
||
|
type Server struct {
|
||
|
Port net.Port
|
||
|
MsgProcessor func(msg []byte) []byte
|
||
|
ShouldClose bool
|
||
|
SendFirst []byte
|
||
|
Listen net.Address
|
||
|
listener net.Listener
|
||
|
}
|
||
|
|
||
|
func (server *Server) Start() (net.Destination, error) {
|
||
|
return server.StartContext(context.Background(), nil)
|
||
|
}
|
||
|
|
||
|
func (server *Server) StartContext(ctx context.Context, sockopt *internet.SocketConfig) (net.Destination, error) {
|
||
|
listenerAddr := server.Listen
|
||
|
if listenerAddr == nil {
|
||
|
listenerAddr = net.LocalHostIP
|
||
|
}
|
||
|
listener, err := internet.ListenSystem(ctx, &net.TCPAddr{
|
||
|
IP: listenerAddr.IP(),
|
||
|
Port: int(server.Port),
|
||
|
}, sockopt)
|
||
|
if err != nil {
|
||
|
return net.Destination{}, err
|
||
|
}
|
||
|
|
||
|
localAddr := listener.Addr().(*net.TCPAddr)
|
||
|
server.Port = net.Port(localAddr.Port)
|
||
|
server.listener = listener
|
||
|
go server.acceptConnections(listener.(*net.TCPListener))
|
||
|
|
||
|
return net.TCPDestination(net.IPAddress(localAddr.IP), net.Port(localAddr.Port)), nil
|
||
|
}
|
||
|
|
||
|
func (server *Server) acceptConnections(listener *net.TCPListener) {
|
||
|
for {
|
||
|
conn, err := listener.Accept()
|
||
|
if err != nil {
|
||
|
fmt.Printf("Failed accept TCP connection: %v\n", err)
|
||
|
return
|
||
|
}
|
||
|
|
||
|
go server.handleConnection(conn)
|
||
|
}
|
||
|
}
|
||
|
|
||
|
func (server *Server) handleConnection(conn net.Conn) {
|
||
|
if len(server.SendFirst) > 0 {
|
||
|
conn.Write(server.SendFirst)
|
||
|
}
|
||
|
|
||
|
pReader, pWriter := pipe.New(pipe.WithoutSizeLimit())
|
||
|
err := task.Run(context.Background(), func() error {
|
||
|
defer pWriter.Close()
|
||
|
|
||
|
for {
|
||
|
b := buf.New()
|
||
|
if _, err := b.ReadFrom(conn); err != nil {
|
||
|
if err == io.EOF {
|
||
|
return nil
|
||
|
}
|
||
|
return err
|
||
|
}
|
||
|
copy(b.Bytes(), server.MsgProcessor(b.Bytes()))
|
||
|
if err := pWriter.WriteMultiBuffer(buf.MultiBuffer{b}); err != nil {
|
||
|
return err
|
||
|
}
|
||
|
}
|
||
|
}, func() error {
|
||
|
defer pReader.Interrupt()
|
||
|
|
||
|
w := buf.NewWriter(conn)
|
||
|
for {
|
||
|
mb, err := pReader.ReadMultiBuffer()
|
||
|
if err != nil {
|
||
|
if err == io.EOF {
|
||
|
return nil
|
||
|
}
|
||
|
return err
|
||
|
}
|
||
|
if err := w.WriteMultiBuffer(mb); err != nil {
|
||
|
return err
|
||
|
}
|
||
|
}
|
||
|
})
|
||
|
|
||
|
if err != nil {
|
||
|
fmt.Println("failed to transfer data: ", err.Error())
|
||
|
}
|
||
|
|
||
|
conn.Close()
|
||
|
}
|
||
|
|
||
|
func (server *Server) Close() error {
|
||
|
return server.listener.Close()
|
||
|
}
|