123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641 |
- package weed_server
- import (
- "context"
- "fmt"
- "io"
- "os"
- "path"
- "strings"
- "time"
- "github.com/seaweedfs/seaweedfs/weed/util/buffered_writer"
- "golang.org/x/net/webdav"
- "google.golang.org/grpc"
- "github.com/seaweedfs/seaweedfs/weed/operation"
- "github.com/seaweedfs/seaweedfs/weed/pb"
- "github.com/seaweedfs/seaweedfs/weed/pb/filer_pb"
- "github.com/seaweedfs/seaweedfs/weed/util"
- "github.com/seaweedfs/seaweedfs/weed/util/chunk_cache"
- "github.com/seaweedfs/seaweedfs/weed/filer"
- "github.com/seaweedfs/seaweedfs/weed/glog"
- "github.com/seaweedfs/seaweedfs/weed/security"
- )
- type WebDavOption struct {
- Filer pb.ServerAddress
- FilerRootPath string
- DomainName string
- BucketsPath string
- GrpcDialOption grpc.DialOption
- Collection string
- Replication string
- DiskType string
- Uid uint32
- Gid uint32
- Cipher bool
- CacheDir string
- CacheSizeMB int64
- MaxMB int
- }
- type WebDavServer struct {
- option *WebDavOption
- secret security.SigningKey
- filer *filer.Filer
- grpcDialOption grpc.DialOption
- Handler *webdav.Handler
- }
- func max(x, y int64) int64 {
- if x <= y {
- return y
- }
- return x
- }
- func NewWebDavServer(option *WebDavOption) (ws *WebDavServer, err error) {
- fs, _ := NewWebDavFileSystem(option)
- // Fix no set filer.path , accessing "/" returns "//"
- if option.FilerRootPath == "/" {
- option.FilerRootPath = ""
- }
- // filer.path non "/" option means we are accessing filer's sub-folders
- if option.FilerRootPath != "" {
- fs = NewWrappedFs(fs, path.Clean(option.FilerRootPath))
- }
- ws = &WebDavServer{
- option: option,
- grpcDialOption: security.LoadClientTLS(util.GetViper(), "grpc.filer"),
- Handler: &webdav.Handler{
- FileSystem: fs,
- LockSystem: webdav.NewMemLS(),
- },
- }
- return ws, nil
- }
- // adapted from https://github.com/mattn/davfs/blob/master/plugin/mysql/mysql.go
- type WebDavFileSystem struct {
- option *WebDavOption
- secret security.SigningKey
- grpcDialOption grpc.DialOption
- chunkCache *chunk_cache.TieredChunkCache
- readerCache *filer.ReaderCache
- signature int32
- }
- type FileInfo struct {
- name string
- size int64
- mode os.FileMode
- modifiedTime time.Time
- etag string
- isDirectory bool
- }
- func (fi *FileInfo) Name() string { return fi.name }
- func (fi *FileInfo) Size() int64 { return fi.size }
- func (fi *FileInfo) Mode() os.FileMode { return fi.mode }
- func (fi *FileInfo) ModTime() time.Time { return fi.modifiedTime }
- func (fi *FileInfo) IsDir() bool { return fi.isDirectory }
- func (fi *FileInfo) Sys() interface{} { return nil }
- func (fi *FileInfo) ETag(ctx context.Context) (string, error) {
- return fi.etag, nil
- }
- type WebDavFile struct {
- fs *WebDavFileSystem
- name string
- isDirectory bool
- off int64
- entry *filer_pb.Entry
- visibleIntervals *filer.IntervalList[*filer.VisibleInterval]
- reader io.ReaderAt
- bufWriter *buffered_writer.BufferedWriteCloser
- }
- func NewWebDavFileSystem(option *WebDavOption) (webdav.FileSystem, error) {
- cacheUniqueId := util.Md5String([]byte("webdav" + string(option.Filer) + util.Version()))[0:8]
- cacheDir := path.Join(option.CacheDir, cacheUniqueId)
- os.MkdirAll(cacheDir, os.FileMode(0755))
- chunkCache := chunk_cache.NewTieredChunkCache(256, cacheDir, option.CacheSizeMB, 1024*1024)
- t := &WebDavFileSystem{
- option: option,
- chunkCache: chunkCache,
- signature: util.RandomInt32(),
- }
- t.readerCache = filer.NewReaderCache(32, chunkCache, filer.LookupFn(t))
- return t, nil
- }
- var _ = filer_pb.FilerClient(&WebDavFileSystem{})
- func (fs *WebDavFileSystem) WithFilerClient(streamingMode bool, fn func(filer_pb.SeaweedFilerClient) error) error {
- return pb.WithGrpcClient(streamingMode, fs.signature, func(grpcConnection *grpc.ClientConn) error {
- client := filer_pb.NewSeaweedFilerClient(grpcConnection)
- return fn(client)
- }, fs.option.Filer.ToGrpcAddress(), false, fs.option.GrpcDialOption)
- }
- func (fs *WebDavFileSystem) AdjustedUrl(location *filer_pb.Location) string {
- return location.Url
- }
- func (fs *WebDavFileSystem) GetDataCenter() string {
- return ""
- }
- func clearName(name string) (string, error) {
- slashed := strings.HasSuffix(name, "/")
- name = path.Clean(name)
- if !strings.HasSuffix(name, "/") && slashed {
- name += "/"
- }
- if !strings.HasPrefix(name, "/") {
- return "", os.ErrInvalid
- }
- return name, nil
- }
- func (fs *WebDavFileSystem) Mkdir(ctx context.Context, fullDirPath string, perm os.FileMode) error {
- glog.V(2).Infof("WebDavFileSystem.Mkdir %v", fullDirPath)
- if !strings.HasSuffix(fullDirPath, "/") {
- fullDirPath += "/"
- }
- var err error
- if fullDirPath, err = clearName(fullDirPath); err != nil {
- return err
- }
- _, err = fs.stat(ctx, fullDirPath)
- if err == nil {
- return os.ErrExist
- }
- return fs.WithFilerClient(false, func(client filer_pb.SeaweedFilerClient) error {
- dir, name := util.FullPath(fullDirPath).DirAndName()
- request := &filer_pb.CreateEntryRequest{
- Directory: dir,
- Entry: &filer_pb.Entry{
- Name: name,
- IsDirectory: true,
- Attributes: &filer_pb.FuseAttributes{
- Mtime: time.Now().Unix(),
- Crtime: time.Now().Unix(),
- FileMode: uint32(perm | os.ModeDir),
- Uid: fs.option.Uid,
- Gid: fs.option.Gid,
- },
- },
- Signatures: []int32{fs.signature},
- }
- glog.V(1).Infof("mkdir: %v", request)
- if err := filer_pb.CreateEntry(client, request); err != nil {
- return fmt.Errorf("mkdir %s/%s: %v", dir, name, err)
- }
- return nil
- })
- }
- func (fs *WebDavFileSystem) OpenFile(ctx context.Context, fullFilePath string, flag int, perm os.FileMode) (webdav.File, error) {
- glog.V(2).Infof("WebDavFileSystem.OpenFile %v %x", fullFilePath, flag)
- var err error
- if fullFilePath, err = clearName(fullFilePath); err != nil {
- return nil, err
- }
- if flag&os.O_CREATE != 0 {
- // file should not have / suffix.
- if strings.HasSuffix(fullFilePath, "/") {
- return nil, os.ErrInvalid
- }
- _, err = fs.stat(ctx, fullFilePath)
- if err == nil {
- if flag&os.O_EXCL != 0 {
- return nil, os.ErrExist
- }
- fs.removeAll(ctx, fullFilePath)
- }
- dir, name := util.FullPath(fullFilePath).DirAndName()
- err = fs.WithFilerClient(false, func(client filer_pb.SeaweedFilerClient) error {
- if err := filer_pb.CreateEntry(client, &filer_pb.CreateEntryRequest{
- Directory: dir,
- Entry: &filer_pb.Entry{
- Name: name,
- IsDirectory: perm&os.ModeDir > 0,
- Attributes: &filer_pb.FuseAttributes{
- Mtime: 0,
- Crtime: time.Now().Unix(),
- FileMode: uint32(perm),
- Uid: fs.option.Uid,
- Gid: fs.option.Gid,
- TtlSec: 0,
- },
- },
- Signatures: []int32{fs.signature},
- }); err != nil {
- return fmt.Errorf("create %s: %v", fullFilePath, err)
- }
- return nil
- })
- if err != nil {
- return nil, err
- }
- return &WebDavFile{
- fs: fs,
- name: fullFilePath,
- isDirectory: false,
- bufWriter: buffered_writer.NewBufferedWriteCloser(fs.option.MaxMB * 1024 * 1024),
- }, nil
- }
- fi, err := fs.stat(ctx, fullFilePath)
- if err != nil {
- return nil, os.ErrNotExist
- }
- if !strings.HasSuffix(fullFilePath, "/") && fi.IsDir() {
- fullFilePath += "/"
- }
- return &WebDavFile{
- fs: fs,
- name: fullFilePath,
- isDirectory: false,
- bufWriter: buffered_writer.NewBufferedWriteCloser(fs.option.MaxMB * 1024 * 1024),
- }, nil
- }
- func (fs *WebDavFileSystem) removeAll(ctx context.Context, fullFilePath string) error {
- var err error
- if fullFilePath, err = clearName(fullFilePath); err != nil {
- return err
- }
- dir, name := util.FullPath(fullFilePath).DirAndName()
- return filer_pb.Remove(fs, dir, name, true, false, false, false, []int32{fs.signature})
- }
- func (fs *WebDavFileSystem) RemoveAll(ctx context.Context, name string) error {
- glog.V(2).Infof("WebDavFileSystem.RemoveAll %v", name)
- return fs.removeAll(ctx, name)
- }
- func (fs *WebDavFileSystem) Rename(ctx context.Context, oldName, newName string) error {
- glog.V(2).Infof("WebDavFileSystem.Rename %v to %v", oldName, newName)
- var err error
- if oldName, err = clearName(oldName); err != nil {
- return err
- }
- if newName, err = clearName(newName); err != nil {
- return err
- }
- of, err := fs.stat(ctx, oldName)
- if err != nil {
- return os.ErrExist
- }
- if of.IsDir() {
- if strings.HasSuffix(oldName, "/") {
- oldName = strings.TrimRight(oldName, "/")
- }
- if strings.HasSuffix(newName, "/") {
- newName = strings.TrimRight(newName, "/")
- }
- }
- _, err = fs.stat(ctx, newName)
- if err == nil {
- return os.ErrExist
- }
- oldDir, oldBaseName := util.FullPath(oldName).DirAndName()
- newDir, newBaseName := util.FullPath(newName).DirAndName()
- return fs.WithFilerClient(false, func(client filer_pb.SeaweedFilerClient) error {
- request := &filer_pb.AtomicRenameEntryRequest{
- OldDirectory: oldDir,
- OldName: oldBaseName,
- NewDirectory: newDir,
- NewName: newBaseName,
- }
- _, err := client.AtomicRenameEntry(ctx, request)
- if err != nil {
- return fmt.Errorf("renaming %s/%s => %s/%s: %v", oldDir, oldBaseName, newDir, newBaseName, err)
- }
- return nil
- })
- }
- func (fs *WebDavFileSystem) stat(ctx context.Context, fullFilePath string) (os.FileInfo, error) {
- var err error
- if fullFilePath, err = clearName(fullFilePath); err != nil {
- return nil, err
- }
- fullpath := util.FullPath(fullFilePath)
- var fi FileInfo
- entry, err := filer_pb.GetEntry(fs, fullpath)
- if entry == nil {
- return nil, os.ErrNotExist
- }
- if err != nil {
- return nil, err
- }
- fi.size = int64(filer.FileSize(entry))
- fi.name = string(fullpath)
- fi.mode = os.FileMode(entry.Attributes.FileMode)
- fi.modifiedTime = time.Unix(entry.Attributes.Mtime, 0)
- fi.etag = filer.ETag(entry)
- fi.isDirectory = entry.IsDirectory
- if fi.name == "/" {
- fi.modifiedTime = time.Now()
- fi.isDirectory = true
- }
- return &fi, nil
- }
- func (fs *WebDavFileSystem) Stat(ctx context.Context, name string) (os.FileInfo, error) {
- glog.V(2).Infof("WebDavFileSystem.Stat %v", name)
- return fs.stat(ctx, name)
- }
- func (f *WebDavFile) saveDataAsChunk(reader io.Reader, name string, offset int64, tsNs int64) (chunk *filer_pb.FileChunk, err error) {
- uploader, uploaderErr := operation.NewUploader()
- if uploaderErr != nil {
- glog.V(0).Infof("upload data %v: %v", f.name, uploaderErr)
- return nil, fmt.Errorf("upload data: %v", uploaderErr)
- }
- fileId, uploadResult, flushErr, _ := uploader.UploadWithRetry(
- f.fs,
- &filer_pb.AssignVolumeRequest{
- Count: 1,
- Replication: f.fs.option.Replication,
- Collection: f.fs.option.Collection,
- DiskType: f.fs.option.DiskType,
- Path: name,
- },
- &operation.UploadOption{
- Filename: f.name,
- Cipher: f.fs.option.Cipher,
- IsInputCompressed: false,
- MimeType: "",
- PairMap: nil,
- },
- func(host, fileId string) string {
- return fmt.Sprintf("http://%s/%s", host, fileId)
- },
- reader,
- )
- if flushErr != nil {
- glog.V(0).Infof("upload data %v: %v", f.name, flushErr)
- return nil, fmt.Errorf("upload data: %v", flushErr)
- }
- if uploadResult.Error != "" {
- glog.V(0).Infof("upload failure %v: %v", f.name, flushErr)
- return nil, fmt.Errorf("upload result: %v", uploadResult.Error)
- }
- return uploadResult.ToPbFileChunk(fileId, offset, tsNs), nil
- }
- func (f *WebDavFile) Write(buf []byte) (int, error) {
- glog.V(2).Infof("WebDavFileSystem.Write %v", f.name)
- fullPath := util.FullPath(f.name)
- dir, _ := fullPath.DirAndName()
- var getErr error
- ctx := context.Background()
- if f.entry == nil {
- f.entry, getErr = filer_pb.GetEntry(f.fs, fullPath)
- }
- if f.entry == nil {
- return 0, getErr
- }
- if getErr != nil {
- return 0, getErr
- }
- if f.bufWriter.FlushFunc == nil {
- f.bufWriter.FlushFunc = func(data []byte, offset int64) (flushErr error) {
- var chunk *filer_pb.FileChunk
- chunk, flushErr = f.saveDataAsChunk(util.NewBytesReader(data), f.name, offset, time.Now().UnixNano())
- if flushErr != nil {
- if f.entry.Attributes.Mtime == 0 {
- if err := f.fs.removeAll(ctx, f.name); err != nil {
- glog.Errorf("bufWriter.Flush remove file error: %+v", f.name)
- }
- }
- return fmt.Errorf("%s upload result: %v", f.name, flushErr)
- }
- f.entry.Content = nil
- f.entry.Chunks = append(f.entry.GetChunks(), chunk)
- return flushErr
- }
- f.bufWriter.CloseFunc = func() error {
- manifestedChunks, manifestErr := filer.MaybeManifestize(f.saveDataAsChunk, f.entry.GetChunks())
- if manifestErr != nil {
- // not good, but should be ok
- glog.V(0).Infof("file %s close MaybeManifestize: %v", f.name, manifestErr)
- } else {
- f.entry.Chunks = manifestedChunks
- }
- flushErr := f.fs.WithFilerClient(false, func(client filer_pb.SeaweedFilerClient) error {
- f.entry.Attributes.Mtime = time.Now().Unix()
- request := &filer_pb.UpdateEntryRequest{
- Directory: dir,
- Entry: f.entry,
- Signatures: []int32{f.fs.signature},
- }
- if _, err := client.UpdateEntry(ctx, request); err != nil {
- return fmt.Errorf("update %s: %v", f.name, err)
- }
- return nil
- })
- return flushErr
- }
- }
- written, err := f.bufWriter.Write(buf)
- if err == nil {
- f.entry.Attributes.FileSize = uint64(max(f.off+int64(written), int64(f.entry.Attributes.FileSize)))
- glog.V(3).Infof("WebDavFileSystem.Write %v: written [%d,%d)", f.name, f.off, f.off+int64(len(buf)))
- f.off += int64(written)
- }
- return written, err
- }
- func (f *WebDavFile) Close() error {
- glog.V(2).Infof("WebDavFileSystem.Close %v", f.name)
- err := f.bufWriter.Close()
- if f.entry != nil {
- f.entry = nil
- f.visibleIntervals = nil
- }
- return err
- }
- func (f *WebDavFile) Read(p []byte) (readSize int, err error) {
- glog.V(2).Infof("WebDavFileSystem.Read %v", f.name)
- if f.entry == nil {
- f.entry, err = filer_pb.GetEntry(f.fs, util.FullPath(f.name))
- }
- if f.entry == nil {
- return 0, err
- }
- if err != nil {
- return 0, err
- }
- fileSize := int64(filer.FileSize(f.entry))
- if fileSize == 0 {
- return 0, io.EOF
- }
- if f.visibleIntervals == nil {
- f.visibleIntervals, _ = filer.NonOverlappingVisibleIntervals(filer.LookupFn(f.fs), f.entry.GetChunks(), 0, fileSize)
- f.reader = nil
- }
- if f.reader == nil {
- chunkViews := filer.ViewFromVisibleIntervals(f.visibleIntervals, 0, fileSize)
- f.reader = filer.NewChunkReaderAtFromClient(f.fs.readerCache, chunkViews, fileSize)
- }
- readSize, err = f.reader.ReadAt(p, f.off)
- glog.V(3).Infof("WebDavFileSystem.Read %v: [%d,%d)", f.name, f.off, f.off+int64(readSize))
- f.off += int64(readSize)
- if err != nil && err != io.EOF {
- glog.Errorf("file read %s: %v", f.name, err)
- }
- return
- }
- func (f *WebDavFile) Readdir(count int) (ret []os.FileInfo, err error) {
- glog.V(2).Infof("WebDavFileSystem.Readdir %v count %d", f.name, count)
- dir, _ := util.FullPath(f.name).DirAndName()
- err = filer_pb.ReadDirAllEntries(f.fs, util.FullPath(dir), "", func(entry *filer_pb.Entry, isLast bool) error {
- fi := FileInfo{
- size: int64(filer.FileSize(entry)),
- name: entry.Name,
- mode: os.FileMode(entry.Attributes.FileMode),
- modifiedTime: time.Unix(entry.Attributes.Mtime, 0),
- isDirectory: entry.IsDirectory,
- }
- if !strings.HasSuffix(fi.name, "/") && fi.IsDir() {
- fi.name += "/"
- }
- glog.V(4).Infof("entry: %v", fi.name)
- ret = append(ret, &fi)
- return nil
- })
- if err != nil {
- return nil, err
- }
- old := f.off
- if old >= int64(len(ret)) {
- if count > 0 {
- return nil, io.EOF
- }
- return nil, nil
- }
- if count > 0 {
- f.off += int64(count)
- if f.off > int64(len(ret)) {
- f.off = int64(len(ret))
- }
- } else {
- f.off = int64(len(ret))
- old = 0
- }
- return ret[old:f.off], nil
- }
- func (f *WebDavFile) Seek(offset int64, whence int) (int64, error) {
- glog.V(2).Infof("WebDavFile.Seek %v %v %v", f.name, offset, whence)
- ctx := context.Background()
- var err error
- switch whence {
- case io.SeekStart:
- f.off = 0
- case io.SeekEnd:
- if fi, err := f.fs.stat(ctx, f.name); err != nil {
- return 0, err
- } else {
- f.off = fi.Size()
- }
- }
- f.off += offset
- return f.off, err
- }
- func (f *WebDavFile) Stat() (os.FileInfo, error) {
- glog.V(2).Infof("WebDavFile.Stat %v", f.name)
- ctx := context.Background()
- return f.fs.stat(ctx, f.name)
- }
|