mirror of https://github.com/minio/minio.git
				
				
				
			
		
			
				
	
	
		
			265 lines
		
	
	
		
			7.1 KiB
		
	
	
	
		
			Go
		
	
	
	
			
		
		
	
	
			265 lines
		
	
	
		
			7.1 KiB
		
	
	
	
		
			Go
		
	
	
	
| /*
 | |
|  * 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
 | |
| 
 | |
| import (
 | |
| 	"io"
 | |
| 	"net"
 | |
| 	"net/rpc"
 | |
| 	"path"
 | |
| 	"strconv"
 | |
| 	"strings"
 | |
| 
 | |
| 	"github.com/minio/minio/pkg/disk"
 | |
| )
 | |
| 
 | |
| type networkStorage struct {
 | |
| 	netAddr   string
 | |
| 	netPath   string
 | |
| 	rpcClient *AuthRPCClient
 | |
| }
 | |
| 
 | |
| const (
 | |
| 	storageRPCPath = reservedBucket + "/storage"
 | |
| )
 | |
| 
 | |
| // Converts rpc.ServerError to underlying error. This function is
 | |
| // written so that the storageAPI errors are consistent across network
 | |
| // disks as well.
 | |
| func toStorageErr(err error) error {
 | |
| 	if err == nil {
 | |
| 		return nil
 | |
| 	}
 | |
| 
 | |
| 	switch err.(type) {
 | |
| 	case *net.OpError:
 | |
| 		return errDiskNotFound
 | |
| 	}
 | |
| 
 | |
| 	switch err.Error() {
 | |
| 	case io.EOF.Error():
 | |
| 		return io.EOF
 | |
| 	case io.ErrUnexpectedEOF.Error():
 | |
| 		return io.ErrUnexpectedEOF
 | |
| 	case rpc.ErrShutdown.Error():
 | |
| 		return errDiskNotFound
 | |
| 	case errUnexpected.Error():
 | |
| 		return errUnexpected
 | |
| 	case errDiskFull.Error():
 | |
| 		return errDiskFull
 | |
| 	case errVolumeNotFound.Error():
 | |
| 		return errVolumeNotFound
 | |
| 	case errVolumeExists.Error():
 | |
| 		return errVolumeExists
 | |
| 	case errFileNotFound.Error():
 | |
| 		return errFileNotFound
 | |
| 	case errFileNameTooLong.Error():
 | |
| 		return errFileNameTooLong
 | |
| 	case errFileAccessDenied.Error():
 | |
| 		return errFileAccessDenied
 | |
| 	case errIsNotRegular.Error():
 | |
| 		return errIsNotRegular
 | |
| 	case errVolumeNotEmpty.Error():
 | |
| 		return errVolumeNotEmpty
 | |
| 	case errVolumeAccessDenied.Error():
 | |
| 		return errVolumeAccessDenied
 | |
| 	case errCorruptedFormat.Error():
 | |
| 		return errCorruptedFormat
 | |
| 	case errUnformattedDisk.Error():
 | |
| 		return errUnformattedDisk
 | |
| 	}
 | |
| 	return err
 | |
| }
 | |
| 
 | |
| // Initialize new rpc client.
 | |
| func newRPCClient(networkPath string) (StorageAPI, error) {
 | |
| 	// Input validation.
 | |
| 	if networkPath == "" || strings.LastIndex(networkPath, ":") == -1 {
 | |
| 		return nil, errInvalidArgument
 | |
| 	}
 | |
| 
 | |
| 	// Split network path into its components.
 | |
| 	netAddr, netPath, err := splitNetPath(networkPath)
 | |
| 	if err != nil {
 | |
| 		return nil, err
 | |
| 	}
 | |
| 
 | |
| 	// Dial minio rpc storage http path.
 | |
| 	rpcPath := path.Join(storageRPCPath, netPath)
 | |
| 	port := getPort(srvConfig.serverAddr)
 | |
| 	rpcAddr := netAddr + ":" + strconv.Itoa(port)
 | |
| 	// Initialize rpc client with network address and rpc path.
 | |
| 	cred := serverConfig.GetCredential()
 | |
| 	rpcClient := newAuthClient(&authConfig{
 | |
| 		accessKey:   cred.AccessKeyID,
 | |
| 		secretKey:   cred.SecretAccessKey,
 | |
| 		address:     rpcAddr,
 | |
| 		path:        rpcPath,
 | |
| 		loginMethod: "Storage.LoginHandler",
 | |
| 	})
 | |
| 	// Initialize network storage.
 | |
| 	ndisk := &networkStorage{
 | |
| 		netAddr:   netAddr,
 | |
| 		netPath:   netPath,
 | |
| 		rpcClient: rpcClient,
 | |
| 	}
 | |
| 
 | |
| 	// Returns successfully here.
 | |
| 	return ndisk, nil
 | |
| }
 | |
| 
 | |
| // DiskInfo - fetch disk information for a remote disk.
 | |
| func (n networkStorage) DiskInfo() (info disk.Info, err error) {
 | |
| 	args := GenericArgs{}
 | |
| 	if err = n.rpcClient.Call("Storage.DiskInfoHandler", &args, &info); err != nil {
 | |
| 		return disk.Info{}, err
 | |
| 	}
 | |
| 	return info, nil
 | |
| }
 | |
| 
 | |
| // MakeVol - create a volume on a remote disk.
 | |
| func (n networkStorage) MakeVol(volume string) error {
 | |
| 	reply := GenericReply{}
 | |
| 	args := GenericVolArgs{Vol: volume}
 | |
| 	if err := n.rpcClient.Call("Storage.MakeVolHandler", &args, &reply); err != nil {
 | |
| 		return toStorageErr(err)
 | |
| 	}
 | |
| 	return nil
 | |
| }
 | |
| 
 | |
| // ListVols - List all volumes on a remote disk.
 | |
| func (n networkStorage) ListVols() (vols []VolInfo, err error) {
 | |
| 	ListVols := ListVolsReply{}
 | |
| 	err = n.rpcClient.Call("Storage.ListVolsHandler", &GenericArgs{}, &ListVols)
 | |
| 	if err != nil {
 | |
| 		return nil, err
 | |
| 	}
 | |
| 	return ListVols.Vols, nil
 | |
| }
 | |
| 
 | |
| // StatVol - get current Stat volume info.
 | |
| func (n networkStorage) StatVol(volume string) (volInfo VolInfo, err error) {
 | |
| 	args := GenericVolArgs{Vol: volume}
 | |
| 	if err = n.rpcClient.Call("Storage.StatVolHandler", &args, &volInfo); err != nil {
 | |
| 		return VolInfo{}, toStorageErr(err)
 | |
| 	}
 | |
| 	return volInfo, nil
 | |
| }
 | |
| 
 | |
| // DeleteVol - Delete a volume.
 | |
| func (n networkStorage) DeleteVol(volume string) error {
 | |
| 	reply := GenericReply{}
 | |
| 	args := GenericVolArgs{Vol: volume}
 | |
| 	if err := n.rpcClient.Call("Storage.DeleteVolHandler", &args, &reply); err != nil {
 | |
| 		return toStorageErr(err)
 | |
| 	}
 | |
| 	return nil
 | |
| }
 | |
| 
 | |
| // File operations.
 | |
| 
 | |
| // CreateFile - create file.
 | |
| func (n networkStorage) AppendFile(volume, path string, buffer []byte) (err error) {
 | |
| 	reply := GenericReply{}
 | |
| 	if err = n.rpcClient.Call("Storage.AppendFileHandler", &AppendFileArgs{
 | |
| 		Vol:    volume,
 | |
| 		Path:   path,
 | |
| 		Buffer: buffer,
 | |
| 	}, &reply); err != nil {
 | |
| 		return toStorageErr(err)
 | |
| 	}
 | |
| 	return nil
 | |
| }
 | |
| 
 | |
| // StatFile - get latest Stat information for a file at path.
 | |
| func (n networkStorage) StatFile(volume, path string) (fileInfo FileInfo, err error) {
 | |
| 	if err = n.rpcClient.Call("Storage.StatFileHandler", &StatFileArgs{
 | |
| 		Vol:  volume,
 | |
| 		Path: path,
 | |
| 	}, &fileInfo); err != nil {
 | |
| 		return FileInfo{}, toStorageErr(err)
 | |
| 	}
 | |
| 	return fileInfo, nil
 | |
| }
 | |
| 
 | |
| // ReadAll - reads entire contents of the file at path until EOF, returns the
 | |
| // contents in a byte slice. Returns buf == nil if err != nil.
 | |
| // This API is meant to be used on files which have small memory footprint, do
 | |
| // not use this on large files as it would cause server to crash.
 | |
| func (n networkStorage) ReadAll(volume, path string) (buf []byte, err error) {
 | |
| 	if err = n.rpcClient.Call("Storage.ReadAllHandler", &ReadAllArgs{
 | |
| 		Vol:  volume,
 | |
| 		Path: path,
 | |
| 	}, &buf); err != nil {
 | |
| 		return nil, toStorageErr(err)
 | |
| 	}
 | |
| 	return buf, nil
 | |
| }
 | |
| 
 | |
| // ReadFile - reads a file.
 | |
| func (n networkStorage) ReadFile(volume string, path string, offset int64, buffer []byte) (m int64, err error) {
 | |
| 	var result []byte
 | |
| 	err = n.rpcClient.Call("Storage.ReadFileHandler", &ReadFileArgs{
 | |
| 		Vol:    volume,
 | |
| 		Path:   path,
 | |
| 		Offset: offset,
 | |
| 		Size:   len(buffer),
 | |
| 	}, &result)
 | |
| 	// Copy results to buffer.
 | |
| 	copy(buffer, result)
 | |
| 	// Return length of result, err if any.
 | |
| 	return int64(len(result)), toStorageErr(err)
 | |
| }
 | |
| 
 | |
| // ListDir - list all entries at prefix.
 | |
| func (n networkStorage) ListDir(volume, path string) (entries []string, err error) {
 | |
| 	if err = n.rpcClient.Call("Storage.ListDirHandler", &ListDirArgs{
 | |
| 		Vol:  volume,
 | |
| 		Path: path,
 | |
| 	}, &entries); err != nil {
 | |
| 		return nil, toStorageErr(err)
 | |
| 	}
 | |
| 	// Return successfully unmarshalled results.
 | |
| 	return entries, nil
 | |
| }
 | |
| 
 | |
| // DeleteFile - Delete a file at path.
 | |
| func (n networkStorage) DeleteFile(volume, path string) (err error) {
 | |
| 	reply := GenericReply{}
 | |
| 	if err = n.rpcClient.Call("Storage.DeleteFileHandler", &DeleteFileArgs{
 | |
| 		Vol:  volume,
 | |
| 		Path: path,
 | |
| 	}, &reply); err != nil {
 | |
| 		return toStorageErr(err)
 | |
| 	}
 | |
| 	return nil
 | |
| }
 | |
| 
 | |
| // RenameFile - Rename file.
 | |
| func (n networkStorage) RenameFile(srcVolume, srcPath, dstVolume, dstPath string) (err error) {
 | |
| 	reply := GenericReply{}
 | |
| 	if err = n.rpcClient.Call("Storage.RenameFileHandler", &RenameFileArgs{
 | |
| 		SrcVol:  srcVolume,
 | |
| 		SrcPath: srcPath,
 | |
| 		DstVol:  dstVolume,
 | |
| 		DstPath: dstPath,
 | |
| 	}, &reply); err != nil {
 | |
| 		return toStorageErr(err)
 | |
| 	}
 | |
| 	return nil
 | |
| }
 |