2017-01-17 23:32:58 +05:30
|
|
|
/*
|
2018-02-15 17:45:57 -08:00
|
|
|
* Minio Cloud Storage, (C) 2017, 2018 Minio, Inc.
|
2017-01-17 23:32:58 +05:30
|
|
|
*
|
|
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
* you may not use this file except in compliance with the License.
|
|
|
|
* You may obtain a copy of the License at
|
|
|
|
*
|
|
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
*
|
|
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
* See the License for the specific language governing permissions and
|
|
|
|
* limitations under the License.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
package madmin
|
|
|
|
|
|
|
|
import (
|
2018-01-22 14:54:55 -08:00
|
|
|
"bytes"
|
2017-04-01 06:25:15 +05:30
|
|
|
"encoding/json"
|
2017-01-17 23:32:58 +05:30
|
|
|
"fmt"
|
2018-01-22 14:54:55 -08:00
|
|
|
"io"
|
2017-04-01 06:25:15 +05:30
|
|
|
"io/ioutil"
|
2017-01-17 23:32:58 +05:30
|
|
|
"net/http"
|
|
|
|
"net/url"
|
|
|
|
"time"
|
|
|
|
)
|
|
|
|
|
2018-01-22 14:54:55 -08:00
|
|
|
// HealOpts - collection of options for a heal sequence
|
|
|
|
type HealOpts struct {
|
|
|
|
Recursive bool `json:"recursive"`
|
|
|
|
DryRun bool `json:"dryRun"`
|
2017-01-19 18:34:18 +01:00
|
|
|
}
|
|
|
|
|
2018-01-22 14:54:55 -08: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 18:34:18 +01:00
|
|
|
}
|
|
|
|
|
2018-01-22 14:54:55 -08: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"`
|
|
|
|
NumDisks int `json:"numDisks"`
|
2017-01-19 18:34:18 +01:00
|
|
|
|
2018-01-22 14:54:55 -08:00
|
|
|
Items []HealResultItem `json:"items,omitempty"`
|
2017-01-19 18:34:18 +01:00
|
|
|
}
|
|
|
|
|
2018-01-22 14:54:55 -08:00
|
|
|
// HealItemType - specify the type of heal operation in a healing
|
|
|
|
// result
|
|
|
|
type HealItemType string
|
2017-01-17 23:32:58 +05:30
|
|
|
|
2018-01-22 14:54:55 -08:00
|
|
|
// HealItemType constants
|
2017-01-17 23:32:58 +05:30
|
|
|
const (
|
2018-01-22 14:54:55 -08:00
|
|
|
HealItemMetadata HealItemType = "metadata"
|
|
|
|
HealItemBucket = "bucket"
|
|
|
|
HealItemBucketMetadata = "bucket-metadata"
|
|
|
|
HealItemObject = "object"
|
2017-01-17 23:32:58 +05:30
|
|
|
)
|
|
|
|
|
2018-01-22 14:54:55 -08:00
|
|
|
// Drive state constants
|
2017-01-17 23:32:58 +05:30
|
|
|
const (
|
2018-01-22 14:54:55 -08:00
|
|
|
DriveStateOk string = "ok"
|
|
|
|
DriveStateOffline = "offline"
|
|
|
|
DriveStateCorrupt = "corrupt"
|
|
|
|
DriveStateMissing = "missing"
|
2017-01-17 23:32:58 +05:30
|
|
|
)
|
|
|
|
|
2018-02-15 17:45:57 -08: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 14:54:55 -08: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"`
|
|
|
|
Detail string `json:"detail"`
|
|
|
|
ParityBlocks int `json:"parityBlocks,omitempty"`
|
|
|
|
DataBlocks int `json:"dataBlocks,omitempty"`
|
|
|
|
DiskCount int `json:"diskCount"`
|
2018-02-15 17:45:57 -08: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 14:54:55 -08:00
|
|
|
ObjectSize int64 `json:"objectSize"`
|
|
|
|
}
|
|
|
|
|
2018-02-15 17:45:57 -08: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 14:54:55 -08:00
|
|
|
}
|
|
|
|
|
2018-02-15 17:45:57 -08:00
|
|
|
// GetOnlineCounts - returns the number of online disks before
|
|
|
|
// and after heal
|
2018-01-22 14:54:55 -08:00
|
|
|
func (hri *HealResultItem) GetOnlineCounts() (b, a int) {
|
|
|
|
if hri == nil {
|
|
|
|
return
|
|
|
|
}
|
2018-02-15 17:45:57 -08:00
|
|
|
for _, v := range hri.Before.Drives {
|
|
|
|
if v.State == DriveStateOk {
|
2018-01-22 14:54:55 -08:00
|
|
|
b++
|
2017-01-17 23:32:58 +05:30
|
|
|
}
|
2017-01-19 18:34:18 +01:00
|
|
|
}
|
2018-02-15 17:45:57 -08:00
|
|
|
for _, v := range hri.After.Drives {
|
|
|
|
if v.State == DriveStateOk {
|
2018-01-22 14:54:55 -08:00
|
|
|
a++
|
2017-01-19 18:34:18 +01:00
|
|
|
}
|
|
|
|
}
|
2018-01-22 14:54:55 -08:00
|
|
|
return
|
2017-01-19 18:34:18 +01:00
|
|
|
}
|
|
|
|
|
2018-01-22 14:54:55 -08:00
|
|
|
// Heal - API endpoint to start heal and to fetch status
|
|
|
|
func (adm *AdminClient) Heal(bucket, prefix string, healOpts HealOpts,
|
|
|
|
clientToken string, forceStart bool) (
|
|
|
|
healStart HealStartSuccess, healTaskStatus HealTaskStatus, err error) {
|
2017-01-17 23:32:58 +05:30
|
|
|
|
2018-01-22 14:54:55 -08:00
|
|
|
body, err := json.Marshal(healOpts)
|
2017-01-17 23:32:58 +05:30
|
|
|
if err != nil {
|
2018-01-22 14:54:55 -08:00
|
|
|
return healStart, healTaskStatus, err
|
2017-01-17 23:32:58 +05:30
|
|
|
}
|
|
|
|
|
2018-01-22 14:54:55 -08:00
|
|
|
path := fmt.Sprintf("/v1/heal/%s", bucket)
|
|
|
|
if bucket != "" && prefix != "" {
|
|
|
|
path += "/" + prefix
|
2017-01-17 23:32:58 +05:30
|
|
|
}
|
|
|
|
|
2018-01-22 14:54:55 -08:00
|
|
|
// execute POST request to heal api
|
|
|
|
queryVals := make(url.Values)
|
|
|
|
var contentBody io.Reader
|
|
|
|
if clientToken != "" {
|
|
|
|
queryVals.Set("clientToken", clientToken)
|
2018-01-25 05:54:00 -08:00
|
|
|
body = []byte{}
|
2018-01-22 14:54:55 -08:00
|
|
|
} else {
|
|
|
|
// Set a body only if clientToken is not given
|
|
|
|
contentBody = bytes.NewReader(body)
|
2017-01-17 23:32:58 +05:30
|
|
|
}
|
2018-01-22 14:54:55 -08:00
|
|
|
if forceStart {
|
|
|
|
queryVals.Set("forceStart", "true")
|
2017-01-17 23:32:58 +05:30
|
|
|
}
|
|
|
|
|
2018-01-22 14:54:55 -08:00
|
|
|
resp, err := adm.executeMethod("POST", requestData{
|
|
|
|
relPath: path,
|
|
|
|
contentBody: contentBody,
|
|
|
|
contentSHA256Bytes: sum256(body),
|
|
|
|
queryValues: queryVals,
|
|
|
|
})
|
2017-01-17 23:32:58 +05:30
|
|
|
defer closeResponse(resp)
|
|
|
|
if err != nil {
|
2018-01-22 14:54:55 -08:00
|
|
|
return healStart, healTaskStatus, err
|
2017-01-17 23:32:58 +05:30
|
|
|
}
|
|
|
|
|
|
|
|
if resp.StatusCode != http.StatusOK {
|
2018-01-22 14:54:55 -08:00
|
|
|
return healStart, healTaskStatus, httpRespToErrorResponse(resp)
|
2017-04-01 06:25:15 +05:30
|
|
|
}
|
|
|
|
|
2018-01-22 14:54:55 -08:00
|
|
|
respBytes, err := ioutil.ReadAll(resp.Body)
|
2017-01-23 14:02:55 +05:30
|
|
|
if err != nil {
|
2018-01-22 14:54:55 -08:00
|
|
|
return healStart, healTaskStatus, err
|
2017-01-23 14:02:55 +05:30
|
|
|
}
|
|
|
|
|
2018-01-22 14:54:55 -08:00
|
|
|
// Was it a status request?
|
|
|
|
if clientToken == "" {
|
|
|
|
err = json.Unmarshal(respBytes, &healStart)
|
|
|
|
} else {
|
|
|
|
err = json.Unmarshal(respBytes, &healTaskStatus)
|
2017-01-23 14:02:55 +05:30
|
|
|
}
|
2018-01-22 14:54:55 -08:00
|
|
|
return healStart, healTaskStatus, err
|
2017-01-23 14:02:55 +05:30
|
|
|
}
|