-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathserver.go
132 lines (117 loc) · 2.78 KB
/
server.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
package sshtest
import (
"fmt"
"net"
"sync"
"golang.org/x/crypto/ssh"
)
// A Server is an SSH server listening on a random port on the loopback interface. This
// server can be used in tests for SSH clients.
type Server struct {
Endpoint string
Listener net.Listener
Config *ssh.ServerConfig
// Handler for incoming sessions, NullHandler if nil
Handler func(ssh.Channel, <-chan *ssh.Request)
mu sync.Mutex
closed bool
}
// NewServer starts and return a new SSH server. The caller is responsible for calling Close() when done.
func NewServer(hostKey ssh.Signer) *Server {
s := NewUnstartedServer()
s.Config = &ssh.ServerConfig{NoClientAuth: true}
s.Config.AddHostKey(hostKey)
s.Start()
return s
}
// NewUnstartedServer returns a new server with the default config but doesn't start it
// allowing the caller to change the config or add keys before starting the server.
func NewUnstartedServer() *Server {
ln := newListener()
return &Server{
Listener: ln,
Endpoint: ln.Addr().String(),
}
}
// Start the SSH server
func (s *Server) Start() {
if s.Config == nil {
panic("sshtest: no server config defined")
}
go func() {
for {
serverConn, err := s.Listener.Accept()
if err != nil {
s.mu.Lock()
if s.closed {
s.mu.Unlock()
return
}
s.mu.Unlock()
continue
}
go func() {
defer serverConn.Close()
_, chans, reqs, err := ssh.NewServerConn(serverConn, s.Config)
if err != nil {
return
}
go ssh.DiscardRequests(reqs)
for newCh := range chans {
if newCh.ChannelType() != "session" {
newCh.Reject(ssh.UnknownChannelType, "unknown channel type")
continue
}
ch, inReqs, err := newCh.Accept()
if err != nil {
continue
}
if s.Handler == nil {
NullHandler(ch, inReqs)
continue
}
s.Handler(ch, inReqs)
}
}()
}
}()
}
// Close stops the Server's listener
func (s *Server) Close() {
s.mu.Lock()
defer s.mu.Unlock()
if s.closed {
return
}
s.closed = true
s.Listener.Close()
}
func newListener() net.Listener {
l, err := net.Listen("tcp", "127.0.0.1:0")
if err != nil {
if l, err = net.Listen("tcp6", "[::1]:0"); err != nil {
panic(fmt.Sprintf("sshtest: failed to listen on a port: %v", err))
}
}
return l
}
// NullHandler is used in SSH servers to discard all incoming requests and respond with success (code 0).
func NullHandler(ch ssh.Channel, in <-chan *ssh.Request) {
defer ch.Close()
req, ok := <-in
if !ok {
return
}
req.Reply(true, nil)
SendStatus(ch, 0)
}
// SendStatus replies with an exits-status message on the provided channel.
func SendStatus(ch ssh.Channel, code uint32) error {
var statusMsg = struct {
Status uint32
}{
Status: code,
}
_, err := ch.SendRequest("exit-status", false, ssh.Marshal(&statusMsg))
return err
}