Replaced legacy `*.orly` module imports with `next.orly.dev/pkg` paths across the codebase for consistency. Removed legacy `go.mod` files from sub-packages, consolidating dependency management. Added Dockerfiles and configurations for benchmarking environments.
80 lines
1.8 KiB
Go
80 lines
1.8 KiB
Go
package ws
|
|
|
|
import (
|
|
"context"
|
|
"crypto/tls"
|
|
"errors"
|
|
"fmt"
|
|
"io"
|
|
"net/http"
|
|
"time"
|
|
|
|
"lol.mleku.dev/errorf"
|
|
"next.orly.dev/pkg/utils/units"
|
|
|
|
ws "github.com/coder/websocket"
|
|
)
|
|
|
|
// Connection represents a websocket connection to a Nostr relay.
|
|
type Connection struct {
|
|
conn *ws.Conn
|
|
}
|
|
|
|
// NewConnection creates a new websocket connection to a Nostr relay.
|
|
func NewConnection(
|
|
ctx context.Context, url string, reqHeader http.Header,
|
|
tlsConfig *tls.Config,
|
|
) (c *Connection, err error) {
|
|
var conn *ws.Conn
|
|
if conn, _, err = ws.Dial(
|
|
ctx, url, getConnectionOptions(reqHeader, tlsConfig),
|
|
); err != nil {
|
|
return
|
|
}
|
|
conn.SetReadLimit(33 * units.Mb)
|
|
return &Connection{
|
|
conn: conn,
|
|
}, nil
|
|
}
|
|
|
|
// WriteMessage writes arbitrary bytes to the websocket connection.
|
|
func (c *Connection) WriteMessage(
|
|
ctx context.Context, data []byte,
|
|
) (err error) {
|
|
if err = c.conn.Write(ctx, ws.MessageText, data); err != nil {
|
|
err = errorf.E("failed to write message: %w", err)
|
|
return
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// ReadMessage reads arbitrary bytes from the websocket connection into the provided buffer.
|
|
func (c *Connection) ReadMessage(
|
|
ctx context.Context, buf io.Writer,
|
|
) (err error) {
|
|
var reader io.Reader
|
|
if _, reader, err = c.conn.Reader(ctx); err != nil {
|
|
err = fmt.Errorf("failed to get reader: %w", err)
|
|
return
|
|
}
|
|
if _, err = io.Copy(buf, reader); err != nil {
|
|
err = fmt.Errorf("failed to read message: %w", err)
|
|
return
|
|
}
|
|
return
|
|
}
|
|
|
|
// Close closes the websocket connection.
|
|
func (c *Connection) Close() error {
|
|
return c.conn.Close(ws.StatusNormalClosure, "")
|
|
}
|
|
|
|
// Ping sends a ping message to the websocket connection.
|
|
func (c *Connection) Ping(ctx context.Context) error {
|
|
ctx, cancel := context.WithTimeoutCause(
|
|
ctx, time.Millisecond*800, errors.New("ping took too long"),
|
|
)
|
|
defer cancel()
|
|
return c.conn.Ping(ctx)
|
|
}
|