github.com/jlevesy/mattermost-server@v5.3.2-0.20181003190404-7468f35cb0c8+incompatible/app/analytics.go (about) 1 // Copyright (c) 2017-present Mattermost, Inc. All Rights Reserved. 2 // See License.txt for license information. 3 4 package app 5 6 import ( 7 "fmt" 8 9 "github.com/mattermost/mattermost-server/mlog" 10 "github.com/mattermost/mattermost-server/model" 11 "github.com/mattermost/mattermost-server/store" 12 ) 13 14 const ( 15 DAY_MILLISECONDS = 24 * 60 * 60 * 1000 16 MONTH_MILLISECONDS = 31 * DAY_MILLISECONDS 17 ) 18 19 func (a *App) GetAnalytics(name string, teamId string) (model.AnalyticsRows, *model.AppError) { 20 skipIntensiveQueries := false 21 var systemUserCount int64 22 r := <-a.Srv.Store.User().AnalyticsUniqueUserCount("") 23 if r.Err != nil { 24 return nil, r.Err 25 } 26 27 systemUserCount = r.Data.(int64) 28 if systemUserCount > int64(*a.Config().AnalyticsSettings.MaxUsersForStatistics) { 29 mlog.Debug(fmt.Sprintf("More than %v users on the system, intensive queries skipped", *a.Config().AnalyticsSettings.MaxUsersForStatistics)) 30 skipIntensiveQueries = true 31 } 32 33 if name == "standard" { 34 var rows model.AnalyticsRows = make([]*model.AnalyticsRow, 11) 35 rows[0] = &model.AnalyticsRow{Name: "channel_open_count", Value: 0} 36 rows[1] = &model.AnalyticsRow{Name: "channel_private_count", Value: 0} 37 rows[2] = &model.AnalyticsRow{Name: "post_count", Value: 0} 38 rows[3] = &model.AnalyticsRow{Name: "unique_user_count", Value: 0} 39 rows[4] = &model.AnalyticsRow{Name: "team_count", Value: 0} 40 rows[5] = &model.AnalyticsRow{Name: "total_websocket_connections", Value: 0} 41 rows[6] = &model.AnalyticsRow{Name: "total_master_db_connections", Value: 0} 42 rows[7] = &model.AnalyticsRow{Name: "total_read_db_connections", Value: 0} 43 rows[8] = &model.AnalyticsRow{Name: "daily_active_users", Value: 0} 44 rows[9] = &model.AnalyticsRow{Name: "monthly_active_users", Value: 0} 45 rows[10] = &model.AnalyticsRow{Name: "inactive_user_count", Value: 0} 46 47 openChan := a.Srv.Store.Channel().AnalyticsTypeCount(teamId, model.CHANNEL_OPEN) 48 privateChan := a.Srv.Store.Channel().AnalyticsTypeCount(teamId, model.CHANNEL_PRIVATE) 49 teamChan := a.Srv.Store.Team().AnalyticsTeamCount() 50 51 var userChan store.StoreChannel 52 var userInactiveChan store.StoreChannel 53 if teamId == "" { 54 userInactiveChan = a.Srv.Store.User().AnalyticsGetInactiveUsersCount() 55 } else { 56 userChan = a.Srv.Store.User().AnalyticsUniqueUserCount(teamId) 57 } 58 59 var postChan store.StoreChannel 60 if !skipIntensiveQueries { 61 postChan = a.Srv.Store.Post().AnalyticsPostCount(teamId, false, false) 62 } 63 64 dailyActiveChan := a.Srv.Store.User().AnalyticsActiveCount(DAY_MILLISECONDS) 65 monthlyActiveChan := a.Srv.Store.User().AnalyticsActiveCount(MONTH_MILLISECONDS) 66 67 r := <-openChan 68 if r.Err != nil { 69 return nil, r.Err 70 } 71 rows[0].Value = float64(r.Data.(int64)) 72 73 r = <-privateChan 74 if r.Err != nil { 75 return nil, r.Err 76 } 77 rows[1].Value = float64(r.Data.(int64)) 78 79 if postChan == nil { 80 rows[2].Value = -1 81 } else { 82 r := <-postChan 83 if r.Err != nil { 84 return nil, r.Err 85 } 86 rows[2].Value = float64(r.Data.(int64)) 87 } 88 89 if userChan == nil { 90 rows[3].Value = float64(systemUserCount) 91 } else { 92 r := <-userChan 93 if r.Err != nil { 94 return nil, r.Err 95 } 96 rows[3].Value = float64(r.Data.(int64)) 97 } 98 99 if userInactiveChan == nil { 100 rows[10].Value = -1 101 } else { 102 r := <-userInactiveChan 103 if r.Err != nil { 104 return nil, r.Err 105 } 106 rows[10].Value = float64(r.Data.(int64)) 107 } 108 109 r = <-teamChan 110 if r.Err != nil { 111 return nil, r.Err 112 } 113 rows[4].Value = float64(r.Data.(int64)) 114 115 // If in HA mode then aggregrate all the stats 116 if a.Cluster != nil && *a.Config().ClusterSettings.Enable { 117 stats, err := a.Cluster.GetClusterStats() 118 if err != nil { 119 return nil, err 120 } 121 122 totalSockets := a.TotalWebsocketConnections() 123 totalMasterDb := a.Srv.Store.TotalMasterDbConnections() 124 totalReadDb := a.Srv.Store.TotalReadDbConnections() 125 126 for _, stat := range stats { 127 totalSockets = totalSockets + stat.TotalWebsocketConnections 128 totalMasterDb = totalMasterDb + stat.TotalMasterDbConnections 129 totalReadDb = totalReadDb + stat.TotalReadDbConnections 130 } 131 132 rows[5].Value = float64(totalSockets) 133 rows[6].Value = float64(totalMasterDb) 134 rows[7].Value = float64(totalReadDb) 135 136 } else { 137 rows[5].Value = float64(a.TotalWebsocketConnections()) 138 rows[6].Value = float64(a.Srv.Store.TotalMasterDbConnections()) 139 rows[7].Value = float64(a.Srv.Store.TotalReadDbConnections()) 140 } 141 142 r = <-dailyActiveChan 143 if r.Err != nil { 144 return nil, r.Err 145 } 146 rows[8].Value = float64(r.Data.(int64)) 147 148 r = <-monthlyActiveChan 149 if r.Err != nil { 150 return nil, r.Err 151 } 152 rows[9].Value = float64(r.Data.(int64)) 153 154 return rows, nil 155 } else if name == "post_counts_day" { 156 if skipIntensiveQueries { 157 rows := model.AnalyticsRows{&model.AnalyticsRow{Name: "", Value: -1}} 158 return rows, nil 159 } 160 161 r := <-a.Srv.Store.Post().AnalyticsPostCountsByDay(teamId) 162 if r.Err != nil { 163 return nil, r.Err 164 } 165 return r.Data.(model.AnalyticsRows), nil 166 } else if name == "user_counts_with_posts_day" { 167 if skipIntensiveQueries { 168 rows := model.AnalyticsRows{&model.AnalyticsRow{Name: "", Value: -1}} 169 return rows, nil 170 } 171 172 r := <-a.Srv.Store.Post().AnalyticsUserCountsWithPostsByDay(teamId) 173 if r.Err != nil { 174 return nil, r.Err 175 } 176 return r.Data.(model.AnalyticsRows), nil 177 } else if name == "extra_counts" { 178 var rows model.AnalyticsRows = make([]*model.AnalyticsRow, 6) 179 rows[0] = &model.AnalyticsRow{Name: "file_post_count", Value: 0} 180 rows[1] = &model.AnalyticsRow{Name: "hashtag_post_count", Value: 0} 181 rows[2] = &model.AnalyticsRow{Name: "incoming_webhook_count", Value: 0} 182 rows[3] = &model.AnalyticsRow{Name: "outgoing_webhook_count", Value: 0} 183 rows[4] = &model.AnalyticsRow{Name: "command_count", Value: 0} 184 rows[5] = &model.AnalyticsRow{Name: "session_count", Value: 0} 185 186 iHookChan := a.Srv.Store.Webhook().AnalyticsIncomingCount(teamId) 187 oHookChan := a.Srv.Store.Webhook().AnalyticsOutgoingCount(teamId) 188 commandChan := a.Srv.Store.Command().AnalyticsCommandCount(teamId) 189 sessionChan := a.Srv.Store.Session().AnalyticsSessionCount() 190 191 var fileChan store.StoreChannel 192 var hashtagChan store.StoreChannel 193 if !skipIntensiveQueries { 194 fileChan = a.Srv.Store.Post().AnalyticsPostCount(teamId, true, false) 195 hashtagChan = a.Srv.Store.Post().AnalyticsPostCount(teamId, false, true) 196 } 197 198 if fileChan == nil { 199 rows[0].Value = -1 200 } else { 201 r := <-fileChan 202 if r.Err != nil { 203 return nil, r.Err 204 } 205 rows[0].Value = float64(r.Data.(int64)) 206 } 207 208 if hashtagChan == nil { 209 rows[1].Value = -1 210 } else { 211 r := <-hashtagChan 212 if r.Err != nil { 213 return nil, r.Err 214 } 215 rows[1].Value = float64(r.Data.(int64)) 216 } 217 218 r := <-iHookChan 219 if r.Err != nil { 220 return nil, r.Err 221 } 222 rows[2].Value = float64(r.Data.(int64)) 223 224 r = <-oHookChan 225 if r.Err != nil { 226 return nil, r.Err 227 } 228 rows[3].Value = float64(r.Data.(int64)) 229 230 r = <-commandChan 231 if r.Err != nil { 232 return nil, r.Err 233 } 234 rows[4].Value = float64(r.Data.(int64)) 235 236 r = <-sessionChan 237 if r.Err != nil { 238 return nil, r.Err 239 } 240 rows[5].Value = float64(r.Data.(int64)) 241 242 return rows, nil 243 } 244 245 return nil, nil 246 } 247 248 func (a *App) GetRecentlyActiveUsersForTeam(teamId string) (map[string]*model.User, *model.AppError) { 249 result := <-a.Srv.Store.User().GetRecentlyActiveUsersForTeam(teamId, 0, 100) 250 if result.Err != nil { 251 return nil, result.Err 252 } 253 254 users := result.Data.([]*model.User) 255 userMap := make(map[string]*model.User) 256 257 for _, user := range users { 258 userMap[user.Id] = user 259 } 260 261 return userMap, nil 262 } 263 264 func (a *App) GetRecentlyActiveUsersForTeamPage(teamId string, page, perPage int, asAdmin bool) ([]*model.User, *model.AppError) { 265 var users []*model.User 266 result := <-a.Srv.Store.User().GetRecentlyActiveUsersForTeam(teamId, page*perPage, perPage) 267 if result.Err != nil { 268 return nil, result.Err 269 } 270 users = result.Data.([]*model.User) 271 272 return a.sanitizeProfiles(users, asAdmin), nil 273 } 274 275 func (a *App) GetNewUsersForTeamPage(teamId string, page, perPage int, asAdmin bool) ([]*model.User, *model.AppError) { 276 var users []*model.User 277 result := <-a.Srv.Store.User().GetNewUsersForTeam(teamId, page*perPage, perPage) 278 if result.Err != nil { 279 return nil, result.Err 280 } 281 users = result.Data.([]*model.User) 282 283 return a.sanitizeProfiles(users, asAdmin), nil 284 }