mirror of
https://github.com/mattermost/mattermost.git
synced 2025-02-25 18:55:24 -06:00
* [MM-36198] - Activating a user in System Console does not respect cloud user limits * feedback impl * improve error check
284 lines
9.0 KiB
Go
284 lines
9.0 KiB
Go
// Copyright (c) 2015-present Mattermost, Inc. All Rights Reserved.
|
|
// See LICENSE.txt for license information.
|
|
|
|
package app
|
|
|
|
import (
|
|
"fmt"
|
|
"net/http"
|
|
"time"
|
|
|
|
"github.com/mattermost/mattermost-server/v5/app/request"
|
|
"github.com/mattermost/mattermost-server/v5/model"
|
|
"github.com/mattermost/mattermost-server/v5/shared/mlog"
|
|
)
|
|
|
|
func (a *App) getSysAdminsEmailRecipients() ([]*model.User, *model.AppError) {
|
|
userOptions := &model.UserGetOptions{
|
|
Page: 0,
|
|
PerPage: 100,
|
|
Role: model.SYSTEM_ADMIN_ROLE_ID,
|
|
Inactive: false,
|
|
}
|
|
return a.GetUsers(userOptions)
|
|
}
|
|
|
|
// SendAdminUpgradeRequestEmail takes the username of user trying to alert admins and then applies rate limit of n (number of admins) emails per user per day
|
|
// before sending the emails.
|
|
func (a *App) SendAdminUpgradeRequestEmail(username string, subscription *model.Subscription, action string) *model.AppError {
|
|
if a.Srv().License() == nil || (a.Srv().License() != nil && !*a.Srv().License().Features.Cloud) {
|
|
return nil
|
|
}
|
|
|
|
if subscription != nil && subscription.IsPaidTier == "true" {
|
|
return nil
|
|
}
|
|
|
|
year, month, day := time.Now().Date()
|
|
key := fmt.Sprintf("%s-%d-%s-%d", action, day, month, year)
|
|
|
|
if a.Srv().EmailService.PerDayEmailRateLimiter == nil {
|
|
return model.NewAppError("app.SendAdminUpgradeRequestEmail", "app.email.no_rate_limiter.app_error", nil, fmt.Sprintf("for key=%s", key), http.StatusInternalServerError)
|
|
}
|
|
|
|
// rate limit based on combination of date and action as key
|
|
rateLimited, result, err := a.Srv().EmailService.PerDayEmailRateLimiter.RateLimit(key, 1)
|
|
if err != nil {
|
|
return model.NewAppError("app.SendAdminUpgradeRequestEmail", "app.email.setup_rate_limiter.app_error", nil, fmt.Sprintf("for key=%s, error=%v", key, err), http.StatusInternalServerError)
|
|
}
|
|
|
|
if rateLimited {
|
|
return model.NewAppError("app.SendAdminUpgradeRequestEmail",
|
|
"app.email.rate_limit_exceeded.app_error", map[string]interface{}{"RetryAfter": result.RetryAfter.String(), "ResetAfter": result.ResetAfter.String()},
|
|
fmt.Sprintf("key=%s, retry_after_secs=%f, reset_after_secs=%f",
|
|
key, result.RetryAfter.Seconds(), result.ResetAfter.Seconds()),
|
|
http.StatusRequestEntityTooLarge)
|
|
}
|
|
|
|
sysAdmins, e := a.getSysAdminsEmailRecipients()
|
|
if e != nil {
|
|
return e
|
|
}
|
|
|
|
// we want to at least have one email sent out to an admin
|
|
countNotOks := 0
|
|
|
|
for admin := range sysAdmins {
|
|
ok, err := a.Srv().EmailService.SendUpgradeEmail(username, sysAdmins[admin].Email, sysAdmins[admin].Locale, *a.Config().ServiceSettings.SiteURL, action)
|
|
if !ok || err != nil {
|
|
a.Log().Error("Error sending upgrade request email", mlog.Err(err))
|
|
countNotOks++
|
|
}
|
|
}
|
|
|
|
// if not even one admin got an email, we consider that this operation errored
|
|
if countNotOks == len(sysAdmins) {
|
|
return model.NewAppError("app.SendAdminUpgradeRequestEmail", "app.user.send_emails.app_error", nil, "", http.StatusInternalServerError)
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func (a *App) GetSubscriptionStats() (*model.SubscriptionStats, *model.AppError) {
|
|
if a.Srv().License() == nil || !*a.Srv().License().Features.Cloud {
|
|
return nil, model.NewAppError("app.GetSubscriptionStats", "api.cloud.license_error", nil, "", http.StatusInternalServerError)
|
|
}
|
|
|
|
subscription, appErr := a.Cloud().GetSubscription("")
|
|
if appErr != nil {
|
|
return nil, model.NewAppError("app.GetSubscriptionStats", "api.cloud.request_error", nil, appErr.Error(), http.StatusInternalServerError)
|
|
}
|
|
|
|
count, err := a.Srv().Store.User().Count(model.UserCountOptions{})
|
|
if err != nil {
|
|
return nil, model.NewAppError("app.GetSubscriptionStats", "app.user.get_total_users_count.app_error", nil, err.Error(), http.StatusInternalServerError)
|
|
}
|
|
cloudUserLimit := *a.Config().ExperimentalSettings.CloudUserLimit
|
|
|
|
s := cloudUserLimit - count
|
|
|
|
return &model.SubscriptionStats{
|
|
RemainingSeats: int(s),
|
|
IsPaidTier: subscription.IsPaidTier,
|
|
}, nil
|
|
}
|
|
|
|
func (a *App) CheckCloudAccountAtLimit() (bool, *model.AppError) {
|
|
if a.Srv().License() == nil || (a.Srv().License() != nil && !*a.Srv().License().Features.Cloud) {
|
|
// Not cloud instance, so no at limit checks
|
|
return false, nil
|
|
}
|
|
|
|
stats, err := a.GetSubscriptionStats()
|
|
if err != nil {
|
|
return false, err
|
|
}
|
|
|
|
if stats.IsPaidTier == "true" {
|
|
return false, nil
|
|
}
|
|
|
|
if stats.RemainingSeats < 1 {
|
|
return true, nil
|
|
}
|
|
|
|
return false, nil
|
|
}
|
|
|
|
func (a *App) CheckAndSendUserLimitWarningEmails(c *request.Context) *model.AppError {
|
|
if a.Srv().License() == nil || (a.Srv().License() != nil && !*a.Srv().License().Features.Cloud) {
|
|
// Not cloud instance, do nothing
|
|
return nil
|
|
}
|
|
|
|
subscription, err := a.Cloud().GetSubscription(c.Session().UserId)
|
|
if err != nil {
|
|
return model.NewAppError(
|
|
"app.CheckAndSendUserLimitWarningEmails",
|
|
"api.cloud.get_subscription.error",
|
|
nil,
|
|
err.Error(),
|
|
http.StatusInternalServerError)
|
|
}
|
|
|
|
if subscription != nil && subscription.IsPaidTier == "true" {
|
|
// Paid subscription, do nothing
|
|
return nil
|
|
}
|
|
|
|
cloudUserLimit := *a.Config().ExperimentalSettings.CloudUserLimit
|
|
systemUserCount, _ := a.Srv().Store.User().Count(model.UserCountOptions{})
|
|
remainingUsers := cloudUserLimit - systemUserCount
|
|
|
|
if remainingUsers > 0 {
|
|
return nil
|
|
}
|
|
sysAdmins, appErr := a.getSysAdminsEmailRecipients()
|
|
if appErr != nil {
|
|
return model.NewAppError(
|
|
"app.CheckAndSendUserLimitWarningEmails",
|
|
"api.cloud.get_admins_emails.error",
|
|
nil,
|
|
appErr.Error(),
|
|
http.StatusInternalServerError)
|
|
}
|
|
|
|
// -1 means they are 1 user over the limit - we only want to send the email for the 11th user
|
|
if remainingUsers == -1 {
|
|
// Over limit by 1 user
|
|
for admin := range sysAdmins {
|
|
_, appErr := a.Srv().EmailService.SendOverUserLimitWarningEmail(sysAdmins[admin].Email, sysAdmins[admin].Locale, *a.Config().ServiceSettings.SiteURL)
|
|
if appErr != nil {
|
|
a.Log().Error(
|
|
"Error sending user limit warning email to admin",
|
|
mlog.String("username", sysAdmins[admin].Username),
|
|
mlog.Err(err),
|
|
)
|
|
}
|
|
}
|
|
} else if remainingUsers == 0 {
|
|
// At limit
|
|
for admin := range sysAdmins {
|
|
_, appErr := a.Srv().EmailService.SendAtUserLimitWarningEmail(sysAdmins[admin].Email, sysAdmins[admin].Locale, *a.Config().ServiceSettings.SiteURL)
|
|
if appErr != nil {
|
|
a.Log().Error(
|
|
"Error sending user limit warning email to admin",
|
|
mlog.String("username", sysAdmins[admin].Username),
|
|
mlog.Err(err),
|
|
)
|
|
}
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (a *App) SendPaymentFailedEmail(failedPayment *model.FailedPayment) *model.AppError {
|
|
sysAdmins, err := a.getSysAdminsEmailRecipients()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
for _, admin := range sysAdmins {
|
|
_, err := a.Srv().EmailService.SendPaymentFailedEmail(admin.Email, admin.Locale, failedPayment, *a.Config().ServiceSettings.SiteURL)
|
|
if err != nil {
|
|
a.Log().Error("Error sending payment failed email", mlog.Err(err))
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// SendNoCardPaymentFailedEmail
|
|
func (a *App) SendNoCardPaymentFailedEmail() *model.AppError {
|
|
sysAdmins, err := a.getSysAdminsEmailRecipients()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
for _, admin := range sysAdmins {
|
|
err := a.Srv().EmailService.SendNoCardPaymentFailedEmail(admin.Email, admin.Locale, *a.Config().ServiceSettings.SiteURL)
|
|
if err != nil {
|
|
a.Log().Error("Error sending payment failed email", mlog.Err(err))
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (a *App) SendCloudTrialEndWarningEmail(trialEndDate, siteURL string) *model.AppError {
|
|
sysAdmins, e := a.getSysAdminsEmailRecipients()
|
|
if e != nil {
|
|
return e
|
|
}
|
|
|
|
// we want to at least have one email sent out to an admin
|
|
countNotOks := 0
|
|
|
|
for admin := range sysAdmins {
|
|
name := sysAdmins[admin].FirstName
|
|
if name == "" {
|
|
name = sysAdmins[admin].Username
|
|
}
|
|
err := a.Srv().EmailService.SendCloudTrialEndWarningEmail(sysAdmins[admin].Email, name, trialEndDate, sysAdmins[admin].Locale, siteURL)
|
|
if err != nil {
|
|
a.Log().Error("Error sending trial ending warning to", mlog.String("email", sysAdmins[admin].Email), mlog.Err(err))
|
|
countNotOks++
|
|
}
|
|
}
|
|
|
|
// if not even one admin got an email, we consider that this operation errored
|
|
if countNotOks == len(sysAdmins) {
|
|
return model.NewAppError("app.SendCloudTrialEndWarningEmail", "app.user.send_emails.app_error", nil, "", http.StatusInternalServerError)
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func (a *App) SendCloudTrialEndedEmail() *model.AppError {
|
|
sysAdmins, e := a.getSysAdminsEmailRecipients()
|
|
if e != nil {
|
|
return e
|
|
}
|
|
|
|
// we want to at least have one email sent out to an admin
|
|
countNotOks := 0
|
|
|
|
for admin := range sysAdmins {
|
|
name := sysAdmins[admin].FirstName
|
|
if name == "" {
|
|
name = sysAdmins[admin].Username
|
|
}
|
|
|
|
err := a.Srv().EmailService.SendCloudTrialEndedEmail(sysAdmins[admin].Email, name, sysAdmins[admin].Locale, *a.Config().ServiceSettings.SiteURL)
|
|
if err != nil {
|
|
a.Log().Error("Error sending trial ended email to", mlog.String("email", sysAdmins[admin].Email), mlog.Err(err))
|
|
countNotOks++
|
|
}
|
|
}
|
|
|
|
// if not even one admin got an email, we consider that this operation errored
|
|
if countNotOks == len(sysAdmins) {
|
|
return model.NewAppError("app.SendCloudTrialEndedEmail", "app.user.send_emails.app_error", nil, "", http.StatusInternalServerError)
|
|
}
|
|
|
|
return nil
|
|
}
|