websocket_test.go 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587
  1. // Copyright 2009 The Go Authors. All rights reserved.
  2. // Use of this source code is governed by a BSD-style
  3. // license that can be found in the LICENSE file.
  4. package websocket
  5. import (
  6. "bytes"
  7. "fmt"
  8. "io"
  9. "log"
  10. "net"
  11. "net/http"
  12. "net/http/httptest"
  13. "net/url"
  14. "reflect"
  15. "runtime"
  16. "strings"
  17. "sync"
  18. "testing"
  19. "time"
  20. )
  21. var serverAddr string
  22. var once sync.Once
  23. func echoServer(ws *Conn) {
  24. defer ws.Close()
  25. io.Copy(ws, ws)
  26. }
  27. type Count struct {
  28. S string
  29. N int
  30. }
  31. func countServer(ws *Conn) {
  32. defer ws.Close()
  33. for {
  34. var count Count
  35. err := JSON.Receive(ws, &count)
  36. if err != nil {
  37. return
  38. }
  39. count.N++
  40. count.S = strings.Repeat(count.S, count.N)
  41. err = JSON.Send(ws, count)
  42. if err != nil {
  43. return
  44. }
  45. }
  46. }
  47. type testCtrlAndDataHandler struct {
  48. hybiFrameHandler
  49. }
  50. func (h *testCtrlAndDataHandler) WritePing(b []byte) (int, error) {
  51. h.hybiFrameHandler.conn.wio.Lock()
  52. defer h.hybiFrameHandler.conn.wio.Unlock()
  53. w, err := h.hybiFrameHandler.conn.frameWriterFactory.NewFrameWriter(PingFrame)
  54. if err != nil {
  55. return 0, err
  56. }
  57. n, err := w.Write(b)
  58. w.Close()
  59. return n, err
  60. }
  61. func ctrlAndDataServer(ws *Conn) {
  62. defer ws.Close()
  63. h := &testCtrlAndDataHandler{hybiFrameHandler: hybiFrameHandler{conn: ws}}
  64. ws.frameHandler = h
  65. go func() {
  66. for i := 0; ; i++ {
  67. var b []byte
  68. if i%2 != 0 { // with or without payload
  69. b = []byte(fmt.Sprintf("#%d-CONTROL-FRAME-FROM-SERVER", i))
  70. }
  71. if _, err := h.WritePing(b); err != nil {
  72. break
  73. }
  74. if _, err := h.WritePong(b); err != nil { // unsolicited pong
  75. break
  76. }
  77. time.Sleep(10 * time.Millisecond)
  78. }
  79. }()
  80. b := make([]byte, 128)
  81. for {
  82. n, err := ws.Read(b)
  83. if err != nil {
  84. break
  85. }
  86. if _, err := ws.Write(b[:n]); err != nil {
  87. break
  88. }
  89. }
  90. }
  91. func subProtocolHandshake(config *Config, req *http.Request) error {
  92. for _, proto := range config.Protocol {
  93. if proto == "chat" {
  94. config.Protocol = []string{proto}
  95. return nil
  96. }
  97. }
  98. return ErrBadWebSocketProtocol
  99. }
  100. func subProtoServer(ws *Conn) {
  101. for _, proto := range ws.Config().Protocol {
  102. io.WriteString(ws, proto)
  103. }
  104. }
  105. func startServer() {
  106. http.Handle("/echo", Handler(echoServer))
  107. http.Handle("/count", Handler(countServer))
  108. http.Handle("/ctrldata", Handler(ctrlAndDataServer))
  109. subproto := Server{
  110. Handshake: subProtocolHandshake,
  111. Handler: Handler(subProtoServer),
  112. }
  113. http.Handle("/subproto", subproto)
  114. server := httptest.NewServer(nil)
  115. serverAddr = server.Listener.Addr().String()
  116. log.Print("Test WebSocket server listening on ", serverAddr)
  117. }
  118. func newConfig(t *testing.T, path string) *Config {
  119. config, _ := NewConfig(fmt.Sprintf("ws://%s%s", serverAddr, path), "http://localhost")
  120. return config
  121. }
  122. func TestEcho(t *testing.T) {
  123. once.Do(startServer)
  124. // websocket.Dial()
  125. client, err := net.Dial("tcp", serverAddr)
  126. if err != nil {
  127. t.Fatal("dialing", err)
  128. }
  129. conn, err := NewClient(newConfig(t, "/echo"), client)
  130. if err != nil {
  131. t.Errorf("WebSocket handshake error: %v", err)
  132. return
  133. }
  134. msg := []byte("hello, world\n")
  135. if _, err := conn.Write(msg); err != nil {
  136. t.Errorf("Write: %v", err)
  137. }
  138. var actual_msg = make([]byte, 512)
  139. n, err := conn.Read(actual_msg)
  140. if err != nil {
  141. t.Errorf("Read: %v", err)
  142. }
  143. actual_msg = actual_msg[0:n]
  144. if !bytes.Equal(msg, actual_msg) {
  145. t.Errorf("Echo: expected %q got %q", msg, actual_msg)
  146. }
  147. conn.Close()
  148. }
  149. func TestAddr(t *testing.T) {
  150. once.Do(startServer)
  151. // websocket.Dial()
  152. client, err := net.Dial("tcp", serverAddr)
  153. if err != nil {
  154. t.Fatal("dialing", err)
  155. }
  156. conn, err := NewClient(newConfig(t, "/echo"), client)
  157. if err != nil {
  158. t.Errorf("WebSocket handshake error: %v", err)
  159. return
  160. }
  161. ra := conn.RemoteAddr().String()
  162. if !strings.HasPrefix(ra, "ws://") || !strings.HasSuffix(ra, "/echo") {
  163. t.Errorf("Bad remote addr: %v", ra)
  164. }
  165. la := conn.LocalAddr().String()
  166. if !strings.HasPrefix(la, "http://") {
  167. t.Errorf("Bad local addr: %v", la)
  168. }
  169. conn.Close()
  170. }
  171. func TestCount(t *testing.T) {
  172. once.Do(startServer)
  173. // websocket.Dial()
  174. client, err := net.Dial("tcp", serverAddr)
  175. if err != nil {
  176. t.Fatal("dialing", err)
  177. }
  178. conn, err := NewClient(newConfig(t, "/count"), client)
  179. if err != nil {
  180. t.Errorf("WebSocket handshake error: %v", err)
  181. return
  182. }
  183. var count Count
  184. count.S = "hello"
  185. if err := JSON.Send(conn, count); err != nil {
  186. t.Errorf("Write: %v", err)
  187. }
  188. if err := JSON.Receive(conn, &count); err != nil {
  189. t.Errorf("Read: %v", err)
  190. }
  191. if count.N != 1 {
  192. t.Errorf("count: expected %d got %d", 1, count.N)
  193. }
  194. if count.S != "hello" {
  195. t.Errorf("count: expected %q got %q", "hello", count.S)
  196. }
  197. if err := JSON.Send(conn, count); err != nil {
  198. t.Errorf("Write: %v", err)
  199. }
  200. if err := JSON.Receive(conn, &count); err != nil {
  201. t.Errorf("Read: %v", err)
  202. }
  203. if count.N != 2 {
  204. t.Errorf("count: expected %d got %d", 2, count.N)
  205. }
  206. if count.S != "hellohello" {
  207. t.Errorf("count: expected %q got %q", "hellohello", count.S)
  208. }
  209. conn.Close()
  210. }
  211. func TestWithQuery(t *testing.T) {
  212. once.Do(startServer)
  213. client, err := net.Dial("tcp", serverAddr)
  214. if err != nil {
  215. t.Fatal("dialing", err)
  216. }
  217. config := newConfig(t, "/echo")
  218. config.Location, err = url.ParseRequestURI(fmt.Sprintf("ws://%s/echo?q=v", serverAddr))
  219. if err != nil {
  220. t.Fatal("location url", err)
  221. }
  222. ws, err := NewClient(config, client)
  223. if err != nil {
  224. t.Errorf("WebSocket handshake: %v", err)
  225. return
  226. }
  227. ws.Close()
  228. }
  229. func testWithProtocol(t *testing.T, subproto []string) (string, error) {
  230. once.Do(startServer)
  231. client, err := net.Dial("tcp", serverAddr)
  232. if err != nil {
  233. t.Fatal("dialing", err)
  234. }
  235. config := newConfig(t, "/subproto")
  236. config.Protocol = subproto
  237. ws, err := NewClient(config, client)
  238. if err != nil {
  239. return "", err
  240. }
  241. msg := make([]byte, 16)
  242. n, err := ws.Read(msg)
  243. if err != nil {
  244. return "", err
  245. }
  246. ws.Close()
  247. return string(msg[:n]), nil
  248. }
  249. func TestWithProtocol(t *testing.T) {
  250. proto, err := testWithProtocol(t, []string{"chat"})
  251. if err != nil {
  252. t.Errorf("SubProto: unexpected error: %v", err)
  253. }
  254. if proto != "chat" {
  255. t.Errorf("SubProto: expected %q, got %q", "chat", proto)
  256. }
  257. }
  258. func TestWithTwoProtocol(t *testing.T) {
  259. proto, err := testWithProtocol(t, []string{"test", "chat"})
  260. if err != nil {
  261. t.Errorf("SubProto: unexpected error: %v", err)
  262. }
  263. if proto != "chat" {
  264. t.Errorf("SubProto: expected %q, got %q", "chat", proto)
  265. }
  266. }
  267. func TestWithBadProtocol(t *testing.T) {
  268. _, err := testWithProtocol(t, []string{"test"})
  269. if err != ErrBadStatus {
  270. t.Errorf("SubProto: expected %v, got %v", ErrBadStatus, err)
  271. }
  272. }
  273. func TestHTTP(t *testing.T) {
  274. once.Do(startServer)
  275. // If the client did not send a handshake that matches the protocol
  276. // specification, the server MUST return an HTTP response with an
  277. // appropriate error code (such as 400 Bad Request)
  278. resp, err := http.Get(fmt.Sprintf("http://%s/echo", serverAddr))
  279. if err != nil {
  280. t.Errorf("Get: error %#v", err)
  281. return
  282. }
  283. if resp == nil {
  284. t.Error("Get: resp is null")
  285. return
  286. }
  287. if resp.StatusCode != http.StatusBadRequest {
  288. t.Errorf("Get: expected %q got %q", http.StatusBadRequest, resp.StatusCode)
  289. }
  290. }
  291. func TestTrailingSpaces(t *testing.T) {
  292. // http://code.google.com/p/go/issues/detail?id=955
  293. // The last runs of this create keys with trailing spaces that should not be
  294. // generated by the client.
  295. once.Do(startServer)
  296. config := newConfig(t, "/echo")
  297. for i := 0; i < 30; i++ {
  298. // body
  299. ws, err := DialConfig(config)
  300. if err != nil {
  301. t.Errorf("Dial #%d failed: %v", i, err)
  302. break
  303. }
  304. ws.Close()
  305. }
  306. }
  307. func TestDialConfigBadVersion(t *testing.T) {
  308. once.Do(startServer)
  309. config := newConfig(t, "/echo")
  310. config.Version = 1234
  311. _, err := DialConfig(config)
  312. if dialerr, ok := err.(*DialError); ok {
  313. if dialerr.Err != ErrBadProtocolVersion {
  314. t.Errorf("dial expected err %q but got %q", ErrBadProtocolVersion, dialerr.Err)
  315. }
  316. }
  317. }
  318. func TestSmallBuffer(t *testing.T) {
  319. // http://code.google.com/p/go/issues/detail?id=1145
  320. // Read should be able to handle reading a fragment of a frame.
  321. once.Do(startServer)
  322. // websocket.Dial()
  323. client, err := net.Dial("tcp", serverAddr)
  324. if err != nil {
  325. t.Fatal("dialing", err)
  326. }
  327. conn, err := NewClient(newConfig(t, "/echo"), client)
  328. if err != nil {
  329. t.Errorf("WebSocket handshake error: %v", err)
  330. return
  331. }
  332. msg := []byte("hello, world\n")
  333. if _, err := conn.Write(msg); err != nil {
  334. t.Errorf("Write: %v", err)
  335. }
  336. var small_msg = make([]byte, 8)
  337. n, err := conn.Read(small_msg)
  338. if err != nil {
  339. t.Errorf("Read: %v", err)
  340. }
  341. if !bytes.Equal(msg[:len(small_msg)], small_msg) {
  342. t.Errorf("Echo: expected %q got %q", msg[:len(small_msg)], small_msg)
  343. }
  344. var second_msg = make([]byte, len(msg))
  345. n, err = conn.Read(second_msg)
  346. if err != nil {
  347. t.Errorf("Read: %v", err)
  348. }
  349. second_msg = second_msg[0:n]
  350. if !bytes.Equal(msg[len(small_msg):], second_msg) {
  351. t.Errorf("Echo: expected %q got %q", msg[len(small_msg):], second_msg)
  352. }
  353. conn.Close()
  354. }
  355. var parseAuthorityTests = []struct {
  356. in *url.URL
  357. out string
  358. }{
  359. {
  360. &url.URL{
  361. Scheme: "ws",
  362. Host: "www.google.com",
  363. },
  364. "www.google.com:80",
  365. },
  366. {
  367. &url.URL{
  368. Scheme: "wss",
  369. Host: "www.google.com",
  370. },
  371. "www.google.com:443",
  372. },
  373. {
  374. &url.URL{
  375. Scheme: "ws",
  376. Host: "www.google.com:80",
  377. },
  378. "www.google.com:80",
  379. },
  380. {
  381. &url.URL{
  382. Scheme: "wss",
  383. Host: "www.google.com:443",
  384. },
  385. "www.google.com:443",
  386. },
  387. // some invalid ones for parseAuthority. parseAuthority doesn't
  388. // concern itself with the scheme unless it actually knows about it
  389. {
  390. &url.URL{
  391. Scheme: "http",
  392. Host: "www.google.com",
  393. },
  394. "www.google.com",
  395. },
  396. {
  397. &url.URL{
  398. Scheme: "http",
  399. Host: "www.google.com:80",
  400. },
  401. "www.google.com:80",
  402. },
  403. {
  404. &url.URL{
  405. Scheme: "asdf",
  406. Host: "127.0.0.1",
  407. },
  408. "127.0.0.1",
  409. },
  410. {
  411. &url.URL{
  412. Scheme: "asdf",
  413. Host: "www.google.com",
  414. },
  415. "www.google.com",
  416. },
  417. }
  418. func TestParseAuthority(t *testing.T) {
  419. for _, tt := range parseAuthorityTests {
  420. out := parseAuthority(tt.in)
  421. if out != tt.out {
  422. t.Errorf("got %v; want %v", out, tt.out)
  423. }
  424. }
  425. }
  426. type closerConn struct {
  427. net.Conn
  428. closed int // count of the number of times Close was called
  429. }
  430. func (c *closerConn) Close() error {
  431. c.closed++
  432. return c.Conn.Close()
  433. }
  434. func TestClose(t *testing.T) {
  435. if runtime.GOOS == "plan9" {
  436. t.Skip("see golang.org/issue/11454")
  437. }
  438. once.Do(startServer)
  439. conn, err := net.Dial("tcp", serverAddr)
  440. if err != nil {
  441. t.Fatal("dialing", err)
  442. }
  443. cc := closerConn{Conn: conn}
  444. client, err := NewClient(newConfig(t, "/echo"), &cc)
  445. if err != nil {
  446. t.Fatalf("WebSocket handshake: %v", err)
  447. }
  448. // set the deadline to ten minutes ago, which will have expired by the time
  449. // client.Close sends the close status frame.
  450. conn.SetDeadline(time.Now().Add(-10 * time.Minute))
  451. if err := client.Close(); err == nil {
  452. t.Errorf("ws.Close(): expected error, got %v", err)
  453. }
  454. if cc.closed < 1 {
  455. t.Fatalf("ws.Close(): expected underlying ws.rwc.Close to be called > 0 times, got: %v", cc.closed)
  456. }
  457. }
  458. var originTests = []struct {
  459. req *http.Request
  460. origin *url.URL
  461. }{
  462. {
  463. req: &http.Request{
  464. Header: http.Header{
  465. "Origin": []string{"http://www.example.com"},
  466. },
  467. },
  468. origin: &url.URL{
  469. Scheme: "http",
  470. Host: "www.example.com",
  471. },
  472. },
  473. {
  474. req: &http.Request{},
  475. },
  476. }
  477. func TestOrigin(t *testing.T) {
  478. conf := newConfig(t, "/echo")
  479. conf.Version = ProtocolVersionHybi13
  480. for i, tt := range originTests {
  481. origin, err := Origin(conf, tt.req)
  482. if err != nil {
  483. t.Error(err)
  484. continue
  485. }
  486. if !reflect.DeepEqual(origin, tt.origin) {
  487. t.Errorf("#%d: got origin %v; want %v", i, origin, tt.origin)
  488. continue
  489. }
  490. }
  491. }
  492. func TestCtrlAndData(t *testing.T) {
  493. once.Do(startServer)
  494. c, err := net.Dial("tcp", serverAddr)
  495. if err != nil {
  496. t.Fatal(err)
  497. }
  498. ws, err := NewClient(newConfig(t, "/ctrldata"), c)
  499. if err != nil {
  500. t.Fatal(err)
  501. }
  502. defer ws.Close()
  503. h := &testCtrlAndDataHandler{hybiFrameHandler: hybiFrameHandler{conn: ws}}
  504. ws.frameHandler = h
  505. b := make([]byte, 128)
  506. for i := 0; i < 2; i++ {
  507. data := []byte(fmt.Sprintf("#%d-DATA-FRAME-FROM-CLIENT", i))
  508. if _, err := ws.Write(data); err != nil {
  509. t.Fatalf("#%d: %v", i, err)
  510. }
  511. var ctrl []byte
  512. if i%2 != 0 { // with or without payload
  513. ctrl = []byte(fmt.Sprintf("#%d-CONTROL-FRAME-FROM-CLIENT", i))
  514. }
  515. if _, err := h.WritePing(ctrl); err != nil {
  516. t.Fatalf("#%d: %v", i, err)
  517. }
  518. n, err := ws.Read(b)
  519. if err != nil {
  520. t.Fatalf("#%d: %v", i, err)
  521. }
  522. if !bytes.Equal(b[:n], data) {
  523. t.Fatalf("#%d: got %v; want %v", i, b[:n], data)
  524. }
  525. }
  526. }