9ff56c9570
From-SVN: r173931
193 lines
5.0 KiB
Go
193 lines
5.0 KiB
Go
// Copyright 2009 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 implements a client and server for the Web Socket protocol.
|
|
// The protocol is defined at http://tools.ietf.org/html/draft-hixie-thewebsocketprotocol
|
|
package websocket
|
|
|
|
// TODO(ukai):
|
|
// better logging.
|
|
|
|
import (
|
|
"bufio"
|
|
"crypto/md5"
|
|
"encoding/binary"
|
|
"http"
|
|
"io"
|
|
"net"
|
|
"os"
|
|
)
|
|
|
|
// WebSocketAddr is an implementation of net.Addr for Web Sockets.
|
|
type WebSocketAddr string
|
|
|
|
// Network returns the network type for a Web Socket, "websocket".
|
|
func (addr WebSocketAddr) Network() string { return "websocket" }
|
|
|
|
// String returns the network address for a Web Socket.
|
|
func (addr WebSocketAddr) String() string { return string(addr) }
|
|
|
|
const (
|
|
stateFrameByte = iota
|
|
stateFrameLength
|
|
stateFrameData
|
|
stateFrameTextData
|
|
)
|
|
|
|
// Conn is a channel to communicate to a Web Socket.
|
|
// It implements the net.Conn interface.
|
|
type Conn struct {
|
|
// The origin URI for the Web Socket.
|
|
Origin string
|
|
// The location URI for the Web Socket.
|
|
Location string
|
|
// The subprotocol for the Web Socket.
|
|
Protocol string
|
|
// The initial http Request (for the Server side only).
|
|
Request *http.Request
|
|
|
|
buf *bufio.ReadWriter
|
|
rwc io.ReadWriteCloser
|
|
|
|
// It holds text data in previous Read() that failed with small buffer.
|
|
data []byte
|
|
reading bool
|
|
}
|
|
|
|
// newConn creates a new Web Socket.
|
|
func newConn(origin, location, protocol string, buf *bufio.ReadWriter, rwc io.ReadWriteCloser) *Conn {
|
|
if buf == nil {
|
|
br := bufio.NewReader(rwc)
|
|
bw := bufio.NewWriter(rwc)
|
|
buf = bufio.NewReadWriter(br, bw)
|
|
}
|
|
ws := &Conn{Origin: origin, Location: location, Protocol: protocol, buf: buf, rwc: rwc}
|
|
return ws
|
|
}
|
|
|
|
// Read implements the io.Reader interface for a Conn.
|
|
func (ws *Conn) Read(msg []byte) (n int, err os.Error) {
|
|
Frame:
|
|
for !ws.reading && len(ws.data) == 0 {
|
|
// Beginning of frame, possibly.
|
|
b, err := ws.buf.ReadByte()
|
|
if err != nil {
|
|
return 0, err
|
|
}
|
|
if b&0x80 == 0x80 {
|
|
// Skip length frame.
|
|
length := 0
|
|
for {
|
|
c, err := ws.buf.ReadByte()
|
|
if err != nil {
|
|
return 0, err
|
|
}
|
|
length = length*128 + int(c&0x7f)
|
|
if c&0x80 == 0 {
|
|
break
|
|
}
|
|
}
|
|
for length > 0 {
|
|
_, err := ws.buf.ReadByte()
|
|
if err != nil {
|
|
return 0, err
|
|
}
|
|
}
|
|
continue Frame
|
|
}
|
|
// In text mode
|
|
if b != 0 {
|
|
// Skip this frame
|
|
for {
|
|
c, err := ws.buf.ReadByte()
|
|
if err != nil {
|
|
return 0, err
|
|
}
|
|
if c == '\xff' {
|
|
break
|
|
}
|
|
}
|
|
continue Frame
|
|
}
|
|
ws.reading = true
|
|
}
|
|
if len(ws.data) == 0 {
|
|
ws.data, err = ws.buf.ReadSlice('\xff')
|
|
if err == nil {
|
|
ws.reading = false
|
|
ws.data = ws.data[:len(ws.data)-1] // trim \xff
|
|
}
|
|
}
|
|
n = copy(msg, ws.data)
|
|
ws.data = ws.data[n:]
|
|
return n, err
|
|
}
|
|
|
|
// Write implements the io.Writer interface for a Conn.
|
|
func (ws *Conn) Write(msg []byte) (n int, err os.Error) {
|
|
ws.buf.WriteByte(0)
|
|
ws.buf.Write(msg)
|
|
ws.buf.WriteByte(0xff)
|
|
err = ws.buf.Flush()
|
|
return len(msg), err
|
|
}
|
|
|
|
// Close implements the io.Closer interface for a Conn.
|
|
func (ws *Conn) Close() os.Error { return ws.rwc.Close() }
|
|
|
|
// LocalAddr returns the WebSocket Origin for the connection.
|
|
func (ws *Conn) LocalAddr() net.Addr { return WebSocketAddr(ws.Origin) }
|
|
|
|
// RemoteAddr returns the WebSocket locations for the connection.
|
|
func (ws *Conn) RemoteAddr() net.Addr { return WebSocketAddr(ws.Location) }
|
|
|
|
// SetTimeout sets the connection's network timeout in nanoseconds.
|
|
func (ws *Conn) SetTimeout(nsec int64) os.Error {
|
|
if conn, ok := ws.rwc.(net.Conn); ok {
|
|
return conn.SetTimeout(nsec)
|
|
}
|
|
return os.EINVAL
|
|
}
|
|
|
|
// SetReadTimeout sets the connection's network read timeout in nanoseconds.
|
|
func (ws *Conn) SetReadTimeout(nsec int64) os.Error {
|
|
if conn, ok := ws.rwc.(net.Conn); ok {
|
|
return conn.SetReadTimeout(nsec)
|
|
}
|
|
return os.EINVAL
|
|
}
|
|
|
|
// SetWritetTimeout sets the connection's network write timeout in nanoseconds.
|
|
func (ws *Conn) SetWriteTimeout(nsec int64) os.Error {
|
|
if conn, ok := ws.rwc.(net.Conn); ok {
|
|
return conn.SetWriteTimeout(nsec)
|
|
}
|
|
return os.EINVAL
|
|
}
|
|
|
|
// getChallengeResponse computes the expected response from the
|
|
// challenge as described in section 5.1 Opening Handshake steps 42 to
|
|
// 43 of http://www.whatwg.org/specs/web-socket-protocol/
|
|
func getChallengeResponse(number1, number2 uint32, key3 []byte) (expected []byte, err os.Error) {
|
|
// 41. Let /challenge/ be the concatenation of /number_1/, expressed
|
|
// a big-endian 32 bit integer, /number_2/, expressed in a big-
|
|
// endian 32 bit integer, and the eight bytes of /key_3/ in the
|
|
// order they were sent to the wire.
|
|
challenge := make([]byte, 16)
|
|
binary.BigEndian.PutUint32(challenge[0:], number1)
|
|
binary.BigEndian.PutUint32(challenge[4:], number2)
|
|
copy(challenge[8:], key3)
|
|
|
|
// 42. Let /expected/ be the MD5 fingerprint of /challenge/ as a big-
|
|
// endian 128 bit string.
|
|
h := md5.New()
|
|
if _, err = h.Write(challenge); err != nil {
|
|
return
|
|
}
|
|
expected = h.Sum()
|
|
return
|
|
}
|
|
|
|
var _ net.Conn = (*Conn)(nil) // compile-time check that *Conn implements net.Conn.
|