| 12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485 | package freedomimport (	"net"	"sync"	"github.com/v2ray/v2ray-core"	"github.com/v2ray/v2ray-core/common/log"	v2net "github.com/v2ray/v2ray-core/common/net")type FreedomConnection struct {}func NewFreedomConnection() *FreedomConnection {	return &FreedomConnection{}}func (vconn *FreedomConnection) Dispatch(firstPacket v2net.Packet, ray core.OutboundRay) error {	conn, err := net.Dial(firstPacket.Destination().Network(), firstPacket.Destination().Address().String())	log.Info("Freedom: Opening connection to %s", firstPacket.Destination().String())	if err != nil {		if ray != nil {			close(ray.OutboundOutput())		}		return log.Error("Freedom: Failed to open connection: %s : %v", firstPacket.Destination().String(), err)	}	input := ray.OutboundInput()	output := ray.OutboundOutput()	var readMutex, writeMutex sync.Mutex	readMutex.Lock()	writeMutex.Lock()	if chunk := firstPacket.Chunk(); chunk != nil {		conn.Write(chunk)	}	if !firstPacket.MoreChunks() {		writeMutex.Unlock()	} else {		go dumpInput(conn, input, &writeMutex)	}	go dumpOutput(conn, output, &readMutex, firstPacket.Destination().IsUDP())	go func() {		writeMutex.Lock()		if tcpConn, ok := conn.(*net.TCPConn); ok {			tcpConn.CloseWrite()		}		readMutex.Lock()		conn.Close()	}()	return nil}func dumpInput(conn net.Conn, input <-chan []byte, finish *sync.Mutex) {	v2net.ChanToWriter(conn, input)	finish.Unlock()}func dumpOutput(conn net.Conn, output chan<- []byte, finish *sync.Mutex, udp bool) {	defer finish.Unlock()	defer close(output)	bufferSize := 4 /* KB */	if udp {		bufferSize = 2	}	response, err := v2net.ReadFrom(conn, bufferSize)	log.Info("Freedom receives %d bytes from %s", len(response), conn.RemoteAddr().String())	if len(response) > 0 {		output <- response	}	if err != nil {		return	}	if udp {		return	}	v2net.ReaderToChan(output, conn)}
 |