2013-11-06 12:40:35 +08:00
|
|
|
package sftp
|
|
|
|
|
|
|
|
// sftp integration tests
|
|
|
|
// enable with -integration
|
|
|
|
|
|
|
|
import (
|
|
|
|
"flag"
|
|
|
|
"io/ioutil"
|
|
|
|
"os"
|
|
|
|
"os/exec"
|
|
|
|
"testing"
|
|
|
|
)
|
|
|
|
|
2013-11-06 16:29:59 +08:00
|
|
|
const (
|
|
|
|
READONLY = true
|
|
|
|
READWRITE = false
|
|
|
|
)
|
|
|
|
|
2013-11-06 12:40:35 +08:00
|
|
|
var testIntegration = flag.Bool("integration", false, "perform integration tests against sftp server process")
|
|
|
|
|
|
|
|
// testClient returns a *Client connected to a localy running sftp-server
|
|
|
|
// the *exec.Cmd returned must be defer Wait'd.
|
2013-11-06 16:29:59 +08:00
|
|
|
func testClient(t *testing.T, readonly bool) (*Client, *exec.Cmd) {
|
2013-11-06 12:40:35 +08:00
|
|
|
cmd := exec.Command("/usr/lib/openssh/sftp-server", "-e", "-R") // log to stderr, read only
|
2013-11-06 16:29:59 +08:00
|
|
|
if !readonly {
|
|
|
|
cmd = exec.Command("/usr/lib/openssh/sftp-server", "-e") // log to stderr
|
|
|
|
}
|
2013-11-06 12:40:35 +08:00
|
|
|
cmd.Stderr = os.Stdout
|
|
|
|
pw, err := cmd.StdinPipe()
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
pr, err := cmd.StdoutPipe()
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
if err := cmd.Start(); err != nil {
|
|
|
|
t.Skipf("could not start sftp-server process: %v", err)
|
|
|
|
}
|
|
|
|
sftp := &Client{
|
|
|
|
w: pw,
|
|
|
|
r: pr,
|
|
|
|
}
|
|
|
|
if err := sftp.sendInit(); err != nil {
|
|
|
|
defer cmd.Wait()
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
if err := sftp.recvVersion(); err != nil {
|
|
|
|
defer cmd.Wait()
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
return sftp, cmd
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestNewClient(t *testing.T) {
|
2013-11-06 16:29:59 +08:00
|
|
|
sftp, cmd := testClient(t, READONLY)
|
2013-11-06 12:40:35 +08:00
|
|
|
defer cmd.Wait()
|
|
|
|
|
|
|
|
if err := sftp.Close(); err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestClientLstat(t *testing.T) {
|
2013-11-06 16:29:59 +08:00
|
|
|
sftp, cmd := testClient(t, READONLY)
|
2013-11-06 12:40:35 +08:00
|
|
|
defer cmd.Wait()
|
|
|
|
defer sftp.Close()
|
|
|
|
|
|
|
|
f, err := ioutil.TempFile("", "sftptest")
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
defer os.Remove(f.Name())
|
|
|
|
|
|
|
|
want, err := os.Lstat(f.Name())
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
got, err := sftp.Lstat(f.Name())
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
if !sameFile(want, got) {
|
|
|
|
t.Fatalf("Lstat(%q): want %#v, got %#v", f.Name(), want, got)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-11-06 16:29:59 +08:00
|
|
|
func TestClientLstatiMissing(t *testing.T) {
|
|
|
|
sftp, cmd := testClient(t, READONLY)
|
|
|
|
defer cmd.Wait()
|
|
|
|
defer sftp.Close()
|
|
|
|
|
|
|
|
f, err := ioutil.TempFile("", "sftptest")
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
os.Remove(f.Name())
|
|
|
|
|
|
|
|
_, err = sftp.Lstat(f.Name())
|
|
|
|
if err1, ok := err.(*StatusError); !ok || err1.Code != ssh_FX_NO_SUCH_FILE {
|
|
|
|
t.Fatalf("Lstat: want: %v, got %#v", ssh_FX_NO_SUCH_FILE, err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-11-06 12:53:14 +08:00
|
|
|
func TestClientOpen(t *testing.T) {
|
2013-11-06 16:29:59 +08:00
|
|
|
sftp, cmd := testClient(t, READONLY)
|
2013-11-06 12:53:14 +08:00
|
|
|
defer cmd.Wait()
|
|
|
|
defer sftp.Close()
|
|
|
|
|
|
|
|
f, err := ioutil.TempFile("", "sftptest")
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
defer os.Remove(f.Name())
|
|
|
|
|
|
|
|
got, err := sftp.Open(f.Name())
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
if err := got.Close(); err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestClientRead(t *testing.T) {
|
2013-11-06 16:29:59 +08:00
|
|
|
sftp, cmd := testClient(t, READONLY)
|
2013-11-06 12:53:14 +08:00
|
|
|
defer cmd.Wait()
|
|
|
|
defer sftp.Close()
|
|
|
|
|
|
|
|
f, err := ioutil.TempFile("", "sftptest")
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
defer os.Remove(f.Name())
|
|
|
|
|
|
|
|
if _, err := f.WriteString("Hello world!"); err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
if err := f.Close(); err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
got, err := sftp.Open(f.Name())
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
defer got.Close()
|
|
|
|
|
|
|
|
b, err := ioutil.ReadAll(got)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
if want, got := "Hello world!", string(b); got != want {
|
|
|
|
t.Fatalf("Read(): want %q, got %q", want, got)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-11-06 16:29:59 +08:00
|
|
|
func TestClientCreate(t *testing.T) {
|
|
|
|
sftp, cmd := testClient(t, READWRITE)
|
|
|
|
defer cmd.Wait()
|
|
|
|
defer sftp.Close()
|
|
|
|
|
|
|
|
f, err := ioutil.TempFile("", "sftptest")
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
defer f.Close()
|
|
|
|
defer os.Remove(f.Name())
|
|
|
|
|
|
|
|
f2, err := sftp.Create(f.Name())
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
defer f2.Close()
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestClientCreateFailed(t *testing.T) {
|
|
|
|
sftp, cmd := testClient(t, READONLY)
|
|
|
|
defer cmd.Wait()
|
|
|
|
defer sftp.Close()
|
|
|
|
|
|
|
|
f, err := ioutil.TempFile("", "sftptest")
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
defer f.Close()
|
|
|
|
defer os.Remove(f.Name())
|
|
|
|
|
|
|
|
f2, err := sftp.Create(f.Name())
|
|
|
|
if err1, ok := err.(*StatusError); !ok || err1.Code != ssh_FX_PERMISSION_DENIED {
|
|
|
|
t.Fatalf("Create: want: %v, got %#v", ssh_FX_PERMISSION_DENIED, err)
|
|
|
|
}
|
|
|
|
if err == nil {
|
|
|
|
f2.Close()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-11-06 13:03:08 +08:00
|
|
|
func TestClientFileStat(t *testing.T) {
|
2013-11-06 16:29:59 +08:00
|
|
|
sftp, cmd := testClient(t, READONLY)
|
2013-11-06 13:03:08 +08:00
|
|
|
defer cmd.Wait()
|
|
|
|
defer sftp.Close()
|
|
|
|
|
|
|
|
f, err := ioutil.TempFile("", "sftptest")
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
defer os.Remove(f.Name())
|
|
|
|
|
|
|
|
want, err := os.Lstat(f.Name())
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
f2, err := sftp.Open(f.Name())
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
got, err := f2.Stat()
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
if !sameFile(want, got) {
|
|
|
|
t.Fatalf("Lstat(%q): want %#v, got %#v", f.Name(), want, got)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-11-06 16:29:59 +08:00
|
|
|
func TestClientWalkdir(t *testing.T) {
|
|
|
|
sftp, cmd := testClient(t, READONLY)
|
|
|
|
defer cmd.Wait()
|
|
|
|
defer sftp.Close()
|
|
|
|
|
|
|
|
d, err := ioutil.TempDir("", "sftptest")
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
defer os.Remove(d)
|
|
|
|
|
|
|
|
w := sftp.Walk(d)
|
|
|
|
for w.Step() {
|
|
|
|
if err := w.Err(); err != nil {
|
|
|
|
t.Error(err)
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
t.Log(w.Path())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-11-06 12:40:35 +08:00
|
|
|
func sameFile(want, got os.FileInfo) bool {
|
|
|
|
return want.Name() == got.Name() &&
|
|
|
|
want.Size() == got.Size()
|
|
|
|
}
|