github.com/crspeller/mattermost-server@v0.0.0-20190328001957-a200beb3d111/app/websocket_router.go (about)

     1  // Copyright (c) 2016-present Mattermost, Inc. All Rights Reserved.
     2  // See License.txt for license information.
     3  
     4  package app
     5  
     6  import (
     7  	"fmt"
     8  	"net/http"
     9  
    10  	"github.com/crspeller/mattermost-server/mlog"
    11  	"github.com/crspeller/mattermost-server/model"
    12  	"github.com/crspeller/mattermost-server/utils"
    13  )
    14  
    15  type webSocketHandler interface {
    16  	ServeWebSocket(*WebConn, *model.WebSocketRequest)
    17  }
    18  
    19  type WebSocketRouter struct {
    20  	app      *App
    21  	handlers map[string]webSocketHandler
    22  }
    23  
    24  func (wr *WebSocketRouter) Handle(action string, handler webSocketHandler) {
    25  	wr.handlers[action] = handler
    26  }
    27  
    28  func (wr *WebSocketRouter) ServeWebSocket(conn *WebConn, r *model.WebSocketRequest) {
    29  	if r.Action == "" {
    30  		err := model.NewAppError("ServeWebSocket", "api.web_socket_router.no_action.app_error", nil, "", http.StatusBadRequest)
    31  		ReturnWebSocketError(conn, r, err)
    32  		return
    33  	}
    34  
    35  	if r.Seq <= 0 {
    36  		err := model.NewAppError("ServeWebSocket", "api.web_socket_router.bad_seq.app_error", nil, "", http.StatusBadRequest)
    37  		ReturnWebSocketError(conn, r, err)
    38  		return
    39  	}
    40  
    41  	if r.Action == model.WEBSOCKET_AUTHENTICATION_CHALLENGE {
    42  		if conn.GetSessionToken() != "" {
    43  			return
    44  		}
    45  
    46  		token, ok := r.Data["token"].(string)
    47  		if !ok {
    48  			conn.WebSocket.Close()
    49  			return
    50  		}
    51  
    52  		session, err := wr.app.GetSession(token)
    53  		if err != nil {
    54  			conn.WebSocket.Close()
    55  			return
    56  		}
    57  
    58  		wr.app.Srv.Go(func() {
    59  			wr.app.SetStatusOnline(session.UserId, false)
    60  			wr.app.UpdateLastActivityAtIfNeeded(*session)
    61  		})
    62  
    63  		conn.SetSession(session)
    64  		conn.SetSessionToken(session.Token)
    65  		conn.UserId = session.UserId
    66  
    67  		wr.app.HubRegister(conn)
    68  
    69  		resp := model.NewWebSocketResponse(model.STATUS_OK, r.Seq, nil)
    70  		conn.Send <- resp
    71  
    72  		return
    73  	}
    74  
    75  	if !conn.IsAuthenticated() {
    76  		err := model.NewAppError("ServeWebSocket", "api.web_socket_router.not_authenticated.app_error", nil, "", http.StatusUnauthorized)
    77  		ReturnWebSocketError(conn, r, err)
    78  		return
    79  	}
    80  
    81  	handler, ok := wr.handlers[r.Action]
    82  	if !ok {
    83  		err := model.NewAppError("ServeWebSocket", "api.web_socket_router.bad_action.app_error", nil, "", http.StatusInternalServerError)
    84  		ReturnWebSocketError(conn, r, err)
    85  		return
    86  	}
    87  
    88  	handler.ServeWebSocket(conn, r)
    89  }
    90  
    91  func ReturnWebSocketError(conn *WebConn, r *model.WebSocketRequest, err *model.AppError) {
    92  	mlog.Error(fmt.Sprintf("websocket routing error: seq=%v uid=%v %v [details: %v]", r.Seq, conn.UserId, err.SystemMessage(utils.T), err.DetailedError))
    93  
    94  	err.DetailedError = ""
    95  	errorResp := model.NewWebSocketError(r.Seq, err)
    96  
    97  	conn.Send <- errorResp
    98  }