2021-04-18 12:41:13 -07:00
|
|
|
// Copyright (c) 2015-2021 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/>.
|
2018-09-19 00:46:35 +01:00
|
|
|
|
|
|
|
package madmin
|
|
|
|
|
|
|
|
import (
|
2020-03-20 15:00:44 -07:00
|
|
|
"context"
|
2018-09-19 00:46:35 +01:00
|
|
|
"encoding/json"
|
|
|
|
"errors"
|
|
|
|
"fmt"
|
|
|
|
"io"
|
|
|
|
"io/ioutil"
|
|
|
|
"net/http"
|
2018-09-26 21:02:05 -07:00
|
|
|
"net/url"
|
2018-09-19 00:46:35 +01:00
|
|
|
)
|
|
|
|
|
|
|
|
// ProfilerType represents the profiler type
|
2018-09-26 21:02:05 -07:00
|
|
|
// passed to the profiler subsystem.
|
2018-09-19 00:46:35 +01:00
|
|
|
type ProfilerType string
|
|
|
|
|
2018-09-26 21:02:05 -07:00
|
|
|
// Different supported profiler types.
|
2018-09-19 00:46:35 +01:00
|
|
|
const (
|
2020-03-04 15:58:12 +01:00
|
|
|
ProfilerCPU ProfilerType = "cpu" // represents CPU profiler type
|
|
|
|
ProfilerMEM ProfilerType = "mem" // represents MEM profiler type
|
|
|
|
ProfilerBlock ProfilerType = "block" // represents Block profiler type
|
|
|
|
ProfilerMutex ProfilerType = "mutex" // represents Mutex profiler type
|
|
|
|
ProfilerTrace ProfilerType = "trace" // represents Trace profiler type
|
|
|
|
ProfilerThreads ProfilerType = "threads" // represents ThreadCreate profiler type
|
|
|
|
ProfilerGoroutines ProfilerType = "goroutines" // represents Goroutine dumps.
|
2018-09-19 00:46:35 +01:00
|
|
|
)
|
|
|
|
|
|
|
|
// StartProfilingResult holds the result of starting
|
|
|
|
// profiler result in a given node.
|
|
|
|
type StartProfilingResult struct {
|
|
|
|
NodeName string `json:"nodeName"`
|
|
|
|
Success bool `json:"success"`
|
|
|
|
Error string `json:"error"`
|
|
|
|
}
|
|
|
|
|
|
|
|
// StartProfiling makes an admin call to remotely start profiling on a standalone
|
|
|
|
// server or the whole cluster in case of a distributed setup.
|
2020-03-20 15:00:44 -07:00
|
|
|
func (adm *AdminClient) StartProfiling(ctx context.Context, profiler ProfilerType) ([]StartProfilingResult, error) {
|
2018-09-26 21:02:05 -07:00
|
|
|
v := url.Values{}
|
|
|
|
v.Set("profilerType", string(profiler))
|
2020-03-20 15:00:44 -07:00
|
|
|
resp, err := adm.executeMethod(ctx,
|
|
|
|
http.MethodPost, requestData{
|
|
|
|
relPath: adminAPIPrefix + "/profiling/start",
|
|
|
|
queryValues: v,
|
|
|
|
},
|
|
|
|
)
|
2018-09-19 00:46:35 +01:00
|
|
|
defer closeResponse(resp)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
if resp.StatusCode != http.StatusOK {
|
|
|
|
return nil, httpRespToErrorResponse(resp)
|
|
|
|
}
|
|
|
|
|
|
|
|
jsonResult, err := ioutil.ReadAll(resp.Body)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
var startResults []StartProfilingResult
|
|
|
|
err = json.Unmarshal(jsonResult, &startResults)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return startResults, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// DownloadProfilingData makes an admin call to download profiling data of a standalone
|
|
|
|
// server or of the whole cluster in case of a distributed setup.
|
2020-03-20 15:00:44 -07:00
|
|
|
func (adm *AdminClient) DownloadProfilingData(ctx context.Context) (io.ReadCloser, error) {
|
2019-10-23 09:31:14 +05:30
|
|
|
path := fmt.Sprintf(adminAPIPrefix + "/profiling/download")
|
2020-03-20 15:00:44 -07:00
|
|
|
resp, err := adm.executeMethod(ctx,
|
|
|
|
http.MethodGet, requestData{
|
|
|
|
relPath: path,
|
|
|
|
},
|
|
|
|
)
|
2018-09-19 00:46:35 +01:00
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
closeResponse(resp)
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
if resp.StatusCode != http.StatusOK {
|
|
|
|
return nil, httpRespToErrorResponse(resp)
|
|
|
|
}
|
|
|
|
|
|
|
|
if resp.Body == nil {
|
|
|
|
return nil, errors.New("body is nil")
|
|
|
|
}
|
|
|
|
|
|
|
|
return resp.Body, nil
|
|
|
|
}
|