cloudflare/cloudflared

Public

mirrored from https://github.com/cloudflare/cloudflaredAvailable

CodeCommitsIssuesPull requestsActionsInsightsSecurity
2020.4.0

Branches

Tags

  • No tags available.
0Branches0Tags
Go to file
Add file
Code

Clone

HTTPS

Download ZIP

carrier/websocket.go

137lines · modecode

1package carrier
2
3import (
4 "fmt"
5 "io"
6 "net"
7 "net/http"
8
9 "github.com/cloudflare/cloudflared/cmd/cloudflared/token"
10 "github.com/cloudflare/cloudflared/socks"
11 cfwebsocket "github.com/cloudflare/cloudflared/websocket"
12 "github.com/gorilla/websocket"
13 "github.com/sirupsen/logrus"
14)
15
16// Websocket is used to carry data via WS binary frames over the tunnel from client to the origin
17// This implements the functions for glider proxy (sock5) and the carrier interface
18type Websocket struct {
19 logger *logrus.Logger
20 isSocks bool
21}
22
23type wsdialer struct {
24 conn *cfwebsocket.Conn
25}
26
27func (d *wsdialer) Dial(address string) (io.ReadWriteCloser, *socks.AddrSpec, error) {
28 local, ok := d.conn.LocalAddr().(*net.TCPAddr)
29 if !ok {
30 return nil, nil, fmt.Errorf("not a tcp connection")
31 }
32
33 addr := socks.AddrSpec{IP: local.IP, Port: local.Port}
34 return d.conn, &addr, nil
35}
36
37// NewWSConnection returns a new connection object
38func NewWSConnection(logger *logrus.Logger, isSocks bool) Connection {
39 return &Websocket{
40 logger: logger,
41 isSocks: isSocks,
42 }
43}
44
45// ServeStream will create a Websocket client stream connection to the edge
46// it blocks and writes the raw data from conn over the tunnel
47func (ws *Websocket) ServeStream(options *StartOptions, conn io.ReadWriter) error {
48 wsConn, err := createWebsocketStream(options)
49 if err != nil {
50 ws.logger.WithError(err).Errorf("failed to connect to %s", options.OriginURL)
51 return err
52 }
53 defer wsConn.Close()
54
55 if ws.isSocks {
56 dialer := &wsdialer{conn: wsConn}
57 requestHandler := socks.NewRequestHandler(dialer)
58 socksServer := socks.NewConnectionHandler(requestHandler)
59
60 socksServer.Serve(conn)
61 } else {
62 cfwebsocket.Stream(wsConn, conn)
63 }
64 return nil
65}
66
67// StartServer creates a Websocket server to listen for connections.
68// This is used on the origin (tunnel) side to take data from the muxer and send it to the origin
69func (ws *Websocket) StartServer(listener net.Listener, remote string, shutdownC <-chan struct{}) error {
70 return cfwebsocket.StartProxyServer(ws.logger, listener, remote, shutdownC, cfwebsocket.DefaultStreamHandler)
71}
72
73// createWebsocketStream will create a WebSocket connection to stream data over
74// It also handles redirects from Access and will present that flow if
75// the token is not present on the request
76func createWebsocketStream(options *StartOptions) (*cfwebsocket.Conn, error) {
77 req, err := http.NewRequest(http.MethodGet, options.OriginURL, nil)
78 if err != nil {
79 return nil, err
80 }
81 req.Header = options.Headers
82
83 wsConn, resp, err := cfwebsocket.ClientConnect(req, nil)
84 defer closeRespBody(resp)
85 if err != nil && IsAccessResponse(resp) {
86 wsConn, err = createAccessAuthenticatedStream(options)
87 if err != nil {
88 return nil, err
89 }
90 } else if err != nil {
91 return nil, err
92 }
93
94 return &cfwebsocket.Conn{Conn: wsConn}, nil
95}
96
97// createAccessAuthenticatedStream will try load a token from storage and make
98// a connection with the token set on the request. If it still get redirect,
99// this probably means the token in storage is invalid (expired/revoked). If that
100// happens it deletes the token and runs the connection again, so the user can
101// login again and generate a new one.
102func createAccessAuthenticatedStream(options *StartOptions) (*websocket.Conn, error) {
103 wsConn, resp, err := createAccessWebSocketStream(options)
104 defer closeRespBody(resp)
105 if err == nil {
106 return wsConn, nil
107 }
108
109 if !IsAccessResponse(resp) {
110 return nil, err
111 }
112
113 // Access Token is invalid for some reason. Go through regen flow
114 originReq, err := http.NewRequest(http.MethodGet, options.OriginURL, nil)
115 if err != nil {
116 return nil, err
117 }
118 if err := token.RemoveTokenIfExists(originReq.URL); err != nil {
119 return nil, err
120 }
121 wsConn, resp, err = createAccessWebSocketStream(options)
122 defer closeRespBody(resp)
123 if err != nil {
124 return nil, err
125 }
126
127 return wsConn, nil
128}
129
130// createAccessWebSocketStream builds an Access request and makes a connection
131func createAccessWebSocketStream(options *StartOptions) (*websocket.Conn, *http.Response, error) {
132 req, err := BuildAccessRequest(options)
133 if err != nil {
134 return nil, nil, err
135 }
136 return cfwebsocket.ClientConnect(req, nil)
137}