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>
201 lines
6.4 KiB
Go
201 lines
6.4 KiB
Go
// Copyright (c) 2015-present Mattermost, Inc. All Rights Reserved.
|
|
// See LICENSE.txt for license information.
|
|
|
|
package platform
|
|
|
|
import (
|
|
"errors"
|
|
"sync"
|
|
"testing"
|
|
|
|
"github.com/stretchr/testify/assert"
|
|
"github.com/stretchr/testify/mock"
|
|
"github.com/stretchr/testify/require"
|
|
|
|
"github.com/mattermost/mattermost/server/public/model"
|
|
smocks "github.com/mattermost/mattermost/server/v8/channels/store/storetest/mocks"
|
|
"github.com/mattermost/mattermost/server/v8/einterfaces/mocks"
|
|
)
|
|
|
|
func TestConfigListener(t *testing.T) {
|
|
mainHelper.Parallel(t)
|
|
th := Setup(t)
|
|
defer th.TearDown()
|
|
|
|
originalSiteName := th.Service.Config().TeamSettings.SiteName
|
|
|
|
listenerCalled := false
|
|
listener := func(oldConfig *model.Config, newConfig *model.Config) {
|
|
assert.False(t, listenerCalled, "listener called twice")
|
|
|
|
assert.Equal(t, *originalSiteName, *oldConfig.TeamSettings.SiteName, "old config contains incorrect site name")
|
|
assert.Equal(t, "test123", *newConfig.TeamSettings.SiteName, "new config contains incorrect site name")
|
|
|
|
listenerCalled = true
|
|
}
|
|
listenerId := th.Service.AddConfigListener(listener)
|
|
defer th.Service.RemoveConfigListener(listenerId)
|
|
|
|
listener2Called := false
|
|
listener2 := func(oldConfig *model.Config, newConfig *model.Config) {
|
|
assert.False(t, listener2Called, "listener2 called twice")
|
|
|
|
listener2Called = true
|
|
}
|
|
listener2Id := th.Service.AddConfigListener(listener2)
|
|
defer th.Service.RemoveConfigListener(listener2Id)
|
|
|
|
th.Service.UpdateConfig(func(cfg *model.Config) {
|
|
*cfg.TeamSettings.SiteName = "test123"
|
|
})
|
|
|
|
assert.True(t, listenerCalled, "listener should've been called")
|
|
assert.True(t, listener2Called, "listener 2 should've been called")
|
|
}
|
|
|
|
func TestConfigSave(t *testing.T) {
|
|
mainHelper.Parallel(t)
|
|
cm := &mocks.ClusterInterface{}
|
|
cm.On("SendClusterMessage", mock.AnythingOfType("*model.ClusterMessage")).Return(nil)
|
|
th := SetupWithCluster(t, cm)
|
|
defer th.TearDown()
|
|
|
|
t.Run("trigger a config changed event for the cluster", func(t *testing.T) {
|
|
oldCfg := th.Service.Config()
|
|
newCfg := oldCfg.Clone()
|
|
newCfg.ServiceSettings.SiteURL = model.NewPointer("http://newhost.me")
|
|
|
|
sanitizedOldCfg := th.Service.configStore.RemoveEnvironmentOverrides(oldCfg)
|
|
sanitizedNewCfg := th.Service.configStore.RemoveEnvironmentOverrides(newCfg)
|
|
cm.On("ConfigChanged", sanitizedOldCfg, sanitizedNewCfg, true).Return(nil)
|
|
|
|
_, _, appErr := th.Service.SaveConfig(newCfg, true)
|
|
require.Nil(t, appErr)
|
|
|
|
updatedCfg := th.Service.Config()
|
|
assert.Equal(t, "http://newhost.me", *updatedCfg.ServiceSettings.SiteURL)
|
|
})
|
|
|
|
t.Run("do not restart the metrics server on a different type of config change", func(t *testing.T) {
|
|
th := Setup(t, StartMetrics())
|
|
defer th.TearDown()
|
|
|
|
metricsMock := &mocks.MetricsInterface{}
|
|
metricsMock.On("IncrementWebsocketEvent", model.WebsocketEventConfigChanged).Return()
|
|
metricsMock.On("IncrementWebSocketBroadcastBufferSize", mock.AnythingOfType("string"), mock.AnythingOfType("float64")).Return()
|
|
metricsMock.On("DecrementWebSocketBroadcastBufferSize", mock.AnythingOfType("string"), mock.AnythingOfType("float64")).Return()
|
|
metricsMock.On("Register").Return()
|
|
th.Service.metricsIFace = metricsMock
|
|
|
|
// Change a random config setting
|
|
cfg := th.Service.Config().Clone()
|
|
cfg.ThemeSettings.EnableThemeSelection = model.NewPointer(!*cfg.ThemeSettings.EnableThemeSelection)
|
|
_, _, appErr := th.Service.SaveConfig(cfg, false)
|
|
require.Nil(t, appErr)
|
|
metricsMock.AssertNumberOfCalls(t, "Register", 0)
|
|
|
|
// Disable metrics
|
|
cfg.MetricsSettings.Enable = model.NewPointer(false)
|
|
_, _, appErr = th.Service.SaveConfig(cfg, false)
|
|
require.Nil(t, appErr)
|
|
|
|
// Change the metrics setting
|
|
cfg.MetricsSettings.Enable = model.NewPointer(true)
|
|
_, _, appErr = th.Service.SaveConfig(cfg, false)
|
|
require.Nil(t, appErr)
|
|
metricsMock.AssertNumberOfCalls(t, "Register", 1)
|
|
})
|
|
}
|
|
|
|
func TestIsFirstUserAccount(t *testing.T) {
|
|
mainHelper.Parallel(t)
|
|
th := SetupWithStoreMock(t)
|
|
defer th.TearDown()
|
|
storeMock := th.Service.Store.(*smocks.Store)
|
|
userStoreMock := &smocks.UserStore{}
|
|
storeMock.On("User").Return(userStoreMock)
|
|
|
|
type test struct {
|
|
name string
|
|
count int64
|
|
err error
|
|
result bool
|
|
shouldCallStore bool
|
|
}
|
|
|
|
tests := []test{
|
|
{"failed request", 0, errors.New("error"), false, true},
|
|
{"success negative users", -100, nil, true, true},
|
|
{"success no users", 0, nil, true, true},
|
|
{"success one user", 1, nil, false, true},
|
|
{"success multiple users - no store call", 42, nil, false, false},
|
|
}
|
|
|
|
// create a session, this should not affect IsFirstUserAccount
|
|
err := th.Service.sessionCache.SetWithDefaultExpiry("mock_session", 1)
|
|
require.NoError(t, err)
|
|
|
|
for _, te := range tests {
|
|
t.Run(te.name, func(t *testing.T) {
|
|
*userStoreMock = smocks.UserStore{}
|
|
|
|
if te.shouldCallStore {
|
|
userStoreMock.On("Count", model.UserCountOptions{IncludeDeleted: true}).Return(te.count, te.err).Once()
|
|
} else {
|
|
userStoreMock.On("Count", model.UserCountOptions{IncludeDeleted: true}).Unset()
|
|
}
|
|
|
|
require.Equal(t, te.result, th.Service.IsFirstUserAccount())
|
|
})
|
|
}
|
|
}
|
|
|
|
func TestIsFirstUserAccountThunderingHerd(t *testing.T) {
|
|
mainHelper.Parallel(t)
|
|
th := SetupWithStoreMock(t)
|
|
defer th.TearDown()
|
|
storeMock := th.Service.Store.(*smocks.Store)
|
|
userStoreMock := &smocks.UserStore{}
|
|
storeMock.On("User").Return(userStoreMock)
|
|
|
|
tests := []struct {
|
|
name string
|
|
count int64
|
|
err error
|
|
concurrentRequest int
|
|
result bool
|
|
numberOfStoreCalls int
|
|
}{
|
|
{"failed request", 0, errors.New("error"), 10, false, 10},
|
|
{"success negative users", -100, nil, 10, true, 10},
|
|
{"success no users", 0, nil, 10, true, 10},
|
|
{"success one user - lot of requests", 1, nil, 1000, false, 1},
|
|
{"success multiple users - no store call", 42, nil, 10, false, 0},
|
|
}
|
|
|
|
for _, te := range tests {
|
|
t.Run(te.name, func(t *testing.T) {
|
|
*userStoreMock = smocks.UserStore{}
|
|
|
|
if te.numberOfStoreCalls != 0 {
|
|
userStoreMock.On("Count", model.UserCountOptions{IncludeDeleted: true}).Return(te.count, te.err).Times(te.numberOfStoreCalls)
|
|
} else {
|
|
userStoreMock.On("Count", model.UserCountOptions{IncludeDeleted: true}).Unset()
|
|
}
|
|
defer userStoreMock.AssertExpectations(t)
|
|
|
|
var wg sync.WaitGroup
|
|
for i := 0; i < te.concurrentRequest; i++ {
|
|
wg.Add(1)
|
|
|
|
go func() {
|
|
defer wg.Done()
|
|
require.Equal(t, te.result, th.Service.IsFirstUserAccount())
|
|
}()
|
|
}
|
|
|
|
wg.Wait()
|
|
})
|
|
}
|
|
}
|