2021-04-18 15:41:13 -04: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/>.
|
2017-01-17 13:02:58 -05:00
|
|
|
|
|
|
|
package madmin
|
|
|
|
|
|
|
|
import (
|
2020-03-20 18:00:44 -04:00
|
|
|
"context"
|
2017-03-31 20:55:15 -04:00
|
|
|
"encoding/json"
|
2017-01-17 13:02:58 -05:00
|
|
|
"fmt"
|
2017-03-31 20:55:15 -04:00
|
|
|
"io/ioutil"
|
2017-01-17 13:02:58 -05:00
|
|
|
"net/http"
|
|
|
|
"net/url"
|
2021-03-04 17:36:23 -05:00
|
|
|
"sort"
|
2017-01-17 13:02:58 -05:00
|
|
|
"time"
|
|
|
|
)
|
|
|
|
|
2019-03-14 16:08:51 -04:00
|
|
|
// HealScanMode represents the type of healing scan
|
|
|
|
type HealScanMode int
|
|
|
|
|
|
|
|
const (
|
2021-01-27 13:21:14 -05:00
|
|
|
// HealUnknownScan default is unknown
|
|
|
|
HealUnknownScan HealScanMode = iota
|
|
|
|
|
2019-03-14 16:08:51 -04:00
|
|
|
// HealNormalScan checks if parts are present and not outdated
|
2021-01-27 13:21:14 -05:00
|
|
|
HealNormalScan
|
|
|
|
|
2019-03-14 16:08:51 -04:00
|
|
|
// HealDeepScan checks for parts bitrot checksums
|
|
|
|
HealDeepScan
|
|
|
|
)
|
|
|
|
|
2018-01-22 17:54:55 -05:00
|
|
|
// HealOpts - collection of options for a heal sequence
|
|
|
|
type HealOpts struct {
|
2019-03-14 16:08:51 -04:00
|
|
|
Recursive bool `json:"recursive"`
|
|
|
|
DryRun bool `json:"dryRun"`
|
|
|
|
Remove bool `json:"remove"`
|
2021-01-05 16:24:22 -05:00
|
|
|
Recreate bool `json:"recreate"` // only used when bucket needs to be healed
|
2019-03-14 16:08:51 -04:00
|
|
|
ScanMode HealScanMode `json:"scanMode"`
|
2021-04-19 13:30:42 -04:00
|
|
|
NoLock bool `json:"nolock"`
|
2017-01-19 12:34:18 -05:00
|
|
|
}
|
|
|
|
|
2020-04-30 23:23:00 -04:00
|
|
|
// Equal returns true if no is same as o.
|
|
|
|
func (o HealOpts) Equal(no HealOpts) bool {
|
|
|
|
if o.Recursive != no.Recursive {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
if o.DryRun != no.DryRun {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
if o.Remove != no.Remove {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
return o.ScanMode == no.ScanMode
|
|
|
|
}
|
|
|
|
|
2018-01-22 17:54:55 -05:00
|
|
|
// HealStartSuccess - holds information about a successfully started
|
|
|
|
// heal operation
|
|
|
|
type HealStartSuccess struct {
|
|
|
|
ClientToken string `json:"clientToken"`
|
|
|
|
ClientAddress string `json:"clientAddress"`
|
|
|
|
StartTime time.Time `json:"startTime"`
|
2017-01-19 12:34:18 -05:00
|
|
|
}
|
|
|
|
|
2018-11-04 22:24:16 -05:00
|
|
|
// HealStopSuccess - holds information about a successfully stopped
|
|
|
|
// heal operation.
|
|
|
|
type HealStopSuccess HealStartSuccess
|
|
|
|
|
2018-01-22 17:54:55 -05:00
|
|
|
// HealTaskStatus - status struct for a heal task
|
|
|
|
type HealTaskStatus struct {
|
|
|
|
Summary string `json:"summary"`
|
|
|
|
FailureDetail string `json:"detail"`
|
|
|
|
StartTime time.Time `json:"startTime"`
|
|
|
|
HealSettings HealOpts `json:"settings"`
|
2017-01-19 12:34:18 -05:00
|
|
|
|
2018-01-22 17:54:55 -05:00
|
|
|
Items []HealResultItem `json:"items,omitempty"`
|
2017-01-19 12:34:18 -05:00
|
|
|
}
|
|
|
|
|
2018-01-22 17:54:55 -05:00
|
|
|
// HealItemType - specify the type of heal operation in a healing
|
|
|
|
// result
|
|
|
|
type HealItemType string
|
2017-01-17 13:02:58 -05:00
|
|
|
|
2018-01-22 17:54:55 -05:00
|
|
|
// HealItemType constants
|
2017-01-17 13:02:58 -05:00
|
|
|
const (
|
2018-01-22 17:54:55 -05:00
|
|
|
HealItemMetadata HealItemType = "metadata"
|
|
|
|
HealItemBucket = "bucket"
|
|
|
|
HealItemBucketMetadata = "bucket-metadata"
|
|
|
|
HealItemObject = "object"
|
2017-01-17 13:02:58 -05:00
|
|
|
)
|
|
|
|
|
2018-01-22 17:54:55 -05:00
|
|
|
// Drive state constants
|
2017-01-17 13:02:58 -05:00
|
|
|
const (
|
2020-03-27 17:48:30 -04:00
|
|
|
DriveStateOk string = "ok"
|
|
|
|
DriveStateOffline = "offline"
|
|
|
|
DriveStateCorrupt = "corrupt"
|
|
|
|
DriveStateMissing = "missing"
|
2020-07-13 12:51:07 -04:00
|
|
|
DriveStatePermission = "permission-denied"
|
|
|
|
DriveStateFaulty = "faulty"
|
|
|
|
DriveStateUnknown = "unknown"
|
2020-03-27 17:48:30 -04:00
|
|
|
DriveStateUnformatted = "unformatted" // only returned by disk
|
2017-01-17 13:02:58 -05:00
|
|
|
)
|
|
|
|
|
2018-02-15 20:45:57 -05:00
|
|
|
// HealDriveInfo - struct for an individual drive info item.
|
|
|
|
type HealDriveInfo struct {
|
|
|
|
UUID string `json:"uuid"`
|
|
|
|
Endpoint string `json:"endpoint"`
|
|
|
|
State string `json:"state"`
|
|
|
|
}
|
|
|
|
|
2018-01-22 17:54:55 -05:00
|
|
|
// HealResultItem - struct for an individual heal result item
|
|
|
|
type HealResultItem struct {
|
|
|
|
ResultIndex int64 `json:"resultId"`
|
|
|
|
Type HealItemType `json:"type"`
|
|
|
|
Bucket string `json:"bucket"`
|
|
|
|
Object string `json:"object"`
|
2020-11-23 12:12:17 -05:00
|
|
|
VersionID string `json:"versionId"`
|
2018-01-22 17:54:55 -05:00
|
|
|
Detail string `json:"detail"`
|
|
|
|
ParityBlocks int `json:"parityBlocks,omitempty"`
|
|
|
|
DataBlocks int `json:"dataBlocks,omitempty"`
|
|
|
|
DiskCount int `json:"diskCount"`
|
2018-02-15 20:45:57 -05:00
|
|
|
SetCount int `json:"setCount"`
|
|
|
|
// below slices are from drive info.
|
|
|
|
Before struct {
|
|
|
|
Drives []HealDriveInfo `json:"drives"`
|
|
|
|
} `json:"before"`
|
|
|
|
After struct {
|
|
|
|
Drives []HealDriveInfo `json:"drives"`
|
|
|
|
} `json:"after"`
|
2018-01-22 17:54:55 -05:00
|
|
|
ObjectSize int64 `json:"objectSize"`
|
|
|
|
}
|
|
|
|
|
2018-02-15 20:45:57 -05:00
|
|
|
// GetMissingCounts - returns the number of missing disks before
|
|
|
|
// and after heal
|
|
|
|
func (hri *HealResultItem) GetMissingCounts() (b, a int) {
|
|
|
|
if hri == nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
for _, v := range hri.Before.Drives {
|
|
|
|
if v.State == DriveStateMissing {
|
|
|
|
b++
|
|
|
|
}
|
|
|
|
}
|
|
|
|
for _, v := range hri.After.Drives {
|
|
|
|
if v.State == DriveStateMissing {
|
|
|
|
a++
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
// GetOfflineCounts - returns the number of offline disks before
|
|
|
|
// and after heal
|
|
|
|
func (hri *HealResultItem) GetOfflineCounts() (b, a int) {
|
|
|
|
if hri == nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
for _, v := range hri.Before.Drives {
|
|
|
|
if v.State == DriveStateOffline {
|
|
|
|
b++
|
|
|
|
}
|
|
|
|
}
|
|
|
|
for _, v := range hri.After.Drives {
|
|
|
|
if v.State == DriveStateOffline {
|
|
|
|
a++
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
// GetCorruptedCounts - returns the number of corrupted disks before
|
|
|
|
// and after heal
|
|
|
|
func (hri *HealResultItem) GetCorruptedCounts() (b, a int) {
|
|
|
|
if hri == nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
for _, v := range hri.Before.Drives {
|
|
|
|
if v.State == DriveStateCorrupt {
|
|
|
|
b++
|
|
|
|
}
|
|
|
|
}
|
|
|
|
for _, v := range hri.After.Drives {
|
|
|
|
if v.State == DriveStateCorrupt {
|
|
|
|
a++
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return
|
2018-01-22 17:54:55 -05:00
|
|
|
}
|
|
|
|
|
2018-02-15 20:45:57 -05:00
|
|
|
// GetOnlineCounts - returns the number of online disks before
|
|
|
|
// and after heal
|
2018-01-22 17:54:55 -05:00
|
|
|
func (hri *HealResultItem) GetOnlineCounts() (b, a int) {
|
|
|
|
if hri == nil {
|
|
|
|
return
|
|
|
|
}
|
2018-02-15 20:45:57 -05:00
|
|
|
for _, v := range hri.Before.Drives {
|
|
|
|
if v.State == DriveStateOk {
|
2018-01-22 17:54:55 -05:00
|
|
|
b++
|
2017-01-17 13:02:58 -05:00
|
|
|
}
|
2017-01-19 12:34:18 -05:00
|
|
|
}
|
2018-02-15 20:45:57 -05:00
|
|
|
for _, v := range hri.After.Drives {
|
|
|
|
if v.State == DriveStateOk {
|
2018-01-22 17:54:55 -05:00
|
|
|
a++
|
2017-01-19 12:34:18 -05:00
|
|
|
}
|
|
|
|
}
|
2018-01-22 17:54:55 -05:00
|
|
|
return
|
2017-01-19 12:34:18 -05:00
|
|
|
}
|
|
|
|
|
2018-01-22 17:54:55 -05:00
|
|
|
// Heal - API endpoint to start heal and to fetch status
|
2018-11-04 22:24:16 -05:00
|
|
|
// forceStart and forceStop are mutually exclusive, you can either
|
|
|
|
// set one of them to 'true'. If both are set 'forceStart' will be
|
|
|
|
// honored.
|
2020-03-20 18:00:44 -04:00
|
|
|
func (adm *AdminClient) Heal(ctx context.Context, bucket, prefix string,
|
|
|
|
healOpts HealOpts, clientToken string, forceStart, forceStop bool) (
|
2018-01-22 17:54:55 -05:00
|
|
|
healStart HealStartSuccess, healTaskStatus HealTaskStatus, err error) {
|
2017-01-17 13:02:58 -05:00
|
|
|
|
2018-11-04 22:24:16 -05:00
|
|
|
if forceStart && forceStop {
|
|
|
|
return healStart, healTaskStatus, ErrInvalidArgument("forceStart and forceStop set to true is not allowed")
|
|
|
|
}
|
|
|
|
|
2018-01-22 17:54:55 -05:00
|
|
|
body, err := json.Marshal(healOpts)
|
2017-01-17 13:02:58 -05:00
|
|
|
if err != nil {
|
2018-01-22 17:54:55 -05:00
|
|
|
return healStart, healTaskStatus, err
|
2017-01-17 13:02:58 -05:00
|
|
|
}
|
|
|
|
|
2019-10-23 00:01:14 -04:00
|
|
|
path := fmt.Sprintf(adminAPIPrefix+"/heal/%s", bucket)
|
2018-01-22 17:54:55 -05:00
|
|
|
if bucket != "" && prefix != "" {
|
|
|
|
path += "/" + prefix
|
2017-01-17 13:02:58 -05:00
|
|
|
}
|
|
|
|
|
2018-01-22 17:54:55 -05:00
|
|
|
// execute POST request to heal api
|
|
|
|
queryVals := make(url.Values)
|
|
|
|
if clientToken != "" {
|
|
|
|
queryVals.Set("clientToken", clientToken)
|
2018-01-25 08:54:00 -05:00
|
|
|
body = []byte{}
|
2017-01-17 13:02:58 -05:00
|
|
|
}
|
2018-11-04 22:24:16 -05:00
|
|
|
|
|
|
|
// Anyone can be set, either force start or forceStop.
|
2018-01-22 17:54:55 -05:00
|
|
|
if forceStart {
|
|
|
|
queryVals.Set("forceStart", "true")
|
2018-11-04 22:24:16 -05:00
|
|
|
} else if forceStop {
|
|
|
|
queryVals.Set("forceStop", "true")
|
2017-01-17 13:02:58 -05:00
|
|
|
}
|
|
|
|
|
2020-03-20 18:00:44 -04:00
|
|
|
resp, err := adm.executeMethod(ctx,
|
|
|
|
http.MethodPost, requestData{
|
|
|
|
relPath: path,
|
|
|
|
content: body,
|
|
|
|
queryValues: queryVals,
|
|
|
|
})
|
2017-01-17 13:02:58 -05:00
|
|
|
defer closeResponse(resp)
|
|
|
|
if err != nil {
|
2018-01-22 17:54:55 -05:00
|
|
|
return healStart, healTaskStatus, err
|
2017-01-17 13:02:58 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
if resp.StatusCode != http.StatusOK {
|
2018-01-22 17:54:55 -05:00
|
|
|
return healStart, healTaskStatus, httpRespToErrorResponse(resp)
|
2017-03-31 20:55:15 -04:00
|
|
|
}
|
|
|
|
|
2018-01-22 17:54:55 -05:00
|
|
|
respBytes, err := ioutil.ReadAll(resp.Body)
|
2017-01-23 03:32:55 -05:00
|
|
|
if err != nil {
|
2018-01-22 17:54:55 -05:00
|
|
|
return healStart, healTaskStatus, err
|
2017-01-23 03:32:55 -05:00
|
|
|
}
|
|
|
|
|
2018-01-22 17:54:55 -05:00
|
|
|
// Was it a status request?
|
|
|
|
if clientToken == "" {
|
2018-11-04 22:24:16 -05:00
|
|
|
// As a special operation forceStop would return a
|
|
|
|
// similar struct as healStart will have the
|
|
|
|
// heal sequence information about the heal which
|
|
|
|
// was stopped.
|
2018-01-22 17:54:55 -05:00
|
|
|
err = json.Unmarshal(respBytes, &healStart)
|
|
|
|
} else {
|
|
|
|
err = json.Unmarshal(respBytes, &healTaskStatus)
|
2017-01-23 03:32:55 -05:00
|
|
|
}
|
2018-11-04 22:24:16 -05:00
|
|
|
if err != nil {
|
|
|
|
// May be the server responded with error after success
|
|
|
|
// message, handle it separately here.
|
|
|
|
var errResp ErrorResponse
|
|
|
|
err = json.Unmarshal(respBytes, &errResp)
|
|
|
|
if err != nil {
|
|
|
|
// Unknown structure return error anyways.
|
|
|
|
return healStart, healTaskStatus, err
|
|
|
|
}
|
|
|
|
return healStart, healTaskStatus, errResp
|
|
|
|
}
|
|
|
|
return healStart, healTaskStatus, nil
|
2017-01-23 03:32:55 -05:00
|
|
|
}
|
2019-06-25 19:42:24 -04:00
|
|
|
|
|
|
|
// BgHealState represents the status of the background heal
|
|
|
|
type BgHealState struct {
|
|
|
|
ScannedItemsCount int64
|
2021-03-04 17:36:23 -05:00
|
|
|
|
|
|
|
HealDisks []string
|
|
|
|
|
|
|
|
// SetStatus contains information for each set.
|
|
|
|
Sets []SetStatus `json:"sets"`
|
|
|
|
}
|
|
|
|
|
|
|
|
// SetStatus contains information about the heal status of a set.
|
|
|
|
type SetStatus struct {
|
|
|
|
ID string `json:"id"`
|
|
|
|
PoolIndex int `json:"pool_index"`
|
|
|
|
SetIndex int `json:"set_index"`
|
|
|
|
HealStatus string `json:"heal_status"`
|
|
|
|
HealPriority string `json:"heal_priority"`
|
|
|
|
Disks []Disk `json:"disks"`
|
|
|
|
}
|
|
|
|
|
|
|
|
// HealingDisk contains information about
|
|
|
|
type HealingDisk struct {
|
|
|
|
// Copied from cmd/background-newdisks-heal-ops.go
|
|
|
|
// When adding new field, update (*healingTracker).toHealingDisk
|
|
|
|
|
|
|
|
ID string `json:"id"`
|
|
|
|
PoolIndex int `json:"pool_index"`
|
|
|
|
SetIndex int `json:"set_index"`
|
|
|
|
DiskIndex int `json:"disk_index"`
|
|
|
|
Endpoint string `json:"endpoint"`
|
|
|
|
Path string `json:"path"`
|
|
|
|
Started time.Time `json:"started"`
|
|
|
|
LastUpdate time.Time `json:"last_update"`
|
|
|
|
ObjectsHealed uint64 `json:"objects_healed"`
|
|
|
|
ObjectsFailed uint64 `json:"objects_failed"`
|
|
|
|
BytesDone uint64 `json:"bytes_done"`
|
|
|
|
BytesFailed uint64 `json:"bytes_failed"`
|
|
|
|
|
|
|
|
// Last object scanned.
|
|
|
|
Bucket string `json:"current_bucket"`
|
|
|
|
Object string `json:"current_object"`
|
|
|
|
|
|
|
|
// Filled on startup/restarts.
|
|
|
|
QueuedBuckets []string `json:"queued_buckets"`
|
|
|
|
|
|
|
|
// Filled during heal.
|
|
|
|
HealedBuckets []string `json:"healed_buckets"`
|
|
|
|
// future add more tracking capabilities
|
|
|
|
}
|
|
|
|
|
|
|
|
// Merge others into b.
|
|
|
|
func (b *BgHealState) Merge(others ...BgHealState) {
|
|
|
|
for _, other := range others {
|
|
|
|
b.ScannedItemsCount += other.ScannedItemsCount
|
|
|
|
if len(b.Sets) == 0 {
|
|
|
|
b.Sets = make([]SetStatus, len(other.Sets))
|
|
|
|
copy(b.Sets, other.Sets)
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
// Add disk if not present.
|
|
|
|
// If present select the one with latest lastupdate.
|
|
|
|
addSet := func(set SetStatus) {
|
|
|
|
for eSetIdx, existing := range b.Sets {
|
|
|
|
if existing.ID != set.ID {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
if len(existing.Disks) < len(set.Disks) {
|
|
|
|
b.Sets[eSetIdx].Disks = set.Disks
|
|
|
|
}
|
|
|
|
if len(existing.Disks) < len(set.Disks) {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
for i, disk := range set.Disks {
|
|
|
|
// Disks should be the same.
|
|
|
|
if disk.HealInfo != nil {
|
|
|
|
existing.Disks[i].HealInfo = disk.HealInfo
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return
|
|
|
|
}
|
|
|
|
b.Sets = append(b.Sets, set)
|
|
|
|
}
|
|
|
|
for _, disk := range other.Sets {
|
|
|
|
addSet(disk)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
sort.Slice(b.Sets, func(i, j int) bool {
|
|
|
|
if b.Sets[i].PoolIndex != b.Sets[j].PoolIndex {
|
|
|
|
return b.Sets[i].PoolIndex < b.Sets[j].PoolIndex
|
|
|
|
}
|
|
|
|
return b.Sets[i].SetIndex < b.Sets[j].SetIndex
|
|
|
|
})
|
2019-06-25 19:42:24 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
// BackgroundHealStatus returns the background heal status of the
|
|
|
|
// current server or cluster.
|
2020-03-20 18:00:44 -04:00
|
|
|
func (adm *AdminClient) BackgroundHealStatus(ctx context.Context) (BgHealState, error) {
|
2019-06-25 19:42:24 -04:00
|
|
|
// Execute POST request to background heal status api
|
2020-03-20 18:00:44 -04:00
|
|
|
resp, err := adm.executeMethod(ctx,
|
|
|
|
http.MethodPost,
|
|
|
|
requestData{relPath: adminAPIPrefix + "/background-heal/status"})
|
2019-06-25 19:42:24 -04:00
|
|
|
if err != nil {
|
|
|
|
return BgHealState{}, err
|
|
|
|
}
|
|
|
|
defer closeResponse(resp)
|
|
|
|
|
|
|
|
if resp.StatusCode != http.StatusOK {
|
|
|
|
return BgHealState{}, httpRespToErrorResponse(resp)
|
|
|
|
}
|
|
|
|
|
|
|
|
respBytes, err := ioutil.ReadAll(resp.Body)
|
|
|
|
if err != nil {
|
|
|
|
return BgHealState{}, err
|
|
|
|
}
|
|
|
|
|
|
|
|
var healState BgHealState
|
|
|
|
|
|
|
|
err = json.Unmarshal(respBytes, &healState)
|
|
|
|
if err != nil {
|
|
|
|
return BgHealState{}, err
|
|
|
|
}
|
|
|
|
return healState, nil
|
|
|
|
}
|