mirror of
https://github.com/mattermost/mattermost.git
synced 2026-04-21 22:29:28 -04:00
* added download system logs * download all logs * download all logs check-lint fix * check lint fix * download logs api * download logs api working * download logs working with error log * linting issues and code cleanup * CI check fix * documented the api and logs from file with error handling * test and final changes done * final changes done * Fix order of server-side translations * Fix incorrect indentation of logs.yaml --------- Co-authored-by: Harrison Healey <harrisonmhealey@gmail.com>
403 lines
12 KiB
Go
403 lines
12 KiB
Go
// Copyright (c) 2015-present Mattermost, Inc. All Rights Reserved.
|
|
// See LICENSE.txt for license information.
|
|
|
|
package app
|
|
|
|
import (
|
|
"bytes"
|
|
"encoding/json"
|
|
"os"
|
|
"runtime"
|
|
"runtime/pprof"
|
|
"strings"
|
|
"time"
|
|
|
|
"github.com/hashicorp/go-multierror"
|
|
"github.com/pkg/errors"
|
|
"gopkg.in/yaml.v2"
|
|
|
|
"github.com/mattermost/mattermost/server/public/model"
|
|
"github.com/mattermost/mattermost/server/public/shared/mlog"
|
|
"github.com/mattermost/mattermost/server/public/shared/request"
|
|
"github.com/mattermost/mattermost/server/v8/config"
|
|
)
|
|
|
|
const (
|
|
cpuProfileDuration = 5 * time.Second
|
|
)
|
|
|
|
func (a *App) GenerateSupportPacket(c request.CTX, options *model.SupportPacketOptions) []model.FileData {
|
|
// If any errors we come across within this function, we will log it in a warning.txt file so that we know why certain files did not get produced if any
|
|
var warnings []string
|
|
|
|
// Creating an array of files that we are going to be adding to our zip file
|
|
fileDatas := []model.FileData{}
|
|
|
|
// A array of the functions that we can iterate through since they all have the same return value
|
|
functions := map[string]func(c request.CTX) (*model.FileData, error){
|
|
"support package": a.generateSupportPacketYaml,
|
|
"plugins": a.createPluginsFile,
|
|
"config": a.createSanitizedConfigFile,
|
|
"cpu profile": a.createCPUProfile,
|
|
"heap profile": a.createHeapProfile,
|
|
"goroutines": a.createGoroutineProfile,
|
|
}
|
|
|
|
if options.IncludeLogs {
|
|
functions["mattermost log"] = a.GetMattermostLog
|
|
functions["notification log"] = a.getNotificationsLog
|
|
}
|
|
|
|
for name, fn := range functions {
|
|
fileData, err := fn(c)
|
|
if err != nil {
|
|
c.Logger().Error("Failed to generate file for support package", mlog.Err(err), mlog.String("file", name))
|
|
warnings = append(warnings, err.Error())
|
|
}
|
|
|
|
if fileData != nil {
|
|
fileDatas = append(fileDatas, *fileData)
|
|
}
|
|
}
|
|
|
|
if pluginsEnvironment := a.GetPluginsEnvironment(); pluginsEnvironment != nil {
|
|
pluginContext := pluginContext(c)
|
|
for _, id := range options.PluginPackets {
|
|
hooks, err := pluginsEnvironment.HooksForPlugin(id)
|
|
if err != nil {
|
|
c.Logger().Error("Failed to call hooks for plugin", mlog.Err(err), mlog.String("plugin", id))
|
|
warnings = append(warnings, err.Error())
|
|
continue
|
|
}
|
|
pluginData, err := hooks.GenerateSupportData(pluginContext)
|
|
if err != nil {
|
|
c.Logger().Warn("Failed to generate plugin file for support package", mlog.Err(err), mlog.String("plugin", id))
|
|
warnings = append(warnings, err.Error())
|
|
continue
|
|
}
|
|
for _, data := range pluginData {
|
|
fileDatas = append(fileDatas, *data)
|
|
}
|
|
}
|
|
}
|
|
|
|
// Adding a warning.txt file to the fileDatas if any warning
|
|
if len(warnings) > 0 {
|
|
finalWarning := strings.Join(warnings, "\n")
|
|
fileDatas = append(fileDatas, model.FileData{
|
|
Filename: "warning.txt",
|
|
Body: []byte(finalWarning),
|
|
})
|
|
}
|
|
|
|
return fileDatas
|
|
}
|
|
|
|
func (a *App) generateSupportPacketYaml(c request.CTX) (*model.FileData, error) {
|
|
var rErr *multierror.Error
|
|
|
|
/* DB */
|
|
|
|
databaseType, databaseSchemaVersion := a.Srv().DatabaseTypeAndSchemaVersion()
|
|
databaseVersion, err := a.Srv().Store().GetDbVersion(false)
|
|
if err != nil {
|
|
rErr = multierror.Append(errors.Wrap(err, "error while getting DB version"))
|
|
}
|
|
|
|
/* Cluster */
|
|
|
|
var clusterID string
|
|
if a.Cluster() != nil {
|
|
clusterID = a.Cluster().GetClusterId()
|
|
}
|
|
|
|
/* File store */
|
|
|
|
fileDriver := a.Srv().Platform().FileBackend().DriverName()
|
|
fileStatus := model.StatusOk
|
|
err = a.Srv().Platform().FileBackend().TestConnection()
|
|
if err != nil {
|
|
fileStatus = model.StatusFail + ": " + err.Error()
|
|
}
|
|
|
|
/* LDAP */
|
|
|
|
var vendorName, vendorVersion string
|
|
if ldapInterface := a.Ldap(); ldapInterface != nil {
|
|
vendorName, vendorVersion = ldapInterface.GetVendorNameAndVendorVersion(c)
|
|
}
|
|
|
|
/* Elastic Search */
|
|
|
|
var elasticServerVersion string
|
|
var elasticServerPlugins []string
|
|
if a.Srv().Platform().SearchEngine.ElasticsearchEngine != nil {
|
|
elasticServerVersion = a.Srv().Platform().SearchEngine.ElasticsearchEngine.GetFullVersion()
|
|
elasticServerPlugins = a.Srv().Platform().SearchEngine.ElasticsearchEngine.GetPlugins()
|
|
}
|
|
|
|
/* License */
|
|
|
|
var (
|
|
licenseTo string
|
|
supportedUsers int
|
|
isTrial bool
|
|
)
|
|
if license := a.Srv().License(); license != nil {
|
|
licenseTo = license.Customer.Company
|
|
supportedUsers = *license.Features.Users
|
|
isTrial = license.IsTrial
|
|
}
|
|
|
|
/* Server stats */
|
|
|
|
uniqueUserCount, err := a.Srv().Store().User().Count(model.UserCountOptions{})
|
|
if err != nil {
|
|
rErr = multierror.Append(errors.Wrap(err, "error while getting user count"))
|
|
}
|
|
|
|
var (
|
|
totalChannels int
|
|
totalPosts int
|
|
totalTeams int
|
|
websocketConnections int
|
|
masterDbConnections int
|
|
replicaDbConnections int
|
|
dailyActiveUsers int
|
|
monthlyActiveUsers int
|
|
inactiveUserCount int
|
|
)
|
|
analytics, appErr := a.GetAnalyticsForSupportPacket(c)
|
|
if appErr != nil {
|
|
rErr = multierror.Append(errors.Wrap(appErr, "error while getting analytics"))
|
|
}
|
|
if len(analytics) < 11 {
|
|
rErr = multierror.Append(errors.New("not enought analytics information found"))
|
|
} else {
|
|
totalChannels = int(analytics[0].Value) + int(analytics[1].Value)
|
|
totalPosts = int(analytics[2].Value)
|
|
totalTeams = int(analytics[4].Value)
|
|
websocketConnections = int(analytics[5].Value)
|
|
masterDbConnections = int(analytics[6].Value)
|
|
replicaDbConnections = int(analytics[7].Value)
|
|
dailyActiveUsers = int(analytics[8].Value)
|
|
monthlyActiveUsers = int(analytics[9].Value)
|
|
inactiveUserCount = int(analytics[10].Value)
|
|
}
|
|
|
|
/* Jobs */
|
|
|
|
dataRetentionJobs, err := a.Srv().Store().Job().GetAllByTypePage(c, model.JobTypeDataRetention, 0, 2)
|
|
if err != nil {
|
|
rErr = multierror.Append(errors.Wrap(err, "error while getting data retention jobs"))
|
|
}
|
|
messageExportJobs, err := a.Srv().Store().Job().GetAllByTypePage(c, model.JobTypeMessageExport, 0, 2)
|
|
if err != nil {
|
|
rErr = multierror.Append(errors.Wrap(err, "error while getting message export jobs"))
|
|
}
|
|
elasticPostIndexingJobs, err := a.Srv().Store().Job().GetAllByTypePage(c, model.JobTypeElasticsearchPostIndexing, 0, 2)
|
|
if err != nil {
|
|
rErr = multierror.Append(errors.Wrap(err, "error while getting ES post indexing jobs"))
|
|
}
|
|
elasticPostAggregationJobs, err := a.Srv().Store().Job().GetAllByTypePage(c, model.JobTypeElasticsearchPostAggregation, 0, 2)
|
|
if err != nil {
|
|
rErr = multierror.Append(errors.Wrap(err, "error while getting ES post aggregation jobs"))
|
|
}
|
|
blevePostIndexingJobs, err := a.Srv().Store().Job().GetAllByTypePage(c, model.JobTypeBlevePostIndexing, 0, 2)
|
|
if err != nil {
|
|
rErr = multierror.Append(errors.Wrap(err, "error while getting bleve post indexing jobs"))
|
|
}
|
|
ldapSyncJobs, err := a.Srv().Store().Job().GetAllByTypePage(c, model.JobTypeLdapSync, 0, 2)
|
|
if err != nil {
|
|
rErr = multierror.Append(errors.Wrap(err, "error while getting LDAP sync jobs"))
|
|
}
|
|
migrationJobs, err := a.Srv().Store().Job().GetAllByTypePage(c, model.JobTypeMigrations, 0, 2)
|
|
if err != nil {
|
|
rErr = multierror.Append(errors.Wrap(err, "error while getting migration jobs"))
|
|
}
|
|
|
|
// Creating the struct for support packet yaml file
|
|
supportPacket := model.SupportPacket{
|
|
/* Build information */
|
|
ServerOS: runtime.GOOS,
|
|
ServerArchitecture: runtime.GOARCH,
|
|
ServerVersion: model.CurrentVersion,
|
|
BuildHash: model.BuildHash,
|
|
|
|
/* DB */
|
|
DatabaseType: databaseType,
|
|
DatabaseVersion: databaseVersion,
|
|
DatabaseSchemaVersion: databaseSchemaVersion,
|
|
WebsocketConnections: websocketConnections,
|
|
MasterDbConnections: masterDbConnections,
|
|
ReplicaDbConnections: replicaDbConnections,
|
|
|
|
/* Cluster */
|
|
ClusterID: clusterID,
|
|
|
|
/* File store */
|
|
FileDriver: fileDriver,
|
|
FileStatus: fileStatus,
|
|
|
|
/* LDAP */
|
|
LdapVendorName: vendorName,
|
|
LdapVendorVersion: vendorVersion,
|
|
|
|
/* Elastic Search */
|
|
ElasticServerVersion: elasticServerVersion,
|
|
ElasticServerPlugins: elasticServerPlugins,
|
|
|
|
/* License */
|
|
LicenseTo: licenseTo,
|
|
LicenseSupportedUsers: supportedUsers,
|
|
LicenseIsTrial: isTrial,
|
|
|
|
/* Server stats */
|
|
ActiveUsers: int(uniqueUserCount),
|
|
DailyActiveUsers: dailyActiveUsers,
|
|
MonthlyActiveUsers: monthlyActiveUsers,
|
|
InactiveUserCount: inactiveUserCount,
|
|
TotalPosts: totalPosts,
|
|
TotalChannels: totalChannels,
|
|
TotalTeams: totalTeams,
|
|
|
|
/* Jobs */
|
|
DataRetentionJobs: dataRetentionJobs,
|
|
MessageExportJobs: messageExportJobs,
|
|
ElasticPostIndexingJobs: elasticPostIndexingJobs,
|
|
ElasticPostAggregationJobs: elasticPostAggregationJobs,
|
|
BlevePostIndexingJobs: blevePostIndexingJobs,
|
|
LdapSyncJobs: ldapSyncJobs,
|
|
MigrationJobs: migrationJobs,
|
|
}
|
|
|
|
// Marshal to a Yaml File
|
|
supportPacketYaml, err := yaml.Marshal(&supportPacket)
|
|
if err != nil {
|
|
rErr = multierror.Append(errors.Wrap(err, "failed to marshal support package into yaml"))
|
|
}
|
|
|
|
fileData := &model.FileData{
|
|
Filename: "support_packet.yaml",
|
|
Body: supportPacketYaml,
|
|
}
|
|
return fileData, rErr.ErrorOrNil()
|
|
}
|
|
|
|
func (a *App) createPluginsFile(_ request.CTX) (*model.FileData, error) {
|
|
// Getting the plugins installed on the server, prettify it, and then add them to the file data array
|
|
pluginsResponse, appErr := a.GetPlugins()
|
|
if appErr != nil {
|
|
return nil, errors.Wrap(appErr, "failed to get plugin list for support package")
|
|
}
|
|
|
|
pluginsPrettyJSON, err := json.MarshalIndent(pluginsResponse, "", " ")
|
|
if err != nil {
|
|
return nil, errors.Wrap(err, "failed to marshal plugin list into json")
|
|
}
|
|
|
|
fileData := &model.FileData{
|
|
Filename: "plugins.json",
|
|
Body: pluginsPrettyJSON,
|
|
}
|
|
return fileData, nil
|
|
}
|
|
|
|
func (a *App) getNotificationsLog(_ request.CTX) (*model.FileData, error) {
|
|
if !*a.Config().NotificationLogSettings.EnableFile {
|
|
return nil, errors.New("Unable to retrieve notifications.log because LogSettings: EnableFile is set to false")
|
|
}
|
|
|
|
notificationsLog := config.GetNotificationsLogFileLocation(*a.Config().LogSettings.FileLocation)
|
|
notificationsLogFileData, err := os.ReadFile(notificationsLog)
|
|
if err != nil {
|
|
return nil, errors.Wrapf(err, "failed read notifcation log file at path %s", notificationsLog)
|
|
}
|
|
|
|
fileData := &model.FileData{
|
|
Filename: "notifications.log",
|
|
Body: notificationsLogFileData,
|
|
}
|
|
return fileData, nil
|
|
}
|
|
|
|
func (a *App) GetMattermostLog(ctx request.CTX) (*model.FileData, error) {
|
|
if !*a.Config().LogSettings.EnableFile {
|
|
return nil, errors.New("Unable to retrieve mattermost.log because LogSettings: EnableFile is set to false")
|
|
}
|
|
|
|
mattermostLog := config.GetLogFileLocation(*a.Config().LogSettings.FileLocation)
|
|
mattermostLogFileData, err := os.ReadFile(mattermostLog)
|
|
if err != nil {
|
|
return nil, errors.Wrapf(err, "failed read mattermost log file at path %s", mattermostLog)
|
|
}
|
|
|
|
fileData := &model.FileData{
|
|
Filename: "mattermost.log",
|
|
Body: mattermostLogFileData,
|
|
}
|
|
return fileData, nil
|
|
}
|
|
|
|
func (a *App) createSanitizedConfigFile(_ request.CTX) (*model.FileData, error) {
|
|
// Getting sanitized config, prettifying it, and then adding it to our file data array
|
|
sanitizedConfigPrettyJSON, err := json.MarshalIndent(a.GetSanitizedConfig(), "", " ")
|
|
if err != nil {
|
|
return nil, errors.Wrap(err, "failed to sanitized config into json")
|
|
}
|
|
|
|
fileData := &model.FileData{
|
|
Filename: "sanitized_config.json",
|
|
Body: sanitizedConfigPrettyJSON,
|
|
}
|
|
return fileData, nil
|
|
}
|
|
|
|
func (a *App) createCPUProfile(_ request.CTX) (*model.FileData, error) {
|
|
var b bytes.Buffer
|
|
|
|
err := pprof.StartCPUProfile(&b)
|
|
if err != nil {
|
|
return nil, errors.Wrap(err, "failed to start CPU profile")
|
|
}
|
|
|
|
time.Sleep(cpuProfileDuration)
|
|
|
|
pprof.StopCPUProfile()
|
|
|
|
fileData := &model.FileData{
|
|
Filename: "cpu.prof",
|
|
Body: b.Bytes(),
|
|
}
|
|
return fileData, nil
|
|
}
|
|
|
|
func (a *App) createHeapProfile(request.CTX) (*model.FileData, error) {
|
|
var b bytes.Buffer
|
|
|
|
err := pprof.Lookup("heap").WriteTo(&b, 0)
|
|
if err != nil {
|
|
return nil, errors.Wrap(err, "failed to lookup heap profile")
|
|
}
|
|
|
|
fileData := &model.FileData{
|
|
Filename: "heap.prof",
|
|
Body: b.Bytes(),
|
|
}
|
|
return fileData, nil
|
|
}
|
|
|
|
func (a *App) createGoroutineProfile(_ request.CTX) (*model.FileData, error) {
|
|
var b bytes.Buffer
|
|
|
|
err := pprof.Lookup("goroutine").WriteTo(&b, 2)
|
|
if err != nil {
|
|
return nil, errors.Wrap(err, "failed to lookup goroutine profile")
|
|
}
|
|
|
|
fileData := &model.FileData{
|
|
Filename: "goroutines",
|
|
Body: b.Bytes(),
|
|
}
|
|
return fileData, nil
|
|
}
|