2021-01-26 17:50:25 +00:00
|
|
|
package localsink
|
|
|
|
|
|
|
|
import (
|
2022-07-29 07:17:28 +00:00
|
|
|
"github.com/seaweedfs/seaweedfs/weed/filer"
|
|
|
|
"github.com/seaweedfs/seaweedfs/weed/glog"
|
|
|
|
"github.com/seaweedfs/seaweedfs/weed/pb/filer_pb"
|
|
|
|
"github.com/seaweedfs/seaweedfs/weed/replication/repl_util"
|
|
|
|
"github.com/seaweedfs/seaweedfs/weed/replication/sink"
|
|
|
|
"github.com/seaweedfs/seaweedfs/weed/replication/source"
|
|
|
|
"github.com/seaweedfs/seaweedfs/weed/s3api/s3_constants"
|
|
|
|
"github.com/seaweedfs/seaweedfs/weed/util"
|
2021-01-26 17:50:25 +00:00
|
|
|
"os"
|
|
|
|
"path/filepath"
|
|
|
|
"strings"
|
|
|
|
)
|
|
|
|
|
|
|
|
type LocalSink struct {
|
2021-05-26 00:19:17 +00:00
|
|
|
Dir string
|
|
|
|
filerSource *source.FilerSource
|
|
|
|
isIncremental bool
|
2021-01-26 17:50:25 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func init() {
|
|
|
|
sink.Sinks = append(sink.Sinks, &LocalSink{})
|
|
|
|
}
|
|
|
|
|
|
|
|
func (localsink *LocalSink) SetSourceFiler(s *source.FilerSource) {
|
|
|
|
localsink.filerSource = s
|
|
|
|
}
|
|
|
|
|
|
|
|
func (localsink *LocalSink) GetName() string {
|
|
|
|
return "local"
|
|
|
|
}
|
|
|
|
|
|
|
|
func (localsink *LocalSink) isMultiPartEntry(key string) bool {
|
2022-06-29 11:21:16 +00:00
|
|
|
return strings.HasSuffix(key, ".part") && strings.Contains(key, "/"+s3_constants.MultipartUploadsFolder+"/")
|
2021-01-26 17:50:25 +00:00
|
|
|
}
|
|
|
|
|
2021-05-26 00:19:17 +00:00
|
|
|
func (localsink *LocalSink) initialize(dir string, isIncremental bool) error {
|
2021-01-27 10:01:33 +00:00
|
|
|
localsink.Dir = dir
|
2021-05-26 00:19:17 +00:00
|
|
|
localsink.isIncremental = isIncremental
|
2021-01-26 17:50:25 +00:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (localsink *LocalSink) Initialize(configuration util.Configuration, prefix string) error {
|
|
|
|
dir := configuration.GetString(prefix + "directory")
|
2021-05-26 00:19:17 +00:00
|
|
|
isIncremental := configuration.GetBool(prefix + "is_incremental")
|
2021-01-26 17:50:25 +00:00
|
|
|
glog.V(4).Infof("sink.local.directory: %v", dir)
|
2021-05-26 00:19:17 +00:00
|
|
|
return localsink.initialize(dir, isIncremental)
|
2021-01-26 17:50:25 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func (localsink *LocalSink) GetSinkToDirectory() string {
|
2021-01-27 10:01:33 +00:00
|
|
|
return localsink.Dir
|
2021-01-26 17:50:25 +00:00
|
|
|
}
|
|
|
|
|
2021-03-01 00:19:03 +00:00
|
|
|
func (localsink *LocalSink) IsIncremental() bool {
|
2021-05-26 00:19:17 +00:00
|
|
|
return localsink.isIncremental
|
2021-03-01 00:19:03 +00:00
|
|
|
}
|
|
|
|
|
2021-01-26 17:50:25 +00:00
|
|
|
func (localsink *LocalSink) DeleteEntry(key string, isDirectory, deleteIncludeChunks bool, signatures []int32) error {
|
|
|
|
if localsink.isMultiPartEntry(key) {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
glog.V(4).Infof("Delete Entry key: %s", key)
|
|
|
|
if err := os.Remove(key); err != nil {
|
2021-01-27 07:45:58 +00:00
|
|
|
glog.V(0).Infof("remove entry key %s: %s", key, err)
|
2021-01-26 17:50:25 +00:00
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (localsink *LocalSink) CreateEntry(key string, entry *filer_pb.Entry, signatures []int32) error {
|
|
|
|
if entry.IsDirectory || localsink.isMultiPartEntry(key) {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
glog.V(4).Infof("Create Entry key: %s", key)
|
|
|
|
|
|
|
|
totalSize := filer.FileSize(entry)
|
2022-11-15 14:33:36 +00:00
|
|
|
chunkViews := filer.ViewFromChunks(localsink.filerSource.LookupFileId, entry.GetChunks(), 0, int64(totalSize))
|
2021-01-26 17:50:25 +00:00
|
|
|
|
|
|
|
dir := filepath.Dir(key)
|
|
|
|
|
|
|
|
if _, err := os.Stat(dir); os.IsNotExist(err) {
|
2022-09-14 17:11:31 +00:00
|
|
|
glog.V(4).Infof("Create Directory key: %s", dir)
|
2021-03-01 00:20:47 +00:00
|
|
|
if err = os.MkdirAll(dir, 0755); err != nil {
|
2021-01-26 17:50:25 +00:00
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-05-26 21:42:18 +00:00
|
|
|
if entry.IsDirectory {
|
|
|
|
return os.Mkdir(key, os.FileMode(entry.Attributes.FileMode))
|
|
|
|
}
|
|
|
|
|
|
|
|
dstFile, err := os.OpenFile(key, os.O_RDWR|os.O_CREATE|os.O_TRUNC, os.FileMode(entry.Attributes.FileMode))
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
defer dstFile.Close()
|
|
|
|
|
2021-01-26 17:50:25 +00:00
|
|
|
writeFunc := func(data []byte) error {
|
2021-05-26 21:42:18 +00:00
|
|
|
_, writeErr := dstFile.Write(data)
|
2021-01-26 17:50:25 +00:00
|
|
|
return writeErr
|
|
|
|
}
|
|
|
|
|
2022-08-20 06:00:56 +00:00
|
|
|
if len(entry.Content) > 0 {
|
|
|
|
return writeFunc(entry.Content)
|
|
|
|
}
|
|
|
|
|
2021-01-26 17:50:25 +00:00
|
|
|
if err := repl_util.CopyFromChunkViews(chunkViews, localsink.filerSource, writeFunc); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (localsink *LocalSink) UpdateEntry(key string, oldEntry *filer_pb.Entry, newParentPath string, newEntry *filer_pb.Entry, deleteIncludeChunks bool, signatures []int32) (foundExistingEntry bool, err error) {
|
|
|
|
if localsink.isMultiPartEntry(key) {
|
|
|
|
return true, nil
|
|
|
|
}
|
|
|
|
glog.V(4).Infof("Update Entry key: %s", key)
|
|
|
|
// do delete and create
|
2021-05-26 21:42:18 +00:00
|
|
|
foundExistingEntry = util.FileExists(key)
|
|
|
|
err = localsink.CreateEntry(key, newEntry, signatures)
|
|
|
|
return
|
2021-01-26 17:50:25 +00:00
|
|
|
}
|