mirror of
https://github.com/TeaOSLab/EdgeNode.git
synced 2025-11-02 14:00:25 +08:00
102 lines
2.5 KiB
Go
102 lines
2.5 KiB
Go
package nodes
|
||
|
||
import (
|
||
"errors"
|
||
"github.com/TeaOSLab/EdgeNode/internal/utils"
|
||
"io"
|
||
"net/http"
|
||
"net/url"
|
||
)
|
||
|
||
// 处理Websocket请求
|
||
func (this *HTTPRequest) doWebsocket(requestHost string) {
|
||
if this.web.WebsocketRef == nil || !this.web.WebsocketRef.IsOn || this.web.Websocket == nil || !this.web.Websocket.IsOn {
|
||
this.writer.WriteHeader(http.StatusForbidden)
|
||
this.addError(errors.New("websocket have not been enabled yet"))
|
||
return
|
||
}
|
||
|
||
// TODO 实现handshakeTimeout
|
||
|
||
// 校验来源
|
||
var requestOrigin = this.RawReq.Header.Get("Origin")
|
||
if len(requestOrigin) > 0 {
|
||
u, err := url.Parse(requestOrigin)
|
||
if err == nil {
|
||
if !this.web.Websocket.MatchOrigin(u.Host) {
|
||
this.writer.WriteHeader(http.StatusForbidden)
|
||
this.addError(errors.New("websocket origin '" + requestOrigin + "' not been allowed"))
|
||
return
|
||
}
|
||
}
|
||
}
|
||
|
||
// 设置指定的来源域
|
||
if !this.web.Websocket.RequestSameOrigin && len(this.web.Websocket.RequestOrigin) > 0 {
|
||
var newRequestOrigin = this.web.Websocket.RequestOrigin
|
||
if this.web.Websocket.RequestOriginHasVariables() {
|
||
newRequestOrigin = this.Format(newRequestOrigin)
|
||
}
|
||
this.RawReq.Header.Set("Origin", newRequestOrigin)
|
||
}
|
||
|
||
// TODO 增加N次错误重试,重试的时候需要尝试不同的源站
|
||
originConn, _, err := OriginConnect(this.origin, this.requestServerPort(), this.RawReq.RemoteAddr, requestHost)
|
||
if err != nil {
|
||
this.write50x(err, http.StatusBadGateway, false)
|
||
|
||
// 增加失败次数
|
||
SharedOriginStateManager.Fail(this.origin, requestHost, this.reverseProxy, func() {
|
||
this.reverseProxy.ResetScheduling()
|
||
})
|
||
|
||
return
|
||
}
|
||
|
||
if !this.origin.IsOk {
|
||
SharedOriginStateManager.Success(this.origin, func() {
|
||
this.reverseProxy.ResetScheduling()
|
||
})
|
||
}
|
||
|
||
defer func() {
|
||
_ = originConn.Close()
|
||
}()
|
||
|
||
err = this.RawReq.Write(originConn)
|
||
if err != nil {
|
||
this.write50x(err, http.StatusBadGateway, false)
|
||
return
|
||
}
|
||
|
||
clientConn, _, err := this.writer.Hijack()
|
||
if err != nil || clientConn == nil {
|
||
this.write50x(err, http.StatusInternalServerError, false)
|
||
return
|
||
}
|
||
defer func() {
|
||
_ = clientConn.Close()
|
||
}()
|
||
|
||
go func() {
|
||
var buf = utils.BytePool4k.Get()
|
||
defer utils.BytePool4k.Put(buf)
|
||
for {
|
||
n, err := originConn.Read(buf)
|
||
if n > 0 {
|
||
this.writer.sentBodyBytes += int64(n)
|
||
_, err = clientConn.Write(buf[:n])
|
||
if err != nil {
|
||
break
|
||
}
|
||
}
|
||
if err != nil {
|
||
break
|
||
}
|
||
}
|
||
_ = clientConn.Close()
|
||
_ = originConn.Close()
|
||
}()
|
||
_, _ = io.Copy(originConn, clientConn)
|
||
}
|