ftp: replace URL parser with url.URL

This commit is contained in:
Nick Craig-Wood 2017-05-03 16:07:40 +01:00
parent fce734662f
commit c13cff37ef
2 changed files with 27 additions and 83 deletions

View file

@ -2,11 +2,9 @@
package ftp package ftp
import ( import (
"fmt"
"io" "io"
"net/url"
"path/filepath" "path/filepath"
"regexp"
"strconv"
"strings" "strings"
"sync" "sync"
"time" "time"
@ -24,7 +22,7 @@ var globalMux = sync.Mutex{}
// Register with Fs // Register with Fs
func init() { func init() {
fs.Register(&fs.RegInfo{ fs.Register(&fs.RegInfo{
Name: "Ftp", Name: "ftp",
Description: "FTP interface", Description: "FTP interface",
NewFs: NewFs, NewFs: NewFs,
Options: []fs.Option{ Options: []fs.Option{
@ -34,9 +32,10 @@ func init() {
}, { }, {
Name: "password", Name: "password",
Help: "Password", Help: "Password",
IsPassword: true,
}, { }, {
Name: "url", Name: "url",
Help: "FTP url", Help: "FTP URL",
}, },
}, },
}) })
@ -48,7 +47,7 @@ type Fs struct {
root string // the path we are working on if any root string // the path we are working on if any
features *fs.Features // optional features features *fs.Features // optional features
c *ftp.ServerConn // the connection to the FTP server c *ftp.ServerConn // the connection to the FTP server
url URL url *url.URL
mu sync.Mutex mu sync.Mutex
} }
@ -67,23 +66,6 @@ type FileInfo struct {
IsDir bool IsDir bool
} }
// URL represents an FTP URL
type URL struct {
Scheme string
Host string
Port int
Path string
}
// ToDial converts the URL to Dial format
func (u *URL) ToDial() string {
return fmt.Sprintf("%s:%d", u.Host, u.Port)
}
func (u *URL) String() string {
return fmt.Sprintf("ftp://%s:%d/%s", u.Host, u.Port, u.Path)
}
// ------------------------------------------------------------ // ------------------------------------------------------------
// Name of this fs // Name of this fs
@ -98,7 +80,7 @@ func (f *Fs) Root() string {
// String returns a description of the FS // String returns a description of the FS
func (f *Fs) String() string { func (f *Fs) String() string {
return fmt.Sprintf("FTP Connection to %s", f.url.String()) return f.url.String()
} }
// Features returns the optional features of this Fs // Features returns the optional features of this Fs
@ -106,39 +88,37 @@ func (f *Fs) Features() *fs.Features {
return f.features return f.features
} }
// Parse a URL
func parseURL(url string) URL {
// This is *similar* to the RFC 3986 regexp but it matches the
// port independently from the host
r, _ := regexp.Compile("^(([^:/?#]+):)?(//([^/?#:]*))?(:([0-9]+))?([^?#]*)(\\?([^#]*))?(#(.*))?")
data := r.FindAllStringSubmatch(url, -1)
if data[0][5] == "" {
data[0][5] = "21"
}
port, _ := strconv.Atoi(data[0][5])
return URL{data[0][2], data[0][4], port, data[0][7]}
}
// Open a new connection to the FTP server. // Open a new connection to the FTP server.
func ftpConnection(name, root string) (*ftp.ServerConn, URL, error) { func ftpConnection(name, root string) (*ftp.ServerConn, *url.URL, error) {
url := fs.ConfigFileGet(name, "url") URL := fs.ConfigFileGet(name, "url")
user := fs.ConfigFileGet(name, "username") user := fs.ConfigFileGet(name, "username")
pass := fs.ConfigFileGet(name, "password") pass := fs.ConfigFileGet(name, "password")
u := parseURL(url) pass, err := fs.Reveal(pass)
if err != nil {
return nil, nil, errors.Wrap(err, "failed to decrypt password")
}
u, err := url.Parse(URL)
if err != nil {
return nil, nil, errors.Wrap(err, "open ftp connection url parse")
}
u.Path = filepath.Join(u.Path, root) u.Path = filepath.Join(u.Path, root)
fs.Debugf(nil, "New ftp Connection with name %s and url %s (path %s)", name, u.String(), u.Path) fs.Debugf(nil, "New ftp Connection with name %s and url %s (path %s)", name, u.String(), u.Path)
globalMux.Lock() globalMux.Lock()
defer globalMux.Unlock() defer globalMux.Unlock()
c, err := ftp.DialTimeout(u.ToDial(), 30*time.Second) dialAddr := u.Hostname()
if u.Port() != "" {
dialAddr += ":" + u.Port()
} else {
dialAddr += ":21"
}
c, err := ftp.DialTimeout(dialAddr, 30*time.Second)
if err != nil { if err != nil {
fs.Errorf(nil, "Error while Dialing %s: %s", u.ToDial(), err) fs.Errorf(nil, "Error while Dialing %s: %s", dialAddr, err)
return nil, u, err return nil, u, err
} }
err = c.Login(user, pass) err = c.Login(user, pass)
if err != nil { if err != nil {
fs.Errorf(nil, "Error while Logging in into %s: %s", u.ToDial(), err) fs.Errorf(nil, "Error while Logging in into %s: %s", dialAddr, err)
return nil, u, err return nil, u, err
} }
return c, u, nil return c, u, nil

View file

@ -1,36 +0,0 @@
package ftp
import "testing"
func TestParseUrlToDial(t *testing.T) {
for _, test := range []struct {
in string
want string
}{
{"ftp://foo.bar", "foo.bar:21"},
{"http://foo.bar", "foo.bar:21"},
{"ftp:/foo.bar:123", "foo.bar:123"},
} {
u := parseURL(test.in)
got := u.ToDial()
if got != test.want {
t.Logf("%q: want %q got %q", test.in, test.want, got)
}
}
}
func TestParseUrlPath(t *testing.T) {
for _, test := range []struct {
in string
want string
}{
{"ftp://foo.bar/", "/"},
{"ftp://foo.bar/debian", "/debian"},
{"ftp://foo.bar", "/"},
} {
u := parseURL(test.in)
if u.Path != test.want {
t.Logf("%q: want %q got %q", test.in, test.want, u.Path)
}
}
}