| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100 | package freedomimport (	"net"	"sync"	"github.com/v2ray/v2ray-core/app"	v2io "github.com/v2ray/v2ray-core/common/io"	"github.com/v2ray/v2ray-core/common/log"	v2net "github.com/v2ray/v2ray-core/common/net"	"github.com/v2ray/v2ray-core/common/retry"	"github.com/v2ray/v2ray-core/transport/dialer"	"github.com/v2ray/v2ray-core/transport/ray")type FreedomConnection struct {	space app.Space}func (this *FreedomConnection) Dispatch(firstPacket v2net.Packet, ray ray.OutboundRay) error {	log.Info("Freedom: Opening connection to ", firstPacket.Destination())	var conn net.Conn	err := retry.Timed(5, 100).On(func() error {		rawConn, err := dialer.Dial(firstPacket.Destination())		if err != nil {			return err		}		conn = rawConn		return nil	})	if err != nil {		close(ray.OutboundOutput())		log.Error("Freedom: Failed to open connection to ", firstPacket.Destination(), ": ", err)		return err	}	defer conn.Close()	input := ray.OutboundInput()	output := ray.OutboundOutput()	var readMutex, writeMutex sync.Mutex	readMutex.Lock()	writeMutex.Lock()	if chunk := firstPacket.Chunk(); chunk != nil {		conn.Write(chunk.Value)		chunk.Release()	}	if !firstPacket.MoreChunks() {		writeMutex.Unlock()	} else {		go func() {			v2io.ChanToWriter(conn, input)			writeMutex.Unlock()		}()	}	go func() {		defer readMutex.Unlock()		defer close(output)		response, err := v2io.ReadFrom(conn, nil)		log.Info("Freedom receives ", response.Len(), " bytes from ", conn.RemoteAddr())		if response.Len() > 0 {			output <- response		} else {			response.Release()		}		if err != nil {			return		}		if firstPacket.Destination().IsUDP() {			return		}		v2io.RawReaderToChan(output, conn)	}()	if this.space.HasDnsCache() {		if firstPacket.Destination().Address().IsDomain() {			domain := firstPacket.Destination().Address().Domain()			addr := conn.RemoteAddr()			switch typedAddr := addr.(type) {			case *net.TCPAddr:				this.space.DnsCache().Add(domain, typedAddr.IP)			case *net.UDPAddr:				this.space.DnsCache().Add(domain, typedAddr.IP)			}		}	}	writeMutex.Lock()	if tcpConn, ok := conn.(*net.TCPConn); ok {		tcpConn.CloseWrite()	}	readMutex.Lock()	return nil}
 |