2018-09-21 15:18:23 +00:00
|
|
|
package carrier
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
|
|
|
"io"
|
|
|
|
"net"
|
|
|
|
"net/http"
|
|
|
|
"net/http/httptest"
|
|
|
|
"sync"
|
|
|
|
"testing"
|
|
|
|
|
|
|
|
ws "github.com/gorilla/websocket"
|
|
|
|
"github.com/sirupsen/logrus"
|
|
|
|
"github.com/stretchr/testify/assert"
|
|
|
|
)
|
|
|
|
|
|
|
|
const (
|
|
|
|
// example in Sec-Websocket-Key in rfc6455
|
|
|
|
testSecWebsocketKey = "dGhlIHNhbXBsZSBub25jZQ=="
|
|
|
|
)
|
|
|
|
|
|
|
|
type testStreamer struct {
|
|
|
|
buf *bytes.Buffer
|
|
|
|
l sync.RWMutex
|
|
|
|
}
|
|
|
|
|
|
|
|
func newTestStream() *testStreamer {
|
|
|
|
return &testStreamer{buf: new(bytes.Buffer)}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *testStreamer) Read(p []byte) (int, error) {
|
|
|
|
s.l.RLock()
|
|
|
|
defer s.l.RUnlock()
|
|
|
|
return s.buf.Read(p)
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *testStreamer) Write(p []byte) (int, error) {
|
|
|
|
s.l.Lock()
|
|
|
|
defer s.l.Unlock()
|
|
|
|
return s.buf.Write(p)
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestStartClient(t *testing.T) {
|
|
|
|
message := "Good morning Austin! Time for another sunny day in the great state of Texas."
|
|
|
|
logger := logrus.New()
|
|
|
|
ts := newTestWebSocketServer()
|
|
|
|
defer ts.Close()
|
|
|
|
|
|
|
|
buf := newTestStream()
|
2019-01-23 21:42:10 +00:00
|
|
|
options := &StartOptions{
|
2019-05-22 20:41:21 +00:00
|
|
|
OriginURL: "http://" + ts.Listener.Addr().String(),
|
|
|
|
Headers: nil,
|
2019-01-23 21:42:10 +00:00
|
|
|
}
|
|
|
|
err := StartClient(logger, buf, options)
|
2018-09-21 15:18:23 +00:00
|
|
|
assert.NoError(t, err)
|
|
|
|
buf.Write([]byte(message))
|
|
|
|
|
|
|
|
readBuffer := make([]byte, len(message))
|
|
|
|
buf.Read(readBuffer)
|
|
|
|
assert.Equal(t, message, string(readBuffer))
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestStartServer(t *testing.T) {
|
2019-04-05 06:57:00 +00:00
|
|
|
listener, err := net.Listen("tcp", "localhost:")
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("Error starting listener: %v", err)
|
|
|
|
}
|
2018-09-21 15:18:23 +00:00
|
|
|
message := "Good morning Austin! Time for another sunny day in the great state of Texas."
|
|
|
|
logger := logrus.New()
|
|
|
|
shutdownC := make(chan struct{})
|
|
|
|
ts := newTestWebSocketServer()
|
|
|
|
defer ts.Close()
|
2019-01-23 21:42:10 +00:00
|
|
|
options := &StartOptions{
|
2019-05-22 20:41:21 +00:00
|
|
|
OriginURL: "http://" + ts.Listener.Addr().String(),
|
|
|
|
Headers: nil,
|
2019-01-23 21:42:10 +00:00
|
|
|
}
|
2018-09-21 15:18:23 +00:00
|
|
|
|
|
|
|
go func() {
|
2019-01-23 21:42:10 +00:00
|
|
|
err := Serve(logger, listener, shutdownC, options)
|
2018-10-26 20:58:43 +00:00
|
|
|
if err != nil {
|
2019-04-05 06:57:00 +00:00
|
|
|
t.Fatalf("Error running server: %v", err)
|
2018-10-26 20:58:43 +00:00
|
|
|
}
|
2018-09-21 15:18:23 +00:00
|
|
|
}()
|
|
|
|
|
2019-04-05 06:57:00 +00:00
|
|
|
conn, err := net.Dial("tcp", listener.Addr().String())
|
2018-09-21 15:18:23 +00:00
|
|
|
conn.Write([]byte(message))
|
|
|
|
|
|
|
|
readBuffer := make([]byte, len(message))
|
|
|
|
conn.Read(readBuffer)
|
|
|
|
assert.Equal(t, string(readBuffer), message)
|
|
|
|
}
|
|
|
|
|
2019-06-26 15:48:45 +00:00
|
|
|
func TestIsAccessResponse(t *testing.T) {
|
|
|
|
validLocationHeader := http.Header{}
|
|
|
|
validLocationHeader.Add("location", "https://test.cloudflareaccess.com/cdn-cgi/access/login/blahblah")
|
|
|
|
invalidLocationHeader := http.Header{}
|
|
|
|
invalidLocationHeader.Add("location", "https://google.com")
|
|
|
|
testCases := []struct {
|
|
|
|
Description string
|
|
|
|
In *http.Response
|
|
|
|
ExpectedOut bool
|
|
|
|
}{
|
|
|
|
{"nil response", nil, false},
|
|
|
|
{"redirect with no location", &http.Response{StatusCode: http.StatusPermanentRedirect}, false},
|
|
|
|
{"200 ok", &http.Response{StatusCode: http.StatusOK}, false},
|
|
|
|
{"redirect with location", &http.Response{StatusCode: http.StatusPermanentRedirect, Header: validLocationHeader}, true},
|
|
|
|
{"redirect with invalid location", &http.Response{StatusCode: http.StatusPermanentRedirect, Header: invalidLocationHeader}, false},
|
|
|
|
}
|
|
|
|
|
|
|
|
for i, tc := range testCases {
|
|
|
|
if isAccessResponse(tc.In) != tc.ExpectedOut {
|
|
|
|
t.Fatalf("Failed case %d -- %s", i, tc.Description)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2018-09-21 15:18:23 +00:00
|
|
|
func newTestWebSocketServer() *httptest.Server {
|
|
|
|
upgrader := ws.Upgrader{
|
|
|
|
ReadBufferSize: 1024,
|
|
|
|
WriteBufferSize: 1024,
|
|
|
|
}
|
|
|
|
|
|
|
|
return httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
|
|
|
|
conn, _ := upgrader.Upgrade(w, r, nil)
|
|
|
|
defer conn.Close()
|
|
|
|
for {
|
|
|
|
mt, message, err := conn.ReadMessage()
|
|
|
|
if err != nil {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
|
|
|
|
if err := conn.WriteMessage(mt, []byte(message)); err != nil {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}))
|
|
|
|
}
|
|
|
|
|
|
|
|
func testRequest(t *testing.T, url string, stream io.ReadWriter) *http.Request {
|
|
|
|
req, err := http.NewRequest("GET", url, stream)
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("testRequestHeader error")
|
|
|
|
}
|
|
|
|
|
|
|
|
req.Header.Add("Connection", "Upgrade")
|
|
|
|
req.Header.Add("Upgrade", "WebSocket")
|
|
|
|
req.Header.Add("Sec-Websocket-Key", testSecWebsocketKey)
|
|
|
|
req.Header.Add("Sec-Websocket-Protocol", "tunnel-protocol")
|
|
|
|
req.Header.Add("Sec-Websocket-Version", "13")
|
|
|
|
req.Header.Add("User-Agent", "curl/7.59.0")
|
|
|
|
|
|
|
|
return req
|
|
|
|
}
|