| 
									
										
										
										
											2016-09-10 03:53:09 +08:00
										 |  |  | /* | 
					
						
							|  |  |  |  * Minio Cloud Storage, (C) 2016 Minio, Inc. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * 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 cmd | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-17 04:06:49 +08:00
										 |  |  | import ( | 
					
						
							|  |  |  | 	"sync" | 
					
						
							| 
									
										
										
										
											2016-10-06 03:48:07 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	"github.com/minio/minio/pkg/disk" | 
					
						
							| 
									
										
										
										
											2016-09-17 04:06:49 +08:00
										 |  |  | ) | 
					
						
							| 
									
										
										
										
											2016-09-10 06:32:08 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-10 03:53:09 +08:00
										 |  |  | // naughtyDisk wraps a POSIX disk and returns programmed errors
 | 
					
						
							|  |  |  | // specified by the developer. The purpose is to simulate errors
 | 
					
						
							| 
									
										
										
										
											2016-10-06 03:48:07 +08:00
										 |  |  | // that are hard to simulate in practice like DiskNotFound.
 | 
					
						
							| 
									
										
										
										
											2016-09-10 03:53:09 +08:00
										 |  |  | // Programmed errors are stored in errors field.
 | 
					
						
							|  |  |  | type naughtyDisk struct { | 
					
						
							|  |  |  | 	// The real disk
 | 
					
						
							| 
									
										
										
										
											2018-02-16 09:45:57 +08:00
										 |  |  | 	disk StorageAPI | 
					
						
							| 
									
										
										
										
											2016-09-10 03:53:09 +08:00
										 |  |  | 	// Programmed errors: API call number => error to return
 | 
					
						
							|  |  |  | 	errors map[int]error | 
					
						
							|  |  |  | 	// The error to return when no error value is programmed
 | 
					
						
							|  |  |  | 	defaultErr error | 
					
						
							|  |  |  | 	// The current API call number
 | 
					
						
							|  |  |  | 	callNR int | 
					
						
							| 
									
										
										
										
											2016-09-17 04:06:49 +08:00
										 |  |  | 	// Data protection
 | 
					
						
							|  |  |  | 	mu sync.Mutex | 
					
						
							| 
									
										
										
										
											2016-09-10 03:53:09 +08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-16 09:45:57 +08:00
										 |  |  | func newNaughtyDisk(d StorageAPI, errs map[int]error, defaultErr error) *naughtyDisk { | 
					
						
							| 
									
										
										
										
											2016-09-10 03:53:09 +08:00
										 |  |  | 	return &naughtyDisk{disk: d, errors: errs, defaultErr: defaultErr} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-06 03:48:07 +08:00
										 |  |  | func (d *naughtyDisk) String() string { | 
					
						
							| 
									
										
										
										
											2016-10-20 10:59:48 +08:00
										 |  |  | 	return d.disk.String() | 
					
						
							| 
									
										
										
										
											2016-10-06 03:48:07 +08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-02-16 09:45:57 +08:00
										 |  |  | func (d *naughtyDisk) IsOnline() bool { | 
					
						
							|  |  |  | 	if err := d.calcError(); err != nil { | 
					
						
							|  |  |  | 		return err == errDiskNotFound | 
					
						
							| 
									
										
										
										
											2016-11-24 07:48:10 +08:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2018-02-16 09:45:57 +08:00
										 |  |  | 	return d.disk.IsOnline() | 
					
						
							| 
									
										
										
										
											2016-11-24 07:48:10 +08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func (d *naughtyDisk) Close() (err error) { | 
					
						
							|  |  |  | 	if err = d.calcError(); err != nil { | 
					
						
							|  |  |  | 		return err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return d.disk.Close() | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-10 03:53:09 +08:00
										 |  |  | func (d *naughtyDisk) calcError() (err error) { | 
					
						
							| 
									
										
										
										
											2016-09-17 04:06:49 +08:00
										 |  |  | 	d.mu.Lock() | 
					
						
							|  |  |  | 	defer d.mu.Unlock() | 
					
						
							| 
									
										
										
										
											2016-09-10 03:53:09 +08:00
										 |  |  | 	d.callNR++ | 
					
						
							|  |  |  | 	if err, ok := d.errors[d.callNR]; ok { | 
					
						
							|  |  |  | 		return err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if d.defaultErr != nil { | 
					
						
							|  |  |  | 		return d.defaultErr | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return nil | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-10 06:32:08 +08:00
										 |  |  | func (d *naughtyDisk) DiskInfo() (info disk.Info, err error) { | 
					
						
							|  |  |  | 	if err := d.calcError(); err != nil { | 
					
						
							|  |  |  | 		return info, err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return d.disk.DiskInfo() | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-10 03:53:09 +08:00
										 |  |  | func (d *naughtyDisk) MakeVol(volume string) (err error) { | 
					
						
							|  |  |  | 	if err := d.calcError(); err != nil { | 
					
						
							|  |  |  | 		return err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return d.disk.MakeVol(volume) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func (d *naughtyDisk) ListVols() (vols []VolInfo, err error) { | 
					
						
							|  |  |  | 	if err := d.calcError(); err != nil { | 
					
						
							|  |  |  | 		return nil, err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return d.disk.ListVols() | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func (d *naughtyDisk) StatVol(volume string) (volInfo VolInfo, err error) { | 
					
						
							|  |  |  | 	if err := d.calcError(); err != nil { | 
					
						
							|  |  |  | 		return VolInfo{}, err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return d.disk.StatVol(volume) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | func (d *naughtyDisk) DeleteVol(volume string) (err error) { | 
					
						
							|  |  |  | 	if err := d.calcError(); err != nil { | 
					
						
							|  |  |  | 		return err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return d.disk.DeleteVol(volume) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func (d *naughtyDisk) ListDir(volume, path string) (entries []string, err error) { | 
					
						
							|  |  |  | 	if err := d.calcError(); err != nil { | 
					
						
							|  |  |  | 		return []string{}, err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return d.disk.ListDir(volume, path) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-09-26 02:32:56 +08:00
										 |  |  | func (d *naughtyDisk) ReadFile(volume string, path string, offset int64, buf []byte, verifier *BitrotVerifier) (n int64, err error) { | 
					
						
							| 
									
										
										
										
											2016-09-10 03:53:09 +08:00
										 |  |  | 	if err := d.calcError(); err != nil { | 
					
						
							|  |  |  | 		return 0, err | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2017-09-26 02:32:56 +08:00
										 |  |  | 	return d.disk.ReadFile(volume, path, offset, buf, verifier) | 
					
						
							| 
									
										
										
										
											2017-05-17 05:21:52 +08:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-30 03:44:44 +08:00
										 |  |  | func (d *naughtyDisk) PrepareFile(volume, path string, length int64) error { | 
					
						
							|  |  |  | 	if err := d.calcError(); err != nil { | 
					
						
							|  |  |  | 		return err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return d.disk.PrepareFile(volume, path, length) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-10 03:53:09 +08:00
										 |  |  | func (d *naughtyDisk) AppendFile(volume, path string, buf []byte) error { | 
					
						
							|  |  |  | 	if err := d.calcError(); err != nil { | 
					
						
							|  |  |  | 		return err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return d.disk.AppendFile(volume, path, buf) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func (d *naughtyDisk) RenameFile(srcVolume, srcPath, dstVolume, dstPath string) error { | 
					
						
							|  |  |  | 	if err := d.calcError(); err != nil { | 
					
						
							|  |  |  | 		return err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return d.disk.RenameFile(srcVolume, srcPath, dstVolume, dstPath) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func (d *naughtyDisk) StatFile(volume string, path string) (file FileInfo, err error) { | 
					
						
							|  |  |  | 	if err := d.calcError(); err != nil { | 
					
						
							|  |  |  | 		return FileInfo{}, err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return d.disk.StatFile(volume, path) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func (d *naughtyDisk) DeleteFile(volume string, path string) (err error) { | 
					
						
							|  |  |  | 	if err := d.calcError(); err != nil { | 
					
						
							|  |  |  | 		return err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return d.disk.DeleteFile(volume, path) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func (d *naughtyDisk) ReadAll(volume string, path string) (buf []byte, err error) { | 
					
						
							|  |  |  | 	if err := d.calcError(); err != nil { | 
					
						
							|  |  |  | 		return nil, err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return d.disk.ReadAll(volume, path) | 
					
						
							|  |  |  | } |