// Copyright (c) 2015-present Mattermost, Inc. All Rights Reserved. // See LICENSE.txt for license information. package app import ( "encoding/json" "fmt" "io" "maps" "net/http" "time" "github.com/mattermost/mattermost/server/public/model" "github.com/mattermost/mattermost/server/public/shared/i18n" "github.com/mattermost/mattermost/server/public/shared/request" "github.com/mattermost/mattermost/server/v8/platform/services/cache" "github.com/mattermost/mattermost/server/v8/platform/shared/mail" ) var latestVersionCache = cache.NewLRU(&cache.CacheOptions{ Size: 1, }) func (s *Server) GetLogs(rctx request.CTX, page, perPage int) ([]string, *model.AppError) { var lines []string license := s.License() if license != nil && *license.Features.Cluster && s.platform.Cluster() != nil && *s.platform.Config().ClusterSettings.Enable { if info := s.platform.Cluster().GetMyClusterInfo(); info != nil { lines = append(lines, "-----------------------------------------------------------------------------------------------------------") lines = append(lines, "-----------------------------------------------------------------------------------------------------------") lines = append(lines, info.Hostname) lines = append(lines, "-----------------------------------------------------------------------------------------------------------") lines = append(lines, "-----------------------------------------------------------------------------------------------------------") } else { rctx.Logger().Error("Could not get cluster info") } } melines, err := s.GetLogsSkipSend(rctx, page, perPage, &model.LogFilter{}) if err != nil { return nil, err } lines = append(lines, melines...) if s.platform.Cluster() != nil && *s.platform.Config().ClusterSettings.Enable { clines, err := s.platform.Cluster().GetLogs(rctx, page, perPage) if err != nil { return nil, err } lines = append(lines, clines...) } return lines, nil } func (s *Server) QueryLogs(rctx request.CTX, page, perPage int, logFilter *model.LogFilter) (map[string][]string, *model.AppError) { logData := make(map[string][]string) serverName := "default" license := s.License() if license != nil && *license.Features.Cluster && s.platform.Cluster() != nil && *s.platform.Config().ClusterSettings.Enable { if info := s.platform.Cluster().GetMyClusterInfo(); info != nil { serverName = info.Hostname } else { rctx.Logger().Error("Could not get cluster info") } } var appErr *model.AppError serverNames := logFilter.ServerNames if len(serverNames) > 0 { for _, nodeName := range serverNames { if nodeName == "default" { appErr = addLocalLogs(rctx, logData, s, page, perPage, nodeName, logFilter) } } } else { appErr = addLocalLogs(rctx, logData, s, page, perPage, serverName, logFilter) } if appErr != nil { return nil, appErr } if s.platform.Cluster() != nil && *s.Config().ClusterSettings.Enable { clusterLogs, err := s.platform.Cluster().QueryLogs(rctx, page, perPage) if err != nil { return nil, err } if clusterLogs != nil && len(serverNames) > 0 { for _, filteredNodeName := range serverNames { logData[filteredNodeName] = clusterLogs[filteredNodeName] } } else { maps.Copy(logData, clusterLogs) } } return logData, nil } func addLocalLogs(rctx request.CTX, logData map[string][]string, s *Server, page, perPage int, serverName string, logFilter *model.LogFilter) *model.AppError { currentServerLogs, err := s.GetLogsSkipSend(rctx, page, perPage, logFilter) if err != nil { return err } logData[serverName] = currentServerLogs return nil } func (a *App) QueryLogs(rctx request.CTX, page, perPage int, logFilter *model.LogFilter) (map[string][]string, *model.AppError) { return a.Srv().QueryLogs(rctx, page, perPage, logFilter) } func (a *App) GetLogs(rctx request.CTX, page, perPage int) ([]string, *model.AppError) { return a.Srv().GetLogs(rctx, page, perPage) } func (s *Server) GetLogsSkipSend(rctx request.CTX, page, perPage int, logFilter *model.LogFilter) ([]string, *model.AppError) { return s.platform.GetLogsSkipSend(rctx, page, perPage, logFilter) } func (a *App) GetLogsSkipSend(rctx request.CTX, page, perPage int, logFilter *model.LogFilter) ([]string, *model.AppError) { return a.Srv().GetLogsSkipSend(rctx, page, perPage, logFilter) } func (a *App) GetClusterStatus(rctx request.CTX) ([]*model.ClusterInfo, error) { if a.Cluster() == nil { return make([]*model.ClusterInfo, 0), nil } return a.Cluster().GetClusterInfos() } func (s *Server) InvalidateAllCaches() *model.AppError { return s.platform.InvalidateAllCaches() } func (s *Server) InvalidateAllCachesSkipSend() *model.AppError { return s.platform.InvalidateAllCachesSkipSend() } func (a *App) RecycleDatabaseConnection(rctx request.CTX) { rctx.Logger().Info("Attempting to recycle database connections.") // This works by setting 10 seconds as the max conn lifetime for all DB connections. // This allows in gradually closing connections as they expire. In future, we can think // of exposing this as a param from the REST api. a.Srv().Store().RecycleDBConnections(10 * time.Second) rctx.Logger().Info("Finished recycling database connections.") } func (a *App) TestSiteURL(rctx request.CTX, siteURL string) *model.AppError { url := fmt.Sprintf("%s/api/v4/system/ping", siteURL) res, err := http.Get(url) if err != nil || res.StatusCode != 200 { return model.NewAppError("testSiteURL", "app.admin.test_site_url.failure", nil, "", http.StatusBadRequest) } defer func() { _, _ = io.Copy(io.Discard, res.Body) _ = res.Body.Close() }() return nil } func (a *App) TestEmail(rctx request.CTX, userID string, cfg *model.Config) *model.AppError { if *cfg.EmailSettings.SMTPServer == "" { return model.NewAppError("testEmail", "api.admin.test_email.missing_server", nil, i18n.T("api.context.invalid_param.app_error", map[string]any{"Name": "SMTPServer"}), http.StatusBadRequest) } // if the user hasn't changed their email settings, fill in the actual SMTP password so that // the user can verify an existing SMTP connection if *cfg.EmailSettings.SMTPPassword == model.FakeSetting { if *cfg.EmailSettings.SMTPServer == *a.Config().EmailSettings.SMTPServer && *cfg.EmailSettings.SMTPPort == *a.Config().EmailSettings.SMTPPort && *cfg.EmailSettings.SMTPUsername == *a.Config().EmailSettings.SMTPUsername { *cfg.EmailSettings.SMTPPassword = *a.Config().EmailSettings.SMTPPassword } else { return model.NewAppError("testEmail", "api.admin.test_email.reenter_password", nil, "", http.StatusBadRequest) } } user, err := a.GetUser(userID) if err != nil { return err } T := i18n.GetUserTranslations(user.Locale) license := a.Srv().License() mailConfig := a.Srv().MailServiceConfig() if err := mail.SendMailUsingConfig(user.Email, T("api.admin.test_email.subject"), T("api.admin.test_email.body"), mailConfig, license != nil && *license.Features.Compliance, "", "", "", "", ""); err != nil { return model.NewAppError("testEmail", "app.admin.test_email.failure", map[string]any{"Error": err.Error()}, "", http.StatusInternalServerError) } return nil } func (a *App) GetLatestVersion(rctx request.CTX, latestVersionUrl string) (*model.GithubReleaseInfo, *model.AppError) { var cachedLatestVersion *model.GithubReleaseInfo if cacheErr := latestVersionCache.Get("latest_version_cache", &cachedLatestVersion); cacheErr == nil { return cachedLatestVersion, nil } res, err := http.Get(latestVersionUrl) if err != nil { return nil, model.NewAppError("GetLatestVersion", model.NoTranslation, nil, "", http.StatusInternalServerError).Wrap(err) } defer res.Body.Close() responseData, err := io.ReadAll(res.Body) if err != nil { return nil, model.NewAppError("GetLatestVersion", model.NoTranslation, nil, "", http.StatusInternalServerError).Wrap(err) } var releaseInfoResponse *model.GithubReleaseInfo err = json.Unmarshal(responseData, &releaseInfoResponse) if err != nil { return nil, model.NewAppError("GetLatestVersion", model.NoTranslation, nil, "", http.StatusInternalServerError).Wrap(err) } if validErr := releaseInfoResponse.IsValid(); validErr != nil { return nil, model.NewAppError("GetLatestVersion", model.NoTranslation, nil, "", http.StatusInternalServerError).Wrap(validErr) } err = latestVersionCache.SetWithExpiry("latest_version_cache", releaseInfoResponse, 24*time.Hour) if err != nil { return nil, model.NewAppError("GetLatestVersion", model.NoTranslation, nil, "", http.StatusInternalServerError).Wrap(err) } return releaseInfoResponse, nil } func (a *App) clearLatestVersionCache() error { return latestVersionCache.Remove("latest_version_cache") }