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/>.
2016-05-05 00:48:20 +05:30
2016-08-18 16:23:42 -07:00
package cmd
2016-05-05 00:48:20 +05:30
2016-05-07 00:59:43 -07:00
import (
2018-04-05 15:04:40 -07:00
"context"
2016-05-07 00:59:43 -07:00
"encoding/json"
2020-09-28 19:39:32 -07:00
"errors"
2016-05-07 00:59:43 -07:00
"fmt"
2022-01-24 11:28:45 -08:00
"io/fs"
2022-09-19 20:05:16 +02:00
"os"
2018-07-18 20:17:35 -07:00
"reflect"
2020-01-13 22:09:10 +01:00
"sync"
2018-02-15 17:45:57 -08:00
2022-09-19 20:05:16 +02:00
"github.com/dustin/go-humanize"
2021-06-01 14:59:40 -07:00
"github.com/minio/minio/internal/color"
"github.com/minio/minio/internal/config"
"github.com/minio/minio/internal/config/storageclass"
xioutil "github.com/minio/minio/internal/ioutil"
"github.com/minio/minio/internal/logger"
2023-04-26 11:27:40 +05:30
"github.com/minio/pkg/sync/errgroup"
2017-06-12 17:40:28 -07:00
)
const (
2020-06-12 20:04:01 -07:00
// Represents Erasure backend.
formatBackendErasure = "xl"
2017-06-12 17:40:28 -07:00
2022-05-30 10:58:37 -07:00
// Represents Erasure backend - single drive
formatBackendErasureSingle = "xl-single"
2020-06-12 20:04:01 -07:00
// formatErasureV1.Erasure.Version - version '1'.
formatErasureVersionV1 = "1"
2018-02-15 17:45:57 -08:00
2020-06-12 20:04:01 -07:00
// formatErasureV2.Erasure.Version - version '2'.
formatErasureVersionV2 = "2"
2018-02-15 17:45:57 -08:00
2020-06-12 20:04:01 -07:00
// formatErasureV3.Erasure.Version - version '3'.
formatErasureVersionV3 = "3"
2018-03-15 13:55:23 -07:00
2020-06-12 20:04:01 -07:00
// Distribution algorithm used, legacy
2021-01-16 12:08:02 -08:00
formatErasureVersionV2DistributionAlgoV1 = "CRCMOD"
2020-06-12 20:04:01 -07:00
2021-01-16 12:08:02 -08:00
// Distributed algorithm used, with N/2 default parity
formatErasureVersionV3DistributionAlgoV2 = "SIPMOD"
// Distributed algorithm used, with EC:4 default parity
formatErasureVersionV3DistributionAlgoV3 = "SIPMOD+PARITY"
2018-01-08 14:30:55 -08:00
)
2017-06-12 17:40:28 -07:00
2018-02-15 17:45:57 -08:00
// Offline disk UUID represents an offline disk.
const offlineDiskUUID = "ffffffff-ffff-ffff-ffff-ffffffffffff"
// Used to detect the version of "xl" format.
2020-06-12 20:04:01 -07:00
type formatErasureVersionDetect struct {
Erasure struct {
2018-02-15 17:45:57 -08:00
Version string ` json:"version" `
} ` json:"xl" `
}
// Represents the V1 backend disk structure version
// under `.minio.sys` and actual data namespace.
2020-06-12 20:04:01 -07:00
// formatErasureV1 - structure holds format config version '1'.
type formatErasureV1 struct {
2018-01-08 14:30:55 -08:00
formatMetaV1
2020-06-12 20:04:01 -07:00
Erasure struct {
2018-01-08 14:30:55 -08:00
Version string ` json:"version" ` // Version of 'xl' format.
2022-08-04 16:10:08 -07:00
Disk string ` json:"drive" ` // Disk field carries assigned disk uuid.
2018-01-08 14:30:55 -08:00
// JBOD field carries the input disk order generated the first
// time when fresh disks were supplied.
JBOD [ ] string ` json:"jbod" `
2020-06-12 20:04:01 -07:00
} ` json:"xl" ` // Erasure field holds xl format.
2017-06-12 17:40:28 -07:00
}
2018-02-15 17:45:57 -08:00
// Represents the V2 backend disk structure version
// under `.minio.sys` and actual data namespace.
2020-06-12 20:04:01 -07:00
// formatErasureV2 - structure holds format config version '2'.
2018-07-16 20:26:42 -07:00
// The V2 format to support "large bucket" support where a bucket
// can span multiple erasure sets.
2020-06-12 20:04:01 -07:00
type formatErasureV2 struct {
2018-07-16 20:26:42 -07:00
formatMetaV1
2020-06-12 20:04:01 -07:00
Erasure struct {
2018-02-15 17:45:57 -08:00
Version string ` json:"version" ` // Version of 'xl' format.
This string ` json:"this" ` // This field carries assigned disk uuid.
// Sets field carries the input disk order generated the first
// time when fresh disks were supplied, it is a two dimensional
// array second dimension represents list of disks used per set.
Sets [ ] [ ] string ` json:"sets" `
// Distribution algorithm represents the hashing algorithm
// to pick the right set index for an object.
DistributionAlgo string ` json:"distributionAlgo" `
} ` json:"xl" `
}
2020-06-12 20:04:01 -07:00
// formatErasureV3 struct is same as formatErasureV2 struct except that formatErasureV3.Erasure.Version is "3" indicating
2018-03-15 13:55:23 -07:00
// the simplified multipart backend which is a flat hierarchy now.
// In .minio.sys/multipart we have:
2020-06-12 20:04:01 -07:00
// sha256(bucket/object)/uploadID/[xl.meta, part.1, part.2 ....]
type formatErasureV3 struct {
2018-07-16 20:26:42 -07:00
formatMetaV1
2020-06-12 20:04:01 -07:00
Erasure struct {
2018-03-15 13:55:23 -07:00
Version string ` json:"version" ` // Version of 'xl' format.
This string ` json:"this" ` // This field carries assigned disk uuid.
// Sets field carries the input disk order generated the first
// time when fresh disks were supplied, it is a two dimensional
// array second dimension represents list of disks used per set.
Sets [ ] [ ] string ` json:"sets" `
// Distribution algorithm represents the hashing algorithm
// to pick the right set index for an object.
DistributionAlgo string ` json:"distributionAlgo" `
} ` json:"xl" `
}
2022-02-04 12:21:21 -08:00
func ( f * formatErasureV3 ) Drives ( ) ( drives int ) {
for _ , set := range f . Erasure . Sets {
drives += len ( set )
}
return drives
}
2020-06-12 20:04:01 -07:00
func ( f * formatErasureV3 ) Clone ( ) * formatErasureV3 {
2019-11-21 04:24:51 -08:00
b , err := json . Marshal ( f )
if err != nil {
panic ( err )
}
2020-06-12 20:04:01 -07:00
var dst formatErasureV3
2019-11-21 04:24:51 -08:00
if err = json . Unmarshal ( b , & dst ) ; err != nil {
panic ( err )
}
return & dst
}
2020-06-12 20:04:01 -07:00
// Returns formatErasure.Erasure.Version
2021-01-16 12:08:02 -08:00
func newFormatErasureV3 ( numSets int , setLen int ) * formatErasureV3 {
2020-06-12 20:04:01 -07:00
format := & formatErasureV3 { }
2018-02-15 17:45:57 -08:00
format . Version = formatMetaVersionV1
2020-06-12 20:04:01 -07:00
format . Format = formatBackendErasure
2022-05-30 10:58:37 -07:00
if setLen == 1 {
format . Format = formatBackendErasureSingle
}
2018-07-18 20:17:35 -07:00
format . ID = mustGetUUID ( )
2020-06-12 20:04:01 -07:00
format . Erasure . Version = formatErasureVersionV3
2021-01-16 12:08:02 -08:00
format . Erasure . DistributionAlgo = formatErasureVersionV3DistributionAlgoV3
2020-06-12 20:04:01 -07:00
format . Erasure . Sets = make ( [ ] [ ] string , numSets )
2018-02-15 17:45:57 -08:00
for i := 0 ; i < numSets ; i ++ {
2020-06-12 20:04:01 -07:00
format . Erasure . Sets [ i ] = make ( [ ] string , setLen )
2018-02-15 17:45:57 -08:00
for j := 0 ; j < setLen ; j ++ {
2020-06-12 20:04:01 -07:00
format . Erasure . Sets [ i ] [ j ] = mustGetUUID ( )
2018-02-15 17:45:57 -08:00
}
}
return format
}
2020-06-12 20:04:01 -07:00
// Returns format Erasure version after reading `format.json`, returns
// successfully the version only if the backend is Erasure.
2022-01-24 11:28:45 -08:00
func formatGetBackendErasureVersion ( b [ ] byte ) ( string , error ) {
2018-02-15 17:45:57 -08:00
meta := & formatMetaV1 { }
2022-01-24 11:28:45 -08:00
if err := json . Unmarshal ( b , meta ) ; err != nil {
2018-02-15 17:45:57 -08:00
return "" , err
}
if meta . Version != formatMetaVersionV1 {
return "" , fmt . Errorf ( ` format.Version expected: %s, got: %s ` , formatMetaVersionV1 , meta . Version )
}
2022-05-30 10:58:37 -07:00
if meta . Format != formatBackendErasure && meta . Format != formatBackendErasureSingle {
2022-11-07 00:11:58 -08:00
return "" , fmt . Errorf ( ` found backend type %s, expected %s or %s - to migrate to a supported backend visit https://min.io/docs/minio/linux/operations/install-deploy-manage/migrate-fs-gateway.html ` , meta . Format , formatBackendErasure , formatBackendErasureSingle )
2019-12-23 16:31:03 -08:00
}
2020-06-12 20:04:01 -07:00
// Erasure backend found, proceed to detect version.
format := & formatErasureVersionDetect { }
2022-01-24 11:28:45 -08:00
if err := json . Unmarshal ( b , format ) ; err != nil {
2019-12-23 16:31:03 -08:00
return "" , err
}
2020-06-12 20:04:01 -07:00
return format . Erasure . Version , nil
2018-02-15 17:45:57 -08:00
}
// Migrates all previous versions to latest version of `format.json`,
// this code calls migration in sequence, such as V1 is migrated to V2
2020-11-23 08:36:49 -08:00
// first before it V2 migrates to V3.n
2022-01-24 11:28:45 -08:00
func formatErasureMigrate ( export string ) ( [ ] byte , fs . FileInfo , error ) {
2018-02-15 17:45:57 -08:00
formatPath := pathJoin ( export , minioMetaBucket , formatConfigFile )
2022-01-24 11:28:45 -08:00
formatData , formatFi , err := xioutil . ReadFileWithFileInfo ( formatPath )
2018-02-15 17:45:57 -08:00
if err != nil {
2022-01-24 11:28:45 -08:00
return nil , nil , err
2018-02-15 17:45:57 -08:00
}
2022-01-24 11:28:45 -08:00
version , err := formatGetBackendErasureVersion ( formatData )
if err != nil {
2022-08-04 16:10:08 -07:00
return nil , nil , fmt . Errorf ( "Drive %s: %w" , export , err )
2022-01-24 11:28:45 -08:00
}
migrate := func ( formatPath string , formatData [ ] byte ) ( [ ] byte , fs . FileInfo , error ) {
2022-09-19 20:05:16 +02:00
if err = os . WriteFile ( formatPath , formatData , 0 o666 ) ; err != nil {
2022-01-24 11:28:45 -08:00
return nil , nil , err
}
formatFi , err := Lstat ( formatPath )
if err != nil {
return nil , nil , err
}
return formatData , formatFi , nil
}
2018-02-15 17:45:57 -08:00
switch version {
2020-06-12 20:04:01 -07:00
case formatErasureVersionV1 :
2022-01-24 11:28:45 -08:00
formatData , err = formatErasureMigrateV1ToV2 ( formatData , version )
if err != nil {
2022-08-04 16:10:08 -07:00
return nil , nil , fmt . Errorf ( "Drive %s: %w" , export , err )
2018-02-15 17:45:57 -08:00
}
2019-12-23 16:31:03 -08:00
// Migrate successful v1 => v2, proceed to v2 => v3
2020-06-12 20:04:01 -07:00
version = formatErasureVersionV2
2018-02-15 17:45:57 -08:00
fallthrough
2020-06-12 20:04:01 -07:00
case formatErasureVersionV2 :
2022-01-24 11:28:45 -08:00
formatData , err = formatErasureMigrateV2ToV3 ( formatData , export , version )
if err != nil {
2022-08-04 16:10:08 -07:00
return nil , nil , fmt . Errorf ( "Drive %s: %w" , export , err )
2018-03-15 13:55:23 -07:00
}
2019-12-23 16:31:03 -08:00
// Migrate successful v2 => v3, v3 is latest
2020-05-18 09:59:45 -07:00
// version = formatXLVersionV3
2022-01-24 11:28:45 -08:00
return migrate ( formatPath , formatData )
2020-06-12 20:04:01 -07:00
case formatErasureVersionV3 :
2019-12-23 16:31:03 -08:00
// v3 is the latest version, return.
2022-01-24 11:28:45 -08:00
return formatData , formatFi , nil
2018-02-15 17:45:57 -08:00
}
2022-01-24 11:28:45 -08:00
return nil , nil , fmt . Errorf ( ` Disk %s: unknown format version %s ` , export , version )
2018-02-15 17:45:57 -08:00
}
// Migrates version V1 of format.json to version V2 of format.json,
// migration fails upon any error.
2022-01-24 11:28:45 -08:00
func formatErasureMigrateV1ToV2 ( data [ ] byte , version string ) ( [ ] byte , error ) {
2020-06-12 20:04:01 -07:00
if version != formatErasureVersionV1 {
2022-01-24 11:28:45 -08:00
return nil , fmt . Errorf ( ` format version expected %s, found %s ` , formatErasureVersionV1 , version )
2018-02-15 17:45:57 -08:00
}
2020-06-12 20:04:01 -07:00
formatV1 := & formatErasureV1 { }
2022-01-24 11:28:45 -08:00
if err := json . Unmarshal ( data , formatV1 ) ; err != nil {
return nil , err
2018-02-15 17:45:57 -08:00
}
2016-06-02 16:34:15 -07:00
2020-06-12 20:04:01 -07:00
formatV2 := & formatErasureV2 { }
2018-03-15 13:55:23 -07:00
formatV2 . Version = formatMetaVersionV1
2020-06-12 20:04:01 -07:00
formatV2 . Format = formatBackendErasure
formatV2 . Erasure . Version = formatErasureVersionV2
2021-01-16 12:08:02 -08:00
formatV2 . Erasure . DistributionAlgo = formatErasureVersionV2DistributionAlgoV1
2020-06-12 20:04:01 -07:00
formatV2 . Erasure . This = formatV1 . Erasure . Disk
formatV2 . Erasure . Sets = make ( [ ] [ ] string , 1 )
formatV2 . Erasure . Sets [ 0 ] = make ( [ ] string , len ( formatV1 . Erasure . JBOD ) )
copy ( formatV2 . Erasure . Sets [ 0 ] , formatV1 . Erasure . JBOD )
2018-02-15 17:45:57 -08:00
2022-01-24 11:28:45 -08:00
return json . Marshal ( formatV2 )
2018-02-15 17:45:57 -08:00
}
2017-08-03 17:07:02 +05:30
2018-03-15 13:55:23 -07:00
// Migrates V2 for format.json to V3 (Flat hierarchy for multipart)
2022-01-24 11:28:45 -08:00
func formatErasureMigrateV2ToV3 ( data [ ] byte , export , version string ) ( [ ] byte , error ) {
2020-06-12 20:04:01 -07:00
if version != formatErasureVersionV2 {
2022-01-24 11:28:45 -08:00
return nil , fmt . Errorf ( ` format version expected %s, found %s ` , formatErasureVersionV2 , version )
2018-03-15 13:55:23 -07:00
}
2019-12-23 16:31:03 -08:00
2020-06-12 20:04:01 -07:00
formatV2 := & formatErasureV2 { }
2022-01-24 11:28:45 -08:00
if err := json . Unmarshal ( data , formatV2 ) ; err != nil {
return nil , err
2018-03-15 13:55:23 -07:00
}
2018-08-05 21:15:28 -07:00
2022-01-24 11:28:45 -08:00
tmpOld := pathJoin ( export , minioMetaTmpDeletedBucket , mustGetUUID ( ) )
if err := renameAll ( pathJoin ( export , minioMetaMultipartBucket ) ,
tmpOld ) ; err != nil && err != errFileNotFound {
logger . LogIf ( GlobalContext , fmt . Errorf ( "unable to rename (%s -> %s) %w, drive may be faulty please investigate" ,
pathJoin ( export , minioMetaMultipartBucket ) ,
tmpOld ,
osErrToFileErr ( err ) ) )
2018-03-15 13:55:23 -07:00
}
// format-V2 struct is exactly same as format-V1 except that version is "3"
// which indicates the simplified multipart backend.
2020-06-12 20:04:01 -07:00
formatV3 := formatErasureV3 { }
2018-03-15 13:55:23 -07:00
formatV3 . Version = formatV2 . Version
formatV3 . Format = formatV2 . Format
2020-06-12 20:04:01 -07:00
formatV3 . Erasure = formatV2 . Erasure
formatV3 . Erasure . Version = formatErasureVersionV3
2018-03-15 13:55:23 -07:00
2022-01-24 11:28:45 -08:00
return json . Marshal ( formatV3 )
2018-03-15 13:55:23 -07:00
}
2018-02-15 17:45:57 -08:00
// countErrs - count a specific error.
func countErrs ( errs [ ] error , err error ) int {
2022-01-02 09:15:06 -08:00
i := 0
2018-02-15 17:45:57 -08:00
for _ , err1 := range errs {
2022-05-30 10:58:37 -07:00
if err1 == err || errors . Is ( err1 , err ) {
2018-02-15 17:45:57 -08:00
i ++
}
}
return i
2016-06-02 16:34:15 -07:00
}
2018-04-12 15:43:38 -07:00
// Check if unformatted disks are equal to write quorum.
func quorumUnformattedDisks ( errs [ ] error ) bool {
return countErrs ( errs , errUnformattedDisk ) >= ( len ( errs ) / 2 ) + 1
}
2020-06-12 20:04:01 -07:00
// loadFormatErasureAll - load all format config from all input disks in parallel.
func loadFormatErasureAll ( storageDisks [ ] StorageAPI , heal bool ) ( [ ] * formatErasureV3 , [ ] error ) {
2016-06-02 16:34:15 -07:00
// Initialize list of errors.
2019-10-14 09:44:51 -07:00
g := errgroup . WithNErrs ( len ( storageDisks ) )
2016-06-02 16:34:15 -07:00
// Initialize format configs.
2022-01-02 09:15:06 -08:00
formats := make ( [ ] * formatErasureV3 , len ( storageDisks ) )
2016-06-02 16:34:15 -07:00
2018-01-22 14:54:55 -08:00
// Load format from each disk in parallel
2019-10-14 09:44:51 -07:00
for index := range storageDisks {
index := index
g . Go ( func ( ) error {
if storageDisks [ index ] == nil {
return errDiskNotFound
}
2020-06-12 20:04:01 -07:00
format , err := loadFormatErasure ( storageDisks [ index ] )
2019-10-14 09:44:51 -07:00
if err != nil {
return err
2016-06-02 16:34:15 -07:00
}
2018-01-08 14:30:55 -08:00
formats [ index ] = format
2020-03-27 14:48:30 -07:00
if ! heal {
// If no healing required, make the disks valid and
// online.
2020-06-12 20:04:01 -07:00
storageDisks [ index ] . SetDiskID ( format . Erasure . This )
2020-03-27 14:48:30 -07:00
}
2019-10-14 09:44:51 -07:00
return nil
} , index )
2016-06-02 16:34:15 -07:00
}
2019-10-14 09:44:51 -07:00
// Return all formats and errors if any.
return formats , g . Wait ( )
2016-06-02 16:34:15 -07:00
}
2023-01-06 05:41:19 +01:00
func saveFormatErasure ( disk StorageAPI , format * formatErasureV3 , healID string ) error {
2020-08-18 14:37:26 -07:00
if disk == nil || format == nil {
2020-03-27 14:48:30 -07:00
return errDiskNotFound
}
2020-08-18 14:37:26 -07:00
diskID := format . Erasure . This
2020-06-12 20:04:01 -07:00
if err := makeFormatErasureMetaVolumes ( disk ) ; err != nil {
2020-03-27 14:48:30 -07:00
return err
}
2018-02-15 17:45:57 -08:00
// Marshal and write to disk.
formatBytes , err := json . Marshal ( format )
if err != nil {
return err
2017-08-03 17:07:02 +05:30
}
2019-12-23 16:31:03 -08:00
tmpFormat := mustGetUUID ( )
2019-09-24 18:47:26 -07:00
2018-02-15 17:45:57 -08:00
// Purge any existing temporary file, okay to ignore errors here.
2022-07-11 21:45:54 +05:30
defer disk . Delete ( context . TODO ( ) , minioMetaBucket , tmpFormat , DeleteOptions {
Recursive : false ,
Force : false ,
} )
2017-08-03 17:07:02 +05:30
2019-12-23 16:31:03 -08:00
// write to unique file.
2020-11-02 16:14:31 -08:00
if err = disk . WriteAll ( context . TODO ( ) , minioMetaBucket , tmpFormat , formatBytes ) ; err != nil {
2018-02-15 17:45:57 -08:00
return err
2017-08-03 17:07:02 +05:30
}
2019-09-24 18:47:26 -07:00
// Rename file `uuid.json` --> `format.json`.
2020-09-04 09:45:06 -07:00
if err = disk . RenameFile ( context . TODO ( ) , minioMetaBucket , tmpFormat , minioMetaBucket , formatConfigFile ) ; err != nil {
2020-03-27 14:48:30 -07:00
return err
}
disk . SetDiskID ( diskID )
2023-01-06 05:41:19 +01:00
if healID != "" {
2021-03-04 14:36:23 -08:00
ctx := context . Background ( )
2023-04-18 22:49:56 +01:00
ht := initHealingTracker ( disk , healID )
2021-03-04 14:36:23 -08:00
return ht . save ( ctx )
2020-09-28 19:39:32 -07:00
}
2020-03-27 14:48:30 -07:00
return nil
2016-05-20 02:22:22 -07:00
}
2020-06-12 20:04:01 -07:00
// loadFormatErasure - loads format.json from disk.
func loadFormatErasure ( disk StorageAPI ) ( format * formatErasureV3 , err error ) {
2020-09-04 09:45:06 -07:00
buf , err := disk . ReadAll ( context . TODO ( ) , minioMetaBucket , formatConfigFile )
2016-06-25 14:51:06 -07:00
if err != nil {
2016-05-20 02:22:22 -07:00
// 'file not found' and 'volume not found' as
// same. 'volume not found' usually means its a fresh disk.
if err == errFileNotFound || err == errVolumeNotFound {
return nil , errUnformattedDisk
}
2016-05-07 00:59:43 -07:00
return nil , err
2016-05-05 00:48:20 +05:30
}
2016-06-24 02:06:23 -07:00
// Try to decode format json into formatConfigV1 struct.
2020-06-12 20:04:01 -07:00
format = & formatErasureV3 { }
2016-06-25 14:51:06 -07:00
if err = json . Unmarshal ( buf , format ) ; err != nil {
2016-05-07 00:59:43 -07:00
return nil , err
2016-05-05 00:48:20 +05:30
}
2016-06-24 02:06:23 -07:00
// Success.
2016-05-20 02:22:22 -07:00
return format , nil
}
2020-06-12 20:04:01 -07:00
// Valid formatErasure basic versions.
2021-01-29 11:40:55 -08:00
func checkFormatErasureValue ( formatErasure * formatErasureV3 , disk StorageAPI ) error {
2018-02-15 17:45:57 -08:00
// Validate format version and format type.
2020-06-12 20:04:01 -07:00
if formatErasure . Version != formatMetaVersionV1 {
2021-01-29 11:40:55 -08:00
return fmt . Errorf ( "Unsupported version of backend format [%s] found on %s" , formatErasure . Version , disk )
2016-07-26 03:18:47 -07:00
}
2022-05-30 10:58:37 -07:00
if formatErasure . Format != formatBackendErasure && formatErasure . Format != formatBackendErasureSingle {
2021-01-29 11:40:55 -08:00
return fmt . Errorf ( "Unsupported backend format [%s] found on %s" , formatErasure . Format , disk )
2017-08-03 17:07:02 +05:30
}
2020-06-12 20:04:01 -07:00
if formatErasure . Erasure . Version != formatErasureVersionV3 {
2021-01-29 11:40:55 -08:00
return fmt . Errorf ( "Unsupported Erasure backend format found [%s] on %s" , formatErasure . Erasure . Version , disk )
2018-01-22 14:54:55 -08:00
}
return nil
2017-08-03 17:07:02 +05:30
}
2018-02-15 17:45:57 -08:00
// Check all format values.
2021-01-29 11:40:55 -08:00
func checkFormatErasureValues ( formats [ ] * formatErasureV3 , disks [ ] StorageAPI , setDriveCount int ) error {
2020-06-12 20:04:01 -07:00
for i , formatErasure := range formats {
if formatErasure == nil {
2018-02-15 17:45:57 -08:00
continue
}
2021-01-29 11:40:55 -08:00
if err := checkFormatErasureValue ( formatErasure , disks [ i ] ) ; err != nil {
2018-02-15 17:45:57 -08:00
return err
}
2020-06-12 20:04:01 -07:00
if len ( formats ) != len ( formatErasure . Erasure . Sets ) * len ( formatErasure . Erasure . Sets [ 0 ] ) {
2022-08-04 16:10:08 -07:00
return fmt . Errorf ( "%s drive is already being used in another erasure deployment. (Number of drives specified: %d but the number of drives found in the %s drive's format.json: %d)" ,
2021-01-29 11:40:55 -08:00
disks [ i ] , len ( formats ) , humanize . Ordinal ( i + 1 ) , len ( formatErasure . Erasure . Sets ) * len ( formatErasure . Erasure . Sets [ 0 ] ) )
2018-02-15 17:45:57 -08:00
}
2021-01-16 12:08:02 -08:00
// Only if custom erasure drive count is set, verify if the
// set_drive_count was manually set - we need to honor what is
// present on the drives.
2020-08-26 19:29:35 -07:00
if globalCustomErasureDriveCount && len ( formatErasure . Erasure . Sets [ 0 ] ) != setDriveCount {
2022-08-04 16:10:08 -07:00
return fmt . Errorf ( "%s drive is already formatted with %d drives per erasure set. This cannot be changed to %d, please revert your MINIO_ERASURE_SET_DRIVE_COUNT setting" , disks [ i ] , len ( formatErasure . Erasure . Sets [ 0 ] ) , setDriveCount )
2020-03-08 13:30:25 -07:00
}
2017-08-03 17:07:02 +05:30
}
2018-02-15 17:45:57 -08:00
return nil
}
2017-08-03 17:07:02 +05:30
2020-06-12 20:04:01 -07:00
// Get Deployment ID for the Erasure sets from format.json.
2018-07-18 20:17:35 -07:00
// This need not be in quorum. Even if one of the format.json
// file has this value, we assume it is valid.
// If more than one format.json's have different id, it is considered a corrupt
// backend format.
2020-06-12 20:04:01 -07:00
func formatErasureGetDeploymentID ( refFormat * formatErasureV3 , formats [ ] * formatErasureV3 ) ( string , error ) {
2018-07-18 20:17:35 -07:00
var deploymentID string
for _ , format := range formats {
if format == nil || format . ID == "" {
continue
}
2020-06-12 20:04:01 -07:00
if reflect . DeepEqual ( format . Erasure . Sets , refFormat . Erasure . Sets ) {
2018-07-18 20:17:35 -07:00
// Found an ID in one of the format.json file
// Set deploymentID for the first time.
if deploymentID == "" {
deploymentID = format . ID
} else if deploymentID != format . ID {
// DeploymentID found earlier doesn't match with the
// current format.json's ID.
2020-08-03 18:17:48 -07:00
return "" , fmt . Errorf ( "Deployment IDs do not match expected %s, got %s: %w" ,
deploymentID , format . ID , errCorruptedFormat )
2018-07-18 20:17:35 -07:00
}
}
}
return deploymentID , nil
}
2020-06-12 20:04:01 -07:00
// formatErasureFixDeploymentID - Add deployment id if it is not present.
func formatErasureFixDeploymentID ( endpoints Endpoints , storageDisks [ ] StorageAPI , refFormat * formatErasureV3 ) ( err error ) {
2018-07-18 20:17:35 -07:00
// Attempt to load all `format.json` from all disks.
2020-08-03 18:17:48 -07:00
formats , _ := loadFormatErasureAll ( storageDisks , false )
2018-07-18 20:17:35 -07:00
for index := range formats {
2020-06-12 20:04:01 -07:00
// If the Erasure sets do not match, set those formats to nil,
2018-07-18 20:17:35 -07:00
// We do not have to update the ID on those format.json file.
2020-06-12 20:04:01 -07:00
if formats [ index ] != nil && ! reflect . DeepEqual ( formats [ index ] . Erasure . Sets , refFormat . Erasure . Sets ) {
2018-07-18 20:17:35 -07:00
formats [ index ] = nil
}
}
2020-08-03 18:17:48 -07:00
2020-06-12 20:04:01 -07:00
refFormat . ID , err = formatErasureGetDeploymentID ( refFormat , formats )
2018-07-18 20:17:35 -07:00
if err != nil {
return err
}
// If ID is set, then some other node got the lock
// before this node could and generated an ID
// for the deployment. No need to generate one.
if refFormat . ID != "" {
return nil
}
// ID is generated for the first time,
// We set the ID in all the formats and update.
refFormat . ID = mustGetUUID ( )
for _ , format := range formats {
if format != nil {
format . ID = refFormat . ID
}
}
// Deployment ID needs to be set on all the disks.
// Save `format.json` across all disks.
2020-06-12 20:04:01 -07:00
return saveFormatErasureAll ( GlobalContext , storageDisks , formats )
2018-07-18 20:17:35 -07:00
}
// Update only the valid local disks which have not been updated before.
2020-06-12 20:04:01 -07:00
func formatErasureFixLocalDeploymentID ( endpoints Endpoints , storageDisks [ ] StorageAPI , refFormat * formatErasureV3 ) error {
2018-07-18 20:17:35 -07:00
// If this server was down when the deploymentID was updated
// then we make sure that we update the local disks with the deploymentID.
2020-03-27 14:48:30 -07:00
// Initialize errs to collect errors inside go-routine.
g := errgroup . WithNErrs ( len ( storageDisks ) )
for index := range storageDisks {
index := index
g . Go ( func ( ) error {
if endpoints [ index ] . IsLocal && storageDisks [ index ] != nil && storageDisks [ index ] . IsOnline ( ) {
2020-06-12 20:04:01 -07:00
format , err := loadFormatErasure ( storageDisks [ index ] )
2020-03-27 14:48:30 -07:00
if err != nil {
// Disk can be offline etc.
// ignore the errors seen here.
return nil
}
if format . ID != "" {
return nil
}
2020-06-12 20:04:01 -07:00
if ! reflect . DeepEqual ( format . Erasure . Sets , refFormat . Erasure . Sets ) {
2020-03-27 14:48:30 -07:00
return nil
}
format . ID = refFormat . ID
2020-09-28 19:39:32 -07:00
// Heal the drive if we fixed its deployment ID.
2023-01-06 05:41:19 +01:00
if err := saveFormatErasure ( storageDisks [ index ] , format , mustGetUUID ( ) ) ; err != nil {
2020-04-09 09:30:02 -07:00
logger . LogIf ( GlobalContext , err )
2020-03-27 14:48:30 -07:00
return fmt . Errorf ( "Unable to save format.json, %w" , err )
}
2018-07-18 20:17:35 -07:00
}
2020-03-27 14:48:30 -07:00
return nil
} , index )
}
for _ , err := range g . Wait ( ) {
if err != nil {
return err
2018-07-18 20:17:35 -07:00
}
}
return nil
}
2020-06-12 20:04:01 -07:00
// Get backend Erasure format in quorum `format.json`.
func getFormatErasureInQuorum ( formats [ ] * formatErasureV3 ) ( * formatErasureV3 , error ) {
2022-02-04 12:21:21 -08:00
formatCountMap := make ( map [ int ] int , len ( formats ) )
for _ , format := range formats {
2018-02-15 17:45:57 -08:00
if format == nil {
2018-01-22 14:54:55 -08:00
continue
}
2022-02-04 12:21:21 -08:00
formatCountMap [ format . Drives ( ) ] ++
2016-06-02 04:45:56 +05:30
}
2016-07-26 03:18:47 -07:00
2022-02-04 12:21:21 -08:00
maxDrives := 0
2018-02-15 17:45:57 -08:00
maxCount := 0
2022-02-04 12:21:21 -08:00
for drives , count := range formatCountMap {
2018-02-15 17:45:57 -08:00
if count > maxCount {
maxCount = count
2022-02-04 12:21:21 -08:00
maxDrives = drives
2016-07-29 01:49:59 +02:00
}
2018-02-15 17:45:57 -08:00
}
2022-02-04 12:21:21 -08:00
if maxDrives == 0 {
return nil , errErasureReadQuorum
}
2018-02-15 17:45:57 -08:00
if maxCount < len ( formats ) / 2 {
2020-06-12 20:04:01 -07:00
return nil , errErasureReadQuorum
2018-02-15 17:45:57 -08:00
}
2022-02-04 12:21:21 -08:00
for i , format := range formats {
if format == nil {
continue
}
if format . Drives ( ) == maxDrives {
2019-11-21 04:24:51 -08:00
format := formats [ i ] . Clone ( )
2020-06-12 20:04:01 -07:00
format . Erasure . This = ""
2019-11-21 04:24:51 -08:00
return format , nil
2016-07-29 01:49:59 +02:00
}
}
2018-02-15 17:45:57 -08:00
2020-06-12 20:04:01 -07:00
return nil , errErasureReadQuorum
2016-07-29 01:49:59 +02:00
}
2020-06-12 20:04:01 -07:00
func formatErasureV3Check ( reference * formatErasureV3 , format * formatErasureV3 ) error {
2019-11-21 04:24:51 -08:00
tmpFormat := format . Clone ( )
2020-06-12 20:04:01 -07:00
this := tmpFormat . Erasure . This
tmpFormat . Erasure . This = ""
if len ( reference . Erasure . Sets ) != len ( format . Erasure . Sets ) {
return fmt . Errorf ( "Expected number of sets %d, got %d" , len ( reference . Erasure . Sets ) , len ( format . Erasure . Sets ) )
2018-02-15 17:45:57 -08:00
}
2017-08-03 17:07:02 +05:30
2018-02-15 17:45:57 -08:00
// Make sure that the sets match.
2020-06-12 20:04:01 -07:00
for i := range reference . Erasure . Sets {
if len ( reference . Erasure . Sets [ i ] ) != len ( format . Erasure . Sets [ i ] ) {
2018-02-15 17:45:57 -08:00
return fmt . Errorf ( "Each set should be of same size, expected %d got %d" ,
2020-06-12 20:04:01 -07:00
len ( reference . Erasure . Sets [ i ] ) , len ( format . Erasure . Sets [ i ] ) )
2016-07-29 01:49:59 +02:00
}
2020-06-12 20:04:01 -07:00
for j := range reference . Erasure . Sets [ i ] {
if reference . Erasure . Sets [ i ] [ j ] != format . Erasure . Sets [ i ] [ j ] {
2020-10-24 13:23:08 -07:00
return fmt . Errorf ( "UUID on positions %d:%d do not match with, expected %s got %s: (%w)" ,
i , j , reference . Erasure . Sets [ i ] [ j ] , format . Erasure . Sets [ i ] [ j ] , errInconsistentDisk )
2016-07-29 01:49:59 +02:00
}
}
2018-02-15 17:45:57 -08:00
}
// Make sure that the diskID is found in the set.
2020-06-12 20:04:01 -07:00
for i := 0 ; i < len ( tmpFormat . Erasure . Sets ) ; i ++ {
for j := 0 ; j < len ( tmpFormat . Erasure . Sets [ i ] ) ; j ++ {
if this == tmpFormat . Erasure . Sets [ i ] [ j ] {
2018-02-15 17:45:57 -08:00
return nil
2016-07-29 01:49:59 +02:00
}
}
}
2022-08-04 16:10:08 -07:00
return fmt . Errorf ( "DriveID %s not found in any drive sets %s" , this , format . Erasure . Sets )
2016-07-29 01:49:59 +02:00
}
2020-06-12 20:04:01 -07:00
// saveFormatErasureAll - populates `format.json` on disks in its order.
func saveFormatErasureAll ( ctx context . Context , storageDisks [ ] StorageAPI , formats [ ] * formatErasureV3 ) error {
2019-10-14 09:44:51 -07:00
g := errgroup . WithNErrs ( len ( storageDisks ) )
2016-07-29 01:49:59 +02:00
2018-02-15 17:45:57 -08:00
// Write `format.json` to all disks.
2019-10-14 09:44:51 -07:00
for index := range storageDisks {
index := index
g . Go ( func ( ) error {
2020-07-17 10:08:04 -07:00
if formats [ index ] == nil {
return errDiskNotFound
}
2023-01-06 05:41:19 +01:00
return saveFormatErasure ( storageDisks [ index ] , formats [ index ] , "" )
2019-10-14 09:44:51 -07:00
} , index )
2016-07-29 01:49:59 +02:00
}
2019-10-14 09:44:51 -07:00
// Wait for the routines to finish.
2022-07-22 09:04:17 +08:00
return reduceWriteQuorumErrs ( ctx , g . Wait ( ) , nil , len ( storageDisks ) )
2018-02-15 17:45:57 -08:00
}
2018-04-03 23:58:48 -05:00
// relinquishes the underlying connection for all storage disks.
2022-05-30 10:58:37 -07:00
func closeStorageDisks ( storageDisks ... StorageAPI ) {
2022-01-24 11:28:45 -08:00
var wg sync . WaitGroup
2018-04-03 23:58:48 -05:00
for _ , disk := range storageDisks {
if disk == nil {
continue
}
2022-01-24 11:28:45 -08:00
wg . Add ( 1 )
go func ( disk StorageAPI ) {
defer wg . Done ( )
disk . Close ( )
} ( disk )
2018-04-03 23:58:48 -05:00
}
2022-01-24 11:28:45 -08:00
wg . Wait ( )
2018-04-03 23:58:48 -05:00
}
2019-09-27 16:47:12 -07:00
// Initialize storage disks for each endpoint.
// Errors are returned for each endpoint with matching index.
2022-10-31 15:27:50 +01:00
func initStorageDisksWithErrors ( endpoints Endpoints , healthCheck bool ) ( [ ] StorageAPI , [ ] error ) {
2018-02-15 17:45:57 -08:00
// Bootstrap disks.
storageDisks := make ( [ ] StorageAPI , len ( endpoints ) )
2019-10-14 09:44:51 -07:00
g := errgroup . WithNErrs ( len ( endpoints ) )
for index := range endpoints {
index := index
2020-08-18 14:37:26 -07:00
g . Go ( func ( ) ( err error ) {
2022-10-31 15:27:50 +01:00
storageDisks [ index ] , err = newStorageAPI ( endpoints [ index ] , healthCheck )
2020-08-18 14:37:26 -07:00
return err
2019-10-14 09:44:51 -07:00
} , index )
2019-08-03 00:47:26 +05:30
}
2019-10-14 09:44:51 -07:00
return storageDisks , g . Wait ( )
2019-08-03 00:47:26 +05:30
}
2020-06-12 20:04:01 -07:00
// formatErasureV3ThisEmpty - find out if '.This' field is empty
2018-03-19 09:13:00 -07:00
// in any of the input `formats`, if yes return true.
2020-06-12 20:04:01 -07:00
func formatErasureV3ThisEmpty ( formats [ ] * formatErasureV3 ) bool {
2018-03-19 09:13:00 -07:00
for _ , format := range formats {
if format == nil {
continue
}
// NOTE: This code is specifically needed when migrating version
// V1 to V2 to V3, in a scenario such as this we only need to handle
// single sets since we never used to support multiple sets in releases
// with V1 format version.
2020-06-12 20:04:01 -07:00
if len ( format . Erasure . Sets ) > 1 {
2018-03-19 09:13:00 -07:00
continue
}
2020-06-12 20:04:01 -07:00
if format . Erasure . This == "" {
2018-03-19 09:13:00 -07:00
return true
}
}
return false
}
2020-06-12 20:04:01 -07:00
// fixFormatErasureV3 - fix format Erasure configuration on all disks.
func fixFormatErasureV3 ( storageDisks [ ] StorageAPI , endpoints Endpoints , formats [ ] * formatErasureV3 ) error {
2020-03-27 14:48:30 -07:00
g := errgroup . WithNErrs ( len ( formats ) )
for i := range formats {
i := i
g . Go ( func ( ) error {
if formats [ i ] == nil || ! endpoints [ i ] . IsLocal {
return nil
}
// NOTE: This code is specifically needed when migrating version
// V1 to V2 to V3, in a scenario such as this we only need to handle
// single sets since we never used to support multiple sets in releases
// with V1 format version.
2020-06-12 20:04:01 -07:00
if len ( formats [ i ] . Erasure . Sets ) > 1 {
2020-03-27 14:48:30 -07:00
return nil
}
2020-06-12 20:04:01 -07:00
if formats [ i ] . Erasure . This == "" {
formats [ i ] . Erasure . This = formats [ i ] . Erasure . Sets [ 0 ] [ i ]
2020-09-28 19:39:32 -07:00
// Heal the drive if drive has .This empty.
2023-01-06 05:41:19 +01:00
if err := saveFormatErasure ( storageDisks [ i ] , formats [ i ] , mustGetUUID ( ) ) ; err != nil {
2020-03-27 14:48:30 -07:00
return err
}
2018-03-19 09:13:00 -07:00
}
2020-03-27 14:48:30 -07:00
return nil
} , i )
}
for _ , err := range g . Wait ( ) {
if err != nil {
return err
2018-03-19 09:13:00 -07:00
}
}
return nil
}
2020-06-12 20:04:01 -07:00
// initFormatErasure - save Erasure format configuration on all disks.
2021-01-19 10:01:31 -08:00
func initFormatErasure ( ctx context . Context , storageDisks [ ] StorageAPI , setCount , setDriveCount int , deploymentID , distributionAlgo string , sErrs [ ] error ) ( * formatErasureV3 , error ) {
2021-01-16 12:08:02 -08:00
format := newFormatErasureV3 ( setCount , setDriveCount )
2020-06-12 20:04:01 -07:00
formats := make ( [ ] * formatErasureV3 , len ( storageDisks ) )
2023-01-10 08:07:45 +01:00
wantAtMost , err := ecDrivesNoConfig ( setDriveCount )
if err != nil {
return nil , err
}
2016-05-05 00:48:20 +05:30
2018-02-15 17:45:57 -08:00
for i := 0 ; i < setCount ; i ++ {
2020-08-26 19:29:35 -07:00
hostCount := make ( map [ string ] int , setDriveCount )
for j := 0 ; j < setDriveCount ; j ++ {
disk := storageDisks [ i * setDriveCount + j ]
2023-02-06 09:26:09 -08:00
if disk == nil {
continue
}
2019-11-21 04:24:51 -08:00
newFormat := format . Clone ( )
2020-06-12 20:04:01 -07:00
newFormat . Erasure . This = format . Erasure . Sets [ i ] [ j ]
2021-01-19 10:01:31 -08:00
if distributionAlgo != "" {
newFormat . Erasure . DistributionAlgo = distributionAlgo
}
2019-11-20 02:09:30 -08:00
if deploymentID != "" {
newFormat . ID = deploymentID
}
2020-01-13 22:09:10 +01:00
hostCount [ disk . Hostname ( ) ] ++
2020-08-26 19:29:35 -07:00
formats [ i * setDriveCount + j ] = newFormat
2018-02-15 17:45:57 -08:00
}
2023-02-06 09:26:09 -08:00
var once sync . Once
for host , count := range hostCount {
if count > wantAtMost {
if host == "" {
host = "local"
2020-01-13 22:09:10 +01:00
}
2023-02-06 09:26:09 -08:00
once . Do ( func ( ) {
if len ( hostCount ) == 1 {
return
}
logger . Info ( " * Set %v:" , i + 1 )
for j := 0 ; j < setDriveCount ; j ++ {
disk := storageDisks [ i * setDriveCount + j ]
logger . Info ( " - Drive: %s" , disk . String ( ) )
}
} )
logger . Info ( color . Yellow ( "WARNING:" ) + " Host %v has more than %v drives of set. " +
"A host failure will result in data becoming unavailable." , host , wantAtMost )
2020-01-13 22:09:10 +01:00
}
}
2016-05-05 00:48:20 +05:30
}
2020-08-18 14:37:26 -07:00
// Mark all root disks down
markRootDisksAsDown ( storageDisks , sErrs )
2018-02-15 17:45:57 -08:00
// Save formats `format.json` across all disks.
2020-06-12 20:04:01 -07:00
if err := saveFormatErasureAll ( ctx , storageDisks , formats ) ; err != nil {
2018-02-15 17:45:57 -08:00
return nil , err
2017-04-18 10:35:17 -07:00
}
2018-02-15 17:45:57 -08:00
2020-06-12 20:04:01 -07:00
return getFormatErasureInQuorum ( formats )
2017-04-18 10:35:17 -07:00
}
2020-01-13 22:09:10 +01:00
// ecDrivesNoConfig returns the erasure coded drives in a set if no config has been set.
// It will attempt to read it from env variable and fall back to drives/2.
2023-01-10 08:07:45 +01:00
func ecDrivesNoConfig ( setDriveCount int ) ( int , error ) {
sc , err := storageclass . LookupConfig ( config . KVS { } , setDriveCount )
if err != nil {
return 0 , err
2020-01-13 22:09:10 +01:00
}
2023-01-10 08:07:45 +01:00
return sc . GetParityForSC ( storageclass . STANDARD ) , nil
2020-01-13 22:09:10 +01:00
}
2020-06-12 20:04:01 -07:00
// Make Erasure backend meta volumes.
func makeFormatErasureMetaVolumes ( disk StorageAPI ) error {
2020-03-27 14:48:30 -07:00
if disk == nil {
return errDiskNotFound
}
2022-01-10 17:26:00 -08:00
volumes := [ ] string {
2022-01-24 11:28:45 -08:00
minioMetaTmpDeletedBucket , // creates .minio.sys/tmp as well as .minio.sys/tmp/.trash
minioMetaMultipartBucket , // creates .minio.sys/multipart
dataUsageBucket , // creates .minio.sys/buckets
minioConfigBucket , // creates .minio.sys/config
2022-01-10 17:26:00 -08:00
}
2019-12-23 16:31:03 -08:00
// Attempt to create MinIO internal buckets.
2022-01-10 17:26:00 -08:00
return disk . MakeVolBulk ( context . TODO ( ) , volumes ... )
2016-12-11 15:18:55 -08:00
}
2018-02-15 17:45:57 -08:00
// Initialize a new set of set formats which will be written to all disks.
2020-08-26 19:29:35 -07:00
func newHealFormatSets ( refFormat * formatErasureV3 , setCount , setDriveCount int , formats [ ] * formatErasureV3 , errs [ ] error ) [ ] [ ] * formatErasureV3 {
2020-06-12 20:04:01 -07:00
newFormats := make ( [ ] [ ] * formatErasureV3 , setCount )
for i := range refFormat . Erasure . Sets {
2020-08-26 19:29:35 -07:00
newFormats [ i ] = make ( [ ] * formatErasureV3 , setDriveCount )
2018-02-15 17:45:57 -08:00
}
2020-06-12 20:04:01 -07:00
for i := range refFormat . Erasure . Sets {
for j := range refFormat . Erasure . Sets [ i ] {
2020-10-26 10:29:29 -07:00
if errors . Is ( errs [ i * setDriveCount + j ] , errUnformattedDisk ) {
2020-06-12 20:04:01 -07:00
newFormats [ i ] [ j ] = & formatErasureV3 { }
2019-01-23 03:32:06 +01:00
newFormats [ i ] [ j ] . ID = refFormat . ID
2018-02-15 17:45:57 -08:00
newFormats [ i ] [ j ] . Format = refFormat . Format
2020-10-26 10:29:29 -07:00
newFormats [ i ] [ j ] . Version = refFormat . Version
newFormats [ i ] [ j ] . Erasure . This = refFormat . Erasure . Sets [ i ] [ j ]
newFormats [ i ] [ j ] . Erasure . Sets = refFormat . Erasure . Sets
2020-06-12 20:04:01 -07:00
newFormats [ i ] [ j ] . Erasure . Version = refFormat . Erasure . Version
newFormats [ i ] [ j ] . Erasure . DistributionAlgo = refFormat . Erasure . DistributionAlgo
2018-02-15 17:45:57 -08:00
}
}
}
return newFormats
2016-05-05 00:48:20 +05:30
}