summaryrefslogtreecommitdiffstats
path: root/api/status.go
diff options
context:
space:
mode:
authorJoram Wilander <jwawilander@gmail.com>2016-07-18 11:10:03 -0400
committerGitHub <noreply@github.com>2016-07-18 11:10:03 -0400
commitc0ab2636d699c8544ce03a58f61b95cfd66ff7ce (patch)
treec7d07934e0ff1a75aafb097a184ae150888199c0 /api/status.go
parent180adc79af3d14de6ce62f6e687a6735db3fe82f (diff)
downloadchat-c0ab2636d699c8544ce03a58f61b95cfd66ff7ce.tar.gz
chat-c0ab2636d699c8544ce03a58f61b95cfd66ff7ce.tar.bz2
chat-c0ab2636d699c8544ce03a58f61b95cfd66ff7ce.zip
PLT-2241 Refactored statuses into a more real-time system (#3573)
* Refactored statuses into a more real-time system * Updated package.json with correct commit and fixed minor bug * Minor updates to statuses based on feedback * When setting status online, update only LastActivityAt if status already exists
Diffstat (limited to 'api/status.go')
-rw-r--r--api/status.go155
1 files changed, 155 insertions, 0 deletions
diff --git a/api/status.go b/api/status.go
new file mode 100644
index 000000000..88f024f4e
--- /dev/null
+++ b/api/status.go
@@ -0,0 +1,155 @@
+// Copyright (c) 2016 Mattermost, Inc. All Rights Reserved.
+// See License.txt for license information.
+
+package api
+
+import (
+ "net/http"
+
+ l4g "github.com/alecthomas/log4go"
+ "github.com/mattermost/platform/model"
+ "github.com/mattermost/platform/store"
+ "github.com/mattermost/platform/utils"
+)
+
+func InitStatus() {
+ l4g.Debug(utils.T("api.status.init.debug"))
+
+ BaseRoutes.Users.Handle("/status", ApiUserRequiredActivity(getStatusesHttp, false)).Methods("GET")
+ BaseRoutes.WebSocket.Handle("get_statuses", ApiWebSocketHandler(getStatusesWebSocket))
+}
+
+func getStatusesHttp(c *Context, w http.ResponseWriter, r *http.Request) {
+ statusMap, err := GetAllStatuses()
+ if err != nil {
+ c.Err = err
+ return
+ }
+
+ w.Write([]byte(model.StringInterfaceToJson(statusMap)))
+}
+
+func getStatusesWebSocket(req *model.WebSocketRequest) (map[string]interface{}, *model.AppError) {
+ statusMap, err := GetAllStatuses()
+ if err != nil {
+ return nil, err
+ }
+
+ return statusMap, nil
+}
+
+func GetAllStatuses() (map[string]interface{}, *model.AppError) {
+ if result := <-Srv.Store.Status().GetOnlineAway(); result.Err != nil {
+ return nil, result.Err
+ } else {
+ statuses := result.Data.([]*model.Status)
+
+ statusMap := map[string]interface{}{}
+ for _, s := range statuses {
+ statusMap[s.UserId] = s.Status
+ }
+
+ return statusMap, nil
+ }
+}
+
+func SetStatusOnline(userId string, sessionId string) {
+ broadcast := false
+ saveStatus := false
+
+ var status *model.Status
+ var err *model.AppError
+ if status, err = GetStatus(userId); err != nil {
+ status = &model.Status{userId, model.STATUS_ONLINE, model.GetMillis()}
+ broadcast = true
+ saveStatus = true
+ } else {
+ if status.Status != model.STATUS_ONLINE {
+ broadcast = true
+ }
+ status.Status = model.STATUS_ONLINE
+ status.LastActivityAt = model.GetMillis()
+ }
+
+ statusCache.Add(status.UserId, status)
+
+ achan := Srv.Store.Session().UpdateLastActivityAt(sessionId, model.GetMillis())
+
+ var schan store.StoreChannel
+ if saveStatus {
+ schan = Srv.Store.Status().SaveOrUpdate(status)
+ } else {
+ schan = Srv.Store.Status().UpdateLastActivityAt(status.UserId, status.LastActivityAt)
+ }
+
+ if result := <-achan; result.Err != nil {
+ l4g.Error(utils.T("api.status.last_activity.error"), userId, sessionId, result.Err)
+ }
+
+ if result := <-schan; result.Err != nil {
+ l4g.Error(utils.T("api.status.save_status.error"), userId, result.Err)
+ }
+
+ if broadcast {
+ event := model.NewWebSocketEvent("", "", status.UserId, model.WEBSOCKET_EVENT_STATUS_CHANGE)
+ event.Add("status", model.STATUS_ONLINE)
+ go Publish(event)
+ }
+}
+
+func SetStatusOffline(userId string) {
+ status := &model.Status{userId, model.STATUS_OFFLINE, model.GetMillis()}
+
+ statusCache.Add(status.UserId, status)
+
+ if result := <-Srv.Store.Status().SaveOrUpdate(status); result.Err != nil {
+ l4g.Error(utils.T("api.status.save_status.error"), userId, result.Err)
+ }
+
+ event := model.NewWebSocketEvent("", "", status.UserId, model.WEBSOCKET_EVENT_STATUS_CHANGE)
+ event.Add("status", model.STATUS_OFFLINE)
+ go Publish(event)
+}
+
+func SetStatusAwayIfNeeded(userId string) {
+ status, err := GetStatus(userId)
+ if err != nil {
+ status = &model.Status{userId, model.STATUS_OFFLINE, 0}
+ }
+
+ if status.Status == model.STATUS_AWAY {
+ return
+ }
+
+ if !IsUserAway(status.LastActivityAt) {
+ return
+ }
+
+ status.Status = model.STATUS_AWAY
+
+ statusCache.Add(status.UserId, status)
+
+ if result := <-Srv.Store.Status().SaveOrUpdate(status); result.Err != nil {
+ l4g.Error(utils.T("api.status.save_status.error"), userId, result.Err)
+ }
+
+ event := model.NewWebSocketEvent("", "", status.UserId, model.WEBSOCKET_EVENT_STATUS_CHANGE)
+ event.Add("status", model.STATUS_AWAY)
+ go Publish(event)
+}
+
+func GetStatus(userId string) (*model.Status, *model.AppError) {
+ if status, ok := statusCache.Get(userId); ok {
+ return status.(*model.Status), nil
+ }
+
+ if result := <-Srv.Store.Status().Get(userId); result.Err != nil {
+ return nil, result.Err
+ } else {
+ return result.Data.(*model.Status), nil
+ }
+}
+
+func IsUserAway(lastActivityAt int64) bool {
+ return model.GetMillis()-lastActivityAt >= *utils.Cfg.TeamSettings.UserStatusAwayTimeout*1000
+}