Newer
Older
"net/http"
"net/http/httptest"
"os"
"os/exec"
"github.com/golang/protobuf/ptypes"
"github.com/golang/protobuf/ptypes/duration"
"nhooyr.io/websocket/internal/test/wstest"
"nhooyr.io/websocket/internal/test/xrand"
copts := func() *websocket.CompressionOptions {
return &websocket.CompressionOptions{
Mode: websocket.CompressionMode(xrand.Int(int(websocket.CompressionDisabled) + 1)),
Threshold: xrand.Int(9999),
}
}
tt, c1, c2 := newConnTest(t, &websocket.DialOptions{
CompressionOptions: copts(),
err := c1.Close(websocket.StatusNormalClosure, "")
assert.Contains(t, err, "failed to marshal close frame: status code StatusCode(-1) cannot be set")
c1.CloseRead(tt.ctx)
c2.CloseRead(tt.ctx)
err := c1.Close(websocket.StatusNormalClosure, "")
ctx, cancel := context.WithTimeout(tt.ctx, time.Millisecond*100)
defer cancel()
err := c1.Ping(ctx)
assert.Contains(t, err, "failed to wait for pong")
})
t.Run("concurrentWrite", func(t *testing.T) {
msg := xrand.Bytes(xrand.Int(9999))
const count = 100
errs := make(chan error, count)
for i := 0; i < count; i++ {
go func() {
errs <- c1.Write(tt.ctx, websocket.MessageBinary, msg)
}()
}
for i := 0; i < count; i++ {
err := <-errs
err := c1.Close(websocket.StatusNormalClosure, "")
t.Run("concurrentWriteError", func(t *testing.T) {
_, err := c1.Writer(tt.ctx, websocket.MessageText)
ctx, cancel := context.WithTimeout(context.Background(), time.Millisecond*100)
defer cancel()
err = c1.Write(ctx, websocket.MessageText, []byte("x"))
assert.Equal(t, "write error", context.DeadlineExceeded, err)
n1 := websocket.NetConn(tt.ctx, c1, websocket.MessageBinary)
n2 := websocket.NetConn(tt.ctx, c2, websocket.MessageBinary)
// Does not give any confidence but at least ensures no crashes.
assert.Equal(t, "remote addr", n1.RemoteAddr(), n1.LocalAddr())
assert.Equal(t, "remote addr string", "websocket/unknown-addr", n1.RemoteAddr().String())
assert.Equal(t, "remote addr network", "websocket", n1.RemoteAddr().Network())
errs := xsync.Go(func() error {
_, err := n2.Write([]byte("hello"))
if err != nil {
return err
}
return n2.Close()
})
b, err := ioutil.ReadAll(n1)
n1 := websocket.NetConn(tt.ctx, c1, websocket.MessageBinary)
n2 := websocket.NetConn(tt.ctx, c2, websocket.MessageText)
errs := xsync.Go(func() error {
_, err := n2.Write([]byte("hello"))
if err != nil {
return err
}
return nil
})
assert.Contains(t, err, `unexpected frame type read (expected MessageBinary): MessageText`)
assert.Success(t, err)
assert.Equal(t, "read msg", exp, act)
assert.Success(t, err)
assert.Equal(t, "read msg", exp, act)
var wg sync.WaitGroup
s := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
wg.Add(1)
defer wg.Done()
c, err := websocket.Accept(w, r, &websocket.AcceptOptions{
Subprotocols: []string{"echo"},
InsecureSkipVerify: true,
})
err = assertCloseStatus(websocket.StatusNormalClosure, err)
if err != nil {
t.Errorf("echo server failed: %v", err)
return
ctx, cancel := context.WithTimeout(context.Background(), time.Minute)
cmd := exec.CommandContext(ctx, "go", "test", "-exec=wasmbrowsertest", "./...")
cmd.Env = append(os.Environ(), "GOOS=js", "GOARCH=wasm", fmt.Sprintf("WS_ECHO_SERVER_URL=%v", wstest.URL(s)))
b, err := cmd.CombinedOutput()
if err != nil {
t.Fatalf("wasm test binary failed: %v:\n%s", err, b)
func assertCloseStatus(exp websocket.StatusCode, err error) error {
if websocket.CloseStatus(err) == -1 {
return xerrors.Errorf("expected websocket.CloseError: %T %v", err, err)
}
if websocket.CloseStatus(err) != exp {
return xerrors.Errorf("expected close status %v but got ", exp, err)
ctx context.Context
doneFuncs []func()
}
func newConnTest(t testing.TB, dialOpts *websocket.DialOptions, acceptOpts *websocket.AcceptOptions) (tt *connTest, c1, c2 *websocket.Conn) {
if t, ok := t.(*testing.T); ok {
t.Parallel()
}
ctx, cancel := context.WithTimeout(context.Background(), time.Second*30)
c1, c2, err := wstest.Pipe(dialOpts, acceptOpts)
assert.Success(tt.t, err)
tt.appendDone(func() {
c2.Close(websocket.StatusInternalError, "")
c1.Close(websocket.StatusInternalError, "")
})
return tt, c1, c2
tt.doneFuncs = append(tt.doneFuncs, f)
}
for i := len(tt.doneFuncs) - 1; i >= 0; i-- {
tt.doneFuncs[i]()
}
}
func (tt *connTest) goEchoLoop(c *websocket.Conn) {
ctx, cancel := context.WithCancel(tt.ctx)
echoLoopErr := xsync.Go(func() error {
err := wstest.EchoLoop(ctx, c)
return assertCloseStatus(websocket.StatusNormalClosure, err)
})
tt.appendDone(func() {
cancel()
err := <-echoLoopErr
if err != nil {
tt.t.Errorf("echo loop error: %v", err)
}
})
}
func (tt *connTest) goDiscardLoop(c *websocket.Conn) {
ctx, cancel := context.WithCancel(tt.ctx)
discardLoopErr := xsync.Go(func() error {
defer c.Close(websocket.StatusInternalError, "")
for {
_, _, err := c.Read(ctx)
if err != nil {
return assertCloseStatus(websocket.StatusNormalClosure, err)
}
}
})
tt.appendDone(func() {
cancel()
err := <-discardLoopErr
if err != nil {
tt.t.Errorf("discard loop error: %v", err)
}
})
}
func BenchmarkConn(b *testing.B) {
var benchCases = []struct {
name string
mode websocket.CompressionMode
}{
{
mode: websocket.CompressionContextTakeover,
},
{
mode: websocket.CompressionNoContextTakeover,
},
}
for _, bc := range benchCases {
b.Run(bc.name, func(b *testing.B) {
bb, c1, c2 := newConnTest(b, &websocket.DialOptions{
CompressionOptions: &websocket.CompressionOptions{Mode: bc.mode},
}, &websocket.AcceptOptions{
CompressionOptions: &websocket.CompressionOptions{Mode: bc.mode},
})
msg := []byte(strings.Repeat("1234", 128))
readBuf := make([]byte, len(msg))
writes := make(chan struct{})
defer close(writes)
werrs := make(chan error)
go func() {
for range writes {
werrs <- c1.Write(bb.ctx, websocket.MessageText, msg)
b.ReportAllocs()
b.ResetTimer()
for i := 0; i < b.N; i++ {
typ, r, err := c1.Reader(bb.ctx)
if err != nil {
b.Fatal(err)
}
if websocket.MessageText != typ {
assert.Equal(b, "data type", websocket.MessageText, typ)
}
_, err = io.ReadFull(r, readBuf)
if err != nil {
b.Fatal(err)
}
n2, err := r.Read(readBuf)
if err != io.EOF {
assert.Equal(b, "read err", io.EOF, err)
}
if n2 != 0 {
assert.Equal(b, "n2", 0, n2)
}
if !bytes.Equal(msg, readBuf) {
assert.Equal(b, "msg", msg, readBuf)
}
err = <-werrs
if err != nil {
b.Fatal(err)
}
}
b.StopTimer()
err := c1.Close(websocket.StatusNormalClosure, "")
assert.Success(b, err)
})
}
}
func TestCompression(t *testing.T) {
t.Parallel()
}