mirror of
https://github.com/itzg/mc-router.git
synced 2024-11-16 10:35:19 +01:00
176 lines
4.5 KiB
Go
176 lines
4.5 KiB
Go
package server
|
|
|
|
import (
|
|
"bytes"
|
|
"context"
|
|
"github.com/itzg/mc-router/mcproto"
|
|
"github.com/sirupsen/logrus"
|
|
"io"
|
|
"net"
|
|
"time"
|
|
)
|
|
|
|
const (
|
|
connectionsLimitPerSec = 1
|
|
handshakeTimeout = 30 * time.Second
|
|
)
|
|
|
|
var noDeadline time.Time
|
|
|
|
type IConnector interface {
|
|
StartAcceptingConnections(ctx context.Context, listenAddress string) error
|
|
}
|
|
|
|
var Connector IConnector = &connectorImpl{}
|
|
|
|
type connectorImpl struct {
|
|
}
|
|
|
|
func (c *connectorImpl) StartAcceptingConnections(ctx context.Context, listenAddress string) error {
|
|
|
|
ln, err := net.Listen("tcp", listenAddress)
|
|
if err != nil {
|
|
logrus.WithError(err).Fatal("Unable to start listening")
|
|
return err
|
|
}
|
|
logrus.WithField("listenAddress", listenAddress).Info("Listening for Minecraft client connections")
|
|
|
|
go c.acceptConnections(ctx, ln)
|
|
|
|
return nil
|
|
}
|
|
|
|
func (c *connectorImpl) acceptConnections(ctx context.Context, ln net.Listener) {
|
|
//noinspection GoUnhandledErrorResult
|
|
defer ln.Close()
|
|
|
|
limiter := time.Tick(time.Second / connectionsLimitPerSec)
|
|
|
|
for {
|
|
select {
|
|
case <-ctx.Done():
|
|
return
|
|
|
|
case <-limiter:
|
|
conn, err := ln.Accept()
|
|
if err != nil {
|
|
logrus.WithError(err).Error("Failed to accept connection")
|
|
} else {
|
|
go c.HandleConnection(ctx, conn)
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
func (c *connectorImpl) HandleConnection(ctx context.Context, frontendConn net.Conn) {
|
|
//noinspection GoUnhandledErrorResult
|
|
defer frontendConn.Close()
|
|
|
|
clientAddr := frontendConn.RemoteAddr()
|
|
logrus.WithFields(logrus.Fields{"clientAddr": clientAddr}).Info("Got connection")
|
|
|
|
inspectionBuffer := new(bytes.Buffer)
|
|
|
|
inspectionReader := io.TeeReader(frontendConn, inspectionBuffer)
|
|
|
|
if err := frontendConn.SetReadDeadline(time.Now().Add(handshakeTimeout)); err != nil {
|
|
logrus.
|
|
WithError(err).
|
|
WithField("client", frontendConn).
|
|
Error("Failed to set read deadline")
|
|
return
|
|
}
|
|
packet, err := mcproto.ReadPacket(inspectionReader, clientAddr)
|
|
if err != nil {
|
|
logrus.WithError(err).WithField("clientAddr", clientAddr).Error("Failed to read packet")
|
|
return
|
|
}
|
|
|
|
logrus.WithFields(logrus.Fields{"length": packet.Length, "packetID": packet.PacketID}).Info("Got packet")
|
|
|
|
if packet.PacketID == mcproto.PacketIdHandshake {
|
|
handshake, err := mcproto.ReadHandshake(packet.Data)
|
|
if err != nil {
|
|
logrus.WithError(err).WithField("clientAddr", clientAddr).Error("Failed to read handshake")
|
|
return
|
|
}
|
|
|
|
logrus.WithFields(logrus.Fields{
|
|
"protocolVersion": handshake.ProtocolVersion,
|
|
"server": handshake.ServerAddress,
|
|
"serverPort": handshake.ServerPort,
|
|
"nextState": handshake.NextState,
|
|
}).Info("Got handshake")
|
|
|
|
backendHostPort := Routes.FindBackendForServerAddress(handshake.ServerAddress)
|
|
if backendHostPort == "" {
|
|
logrus.WithField("serverAddress", handshake.ServerAddress).Warn("Unable to find registered backend")
|
|
return
|
|
}
|
|
|
|
logrus.WithField("backendHostPort", backendHostPort).Info("Connecting to backend")
|
|
backendConn, err := net.Dial("tcp", backendHostPort)
|
|
if err != nil {
|
|
logrus.WithError(err).WithFields(logrus.Fields{
|
|
"serverAddress": handshake.ServerAddress,
|
|
"backend": backendHostPort,
|
|
}).Warn("Unable to connect to backend")
|
|
return
|
|
}
|
|
|
|
amount, err := io.Copy(backendConn, inspectionBuffer)
|
|
if err != nil {
|
|
logrus.WithError(err).Error("Failed to write handshake to backend connection")
|
|
return
|
|
}
|
|
logrus.WithField("amount", amount).Debug("Relayed handshake to backend")
|
|
|
|
if err = frontendConn.SetReadDeadline(noDeadline); err != nil {
|
|
logrus.
|
|
WithError(err).
|
|
WithField("client", frontendConn).
|
|
Error("Failed to clear read deadline")
|
|
return
|
|
}
|
|
pumpConnections(ctx, frontendConn, backendConn)
|
|
} else {
|
|
logrus.
|
|
WithField("client", frontendConn).
|
|
WithField("packetID", packet.PacketID).
|
|
Error("Unexpected packetID, expected handshake")
|
|
return
|
|
}
|
|
}
|
|
|
|
func pumpConnections(ctx context.Context, frontendConn, backendConn net.Conn) {
|
|
//noinspection GoUnhandledErrorResult
|
|
defer backendConn.Close()
|
|
|
|
errors := make(chan error, 2)
|
|
|
|
go pumpFrames(backendConn, frontendConn, errors, "backend", "frontend")
|
|
go pumpFrames(frontendConn, backendConn, errors, "frontend", "backend")
|
|
|
|
for {
|
|
select {
|
|
case err := <-errors:
|
|
if err != io.EOF {
|
|
logrus.WithError(err).Error("Error observed on connection relay")
|
|
}
|
|
|
|
return
|
|
|
|
case <-ctx.Done():
|
|
return
|
|
}
|
|
}
|
|
}
|
|
|
|
func pumpFrames(incoming io.Reader, outgoing io.Writer, errors chan<- error, from, to string) {
|
|
amount, err := io.Copy(outgoing, incoming)
|
|
if err != nil {
|
|
errors <- err
|
|
}
|
|
logrus.WithField("amount", amount).Infof("Finished relay %s->%s", from, to)
|
|
}
|