// Copyright 2011 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 websocket import ( "bufio" "bytes" "fmt" "io" "net/http" "net/url" "strings" "testing" ) // Test the getChallengeResponse function with values from section // 5.1 of the specification steps 18, 26, and 43 from // http://tools.ietf.org/html/draft-ietf-hybi-thewebsocketprotocol-00 func TestHixie76Challenge(t *testing.T) { var part1 uint32 = 777007543 var part2 uint32 = 114997259 key3 := []byte{0x47, 0x30, 0x22, 0x2D, 0x5A, 0x3F, 0x47, 0x58} expected := []byte("0st3Rl&q-2ZU^weu") response, err := getChallengeResponse(part1, part2, key3) if err != nil { t.Errorf("getChallengeResponse: returned error %v", err) return } if !bytes.Equal(expected, response) { t.Errorf("getChallengeResponse: expected %q got %q", expected, response) } } func TestHixie76ClientHandshake(t *testing.T) { b := bytes.NewBuffer([]byte{}) bw := bufio.NewWriter(b) br := bufio.NewReader(strings.NewReader(`HTTP/1.1 101 WebSocket Protocol Handshake Upgrade: WebSocket Connection: Upgrade Sec-WebSocket-Origin: http://example.com Sec-WebSocket-Location: ws://example.com/demo Sec-WebSocket-Protocol: sample 8jKS'y:G*Co,Wxa-`)) var err error config := new(Config) config.Location, err = url.ParseRequest("ws://example.com/demo") if err != nil { t.Fatal("location url", err) } config.Origin, err = url.ParseRequest("http://example.com") if err != nil { t.Fatal("origin url", err) } config.Protocol = append(config.Protocol, "sample") config.Version = ProtocolVersionHixie76 config.handshakeData = map[string]string{ "key1": "4 @1 46546xW%0l 1 5", "number1": "829309203", "key2": "12998 5 Y3 1 .P00", "number2": "259970620", "key3": "^n:ds[4U", } err = hixie76ClientHandshake(config, br, bw) if err != nil { t.Errorf("handshake failed: %v", err) } req, err := http.ReadRequest(bufio.NewReader(b)) if err != nil { t.Fatalf("read request: %v", err) } if req.Method != "GET" { t.Errorf("request method expected GET, but got %q", req.Method) } if req.URL.Path != "/demo" { t.Errorf("request path expected /demo, but got %q", req.URL.Path) } if req.Proto != "HTTP/1.1" { t.Errorf("request proto expected HTTP/1.1, but got %q", req.Proto) } if req.Host != "example.com" { t.Errorf("request Host expected example.com, but got %v", req.Host) } var expectedHeader = map[string]string{ "Connection": "Upgrade", "Upgrade": "WebSocket", "Origin": "http://example.com", "Sec-Websocket-Key1": config.handshakeData["key1"], "Sec-Websocket-Key2": config.handshakeData["key2"], "Sec-WebSocket-Protocol": config.Protocol[0], } for k, v := range expectedHeader { if req.Header.Get(k) != v { t.Errorf(fmt.Sprintf("%s expected %q but got %q", k, v, req.Header.Get(k))) } } } func TestHixie76ServerHandshake(t *testing.T) { config := new(Config) handshaker := &hixie76ServerHandshaker{Config: config} br := bufio.NewReader(strings.NewReader(`GET /demo HTTP/1.1 Host: example.com Connection: Upgrade Sec-WebSocket-Key2: 12998 5 Y3 1 .P00 Sec-WebSocket-Protocol: sample Upgrade: WebSocket Sec-WebSocket-Key1: 4 @1 46546xW%0l 1 5 Origin: http://example.com ^n:ds[4U`)) req, err := http.ReadRequest(br) if err != nil { t.Fatal("request", err) } code, err := handshaker.ReadHandshake(br, req) if err != nil { t.Errorf("handshake failed: %v", err) } if code != http.StatusSwitchingProtocols { t.Errorf("status expected %q but got %q", http.StatusSwitchingProtocols, code) } b := bytes.NewBuffer([]byte{}) bw := bufio.NewWriter(b) err = handshaker.AcceptHandshake(bw) if err != nil { t.Errorf("handshake response failed: %v", err) } expectedResponse := strings.Join([]string{ "HTTP/1.1 101 WebSocket Protocol Handshake", "Upgrade: WebSocket", "Connection: Upgrade", "Sec-WebSocket-Origin: http://example.com", "Sec-WebSocket-Location: ws://example.com/demo", "Sec-WebSocket-Protocol: sample", "", ""}, "\r\n") + "8jKS'y:G*Co,Wxa-" if b.String() != expectedResponse { t.Errorf("handshake expected %q but got %q", expectedResponse, b.String()) } } func TestHixie76SkipLengthFrame(t *testing.T) { b := []byte{'\x80', '\x01', 'x', 0, 'h', 'e', 'l', 'l', 'o', '\xff'} buf := bytes.NewBuffer(b) br := bufio.NewReader(buf) bw := bufio.NewWriter(buf) config := newConfig(t, "/") ws := newHixieConn(config, bufio.NewReadWriter(br, bw), nil, nil) msg := make([]byte, 5) n, err := ws.Read(msg) if err != nil { t.Errorf("Read: %v", err) } if !bytes.Equal(b[4:9], msg[0:n]) { t.Errorf("Read: expected %q got %q", b[4:9], msg[0:n]) } } func TestHixie76SkipNoUTF8Frame(t *testing.T) { b := []byte{'\x01', 'n', '\xff', 0, 'h', 'e', 'l', 'l', 'o', '\xff'} buf := bytes.NewBuffer(b) br := bufio.NewReader(buf) bw := bufio.NewWriter(buf) config := newConfig(t, "/") ws := newHixieConn(config, bufio.NewReadWriter(br, bw), nil, nil) msg := make([]byte, 5) n, err := ws.Read(msg) if err != nil { t.Errorf("Read: %v", err) } if !bytes.Equal(b[4:9], msg[0:n]) { t.Errorf("Read: expected %q got %q", b[4:9], msg[0:n]) } } func TestHixie76ClosingFrame(t *testing.T) { b := []byte{0, 'h', 'e', 'l', 'l', 'o', '\xff'} buf := bytes.NewBuffer(b) br := bufio.NewReader(buf) bw := bufio.NewWriter(buf) config := newConfig(t, "/") ws := newHixieConn(config, bufio.NewReadWriter(br, bw), nil, nil) msg := make([]byte, 5) n, err := ws.Read(msg) if err != nil { t.Errorf("read: %v", err) } if !bytes.Equal(b[1:6], msg[0:n]) { t.Errorf("Read: expected %q got %q", b[1:6], msg[0:n]) } n, err = ws.Read(msg) if err != io.EOF { t.Errorf("read: %v", err) } }