mirror of
https://github.com/rclone/rclone.git
synced 2025-01-12 17:28:46 +01:00
http: fix panic with url encoded content - fixes #1565
This fixes the issue which caused the panic (carrying on after an error) and the issue which caused the error (double unescaping the URL).
This commit is contained in:
parent
c3cd247d4b
commit
b0fd187cba
28
http/http.go
28
http/http.go
@ -63,12 +63,12 @@ type Object struct {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Join a URL and a path returning a new URL
|
// Join a URL and a path returning a new URL
|
||||||
func urlJoin(base *url.URL, path string) *url.URL {
|
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 {
|
||||||
fs.Errorf(nil, "Error parsing %q as URL: %v", path, err)
|
return nil, errors.Wrapf(err, "Error parsing %q as URL", path)
|
||||||
}
|
}
|
||||||
return base.ResolveReference(rel)
|
return base.ResolveReference(rel), nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// statusError returns an error if the res contained an error
|
// statusError returns an error if the res contained an error
|
||||||
@ -195,12 +195,11 @@ func parseInt64(s string) int64 {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// parseName turns a name as found in the page into a remote path or returns false
|
// parseName turns a name as found in the page into a remote path or returns false
|
||||||
func parseName(base *url.URL, val string) (string, bool) {
|
func parseName(base *url.URL, name string) (string, bool) {
|
||||||
name, err := url.QueryUnescape(val)
|
u, err := urlJoin(base, name)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return "", false
|
return "", false
|
||||||
}
|
}
|
||||||
u := urlJoin(base, name)
|
|
||||||
uStr := u.String()
|
uStr := u.String()
|
||||||
if strings.Index(uStr, "?") >= 0 {
|
if strings.Index(uStr, "?") >= 0 {
|
||||||
return "", false
|
return "", false
|
||||||
@ -258,7 +257,10 @@ 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 := urlJoin(f.endpoint, dir)
|
u, err := urlJoin(f.endpoint, dir)
|
||||||
|
if err != nil {
|
||||||
|
return nil, errors.Wrap(err, "failed to readDir")
|
||||||
|
}
|
||||||
if !strings.HasSuffix(u.String(), "/") {
|
if !strings.HasSuffix(u.String(), "/") {
|
||||||
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 /", u.String())
|
||||||
}
|
}
|
||||||
@ -372,7 +374,11 @@ func (o *Object) path() string {
|
|||||||
|
|
||||||
// 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 {
|
||||||
endpoint := urlJoin(o.fs.endpoint, o.remote).String()
|
url, err := urlJoin(o.fs.endpoint, o.remote)
|
||||||
|
if err != nil {
|
||||||
|
return errors.Wrap(err, "failed to stat")
|
||||||
|
}
|
||||||
|
endpoint := url.String()
|
||||||
res, err := o.fs.httpClient.Head(endpoint)
|
res, err := o.fs.httpClient.Head(endpoint)
|
||||||
err = statusError(res, err)
|
err = statusError(res, err)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
@ -402,7 +408,11 @@ 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) {
|
||||||
endpoint := urlJoin(o.fs.endpoint, o.remote).String()
|
url, err := urlJoin(o.fs.endpoint, o.remote)
|
||||||
|
if err != nil {
|
||||||
|
return nil, errors.Wrap(err, "Open failed")
|
||||||
|
}
|
||||||
|
endpoint := url.String()
|
||||||
req, err := http.NewRequest("GET", endpoint, nil)
|
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")
|
||||||
|
@ -220,6 +220,7 @@ func TestParseName(t *testing.T) {
|
|||||||
{"http://example.com/dir/", "/dir/", false, ""},
|
{"http://example.com/dir/", "/dir/", false, ""},
|
||||||
{"http://example.com/dir/", "/dir/potato", true, "potato"},
|
{"http://example.com/dir/", "/dir/potato", true, "potato"},
|
||||||
{"http://example.com/dir/", "subdir/potato", false, ""},
|
{"http://example.com/dir/", "subdir/potato", false, ""},
|
||||||
|
{"http://example.com/dir/", "With percent %25.txt", true, "With percent %.txt"},
|
||||||
} {
|
} {
|
||||||
u, err := url.Parse(test.base)
|
u, err := url.Parse(test.base)
|
||||||
require.NoError(t, err)
|
require.NoError(t, err)
|
||||||
|
Loading…
Reference in New Issue
Block a user