From 560423fa8a0d5487c0d066e42a00e65196be7c68 Mon Sep 17 00:00:00 2001 From: Julien Laffaye Date: Wed, 17 Aug 2022 18:41:24 -0400 Subject: [PATCH 01/35] DialWithNetConn is a special case of DialWithDialFunc --- ftp.go | 35 +++++++++++++++++++---------------- 1 file changed, 19 insertions(+), 16 deletions(-) diff --git a/ftp.go b/ftp.go index abe944f..6e994ab 100644 --- a/ftp.go +++ b/ftp.go @@ -70,7 +70,6 @@ type dialOptions struct { dialer net.Dialer tlsConfig *tls.Config explicitTLS bool - conn net.Conn disableEPSV bool disableUTF8 bool disableMLSD bool @@ -108,14 +107,13 @@ func Dial(addr string, options ...DialOption) (*ServerConn, error) { do.location = time.UTC } - tconn := do.conn - if tconn == nil { - var err error + dialFunc := do.dialFunc - if do.dialFunc != nil { - tconn, err = do.dialFunc("tcp", addr) - } else if do.tlsConfig != nil && !do.explicitTLS { - tconn, err = tls.DialWithDialer(&do.dialer, "tcp", addr, do.tlsConfig) + if dialFunc == nil { + if do.tlsConfig != nil && !do.explicitTLS { + dialFunc = func(network, address string) (net.Conn, error) { + return tls.DialWithDialer(&do.dialer, network, addr, do.tlsConfig) + } } else { ctx := do.context @@ -123,12 +121,15 @@ func Dial(addr string, options ...DialOption) (*ServerConn, error) { ctx = context.Background() } - tconn, err = do.dialer.DialContext(ctx, "tcp", addr) + dialFunc = func(network, address string) (net.Conn, error) { + return do.dialer.DialContext(ctx, network, addr) + } } + } - if err != nil { - return nil, err - } + tconn, err := dialFunc("tcp", addr) + if err != nil { + return nil, err } // Use the resolved IP address in case addr contains a domain name @@ -143,7 +144,7 @@ func Dial(addr string, options ...DialOption) (*ServerConn, error) { host: remoteAddr.IP.String(), } - _, _, err := c.conn.ReadResponse(StatusReady) + _, _, err = c.conn.ReadResponse(StatusReady) if err != nil { _ = c.Quit() return nil, err @@ -185,10 +186,12 @@ func DialWithDialer(dialer net.Dialer) DialOption { } // DialWithNetConn returns a DialOption that configures the ServerConn with the underlying net.Conn +// +// Deprecated: Use [DialWithDialFunc] instead func DialWithNetConn(conn net.Conn) DialOption { - return DialOption{func(do *dialOptions) { - do.conn = conn - }} + return DialWithDialFunc(func(network, address string) (net.Conn, error) { + return conn, nil + }) } // DialWithDisabledEPSV returns a DialOption that configures the ServerConn with EPSV disabled From 45482d097e71da81197b9610d839b7c3d8141ef3 Mon Sep 17 00:00:00 2001 From: Julien Laffaye Date: Wed, 17 Aug 2022 19:24:40 -0400 Subject: [PATCH 02/35] Use assert package to simplify tests --- client_test.go | 147 ++++++++++++++----------------------------------- conn_test.go | 20 +++---- 2 files changed, 47 insertions(+), 120 deletions(-) diff --git a/client_test.go b/client_test.go index d23e5b5..63899a9 100644 --- a/client_test.go +++ b/client_test.go @@ -5,8 +5,6 @@ import ( "fmt" "io/ioutil" "net" - "net/textproto" - "strings" "syscall" "testing" "time" @@ -28,174 +26,111 @@ func TestConnEPSV(t *testing.T) { } func testConn(t *testing.T, disableEPSV bool) { + assert := assert.New(t) mock, c := openConn(t, "127.0.0.1", DialWithTimeout(5*time.Second), DialWithDisabledEPSV(disableEPSV)) err := c.Login("anonymous", "anonymous") - if err != nil { - t.Fatal(err) - } + assert.NoError(err) err = c.NoOp() - if err != nil { - t.Error(err) - } + assert.NoError(err) err = c.ChangeDir("incoming") - if err != nil { - t.Error(err) - } + assert.NoError(err) dir, err := c.CurrentDir() - if err != nil { - t.Error(err) - } else { - if dir != "/incoming" { - t.Error("Wrong dir: " + dir) - } + if assert.NoError(err) { + assert.Equal("/incoming", dir) } data := bytes.NewBufferString(testData) err = c.Stor("test", data) - if err != nil { - t.Error(err) - } + assert.NoError(err) _, err = c.List(".") - if err != nil { - t.Error(err) - } + assert.NoError(err) err = c.Rename("test", "tset") - if err != nil { - t.Error(err) - } + assert.NoError(err) // Read without deadline r, err := c.Retr("tset") - if err != nil { - t.Error(err) - } else { - buf, errRead := ioutil.ReadAll(r) - if err != nil { - t.Error(errRead) - } - if string(buf) != testData { - t.Errorf("'%s'", buf) + if assert.NoError(err) { + buf, err := ioutil.ReadAll(r) + if assert.NoError(err) { + assert.Equal(testData, string(buf)) } + r.Close() r.Close() // test we can close two times } // Read with deadline r, err = c.Retr("tset") - if err != nil { - t.Error(err) - } else { + if assert.NoError(err) { if err := r.SetDeadline(time.Now()); err != nil { t.Fatal(err) } _, err = ioutil.ReadAll(r) - if err == nil { - t.Error("deadline should have caused error") - } else if !strings.HasSuffix(err.Error(), "i/o timeout") { - t.Error(err) - } + assert.ErrorContains(err, "i/o timeout") r.Close() } // Read with offset r, err = c.RetrFrom("tset", 5) - if err != nil { - t.Error(err) - } else { - buf, errRead := ioutil.ReadAll(r) - if errRead != nil { - t.Error(errRead) - } - expected := testData[5:] - if string(buf) != expected { - t.Errorf("read %q, expected %q", buf, expected) + if assert.NoError(err) { + buf, err := ioutil.ReadAll(r) + if assert.NoError(err) { + expected := testData[5:] + assert.Equal(expected, string(buf)) } + r.Close() } data2 := bytes.NewBufferString(testData) err = c.Append("tset", data2) - if err != nil { - t.Error(err) - } + assert.NoError(err) // Read without deadline, after append r, err = c.Retr("tset") - if err != nil { - t.Error(err) - } else { - buf, errRead := ioutil.ReadAll(r) - if err != nil { - t.Error(errRead) - } - if string(buf) != testData+testData { - t.Errorf("'%s'", buf) + if assert.NoError(err) { + buf, err := ioutil.ReadAll(r) + if assert.NoError(err) { + assert.Equal(testData+testData, string(buf)) } + r.Close() } fileSize, err := c.FileSize("magic-file") - if err != nil { - t.Error(err) - } - if fileSize != 42 { - t.Errorf("file size %q, expected %q", fileSize, 42) - } + assert.NoError(err) + assert.Equal(int64(42), fileSize) _, err = c.FileSize("not-found") - if err == nil { - t.Fatal("expected error, got nil") - } + assert.Error(err) err = c.Delete("tset") - if err != nil { - t.Error(err) - } + assert.NoError(err) err = c.MakeDir(testDir) - if err != nil { - t.Error(err) - } + assert.NoError(err) err = c.ChangeDir(testDir) - if err != nil { - t.Error(err) - } + assert.NoError(err) err = c.ChangeDirToParent() - if err != nil { - t.Error(err) - } + assert.NoError(err) entries, err := c.NameList("/") - if err != nil { - t.Error(err) - } - if len(entries) != 1 || entries[0] != "/incoming" { - t.Errorf("Unexpected entries: %v", entries) - } + assert.NoError(err) + assert.Equal([]string{"/incoming"}, entries) err = c.RemoveDir(testDir) - if err != nil { - t.Error(err) - } + assert.NoError(err) err = c.Logout() - if err != nil { - if protoErr := err.(*textproto.Error); protoErr != nil { - if protoErr.Code != StatusNotImplemented { - t.Error(err) - } - } else { - t.Error(err) - } - } + assert.NoError(err) if err = c.Quit(); err != nil { t.Fatal(err) @@ -205,9 +140,7 @@ func testConn(t *testing.T, disableEPSV bool) { mock.Wait() err = c.NoOp() - if err == nil { - t.Error("Expected error") - } + assert.Error(err, "should error on closed conn") } // TestConnect tests the legacy Connect function diff --git a/conn_test.go b/conn_test.go index 130d2c6..cbe4b74 100644 --- a/conn_test.go +++ b/conn_test.go @@ -6,12 +6,14 @@ import ( "io" "net" "net/textproto" - "reflect" "strconv" "strings" "sync" "testing" "time" + + "github.com/stretchr/testify/assert" + "github.com/stretchr/testify/require" ) type ftpMock struct { @@ -386,20 +388,14 @@ func openConn(t *testing.T, addr string, options ...DialOption) (*ftpMock, *Serv func openConnExt(t *testing.T, addr, modtime string, options ...DialOption) (*ftpMock, *ServerConn) { mock, err := newFtpMockExt(t, addr, modtime) - if err != nil { - t.Fatal(err) - } + require.NoError(t, err) defer mock.Close() c, err := Dial(mock.Addr(), options...) - if err != nil { - t.Fatal(err) - } + require.NoError(t, err) err = c.Login("anonymous", "anonymous") - if err != nil { - t.Fatal(err) - } + require.NoError(t, err) return mock, c } @@ -417,9 +413,7 @@ func closeConn(t *testing.T, mock *ftpMock, c *ServerConn, commands []string) { // Wait for the connection to close mock.Wait() - if !reflect.DeepEqual(mock.commands, expected) { - t.Fatal("unexpected sequence of commands:", mock.commands, "expected:", expected) - } + assert.Equal(t, expected, mock.commands, "unexpected sequence of commands") } func TestConn4(t *testing.T) { From fa83b53d0e35def1ee9ca32470d08736bbc57094 Mon Sep 17 00:00:00 2001 From: Julien Laffaye Date: Wed, 17 Aug 2022 19:35:55 -0400 Subject: [PATCH 03/35] Use tls.Dialer to be able to use DialContext --- ftp.go | 16 ++++++++++------ 1 file changed, 10 insertions(+), 6 deletions(-) diff --git a/ftp.go b/ftp.go index 6e994ab..7ad3ca1 100644 --- a/ftp.go +++ b/ftp.go @@ -110,16 +110,20 @@ func Dial(addr string, options ...DialOption) (*ServerConn, error) { dialFunc := do.dialFunc if dialFunc == nil { + ctx := do.context + + if ctx == nil { + ctx = context.Background() + } if do.tlsConfig != nil && !do.explicitTLS { dialFunc = func(network, address string) (net.Conn, error) { - return tls.DialWithDialer(&do.dialer, network, addr, do.tlsConfig) + tlsDialer := &tls.Dialer{ + NetDialer: &do.dialer, + Config: do.tlsConfig, + } + return tlsDialer.DialContext(ctx, network, addr) } } else { - ctx := do.context - - if ctx == nil { - ctx = context.Background() - } dialFunc = func(network, address string) (net.Conn, error) { return do.dialer.DialContext(ctx, network, addr) From 39592b91e4d6bca360fcc2ab38aad6bbfab5de15 Mon Sep 17 00:00:00 2001 From: Julien Laffaye Date: Wed, 17 Aug 2022 19:43:45 -0400 Subject: [PATCH 04/35] Assign type to TransferType constants --- ftp.go | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/ftp.go b/ftp.go index 7ad3ca1..89f5197 100644 --- a/ftp.go +++ b/ftp.go @@ -33,8 +33,8 @@ type TransferType string // The different transfer types const ( - TransferTypeBinary = "I" - TransferTypeASCII = "A" + TransferTypeBinary = TransferType("I") + TransferTypeASCII = TransferType("A") ) // Time format used by the MDTM and MFMT commands From 9cda78131d0cf0a3cee5b6c9b4fefd15a44b9b6e Mon Sep 17 00:00:00 2001 From: Julien Laffaye Date: Thu, 18 Aug 2022 10:17:32 -0400 Subject: [PATCH 05/35] Fix transferring spelling. --- ftp.go | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/ftp.go b/ftp.go index 89f5197..1b5e1a5 100644 --- a/ftp.go +++ b/ftp.go @@ -28,7 +28,7 @@ const ( EntryTypeLink ) -// TransferType denotes the formats for transfering Entries. +// TransferType denotes the formats for transferring Entries. type TransferType string // The different transfer types From 5a2fd50da8bafe8f7bce92c6fc91802ae38687c5 Mon Sep 17 00:00:00 2001 From: Julien Laffaye Date: Thu, 18 Aug 2022 11:18:54 -0400 Subject: [PATCH 06/35] ioutil is deprecated, use io package instead --- client_test.go | 10 +++++----- 1 file changed, 5 insertions(+), 5 deletions(-) diff --git a/client_test.go b/client_test.go index 63899a9..01843d0 100644 --- a/client_test.go +++ b/client_test.go @@ -3,7 +3,7 @@ package ftp import ( "bytes" "fmt" - "io/ioutil" + "io" "net" "syscall" "testing" @@ -56,7 +56,7 @@ func testConn(t *testing.T, disableEPSV bool) { // Read without deadline r, err := c.Retr("tset") if assert.NoError(err) { - buf, err := ioutil.ReadAll(r) + buf, err := io.ReadAll(r) if assert.NoError(err) { assert.Equal(testData, string(buf)) } @@ -71,7 +71,7 @@ func testConn(t *testing.T, disableEPSV bool) { if err := r.SetDeadline(time.Now()); err != nil { t.Fatal(err) } - _, err = ioutil.ReadAll(r) + _, err = io.ReadAll(r) assert.ErrorContains(err, "i/o timeout") r.Close() } @@ -79,7 +79,7 @@ func testConn(t *testing.T, disableEPSV bool) { // Read with offset r, err = c.RetrFrom("tset", 5) if assert.NoError(err) { - buf, err := ioutil.ReadAll(r) + buf, err := io.ReadAll(r) if assert.NoError(err) { expected := testData[5:] assert.Equal(expected, string(buf)) @@ -95,7 +95,7 @@ func testConn(t *testing.T, disableEPSV bool) { // Read without deadline, after append r, err = c.Retr("tset") if assert.NoError(err) { - buf, err := ioutil.ReadAll(r) + buf, err := io.ReadAll(r) if assert.NoError(err) { assert.Equal(testData+testData, string(buf)) } From 4d1d644cf19d7547d270aab65c7570db1ccf2279 Mon Sep 17 00:00:00 2001 From: Julien Laffaye Date: Thu, 18 Aug 2022 12:44:22 -0400 Subject: [PATCH 07/35] Mark Connect() and DialWithTimeout() as deprecated --- ftp.go | 5 +++-- 1 file changed, 3 insertions(+), 2 deletions(-) diff --git a/ftp.go b/ftp.go index 1b5e1a5..9ba5651 100644 --- a/ftp.go +++ b/ftp.go @@ -308,14 +308,15 @@ func (o *dialOptions) wrapStream(rd io.ReadCloser) io.ReadCloser { } // Connect is an alias to Dial, for backward compatibility +// +// Deprecated: Use [Dial] instead func Connect(addr string) (*ServerConn, error) { return Dial(addr) } // DialTimeout initializes the connection to the specified ftp server address. // -// It is generally followed by a call to Login() as most FTP commands require -// an authenticated user. +// Deprecated: Use [Dial] with [DialWithTimeout] option instead func DialTimeout(addr string, timeout time.Duration) (*ServerConn, error) { return Dial(addr, DialWithTimeout(timeout)) } From 0aeb8660a7e2778621111d4ad3ef997a782fe9f2 Mon Sep 17 00:00:00 2001 From: Thomas Hallgren Date: Sun, 21 Aug 2022 23:25:29 +0200 Subject: [PATCH 08/35] Add MLST command in the form of a Get method (#269) * Add MLST command in the form of a Get method The `LIST` and `MLSD` commands are inefficient when the objective is to retrieve one single `Entry` for a known path, because the only way to get such an entry is to list the parent directory using a data connection. The `MLST` fixes this by allowing one single `Entry` to be returned using the control connection. The name `Get` was chosen because it is often used in conjunction with `List` as a mean to get one single entry. Signed-off-by: Thomas Hallgren * Changes in response to code review: - Rename `Get` to `GetEntry` (because it returns an `*Entry`) - Add test-case for multiline response on the control connection - Fix issues with parsing the multiline response. Signed-off-by: Thomas Hallgren * Add sample output from MLST to GetEntry comment. Signed-off-by: Thomas Hallgren * Changes in response to code review: - Remove unused `time.Time` argument - Add struct labels to make `govet` happy Signed-off-by: Thomas Hallgren * Remove time arg when calling parseNextRFC3659ListLine Signed-off-by: Thomas Hallgren Signed-off-by: Thomas Hallgren --- client_test.go | 36 +++++++++++++++++++++++++++++++++- conn_test.go | 19 +++++++++++++++++- ftp.go | 52 ++++++++++++++++++++++++++++++++++++++++++++++++++ parse.go | 14 +++++++++++--- 4 files changed, 116 insertions(+), 5 deletions(-) diff --git a/client_test.go b/client_test.go index 01843d0..b4faae5 100644 --- a/client_test.go +++ b/client_test.go @@ -110,6 +110,40 @@ func testConn(t *testing.T, disableEPSV bool) { _, err = c.FileSize("not-found") assert.Error(err) + entry, err := c.GetEntry("magic-file") + if err != nil { + t.Error(err) + } + if entry == nil { + t.Fatal("expected entry, got nil") + } + if entry.Size != 42 { + t.Errorf("entry size %q, expected %q", entry.Size, 42) + } + if entry.Type != EntryTypeFile { + t.Errorf("entry type %q, expected %q", entry.Type, EntryTypeFile) + } + if entry.Name != "magic-file" { + t.Errorf("entry name %q, expected %q", entry.Name, "magic-file") + } + + entry, err = c.GetEntry("multiline-dir") + if err != nil { + t.Error(err) + } + if entry == nil { + t.Fatal("expected entry, got nil") + } + if entry.Size != 0 { + t.Errorf("entry size %q, expected %q", entry.Size, 0) + } + if entry.Type != EntryTypeFolder { + t.Errorf("entry type %q, expected %q", entry.Type, EntryTypeFolder) + } + if entry.Name != "multiline-dir" { + t.Errorf("entry name %q, expected %q", entry.Name, "multiline-dir") + } + err = c.Delete("tset") assert.NoError(err) @@ -245,7 +279,7 @@ func TestMissingFolderDeleteDirRecur(t *testing.T) { } func TestListCurrentDir(t *testing.T) { - mock, c := openConn(t, "127.0.0.1") + mock, c := openConnExt(t, "127.0.0.1", "no-time", DialWithDisabledMLSD(true)) _, err := c.List("") assert.NoError(t, err) diff --git a/conn_test.go b/conn_test.go index cbe4b74..7d3c299 100644 --- a/conn_test.go +++ b/conn_test.go @@ -90,7 +90,7 @@ func (mock *ftpMock) listen() { // At least one command must have a multiline response switch cmdParts[0] { case "FEAT": - features := "211-Features:\r\n FEAT\r\n PASV\r\n EPSV\r\n UTF8\r\n SIZE\r\n" + features := "211-Features:\r\n FEAT\r\n PASV\r\n EPSV\r\n UTF8\r\n SIZE\r\n MLST\r\n" switch mock.modtime { case "std-time": features += " MDTM\r\n MFMT\r\n" @@ -178,6 +178,23 @@ func (mock *ftpMock) listen() { mock.dataConn.write([]byte("-rw-r--r-- 1 ftp wheel 0 Jan 29 10:29 lo\r\ntotal 1")) mock.printfLine("226 Transfer complete") mock.closeDataConn() + case "MLSD": + if mock.dataConn == nil { + mock.printfLine("425 Unable to build data connection: Connection refused") + break + } + + mock.dataConn.Wait() + mock.printfLine("150 Opening data connection for file list") + mock.dataConn.write([]byte("Type=file;Size=0;Modify=20201213202400; lo\r\n")) + mock.printfLine("226 Transfer complete") + mock.closeDataConn() + case "MLST": + if cmdParts[1] == "multiline-dir" { + mock.printfLine("250-File data\r\n Type=dir;Size=0; multiline-dir\r\n Modify=20201213202400; multiline-dir\r\n250 End") + } else { + mock.printfLine("250-File data\r\n Type=file;Size=42;Modify=20201213202400; magic-file\r\n250 End") + } case "NLST": if mock.dataConn == nil { mock.printfLine("425 Unable to build data connection: Connection refused") diff --git a/ftp.go b/ftp.go index 9ba5651..80975e8 100644 --- a/ftp.go +++ b/ftp.go @@ -678,6 +678,58 @@ func (c *ServerConn) List(path string) (entries []*Entry, err error) { return entries, errs.ErrorOrNil() } +// GetEntry issues a MLST FTP command which retrieves one single Entry using the +// control connection. The returnedEntry will describe the current directory +// when no path is given. +func (c *ServerConn) GetEntry(path string) (entry *Entry, err error) { + if !c.mlstSupported { + return nil, &textproto.Error{Code: StatusNotImplemented, Msg: StatusText(StatusNotImplemented)} + } + space := " " + if path == "" { + space = "" + } + _, msg, err := c.cmd(StatusRequestedFileActionOK, "%s%s%s", "MLST", space, path) + if err != nil { + return nil, err + } + + // The expected reply will look something like: + // + // 250-File details + // Type=file;Size=1024;Modify=20220813133357; path + // 250 End + // + // Multiple lines are allowed though, so it can also be in the form: + // + // 250-File details + // Type=file;Size=1024; path + // Modify=20220813133357; path + // 250 End + lines := strings.Split(msg, "\n") + lc := len(lines) + + // lines must be a multi-line message with a length of 3 or more, and we + // don't care about the first and last line + if lc < 3 { + return nil, errors.New("invalid response") + } + + e := &Entry{} + for _, l := range lines[1 : lc-1] { + // According to RFC 3659, the entry lines must start with a space when passed over the + // control connection. Some servers don't seem to add that space though. Both forms are + // accepted here. + if len(l) > 0 && l[0] == ' ' { + l = l[1:] + } + if e, err = parseNextRFC3659ListLine(l, c.options.location, e); err != nil { + return nil, err + } + } + return e, nil +} + // IsTimePreciseInList returns true if client and server support the MLSD // command so List can return time with 1-second precision for all files. func (c *ServerConn) IsTimePreciseInList() bool { diff --git a/parse.go b/parse.go index dbe0324..2decf1e 100644 --- a/parse.go +++ b/parse.go @@ -27,7 +27,11 @@ var dirTimeFormats = []string{ } // parseRFC3659ListLine parses the style of directory line defined in RFC 3659. -func parseRFC3659ListLine(line string, now time.Time, loc *time.Location) (*Entry, error) { +func parseRFC3659ListLine(line string, _ time.Time, loc *time.Location) (*Entry, error) { + return parseNextRFC3659ListLine(line, loc, &Entry{}) +} + +func parseNextRFC3659ListLine(line string, loc *time.Location, e *Entry) (*Entry, error) { iSemicolon := strings.Index(line, ";") iWhitespace := strings.Index(line, " ") @@ -35,8 +39,12 @@ func parseRFC3659ListLine(line string, now time.Time, loc *time.Location) (*Entr return nil, errUnsupportedListLine } - e := &Entry{ - Name: line[iWhitespace+1:], + name := line[iWhitespace+1:] + if e.Name == "" { + e.Name = name + } else if e.Name != name { + // All lines must have the same name + return nil, errUnsupportedListLine } for _, field := range strings.Split(line[:iWhitespace-1], ";") { From b85cf1edccd44c48daaf71b6880941c4b1856a54 Mon Sep 17 00:00:00 2001 From: Julien Laffaye Date: Sun, 28 Aug 2022 21:58:25 -0400 Subject: [PATCH 09/35] Add default timeout to instantiate connection --- ftp.go | 14 +++++++++++++- 1 file changed, 13 insertions(+), 1 deletion(-) diff --git a/ftp.go b/ftp.go index 80975e8..7b13628 100644 --- a/ftp.go +++ b/ftp.go @@ -18,6 +18,12 @@ import ( "github.com/hashicorp/go-multierror" ) +const ( + // 30 seconds was chosen as it's the + // same duration as http.DefaultTransport's timeout. + DefaultDialTimeout = 30 * time.Second +) + // EntryType describes the different types of an Entry. type EntryType int @@ -115,6 +121,12 @@ func Dial(addr string, options ...DialOption) (*ServerConn, error) { if ctx == nil { ctx = context.Background() } + if _, ok := ctx.Deadline(); !ok { + var cancel context.CancelFunc + ctx, cancel = context.WithTimeout(ctx, DefaultDialTimeout) + defer cancel() + } + if do.tlsConfig != nil && !do.explicitTLS { dialFunc = func(network, address string) (net.Conn, error) { tlsDialer := &tls.Dialer{ @@ -1001,7 +1013,7 @@ func (c *ServerConn) RemoveDir(path string) error { return err } -//Walk prepares the internal walk function so that the caller can begin traversing the directory +// Walk prepares the internal walk function so that the caller can begin traversing the directory func (c *ServerConn) Walk(root string) *Walker { w := new(Walker) w.serverConn = c From 99be0634ab9a1a9a61630647096d3e63b6a4873e Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=C3=98yvind=20Heddeland=20Instefjord?= Date: Sun, 4 Sep 2022 20:43:06 +0200 Subject: [PATCH 10/35] Add forceListHidden dial option to force the use of 'LIST -a' command (#271) This is useful for servers that do not offer up hidden folders/files by default when using LIST/MLSD commands. Setting forceListHidden to true will force the use of the 'LIST -a' command even when MLST support has been detected. --- client_test.go | 14 ++++++++++++++ ftp.go | 40 +++++++++++++++++++++++++++------------- walker.go | 2 +- 3 files changed, 42 insertions(+), 14 deletions(-) diff --git a/client_test.go b/client_test.go index b4faae5..75be766 100644 --- a/client_test.go +++ b/client_test.go @@ -295,6 +295,20 @@ func TestListCurrentDir(t *testing.T) { mock.Wait() } +func TestListCurrentDirWithForceListHidden(t *testing.T) { + mock, c := openConnExt(t, "127.0.0.1", "no-time", DialWithDisabledMLSD(true), DialWithForceListHidden(true)) + + assert.True(t, c.options.forceListHidden) + _, err := c.List("") + assert.NoError(t, err) + assert.Equal(t, "LIST -a", mock.lastFull, "LIST -a must not have a trailing whitespace") + + err = c.Quit() + assert.NoError(t, err) + + mock.Wait() +} + func TestTimeUnsupported(t *testing.T) { mock, c := openConnExt(t, "127.0.0.1", "no-time") diff --git a/ftp.go b/ftp.go index 7b13628..7059630 100644 --- a/ftp.go +++ b/ftp.go @@ -72,18 +72,19 @@ type DialOption struct { // dialOptions contains all the options set by DialOption.setup type dialOptions struct { - context context.Context - dialer net.Dialer - tlsConfig *tls.Config - explicitTLS bool - disableEPSV bool - disableUTF8 bool - disableMLSD bool - writingMDTM bool - location *time.Location - debugOutput io.Writer - dialFunc func(network, address string) (net.Conn, error) - shutTimeout time.Duration // time to wait for data connection closing status + context context.Context + dialer net.Dialer + tlsConfig *tls.Config + explicitTLS bool + disableEPSV bool + disableUTF8 bool + disableMLSD bool + writingMDTM bool + forceListHidden bool + location *time.Location + debugOutput io.Writer + dialFunc func(network, address string) (net.Conn, error) + shutTimeout time.Duration // time to wait for data connection closing status } // Entry describes a file and is returned by List(). @@ -247,6 +248,16 @@ func DialWithWritingMDTM(enabled bool) DialOption { }} } +// DialWithForceListHidden returns a DialOption making ServerConn use LIST -a to include hidden files and folders in directory listings +// +// This is useful for servers that do not do this by default, but it forces the use of the LIST command +// even if the server supports MLST. +func DialWithForceListHidden(enabled bool) DialOption { + return DialOption{func(do *dialOptions) { + do.forceListHidden = enabled + }} +} + // DialWithLocation returns a DialOption that configures the ServerConn with specified time.Location // The location is used to parse the dates sent by the server which are in server's timezone func DialWithLocation(location *time.Location) DialOption { @@ -650,11 +661,14 @@ func (c *ServerConn) List(path string) (entries []*Entry, err error) { var cmd string var parser parseFunc - if c.mlstSupported { + if c.mlstSupported && !c.options.forceListHidden { cmd = "MLSD" parser = parseRFC3659ListLine } else { cmd = "LIST" + if c.options.forceListHidden { + cmd += " -a" + } parser = parseListLine } diff --git a/walker.go b/walker.go index 3e0acc3..81735f1 100644 --- a/walker.go +++ b/walker.go @@ -4,7 +4,7 @@ import ( "path" ) -//Walker traverses the directory tree of a remote FTP server +// Walker traverses the directory tree of a remote FTP server type Walker struct { serverConn *ServerConn root string From be12fe0263fcc82a18c8d7931a24fb28f10bd2f1 Mon Sep 17 00:00:00 2001 From: Julien Laffaye Date: Wed, 7 Sep 2022 17:48:18 -0400 Subject: [PATCH 11/35] Create codeql-analysis.yml --- .github/workflows/codeql-analysis.yml | 72 +++++++++++++++++++++++++++ 1 file changed, 72 insertions(+) create mode 100644 .github/workflows/codeql-analysis.yml diff --git a/.github/workflows/codeql-analysis.yml b/.github/workflows/codeql-analysis.yml new file mode 100644 index 0000000..5dfabbc --- /dev/null +++ b/.github/workflows/codeql-analysis.yml @@ -0,0 +1,72 @@ +# For most projects, this workflow file will not need changing; you simply need +# to commit it to your repository. +# +# You may wish to alter this file to override the set of languages analyzed, +# or to provide custom queries or build logic. +# +# ******** NOTE ******** +# We have attempted to detect the languages in your repository. Please check +# the `language` matrix defined below to confirm you have the correct set of +# supported CodeQL languages. +# +name: "CodeQL" + +on: + push: + branches: [ "master" ] + pull_request: + # The branches below must be a subset of the branches above + branches: [ "master" ] + schedule: + - cron: '20 19 * * 2' + +jobs: + analyze: + name: Analyze + runs-on: ubuntu-latest + permissions: + actions: read + contents: read + security-events: write + + strategy: + fail-fast: false + matrix: + language: [ 'go' ] + # CodeQL supports [ 'cpp', 'csharp', 'go', 'java', 'javascript', 'python', 'ruby' ] + # Learn more about CodeQL language support at https://aka.ms/codeql-docs/language-support + + steps: + - name: Checkout repository + uses: actions/checkout@v3 + + # Initializes the CodeQL tools for scanning. + - name: Initialize CodeQL + uses: github/codeql-action/init@v2 + with: + languages: ${{ matrix.language }} + # If you wish to specify custom queries, you can do so here or in a config file. + # By default, queries listed here will override any specified in a config file. + # Prefix the list here with "+" to use these queries and those in the config file. + + # Details on CodeQL's query packs refer to : https://docs.github.com/en/code-security/code-scanning/automatically-scanning-your-code-for-vulnerabilities-and-errors/configuring-code-scanning#using-queries-in-ql-packs + # queries: security-extended,security-and-quality + + + # Autobuild attempts to build any compiled languages (C/C++, C#, or Java). + # If this step fails, then you should remove it and run the build manually (see below) + - name: Autobuild + uses: github/codeql-action/autobuild@v2 + + # ℹī¸ Command-line programs to run using the OS shell. + # 📚 See https://docs.github.com/en/actions/using-workflows/workflow-syntax-for-github-actions#jobsjob_idstepsrun + + # If the Autobuild fails above, remove it and uncomment the following three lines. + # modify them (or add more) to build your code if your project, please refer to the EXAMPLE below for guidance. + + # - run: | + # echo "Run, Build Application using script" + # ./location_of_script_within_repo/buildscript.sh + + - name: Perform CodeQL Analysis + uses: github/codeql-action/analyze@v2 From 9c3c2fd7407b322bbdd5f110b75019158ba27082 Mon Sep 17 00:00:00 2001 From: Julien Laffaye Date: Wed, 7 Sep 2022 18:02:26 -0400 Subject: [PATCH 12/35] Add CodeQL and golangci-lint badges --- README.md | 2 ++ 1 file changed, 2 insertions(+) diff --git a/README.md b/README.md index 6da76c1..8436b0e 100644 --- a/README.md +++ b/README.md @@ -2,6 +2,8 @@ [![Units tests](https://github.com/jlaffaye/ftp/actions/workflows/unit_tests.yaml/badge.svg)](https://github.com/jlaffaye/ftp/actions/workflows/unit_tests.yaml) [![Coverage Status](https://coveralls.io/repos/jlaffaye/ftp/badge.svg?branch=master&service=github)](https://coveralls.io/github/jlaffaye/ftp?branch=master) +[![golangci-lint](https://github.com/jlaffaye/ftp/actions/workflows/golangci-lint.yaml/badge.svg)](https://github.com/jlaffaye/ftp/actions/workflows/golangci-lint.yaml) +[![CodeQL](https://github.com/jlaffaye/ftp/actions/workflows/codeql-analysis.yml/badge.svg)](https://github.com/jlaffaye/ftp/actions/workflows/codeql-analysis.yml) [![Go ReportCard](https://goreportcard.com/badge/jlaffaye/ftp)](http://goreportcard.com/report/jlaffaye/ftp) [![Go Reference](https://pkg.go.dev/badge/github.com/jlaffaye/ftp.svg)](https://pkg.go.dev/github.com/jlaffaye/ftp) From f1ba13192bcdb1f34ec315b671fbd2d047c90a54 Mon Sep 17 00:00:00 2001 From: Julien Laffaye Date: Wed, 7 Sep 2022 18:03:34 -0400 Subject: [PATCH 13/35] Add dependabot for github-actions --- .github/dependabot.yml | 4 ++++ 1 file changed, 4 insertions(+) diff --git a/.github/dependabot.yml b/.github/dependabot.yml index 66efae6..2d34a20 100644 --- a/.github/dependabot.yml +++ b/.github/dependabot.yml @@ -11,3 +11,7 @@ updates: interval: "daily" assignees: - "jlaffaye" + - package-ecosystem: "github-actions" + directory: "/" + schedule: + interval: "daily" From 260999f2decb8da1fe5292c852b4c307d6b94624 Mon Sep 17 00:00:00 2001 From: Julien Laffaye Date: Wed, 7 Sep 2022 18:14:30 -0400 Subject: [PATCH 14/35] Secure GitHub workflows --- .github/workflows/codeql-analysis.yml | 11 ++++------- .github/workflows/golangci-lint.yaml | 7 +++++-- .github/workflows/unit_tests.yaml | 12 ++++++------ 3 files changed, 15 insertions(+), 15 deletions(-) diff --git a/.github/workflows/codeql-analysis.yml b/.github/workflows/codeql-analysis.yml index 5dfabbc..c5f8e91 100644 --- a/.github/workflows/codeql-analysis.yml +++ b/.github/workflows/codeql-analysis.yml @@ -20,6 +20,9 @@ on: schedule: - cron: '20 19 * * 2' +permissions: + contents: read + jobs: analyze: name: Analyze @@ -52,12 +55,6 @@ jobs: # Details on CodeQL's query packs refer to : https://docs.github.com/en/code-security/code-scanning/automatically-scanning-your-code-for-vulnerabilities-and-errors/configuring-code-scanning#using-queries-in-ql-packs # queries: security-extended,security-and-quality - - # Autobuild attempts to build any compiled languages (C/C++, C#, or Java). - # If this step fails, then you should remove it and run the build manually (see below) - - name: Autobuild - uses: github/codeql-action/autobuild@v2 - # ℹī¸ Command-line programs to run using the OS shell. # 📚 See https://docs.github.com/en/actions/using-workflows/workflow-syntax-for-github-actions#jobsjob_idstepsrun @@ -69,4 +66,4 @@ jobs: # ./location_of_script_within_repo/buildscript.sh - name: Perform CodeQL Analysis - uses: github/codeql-action/analyze@v2 + uses: github/codeql-action/analyze@b398f525a5587552e573b247ac661067fafa920b diff --git a/.github/workflows/golangci-lint.yaml b/.github/workflows/golangci-lint.yaml index 45cf1fe..3beffad 100644 --- a/.github/workflows/golangci-lint.yaml +++ b/.github/workflows/golangci-lint.yaml @@ -5,9 +5,12 @@ jobs: golangci-lint: name: lint runs-on: ubuntu-latest + permissions: + contents: read # for actions/checkout to fetch code + pull-requests: read # for golangci/golangci-lint-action to fetch pull requests steps: - - uses: actions/checkout@v2 + - uses: actions/checkout@7884fcad6b5d53d10323aee724dc68d8b9096a2e - name: golangci-lint - uses: golangci/golangci-lint-action@v2 + uses: golangci/golangci-lint-action@5c56cd6c9dc07901af25baab6f2b0d9f3b7c3018 with: only-new-issues: true diff --git a/.github/workflows/unit_tests.yaml b/.github/workflows/unit_tests.yaml index 48b2ade..ea2ce8c 100644 --- a/.github/workflows/unit_tests.yaml +++ b/.github/workflows/unit_tests.yaml @@ -5,12 +5,12 @@ jobs: name: test runs-on: ubuntu-latest steps: - - uses: actions/checkout@master + - uses: actions/checkout@61b9e3751b92087fd0b06925ba6dd6314e06f089 - name: Setup go - uses: actions/setup-go@v2 + uses: actions/setup-go@bfdd3570ce990073878bf10f6b2d79082de49492 with: - go-version: 1.17 - - uses: actions/cache@v2 + go-version: 1.19 + - uses: actions/cache@937d24475381cd9c75ae6db12cb4e79714b926ed with: path: | ~/go/pkg/mod @@ -21,9 +21,9 @@ jobs: - name: Run tests run: go test -v -covermode=count -coverprofile=coverage.out - name: Convert coverage to lcov - uses: jandelgado/gcov2lcov-action@v1.0.8 + uses: jandelgado/gcov2lcov-action@2477d9ec11814a9326e7fdde8d315783d163edb7 - name: Coveralls - uses: coverallsapp/github-action@v1.1.2 + uses: coverallsapp/github-action@8cbef1dea373ebce56de0a14c68d6267baa10b44 with: github-token: ${{ secrets.github_token }} path-to-lcov: coverage.lcov From 08a9ec380c0359748eeedf40e63db18f3a3facfd Mon Sep 17 00:00:00 2001 From: "dependabot[bot]" <49699333+dependabot[bot]@users.noreply.github.com> Date: Wed, 7 Sep 2022 18:42:48 -0400 Subject: [PATCH 15/35] Bump actions/cache from 2.1.7 to 3.0.8 (#279) Bumps [actions/cache](https://github.com/actions/cache) from 2.1.7 to 3.0.8. - [Release notes](https://github.com/actions/cache/releases) - [Changelog](https://github.com/actions/cache/blob/main/RELEASES.md) - [Commits](https://github.com/actions/cache/compare/937d24475381cd9c75ae6db12cb4e79714b926ed...fd5de65bc895cf536527842281bea11763fefd77) --- updated-dependencies: - dependency-name: actions/cache dependency-type: direct:production update-type: version-update:semver-major ... Signed-off-by: dependabot[bot] Signed-off-by: dependabot[bot] Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com> --- .github/workflows/unit_tests.yaml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/.github/workflows/unit_tests.yaml b/.github/workflows/unit_tests.yaml index ea2ce8c..3d3243b 100644 --- a/.github/workflows/unit_tests.yaml +++ b/.github/workflows/unit_tests.yaml @@ -10,7 +10,7 @@ jobs: uses: actions/setup-go@bfdd3570ce990073878bf10f6b2d79082de49492 with: go-version: 1.19 - - uses: actions/cache@937d24475381cd9c75ae6db12cb4e79714b926ed + - uses: actions/cache@fd5de65bc895cf536527842281bea11763fefd77 with: path: | ~/go/pkg/mod From dca029e125096fcd00f96d2818b577637599ff06 Mon Sep 17 00:00:00 2001 From: "dependabot[bot]" <49699333+dependabot[bot]@users.noreply.github.com> Date: Wed, 7 Sep 2022 18:45:22 -0400 Subject: [PATCH 16/35] Bump golangci/golangci-lint-action from 2.5.2 to 3.2.0 (#278) Bumps [golangci/golangci-lint-action](https://github.com/golangci/golangci-lint-action) from 2.5.2 to 3.2.0. - [Release notes](https://github.com/golangci/golangci-lint-action/releases) - [Commits](https://github.com/golangci/golangci-lint-action/compare/5c56cd6c9dc07901af25baab6f2b0d9f3b7c3018...537aa1903e5d359d0b27dbc19ddd22c5087f3fbc) --- updated-dependencies: - dependency-name: golangci/golangci-lint-action dependency-type: direct:production update-type: version-update:semver-major ... Signed-off-by: dependabot[bot] Signed-off-by: dependabot[bot] Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com> --- .github/workflows/golangci-lint.yaml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/.github/workflows/golangci-lint.yaml b/.github/workflows/golangci-lint.yaml index 3beffad..547e64d 100644 --- a/.github/workflows/golangci-lint.yaml +++ b/.github/workflows/golangci-lint.yaml @@ -11,6 +11,6 @@ jobs: steps: - uses: actions/checkout@7884fcad6b5d53d10323aee724dc68d8b9096a2e - name: golangci-lint - uses: golangci/golangci-lint-action@5c56cd6c9dc07901af25baab6f2b0d9f3b7c3018 + uses: golangci/golangci-lint-action@537aa1903e5d359d0b27dbc19ddd22c5087f3fbc with: only-new-issues: true From b40593bd5a77be5cee94e3bd44a893bd2189abbe Mon Sep 17 00:00:00 2001 From: "dependabot[bot]" <49699333+dependabot[bot]@users.noreply.github.com> Date: Wed, 7 Sep 2022 18:45:43 -0400 Subject: [PATCH 17/35] Bump jandelgado/gcov2lcov-action from 1.0.8 to 1.0.9 (#273) Bumps [jandelgado/gcov2lcov-action](https://github.com/jandelgado/gcov2lcov-action) from 1.0.8 to 1.0.9. - [Release notes](https://github.com/jandelgado/gcov2lcov-action/releases) - [Changelog](https://github.com/jandelgado/gcov2lcov-action/blob/master/CHANGELOG.md) - [Commits](https://github.com/jandelgado/gcov2lcov-action/compare/v1.0.8...v1.0.9) --- updated-dependencies: - dependency-name: jandelgado/gcov2lcov-action dependency-type: direct:production update-type: version-update:semver-patch ... Signed-off-by: dependabot[bot] Signed-off-by: dependabot[bot] Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com> --- .github/workflows/unit_tests.yaml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/.github/workflows/unit_tests.yaml b/.github/workflows/unit_tests.yaml index 3d3243b..fb0add8 100644 --- a/.github/workflows/unit_tests.yaml +++ b/.github/workflows/unit_tests.yaml @@ -21,7 +21,7 @@ jobs: - name: Run tests run: go test -v -covermode=count -coverprofile=coverage.out - name: Convert coverage to lcov - uses: jandelgado/gcov2lcov-action@2477d9ec11814a9326e7fdde8d315783d163edb7 + uses: jandelgado/gcov2lcov-action@c680c0f7c7442485f1749eb2a13e54a686e76eb5 - name: Coveralls uses: coverallsapp/github-action@8cbef1dea373ebce56de0a14c68d6267baa10b44 with: From a2b3f0878b3cbf5368fe3dec4303194b34d2f683 Mon Sep 17 00:00:00 2001 From: "dependabot[bot]" <49699333+dependabot[bot]@users.noreply.github.com> Date: Wed, 7 Sep 2022 18:46:14 -0400 Subject: [PATCH 18/35] Bump coverallsapp/github-action from 1.1.2 to 1.1.3 (#274) Bumps [coverallsapp/github-action](https://github.com/coverallsapp/github-action) from 1.1.2 to 1.1.3. - [Release notes](https://github.com/coverallsapp/github-action/releases) - [Commits](https://github.com/coverallsapp/github-action/compare/v1.1.2...1.1.3) --- updated-dependencies: - dependency-name: coverallsapp/github-action dependency-type: direct:production update-type: version-update:semver-patch ... Signed-off-by: dependabot[bot] Signed-off-by: dependabot[bot] Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com> --- .github/workflows/unit_tests.yaml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/.github/workflows/unit_tests.yaml b/.github/workflows/unit_tests.yaml index fb0add8..2863e15 100644 --- a/.github/workflows/unit_tests.yaml +++ b/.github/workflows/unit_tests.yaml @@ -23,7 +23,7 @@ jobs: - name: Convert coverage to lcov uses: jandelgado/gcov2lcov-action@c680c0f7c7442485f1749eb2a13e54a686e76eb5 - name: Coveralls - uses: coverallsapp/github-action@8cbef1dea373ebce56de0a14c68d6267baa10b44 + uses: coverallsapp/github-action@9ba913c152ae4be1327bfb9085dc806cedb44057 with: github-token: ${{ secrets.github_token }} path-to-lcov: coverage.lcov From 6512c2a4ae1a4090fd6e9a10f98bb11aebe3c44f Mon Sep 17 00:00:00 2001 From: "dependabot[bot]" <49699333+dependabot[bot]@users.noreply.github.com> Date: Wed, 7 Sep 2022 18:52:58 -0400 Subject: [PATCH 19/35] Bump actions/setup-go from 2.2.0 to 3.3.0 (#280) Bumps [actions/setup-go](https://github.com/actions/setup-go) from 2.2.0 to 3.3.0. - [Release notes](https://github.com/actions/setup-go/releases) - [Commits](https://github.com/actions/setup-go/compare/bfdd3570ce990073878bf10f6b2d79082de49492...268d8c0ca0432bb2cf416faae41297df9d262d7f) --- updated-dependencies: - dependency-name: actions/setup-go dependency-type: direct:production update-type: version-update:semver-major ... Signed-off-by: dependabot[bot] Signed-off-by: dependabot[bot] Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com> --- .github/workflows/unit_tests.yaml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/.github/workflows/unit_tests.yaml b/.github/workflows/unit_tests.yaml index 2863e15..4db90d5 100644 --- a/.github/workflows/unit_tests.yaml +++ b/.github/workflows/unit_tests.yaml @@ -7,7 +7,7 @@ jobs: steps: - uses: actions/checkout@61b9e3751b92087fd0b06925ba6dd6314e06f089 - name: Setup go - uses: actions/setup-go@bfdd3570ce990073878bf10f6b2d79082de49492 + uses: actions/setup-go@268d8c0ca0432bb2cf416faae41297df9d262d7f with: go-version: 1.19 - uses: actions/cache@fd5de65bc895cf536527842281bea11763fefd77 From 917837ffbd127430e33ef83f6029da60bcd5452f Mon Sep 17 00:00:00 2001 From: "dependabot[bot]" <49699333+dependabot[bot]@users.noreply.github.com> Date: Wed, 21 Dec 2022 17:31:46 -0500 Subject: [PATCH 20/35] Bump actions/cache from 3.0.8 to 3.2.0 (#301) Bumps [actions/cache](https://github.com/actions/cache) from 3.0.8 to 3.2.0. - [Release notes](https://github.com/actions/cache/releases) - [Changelog](https://github.com/actions/cache/blob/main/RELEASES.md) - [Commits](https://github.com/actions/cache/compare/fd5de65bc895cf536527842281bea11763fefd77...c17f4bf4666a8001b1a45c09eb7a485c41aa64c3) --- updated-dependencies: - dependency-name: actions/cache dependency-type: direct:production update-type: version-update:semver-minor ... Signed-off-by: dependabot[bot] Signed-off-by: dependabot[bot] Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com> --- .github/workflows/unit_tests.yaml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/.github/workflows/unit_tests.yaml b/.github/workflows/unit_tests.yaml index 4db90d5..2f5aebd 100644 --- a/.github/workflows/unit_tests.yaml +++ b/.github/workflows/unit_tests.yaml @@ -10,7 +10,7 @@ jobs: uses: actions/setup-go@268d8c0ca0432bb2cf416faae41297df9d262d7f with: go-version: 1.19 - - uses: actions/cache@fd5de65bc895cf536527842281bea11763fefd77 + - uses: actions/cache@c17f4bf4666a8001b1a45c09eb7a485c41aa64c3 with: path: | ~/go/pkg/mod From a0ec57882baad191fe6a89126d1caf1cbf0b0804 Mon Sep 17 00:00:00 2001 From: "dependabot[bot]" <49699333+dependabot[bot]@users.noreply.github.com> Date: Wed, 21 Dec 2022 17:32:07 -0500 Subject: [PATCH 21/35] Bump actions/setup-go from 3.3.0 to 3.5.0 (#300) Bumps [actions/setup-go](https://github.com/actions/setup-go) from 3.3.0 to 3.5.0. - [Release notes](https://github.com/actions/setup-go/releases) - [Commits](https://github.com/actions/setup-go/compare/268d8c0ca0432bb2cf416faae41297df9d262d7f...6edd4406fa81c3da01a34fa6f6343087c207a568) --- updated-dependencies: - dependency-name: actions/setup-go dependency-type: direct:production update-type: version-update:semver-minor ... Signed-off-by: dependabot[bot] Signed-off-by: dependabot[bot] Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com> --- .github/workflows/unit_tests.yaml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/.github/workflows/unit_tests.yaml b/.github/workflows/unit_tests.yaml index 2f5aebd..bff2b6f 100644 --- a/.github/workflows/unit_tests.yaml +++ b/.github/workflows/unit_tests.yaml @@ -7,7 +7,7 @@ jobs: steps: - uses: actions/checkout@61b9e3751b92087fd0b06925ba6dd6314e06f089 - name: Setup go - uses: actions/setup-go@268d8c0ca0432bb2cf416faae41297df9d262d7f + uses: actions/setup-go@6edd4406fa81c3da01a34fa6f6343087c207a568 with: go-version: 1.19 - uses: actions/cache@c17f4bf4666a8001b1a45c09eb7a485c41aa64c3 From 5b1ec4dc6cd780a9f3d93eacc0aa194baf09f4cb Mon Sep 17 00:00:00 2001 From: "dependabot[bot]" <49699333+dependabot[bot]@users.noreply.github.com> Date: Wed, 21 Dec 2022 17:32:28 -0500 Subject: [PATCH 22/35] Bump golangci/golangci-lint-action from 3.2.0 to 3.3.1 (#297) Bumps [golangci/golangci-lint-action](https://github.com/golangci/golangci-lint-action) from 3.2.0 to 3.3.1. - [Release notes](https://github.com/golangci/golangci-lint-action/releases) - [Commits](https://github.com/golangci/golangci-lint-action/compare/537aa1903e5d359d0b27dbc19ddd22c5087f3fbc...0ad9a0988b3973e851ab0a07adf248ec2e100376) --- updated-dependencies: - dependency-name: golangci/golangci-lint-action dependency-type: direct:production update-type: version-update:semver-minor ... Signed-off-by: dependabot[bot] Signed-off-by: dependabot[bot] Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com> --- .github/workflows/golangci-lint.yaml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/.github/workflows/golangci-lint.yaml b/.github/workflows/golangci-lint.yaml index 547e64d..c784f1e 100644 --- a/.github/workflows/golangci-lint.yaml +++ b/.github/workflows/golangci-lint.yaml @@ -11,6 +11,6 @@ jobs: steps: - uses: actions/checkout@7884fcad6b5d53d10323aee724dc68d8b9096a2e - name: golangci-lint - uses: golangci/golangci-lint-action@537aa1903e5d359d0b27dbc19ddd22c5087f3fbc + uses: golangci/golangci-lint-action@0ad9a0988b3973e851ab0a07adf248ec2e100376 with: only-new-issues: true From b10ce93530e360831073440bd7be8622b75feb01 Mon Sep 17 00:00:00 2001 From: "dependabot[bot]" <49699333+dependabot[bot]@users.noreply.github.com> Date: Wed, 21 Dec 2022 17:33:01 -0500 Subject: [PATCH 23/35] Bump github.com/stretchr/testify from 1.8.0 to 1.8.1 (#294) Bumps [github.com/stretchr/testify](https://github.com/stretchr/testify) from 1.8.0 to 1.8.1. - [Release notes](https://github.com/stretchr/testify/releases) - [Commits](https://github.com/stretchr/testify/compare/v1.8.0...v1.8.1) --- updated-dependencies: - dependency-name: github.com/stretchr/testify dependency-type: direct:production update-type: version-update:semver-patch ... Signed-off-by: dependabot[bot] Signed-off-by: dependabot[bot] Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com> --- go.mod | 2 +- go.sum | 4 +++- 2 files changed, 4 insertions(+), 2 deletions(-) diff --git a/go.mod b/go.mod index 1f6b68e..b5bce9c 100644 --- a/go.mod +++ b/go.mod @@ -4,7 +4,7 @@ go 1.17 require ( github.com/hashicorp/go-multierror v1.1.1 - github.com/stretchr/testify v1.8.0 + github.com/stretchr/testify v1.8.1 ) require ( diff --git a/go.sum b/go.sum index de9418d..61372e7 100644 --- a/go.sum +++ b/go.sum @@ -9,9 +9,11 @@ github.com/pmezard/go-difflib v1.0.0 h1:4DBwDE0NGyQoBHbLQYPwSUPoCMWR5BEzIk/f1lZb github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4= github.com/stretchr/objx v0.1.0/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME= github.com/stretchr/objx v0.4.0/go.mod h1:YvHI0jy2hoMjB+UWwv71VJQ9isScKT/TqJzVSSt89Yw= +github.com/stretchr/objx v0.5.0/go.mod h1:Yh+to48EsGEfYuaHDzXPcE3xhTkx73EhmCGUpEOglKo= github.com/stretchr/testify v1.7.1/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg= -github.com/stretchr/testify v1.8.0 h1:pSgiaMZlXftHpm5L7V1+rVB+AZJydKsMxsQBIJw4PKk= github.com/stretchr/testify v1.8.0/go.mod h1:yNjHg4UonilssWZ8iaSj1OCr/vHnekPRkoO+kdMU+MU= +github.com/stretchr/testify v1.8.1 h1:w7B6lhMri9wdJUVmEZPGGhZzrYTPvgJArz7wNPgYKsk= +github.com/stretchr/testify v1.8.1/go.mod h1:w2LPCIKwWwSfY2zedu0+kehJoqGctiVI29o6fzry7u4= gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405 h1:yhCVgyC4o1eVCa2tZl7eS0r+SDo693bJlVdllGtEeKM= gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= gopkg.in/yaml.v3 v3.0.0-20200313102051-9f266ea9e77c/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM= From 5fc8c694ec1fefb04970605b67cfc91a35964267 Mon Sep 17 00:00:00 2001 From: "dependabot[bot]" <49699333+dependabot[bot]@users.noreply.github.com> Date: Mon, 16 Jan 2023 13:19:23 -0500 Subject: [PATCH 24/35] Bump actions/cache from 3.2.0 to 3.2.3 (#304) Bumps [actions/cache](https://github.com/actions/cache) from 3.2.0 to 3.2.3. - [Release notes](https://github.com/actions/cache/releases) - [Changelog](https://github.com/actions/cache/blob/main/RELEASES.md) - [Commits](https://github.com/actions/cache/compare/c17f4bf4666a8001b1a45c09eb7a485c41aa64c3...58c146cc91c5b9e778e71775dfe9bf1442ad9a12) --- updated-dependencies: - dependency-name: actions/cache dependency-type: direct:production update-type: version-update:semver-patch ... Signed-off-by: dependabot[bot] Signed-off-by: dependabot[bot] Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com> --- .github/workflows/unit_tests.yaml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/.github/workflows/unit_tests.yaml b/.github/workflows/unit_tests.yaml index bff2b6f..1e85a6e 100644 --- a/.github/workflows/unit_tests.yaml +++ b/.github/workflows/unit_tests.yaml @@ -10,7 +10,7 @@ jobs: uses: actions/setup-go@6edd4406fa81c3da01a34fa6f6343087c207a568 with: go-version: 1.19 - - uses: actions/cache@c17f4bf4666a8001b1a45c09eb7a485c41aa64c3 + - uses: actions/cache@58c146cc91c5b9e778e71775dfe9bf1442ad9a12 with: path: | ~/go/pkg/mod From 0257a7bc42044ca5b4538b9c7f53c68f2078e630 Mon Sep 17 00:00:00 2001 From: "dependabot[bot]" <49699333+dependabot[bot]@users.noreply.github.com> Date: Mon, 23 Jan 2023 20:58:31 -0500 Subject: [PATCH 25/35] Bump golangci/golangci-lint-action from 3.3.1 to 3.4.0 (#306) Bumps [golangci/golangci-lint-action](https://github.com/golangci/golangci-lint-action) from 3.3.1 to 3.4.0. - [Release notes](https://github.com/golangci/golangci-lint-action/releases) - [Commits](https://github.com/golangci/golangci-lint-action/compare/0ad9a0988b3973e851ab0a07adf248ec2e100376...08e2f20817b15149a52b5b3ebe7de50aff2ba8c5) --- updated-dependencies: - dependency-name: golangci/golangci-lint-action dependency-type: direct:production update-type: version-update:semver-minor ... Signed-off-by: dependabot[bot] Signed-off-by: dependabot[bot] Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com> --- .github/workflows/golangci-lint.yaml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/.github/workflows/golangci-lint.yaml b/.github/workflows/golangci-lint.yaml index c784f1e..b06aaec 100644 --- a/.github/workflows/golangci-lint.yaml +++ b/.github/workflows/golangci-lint.yaml @@ -11,6 +11,6 @@ jobs: steps: - uses: actions/checkout@7884fcad6b5d53d10323aee724dc68d8b9096a2e - name: golangci-lint - uses: golangci/golangci-lint-action@0ad9a0988b3973e851ab0a07adf248ec2e100376 + uses: golangci/golangci-lint-action@08e2f20817b15149a52b5b3ebe7de50aff2ba8c5 with: only-new-issues: true From 58cb52405287644d64f7a5d993b8d3776d4a7120 Mon Sep 17 00:00:00 2001 From: "dependabot[bot]" <49699333+dependabot[bot]@users.noreply.github.com> Date: Wed, 8 Feb 2023 11:44:47 -0500 Subject: [PATCH 26/35] Bump actions/cache from 3.2.3 to 3.2.4 (#307) Bumps [actions/cache](https://github.com/actions/cache) from 3.2.3 to 3.2.4. - [Release notes](https://github.com/actions/cache/releases) - [Changelog](https://github.com/actions/cache/blob/main/RELEASES.md) - [Commits](https://github.com/actions/cache/compare/58c146cc91c5b9e778e71775dfe9bf1442ad9a12...627f0f41f6904a5b1efbaed9f96d9eb58e92e920) --- updated-dependencies: - dependency-name: actions/cache dependency-type: direct:production update-type: version-update:semver-patch ... Signed-off-by: dependabot[bot] Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com> --- .github/workflows/unit_tests.yaml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/.github/workflows/unit_tests.yaml b/.github/workflows/unit_tests.yaml index 1e85a6e..5e71700 100644 --- a/.github/workflows/unit_tests.yaml +++ b/.github/workflows/unit_tests.yaml @@ -10,7 +10,7 @@ jobs: uses: actions/setup-go@6edd4406fa81c3da01a34fa6f6343087c207a568 with: go-version: 1.19 - - uses: actions/cache@58c146cc91c5b9e778e71775dfe9bf1442ad9a12 + - uses: actions/cache@627f0f41f6904a5b1efbaed9f96d9eb58e92e920 with: path: | ~/go/pkg/mod From 9e39e2c406832ca08f0f429e53fa85f8117275d8 Mon Sep 17 00:00:00 2001 From: Nick Craig-Wood Date: Wed, 8 Feb 2023 16:46:41 +0000 Subject: [PATCH 27/35] Fix hang when using ExplicitTLS to certain servers. (#283) In #282 it was discovered that doing the tls Handshake immediately on connection causes some FTP servers (proftpd and pureftpd) to hang. The exact cause of this is unknown, but this patch works around the problem by not doing the Handsake initially, and only doing it at the end if we were attempting to upload a zero length file. Doing the Handshake at the end was originally added in a4e9650823896675 however it got reverted in 212daf295f0e6ae4 which used tls.DialWithDialer to do the handshake. Unfortunately tls.DialWithDialer seems to trigger the hanging bug. See: https://forum.rclone.org/t/rclone-ftps-explicit-rclone-touch-empty-files-proftpd-unable-to-build-data-connection-operation-not-permitted/22522 See: https://github.com/rclone/rclone/issues/6426#issuecomment-1243993039 Fixes #282 --- ftp.go | 34 ++++++++++++++++++++++++++++++++-- 1 file changed, 32 insertions(+), 2 deletions(-) diff --git a/ftp.go b/ftp.go index 7059630..bbc1799 100644 --- a/ftp.go +++ b/ftp.go @@ -559,7 +559,24 @@ func (c *ServerConn) openDataConn() (net.Conn, error) { } if c.options.tlsConfig != nil { - return tls.DialWithDialer(&c.options.dialer, "tcp", addr, c.options.tlsConfig) + // We don't use tls.DialWithDialer here (which does Dial, create + // the Client and then do the Handshake) because it seems to + // hang with some FTP servers, namely proftpd and pureftpd. + // + // Instead we do Dial, create the Client and wait for the first + // Read or Write to trigger the Handshake. + // + // This means that if we are uploading a zero sized file, we + // need to make sure we do the Handshake explicitly as Write + // won't have been called. This is done in StorFrom(). + // + // See: https://github.com/jlaffaye/ftp/issues/282 + conn, err := c.options.dialer.Dial("tcp", addr) + if err != nil { + return nil, err + } + tlsConn := tls.Client(conn, c.options.tlsConfig) + return tlsConn, nil } return c.options.dialer.Dial("tcp", addr) @@ -912,8 +929,21 @@ func (c *ServerConn) StorFrom(path string, r io.Reader, offset uint64) error { // response otherwise if the failure is not due to a connection problem, // for example the server denied the upload for quota limits, we miss // the response and we cannot use the connection to send other commands. - if _, err := io.Copy(conn, r); err != nil { + if n, err := io.Copy(conn, r); err != nil { errs = multierror.Append(errs, err) + } else if n == 0 { + // If we wrote no bytes and got no error, make sure we call + // tls.Handshake on the connection as it won't get called + // unless Write() is called. (See comment in openDataConn()). + // + // ProFTP doesn't like this and returns "Unable to build data + // connection: Operation not permitted" when trying to upload + // an empty file without this. + if do, ok := conn.(interface{ Handshake() error }); ok { + if err := do.Handshake(); err != nil { + errs = multierror.Append(errs, err) + } + } } if err := conn.Close(); err != nil { From d84bf4be2b6e9dca4a58a2d1f326d96294e5c339 Mon Sep 17 00:00:00 2001 From: "dependabot[bot]" <49699333+dependabot[bot]@users.noreply.github.com> Date: Mon, 13 Feb 2023 19:46:52 -0500 Subject: [PATCH 28/35] Bump actions/cache from 3.2.4 to 3.2.5 (#308) Bumps [actions/cache](https://github.com/actions/cache) from 3.2.4 to 3.2.5. - [Release notes](https://github.com/actions/cache/releases) - [Changelog](https://github.com/actions/cache/blob/main/RELEASES.md) - [Commits](https://github.com/actions/cache/compare/627f0f41f6904a5b1efbaed9f96d9eb58e92e920...6998d139ddd3e68c71e9e398d8e40b71a2f39812) --- updated-dependencies: - dependency-name: actions/cache dependency-type: direct:production update-type: version-update:semver-patch ... Signed-off-by: dependabot[bot] Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com> --- .github/workflows/unit_tests.yaml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/.github/workflows/unit_tests.yaml b/.github/workflows/unit_tests.yaml index 5e71700..e2fc836 100644 --- a/.github/workflows/unit_tests.yaml +++ b/.github/workflows/unit_tests.yaml @@ -10,7 +10,7 @@ jobs: uses: actions/setup-go@6edd4406fa81c3da01a34fa6f6343087c207a568 with: go-version: 1.19 - - uses: actions/cache@627f0f41f6904a5b1efbaed9f96d9eb58e92e920 + - uses: actions/cache@6998d139ddd3e68c71e9e398d8e40b71a2f39812 with: path: | ~/go/pkg/mod From f568a0d846891ab000168e0d7c94cddf8069b511 Mon Sep 17 00:00:00 2001 From: "dependabot[bot]" <49699333+dependabot[bot]@users.noreply.github.com> Date: Fri, 12 May 2023 22:27:20 -0400 Subject: [PATCH 29/35] Bump coverallsapp/github-action from 1.1.3 to 2.1.2 (#327) Bumps [coverallsapp/github-action](https://github.com/coverallsapp/github-action) from 1.1.3 to 2.1.2. - [Release notes](https://github.com/coverallsapp/github-action/releases) - [Commits](https://github.com/coverallsapp/github-action/compare/9ba913c152ae4be1327bfb9085dc806cedb44057...f350da2c033043742f89e8c0b7b5145a1616da6d) --- updated-dependencies: - dependency-name: coverallsapp/github-action dependency-type: direct:production update-type: version-update:semver-major ... Signed-off-by: dependabot[bot] Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com> --- .github/workflows/unit_tests.yaml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/.github/workflows/unit_tests.yaml b/.github/workflows/unit_tests.yaml index e2fc836..b8bc1a9 100644 --- a/.github/workflows/unit_tests.yaml +++ b/.github/workflows/unit_tests.yaml @@ -23,7 +23,7 @@ jobs: - name: Convert coverage to lcov uses: jandelgado/gcov2lcov-action@c680c0f7c7442485f1749eb2a13e54a686e76eb5 - name: Coveralls - uses: coverallsapp/github-action@9ba913c152ae4be1327bfb9085dc806cedb44057 + uses: coverallsapp/github-action@f350da2c033043742f89e8c0b7b5145a1616da6d with: github-token: ${{ secrets.github_token }} path-to-lcov: coverage.lcov From 3eddb768d070a5f510fa030fcb7f1e5d0d409039 Mon Sep 17 00:00:00 2001 From: "dependabot[bot]" <49699333+dependabot[bot]@users.noreply.github.com> Date: Fri, 12 May 2023 22:27:33 -0400 Subject: [PATCH 30/35] Bump actions/setup-go from 3.5.0 to 4.0.0 (#319) Bumps [actions/setup-go](https://github.com/actions/setup-go) from 3.5.0 to 4.0.0. - [Release notes](https://github.com/actions/setup-go/releases) - [Commits](https://github.com/actions/setup-go/compare/6edd4406fa81c3da01a34fa6f6343087c207a568...4d34df0c2316fe8122ab82dc22947d607c0c91f9) --- updated-dependencies: - dependency-name: actions/setup-go dependency-type: direct:production update-type: version-update:semver-major ... Signed-off-by: dependabot[bot] Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com> --- .github/workflows/unit_tests.yaml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/.github/workflows/unit_tests.yaml b/.github/workflows/unit_tests.yaml index b8bc1a9..a952d4e 100644 --- a/.github/workflows/unit_tests.yaml +++ b/.github/workflows/unit_tests.yaml @@ -7,7 +7,7 @@ jobs: steps: - uses: actions/checkout@61b9e3751b92087fd0b06925ba6dd6314e06f089 - name: Setup go - uses: actions/setup-go@6edd4406fa81c3da01a34fa6f6343087c207a568 + uses: actions/setup-go@4d34df0c2316fe8122ab82dc22947d607c0c91f9 with: go-version: 1.19 - uses: actions/cache@6998d139ddd3e68c71e9e398d8e40b71a2f39812 From 67ffbbc0de62c0528cf16086d71b335a6c72f71a Mon Sep 17 00:00:00 2001 From: "dependabot[bot]" <49699333+dependabot[bot]@users.noreply.github.com> Date: Fri, 12 May 2023 22:27:45 -0400 Subject: [PATCH 31/35] Bump actions/cache from 3.2.5 to 3.3.1 (#318) Bumps [actions/cache](https://github.com/actions/cache) from 3.2.5 to 3.3.1. - [Release notes](https://github.com/actions/cache/releases) - [Changelog](https://github.com/actions/cache/blob/main/RELEASES.md) - [Commits](https://github.com/actions/cache/compare/6998d139ddd3e68c71e9e398d8e40b71a2f39812...88522ab9f39a2ea568f7027eddc7d8d8bc9d59c8) --- updated-dependencies: - dependency-name: actions/cache dependency-type: direct:production update-type: version-update:semver-minor ... Signed-off-by: dependabot[bot] Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com> --- .github/workflows/unit_tests.yaml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/.github/workflows/unit_tests.yaml b/.github/workflows/unit_tests.yaml index a952d4e..0edc322 100644 --- a/.github/workflows/unit_tests.yaml +++ b/.github/workflows/unit_tests.yaml @@ -10,7 +10,7 @@ jobs: uses: actions/setup-go@4d34df0c2316fe8122ab82dc22947d607c0c91f9 with: go-version: 1.19 - - uses: actions/cache@6998d139ddd3e68c71e9e398d8e40b71a2f39812 + - uses: actions/cache@88522ab9f39a2ea568f7027eddc7d8d8bc9d59c8 with: path: | ~/go/pkg/mod From 0feadd74ba91cd0d5f5e0c0e013e5fd449e97177 Mon Sep 17 00:00:00 2001 From: "dependabot[bot]" <49699333+dependabot[bot]@users.noreply.github.com> Date: Fri, 12 May 2023 22:27:57 -0400 Subject: [PATCH 32/35] Bump github.com/stretchr/testify from 1.8.1 to 1.8.2 (#311) Bumps [github.com/stretchr/testify](https://github.com/stretchr/testify) from 1.8.1 to 1.8.2. - [Release notes](https://github.com/stretchr/testify/releases) - [Commits](https://github.com/stretchr/testify/compare/v1.8.1...v1.8.2) --- updated-dependencies: - dependency-name: github.com/stretchr/testify dependency-type: direct:production update-type: version-update:semver-patch ... Signed-off-by: dependabot[bot] Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com> --- go.mod | 2 +- go.sum | 4 ++-- 2 files changed, 3 insertions(+), 3 deletions(-) diff --git a/go.mod b/go.mod index b5bce9c..927bbe7 100644 --- a/go.mod +++ b/go.mod @@ -4,7 +4,7 @@ go 1.17 require ( github.com/hashicorp/go-multierror v1.1.1 - github.com/stretchr/testify v1.8.1 + github.com/stretchr/testify v1.8.2 ) require ( diff --git a/go.sum b/go.sum index 61372e7..1a70df0 100644 --- a/go.sum +++ b/go.sum @@ -12,8 +12,8 @@ github.com/stretchr/objx v0.4.0/go.mod h1:YvHI0jy2hoMjB+UWwv71VJQ9isScKT/TqJzVSS github.com/stretchr/objx v0.5.0/go.mod h1:Yh+to48EsGEfYuaHDzXPcE3xhTkx73EhmCGUpEOglKo= github.com/stretchr/testify v1.7.1/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg= github.com/stretchr/testify v1.8.0/go.mod h1:yNjHg4UonilssWZ8iaSj1OCr/vHnekPRkoO+kdMU+MU= -github.com/stretchr/testify v1.8.1 h1:w7B6lhMri9wdJUVmEZPGGhZzrYTPvgJArz7wNPgYKsk= -github.com/stretchr/testify v1.8.1/go.mod h1:w2LPCIKwWwSfY2zedu0+kehJoqGctiVI29o6fzry7u4= +github.com/stretchr/testify v1.8.2 h1:+h33VjcLVPDHtOdpUCuF+7gSuG3yGIftsP1YvFihtJ8= +github.com/stretchr/testify v1.8.2/go.mod h1:w2LPCIKwWwSfY2zedu0+kehJoqGctiVI29o6fzry7u4= gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405 h1:yhCVgyC4o1eVCa2tZl7eS0r+SDo693bJlVdllGtEeKM= gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= gopkg.in/yaml.v3 v3.0.0-20200313102051-9f266ea9e77c/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM= From 36e873b5133014eac5792e69342c0afe70185fa7 Mon Sep 17 00:00:00 2001 From: Nick Craig-Wood Date: Sat, 13 May 2023 03:29:37 +0100 Subject: [PATCH 33/35] Fix GetEntry / MLST for some servers - Fixes #321 (#322) Some servers seem to send a blank line at the end of an MLST response. MLST Download 250- Size=53248;Modify=20230327134339.000;Type=dir; Download 250 Requested file action okay, completed. Before this change this would cause the GetEntry method to return this error. unsupported LIST line This patch ignores blank lines in the MLST response. --- conn_test.go | 2 +- ftp.go | 4 ++++ 2 files changed, 5 insertions(+), 1 deletion(-) diff --git a/conn_test.go b/conn_test.go index 7d3c299..ab26840 100644 --- a/conn_test.go +++ b/conn_test.go @@ -193,7 +193,7 @@ func (mock *ftpMock) listen() { if cmdParts[1] == "multiline-dir" { mock.printfLine("250-File data\r\n Type=dir;Size=0; multiline-dir\r\n Modify=20201213202400; multiline-dir\r\n250 End") } else { - mock.printfLine("250-File data\r\n Type=file;Size=42;Modify=20201213202400; magic-file\r\n250 End") + mock.printfLine("250-File data\r\n Type=file;Size=42;Modify=20201213202400; magic-file\r\n \r\n250 End") } case "NLST": if mock.dataConn == nil { diff --git a/ftp.go b/ftp.go index bbc1799..9892882 100644 --- a/ftp.go +++ b/ftp.go @@ -766,6 +766,10 @@ func (c *ServerConn) GetEntry(path string) (entry *Entry, err error) { if len(l) > 0 && l[0] == ' ' { l = l[1:] } + // Some severs seem to send a blank line at the end which we ignore + if l == "" { + continue + } if e, err = parseNextRFC3659ListLine(l, c.options.location, e); err != nil { return nil, err } From f5852338bb59978cb0e79c5aff9bac2f59c8d36a Mon Sep 17 00:00:00 2001 From: "dependabot[bot]" <49699333+dependabot[bot]@users.noreply.github.com> Date: Tue, 23 May 2023 17:29:21 -0400 Subject: [PATCH 34/35] Bump actions/setup-go from 4.0.0 to 4.0.1 (#328) Bumps [actions/setup-go](https://github.com/actions/setup-go) from 4.0.0 to 4.0.1. - [Release notes](https://github.com/actions/setup-go/releases) - [Commits](https://github.com/actions/setup-go/compare/4d34df0c2316fe8122ab82dc22947d607c0c91f9...fac708d6674e30b6ba41289acaab6d4b75aa0753) --- updated-dependencies: - dependency-name: actions/setup-go dependency-type: direct:production update-type: version-update:semver-patch ... Signed-off-by: dependabot[bot] Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com> --- .github/workflows/unit_tests.yaml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/.github/workflows/unit_tests.yaml b/.github/workflows/unit_tests.yaml index 0edc322..5c1bde7 100644 --- a/.github/workflows/unit_tests.yaml +++ b/.github/workflows/unit_tests.yaml @@ -7,7 +7,7 @@ jobs: steps: - uses: actions/checkout@61b9e3751b92087fd0b06925ba6dd6314e06f089 - name: Setup go - uses: actions/setup-go@4d34df0c2316fe8122ab82dc22947d607c0c91f9 + uses: actions/setup-go@fac708d6674e30b6ba41289acaab6d4b75aa0753 with: go-version: 1.19 - uses: actions/cache@88522ab9f39a2ea568f7027eddc7d8d8bc9d59c8 From 2c2aa379fd3e5e6b6126576a19fd19da9e091c19 Mon Sep 17 00:00:00 2001 From: "dependabot[bot]" <49699333+dependabot[bot]@users.noreply.github.com> Date: Tue, 23 May 2023 17:29:40 -0400 Subject: [PATCH 35/35] Bump github.com/stretchr/testify from 1.8.2 to 1.8.3 (#330) Bumps [github.com/stretchr/testify](https://github.com/stretchr/testify) from 1.8.2 to 1.8.3. - [Release notes](https://github.com/stretchr/testify/releases) - [Commits](https://github.com/stretchr/testify/compare/v1.8.2...v1.8.3) --- updated-dependencies: - dependency-name: github.com/stretchr/testify dependency-type: direct:production update-type: version-update:semver-patch ... Signed-off-by: dependabot[bot] Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com> --- go.mod | 2 +- go.sum | 4 ++-- 2 files changed, 3 insertions(+), 3 deletions(-) diff --git a/go.mod b/go.mod index 927bbe7..d8b3596 100644 --- a/go.mod +++ b/go.mod @@ -4,7 +4,7 @@ go 1.17 require ( github.com/hashicorp/go-multierror v1.1.1 - github.com/stretchr/testify v1.8.2 + github.com/stretchr/testify v1.8.3 ) require ( diff --git a/go.sum b/go.sum index 1a70df0..b3e94cf 100644 --- a/go.sum +++ b/go.sum @@ -12,8 +12,8 @@ github.com/stretchr/objx v0.4.0/go.mod h1:YvHI0jy2hoMjB+UWwv71VJQ9isScKT/TqJzVSS github.com/stretchr/objx v0.5.0/go.mod h1:Yh+to48EsGEfYuaHDzXPcE3xhTkx73EhmCGUpEOglKo= github.com/stretchr/testify v1.7.1/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg= github.com/stretchr/testify v1.8.0/go.mod h1:yNjHg4UonilssWZ8iaSj1OCr/vHnekPRkoO+kdMU+MU= -github.com/stretchr/testify v1.8.2 h1:+h33VjcLVPDHtOdpUCuF+7gSuG3yGIftsP1YvFihtJ8= -github.com/stretchr/testify v1.8.2/go.mod h1:w2LPCIKwWwSfY2zedu0+kehJoqGctiVI29o6fzry7u4= +github.com/stretchr/testify v1.8.3 h1:RP3t2pwF7cMEbC1dqtB6poj3niw/9gnV4Cjg5oW5gtY= +github.com/stretchr/testify v1.8.3/go.mod h1:sz/lmYIOXD/1dqDmKjjqLyZ2RngseejIcXlSw2iwfAo= gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405 h1:yhCVgyC4o1eVCa2tZl7eS0r+SDo693bJlVdllGtEeKM= gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= gopkg.in/yaml.v3 v3.0.0-20200313102051-9f266ea9e77c/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM=