Files
mattermost/model/websocket_client_test.go
Agniva De Sarker abce5d620f MM-25439: Skip SuddenClose flaky test (#14628)
Automatic Merge
2020-05-22 02:44:42 -07:00

152 lines
3.7 KiB
Go

// Copyright (c) 2015-present Mattermost, Inc. All Rights Reserved.
// See LICENSE.txt for license information.
package model
import (
"net/http"
"net/http/httptest"
"strings"
"testing"
"time"
"github.com/gorilla/websocket"
"github.com/stretchr/testify/assert"
"github.com/stretchr/testify/require"
)
func dummyWebsocketHandler(t *testing.T) http.HandlerFunc {
return func(w http.ResponseWriter, req *http.Request) {
upgrader := &websocket.Upgrader{
ReadBufferSize: 1024,
WriteBufferSize: 1024,
}
conn, err := upgrader.Upgrade(w, req, nil)
require.Nil(t, err)
var buf []byte
for {
_, buf, err = conn.ReadMessage()
if err != nil {
break
}
t.Logf("%s\n", buf)
err = conn.WriteMessage(websocket.PingMessage, []byte("ping"))
if err != nil {
break
}
}
}
}
// TestWebSocketRace needs to be run with -race to verify that
// there is no race.
func TestWebSocketRace(t *testing.T) {
s := httptest.NewServer(dummyWebsocketHandler(t))
defer s.Close()
url := strings.Replace(s.URL, "http://", "ws://", 1)
cli, err := NewWebSocketClient4(url, "authToken")
require.Nil(t, err)
cli.Listen()
for i := 0; i < 10; i++ {
time.Sleep(500 * time.Millisecond)
cli.UserTyping("channel", "parentId")
}
}
func TestWebSocketClose(t *testing.T) {
// This fails in SuddenClose because we check for closing the writeChan
// only after waiting the closure of Event and Response channels.
// Therefore, it is still racy and can panic. There is no use chasing this
// again because it will be completely overhauled in v6.
t.Skip("Skipping the test. Will be changed in v6.")
s := httptest.NewServer(dummyWebsocketHandler(t))
defer s.Close()
url := strings.Replace(s.URL, "http://", "ws://", 1)
// Check whether the Event and Response channels
// have been closed or not.
waitClose := func(doneChan chan struct{}) int {
numClosed := 0
timeout := time.After(300 * time.Millisecond)
for {
select {
case <-doneChan:
numClosed++
if numClosed == 2 {
return numClosed
}
case <-timeout:
require.Fail(t, "timed out waiting for channels to be closed")
return numClosed
}
}
}
checkWriteChan := func(writeChan chan writeMessage) {
defer func() {
if x := recover(); x == nil {
require.Fail(t, "should have panicked due to closing a closed channel")
}
}()
close(writeChan)
}
waitForResponses := func(doneChan chan struct{}, cli *WebSocketClient) {
go func() {
for range cli.EventChannel {
}
doneChan <- struct{}{}
}()
go func() {
for range cli.ResponseChannel {
}
doneChan <- struct{}{}
}()
}
t.Run("SuddenClose", func(t *testing.T) {
cli, err := NewWebSocketClient4(url, "authToken")
require.Nil(t, err)
cli.Listen()
doneChan := make(chan struct{}, 2)
waitForResponses(doneChan, cli)
cli.UserTyping("channelId", "parentId")
cli.Conn.Close()
numClosed := waitClose(doneChan)
assert.Equal(t, 2, numClosed, "unexpected number of channels closed")
// Check whether the write channel was closed or not.
checkWriteChan(cli.writeChan)
require.NotNil(t, cli.ListenError, "non-nil listen error")
assert.Equal(t, "model.websocket_client.connect_fail.app_error", cli.ListenError.Id, "unexpected error id")
})
t.Run("ExplicitClose", func(t *testing.T) {
cli, err := NewWebSocketClient4(url, "authToken")
require.Nil(t, err)
cli.Listen()
doneChan := make(chan struct{}, 2)
waitForResponses(doneChan, cli)
cli.UserTyping("channelId", "parentId")
cli.Close()
numClosed := waitClose(doneChan)
assert.Equal(t, 2, numClosed, "unexpected number of channels closed")
// Check whether the write channel was closed or not.
checkWriteChan(cli.writeChan)
})
}