mirror of
https://github.com/yusing/godoxy.git
synced 2025-05-19 20:32:35 +02:00

- Updated Dockerfile and Makefile for socket-proxy build. - Modified go.mod to include necessary dependencies. - Updated CI workflows for socket-proxy integration. - Better module isolation - Code refactor
94 lines
2.4 KiB
Go
94 lines
2.4 KiB
Go
package gpwebsocket
|
|
|
|
import (
|
|
"net/http"
|
|
"sync"
|
|
"time"
|
|
|
|
"github.com/coder/websocket"
|
|
"github.com/yusing/go-proxy/internal/logging"
|
|
)
|
|
|
|
func warnNoMatchDomains() {
|
|
logging.Warn().Msg("no match domains configured, accepting websocket API request from all origins")
|
|
}
|
|
|
|
var warnNoMatchDomainOnce sync.Once
|
|
|
|
const (
|
|
HeaderXGoDoxyWebsocketAllowedDomains = "X-GoDoxy-Websocket-Allowed-Domains"
|
|
)
|
|
|
|
func WebsocketAllowedDomains(h http.Header) []string {
|
|
return h[HeaderXGoDoxyWebsocketAllowedDomains]
|
|
}
|
|
|
|
func SetWebsocketAllowedDomains(h http.Header, domains []string) {
|
|
h[HeaderXGoDoxyWebsocketAllowedDomains] = domains
|
|
}
|
|
|
|
func Initiate(w http.ResponseWriter, r *http.Request) (*websocket.Conn, error) {
|
|
var originPats []string
|
|
|
|
localAddresses := []string{"127.0.0.1", "10.0.*.*", "172.16.*.*", "192.168.*.*"}
|
|
|
|
allowedDomains := WebsocketAllowedDomains(r.Header)
|
|
if len(allowedDomains) == 0 {
|
|
warnNoMatchDomainOnce.Do(warnNoMatchDomains)
|
|
originPats = []string{"*"}
|
|
} else {
|
|
originPats = make([]string, len(allowedDomains))
|
|
for i, domain := range allowedDomains {
|
|
if domain[0] != '.' {
|
|
originPats[i] = "*." + domain
|
|
} else {
|
|
originPats[i] = "*" + domain
|
|
}
|
|
}
|
|
originPats = append(originPats, localAddresses...)
|
|
}
|
|
return websocket.Accept(w, r, &websocket.AcceptOptions{
|
|
OriginPatterns: originPats,
|
|
})
|
|
}
|
|
|
|
func Periodic(w http.ResponseWriter, r *http.Request, interval time.Duration, do func(conn *websocket.Conn) error) {
|
|
conn, err := Initiate(w, r)
|
|
if err != nil {
|
|
http.Error(w, err.Error(), http.StatusInternalServerError)
|
|
return
|
|
}
|
|
//nolint:errcheck
|
|
defer conn.CloseNow()
|
|
|
|
if err := do(conn); err != nil {
|
|
http.Error(w, err.Error(), http.StatusInternalServerError)
|
|
return
|
|
}
|
|
|
|
ticker := time.NewTicker(interval)
|
|
defer ticker.Stop()
|
|
|
|
for {
|
|
select {
|
|
case <-r.Context().Done():
|
|
return
|
|
case <-ticker.C:
|
|
if err := do(conn); err != nil {
|
|
http.Error(w, err.Error(), http.StatusInternalServerError)
|
|
return
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// WriteText writes a text message to the websocket connection.
|
|
// It returns true if the message was written successfully, false otherwise.
|
|
// It logs an error if the message is not written successfully.
|
|
func WriteText(r *http.Request, conn *websocket.Conn, msg string) bool {
|
|
if err := conn.Write(r.Context(), websocket.MessageText, []byte(msg)); err != nil {
|
|
logging.Err(err).Msg("failed to write text message")
|
|
return false
|
|
}
|
|
return true
|
|
}
|