2021-07-27 05:53:44 +00:00
|
|
|
package remote_storage
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
"github.com/chrislusf/seaweedfs/weed/pb/filer_pb"
|
2021-08-26 22:18:34 +00:00
|
|
|
"github.com/chrislusf/seaweedfs/weed/pb/remote_pb"
|
2021-08-08 08:21:42 +00:00
|
|
|
"io"
|
2021-07-27 10:26:35 +00:00
|
|
|
"strings"
|
2021-07-27 05:53:44 +00:00
|
|
|
"sync"
|
|
|
|
)
|
|
|
|
|
2021-08-26 22:18:34 +00:00
|
|
|
func ParseLocation(remote string) (loc *remote_pb.RemoteStorageLocation) {
|
|
|
|
loc = &remote_pb.RemoteStorageLocation{}
|
2021-07-27 10:26:35 +00:00
|
|
|
if strings.HasSuffix(string(remote), "/") {
|
|
|
|
remote = remote[:len(remote)-1]
|
|
|
|
}
|
|
|
|
parts := strings.SplitN(string(remote), "/", 3)
|
2021-07-29 05:43:12 +00:00
|
|
|
if len(parts) >= 1 {
|
2021-07-29 09:08:55 +00:00
|
|
|
loc.Name = parts[0]
|
2021-07-27 10:26:35 +00:00
|
|
|
}
|
2021-07-29 05:43:12 +00:00
|
|
|
if len(parts) >= 2 {
|
2021-07-29 09:08:55 +00:00
|
|
|
loc.Bucket = parts[1]
|
2021-07-27 10:26:35 +00:00
|
|
|
}
|
2021-07-29 09:08:55 +00:00
|
|
|
loc.Path = string(remote[len(loc.Name)+1+len(loc.Bucket):])
|
|
|
|
if loc.Path == "" {
|
|
|
|
loc.Path = "/"
|
2021-07-27 10:26:35 +00:00
|
|
|
}
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2021-08-26 22:18:34 +00:00
|
|
|
func FormatLocation(loc *remote_pb.RemoteStorageLocation) string {
|
2021-08-16 03:07:13 +00:00
|
|
|
return fmt.Sprintf("%s/%s%s", loc.Name, loc.Bucket, loc.Path)
|
|
|
|
}
|
|
|
|
|
2021-07-27 05:53:44 +00:00
|
|
|
type VisitFunc func(dir string, name string, isDirectory bool, remoteEntry *filer_pb.RemoteEntry) error
|
|
|
|
|
|
|
|
type RemoteStorageClient interface {
|
2021-08-26 22:18:34 +00:00
|
|
|
Traverse(loc *remote_pb.RemoteStorageLocation, visitFn VisitFunc) error
|
|
|
|
ReadFile(loc *remote_pb.RemoteStorageLocation, offset int64, size int64) (data []byte, err error)
|
|
|
|
WriteDirectory(loc *remote_pb.RemoteStorageLocation, entry *filer_pb.Entry) (err error)
|
|
|
|
WriteFile(loc *remote_pb.RemoteStorageLocation, entry *filer_pb.Entry, reader io.Reader) (remoteEntry *filer_pb.RemoteEntry, err error)
|
|
|
|
UpdateFileMetadata(loc *remote_pb.RemoteStorageLocation, oldEntry *filer_pb.Entry, newEntry *filer_pb.Entry) (err error)
|
|
|
|
DeleteFile(loc *remote_pb.RemoteStorageLocation) (err error)
|
2021-07-27 05:53:44 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
type RemoteStorageClientMaker interface {
|
2021-08-26 22:18:34 +00:00
|
|
|
Make(remoteConf *remote_pb.RemoteConf) (RemoteStorageClient, error)
|
2021-07-27 05:53:44 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
var (
|
|
|
|
RemoteStorageClientMakers = make(map[string]RemoteStorageClientMaker)
|
|
|
|
remoteStorageClients = make(map[string]RemoteStorageClient)
|
|
|
|
remoteStorageClientsLock sync.Mutex
|
|
|
|
)
|
|
|
|
|
2021-08-26 22:18:34 +00:00
|
|
|
func makeRemoteStorageClient(remoteConf *remote_pb.RemoteConf) (RemoteStorageClient, error) {
|
2021-07-27 05:53:44 +00:00
|
|
|
maker, found := RemoteStorageClientMakers[remoteConf.Type]
|
|
|
|
if !found {
|
|
|
|
return nil, fmt.Errorf("remote storage type %s not found", remoteConf.Type)
|
|
|
|
}
|
|
|
|
return maker.Make(remoteConf)
|
|
|
|
}
|
|
|
|
|
2021-08-26 22:18:34 +00:00
|
|
|
func GetRemoteStorage(remoteConf *remote_pb.RemoteConf) (RemoteStorageClient, error) {
|
2021-07-27 05:53:44 +00:00
|
|
|
remoteStorageClientsLock.Lock()
|
|
|
|
defer remoteStorageClientsLock.Unlock()
|
|
|
|
|
|
|
|
existingRemoteStorageClient, found := remoteStorageClients[remoteConf.Name]
|
|
|
|
if found {
|
|
|
|
return existingRemoteStorageClient, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
newRemoteStorageClient, err := makeRemoteStorageClient(remoteConf)
|
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("make remote storage client %s: %v", remoteConf.Name, err)
|
|
|
|
}
|
|
|
|
|
|
|
|
remoteStorageClients[remoteConf.Name] = newRemoteStorageClient
|
|
|
|
|
|
|
|
return newRemoteStorageClient, nil
|
|
|
|
}
|