rclone/ftp/ftp.go

585 lines
13 KiB
Go
Raw Normal View History

// Package ftp interfaces with FTP servers
2016-12-27 20:52:30 +00:00
package ftp
// FIXME Mover and DirMover are possible using c.Rename
2016-12-27 20:52:30 +00:00
import (
"io"
"net/textproto"
2017-05-03 15:07:40 +00:00
"net/url"
2017-05-03 15:42:04 +00:00
"path"
2016-12-27 20:52:30 +00:00
"strings"
2017-05-03 12:18:17 +00:00
"sync"
"time"
"github.com/jlaffaye/ftp"
"github.com/ncw/rclone/fs"
"github.com/pkg/errors"
2016-12-27 20:52:30 +00:00
)
// Register with Fs
func init() {
fs.Register(&fs.RegInfo{
2017-05-03 15:07:40 +00:00
Name: "ftp",
2016-12-27 20:52:30 +00:00
Description: "FTP interface",
2017-05-03 12:18:17 +00:00
NewFs: NewFs,
2016-12-27 20:52:30 +00:00
Options: []fs.Option{
{
Name: "username",
Help: "Username",
}, {
2017-05-03 15:07:40 +00:00
Name: "password",
Help: "Password",
IsPassword: true,
2016-12-27 20:52:30 +00:00
}, {
Name: "url",
2017-05-03 15:07:40 +00:00
Help: "FTP URL",
2016-12-27 20:52:30 +00:00
},
},
})
}
// Fs represents a remote FTP server
2016-12-27 20:52:30 +00:00
type Fs struct {
name string // name of this remote
root string // the path we are working on if any
features *fs.Features // optional features
2017-05-03 15:07:40 +00:00
url *url.URL
user string
pass string
dialAddr string
poolMu sync.Mutex
pool []*ftp.ServerConn
2016-12-27 20:52:30 +00:00
}
// Object describes an FTP file
2016-12-27 20:52:30 +00:00
type Object struct {
2017-05-03 12:18:17 +00:00
fs *Fs
remote string
info *FileInfo
2016-12-27 20:52:30 +00:00
}
// FileInfo is the metadata known about an FTP file
2016-12-27 20:52:30 +00:00
type FileInfo struct {
Name string
Size uint64
ModTime time.Time
IsDir bool
}
// ------------------------------------------------------------
2016-12-27 20:52:30 +00:00
// Name of this fs
2016-12-27 20:52:30 +00:00
func (f *Fs) Name() string {
2017-05-03 12:18:17 +00:00
return f.name
2016-12-27 20:52:30 +00:00
}
// Root of the remote (as passed into NewFs)
func (f *Fs) Root() string {
return f.root
}
// String returns a description of the FS
2016-12-27 20:52:30 +00:00
func (f *Fs) String() string {
2017-05-03 15:07:40 +00:00
return f.url.String()
2016-12-27 20:52:30 +00:00
}
2017-05-03 12:18:17 +00:00
// Features returns the optional features of this Fs
func (f *Fs) Features() *fs.Features {
return f.features
}
// Open a new connection to the FTP server.
func (f *Fs) ftpConnection() (*ftp.ServerConn, error) {
fs.Debugf(f, "Connecting to FTP server")
2017-05-18 15:20:26 +00:00
c, err := ftp.DialTimeout(f.dialAddr, fs.Config.ConnectTimeout)
if err != nil {
fs.Errorf(f, "Error while Dialing %s: %s", f.dialAddr, err)
return nil, errors.Wrap(err, "ftpConnection Dial")
}
err = c.Login(f.user, f.pass)
if err != nil {
_ = c.Quit()
fs.Errorf(f, "Error while Logging in into %s: %s", f.dialAddr, err)
return nil, errors.Wrap(err, "ftpConnection Login")
}
return c, nil
}
// Get an FTP connection from the pool, or open a new one
func (f *Fs) getFtpConnection() (c *ftp.ServerConn, err error) {
f.poolMu.Lock()
if len(f.pool) > 0 {
c = f.pool[0]
f.pool = f.pool[1:]
}
f.poolMu.Unlock()
if c != nil {
return c, nil
}
return f.ftpConnection()
}
// Return an FTP connection to the pool
//
// It nils the pointed to connection out so it can't be reused
func (f *Fs) putFtpConnection(c **ftp.ServerConn) {
f.poolMu.Lock()
f.pool = append(f.pool, *c)
f.poolMu.Unlock()
}
// NewFs contstructs an Fs from the path, container:path
func NewFs(name, root string) (ff fs.Fs, err error) {
// defer fs.Trace(nil, "name=%q, root=%q", name, root)("fs=%v, err=%v", &ff, &err)
2017-05-03 15:07:40 +00:00
URL := fs.ConfigFileGet(name, "url")
user := fs.ConfigFileGet(name, "username")
pass := fs.ConfigFileGet(name, "password")
pass, err = fs.Reveal(pass)
2017-05-03 15:07:40 +00:00
if err != nil {
return nil, errors.Wrap(err, "NewFS decrypt password")
2017-05-03 15:07:40 +00:00
}
u, err := url.Parse(URL)
if err != nil {
return nil, errors.Wrap(err, "NewFS URL parse")
2017-05-03 15:07:40 +00:00
}
urlPath := strings.Trim(u.Path, "/")
fullPath := root
if urlPath != "" && !strings.HasPrefix("/", root) {
fullPath = path.Join(u.Path, root)
}
root = fullPath
2017-05-16 19:19:03 +00:00
dialAddr := u.Host
if strings.IndexRune(dialAddr, ':') < 0 {
2017-05-03 15:07:40 +00:00
dialAddr += ":21"
}
f := &Fs{
name: name,
root: root,
url: u,
user: user,
pass: pass,
dialAddr: dialAddr,
2016-12-27 20:52:30 +00:00
}
f.features = (&fs.Features{}).Fill(f)
// Make a connection and pool it to return errors early
c, err := f.getFtpConnection()
if err != nil {
return nil, errors.Wrap(err, "NewFs")
}
f.putFtpConnection(&c)
if root != "" {
// Check to see if the root actually an existing file
remote := path.Base(root)
f.root = path.Dir(root)
if f.root == "." {
f.root = ""
}
_, err := f.NewObject(remote)
if err != nil {
if err == fs.ErrorObjectNotFound || errors.Cause(err) == fs.ErrorNotAFile {
// File doesn't exist so return old f
f.root = root
return f, nil
}
return nil, err
}
// return an error with an fs which points to the parent
return f, fs.ErrorIsFile
}
return f, err
2016-12-27 20:52:30 +00:00
}
// translateErrorFile turns FTP errors into rclone errors if possible for a file
func translateErrorFile(err error) error {
switch errX := err.(type) {
case *textproto.Error:
switch errX.Code {
case ftp.StatusFileUnavailable:
err = fs.ErrorObjectNotFound
}
2016-12-27 20:52:30 +00:00
}
return err
}
// translateErrorDir turns FTP errors into rclone errors if possible for a directory
func translateErrorDir(err error) error {
switch errX := err.(type) {
case *textproto.Error:
switch errX.Code {
case ftp.StatusFileUnavailable:
err = fs.ErrorDirNotFound
}
}
return err
2016-12-27 20:52:30 +00:00
}
// NewObject finds the Object at remote. If it can't be found
// it returns the error fs.ErrorObjectNotFound.
func (f *Fs) NewObject(remote string) (o fs.Object, err error) {
// defer fs.Trace(remote, "")("o=%v, err=%v", &o, &err)
fullPath := path.Join(f.root, remote)
dir := path.Dir(fullPath)
base := path.Base(fullPath)
2016-12-27 20:52:30 +00:00
c, err := f.getFtpConnection()
if err != nil {
return nil, errors.Wrap(err, "NewObject")
}
files, err := c.List(dir)
f.putFtpConnection(&c)
2017-05-03 15:42:04 +00:00
if err != nil {
return nil, translateErrorFile(err)
2017-05-03 15:42:04 +00:00
}
for i, file := range files {
if file.Type != ftp.EntryTypeFolder && file.Name == base {
2016-12-27 20:52:30 +00:00
o := &Object{
fs: f,
remote: remote,
}
info := &FileInfo{
2017-05-03 12:18:17 +00:00
Name: remote,
Size: files[i].Size,
2016-12-27 20:52:30 +00:00
ModTime: files[i].Time,
}
o.info = info
2017-05-03 12:18:17 +00:00
2016-12-27 20:52:30 +00:00
return o, nil
}
}
return nil, fs.ErrorObjectNotFound
}
2017-05-03 12:18:17 +00:00
func (f *Fs) list(out fs.ListOpts, dir string, curlevel int) {
// defer fs.Trace(dir, "curlevel=%d", curlevel)("")
c, err := f.getFtpConnection()
if err != nil {
out.SetError(errors.Wrap(err, "list"))
return
}
files, err := c.List(path.Join(f.root, dir))
f.putFtpConnection(&c)
2017-05-03 15:42:04 +00:00
if err != nil {
out.SetError(translateErrorDir(err))
2017-05-03 15:42:04 +00:00
return
}
2017-05-03 12:18:17 +00:00
for i := range files {
2016-12-27 20:52:30 +00:00
object := files[i]
2017-05-03 15:42:04 +00:00
newremote := path.Join(dir, object.Name)
2016-12-27 20:52:30 +00:00
switch object.Type {
case ftp.EntryTypeFolder:
if object.Name == "." || object.Name == ".." {
continue
}
2017-05-03 12:18:17 +00:00
if out.IncludeDirectory(newremote) {
2016-12-27 20:52:30 +00:00
d := &fs.Dir{
Name: newremote,
When: object.Time,
Bytes: 0,
Count: -1,
}
2017-05-03 12:18:17 +00:00
if curlevel < out.Level() {
2017-05-03 15:42:04 +00:00
f.list(out, path.Join(dir, object.Name), curlevel+1)
2016-12-27 20:52:30 +00:00
}
if out.AddDir(d) {
return
}
}
default:
o := &Object{
fs: f,
remote: newremote,
}
info := &FileInfo{
2017-05-03 12:18:17 +00:00
Name: newremote,
Size: object.Size,
2016-12-27 20:52:30 +00:00
ModTime: object.Time,
}
o.info = info
if out.Add(o) {
return
}
}
}
}
// List the objects and directories of the Fs starting from dir
//
// dir should be "" to start from the root, and should not
// have trailing slashes.
//
// This should return ErrDirNotFound (using out.SetError())
// if the directory isn't found.
//
// Fses must support recursion levels of fs.MaxLevel and 1.
// They may return ErrorLevelNotSupported otherwise.
2016-12-27 20:52:30 +00:00
func (f *Fs) List(out fs.ListOpts, dir string) {
// defer fs.Trace(dir, "")("")
2016-12-27 20:52:30 +00:00
f.list(out, dir, 1)
out.Finished()
}
// Hashes are not supported
func (f *Fs) Hashes() fs.HashSet {
return 0
}
2016-12-27 20:52:30 +00:00
// Precision shows Modified Time not supported
func (f *Fs) Precision() time.Duration {
return fs.ModTimeNotSupported
2016-12-27 20:52:30 +00:00
}
// Put in to the remote path with the modTime given of the given size
//
// May create the object even if it returns an error - if so
// will return the object and the error, otherwise will return
// nil and the error
func (f *Fs) Put(in io.Reader, src fs.ObjectInfo) (fs.Object, error) {
// fs.Debugf(f, "Trying to put file %s", src.Remote())
2017-05-18 16:23:02 +00:00
err := f.mkParentDir(src.Remote())
if err != nil {
return nil, errors.Wrap(err, "Put mkParentDir failed")
}
o := &Object{
fs: f,
remote: src.Remote(),
2016-12-27 20:52:30 +00:00
}
2017-05-18 16:23:02 +00:00
err = o.Update(in, src)
return o, err
2016-12-27 20:52:30 +00:00
}
// getInfo reads the FileInfo for a path
func (f *Fs) getInfo(remote string) (fi *FileInfo, err error) {
// defer fs.Trace(remote, "")("fi=%v, err=%v", &fi, &err)
2017-05-03 15:42:04 +00:00
dir := path.Dir(remote)
base := path.Base(remote)
c, err := f.getFtpConnection()
if err != nil {
return nil, errors.Wrap(err, "getInfo")
}
files, err := c.List(dir)
f.putFtpConnection(&c)
2017-05-03 15:42:04 +00:00
if err != nil {
return nil, translateErrorFile(err)
2017-05-03 15:42:04 +00:00
}
for i := range files {
if files[i].Name == base {
info := &FileInfo{
Name: remote,
Size: files[i].Size,
ModTime: files[i].Time,
IsDir: files[i].Type == ftp.EntryTypeFolder,
}
return info, nil
}
}
return nil, fs.ErrorObjectNotFound
2016-12-27 20:52:30 +00:00
}
2017-05-18 16:23:02 +00:00
// mkdir makes the directory and parents using unrooted paths
func (f *Fs) mkdir(abspath string) error {
2017-05-18 16:23:02 +00:00
if abspath == "." || abspath == "/" {
return nil
}
fi, err := f.getInfo(abspath)
if err == nil {
if fi.IsDir {
return nil
}
2017-05-18 16:23:02 +00:00
return fs.ErrorIsFile
} else if err != fs.ErrorObjectNotFound {
return errors.Wrapf(err, "mkdir %q failed", abspath)
}
parent := path.Dir(abspath)
err = f.mkdir(parent)
if err != nil {
return err
}
c, connErr := f.getFtpConnection()
if connErr != nil {
return errors.Wrap(connErr, "mkdir")
2016-12-27 20:52:30 +00:00
}
2017-05-18 16:23:02 +00:00
err = c.MakeDir(abspath)
f.putFtpConnection(&c)
2016-12-27 20:52:30 +00:00
return err
}
2017-05-18 16:23:02 +00:00
// mkParentDir makes the parent of remote if necessary and any
// directories above that
func (f *Fs) mkParentDir(remote string) error {
parent := path.Dir(remote)
return f.mkdir(path.Join(f.root, parent))
}
2017-05-03 15:42:04 +00:00
// Mkdir creates the directory if it doesn't exist
func (f *Fs) Mkdir(dir string) (err error) {
// defer fs.Trace(dir, "")("err=%v", &err)
2017-05-18 16:23:02 +00:00
root := path.Join(f.root, dir)
return f.mkdir(root)
2016-12-27 20:52:30 +00:00
}
// Rmdir removes the directory (container, bucket) if empty
//
// Return an error if it doesn't exist or isn't empty
func (f *Fs) Rmdir(dir string) error {
c, err := f.getFtpConnection()
if err != nil {
2017-05-18 14:52:28 +00:00
return errors.Wrap(translateErrorFile(err), "Rmdir")
}
err = c.RemoveDir(path.Join(f.root, dir))
f.putFtpConnection(&c)
return translateErrorDir(err)
}
// ------------------------------------------------------------
// Fs returns the parent Fs
2016-12-27 20:52:30 +00:00
func (o *Object) Fs() fs.Info {
return o.fs
}
// String version of o
func (o *Object) String() string {
if o == nil {
return "<nil>"
}
return o.remote
2016-12-27 20:52:30 +00:00
}
// Remote returns the remote path
func (o *Object) Remote() string {
return o.remote
}
// Hash returns the hash of an object returning a lowercase hex string
func (o *Object) Hash(t fs.HashType) (string, error) {
return "", fs.ErrHashUnsupported
}
// Size returns the size of an object in bytes
2016-12-27 20:52:30 +00:00
func (o *Object) Size() int64 {
return int64(o.info.Size)
}
// ModTime returns the modification time of the object
func (o *Object) ModTime() time.Time {
return o.info.ModTime
}
// SetModTime sets the modification time of the object
func (o *Object) SetModTime(modTime time.Time) error {
return nil
}
// Storable returns a boolean as to whether this object is storable
2016-12-27 20:52:30 +00:00
func (o *Object) Storable() bool {
return true
}
// ftpReadCloser implements io.ReadCloser for FTP objects.
type ftpReadCloser struct {
io.ReadCloser
c *ftp.ServerConn
f *Fs
}
// Close the FTP reader and return the connection to the pool
func (f *ftpReadCloser) Close() error {
err := f.ReadCloser.Close()
f.f.putFtpConnection(&f.c)
return err
2016-12-27 20:52:30 +00:00
}
// Open an object for read
func (o *Object) Open(options ...fs.OpenOption) (rc io.ReadCloser, err error) {
// defer fs.Trace(o, "")("rc=%v, err=%v", &rc, &err)
2017-05-03 15:42:04 +00:00
path := path.Join(o.fs.root, o.remote)
var offset int64
for _, option := range options {
switch x := option.(type) {
case *fs.SeekOption:
offset = x.Offset
default:
if option.Mandatory() {
fs.Logf(o, "Unsupported mandatory option: %v", option)
}
}
}
c, err := o.fs.getFtpConnection()
if err != nil {
return nil, errors.Wrap(err, "open")
}
fd, err := c.RetrFrom(path, uint64(offset))
if err != nil {
o.fs.putFtpConnection(&c)
return nil, errors.Wrap(err, "open")
}
rc = &ftpReadCloser{ReadCloser: fd, c: c, f: o.fs}
return rc, nil
}
// Update the already existing object
//
// Copy the reader into the object updating modTime and size
//
// The new object may have been created if an error is returned
func (o *Object) Update(in io.Reader, src fs.ObjectInfo) (err error) {
// defer fs.Trace(o, "src=%v", src)("err=%v", &err)
2017-05-03 15:42:04 +00:00
path := path.Join(o.fs.root, o.remote)
// remove the file if upload failed
remove := func() {
removeErr := o.Remove()
if removeErr != nil {
fs.Debugf(o, "Failed to remove: %v", removeErr)
} else {
fs.Debugf(o, "Removed after failed upload: %v", err)
}
2016-12-27 20:52:30 +00:00
}
c, err := o.fs.getFtpConnection()
if err != nil {
return errors.Wrap(err, "Update")
}
err = c.Stor(path, in)
2016-12-27 20:52:30 +00:00
if err != nil {
_ = c.Quit()
remove()
return errors.Wrap(err, "update stor")
2016-12-27 20:52:30 +00:00
}
o.fs.putFtpConnection(&c)
o.info, err = o.fs.getInfo(path)
2016-12-27 20:52:30 +00:00
if err != nil {
return errors.Wrap(err, "update getinfo")
2016-12-27 20:52:30 +00:00
}
return nil
2016-12-27 20:52:30 +00:00
}
// Remove an object
func (o *Object) Remove() (err error) {
// defer fs.Trace(o, "")("err=%v", &err)
2017-05-03 15:42:04 +00:00
path := path.Join(o.fs.root, o.remote)
// Check if it's a directory or a file
2017-05-03 15:42:04 +00:00
info, err := o.fs.getInfo(path)
if err != nil {
return err
}
if info.IsDir {
err = o.fs.Rmdir(o.remote)
} else {
c, err := o.fs.getFtpConnection()
if err != nil {
return errors.Wrap(err, "Remove")
}
err = c.Delete(path)
o.fs.putFtpConnection(&c)
2016-12-27 20:52:30 +00:00
}
return err
2016-12-27 20:52:30 +00:00
}
2017-05-03 12:18:17 +00:00
// Check the interfaces are satisfied
2016-12-27 20:52:30 +00:00
var (
2017-05-03 12:18:17 +00:00
_ fs.Fs = &Fs{}
_ fs.Object = &Object{}
2016-12-27 20:52:30 +00:00
)