1
0
mirror of https://github.com/minio/mc.git synced 2025-12-08 22:28:28 +03:00
Files
mc/cmd/support-profile.go
Shireesh Anjal 206d017878 Expose health diagnostics related functions (#4885)
So that dependent projects (e.g. console) can make use of them instead
of duplicating the logic.
2024-03-30 08:28:09 -07:00

267 lines
7.7 KiB
Go

// Copyright (c) 2015-2022 MinIO, Inc.
//
// This file is part of MinIO Object Storage stack
//
// This program is free software: you can redistribute it and/or modify
// it under the terms of the GNU Affero General Public License as published by
// the Free Software Foundation, either version 3 of the License, or
// (at your option) any later version.
//
// This program is distributed in the hope that it will be useful
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
// GNU Affero General Public License for more details.
//
// You should have received a copy of the GNU Affero General Public License
// along with this program. If not, see <http://www.gnu.org/licenses/>.
package cmd
import (
"fmt"
"io"
"os"
"strings"
"time"
"github.com/minio/cli"
"github.com/minio/madmin-go/v3"
"github.com/minio/mc/pkg/probe"
"github.com/minio/minio-go/v7/pkg/set"
"github.com/minio/pkg/v2/console"
)
// profile command flags.
var (
profileFlags = append([]cli.Flag{
cli.IntFlag{
Name: "duration",
Usage: "profile for the specified duration in seconds",
Value: 10,
},
cli.StringFlag{
Name: "type",
Usage: "profiler type, possible values are 'cpu', 'cpuio', 'mem', 'block', 'mutex', 'trace', 'threads' and 'goroutines'",
Value: "cpu,mem,block,mutex,goroutines",
},
}, subnetCommonFlags...)
)
const profileFile = "profile.zip"
type supportProfileMessage struct {
Status string `json:"status"`
File string `json:"file,omitempty"`
Error string `json:"error,omitempty"`
}
// Colorized message for console printing.
func (s supportProfileMessage) String() string {
var msg string
if s.Error != "" {
errMsg := fmt.Sprintln("Unable to upload profile file to SUBNET: ", s.Error)
msg := console.Colorize(supportErrorMsgTag, errMsg)
infoMsg := fmt.Sprintf("Profiling data saved locally at '%s'", profileFile)
msg += console.Colorize(supportSuccessMsgTag, infoMsg)
return msg
}
if globalAirgapped {
msg = fmt.Sprintf("Profiling data saved successfully at %s", s.File)
} else {
msg = "Profiling data uploaded to SUBNET successfully"
}
return console.Colorize(supportSuccessMsgTag, msg)
}
// JSON jsonified proxy remove message
func (s supportProfileMessage) JSON() string {
return toJSON(s)
}
var supportProfileCmd = cli.Command{
Name: "profile",
Usage: "upload profile data for debugging",
Action: mainSupportProfile,
OnUsageError: onUsageError,
Before: setGlobalsFromContext,
Flags: profileFlags,
HideHelpCommand: true,
CustomHelpTemplate: `NAME:
{{.HelpName}} - {{.Usage}}
USAGE:
{{.HelpName}} [FLAGS] TARGET
FLAGS:
{{range .VisibleFlags}}{{.}}
{{end}}
EXAMPLES:
1. Profile CPU for 10 seconds on cluster with alias 'myminio' and upload results to SUBNET
{{.Prompt}} {{.HelpName}} --type cpu myminio
2. Profile CPU, Memory, Goroutines for 10 seconds on cluster with alias 'myminio' and upload results to SUBNET
{{.Prompt}} {{.HelpName}} --type cpu,mem,goroutines myminio
3. Profile CPU, Memory, Goroutines for 10 minutes on cluster with alias 'myminio' and upload results to SUBNET
{{.Prompt}} {{.HelpName}} --type cpu,mem,goroutines --duration 600 myminio
4. Profile CPU for 10 seconds on cluster with alias 'myminio', save and upload to SUBNET manually
{{.Prompt}} {{.HelpName}} --type cpu --airgap myminio
`,
}
func checkAdminProfileSyntax(ctx *cli.Context) {
s := set.CreateStringSet(string(madmin.ProfilerCPU),
string(madmin.ProfilerMEM),
string(madmin.ProfilerBlock),
string(madmin.ProfilerMutex),
string(madmin.ProfilerTrace),
string(madmin.ProfilerThreads),
string(madmin.ProfilerGoroutines),
string(madmin.ProfilerCPUIO))
// Check if the provided profiler type is known and supported
profilers := strings.Split(strings.ToLower(ctx.String("type")), ",")
for _, profiler := range profilers {
if profiler != "" {
if !s.Contains(profiler) {
fatalIf(errDummy().Trace(ctx.String("type")),
"Profiler type %s unrecognized. Possible values are: %v.", profiler, s)
}
}
}
if len(ctx.Args()) != 1 {
showCommandHelpAndExit(ctx, 1) // last argument is exit code
}
if ctx.Int("duration") < 10 {
fatal(errDummy().Trace(), "for any useful profiling one must run it for atleast 10 seconds")
}
}
// moveFile - os.Rename cannot handle cross device renames, in our situation
// it is possible that /tmp is mounted from a separate partition and current
// working directory is a different partition. To allow all situations to
// be handled appropriately use this function instead of os.Rename()
func moveFile(sourcePath, destPath string) error {
inputFile, e := os.Open(sourcePath)
if e != nil {
return e
}
outputFile, e := os.Create(destPath)
if e != nil {
inputFile.Close()
return e
}
defer outputFile.Close()
if _, e = io.Copy(outputFile, inputFile); e != nil {
inputFile.Close()
return e
}
// The copy was successful, so now delete the original file
inputFile.Close()
return os.Remove(sourcePath)
}
func saveProfileFile(data io.ReadCloser) {
// Create profile zip file
tmpFile, e := os.CreateTemp("", "mc-profile-")
fatalIf(probe.NewError(e), "Unable to download profile data.")
// Copy zip content to target download file
_, e = io.Copy(tmpFile, data)
fatalIf(probe.NewError(e), "Unable to download profile data.")
// Close everything
data.Close()
tmpFile.Close()
downloadedFile := profileFile + "." + time.Now().Format(dateTimeFormatFilename)
fi, e := os.Stat(profileFile)
if e == nil && !fi.IsDir() {
e = moveFile(profileFile, downloadedFile)
fatalIf(probe.NewError(e), "Unable to create a backup of profile.zip")
} else {
if !os.IsNotExist(e) {
fatal(probe.NewError(e), "Unable to save profile data")
}
}
fatalIf(probe.NewError(moveFile(tmpFile.Name(), profileFile)), "Unable to save profile data")
}
// mainSupportProfile is the handle for "mc support profile" command.
func mainSupportProfile(ctx *cli.Context) error {
// Check for command syntax
checkAdminProfileSyntax(ctx)
setSuccessMessageColor()
setErrorMessageColor()
// Get the alias parameter from cli
aliasedURL := ctx.Args().Get(0)
alias, apiKey := initSubnetConnectivity(ctx, aliasedURL, true)
if len(apiKey) == 0 {
// api key not passed as flag. Check that the cluster is registered.
apiKey = validateClusterRegistered(alias, true)
}
// Create a new MinIO Admin Client
client := getClient(aliasedURL)
// Main execution
execSupportProfile(ctx, client, alias, apiKey)
return nil
}
func execSupportProfile(ctx *cli.Context, client *madmin.AdminClient, alias, apiKey string) {
var reqURL string
var headers map[string]string
profilers := ctx.String("type")
duration := ctx.Int("duration")
if !globalAirgapped {
// Retrieve subnet credentials (login/license) beforehand as
// it can take a long time to fetch the profile data
uploadURL := SubnetUploadURL("profile")
reqURL, headers = prepareSubnetUploadURL(uploadURL, alias, apiKey)
}
if !globalJSON {
console.Infof("Profiling '%s' for %d seconds... \n", alias, duration)
}
data, e := client.Profile(globalContext, madmin.ProfilerType(profilers), time.Second*time.Duration(duration))
fatalIf(probe.NewError(e), "Unable to save profile data")
saveProfileFile(data)
if !globalAirgapped {
_, e = (&SubnetFileUploader{
alias: alias,
FilePath: profileFile,
ReqURL: reqURL,
Headers: headers,
DeleteAfterUpload: true,
}).UploadFileToSubnet()
if e != nil {
printMsg(supportProfileMessage{
Status: "error",
Error: e.Error(),
File: profileFile,
})
return
}
printMsg(supportProfileMessage{
Status: "success",
})
} else {
printMsg(supportProfileMessage{
Status: "success",
File: profileFile,
})
}
}