mirror of
https://github.com/mattermost/mattermost.git
synced 2026-02-03 20:40:00 -05:00
* Standardize request.CTX parameter naming to rctx - Migrate 886 request.CTX parameters across 147 files to use consistent 'rctx' naming - Updated function signatures from 'c', 'ctx', and 'cancelContext' to 'rctx' - Updated function bodies to reference the new parameter names - Preserved underscore parameters unchanged as they are unused - Fixed method receiver context issue in store.go 🤖 Generated with [Claude Code](https://claude.ai/code) Co-Authored-By: Claude <noreply@anthropic.com> * Use request.CTX interface in batch worker * Manual fixes * Fix parameter naming * Add linter check --------- Co-authored-by: Claude <noreply@anthropic.com>
280 lines
9.6 KiB
Go
280 lines
9.6 KiB
Go
// Copyright (c) 2015-present Mattermost, Inc. All Rights Reserved.
|
|
// See LICENSE.txt for license information.
|
|
|
|
package app
|
|
|
|
import (
|
|
"errors"
|
|
"fmt"
|
|
"net/http"
|
|
"os"
|
|
"strconv"
|
|
"strings"
|
|
|
|
"github.com/mattermost/mattermost/server/public/model"
|
|
"github.com/mattermost/mattermost/server/public/shared/i18n"
|
|
"github.com/mattermost/mattermost/server/public/shared/mlog"
|
|
"github.com/mattermost/mattermost/server/public/shared/request"
|
|
"github.com/mattermost/mattermost/server/v8/channels/store"
|
|
)
|
|
|
|
const lastTrialNotificationTimeStamp = "LAST_TRIAL_NOTIFICATION_TIMESTAMP"
|
|
const lastUpgradeNotificationTimeStamp = "LAST_UPGRADE_NOTIFICATION_TIMESTAMP"
|
|
const defaultNotifyAdminCoolOffDays = 14
|
|
|
|
func (a *App) SaveAdminNotification(userId string, notifyData *model.NotifyAdminToUpgradeRequest) *model.AppError {
|
|
requiredFeature := notifyData.RequiredFeature
|
|
requiredPlan := notifyData.RequiredPlan
|
|
trial := notifyData.TrialNotification
|
|
|
|
isUserAlreadyNotified := a.UserAlreadyNotifiedOnRequiredFeature(userId, requiredFeature)
|
|
if isUserAlreadyNotified {
|
|
return model.NewAppError("app.SaveAdminNotification", "api.cloud.notify_admin_to_upgrade_error.already_notified", nil, "", http.StatusForbidden)
|
|
}
|
|
|
|
_, appErr := a.SaveAdminNotifyData(&model.NotifyAdminData{
|
|
UserId: userId,
|
|
RequiredPlan: requiredPlan,
|
|
RequiredFeature: requiredFeature,
|
|
Trial: trial,
|
|
})
|
|
|
|
if appErr != nil {
|
|
return appErr
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func (a *App) DoCheckForAdminNotifications(trial bool) *model.AppError {
|
|
ctx := request.EmptyContext(a.Srv().Log())
|
|
currentSKU := "starter"
|
|
license := a.Srv().License()
|
|
if license != nil {
|
|
currentSKU = license.SkuShortName
|
|
}
|
|
|
|
workspaceName := ""
|
|
|
|
return a.SendNotifyAdminPosts(ctx, workspaceName, currentSKU, trial)
|
|
}
|
|
|
|
func (a *App) SaveAdminNotifyData(data *model.NotifyAdminData) (*model.NotifyAdminData, *model.AppError) {
|
|
d, err := a.Srv().Store().NotifyAdmin().Save(data)
|
|
if err != nil {
|
|
var nfErr *store.ErrNotFound
|
|
switch {
|
|
case errors.As(err, &nfErr):
|
|
return nil, model.NewAppError("SaveAdminNotifyData", "app.notify_admin.save.app_error", nil, "", http.StatusNotFound).Wrap(nfErr)
|
|
default:
|
|
return nil, model.NewAppError("SaveAdminNotifyData", "app.notify_admin.save.app_error", nil, "", http.StatusInternalServerError).Wrap(err)
|
|
}
|
|
}
|
|
|
|
return d, nil
|
|
}
|
|
|
|
func filterNotificationData(data []*model.NotifyAdminData, test func(*model.NotifyAdminData) bool) (ret []*model.NotifyAdminData) {
|
|
for _, d := range data {
|
|
if test(d) {
|
|
ret = append(ret, d)
|
|
}
|
|
}
|
|
return
|
|
}
|
|
|
|
func (a *App) SendNotifyAdminPosts(rctx request.CTX, workspaceName string, currentSKU string, trial bool) *model.AppError {
|
|
if !a.CanNotifyAdmin(rctx, trial) {
|
|
return model.NewAppError("SendNotifyAdminPosts", "app.notify_admin.send_notification_post.app_error", nil, "Cannot notify yet", http.StatusForbidden)
|
|
}
|
|
|
|
sysadmins, appErr := a.GetUsersFromProfiles(&model.UserGetOptions{
|
|
Page: 0,
|
|
PerPage: 100,
|
|
Role: model.SystemAdminRoleId,
|
|
Inactive: false,
|
|
})
|
|
if appErr != nil {
|
|
return appErr
|
|
}
|
|
|
|
systemBot, appErr := a.GetSystemBot(rctx)
|
|
if appErr != nil {
|
|
return appErr
|
|
}
|
|
|
|
now := model.GetMillis()
|
|
|
|
data, err := a.Srv().Store().NotifyAdmin().Get(trial)
|
|
if err != nil {
|
|
return model.NewAppError("SendNotifyAdminPosts", "app.notify_admin.send_notification_post.app_error", nil, "", http.StatusInternalServerError).Wrap(err)
|
|
}
|
|
|
|
data = filterNotificationData(data, func(nad *model.NotifyAdminData) bool { return nad.RequiredPlan != currentSKU })
|
|
|
|
if len(data) == 0 {
|
|
rctx.Logger().Warn("No notification data available")
|
|
return nil
|
|
}
|
|
|
|
userBasedPaidFeatureData := a.groupNotifyAdminByUser(data)
|
|
featureBasedData := a.groupNotifyAdminByPaidFeature(data)
|
|
pluginBasedData := a.groupNotifyAdminByPlugin(data)
|
|
|
|
for _, admin := range sysadmins {
|
|
if len(userBasedPaidFeatureData) > 0 && len(featureBasedData) > 0 {
|
|
a.upgradePlanAdminNotifyPost(rctx, workspaceName, userBasedPaidFeatureData, featureBasedData, systemBot, admin, trial)
|
|
}
|
|
}
|
|
|
|
a.FinishSendAdminNotifyPost(rctx, trial, now, pluginBasedData)
|
|
return nil
|
|
}
|
|
|
|
func (a *App) upgradePlanAdminNotifyPost(rctx request.CTX, workspaceName string, userBasedData map[string][]*model.NotifyAdminData, featureBasedData map[model.MattermostFeature][]*model.NotifyAdminData, systemBot *model.Bot, admin *model.User, trial bool) {
|
|
props := make(model.StringInterface)
|
|
T := i18n.GetUserTranslations(admin.Locale)
|
|
|
|
message := T("app.cloud.upgrade_plan_bot_message", map[string]any{"UsersNum": len(userBasedData), "WorkspaceName": workspaceName})
|
|
if len(userBasedData) == 1 {
|
|
message = T("app.cloud.upgrade_plan_bot_message_single", map[string]any{"UsersNum": len(userBasedData), "WorkspaceName": workspaceName}) // todo (allan): investigate if translations library can do this
|
|
}
|
|
if trial {
|
|
message = T("app.cloud.trial_plan_bot_message", map[string]any{"UsersNum": len(userBasedData), "WorkspaceName": workspaceName})
|
|
if len(userBasedData) == 1 {
|
|
message = T("app.cloud.trial_plan_bot_message_single", map[string]any{"UsersNum": len(userBasedData), "WorkspaceName": workspaceName})
|
|
}
|
|
}
|
|
|
|
channel, appErr := a.GetOrCreateDirectChannel(rctx, systemBot.UserId, admin.Id)
|
|
if appErr != nil {
|
|
rctx.Logger().Warn("Error getting direct channel", mlog.Err(appErr))
|
|
return
|
|
}
|
|
|
|
post := &model.Post{
|
|
Message: message,
|
|
UserId: systemBot.UserId,
|
|
ChannelId: channel.Id,
|
|
Type: fmt.Sprintf("%sup_notification", model.PostCustomTypePrefix), // webapp will have to create renderer for this custom post type
|
|
|
|
}
|
|
|
|
props["requested_features"] = featureBasedData
|
|
props["trial"] = trial
|
|
post.SetProps(props)
|
|
|
|
_, appErr = a.CreatePost(rctx, post, channel, model.CreatePostFlags{SetOnline: true})
|
|
|
|
if appErr != nil {
|
|
rctx.Logger().Warn("Error creating post", mlog.Err(appErr))
|
|
}
|
|
}
|
|
|
|
func (a *App) UserAlreadyNotifiedOnRequiredFeature(user string, feature model.MattermostFeature) bool {
|
|
data, err := a.Srv().Store().NotifyAdmin().GetDataByUserIdAndFeature(user, feature)
|
|
if err != nil {
|
|
return false
|
|
}
|
|
if len(data) > 0 {
|
|
return true // if we find data, it means this user already notified on the need for this feature
|
|
}
|
|
|
|
return false
|
|
}
|
|
|
|
func (a *App) CanNotifyAdmin(rctx request.CTX, trial bool) bool {
|
|
systemVarName := lastUpgradeNotificationTimeStamp
|
|
if trial {
|
|
systemVarName = lastTrialNotificationTimeStamp
|
|
}
|
|
|
|
sysVal, sysValErr := a.Srv().Store().System().GetByName(systemVarName)
|
|
if sysValErr != nil {
|
|
var nfErr *store.ErrNotFound
|
|
if errors.As(sysValErr, &nfErr) { // if no timestamps have been recorded before, system is free to notify
|
|
return true
|
|
}
|
|
rctx.Logger().Error("Cannot notify", mlog.Err(sysValErr))
|
|
return false
|
|
}
|
|
|
|
lastNotificationTimestamp, err := strconv.ParseFloat(sysVal.Value, 64)
|
|
if err != nil {
|
|
rctx.Logger().Error("Cannot notify", mlog.Err(err))
|
|
return false
|
|
}
|
|
|
|
coolOffPeriodDaysEnv := os.Getenv("MM_NOTIFY_ADMIN_COOL_OFF_DAYS")
|
|
coolOffPeriodDays, parseError := strconv.ParseFloat(coolOffPeriodDaysEnv, 64)
|
|
if parseError != nil {
|
|
coolOffPeriodDays = defaultNotifyAdminCoolOffDays
|
|
}
|
|
daysToMillis := coolOffPeriodDays * 24 * 60 * 60 * 1000
|
|
timeDiff := model.GetMillis() - int64(lastNotificationTimestamp)
|
|
return timeDiff >= int64(daysToMillis)
|
|
}
|
|
|
|
func (a *App) FinishSendAdminNotifyPost(rctx request.CTX, trial bool, now int64, pluginBasedData map[string][]*model.NotifyAdminData) {
|
|
systemVarName := lastUpgradeNotificationTimeStamp
|
|
if trial {
|
|
systemVarName = lastTrialNotificationTimeStamp
|
|
}
|
|
|
|
val := strconv.FormatInt(model.GetMillis(), 10)
|
|
sysVar := &model.System{Name: systemVarName, Value: val}
|
|
if err := a.Srv().Store().System().SaveOrUpdate(sysVar); err != nil {
|
|
rctx.Logger().Error("Unable to finish send admin notify post job", mlog.Err(err))
|
|
}
|
|
|
|
// All the requested features notifications are now sent in a post and can safely be removed except
|
|
// the plugin notify admin. We keep it as we do not want the same user to send the notification for the same plugin.
|
|
// We update the NotifyAdmin SentAt to keep track of it.
|
|
for pluginId := range pluginBasedData {
|
|
notifications := pluginBasedData[pluginId]
|
|
for _, notification := range notifications {
|
|
requiredFeature := notification.RequiredFeature
|
|
requiredPlan := notification.RequiredPlan
|
|
userId := notification.UserId
|
|
if err := a.Srv().Store().NotifyAdmin().Update(userId, requiredPlan, requiredFeature, now); err != nil {
|
|
rctx.Logger().Error("Unable to update SentAt for work template feature", mlog.Err(err))
|
|
}
|
|
}
|
|
}
|
|
|
|
if err := a.Srv().Store().NotifyAdmin().DeleteBefore(trial, now); err != nil {
|
|
rctx.Logger().Error("Unable to finish send admin notify post job", mlog.Err(err))
|
|
}
|
|
}
|
|
|
|
func (a *App) groupNotifyAdminByUser(data []*model.NotifyAdminData) map[string][]*model.NotifyAdminData {
|
|
userBasedPaidFeatureData := make(map[string][]*model.NotifyAdminData)
|
|
for _, d := range data {
|
|
userBasedPaidFeatureData[d.UserId] = append(userBasedPaidFeatureData[d.UserId], d)
|
|
}
|
|
return userBasedPaidFeatureData
|
|
}
|
|
|
|
func (a *App) groupNotifyAdminByPaidFeature(data []*model.NotifyAdminData) map[model.MattermostFeature][]*model.NotifyAdminData {
|
|
myMap := make(map[model.MattermostFeature][]*model.NotifyAdminData)
|
|
for _, d := range data {
|
|
if strings.HasPrefix(string(d.RequiredFeature), string(model.PluginFeature)) {
|
|
continue
|
|
}
|
|
myMap[d.RequiredFeature] = append(myMap[d.RequiredFeature], d)
|
|
}
|
|
return myMap
|
|
}
|
|
|
|
func (a *App) groupNotifyAdminByPlugin(data []*model.NotifyAdminData) map[string][]*model.NotifyAdminData {
|
|
myMap := make(map[string][]*model.NotifyAdminData)
|
|
for _, d := range data {
|
|
if strings.HasPrefix(string(d.RequiredFeature), string(model.PluginFeature)) {
|
|
plugins := strings.SplitSeq(d.RequiredPlan, ",")
|
|
for plugin := range plugins {
|
|
myMap[plugin] = append(myMap[plugin], d)
|
|
}
|
|
}
|
|
}
|
|
return myMap
|
|
}
|