Платформа ЦРНП "Мирокод" для разработки проектов
https://git.mirocod.ru
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
525 lines
11 KiB
525 lines
11 KiB
// 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" |
|
"io/ioutil" |
|
"sync" |
|
"testing" |
|
) |
|
|
|
func muxPair() (*mux, *mux) { |
|
a, b := memPipe() |
|
|
|
s := newMux(a) |
|
c := newMux(b) |
|
|
|
return s, c |
|
} |
|
|
|
// Returns both ends of a channel, and the mux for the the 2nd |
|
// channel. |
|
func channelPair(t *testing.T) (*channel, *channel, *mux) { |
|
c, s := muxPair() |
|
|
|
res := make(chan *channel, 1) |
|
go func() { |
|
newCh, ok := <-s.incomingChannels |
|
if !ok { |
|
t.Fatalf("No incoming channel") |
|
} |
|
if newCh.ChannelType() != "chan" { |
|
t.Fatalf("got type %q want chan", newCh.ChannelType()) |
|
} |
|
ch, _, err := newCh.Accept() |
|
if err != nil { |
|
t.Fatalf("Accept %v", err) |
|
} |
|
res <- ch.(*channel) |
|
}() |
|
|
|
ch, err := c.openChannel("chan", nil) |
|
if err != nil { |
|
t.Fatalf("OpenChannel: %v", err) |
|
} |
|
|
|
return <-res, ch, c |
|
} |
|
|
|
// Test that stderr and stdout can be addressed from different |
|
// goroutines. This is intended for use with the race detector. |
|
func TestMuxChannelExtendedThreadSafety(t *testing.T) { |
|
writer, reader, mux := channelPair(t) |
|
defer writer.Close() |
|
defer reader.Close() |
|
defer mux.Close() |
|
|
|
var wr, rd sync.WaitGroup |
|
magic := "hello world" |
|
|
|
wr.Add(2) |
|
go func() { |
|
io.WriteString(writer, magic) |
|
wr.Done() |
|
}() |
|
go func() { |
|
io.WriteString(writer.Stderr(), magic) |
|
wr.Done() |
|
}() |
|
|
|
rd.Add(2) |
|
go func() { |
|
c, err := ioutil.ReadAll(reader) |
|
if string(c) != magic { |
|
t.Fatalf("stdout read got %q, want %q (error %s)", c, magic, err) |
|
} |
|
rd.Done() |
|
}() |
|
go func() { |
|
c, err := ioutil.ReadAll(reader.Stderr()) |
|
if string(c) != magic { |
|
t.Fatalf("stderr read got %q, want %q (error %s)", c, magic, err) |
|
} |
|
rd.Done() |
|
}() |
|
|
|
wr.Wait() |
|
writer.CloseWrite() |
|
rd.Wait() |
|
} |
|
|
|
func TestMuxReadWrite(t *testing.T) { |
|
s, c, mux := channelPair(t) |
|
defer s.Close() |
|
defer c.Close() |
|
defer mux.Close() |
|
|
|
magic := "hello world" |
|
magicExt := "hello stderr" |
|
go func() { |
|
_, err := s.Write([]byte(magic)) |
|
if err != nil { |
|
t.Fatalf("Write: %v", err) |
|
} |
|
_, err = s.Extended(1).Write([]byte(magicExt)) |
|
if err != nil { |
|
t.Fatalf("Write: %v", err) |
|
} |
|
err = s.Close() |
|
if err != nil { |
|
t.Fatalf("Close: %v", err) |
|
} |
|
}() |
|
|
|
var buf [1024]byte |
|
n, err := c.Read(buf[:]) |
|
if err != nil { |
|
t.Fatalf("server Read: %v", err) |
|
} |
|
got := string(buf[:n]) |
|
if got != magic { |
|
t.Fatalf("server: got %q want %q", got, magic) |
|
} |
|
|
|
n, err = c.Extended(1).Read(buf[:]) |
|
if err != nil { |
|
t.Fatalf("server Read: %v", err) |
|
} |
|
|
|
got = string(buf[:n]) |
|
if got != magicExt { |
|
t.Fatalf("server: got %q want %q", got, magic) |
|
} |
|
} |
|
|
|
func TestMuxChannelOverflow(t *testing.T) { |
|
reader, writer, mux := channelPair(t) |
|
defer reader.Close() |
|
defer writer.Close() |
|
defer mux.Close() |
|
|
|
wDone := make(chan int, 1) |
|
go func() { |
|
if _, err := writer.Write(make([]byte, channelWindowSize)); err != nil { |
|
t.Errorf("could not fill window: %v", err) |
|
} |
|
writer.Write(make([]byte, 1)) |
|
wDone <- 1 |
|
}() |
|
writer.remoteWin.waitWriterBlocked() |
|
|
|
// Send 1 byte. |
|
packet := make([]byte, 1+4+4+1) |
|
packet[0] = msgChannelData |
|
marshalUint32(packet[1:], writer.remoteId) |
|
marshalUint32(packet[5:], uint32(1)) |
|
packet[9] = 42 |
|
|
|
if err := writer.mux.conn.writePacket(packet); err != nil { |
|
t.Errorf("could not send packet") |
|
} |
|
if _, err := reader.SendRequest("hello", true, nil); err == nil { |
|
t.Errorf("SendRequest succeeded.") |
|
} |
|
<-wDone |
|
} |
|
|
|
func TestMuxChannelCloseWriteUnblock(t *testing.T) { |
|
reader, writer, mux := channelPair(t) |
|
defer reader.Close() |
|
defer writer.Close() |
|
defer mux.Close() |
|
|
|
wDone := make(chan int, 1) |
|
go func() { |
|
if _, err := writer.Write(make([]byte, channelWindowSize)); err != nil { |
|
t.Errorf("could not fill window: %v", err) |
|
} |
|
if _, err := writer.Write(make([]byte, 1)); err != io.EOF { |
|
t.Errorf("got %v, want EOF for unblock write", err) |
|
} |
|
wDone <- 1 |
|
}() |
|
|
|
writer.remoteWin.waitWriterBlocked() |
|
reader.Close() |
|
<-wDone |
|
} |
|
|
|
func TestMuxConnectionCloseWriteUnblock(t *testing.T) { |
|
reader, writer, mux := channelPair(t) |
|
defer reader.Close() |
|
defer writer.Close() |
|
defer mux.Close() |
|
|
|
wDone := make(chan int, 1) |
|
go func() { |
|
if _, err := writer.Write(make([]byte, channelWindowSize)); err != nil { |
|
t.Errorf("could not fill window: %v", err) |
|
} |
|
if _, err := writer.Write(make([]byte, 1)); err != io.EOF { |
|
t.Errorf("got %v, want EOF for unblock write", err) |
|
} |
|
wDone <- 1 |
|
}() |
|
|
|
writer.remoteWin.waitWriterBlocked() |
|
mux.Close() |
|
<-wDone |
|
} |
|
|
|
func TestMuxReject(t *testing.T) { |
|
client, server := muxPair() |
|
defer server.Close() |
|
defer client.Close() |
|
|
|
go func() { |
|
ch, ok := <-server.incomingChannels |
|
if !ok { |
|
t.Fatalf("Accept") |
|
} |
|
if ch.ChannelType() != "ch" || string(ch.ExtraData()) != "extra" { |
|
t.Fatalf("unexpected channel: %q, %q", ch.ChannelType(), ch.ExtraData()) |
|
} |
|
ch.Reject(RejectionReason(42), "message") |
|
}() |
|
|
|
ch, err := client.openChannel("ch", []byte("extra")) |
|
if ch != nil { |
|
t.Fatal("openChannel not rejected") |
|
} |
|
|
|
ocf, ok := err.(*OpenChannelError) |
|
if !ok { |
|
t.Errorf("got %#v want *OpenChannelError", err) |
|
} else if ocf.Reason != 42 || ocf.Message != "message" { |
|
t.Errorf("got %#v, want {Reason: 42, Message: %q}", ocf, "message") |
|
} |
|
|
|
want := "ssh: rejected: unknown reason 42 (message)" |
|
if err.Error() != want { |
|
t.Errorf("got %q, want %q", err.Error(), want) |
|
} |
|
} |
|
|
|
func TestMuxChannelRequest(t *testing.T) { |
|
client, server, mux := channelPair(t) |
|
defer server.Close() |
|
defer client.Close() |
|
defer mux.Close() |
|
|
|
var received int |
|
var wg sync.WaitGroup |
|
wg.Add(1) |
|
go func() { |
|
for r := range server.incomingRequests { |
|
received++ |
|
r.Reply(r.Type == "yes", nil) |
|
} |
|
wg.Done() |
|
}() |
|
_, err := client.SendRequest("yes", false, nil) |
|
if err != nil { |
|
t.Fatalf("SendRequest: %v", err) |
|
} |
|
ok, err := client.SendRequest("yes", true, nil) |
|
if err != nil { |
|
t.Fatalf("SendRequest: %v", err) |
|
} |
|
|
|
if !ok { |
|
t.Errorf("SendRequest(yes): %v", ok) |
|
|
|
} |
|
|
|
ok, err = client.SendRequest("no", true, nil) |
|
if err != nil { |
|
t.Fatalf("SendRequest: %v", err) |
|
} |
|
if ok { |
|
t.Errorf("SendRequest(no): %v", ok) |
|
|
|
} |
|
|
|
client.Close() |
|
wg.Wait() |
|
|
|
if received != 3 { |
|
t.Errorf("got %d requests, want %d", received, 3) |
|
} |
|
} |
|
|
|
func TestMuxGlobalRequest(t *testing.T) { |
|
clientMux, serverMux := muxPair() |
|
defer serverMux.Close() |
|
defer clientMux.Close() |
|
|
|
var seen bool |
|
go func() { |
|
for r := range serverMux.incomingRequests { |
|
seen = seen || r.Type == "peek" |
|
if r.WantReply { |
|
err := r.Reply(r.Type == "yes", |
|
append([]byte(r.Type), r.Payload...)) |
|
if err != nil { |
|
t.Errorf("AckRequest: %v", err) |
|
} |
|
} |
|
} |
|
}() |
|
|
|
_, _, err := clientMux.SendRequest("peek", false, nil) |
|
if err != nil { |
|
t.Errorf("SendRequest: %v", err) |
|
} |
|
|
|
ok, data, err := clientMux.SendRequest("yes", true, []byte("a")) |
|
if !ok || string(data) != "yesa" || err != nil { |
|
t.Errorf("SendRequest(\"yes\", true, \"a\"): %v %v %v", |
|
ok, data, err) |
|
} |
|
if ok, data, err := clientMux.SendRequest("yes", true, []byte("a")); !ok || string(data) != "yesa" || err != nil { |
|
t.Errorf("SendRequest(\"yes\", true, \"a\"): %v %v %v", |
|
ok, data, err) |
|
} |
|
|
|
if ok, data, err := clientMux.SendRequest("no", true, []byte("a")); ok || string(data) != "noa" || err != nil { |
|
t.Errorf("SendRequest(\"no\", true, \"a\"): %v %v %v", |
|
ok, data, err) |
|
} |
|
|
|
clientMux.Disconnect(0, "") |
|
if !seen { |
|
t.Errorf("never saw 'peek' request") |
|
} |
|
} |
|
|
|
func TestMuxGlobalRequestUnblock(t *testing.T) { |
|
clientMux, serverMux := muxPair() |
|
defer serverMux.Close() |
|
defer clientMux.Close() |
|
|
|
result := make(chan error, 1) |
|
go func() { |
|
_, _, err := clientMux.SendRequest("hello", true, nil) |
|
result <- err |
|
}() |
|
|
|
<-serverMux.incomingRequests |
|
serverMux.conn.Close() |
|
err := <-result |
|
|
|
if err != io.EOF { |
|
t.Errorf("want EOF, got %v", io.EOF) |
|
} |
|
} |
|
|
|
func TestMuxChannelRequestUnblock(t *testing.T) { |
|
a, b, connB := channelPair(t) |
|
defer a.Close() |
|
defer b.Close() |
|
defer connB.Close() |
|
|
|
result := make(chan error, 1) |
|
go func() { |
|
_, err := a.SendRequest("hello", true, nil) |
|
result <- err |
|
}() |
|
|
|
<-b.incomingRequests |
|
connB.conn.Close() |
|
err := <-result |
|
|
|
if err != io.EOF { |
|
t.Errorf("want EOF, got %v", err) |
|
} |
|
} |
|
|
|
func TestMuxDisconnect(t *testing.T) { |
|
a, b := muxPair() |
|
defer a.Close() |
|
defer b.Close() |
|
|
|
go func() { |
|
for r := range b.incomingRequests { |
|
r.Reply(true, nil) |
|
} |
|
}() |
|
|
|
a.Disconnect(42, "whatever") |
|
ok, _, err := a.SendRequest("hello", true, nil) |
|
if ok || err == nil { |
|
t.Errorf("got reply after disconnecting") |
|
} |
|
err = b.Wait() |
|
if d, ok := err.(*disconnectMsg); !ok || d.Reason != 42 { |
|
t.Errorf("got %#v, want disconnectMsg{Reason:42}", err) |
|
} |
|
} |
|
|
|
func TestMuxCloseChannel(t *testing.T) { |
|
r, w, mux := channelPair(t) |
|
defer mux.Close() |
|
defer r.Close() |
|
defer w.Close() |
|
|
|
result := make(chan error, 1) |
|
go func() { |
|
var b [1024]byte |
|
_, err := r.Read(b[:]) |
|
result <- err |
|
}() |
|
if err := w.Close(); err != nil { |
|
t.Errorf("w.Close: %v", err) |
|
} |
|
|
|
if _, err := w.Write([]byte("hello")); err != io.EOF { |
|
t.Errorf("got err %v, want io.EOF after Close", err) |
|
} |
|
|
|
if err := <-result; err != io.EOF { |
|
t.Errorf("got %v (%T), want io.EOF", err, err) |
|
} |
|
} |
|
|
|
func TestMuxCloseWriteChannel(t *testing.T) { |
|
r, w, mux := channelPair(t) |
|
defer mux.Close() |
|
|
|
result := make(chan error, 1) |
|
go func() { |
|
var b [1024]byte |
|
_, err := r.Read(b[:]) |
|
result <- err |
|
}() |
|
if err := w.CloseWrite(); err != nil { |
|
t.Errorf("w.CloseWrite: %v", err) |
|
} |
|
|
|
if _, err := w.Write([]byte("hello")); err != io.EOF { |
|
t.Errorf("got err %v, want io.EOF after CloseWrite", err) |
|
} |
|
|
|
if err := <-result; err != io.EOF { |
|
t.Errorf("got %v (%T), want io.EOF", err, err) |
|
} |
|
} |
|
|
|
func TestMuxInvalidRecord(t *testing.T) { |
|
a, b := muxPair() |
|
defer a.Close() |
|
defer b.Close() |
|
|
|
packet := make([]byte, 1+4+4+1) |
|
packet[0] = msgChannelData |
|
marshalUint32(packet[1:], 29348723 /* invalid channel id */) |
|
marshalUint32(packet[5:], 1) |
|
packet[9] = 42 |
|
|
|
a.conn.writePacket(packet) |
|
go a.SendRequest("hello", false, nil) |
|
// 'a' wrote an invalid packet, so 'b' has exited. |
|
req, ok := <-b.incomingRequests |
|
if ok { |
|
t.Errorf("got request %#v after receiving invalid packet", req) |
|
} |
|
} |
|
|
|
func TestZeroWindowAdjust(t *testing.T) { |
|
a, b, mux := channelPair(t) |
|
defer a.Close() |
|
defer b.Close() |
|
defer mux.Close() |
|
|
|
go func() { |
|
io.WriteString(a, "hello") |
|
// bogus adjust. |
|
a.sendMessage(windowAdjustMsg{}) |
|
io.WriteString(a, "world") |
|
a.Close() |
|
}() |
|
|
|
want := "helloworld" |
|
c, _ := ioutil.ReadAll(b) |
|
if string(c) != want { |
|
t.Errorf("got %q want %q", c, want) |
|
} |
|
} |
|
|
|
func TestMuxMaxPacketSize(t *testing.T) { |
|
a, b, mux := channelPair(t) |
|
defer a.Close() |
|
defer b.Close() |
|
defer mux.Close() |
|
|
|
large := make([]byte, a.maxRemotePayload+1) |
|
packet := make([]byte, 1+4+4+1+len(large)) |
|
packet[0] = msgChannelData |
|
marshalUint32(packet[1:], a.remoteId) |
|
marshalUint32(packet[5:], uint32(len(large))) |
|
packet[9] = 42 |
|
|
|
if err := a.mux.conn.writePacket(packet); err != nil { |
|
t.Errorf("could not send packet") |
|
} |
|
|
|
go a.SendRequest("hello", false, nil) |
|
|
|
_, ok := <-b.incomingRequests |
|
if ok { |
|
t.Errorf("connection still alive after receiving large packet.") |
|
} |
|
} |
|
|
|
// Don't ship code with debug=true. |
|
func TestDebug(t *testing.T) { |
|
if debugMux { |
|
t.Error("mux debug switched on") |
|
} |
|
if debugHandshake { |
|
t.Error("handshake debug switched on") |
|
} |
|
}
|
|
|