mirror of https://github.com/XTLS/Xray-core
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
109 lines
2.4 KiB
109 lines
2.4 KiB
package tcp |
|
|
|
import ( |
|
"context" |
|
"fmt" |
|
"io" |
|
|
|
"github.com/xtls/xray-core/common/buf" |
|
"github.com/xtls/xray-core/common/net" |
|
"github.com/xtls/xray-core/common/task" |
|
"github.com/xtls/xray-core/transport/internet" |
|
"github.com/xtls/xray-core/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() |
|
}
|
|
|