mirror of https://github.com/Xhofe/alist
159 lines
3.7 KiB
Go
159 lines
3.7 KiB
Go
package ftp
|
|
|
|
import (
|
|
"context"
|
|
ftpserver "github.com/KirCute/ftpserverlib-pasvportmap"
|
|
"github.com/alist-org/alist/v3/internal/errs"
|
|
"github.com/alist-org/alist/v3/internal/fs"
|
|
"github.com/alist-org/alist/v3/internal/model"
|
|
"github.com/alist-org/alist/v3/internal/op"
|
|
"github.com/alist-org/alist/v3/internal/stream"
|
|
"github.com/alist-org/alist/v3/server/common"
|
|
"github.com/pkg/errors"
|
|
fs2 "io/fs"
|
|
"net/http"
|
|
"os"
|
|
"time"
|
|
)
|
|
|
|
type FileDownloadProxy struct {
|
|
ftpserver.FileTransfer
|
|
reader stream.SStreamReadAtSeeker
|
|
}
|
|
|
|
func OpenDownload(ctx context.Context, reqPath string, offset int64) (*FileDownloadProxy, error) {
|
|
user := ctx.Value("user").(*model.User)
|
|
meta, err := op.GetNearestMeta(reqPath)
|
|
if err != nil {
|
|
if !errors.Is(errors.Cause(err), errs.MetaNotFound) {
|
|
return nil, err
|
|
}
|
|
}
|
|
ctx = context.WithValue(ctx, "meta", meta)
|
|
if !common.CanAccess(user, meta, reqPath, ctx.Value("meta_pass").(string)) {
|
|
return nil, errs.PermissionDenied
|
|
}
|
|
|
|
// directly use proxy
|
|
header := *(ctx.Value("proxy_header").(*http.Header))
|
|
link, obj, err := fs.Link(ctx, reqPath, model.LinkArgs{
|
|
IP: ctx.Value("client_ip").(string),
|
|
Header: header,
|
|
})
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
fileStream := stream.FileStream{
|
|
Obj: obj,
|
|
Ctx: ctx,
|
|
}
|
|
ss, err := stream.NewSeekableStream(fileStream, link)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
reader, err := stream.NewReadAtSeeker(ss, offset)
|
|
if err != nil {
|
|
_ = ss.Close()
|
|
return nil, err
|
|
}
|
|
return &FileDownloadProxy{reader: reader}, nil
|
|
}
|
|
|
|
func (f *FileDownloadProxy) Read(p []byte) (n int, err error) {
|
|
return f.reader.Read(p)
|
|
}
|
|
|
|
func (f *FileDownloadProxy) Write(p []byte) (n int, err error) {
|
|
return 0, errs.NotSupport
|
|
}
|
|
|
|
func (f *FileDownloadProxy) Seek(offset int64, whence int) (int64, error) {
|
|
return f.reader.Seek(offset, whence)
|
|
}
|
|
|
|
func (f *FileDownloadProxy) Close() error {
|
|
return f.reader.Close()
|
|
}
|
|
|
|
type OsFileInfoAdapter struct {
|
|
obj model.Obj
|
|
}
|
|
|
|
func (o *OsFileInfoAdapter) Name() string {
|
|
return o.obj.GetName()
|
|
}
|
|
|
|
func (o *OsFileInfoAdapter) Size() int64 {
|
|
return o.obj.GetSize()
|
|
}
|
|
|
|
func (o *OsFileInfoAdapter) Mode() fs2.FileMode {
|
|
var mode fs2.FileMode = 0755
|
|
if o.IsDir() {
|
|
mode |= fs2.ModeDir
|
|
}
|
|
return mode
|
|
}
|
|
|
|
func (o *OsFileInfoAdapter) ModTime() time.Time {
|
|
return o.obj.ModTime()
|
|
}
|
|
|
|
func (o *OsFileInfoAdapter) IsDir() bool {
|
|
return o.obj.IsDir()
|
|
}
|
|
|
|
func (o *OsFileInfoAdapter) Sys() any {
|
|
return o.obj
|
|
}
|
|
|
|
func Stat(ctx context.Context, path string) (os.FileInfo, error) {
|
|
user := ctx.Value("user").(*model.User)
|
|
reqPath, err := user.JoinPath(path)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
meta, err := op.GetNearestMeta(reqPath)
|
|
if err != nil {
|
|
if !errors.Is(errors.Cause(err), errs.MetaNotFound) {
|
|
return nil, err
|
|
}
|
|
}
|
|
ctx = context.WithValue(ctx, "meta", meta)
|
|
if !common.CanAccess(user, meta, reqPath, ctx.Value("meta_pass").(string)) {
|
|
return nil, errs.PermissionDenied
|
|
}
|
|
obj, err := fs.Get(ctx, reqPath, &fs.GetArgs{})
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
return &OsFileInfoAdapter{obj: obj}, nil
|
|
}
|
|
|
|
func List(ctx context.Context, path string) ([]os.FileInfo, error) {
|
|
user := ctx.Value("user").(*model.User)
|
|
reqPath, err := user.JoinPath(path)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
meta, err := op.GetNearestMeta(reqPath)
|
|
if err != nil {
|
|
if !errors.Is(errors.Cause(err), errs.MetaNotFound) {
|
|
return nil, err
|
|
}
|
|
}
|
|
ctx = context.WithValue(ctx, "meta", meta)
|
|
if !common.CanAccess(user, meta, reqPath, ctx.Value("meta_pass").(string)) {
|
|
return nil, errs.PermissionDenied
|
|
}
|
|
objs, err := fs.List(ctx, reqPath, &fs.ListArgs{})
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
ret := make([]os.FileInfo, len(objs))
|
|
for i, obj := range objs {
|
|
ret[i] = &OsFileInfoAdapter{obj: obj}
|
|
}
|
|
return ret, nil
|
|
}
|