mirror of
https://github.com/v2fly/v2ray-core.git
synced 2024-11-04 17:27:23 -05:00
Keep sending data after request is sent
This commit is contained in:
parent
9265f50b3d
commit
6a8014dbef
@ -3,7 +3,6 @@ package vmess
|
|||||||
import (
|
import (
|
||||||
"crypto/md5"
|
"crypto/md5"
|
||||||
"crypto/rand"
|
"crypto/rand"
|
||||||
"io"
|
|
||||||
mrand "math/rand"
|
mrand "math/rand"
|
||||||
"net"
|
"net"
|
||||||
|
|
||||||
@ -62,11 +61,11 @@ func (handler *VMessOutboundHandler) Start(ray core.OutboundRay) error {
|
|||||||
request.Command = byte(0x01)
|
request.Command = byte(0x01)
|
||||||
request.Address = handler.dest
|
request.Address = handler.dest
|
||||||
|
|
||||||
go handler.startCommunicate(request, vNextAddress, ray)
|
go startCommunicate(request, vNextAddress, ray)
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (handler *VMessOutboundHandler) startCommunicate(request *vmessio.VMessRequest, dest v2net.Address, ray core.OutboundRay) error {
|
func startCommunicate(request *vmessio.VMessRequest, dest v2net.Address, ray core.OutboundRay) error {
|
||||||
input := ray.OutboundInput()
|
input := ray.OutboundInput()
|
||||||
output := ray.OutboundOutput()
|
output := ray.OutboundOutput()
|
||||||
|
|
||||||
@ -78,73 +77,61 @@ func (handler *VMessOutboundHandler) startCommunicate(request *vmessio.VMessRequ
|
|||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
defer conn.Close()
|
defer conn.Close()
|
||||||
|
defer close(output)
|
||||||
|
|
||||||
requestWriter := vmessio.NewVMessRequestWriter()
|
requestFinish := make(chan bool)
|
||||||
err = requestWriter.Write(conn, request)
|
responseFinish := make(chan bool)
|
||||||
|
|
||||||
|
go handleRequest(conn, request, input, requestFinish)
|
||||||
|
go handleResponse(conn, request, output, responseFinish)
|
||||||
|
|
||||||
|
<-requestFinish
|
||||||
|
conn.CloseWrite()
|
||||||
|
<-responseFinish
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func handleRequest(conn *net.TCPConn, request *vmessio.VMessRequest, input <-chan []byte, finish chan<- bool) error {
|
||||||
|
defer close(finish)
|
||||||
|
requestWriter := vmessio.NewVMessRequestWriter()
|
||||||
|
err := requestWriter.Write(conn, request)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Error("Failed to write VMess request: %v", err)
|
log.Error("Failed to write VMess request: %v", err)
|
||||||
close(output)
|
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
requestKey := request.RequestKey[:]
|
encryptRequestWriter, err := v2io.NewAesEncryptWriter(request.RequestKey[:], request.RequestIV[:], conn)
|
||||||
requestIV := request.RequestIV[:]
|
if err != nil {
|
||||||
responseKey := md5.Sum(requestKey)
|
log.Error("Failed to create encrypt writer: %v", err)
|
||||||
responseIV := md5.Sum(requestIV)
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
v2net.ChanToWriter(encryptRequestWriter, input)
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func handleResponse(conn *net.TCPConn, request *vmessio.VMessRequest, output chan<- []byte, finish chan<- bool) error {
|
||||||
|
defer close(finish)
|
||||||
|
responseKey := md5.Sum(request.RequestKey[:])
|
||||||
|
responseIV := md5.Sum(request.RequestIV[:])
|
||||||
|
|
||||||
response := vmessio.VMessResponse{}
|
response := vmessio.VMessResponse{}
|
||||||
nBytes, err := conn.Read(response[:])
|
nBytes, err := conn.Read(response[:])
|
||||||
if err != nil {
|
if err != nil {
|
||||||
close(output)
|
|
||||||
log.Error("Failed to read VMess response (%d bytes): %v", nBytes, err)
|
log.Error("Failed to read VMess response (%d bytes): %v", nBytes, err)
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
log.Debug("Got response %v", response)
|
log.Debug("Got response %v", response)
|
||||||
// TODO: check response
|
// TODO: check response
|
||||||
|
|
||||||
encryptRequestWriter, err := v2io.NewAesEncryptWriter(requestKey, requestIV, conn)
|
|
||||||
if err != nil {
|
|
||||||
close(output)
|
|
||||||
log.Error("Failed to create encrypt writer: %v", err)
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
decryptResponseReader, err := v2io.NewAesDecryptReader(responseKey[:], responseIV[:], conn)
|
decryptResponseReader, err := v2io.NewAesDecryptReader(responseKey[:], responseIV[:], conn)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
close(output)
|
|
||||||
log.Error("Failed to create decrypt reader: %v", err)
|
log.Error("Failed to create decrypt reader: %v", err)
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
readFinish := make(chan bool)
|
v2net.ReaderToChan(output, decryptResponseReader)
|
||||||
writeFinish := make(chan bool)
|
return nil
|
||||||
|
|
||||||
go handler.dumpInput(encryptRequestWriter, input, readFinish)
|
|
||||||
go handler.dumpOutput(decryptResponseReader, output, writeFinish)
|
|
||||||
|
|
||||||
<-readFinish
|
|
||||||
conn.CloseWrite()
|
|
||||||
log.Debug("VMessOut closing write")
|
|
||||||
<-writeFinish
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func (handler *VMessOutboundHandler) dumpOutput(reader io.Reader, output chan<- []byte, finish chan<- bool) {
|
|
||||||
v2net.ReaderToChan(output, reader)
|
|
||||||
close(output)
|
|
||||||
log.Debug("VMessOut closing output")
|
|
||||||
finish <- true
|
|
||||||
}
|
|
||||||
|
|
||||||
func (handler *VMessOutboundHandler) dumpInput(writer io.Writer, input <-chan []byte, finish chan<- bool) {
|
|
||||||
v2net.ChanToWriter(writer, input)
|
|
||||||
log.Debug("VMessOut closing input")
|
|
||||||
finish <- true
|
|
||||||
}
|
|
||||||
|
|
||||||
func (handler *VMessOutboundHandler) waitForFinish(finish <-chan bool) {
|
|
||||||
<-finish
|
|
||||||
<-finish
|
|
||||||
log.Debug("Finishing waiting for VMessOutbound ending.")
|
|
||||||
}
|
}
|
||||||
|
|
||||||
type VMessOutboundHandlerFactory struct {
|
type VMessOutboundHandlerFactory struct {
|
||||||
|
Loading…
Reference in New Issue
Block a user