|
|
@@ -0,0 +1,102 @@
|
|
|
+// Copyright 2013 The Go Authors. All rights reserved.
|
|
|
+// Use of this source code is governed by a BSD-style
|
|
|
+// license that can be found in the LICENSE file.
|
|
|
+
|
|
|
+package ssh
|
|
|
+
|
|
|
+import (
|
|
|
+ "io"
|
|
|
+ "sync"
|
|
|
+ "testing"
|
|
|
+)
|
|
|
+
|
|
|
+// An in-memory packetConn. It is safe to call Close and writePacket
|
|
|
+// from different goroutines.
|
|
|
+type memTransport struct {
|
|
|
+ eof bool
|
|
|
+ pending [][]byte
|
|
|
+ write *memTransport
|
|
|
+ sync.Mutex
|
|
|
+ *sync.Cond
|
|
|
+}
|
|
|
+
|
|
|
+func (t *memTransport) readPacket() ([]byte, error) {
|
|
|
+ t.Lock()
|
|
|
+ defer t.Unlock()
|
|
|
+ for {
|
|
|
+ if len(t.pending) > 0 {
|
|
|
+ r := t.pending[0]
|
|
|
+ t.pending = t.pending[1:]
|
|
|
+ return r, nil
|
|
|
+ }
|
|
|
+ if t.eof {
|
|
|
+ return nil, io.EOF
|
|
|
+ }
|
|
|
+ t.Cond.Wait()
|
|
|
+ }
|
|
|
+}
|
|
|
+
|
|
|
+func (t *memTransport) Close() error {
|
|
|
+ t.write.Lock()
|
|
|
+ defer t.write.Unlock()
|
|
|
+ if t.write.eof {
|
|
|
+ return io.EOF
|
|
|
+ }
|
|
|
+ t.write.eof = true
|
|
|
+ t.write.Cond.Broadcast()
|
|
|
+ return nil
|
|
|
+}
|
|
|
+
|
|
|
+func (t *memTransport) writePacket(p []byte) error {
|
|
|
+ t.write.Lock()
|
|
|
+ defer t.write.Unlock()
|
|
|
+ if t.write.eof {
|
|
|
+ return io.EOF
|
|
|
+ }
|
|
|
+ t.write.pending = append(t.write.pending, p)
|
|
|
+ t.write.Cond.Signal()
|
|
|
+ return nil
|
|
|
+}
|
|
|
+
|
|
|
+func memPipe() (a, b packetConn) {
|
|
|
+ t1 := memTransport{}
|
|
|
+ t2 := memTransport{}
|
|
|
+ t1.write = &t2
|
|
|
+ t2.write = &t1
|
|
|
+ t1.Cond = sync.NewCond(&t1.Mutex)
|
|
|
+ t2.Cond = sync.NewCond(&t2.Mutex)
|
|
|
+ return &t1, &t2
|
|
|
+}
|
|
|
+
|
|
|
+func TestmemPipe(t *testing.T) {
|
|
|
+ a, b := memPipe()
|
|
|
+ if err := a.writePacket([]byte{42}); err != nil {
|
|
|
+ t.Fatalf("writePacket: %v", err)
|
|
|
+ }
|
|
|
+ if err := a.Close(); err != nil {
|
|
|
+ t.Fatal("Close: ", err)
|
|
|
+ }
|
|
|
+ p, err := b.readPacket()
|
|
|
+ if err != nil {
|
|
|
+ t.Fatal("readPacket: ", err)
|
|
|
+ }
|
|
|
+ if len(p) != 1 || p[0] != 42 {
|
|
|
+ t.Fatalf("got %v, want {42}", p)
|
|
|
+ }
|
|
|
+ p, err = b.readPacket()
|
|
|
+ if err != io.EOF {
|
|
|
+ t.Fatalf("got %v, %v, want EOF", p, err)
|
|
|
+ }
|
|
|
+}
|
|
|
+
|
|
|
+func TestDoubleClose(t *testing.T) {
|
|
|
+ a, _ := memPipe()
|
|
|
+ err := a.Close()
|
|
|
+ if err != nil {
|
|
|
+ t.Errorf("Close: %v", err)
|
|
|
+ }
|
|
|
+ err = a.Close()
|
|
|
+ if err != io.EOF {
|
|
|
+ t.Errorf("expect EOF on double close.")
|
|
|
+ }
|
|
|
+}
|