Full Mattermost server source with integrated Community Enterprise features. Includes vendor directory for offline/air-gapped builds. Structure: - enterprise-impl/: Enterprise feature implementations - enterprise-community/: Init files that register implementations - enterprise/: Bridge imports (community_imports.go) - vendor/: All dependencies for offline builds Build (online): go build ./cmd/mattermost Build (offline/air-gapped): go build -mod=vendor ./cmd/mattermost 🤖 Generated with [Claude Code](https://claude.com/claude-code) Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
260 lines
8.7 KiB
Go
260 lines
8.7 KiB
Go
// Copyright (c) 2015-present Mattermost, Inc. All Rights Reserved.
|
|
// See LICENSE.txt for license information.
|
|
|
|
package app
|
|
|
|
import (
|
|
"fmt"
|
|
"os"
|
|
"testing"
|
|
"time"
|
|
|
|
"github.com/stretchr/testify/require"
|
|
|
|
"github.com/mattermost/mattermost/server/public/model"
|
|
)
|
|
|
|
func Test_SendNotifyAdminPosts(t *testing.T) {
|
|
mainHelper.Parallel(t)
|
|
t.Run("no error sending non trial upgrade post when no notifications are available", func(t *testing.T) {
|
|
th := Setup(t).InitBasic()
|
|
defer th.TearDown()
|
|
|
|
th.App.Srv().SetLicense(model.NewTestLicense("cloud"))
|
|
|
|
err := th.App.SendNotifyAdminPosts(th.Context, "", "", false)
|
|
require.Nil(t, err)
|
|
})
|
|
|
|
t.Run("no error sending trial upgrade post when no notifications are available", func(t *testing.T) {
|
|
th := Setup(t).InitBasic()
|
|
defer th.TearDown()
|
|
|
|
th.App.Srv().SetLicense(model.NewTestLicense("cloud"))
|
|
|
|
err := th.App.SendNotifyAdminPosts(th.Context, "", "", true)
|
|
require.Nil(t, err)
|
|
})
|
|
|
|
t.Run("successfully send upgrade notification", func(t *testing.T) {
|
|
th := Setup(t).InitBasic()
|
|
defer th.TearDown()
|
|
|
|
th.App.Srv().SetLicense(model.NewTestLicense("cloud"))
|
|
|
|
// some notifications
|
|
_, appErr := th.App.SaveAdminNotifyData(&model.NotifyAdminData{
|
|
UserId: th.BasicUser.Id,
|
|
RequiredPlan: model.LicenseShortSkuProfessional,
|
|
RequiredFeature: model.PaidFeatureGuestAccounts,
|
|
})
|
|
require.Nil(t, appErr)
|
|
|
|
_, appErr = th.App.SaveAdminNotifyData(&model.NotifyAdminData{
|
|
UserId: th.BasicUser2.Id,
|
|
RequiredPlan: model.LicenseShortSkuProfessional,
|
|
RequiredFeature: model.PaidFeatureGuestAccounts,
|
|
})
|
|
require.Nil(t, appErr)
|
|
|
|
appErr = th.App.SendNotifyAdminPosts(th.Context, "test", "", false)
|
|
require.Nil(t, appErr)
|
|
|
|
bot, appErr := th.App.GetSystemBot(th.Context)
|
|
require.Nil(t, appErr)
|
|
|
|
// message sending is async, wait time for it
|
|
var channel *model.Channel
|
|
var err error
|
|
timeout := 5 * time.Second
|
|
begin := time.Now()
|
|
for {
|
|
if time.Since(begin) > timeout {
|
|
break
|
|
}
|
|
channel, err = th.App.Srv().Store().Channel().GetByName("", model.GetDMNameFromIds(bot.UserId, th.SystemAdminUser.Id), false)
|
|
if err == nil && channel != nil {
|
|
break
|
|
}
|
|
time.Sleep(100 * time.Millisecond)
|
|
}
|
|
require.NoError(t, err, "Expected message to have been sent within %d seconds", timeout)
|
|
|
|
postList, err := th.App.Srv().Store().Post().GetPosts(th.Context, model.GetPostsOptions{ChannelId: channel.Id, Page: 0, PerPage: 1}, false, map[string]bool{})
|
|
require.NoError(t, err)
|
|
|
|
post := postList.Posts[postList.Order[0]]
|
|
require.Equal(t, fmt.Sprintf("%sup_notification", model.PostCustomTypePrefix), post.Type)
|
|
require.Equal(t, bot.UserId, post.UserId)
|
|
require.Equal(t, "2 members of the test workspace have requested a workspace upgrade for: ", post.Message)
|
|
})
|
|
|
|
t.Run("successfully send trial upgrade notification", func(t *testing.T) {
|
|
th := Setup(t).InitBasic()
|
|
defer th.TearDown()
|
|
|
|
th.App.Srv().SetLicense(model.NewTestLicense("cloud"))
|
|
|
|
// some notifications
|
|
_, appErr := th.App.SaveAdminNotifyData(&model.NotifyAdminData{
|
|
UserId: th.BasicUser.Id,
|
|
RequiredPlan: model.LicenseShortSkuProfessional,
|
|
RequiredFeature: model.PaidFeatureAllProfessionalfeatures,
|
|
Trial: true,
|
|
})
|
|
require.Nil(t, appErr)
|
|
|
|
appErr = th.App.SendNotifyAdminPosts(th.Context, "test", "", true)
|
|
require.Nil(t, appErr)
|
|
|
|
bot, appErr := th.App.GetSystemBot(th.Context)
|
|
require.Nil(t, appErr)
|
|
|
|
// message sending is async, wait time for it
|
|
var channel *model.Channel
|
|
var err error
|
|
timeout := 5 * time.Second
|
|
begin := time.Now()
|
|
for {
|
|
if time.Since(begin) > timeout {
|
|
break
|
|
}
|
|
channel, err = th.App.Srv().Store().Channel().GetByName("", model.GetDMNameFromIds(bot.UserId, th.SystemAdminUser.Id), false)
|
|
if err == nil && channel != nil {
|
|
break
|
|
}
|
|
time.Sleep(100 * time.Millisecond)
|
|
}
|
|
require.NoError(t, err, "Expected message to have been sent within %d seconds", timeout)
|
|
|
|
postList, err := th.App.Srv().Store().Post().GetPosts(th.Context, model.GetPostsOptions{ChannelId: channel.Id, Page: 0, PerPage: 1}, false, map[string]bool{})
|
|
require.NoError(t, err)
|
|
|
|
post := postList.Posts[postList.Order[0]]
|
|
require.Equal(t, fmt.Sprintf("%sup_notification", model.PostCustomTypePrefix), post.Type)
|
|
require.Equal(t, bot.UserId, post.UserId)
|
|
require.Equal(t, "1 member of the test workspace has requested starting the Enterprise trial for access to: ", post.Message)
|
|
})
|
|
|
|
t.Run("error when trying to send upgrade post before end of cool off period", func(t *testing.T) {
|
|
th := Setup(t).InitBasic()
|
|
defer th.TearDown()
|
|
|
|
th.App.Srv().SetLicense(model.NewTestLicense("cloud"))
|
|
|
|
// some notifications
|
|
_, appErr := th.App.SaveAdminNotifyData(&model.NotifyAdminData{
|
|
UserId: th.BasicUser.Id,
|
|
RequiredPlan: model.LicenseShortSkuProfessional,
|
|
RequiredFeature: model.PaidFeatureAllProfessionalfeatures,
|
|
})
|
|
require.Nil(t, appErr)
|
|
|
|
appErr = th.App.SendNotifyAdminPosts(th.Context, "", "", false)
|
|
require.Nil(t, appErr)
|
|
|
|
// add some more notifications while in cool off
|
|
_, appErr = th.App.SaveAdminNotifyData(&model.NotifyAdminData{
|
|
UserId: th.BasicUser.Id,
|
|
RequiredPlan: model.LicenseShortSkuProfessional,
|
|
RequiredFeature: model.PaidFeatureCustomUsergroups,
|
|
})
|
|
require.Nil(t, appErr)
|
|
|
|
// second time trying to notify is forbidden
|
|
appErr = th.App.SendNotifyAdminPosts(th.Context, "", "", false)
|
|
require.NotNil(t, appErr)
|
|
require.Equal(t, appErr.Error(), "SendNotifyAdminPosts: Unable to send notification post., Cannot notify yet")
|
|
})
|
|
|
|
t.Run("can send upgrade post at the end of cool off period", func(t *testing.T) {
|
|
th := Setup(t).InitBasic()
|
|
defer th.TearDown()
|
|
|
|
th.App.Srv().SetLicense(model.NewTestLicense("cloud"))
|
|
|
|
os.Setenv("MM_NOTIFY_ADMIN_COOL_OFF_DAYS", "0.00003472222222") // set to 3 seconds
|
|
defer os.Unsetenv("MM_NOTIFY_ADMIN_COOL_OFF_DAYS")
|
|
|
|
// some notifications
|
|
_, appErr := th.App.SaveAdminNotifyData(&model.NotifyAdminData{
|
|
UserId: th.BasicUser.Id,
|
|
RequiredPlan: model.LicenseShortSkuProfessional,
|
|
RequiredFeature: model.PaidFeatureAllProfessionalfeatures,
|
|
})
|
|
require.Nil(t, appErr)
|
|
|
|
appErr = th.App.SendNotifyAdminPosts(th.Context, "", "", false)
|
|
require.Nil(t, appErr)
|
|
|
|
// add some more notifications while in cool off
|
|
_, appErr = th.App.SaveAdminNotifyData(&model.NotifyAdminData{
|
|
UserId: th.BasicUser.Id,
|
|
RequiredPlan: model.LicenseShortSkuProfessional,
|
|
RequiredFeature: model.PaidFeatureCustomUsergroups,
|
|
})
|
|
require.Nil(t, appErr)
|
|
|
|
time.Sleep(5 * time.Second)
|
|
|
|
// no error sending second time
|
|
appErr = th.App.SendNotifyAdminPosts(th.Context, "", "", false)
|
|
require.Nil(t, appErr)
|
|
})
|
|
|
|
t.Run("can filter notifications when plan changes within cool off period", func(t *testing.T) {
|
|
th := Setup(t).InitBasic()
|
|
defer th.TearDown()
|
|
|
|
th.App.Srv().SetLicense(model.NewTestLicense("cloud"))
|
|
|
|
// some notifications
|
|
_, appErr := th.App.SaveAdminNotifyData(&model.NotifyAdminData{
|
|
UserId: th.BasicUser.Id,
|
|
RequiredPlan: model.LicenseShortSkuProfessional,
|
|
RequiredFeature: model.PaidFeatureAllProfessionalfeatures,
|
|
Trial: false,
|
|
})
|
|
require.Nil(t, appErr)
|
|
|
|
_, appErr = th.App.SaveAdminNotifyData(&model.NotifyAdminData{
|
|
UserId: th.BasicUser2.Id,
|
|
RequiredPlan: model.LicenseShortSkuEnterprise,
|
|
RequiredFeature: model.PaidFeatureAllEnterprisefeatures,
|
|
Trial: false,
|
|
})
|
|
require.Nil(t, appErr)
|
|
|
|
appErr = th.App.SendNotifyAdminPosts(th.Context, "test", model.LicenseShortSkuProfessional, false) // try and send notification but workspace currentSKU has since changed to cloud-professional
|
|
require.Nil(t, appErr)
|
|
|
|
bot, appErr := th.App.GetSystemBot(th.Context)
|
|
require.Nil(t, appErr)
|
|
|
|
// message sending is async, wait time for it
|
|
var channel *model.Channel
|
|
var err error
|
|
timeout := 5 * time.Second
|
|
begin := time.Now()
|
|
for {
|
|
if time.Since(begin) > timeout {
|
|
break
|
|
}
|
|
channel, err = th.App.Srv().Store().Channel().GetByName("", model.GetDMNameFromIds(bot.UserId, th.SystemAdminUser.Id), false)
|
|
if err == nil && channel != nil {
|
|
break
|
|
}
|
|
time.Sleep(100 * time.Millisecond)
|
|
}
|
|
require.NoError(t, err, "Expected message to have been sent within %d seconds", timeout)
|
|
|
|
postList, err := th.App.Srv().Store().Post().GetPosts(th.Context, model.GetPostsOptions{ChannelId: channel.Id, Page: 0, PerPage: 1}, false, map[string]bool{})
|
|
require.NoError(t, err)
|
|
|
|
post := postList.Posts[postList.Order[0]]
|
|
require.Equal(t, fmt.Sprintf("%sup_notification", model.PostCustomTypePrefix), post.Type)
|
|
require.Equal(t, bot.UserId, post.UserId)
|
|
require.Equal(t, "1 member of the test workspace has requested a workspace upgrade for: ", post.Message) // expect only one member's notification even though 2 were added
|
|
})
|
|
}
|