123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124 |
- package localsink
- import (
- "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"
- "os"
- "path/filepath"
- "strings"
- )
- type LocalSink struct {
- Dir string
- filerSource *source.FilerSource
- isIncremental bool
- }
- 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 {
- return strings.HasSuffix(key, ".part") && strings.Contains(key, "/"+s3_constants.MultipartUploadsFolder+"/")
- }
- func (localsink *LocalSink) initialize(dir string, isIncremental bool) error {
- localsink.Dir = dir
- localsink.isIncremental = isIncremental
- return nil
- }
- func (localsink *LocalSink) Initialize(configuration util.Configuration, prefix string) error {
- dir := configuration.GetString(prefix + "directory")
- isIncremental := configuration.GetBool(prefix + "is_incremental")
- glog.V(4).Infof("sink.local.directory: %v", dir)
- return localsink.initialize(dir, isIncremental)
- }
- func (localsink *LocalSink) GetSinkToDirectory() string {
- return localsink.Dir
- }
- func (localsink *LocalSink) IsIncremental() bool {
- return localsink.isIncremental
- }
- 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 {
- glog.V(0).Infof("remove entry key %s: %s", key, err)
- }
- 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)
- chunkViews := filer.ViewFromChunks(localsink.filerSource.LookupFileId, entry.GetChunks(), 0, int64(totalSize))
- dir := filepath.Dir(key)
- if _, err := os.Stat(dir); os.IsNotExist(err) {
- glog.V(4).Infof("Create Directory key: %s", dir)
- if err = os.MkdirAll(dir, 0755); err != nil {
- return err
- }
- }
- 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()
- writeFunc := func(data []byte) error {
- _, writeErr := dstFile.Write(data)
- return writeErr
- }
- if len(entry.Content) > 0 {
- return writeFunc(entry.Content)
- }
- 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
- foundExistingEntry = util.FileExists(key)
- err = localsink.CreateEntry(key, newEntry, signatures)
- return
- }
|