2024-07-23 21:58:11 +00:00
|
|
|
package core
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"net"
|
|
|
|
"net/http"
|
|
|
|
"net/url"
|
|
|
|
"time"
|
|
|
|
|
|
|
|
"github.com/Arceliar/phony"
|
2024-09-22 16:54:58 +01:00
|
|
|
"github.com/coder/websocket"
|
2024-07-23 21:58:11 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
type linkWS struct {
|
|
|
|
phony.Inbox
|
|
|
|
*links
|
2024-08-11 10:42:25 +01:00
|
|
|
listenconfig *net.ListenConfig
|
2024-07-23 21:58:11 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
type linkWSConn struct {
|
|
|
|
net.Conn
|
|
|
|
}
|
|
|
|
|
|
|
|
type linkWSListener struct {
|
|
|
|
ch chan *linkWSConn
|
|
|
|
ctx context.Context
|
|
|
|
httpServer *http.Server
|
|
|
|
listener net.Listener
|
|
|
|
}
|
|
|
|
|
|
|
|
type wsServer struct {
|
|
|
|
ch chan *linkWSConn
|
|
|
|
ctx context.Context
|
|
|
|
}
|
|
|
|
|
|
|
|
func (l *linkWSListener) Accept() (net.Conn, error) {
|
|
|
|
qs := <-l.ch
|
|
|
|
if qs == nil {
|
|
|
|
return nil, context.Canceled
|
|
|
|
}
|
|
|
|
return qs, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (l *linkWSListener) Addr() net.Addr {
|
|
|
|
return l.listener.Addr()
|
|
|
|
}
|
|
|
|
|
|
|
|
func (l *linkWSListener) Close() error {
|
|
|
|
if err := l.httpServer.Shutdown(l.ctx); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
return l.listener.Close()
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *wsServer) ServeHTTP(w http.ResponseWriter, r *http.Request) {
|
|
|
|
if r.URL.Path == "/health" || r.URL.Path == "/healthz" {
|
|
|
|
w.WriteHeader(http.StatusOK)
|
|
|
|
_, _ = w.Write([]byte("OK"))
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
c, err := websocket.Accept(w, r, &websocket.AcceptOptions{
|
|
|
|
Subprotocols: []string{"ygg-ws"},
|
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
if c.Subprotocol() != "ygg-ws" {
|
|
|
|
c.Close(websocket.StatusPolicyViolation, "client must speak the ygg-ws subprotocol")
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
s.ch <- &linkWSConn{
|
|
|
|
Conn: websocket.NetConn(s.ctx, c, websocket.MessageBinary),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (l *links) newLinkWS() *linkWS {
|
|
|
|
lt := &linkWS{
|
|
|
|
links: l,
|
2024-08-11 10:42:25 +01:00
|
|
|
listenconfig: &net.ListenConfig{
|
|
|
|
KeepAlive: -1,
|
|
|
|
},
|
2024-07-23 21:58:11 +00:00
|
|
|
}
|
|
|
|
return lt
|
|
|
|
}
|
|
|
|
|
|
|
|
func (l *linkWS) dial(ctx context.Context, url *url.URL, info linkInfo, options linkOptions) (net.Conn, error) {
|
2024-10-27 21:06:56 +00:00
|
|
|
if options.tlsSNI != "" {
|
|
|
|
return nil, ErrLinkSNINotSupported
|
|
|
|
}
|
2024-07-23 21:58:11 +00:00
|
|
|
wsconn, _, err := websocket.Dial(ctx, url.String(), &websocket.DialOptions{
|
|
|
|
Subprotocols: []string{"ygg-ws"},
|
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
return &linkWSConn{
|
|
|
|
Conn: websocket.NetConn(ctx, wsconn, websocket.MessageBinary),
|
|
|
|
}, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (l *linkWS) listen(ctx context.Context, url *url.URL, _ string) (net.Listener, error) {
|
2024-08-11 10:42:25 +01:00
|
|
|
nl, err := l.listenconfig.Listen(ctx, "tcp", url.Host)
|
2024-07-23 21:58:11 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
ch := make(chan *linkWSConn)
|
|
|
|
|
|
|
|
httpServer := &http.Server{
|
|
|
|
Handler: &wsServer{
|
|
|
|
ch: ch,
|
|
|
|
ctx: ctx,
|
|
|
|
},
|
|
|
|
BaseContext: func(_ net.Listener) context.Context { return ctx },
|
|
|
|
ReadTimeout: time.Second * 10,
|
|
|
|
WriteTimeout: time.Second * 10,
|
|
|
|
}
|
|
|
|
|
|
|
|
lwl := &linkWSListener{
|
|
|
|
ch: ch,
|
|
|
|
ctx: ctx,
|
|
|
|
httpServer: httpServer,
|
|
|
|
listener: nl,
|
|
|
|
}
|
|
|
|
go lwl.httpServer.Serve(nl) // nolint:errcheck
|
|
|
|
return lwl, nil
|
|
|
|
}
|