2020-02-25 15:20:57 +01:00
|
|
|
// Copyright 2020 The goftp Authors. All rights reserved.
|
|
|
|
// Use of this source code is governed by a MIT-style
|
|
|
|
// license that can be found in the LICENSE file.
|
|
|
|
|
|
|
|
package server
|
|
|
|
|
|
|
|
import (
|
|
|
|
"errors"
|
|
|
|
"fmt"
|
|
|
|
"io"
|
|
|
|
"os"
|
|
|
|
"path/filepath"
|
|
|
|
"strings"
|
|
|
|
)
|
|
|
|
|
2020-06-30 10:16:43 +02:00
|
|
|
// FileDriver implements Driver directly read local file system
|
2020-02-25 15:20:57 +01:00
|
|
|
type FileDriver struct {
|
|
|
|
RootPath string
|
|
|
|
Perm
|
|
|
|
}
|
|
|
|
|
|
|
|
func (driver *FileDriver) realPath(path string) string {
|
|
|
|
paths := strings.Split(path, "/")
|
|
|
|
return filepath.Join(append([]string{driver.RootPath}, paths...)...)
|
|
|
|
}
|
|
|
|
|
2020-06-30 10:16:43 +02:00
|
|
|
// Stat implements Driver
|
2020-02-25 15:20:57 +01:00
|
|
|
func (driver *FileDriver) Stat(path string) (FileInfo, error) {
|
|
|
|
basepath := driver.realPath(path)
|
|
|
|
rPath, err := filepath.Abs(basepath)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
f, err := os.Lstat(rPath)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
mode, err := driver.Perm.GetMode(path)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
if f.IsDir() {
|
|
|
|
mode |= os.ModeDir
|
|
|
|
}
|
|
|
|
owner, err := driver.Perm.GetOwner(path)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
group, err := driver.Perm.GetGroup(path)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
return &fileInfo{f, mode, owner, group}, nil
|
|
|
|
}
|
|
|
|
|
2020-06-30 10:16:43 +02:00
|
|
|
// ListDir implements Driver
|
2020-02-25 15:20:57 +01:00
|
|
|
func (driver *FileDriver) ListDir(path string, callback func(FileInfo) error) error {
|
|
|
|
basepath := driver.realPath(path)
|
|
|
|
return filepath.Walk(basepath, func(f string, info os.FileInfo, err error) error {
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
rPath, _ := filepath.Rel(basepath, f)
|
|
|
|
if rPath == info.Name() {
|
|
|
|
mode, err := driver.Perm.GetMode(rPath)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if info.IsDir() {
|
|
|
|
mode |= os.ModeDir
|
|
|
|
}
|
|
|
|
owner, err := driver.Perm.GetOwner(rPath)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
group, err := driver.Perm.GetGroup(rPath)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
err = callback(&fileInfo{info, mode, owner, group})
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if info.IsDir() {
|
|
|
|
return filepath.SkipDir
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2020-06-30 10:16:43 +02:00
|
|
|
// DeleteDir implements Driver
|
2020-02-25 15:20:57 +01:00
|
|
|
func (driver *FileDriver) DeleteDir(path string) error {
|
|
|
|
rPath := driver.realPath(path)
|
|
|
|
f, err := os.Lstat(rPath)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if f.IsDir() {
|
2020-06-27 16:45:12 +02:00
|
|
|
return os.RemoveAll(rPath)
|
2020-02-25 15:20:57 +01:00
|
|
|
}
|
|
|
|
return errors.New("Not a directory")
|
|
|
|
}
|
|
|
|
|
2020-06-30 10:16:43 +02:00
|
|
|
// DeleteFile implements Driver
|
2020-02-25 15:20:57 +01:00
|
|
|
func (driver *FileDriver) DeleteFile(path string) error {
|
|
|
|
rPath := driver.realPath(path)
|
|
|
|
f, err := os.Lstat(rPath)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if !f.IsDir() {
|
|
|
|
return os.Remove(rPath)
|
|
|
|
}
|
|
|
|
return errors.New("Not a file")
|
|
|
|
}
|
|
|
|
|
2020-06-30 10:16:43 +02:00
|
|
|
// Rename implements Driver
|
2020-02-25 15:20:57 +01:00
|
|
|
func (driver *FileDriver) Rename(fromPath string, toPath string) error {
|
|
|
|
oldPath := driver.realPath(fromPath)
|
|
|
|
newPath := driver.realPath(toPath)
|
|
|
|
return os.Rename(oldPath, newPath)
|
|
|
|
}
|
|
|
|
|
2020-06-30 10:16:43 +02:00
|
|
|
// MakeDir implements Driver
|
2020-02-25 15:20:57 +01:00
|
|
|
func (driver *FileDriver) MakeDir(path string) error {
|
|
|
|
rPath := driver.realPath(path)
|
|
|
|
return os.MkdirAll(rPath, os.ModePerm)
|
|
|
|
}
|
|
|
|
|
2020-06-30 10:16:43 +02:00
|
|
|
// GetFile implements Driver
|
2020-02-25 15:20:57 +01:00
|
|
|
func (driver *FileDriver) GetFile(path string, offset int64) (int64, io.ReadCloser, error) {
|
|
|
|
rPath := driver.realPath(path)
|
|
|
|
f, err := os.Open(rPath)
|
|
|
|
if err != nil {
|
|
|
|
return 0, nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
info, err := f.Stat()
|
|
|
|
if err != nil {
|
|
|
|
return 0, nil, err
|
|
|
|
}
|
|
|
|
|
2020-06-27 16:45:12 +02:00
|
|
|
f.Seek(offset, io.SeekStart)
|
2020-02-25 15:20:57 +01:00
|
|
|
|
2020-06-27 16:45:12 +02:00
|
|
|
return info.Size() - offset, f, nil
|
2020-02-25 15:20:57 +01:00
|
|
|
}
|
|
|
|
|
2020-06-30 10:16:43 +02:00
|
|
|
// PutFile implements Driver
|
2020-02-25 15:20:57 +01:00
|
|
|
func (driver *FileDriver) PutFile(destPath string, data io.Reader, appendData bool) (int64, error) {
|
|
|
|
rPath := driver.realPath(destPath)
|
|
|
|
var isExist bool
|
|
|
|
f, err := os.Lstat(rPath)
|
|
|
|
if err == nil {
|
|
|
|
isExist = true
|
|
|
|
if f.IsDir() {
|
|
|
|
return 0, errors.New("A dir has the same name")
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if os.IsNotExist(err) {
|
|
|
|
isExist = false
|
|
|
|
} else {
|
|
|
|
return 0, errors.New(fmt.Sprintln("Put File error:", err))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if appendData && !isExist {
|
|
|
|
appendData = false
|
|
|
|
}
|
|
|
|
|
|
|
|
if !appendData {
|
|
|
|
if isExist {
|
|
|
|
err = os.Remove(rPath)
|
|
|
|
if err != nil {
|
|
|
|
return 0, err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
f, err := os.Create(rPath)
|
|
|
|
if err != nil {
|
|
|
|
return 0, err
|
|
|
|
}
|
|
|
|
defer f.Close()
|
|
|
|
bytes, err := io.Copy(f, data)
|
|
|
|
if err != nil {
|
|
|
|
return 0, err
|
|
|
|
}
|
|
|
|
return bytes, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
of, err := os.OpenFile(rPath, os.O_APPEND|os.O_RDWR, 0660)
|
|
|
|
if err != nil {
|
|
|
|
return 0, err
|
|
|
|
}
|
|
|
|
defer of.Close()
|
|
|
|
|
|
|
|
_, err = of.Seek(0, os.SEEK_END)
|
|
|
|
if err != nil {
|
|
|
|
return 0, err
|
|
|
|
}
|
|
|
|
|
|
|
|
bytes, err := io.Copy(of, data)
|
|
|
|
if err != nil {
|
|
|
|
return 0, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return bytes, nil
|
|
|
|
}
|
|
|
|
|
2020-06-30 10:16:43 +02:00
|
|
|
// FileDriverFactory implements DriverFactory
|
2020-02-25 15:20:57 +01:00
|
|
|
type FileDriverFactory struct {
|
|
|
|
RootPath string
|
|
|
|
Perm
|
|
|
|
}
|
|
|
|
|
2020-06-30 10:16:43 +02:00
|
|
|
// NewDriver implements DriverFactory
|
2020-02-25 15:20:57 +01:00
|
|
|
func (factory *FileDriverFactory) NewDriver() (Driver, error) {
|
2020-06-30 10:16:43 +02:00
|
|
|
var err error
|
|
|
|
factory.RootPath, err = filepath.Abs(factory.RootPath)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2020-02-25 15:20:57 +01:00
|
|
|
return &FileDriver{factory.RootPath, factory.Perm}, nil
|
|
|
|
}
|