package local import ( "context" "errors" "fmt" "io" "net/http" "os" stdpath "path" "path/filepath" "strconv" "strings" "github.com/alist-org/alist/v3/internal/conf" "github.com/alist-org/alist/v3/internal/driver" "github.com/alist-org/alist/v3/internal/errs" "github.com/alist-org/alist/v3/internal/model" "github.com/alist-org/alist/v3/internal/sign" "github.com/alist-org/alist/v3/pkg/utils" "github.com/alist-org/alist/v3/server/common" _ "golang.org/x/image/webp" ) type Local struct { model.Storage Addition mkdirPerm int32 } func (d *Local) Config() driver.Config { return config } func (d *Local) Init(ctx context.Context) error { if d.MkdirPerm == "" { d.mkdirPerm = 0777 } else { v, err := strconv.ParseUint(d.MkdirPerm, 8, 32) if err != nil { return err } d.mkdirPerm = int32(v) } if !utils.Exists(d.GetRootPath()) { return fmt.Errorf("root folder %s not exists", d.GetRootPath()) } if !filepath.IsAbs(d.GetRootPath()) { abs, err := filepath.Abs(d.GetRootPath()) if err != nil { return err } d.Addition.RootFolderPath = abs } if d.ThumbCacheFolder != "" && !utils.Exists(d.ThumbCacheFolder) { err := os.MkdirAll(d.ThumbCacheFolder, os.FileMode(d.mkdirPerm)) if err != nil { return err } } return nil } func (d *Local) Drop(ctx context.Context) error { return nil } func (d *Local) GetAddition() driver.Additional { return &d.Addition } func (d *Local) List(ctx context.Context, dir model.Obj, args model.ListArgs) ([]model.Obj, error) { fullPath := dir.GetPath() rawFiles, err := readDir(fullPath) if err != nil { return nil, err } var files []model.Obj for _, f := range rawFiles { if !d.ShowHidden && strings.HasPrefix(f.Name(), ".") { continue } thumb := "" if d.Thumbnail { typeName := utils.GetFileType(f.Name()) if typeName == conf.IMAGE || typeName == conf.VIDEO { thumb = common.GetApiUrl(nil) + stdpath.Join("/d", args.ReqPath, f.Name()) thumb = utils.EncodePath(thumb, true) thumb += "?type=thumb&sign=" + sign.Sign(stdpath.Join(args.ReqPath, f.Name())) } } isFolder := f.IsDir() || isSymlinkDir(f, fullPath) var size int64 if !isFolder { size = f.Size() } file := model.ObjThumb{ Object: model.Object{ Path: filepath.Join(dir.GetPath(), f.Name()), Name: f.Name(), Modified: f.ModTime(), Size: size, IsFolder: isFolder, }, Thumbnail: model.Thumbnail{ Thumbnail: thumb, }, } files = append(files, &file) } return files, nil } func (d *Local) Get(ctx context.Context, path string) (model.Obj, error) { path = filepath.Join(d.GetRootPath(), path) f, err := os.Stat(path) if err != nil { if strings.Contains(err.Error(), "cannot find the file") { return nil, errs.ObjectNotFound } return nil, err } isFolder := f.IsDir() || isSymlinkDir(f, path) size := f.Size() if isFolder { size = 0 } file := model.Object{ Path: path, Name: f.Name(), Modified: f.ModTime(), Size: size, IsFolder: isFolder, } return &file, nil } func (d *Local) Link(ctx context.Context, file model.Obj, args model.LinkArgs) (*model.Link, error) { fullPath := file.GetPath() var link model.Link if args.Type == "thumb" && utils.Ext(file.GetName()) != "svg" { buf, thumbPath, err := d.getThumb(file) if err != nil { return nil, err } link.Header = http.Header{ "Content-Type": []string{"image/png"}, } if thumbPath != nil { link.FilePath = thumbPath } else { link.Data = io.NopCloser(buf) link.Header.Set("Content-Length", strconv.Itoa(buf.Len())) } } else { link.FilePath = &fullPath } return &link, nil } func (d *Local) MakeDir(ctx context.Context, parentDir model.Obj, dirName string) error { fullPath := filepath.Join(parentDir.GetPath(), dirName) err := os.MkdirAll(fullPath, os.FileMode(d.mkdirPerm)) if err != nil { return err } return nil } func (d *Local) Move(ctx context.Context, srcObj, dstDir model.Obj) error { srcPath := srcObj.GetPath() dstPath := filepath.Join(dstDir.GetPath(), srcObj.GetName()) if utils.IsSubPath(srcPath, dstPath) { return fmt.Errorf("the destination folder is a subfolder of the source folder") } err := os.Rename(srcPath, dstPath) if err != nil { return err } return nil } func (d *Local) Rename(ctx context.Context, srcObj model.Obj, newName string) error { srcPath := srcObj.GetPath() dstPath := filepath.Join(filepath.Dir(srcPath), newName) err := os.Rename(srcPath, dstPath) if err != nil { return err } return nil } func (d *Local) Copy(ctx context.Context, srcObj, dstDir model.Obj) error { srcPath := srcObj.GetPath() dstPath := filepath.Join(dstDir.GetPath(), srcObj.GetName()) if utils.IsSubPath(srcPath, dstPath) { return fmt.Errorf("the destination folder is a subfolder of the source folder") } var err error if srcObj.IsDir() { err = utils.CopyDir(srcPath, dstPath) } else { err = utils.CopyFile(srcPath, dstPath) } if err != nil { return err } return nil } func (d *Local) Remove(ctx context.Context, obj model.Obj) error { var err error if obj.IsDir() { err = os.RemoveAll(obj.GetPath()) } else { err = os.Remove(obj.GetPath()) } if err != nil { return err } return nil } func (d *Local) Put(ctx context.Context, dstDir model.Obj, stream model.FileStreamer, up driver.UpdateProgress) error { fullPath := filepath.Join(dstDir.GetPath(), stream.GetName()) out, err := os.Create(fullPath) if err != nil { return err } defer func() { _ = out.Close() if errors.Is(err, context.Canceled) { _ = os.Remove(fullPath) } }() err = utils.CopyWithCtx(ctx, out, stream, stream.GetSize(), up) if err != nil { return err } return nil } var _ driver.Driver = (*Local)(nil)