diff --git a/cmd/cmd.go b/cmd/cmd.go index 760908b90..82a01c8b3 100644 --- a/cmd/cmd.go +++ b/cmd/cmd.go @@ -40,7 +40,7 @@ var Root = &cobra.Command{ Short: "Sync files and directories to and from local and remote object stores - " + fs.Version, Long: ` Rclone is a command line program to sync files and directories to and -from various cloud storage systems, such as: +from various cloud storage systems and using file transfer services, such as: * Google Drive * Amazon S3 @@ -52,6 +52,8 @@ from various cloud storage systems, such as: * Hubic * Backblaze B2 * Yandex Disk + * SFTP + * FTP * The local filesystem Features diff --git a/docs/content/ftp.md b/docs/content/ftp.md index 7a32439d6..6ea0f0a8e 100644 --- a/docs/content/ftp.md +++ b/docs/content/ftp.md @@ -7,7 +7,7 @@ date: "2017-01-01" FTP ------------------------------ -FTP is the File Transfer Protocl. FTP support is provided using the +FTP is the File Transfer Protocol. FTP support is provided using the [github.com/jlaffaye/ftp](https://godoc.org/github.com/jlaffaye/ftp) package. @@ -15,12 +15,10 @@ Here is an example of making an FTP configuration. First run rclone config -This will guide you through an interactive setup process. An FTP -backend only needs an URL and username and password. With -anonymous FTP server, you will need to use `anonymous` as username and -your email address as the password. If you want to use a non-standard FTP -port, please specify this with a colon and the port number at the end of -the URL. For example `ftp://ftp.mirrorservice.org:5555` +This will guide you through an interactive setup process. An FTP remote only +needs a host together with and a username and a password. With anonymous FTP +server, you will need to use `anonymous` as username and your email address as +the password. ``` No remotes found - make a new one @@ -43,7 +41,7 @@ Choose a number from below, or type in your own value \ "dropbox" 5 / Encrypt/Decrypt a remote \ "crypt" - 6 / FTP interface + 6 / FTP Connection \ "ftp" 7 / Google Cloud Storage (this is not Google Drive) \ "google cloud storage" @@ -62,9 +60,16 @@ Choose a number from below, or type in your own value 14 / Yandex Disk \ "yandex" Storage> ftp -Username -username> anonymous -Password +FTP host to connect to +Choose a number from below, or type in your own value + 1 / Connect to ftp.example.com + \ "ftp.example.com" +host> ftp.example.com +FTP username, leave blank for current username, ncw +user> +FTP port, leave blank to use default (21) +port> +FTP password y) Yes type in my own password g) Generate random password y/g> y @@ -72,14 +77,13 @@ Enter the password: password: Confirm the password: password: -FTP URL -url> ftp://ftp.mirrorservice.org/ Remote config -------------------- [remote] -username = anonymous -password = *** ENCRYPTED *** -url = ftp://ftp.mirrorservice.org/ +host = ftp.example.com +user = +port = +pass = *** ENCRYPTED *** -------------------- y) Yes this is OK e) Edit this remote diff --git a/docs/content/sftp.md b/docs/content/sftp.md index 1aa082345..f2552ab98 100644 --- a/docs/content/sftp.md +++ b/docs/content/sftp.md @@ -45,23 +45,25 @@ Choose a number from below, or type in your own value \ "dropbox" 5 / Encrypt/Decrypt a remote \ "crypt" - 6 / Google Cloud Storage (this is not Google Drive) + 6 / FTP Connection + \ "ftp" + 7 / Google Cloud Storage (this is not Google Drive) \ "google cloud storage" - 7 / Google Drive + 8 / Google Drive \ "drive" - 8 / Hubic + 9 / Hubic \ "hubic" - 9 / Local Disk +10 / Local Disk \ "local" -10 / Microsoft OneDrive +11 / Microsoft OneDrive \ "onedrive" -11 / Openstack Swift (Rackspace Cloud Files, Memset Memstore, OVH) +12 / Openstack Swift (Rackspace Cloud Files, Memset Memstore, OVH) \ "swift" -12 / SSH/SFTP Connection +13 / SSH/SFTP Connection \ "sftp" -13 / Yandex Disk +14 / Yandex Disk \ "yandex" -Storage> 12 +Storage> sftp SSH host to connect to Choose a number from below, or type in your own value 1 / Connect to example.com @@ -69,7 +71,7 @@ Choose a number from below, or type in your own value host> example.com SSH username, leave blank for current username, ncw user> -SSH port +SSH port, leave blank to use default (22) port> SSH password, leave blank to use ssh-agent y) Yes type in my own password diff --git a/ftp/ftp.go b/ftp/ftp.go index bd40a0665..5e183518e 100644 --- a/ftp/ftp.go +++ b/ftp/ftp.go @@ -5,8 +5,8 @@ import ( "io" "net/textproto" "net/url" + "os" "path" - "strings" "sync" "time" @@ -19,19 +19,30 @@ import ( func init() { fs.Register(&fs.RegInfo{ Name: "ftp", - Description: "FTP interface", + Description: "FTP Connection", NewFs: NewFs, Options: []fs.Option{ { - Name: "username", - Help: "Username", + Name: "host", + Help: "FTP host to connect to", + Optional: false, + Examples: []fs.OptionExample{{ + Value: "ftp.example.com", + Help: "Connect to ftp.example.com", + }}, }, { - Name: "password", - Help: "Password", + Name: "user", + Help: "FTP username, leave blank for current username, " + os.Getenv("USER"), + Optional: true, + }, { + Name: "port", + Help: "FTP port, leave blank to use default (21) ", + Optional: true, + }, { + Name: "pass", + Help: "FTP password", IsPassword: true, - }, { - Name: "url", - Help: "FTP URL", + Optional: false, }, }, }) @@ -147,27 +158,27 @@ func (f *Fs) putFtpConnection(pc **ftp.ServerConn, err error) { // 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) - URL := fs.ConfigFileGet(name, "url") - user := fs.ConfigFileGet(name, "username") - pass := fs.ConfigFileGet(name, "password") + host := fs.ConfigFileGet(name, "host") + user := fs.ConfigFileGet(name, "user") + pass := fs.ConfigFileGet(name, "pass") + port := fs.ConfigFileGet(name, "port") pass, err = fs.Reveal(pass) if err != nil { return nil, errors.Wrap(err, "NewFS decrypt password") } - u, err := url.Parse(URL) + if user == "" { + user = os.Getenv("USER") + } + if port == "" { + port = "21" + } + + dialAddr := host + ":" + port + u, err := url.Parse("ftp://" + dialAddr) if err != nil { return nil, errors.Wrap(err, "NewFS URL parse") } - urlPath := strings.Trim(u.Path, "/") - fullPath := root - if urlPath != "" && !strings.HasPrefix("/", root) { - fullPath = path.Join(u.Path, root) - } - root = fullPath - dialAddr := u.Host - if strings.IndexRune(dialAddr, ':') < 0 { - dialAddr += ":21" - } + f := &Fs{ name: name, root: root, diff --git a/sftp/sftp.go b/sftp/sftp.go index 23ba7eadb..d52878d37 100644 --- a/sftp/sftp.go +++ b/sftp/sftp.go @@ -38,7 +38,7 @@ func init() { Optional: true, }, { Name: "port", - Help: "SSH port", + Help: "SSH port, leave blank to use default (22)", Optional: true, }, { Name: "pass",