aboutsummaryrefslogtreecommitdiffstats
path: root/vendor/golang.org/x/crypto/ssh/streamlocal.go
blob: b171b330bc380ae8e7b57527307b3b133ca9c694 (plain) (blame)
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
package ssh

import (
    "errors"
    "io"
    "net"
)

// streamLocalChannelOpenDirectMsg is a struct used for SSH_MSG_CHANNEL_OPEN message
// with "direct-streamlocal@openssh.com" string.
//
// See openssh-portable/PROTOCOL, section 2.4. connection: Unix domain socket forwarding
// https://github.com/openssh/openssh-portable/blob/master/PROTOCOL#L235
type streamLocalChannelOpenDirectMsg struct {
    socketPath string
    reserved0  string
    reserved1  uint32
}

// forwardedStreamLocalPayload is a struct used for SSH_MSG_CHANNEL_OPEN message
// with "forwarded-streamlocal@openssh.com" string.
type forwardedStreamLocalPayload struct {
    SocketPath string
    Reserved0  string
}

// streamLocalChannelForwardMsg is a struct used for SSH2_MSG_GLOBAL_REQUEST message
// with "streamlocal-forward@openssh.com"/"cancel-streamlocal-forward@openssh.com" string.
type streamLocalChannelForwardMsg struct {
    socketPath string
}

// ListenUnix is similar to ListenTCP but uses a Unix domain socket.
func (c *Client) ListenUnix(socketPath string) (net.Listener, error) {
    c.handleForwardsOnce.Do(c.handleForwards)
    m := streamLocalChannelForwardMsg{
        socketPath,
    }
    // send message
    ok, _, err := c.SendRequest("streamlocal-forward@openssh.com", true, Marshal(&m))
    if err != nil {
        return nil, err
    }
    if !ok {
        return nil, errors.New("ssh: streamlocal-forward@openssh.com request denied by peer")
    }
    ch := c.forwards.add(&net.UnixAddr{Name: socketPath, Net: "unix"})

    return &unixListener{socketPath, c, ch}, nil
}

func (c *Client) dialStreamLocal(socketPath string) (Channel, error) {
    msg := streamLocalChannelOpenDirectMsg{
        socketPath: socketPath,
    }
    ch, in, err := c.OpenChannel("direct-streamlocal@openssh.com", Marshal(&msg))
    if err != nil {
        return nil, err
    }
    go DiscardRequests(in)
    return ch, err
}

type unixListener struct {
    socketPath string

    conn *Client
    in   <-chan forward
}

// Accept waits for and returns the next connection to the listener.
func (l *unixListener) Accept() (net.Conn, error) {
    s, ok := <-l.in
    if !ok {
        return nil, io.EOF
    }
    ch, incoming, err := s.newCh.Accept()
    if err != nil {
        return nil, err
    }
    go DiscardRequests(incoming)

    return &chanConn{
        Channel: ch,
        laddr: &net.UnixAddr{
            Name: l.socketPath,
            Net:  "unix",
        },
        raddr: &net.UnixAddr{
            Name: "@",
            Net:  "unix",
        },
    }, nil
}

// Close closes the listener.
func (l *unixListener) Close() error {
    // this also closes the listener.
    l.conn.forwards.remove(&net.UnixAddr{Name: l.socketPath, Net: "unix"})
    m := streamLocalChannelForwardMsg{
        l.socketPath,
    }
    ok, _, err := l.conn.SendRequest("cancel-streamlocal-forward@openssh.com", true, Marshal(&m))
    if err == nil && !ok {
        err = errors.New("ssh: cancel-streamlocal-forward@openssh.com failed")
    }
    return err
}

// Addr returns the listener's network address.
func (l *unixListener) Addr() net.Addr {
    return &net.UnixAddr{
        Name: l.socketPath,
        Net:  "unix",
    }
}