package ftp import ( "bytes" "fmt" "io/ioutil" "net" "net/textproto" "strings" "syscall" "testing" "time" "github.com/stretchr/testify/assert" ) const ( testData = "Just some text" testDir = "mydir" ) func TestConnPASV(t *testing.T) { testConn(t, true) } func TestConnEPSV(t *testing.T) { testConn(t, false) } func testConn(t *testing.T, disableEPSV bool) { 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) } err = c.NoOp() if err != nil { t.Error(err) } err = c.ChangeDir("incoming") if err != nil { t.Error(err) } dir, err := c.CurrentDir() if err != nil { t.Error(err) } else { if dir != "/incoming" { t.Error("Wrong dir: " + dir) } } data := bytes.NewBufferString(testData) err = c.Stor("test", data) if err != nil { t.Error(err) } _, err = c.List(".") if err != nil { t.Error(err) } err = c.Rename("test", "tset") if err != nil { t.Error(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) } 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 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) } 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) } r.Close() } data2 := bytes.NewBufferString(testData) err = c.Append("tset", data2) if err != nil { t.Error(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) } 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) } _, err = c.FileSize("not-found") if err == nil { t.Fatal("expected error, got nil") } err = c.Delete("tset") if err != nil { t.Error(err) } err = c.MakeDir(testDir) if err != nil { t.Error(err) } err = c.ChangeDir(testDir) if err != nil { t.Error(err) } err = c.ChangeDirToParent() if err != nil { t.Error(err) } entries, err := c.NameList("/") if err != nil { t.Error(err) } if len(entries) != 1 || entries[0] != "/incoming" { t.Errorf("Unexpected entries: %v", entries) } err = c.RemoveDir(testDir) if err != nil { t.Error(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) } } if err = c.Quit(); err != nil { t.Fatal(err) } // Wait for the connection to close mock.Wait() err = c.NoOp() if err == nil { t.Error("Expected error") } } // TestConnect tests the legacy Connect function func TestConnect(t *testing.T) { mock, err := newFtpMock(t, "127.0.0.1") if err != nil { t.Fatal(err) } defer mock.Close() c, err := Connect(mock.Addr()) if err != nil { t.Fatal(err) } if err := c.Quit(); err != nil { t.Fatal(err) } mock.Wait() } func TestTimeout(t *testing.T) { if testing.Short() { t.Skip("skipping test in short mode.") } if c, err := DialTimeout("localhost:2121", 1*time.Second); err == nil { _ = c.Quit() t.Fatal("expected timeout, got nil error") } } func TestWrongLogin(t *testing.T) { mock, err := newFtpMock(t, "127.0.0.1") if err != nil { t.Fatal(err) } defer mock.Close() c, err := DialTimeout(mock.Addr(), 5*time.Second) if err != nil { t.Fatal(err) } defer func() { if err := c.Quit(); err != nil { t.Errorf("can not quit: %s", err) } }() err = c.Login("zoo2Shia", "fei5Yix9") if err == nil { t.Fatal("expected error, got nil") } } func TestDeleteDirRecur(t *testing.T) { mock, c := openConn(t, "127.0.0.1") err := c.RemoveDirRecur("testDir") if err != nil { t.Error(err) } if err := c.Quit(); err != nil { t.Fatal(err) } // Wait for the connection to close mock.Wait() } // func TestFileDeleteDirRecur(t *testing.T) { // mock, c := openConn(t, "127.0.0.1") // err := c.RemoveDirRecur("testFile") // if err == nil { // t.Fatal("expected error got nil") // } // if err := c.Quit(); err != nil { // t.Fatal(err) // } // // Wait for the connection to close // mock.Wait() // } func TestMissingFolderDeleteDirRecur(t *testing.T) { mock, c := openConn(t, "127.0.0.1") err := c.RemoveDirRecur("missing-dir") if err == nil { t.Fatal("expected error got nil") } if err := c.Quit(); err != nil { t.Fatal(err) } // Wait for the connection to close mock.Wait() } func TestListCurrentDir(t *testing.T) { mock, c := openConn(t, "127.0.0.1") _, err := c.List("") assert.NoError(t, err) assert.Equal(t, "LIST", mock.lastFull, "LIST must not have a trailing whitespace") _, err = c.NameList("") assert.NoError(t, err) assert.Equal(t, "NLST", mock.lastFull, "NLST 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") assert.False(t, c.mdtmSupported, "MDTM must NOT be supported") assert.False(t, c.mfmtSupported, "MFMT must NOT be supported") assert.False(t, c.IsGetTimeSupported(), "GetTime must NOT be supported") assert.False(t, c.IsSetTimeSupported(), "SetTime must NOT be supported") _, err := c.GetTime("file1") assert.NotNil(t, err) err = c.SetTime("file1", time.Now()) assert.NotNil(t, err) assert.NoError(t, c.Quit()) mock.Wait() } func TestTimeStandard(t *testing.T) { mock, c := openConnExt(t, "127.0.0.1", "std-time") assert.True(t, c.mdtmSupported, "MDTM must be supported") assert.True(t, c.mfmtSupported, "MFMT must be supported") assert.True(t, c.IsGetTimeSupported(), "GetTime must be supported") assert.True(t, c.IsSetTimeSupported(), "SetTime must be supported") tm, err := c.GetTime("file1") assert.NoError(t, err) assert.False(t, tm.IsZero(), "GetTime must return valid time") err = c.SetTime("file1", time.Now()) assert.NoError(t, err) assert.NoError(t, c.Quit()) mock.Wait() } func TestTimeVsftpdPartial(t *testing.T) { mock, c := openConnExt(t, "127.0.0.1", "vsftpd") assert.True(t, c.mdtmSupported, "MDTM must be supported") assert.False(t, c.mfmtSupported, "MFMT must NOT be supported") assert.True(t, c.IsGetTimeSupported(), "GetTime must be supported") assert.False(t, c.IsSetTimeSupported(), "SetTime must NOT be supported") tm, err := c.GetTime("file1") assert.NoError(t, err) assert.False(t, tm.IsZero(), "GetTime must return valid time") err = c.SetTime("file1", time.Now()) assert.NotNil(t, err) assert.NoError(t, c.Quit()) mock.Wait() } func TestTimeVsftpdFull(t *testing.T) { mock, c := openConnExt(t, "127.0.0.1", "vsftpd", DialWithWritingMDTM(true)) assert.True(t, c.mdtmSupported, "MDTM must be supported") assert.False(t, c.mfmtSupported, "MFMT must NOT be supported") assert.True(t, c.IsGetTimeSupported(), "GetTime must be supported") assert.True(t, c.IsSetTimeSupported(), "SetTime must be supported") tm, err := c.GetTime("file1") assert.NoError(t, err) assert.False(t, tm.IsZero(), "GetTime must return valid time") err = c.SetTime("file1", time.Now()) assert.NoError(t, err) assert.NoError(t, c.Quit()) mock.Wait() } func TestDialWithDialFunc(t *testing.T) { dialErr := fmt.Errorf("this is proof that dial function was called") f := func(network, address string) (net.Conn, error) { return nil, dialErr } _, err := Dial("bogus-address", DialWithDialFunc(f)) assert.Equal(t, dialErr, err) } func TestDialWithDialer(t *testing.T) { dialerCalled := false dialer := net.Dialer{ Control: func(network, address string, c syscall.RawConn) error { dialerCalled = true return nil }, } mock, err := newFtpMock(t, "127.0.0.1") assert.NoError(t, err) c, err := Dial(mock.Addr(), DialWithDialer(dialer)) assert.NoError(t, err) assert.NoError(t, c.Quit()) assert.Equal(t, true, dialerCalled) }