-
-
Notifications
You must be signed in to change notification settings - Fork 309
/
trans-websocket-hixie76.coffee
156 lines (133 loc) · 4.69 KB
/
trans-websocket-hixie76.coffee
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
# http://tools.ietf.org/html/draft-hixie-thewebsocketprotocol-76
#
crypto = require('crypto')
utils = require('./utils')
transport = require('./transport')
validateCrypto = (req_headers, nonce) ->
k1 = req_headers['sec-websocket-key1']
k2 = req_headers['sec-websocket-key2']
if not k1 or not k2
return false
md5 = crypto.createHash('md5')
for k in [k1, k2]
n = parseInt(k.replace(/[^\d]/g, ''))
spaces = k.replace(/[^ ]/g, '').length
if spaces is 0 or n % spaces isnt 0
return false
n /= spaces
s = String.fromCharCode(
n >> 24 & 0xFF,
n >> 16 & 0xFF,
n >> 8 & 0xFF,
n & 0xFF)
md5.update(s)
md5.update(nonce.toString('binary'))
return md5.digest('binary')
class WebHandshakeHixie76
constructor: (@req, @connection, head, origin, location) ->
@sec = ('sec-websocket-key1' of @req.headers)
wsp = (@sec and ('sec-websocket-protocol' of @req.headers))
prefix = if @sec then 'Sec-' else ''
blob = [
'HTTP/1.1 101 WebSocket Protocol Handshake',
'Upgrade: websocket',
'Connection: Upgrade',
prefix + 'WebSocket-Origin: ' + origin,
prefix + 'WebSocket-Location: ' + location,
]
if wsp
blob.push('Sec-WebSocket-Protocol: ' +
@req.headers['sec-websocket-protocol'].split(',')[0])
@_setup()
try
@connection.write(blob.concat('', '').join('\r\n'), 'utf8')
@connection.setTimeout(0)
@connection.setNoDelay(true)
@connection.setEncoding('binary')
catch e
@didClose()
return
@buffer = new Buffer(0)
@didMessage(head)
return
_setup: ->
@close_cb = () => @didClose()
@connection.addListener('end', @close_cb)
@data_cb = (data) => @didMessage(data)
@connection.addListener('data', @data_cb)
_cleanup: ->
@connection.removeListener('end', @close_cb)
@connection.removeListener('data', @data_cb)
@close_cb = @data_cb = undefined
didClose: ->
if @connection
@_cleanup()
try
@connection.end()
catch x
@connection = undefined
didMessage: (bin_data) ->
@buffer = utils.buffer_concat(@buffer, new Buffer(bin_data, 'binary'))
if @sec is false or @buffer.length >= 8
@gotEnough()
gotEnough: ->
@_cleanup()
if @sec
nonce = @buffer.slice(0, 8)
@buffer = @buffer.slice(8)
reply = validateCrypto(@req.headers, nonce)
if reply is false
@didClose()
return false
try
@connection.write(reply, 'binary')
catch x
@didClose()
return false
# websockets possess no session_id
session = transport.Session.bySessionIdOrNew(undefined, @req.sockjs_server)
session.register( new WebSocketReceiver(@connection) )
class WebSocketReceiver extends transport.ConnectionReceiver
protocol: "websocket"
constructor: ->
@recv_buffer = new Buffer(0)
super
setUp: ->
@data_cb = (data) => @didMessage(data)
@connection.addListener('data', @data_cb)
super
tearDown: ->
@connection.removeListener('data', @data_cb)
@data_cb = undefined
super
didMessage: (bin_data) ->
if bin_data
@recv_buffer = utils.buffer_concat(@recv_buffer, new Buffer(bin_data, 'binary'))
buf = @recv_buffer
# TODO: support length in framing
if buf.length is 0
return
if buf[0] is 0x00
for i in [1...buf.length]
if buf[i] is 0xff
payload = buf.slice(1, i).toString('utf8')
@recv_buffer = buf.slice(i+1)
if @session
@session.didMessage(JSON.parse(payload))
return @didMessage()
# wait for more data
return
else if buf[0] is 0xff and buf[1] is 0x00
@didClose(1001, "Socket closed by the client")
else
@didClose(1002, "Broken framing")
return
doSendFrame: (payload) ->
# 6 bytes for every char shall be enough for utf8
a = new Buffer((payload.length+2)*6)
l = 0
l += a.write('\u0000', l, 'binary')
l += a.write('' + payload, l, 'utf-8')
l += a.write('\uffff', l, 'binary')
super(a.slice(0, l), 'binary')
exports.WebHandshakeHixie76 = WebHandshakeHixie76