Skip to content
Snippets Groups Projects
tcp.go 3.77 KiB
Newer Older
  • Learn to ignore specific revisions
  • Konrad Zemek's avatar
    Konrad Zemek committed
    // Copyright 2019 Path Network, Inc. All rights reserved.
    // Use of this source code is governed by a BSD-style
    // license that can be found in the LICENSE file.
    
    package main
    
    import (
    	"context"
    	"io"
    
    Konrad Zemek's avatar
    Konrad Zemek committed
    	"net"
    
    Konrad Zemek's avatar
    Konrad Zemek committed
    )
    
    func tcpCopyData(dst net.Conn, src net.Conn, ch chan<- error) {
    	_, err := io.Copy(dst, src)
    	ch <- err
    }
    
    
    func tcpHandleConnection(conn net.Conn, logger *slog.Logger) {
    
    Konrad Zemek's avatar
    Konrad Zemek committed
    	defer conn.Close()
    
    	logger = logger.With(slog.String("remoteAddr", conn.RemoteAddr().String()),
    		slog.String("localAddr", conn.LocalAddr().String()))
    
    Konrad Zemek's avatar
    Konrad Zemek committed
    
    	if !CheckOriginAllowed(conn.RemoteAddr().(*net.TCPAddr).IP) {
    
    		logger.Debug("connection origin not in allowed subnets", slog.Bool("dropConnection", true))
    
    Konrad Zemek's avatar
    Konrad Zemek committed
    		return
    	}
    
    	if Opts.Verbose > 1 {
    		logger.Debug("new connection")
    	}
    
    	buffer := GetBuffer()
    	defer func() {
    		if buffer != nil {
    			PutBuffer(buffer)
    		}
    	}()
    
    	n, err := conn.Read(buffer)
    	if err != nil {
    
    		logger.Debug("failed to read PROXY header", "error", err, slog.Bool("dropConnection", true))
    
    Konrad Zemek's avatar
    Konrad Zemek committed
    		return
    	}
    
    	saddr, _, restBytes, err := PROXYReadRemoteAddr(buffer[:n], TCP)
    	if err != nil {
    
    		logger.Debug("failed to parse PROXY header", "error", err, slog.Bool("dropConnection", true))
    
    Konrad Zemek's avatar
    Konrad Zemek committed
    		return
    	}
    
    	targetAddr := Opts.TargetAddr6
    
    		if netip.MustParseAddrPort(conn.RemoteAddr().String()).Addr().Is4() {
    
    			targetAddr = Opts.TargetAddr4
    		}
    
    	} else if netip.MustParseAddrPort(saddr.String()).Addr().Is4() {
    
    Konrad Zemek's avatar
    Konrad Zemek committed
    		targetAddr = Opts.TargetAddr4
    	}
    
    
    	clientAddr := "UNKNOWN"
    	if saddr != nil {
    		clientAddr = saddr.String()
    	}
    
    Konrad Zemek's avatar
    Konrad Zemek committed
    	logger = logger.With(slog.String("clientAddr", clientAddr), slog.String("targetAddr", targetAddr.String()))
    
    Konrad Zemek's avatar
    Konrad Zemek committed
    	if Opts.Verbose > 1 {
    
    		logger.Debug("successfully parsed PROXY header")
    
    Konrad Zemek's avatar
    Konrad Zemek committed
    	}
    
    	dialer := net.Dialer{LocalAddr: saddr}
    	if saddr != nil {
    		dialer.Control = DialUpstreamControl(saddr.(*net.TCPAddr).Port)
    	}
    
    Konrad Zemek's avatar
    Konrad Zemek committed
    	upstreamConn, err := dialer.Dial("tcp", targetAddr.String())
    
    Konrad Zemek's avatar
    Konrad Zemek committed
    	if err != nil {
    
    		logger.Debug("failed to establish upstream connection", "error", err, slog.Bool("dropConnection", true))
    
    Konrad Zemek's avatar
    Konrad Zemek committed
    		return
    	}
    
    	defer upstreamConn.Close()
    	if Opts.Verbose > 1 {
    
    		logger.Debug("successfully established upstream connection")
    
    Konrad Zemek's avatar
    Konrad Zemek committed
    	}
    
    	if err := conn.(*net.TCPConn).SetNoDelay(true); err != nil {
    
    		logger.Debug("failed to set nodelay on downstream connection", "error", err, slog.Bool("dropConnection", true))
    
    Konrad Zemek's avatar
    Konrad Zemek committed
    	} else if Opts.Verbose > 1 {
    
    		logger.Debug("successfully set NoDelay on downstream connection")
    
    Konrad Zemek's avatar
    Konrad Zemek committed
    	}
    
    	if err := upstreamConn.(*net.TCPConn).SetNoDelay(true); err != nil {
    
    		logger.Debug("failed to set nodelay on upstream connection", "error", err, slog.Bool("dropConnection", true))
    
    Konrad Zemek's avatar
    Konrad Zemek committed
    	} else if Opts.Verbose > 1 {
    
    		logger.Debug("successfully set NoDelay on upstream connection")
    
    Konrad Zemek's avatar
    Konrad Zemek committed
    	}
    
    	for len(restBytes) > 0 {
    		n, err := upstreamConn.Write(restBytes)
    		if err != nil {
    			logger.Debug("failed to write data to upstream connection",
    
    				"error", err, slog.Bool("dropConnection", true))
    
    Konrad Zemek's avatar
    Konrad Zemek committed
    			return
    		}
    		restBytes = restBytes[n:]
    	}
    
    	PutBuffer(buffer)
    	buffer = nil
    
    	outErr := make(chan error, 2)
    	go tcpCopyData(upstreamConn, conn, outErr)
    	go tcpCopyData(conn, upstreamConn, outErr)
    
    	err = <-outErr
    	if err != nil {
    
    		logger.Debug("connection broken", "error", err, slog.Bool("dropConnection", true))
    
    Konrad Zemek's avatar
    Konrad Zemek committed
    	} else if Opts.Verbose > 1 {
    		logger.Debug("connection closing")
    	}
    }
    
    
    func TCPListen(listenConfig *net.ListenConfig, logger *slog.Logger, errors chan<- error) {
    
    Konrad Zemek's avatar
    Konrad Zemek committed
    	ctx := context.Background()
    
    Konrad Zemek's avatar
    Konrad Zemek committed
    	ln, err := listenConfig.Listen(ctx, "tcp", Opts.ListenAddr.String())
    
    Konrad Zemek's avatar
    Konrad Zemek committed
    	if err != nil {
    
    		logger.Error("failed to bind listener", "error", err)
    
    Konrad Zemek's avatar
    Konrad Zemek committed
    		errors <- err
    		return
    	}
    
    	logger.Info("listening")
    
    	for {
    		conn, err := ln.Accept()
    		if err != nil {
    
    			logger.Error("failed to accept new connection", "error", err)
    
    Konrad Zemek's avatar
    Konrad Zemek committed
    			errors <- err
    			return
    		}
    
    		go tcpHandleConnection(conn, logger)
    	}
    }