forked from TrueCloudLab/rclone
Fix URL encoding issues - fixes #1573
This fixes the confusion between paths which were URL encoded and paths which weren't. In particular it allows files to have % in the name.
This commit is contained in:
parent
21aca68680
commit
6d59887487
4 changed files with 43 additions and 31 deletions
64
http/http.go
64
http/http.go
|
@ -46,11 +46,12 @@ func init() {
|
||||||
|
|
||||||
// Fs stores the interface to the remote HTTP files
|
// Fs stores the interface to the remote HTTP files
|
||||||
type Fs struct {
|
type Fs struct {
|
||||||
name string
|
name string
|
||||||
root string
|
root string
|
||||||
features *fs.Features // optional features
|
features *fs.Features // optional features
|
||||||
endpoint *url.URL
|
endpoint *url.URL
|
||||||
httpClient *http.Client
|
endpointURL string // endpoint as a string
|
||||||
|
httpClient *http.Client
|
||||||
}
|
}
|
||||||
|
|
||||||
// Object is a remote object that has been stat'd (so it exists, but is not necessarily open for reading)
|
// Object is a remote object that has been stat'd (so it exists, but is not necessarily open for reading)
|
||||||
|
@ -63,6 +64,8 @@ type Object struct {
|
||||||
}
|
}
|
||||||
|
|
||||||
// Join a URL and a path returning a new URL
|
// Join a URL and a path returning a new URL
|
||||||
|
//
|
||||||
|
// path should be URL escaped
|
||||||
func urlJoin(base *url.URL, path string) (*url.URL, error) {
|
func urlJoin(base *url.URL, path string) (*url.URL, error) {
|
||||||
rel, err := url.Parse(path)
|
rel, err := url.Parse(path)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
@ -142,15 +145,19 @@ func NewFs(name, root string) (fs.Fs, error) {
|
||||||
}
|
}
|
||||||
|
|
||||||
f := &Fs{
|
f := &Fs{
|
||||||
name: name,
|
name: name,
|
||||||
root: root,
|
root: root,
|
||||||
httpClient: client,
|
httpClient: client,
|
||||||
endpoint: u,
|
endpoint: u,
|
||||||
|
endpointURL: u.String(),
|
||||||
}
|
}
|
||||||
f.features = (&fs.Features{}).Fill(f)
|
f.features = (&fs.Features{}).Fill(f)
|
||||||
if isFile {
|
if isFile {
|
||||||
return f, fs.ErrorIsFile
|
return f, fs.ErrorIsFile
|
||||||
}
|
}
|
||||||
|
if !strings.HasSuffix(f.endpointURL, "/") {
|
||||||
|
return nil, errors.New("internal error: url doesn't end with /")
|
||||||
|
}
|
||||||
return f, nil
|
return f, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -166,7 +173,7 @@ func (f *Fs) Root() string {
|
||||||
|
|
||||||
// String returns the URL for the filesystem
|
// String returns the URL for the filesystem
|
||||||
func (f *Fs) String() string {
|
func (f *Fs) String() string {
|
||||||
return f.endpoint.String()
|
return f.endpointURL
|
||||||
}
|
}
|
||||||
|
|
||||||
// Features returns the optional features of this Fs
|
// Features returns the optional features of this Fs
|
||||||
|
@ -192,6 +199,11 @@ func (f *Fs) NewObject(remote string) (fs.Object, error) {
|
||||||
return o, nil
|
return o, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Join's the remote onto the base URL
|
||||||
|
func (f *Fs) url(remote string) string {
|
||||||
|
return f.endpointURL + urlEscape(remote)
|
||||||
|
}
|
||||||
|
|
||||||
func parseInt64(s string) int64 {
|
func parseInt64(s string) int64 {
|
||||||
n, e := strconv.ParseInt(s, 10, 64)
|
n, e := strconv.ParseInt(s, 10, 64)
|
||||||
if e != nil {
|
if e != nil {
|
||||||
|
@ -263,14 +275,15 @@ func parse(base *url.URL, in io.Reader) (names []string, err error) {
|
||||||
|
|
||||||
// Read the directory passed in
|
// Read the directory passed in
|
||||||
func (f *Fs) readDir(dir string) (names []string, err error) {
|
func (f *Fs) readDir(dir string) (names []string, err error) {
|
||||||
u, err := urlJoin(f.endpoint, dir)
|
URL := f.url(dir)
|
||||||
|
u, err := url.Parse(URL)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, errors.Wrap(err, "failed to readDir")
|
return nil, errors.Wrap(err, "failed to readDir")
|
||||||
}
|
}
|
||||||
if !strings.HasSuffix(u.String(), "/") {
|
if !strings.HasSuffix(URL, "/") {
|
||||||
return nil, errors.Errorf("internal error: readDir URL %q didn't end in /", u.String())
|
return nil, errors.Errorf("internal error: readDir URL %q didn't end in /", URL)
|
||||||
}
|
}
|
||||||
res, err := f.httpClient.Get(u.String())
|
res, err := f.httpClient.Get(URL)
|
||||||
if err == nil && res.StatusCode == http.StatusNotFound {
|
if err == nil && res.StatusCode == http.StatusNotFound {
|
||||||
return nil, fs.ErrorDirNotFound
|
return nil, fs.ErrorDirNotFound
|
||||||
}
|
}
|
||||||
|
@ -323,6 +336,7 @@ func (f *Fs) List(dir string) (entries fs.DirEntries, err error) {
|
||||||
remote: remote,
|
remote: remote,
|
||||||
}
|
}
|
||||||
if err = file.stat(); err != nil {
|
if err = file.stat(); err != nil {
|
||||||
|
fs.Debugf(remote, "skipping because of error: %v", err)
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
entries = append(entries, file)
|
entries = append(entries, file)
|
||||||
|
@ -373,19 +387,15 @@ func (o *Object) ModTime() time.Time {
|
||||||
return o.modTime
|
return o.modTime
|
||||||
}
|
}
|
||||||
|
|
||||||
// path returns the native path of the object
|
// url returns the native url of the object
|
||||||
func (o *Object) path() string {
|
func (o *Object) url() string {
|
||||||
return path.Join(o.fs.root, o.remote)
|
return o.fs.url(o.remote)
|
||||||
}
|
}
|
||||||
|
|
||||||
// stat updates the info field in the Object
|
// stat updates the info field in the Object
|
||||||
func (o *Object) stat() error {
|
func (o *Object) stat() error {
|
||||||
url, err := urlJoin(o.fs.endpoint, o.remote)
|
url := o.url()
|
||||||
if err != nil {
|
res, err := o.fs.httpClient.Head(url)
|
||||||
return errors.Wrap(err, "failed to stat")
|
|
||||||
}
|
|
||||||
endpoint := url.String()
|
|
||||||
res, err := o.fs.httpClient.Head(endpoint)
|
|
||||||
err = statusError(res, err)
|
err = statusError(res, err)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return errors.Wrap(err, "failed to stat")
|
return errors.Wrap(err, "failed to stat")
|
||||||
|
@ -414,12 +424,8 @@ func (o *Object) Storable() bool {
|
||||||
|
|
||||||
// Open a remote http file object for reading. Seek is supported
|
// Open a remote http file object for reading. Seek is supported
|
||||||
func (o *Object) Open(options ...fs.OpenOption) (in io.ReadCloser, err error) {
|
func (o *Object) Open(options ...fs.OpenOption) (in io.ReadCloser, err error) {
|
||||||
url, err := urlJoin(o.fs.endpoint, o.remote)
|
url := o.url()
|
||||||
if err != nil {
|
req, err := http.NewRequest("GET", url, nil)
|
||||||
return nil, errors.Wrap(err, "Open failed")
|
|
||||||
}
|
|
||||||
endpoint := url.String()
|
|
||||||
req, err := http.NewRequest("GET", endpoint, nil)
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, errors.Wrap(err, "Open failed")
|
return nil, errors.Wrap(err, "Open failed")
|
||||||
}
|
}
|
||||||
|
|
|
@ -72,7 +72,7 @@ func testListRoot(t *testing.T, f fs.Fs) {
|
||||||
assert.True(t, ok)
|
assert.True(t, ok)
|
||||||
|
|
||||||
e = entries[1]
|
e = entries[1]
|
||||||
assert.Equal(t, "one.txt", e.Remote())
|
assert.Equal(t, "one%.txt", e.Remote())
|
||||||
assert.Equal(t, int64(6), e.Size())
|
assert.Equal(t, int64(6), e.Size())
|
||||||
_, ok = e.(*Object)
|
_, ok = e.(*Object)
|
||||||
assert.True(t, ok)
|
assert.True(t, ok)
|
||||||
|
@ -176,7 +176,7 @@ func TestIsAFileRoot(t *testing.T) {
|
||||||
tidy := prepareServer(t)
|
tidy := prepareServer(t)
|
||||||
defer tidy()
|
defer tidy()
|
||||||
|
|
||||||
f, err := NewFs(remoteName, "one.txt")
|
f, err := NewFs(remoteName, "one%.txt")
|
||||||
assert.Equal(t, err, fs.ErrorIsFile)
|
assert.Equal(t, err, fs.ErrorIsFile)
|
||||||
|
|
||||||
testListRoot(t, f)
|
testListRoot(t, f)
|
||||||
|
@ -323,6 +323,8 @@ func TestParseApache(t *testing.T) {
|
||||||
"stressdisk/",
|
"stressdisk/",
|
||||||
"timer-test",
|
"timer-test",
|
||||||
"words-to-regexp.pl",
|
"words-to-regexp.pl",
|
||||||
|
"Now 100% better.mp3",
|
||||||
|
"Now better.mp3",
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -24,5 +24,9 @@
|
||||||
<tr><td valign="top"><img src="/icons/unknown.gif" alt="[ ]"></td><td><a href="timer-test">timer-test</a></td><td align="right">09-May-2017 17:05 </td><td align="right">1.5M</td><td> </td></tr>
|
<tr><td valign="top"><img src="/icons/unknown.gif" alt="[ ]"></td><td><a href="timer-test">timer-test</a></td><td align="right">09-May-2017 17:05 </td><td align="right">1.5M</td><td> </td></tr>
|
||||||
<tr><td valign="top"><img src="/icons/text.gif" alt="[TXT]"></td><td><a href="words-to-regexp.pl">words-to-regexp.pl</a></td><td align="right">01-Mar-2005 20:43 </td><td align="right">6.0K</td><td> </td></tr>
|
<tr><td valign="top"><img src="/icons/text.gif" alt="[TXT]"></td><td><a href="words-to-regexp.pl">words-to-regexp.pl</a></td><td align="right">01-Mar-2005 20:43 </td><td align="right">6.0K</td><td> </td></tr>
|
||||||
<tr><th colspan="5"><hr></th></tr>
|
<tr><th colspan="5"><hr></th></tr>
|
||||||
|
<!-- some extras from https://github.com/ncw/rclone/issues/1573 -->
|
||||||
|
<tr><td valign="top"><img src="/icons/sound2.gif" alt="[SND]"></td><td><a href="Now%20100%25%20better.mp3">Now 100% better.mp3</a></td><td align="right">2017-08-01 11:41 </td><td align="right"> 0 </td><td> </td></tr>
|
||||||
|
<tr><td valign="top"><img src="/icons/sound2.gif" alt="[SND]"></td><td><a href="Now%20better.mp3">Now better.mp3</a></td><td align="right">2017-08-01 11:41 </td><td align="right"> 0 </td><td> </td></tr>
|
||||||
|
|
||||||
</table>
|
</table>
|
||||||
</body></html>
|
</body></html>
|
||||||
|
|
Loading…
Reference in a new issue