From 23de6eff18cbf0fb4576b06e4cf00f2d783bb372 Mon Sep 17 00:00:00 2001 From: Eric Forgy Date: Tue, 13 Feb 2018 12:49:48 +0800 Subject: [PATCH] Use proposed new WebSockets.jl Supports SSLContext --- src/WebSockets.jl | 687 ++++++++++++++++++++++++++++++--------------- test/WebSockets.jl | 55 +++- 2 files changed, 501 insertions(+), 241 deletions(-) diff --git a/src/WebSockets.jl b/src/WebSockets.jl index 9019f2052..1c93a2fb4 100644 --- a/src/WebSockets.jl +++ b/src/WebSockets.jl @@ -1,90 +1,467 @@ +__precompile__() +""" + WebSockets +This module implements the server side of the WebSockets protocol. Some +things would need to be added to implement a WebSockets client, such as +masking of sent frames. + +WebSockets expects to be used with HttpServer to provide the HttpServer +for accepting the HTTP request that begins the opening handshake. WebSockets +implements a subtype of the WebSocketInterface from HttpServer; this means +that you can create a WebSocketsHandler and pass it into the constructor for +an http server. + + Future improvements: +1. Logging of refused requests and closures due to bad behavior of client. +2. Better error handling (should we always be using "error"?) +3. Unit tests with an actual client -- to automatically test the examples. +4. Send close messages with status codes. +5. Allow users to receive control messages if they want to. +""" module WebSockets -using ..Base64 -using MbedTLS: digest, MD_SHA1, SSLContext -import ..HTTP -using ..IOExtras -using ..Streams -import ..ConnectionPool -using HTTP: header -import ..@debug, ..DEBUG_LEVEL, ..@require, ..precondition_error +using HTTP +import HTTP: digest, MD_SHA1 +export WebSocket, + write, + read, + close, + send_ping, + send_pong -const WS_FINAL = 0x80 -const WS_CONTINUATION = 0x00 -const WS_TEXT = 0x01 -const WS_BINARY = 0x02 -const WS_CLOSE = 0x08 -const WS_PING = 0x09 -const WS_PONG = 0x0A +const TCPSock = Base.TCPSocket -const WS_MASK = 0x80 +@enum ReadyState CONNECTED=0x1 CLOSING=0x2 CLOSED=0x3 +""" Buffer writes to socket till flush (sock)""" +init_socket(sock) = Base.buffer_writes(sock) + + +struct WebSocketClosedError <: Exception end +Base.showerror(io::IO, e::WebSocketClosedError) = print(io, "Error: client disconnected") struct WebSocketError <: Exception status::Int16 message::String end +""" +A WebSocket is a wrapper over a TcpSocket. It takes care of wrapping outgoing +data in a frame and unwrapping (and concatenating) incoming data. +""" +mutable struct WebSocket{T <: IO} <: IO + socket::T + server::Bool + state::ReadyState -struct WebSocketHeader - opcode::UInt8 - final::Bool - length::UInt - hasmask::Bool - mask::UInt32 + function WebSocket{T}(socket::T,server::Bool) where T + init_socket(socket) + new(socket, server, CONNECTED) + end +end +WebSocket(socket,server) = WebSocket{typeof(socket)}(socket,server) + +# WebSocket Frames +# +# 0 1 2 3 +# 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1 +# +-+-+-+-+-------+-+-------------+-------------------------------+ +# |F|R|R|R| opcode|M| Payload len | Extended payload length | +# |I|S|S|S| (4) |A| (7) | (16/64) | +# |N|V|V|V| |S| | (if payload len==126/127) | +# | |1|2|3| |K| | | +# +-+-+-+-+-------+-+-------------+ - - - - - - - - - - - - - - - + +# | Extended payload length continued, if payload len == 127 | +# + - - - - - - - - - - - - - - - +-------------------------------+ +# | |Masking-key, if MASK set to 1 | +# +-------------------------------+-------------------------------+ +# | Masking-key (continued) | Payload Data | +# +-------------------------------- - - - - - - - - - - - - - - - + +# : Payload Data continued ... : +# + - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - + +# | Payload Data continued ... | +# +---------------------------------------------------------------+ +# + + +# Opcode values +""" * %x0 denotes a continuation frame""" +const OPCODE_CONTINUATION = 0x00 +""" * %x1 denotes a text frame""" +const OPCODE_TEXT = 0x1 +""" * %x2 denotes a binary frame""" +const OPCODE_BINARY = 0x2 +# * %x3-7 are reserved for further non-control frames +# +""" * %x8 denotes a connection close""" +const OPCODE_CLOSE = 0x8 +""" * %x9 denotes a ping""" +const OPCODE_PING = 0x9 +""" * %xA denotes a pong""" +const OPCODE_PONG = 0xA +# * %xB-F are reserved for further control frames + + +""" +Handshakes with subprotocols are rejected by default. +Add to supported SUBProtocols through e.g. +# Examples +``` + WebSockets.addsubproto("special-protocol") + WebSockets.addsubproto("json") +``` +In the general websocket handler function, specialize +further by checking +# Example +``` +if get(wsrequest.headers, "Sec-WebSocket-Protocol", "") = "special-protocol" + specialhandler(websocket) +else + generalhandler(websocket) end +``` +""" +const SUBProtocols= Array{String,1}() +"Used in handshake. See SUBProtocols" +hasprotocol(s::String) = in(s,SUBProtocols) -mutable struct WebSocket{T <: IO} <: IO - io::T - frame_type::UInt8 - server::Bool - rxpayload::Vector{UInt8} - txpayload::Vector{UInt8} - txclosed::Bool - rxclosed::Bool +"Used to specify handshake response. See SUBProtocols" +function addsubproto(name) + push!(SUBProtocols, string(name)) + return true +end +""" + write_fragment(io, islast, opcode, hasmask, data::Array{UInt8}) +Write the raw frame to a bufffer +""" +function write_fragment(io::IO, islast::Bool, opcode, hasmask::Bool, data::Array{UInt8}) + l = length(data) + b1::UInt8 = (islast ? 0b1000_0000 : 0b0000_0000) | opcode + + mask::UInt8 = hasmask ? 0b1000_0000 : 0b0000_0000 + + write(io, b1) + if l <= 125 + write(io, mask | UInt8(l)) + elseif l <= typemax(UInt16) + write(io, mask | UInt8(126)) + write(io, hton(UInt16(l))) + elseif l <= typemax(UInt64) + write(io, mask | UInt8(127)) + write(io, hton(UInt64(l))) + else + error("Attempted to send too much data for one websocket fragment\n") + end + hasmask && write(io,mask!(data)) + write(io, data) end -function WebSocket(io::T; server=false, binary=false) where T <: IO - WebSocket{T}(io, binary ? WS_BINARY : WS_TEXT, server, - UInt8[], UInt8[], false, false) +""" Write without interruptions""" +function locked_write(io::IO, islast::Bool, opcode, hasmask, data) + isa(io, TCPSock) && lock(io.lock) + try + write_fragment(io, islast, opcode, hasmask, Vector{UInt8}(data)) + finally + if isa(io, TCPSock) + flush(io) + unlock(io.lock) + end + end end +""" Write text data; will be sent as one frame.""" +function Base.write(ws::WebSocket,data::String) + if !isopen(ws) + @show ws + error("Attempted write to closed WebSocket\n") + end + locked_write(ws.socket, true, OPCODE_TEXT, !ws.server, data) +end +""" Write binary data; will be sent as one frame.""" +function Base.write(ws::WebSocket, data::Array{UInt8}) + if !isopen(ws) + @show ws + error("attempt to write to closed WebSocket\n") + end + locked_write(ws.socket, true, OPCODE_BINARY, !ws.server, data) +end -# Handshake -is_websocket_upgrade(r::HTTP.Message) = - (r isa HTTP.Request && r.method == "GET" || r.status == 101) && - HTTP.hasheader(r, "Connection", "upgrade") && - HTTP.hasheader(r, "Upgrade", "webscoket") +function write_ping(io::IO, hasmask, data = "") + locked_write(io, true, OPCODE_PING, hasmask, data) +end +""" Send a ping message, optionally with data.""" +send_ping(ws, data...) = write_ping(ws.socket, !ws.server, data...) -function check_upgrade(http) +function write_pong(io::IO, hasmask, data = "") + locked_write(io, true, OPCODE_PONG, hasmask, data) +end +""" Send a pong message, optionally with data.""" +send_pong(ws, data...) = write_pong(ws.socket, !ws.server, data...) + +""" + close(ws::WebSocket) +Send a close message. +""" +function Base.close(ws::WebSocket) + if !isopen(ws) + error("Attempt to close closed WebSocket") + end + + # Ask client to acknowledge closing the connection + locked_write(ws.socket, true, OPCODE_CLOSE, !ws.server, "") + ws.state = CLOSING + + # Wait till the client responds with an OPCODE_CLOSE. This process is + # complicated by potential blocking reads on the WebSocket in other Tasks + # which may receive the response control frame. Synchronization of who is + # responsible for closing the underlying socket is done using the + # WebSocket's state. When this side initiates closing the connection it is + # responsible for cleaning up, when the other side initiates the close the + # read method is + # + # The exception handling is necessary as read_frame will error when the + # OPCODE_CLOSE control frame is received by a potentially blocking read in + # another Task + try + while ws.state === CLOSING + wsf = read_frame(ws.socket) + # ALERT: stuff might get lost in ether here + if is_control_frame(wsf) && (wsf.opcode == OPCODE_CLOSE) + ws.state = CLOSED + end + end + + close(ws.socket) + catch exception + !isa(exception, EOFError) && rethrow(exception) + end +end +""" + isopen(WebSocket)-> Bool +A WebSocket is closed if the underlying TCP socket closes, or if we send or +receive a close message. +""" +Base.isopen(ws::WebSocket) = (ws.state === CONNECTED) && isopen(ws.socket) + + +""" Represents one (received) message frame.""" +mutable struct WebSocketFragment + is_last::Bool + rsv1::Bool + rsv2::Bool + rsv3::Bool + opcode::UInt8 # This is actually a UInt4 value. + is_masked::Bool + payload_len::UInt64 + maskkey::Vector{UInt8} # This will be 4 bytes on frames from the client. + data::Vector{UInt8} # For text messages, this is a String. +end + +""" This constructor handles conversions from bytes to bools.""" +function WebSocketFragment( + fin::UInt8 + , rsv1::UInt8 + , rsv2::UInt8 + , rsv3::UInt8 + , opcode::UInt8 + , masked::UInt8 + , payload_len::UInt64 + , maskkey::Vector{UInt8} + , data::Vector{UInt8}) + + WebSocketFragment( + fin != 0 + , rsv1 != 0 + , rsv2 != 0 + , rsv3 != 0 + , opcode + , masked != 0 + , payload_len + , maskkey + , data) +end + +""" Control frames have opcodes with the highest bit = 1.""" +is_control_frame(msg::WebSocketFragment) = (msg.opcode & 0b0000_1000) > 0 + +""" Respond to pings, ignore pongs, respond to close.""" +function handle_control_frame(ws::WebSocket,wsf::WebSocketFragment) + if wsf.opcode == OPCODE_CLOSE + # A close OPCODE can be received for two reasons. Either the other side + # is initiating a disconnection, or the this side is (through a call to + # close on the WebSocket) and the client has replied that it is okay + # with closing the connection. This can be derived from the current + # state of the WebSocket + if ws.state !== CLOSING + # The other side initiated the disconnect, so the action must be + # acknowledged by replying with an empty CLOSE frame and cleaning + # up + try + locked_write(ws.socket, true, OPCODE_CLOSE, !ws.server, "") + catch exception + # On sudden disconnects, the other side may be gone before the + # close acknowledgement can be sent. This will cause an + # ArgumentError to be thrown due to the underlying stream being + # closed. These are swallowed here and will be replaced by a + # WebSocketClosedError below + !isa(exception, ArgumentError) && rethrow(exception) + end - if !hasheader(http, "Upgrade", "websocket") - throw(WebSocketError(0, "Expected \"Upgrade: websocket\"!\n" * - "$(http.message)")) + close(ws.socket) + end + + # In the other case the close method is expected to clean-up, which can + # be triggered by changing the state of the WebSocket + ws.state = CLOSED + + throw(WebSocketClosedError()) + elseif wsf.opcode == OPCODE_PING + send_pong(ws,wsf.data) + elseif wsf.opcode == OPCODE_PONG + # Nothing to do here; no reply is needed for a pong message. + else # %xB-F are reserved for further control frames + error("Unknown opcode $(wsf.opcode)") end +end - if !hasheader(http, "Connection", "upgrade") - throw(WebSocketError(0, "Expected \"Connection: upgrade\"!\n" * - "$(http.message)")) +""" Read a frame: turn bytes from the websocket into a WebSocketFragment.""" +function read_frame(io::IO) + ab = read(io,2) + a = ab[1] + fin = a & 0b1000_0000 >>> 7 # If fin, then is final fragment + rsv1 = a & 0b0100_0000 # If not 0, fail. + rsv2 = a & 0b0010_0000 # If not 0, fail. + rsv3 = a & 0b0001_0000 # If not 0, fail. + opcode = a & 0b0000_1111 # If not known code, fail. + # TODO: add validation somewhere to ensure rsv, opcode, mask, etc are valid. + + b = ab[2] + mask = b & 0b1000_0000 >>> 7 + hasmask = mask != 0 + + # if mask != 1 + # error("WebSocket reader cannot handle incoming messages without mask. " * + # "See http://tools.ietf.org/html/rfc6455#section-5.3") + # end + + payload_len::UInt64 = b & 0b0111_1111 + if payload_len == 126 + payload_len = ntoh(read(io,UInt16)) # 2 bytes + elseif payload_len == 127 + payload_len = ntoh(read(io,UInt64)) # 8 bytes end + + maskkey = hasmask ? read(io,4) : UInt8[] + + data = read(io,Int(payload_len)) + hasmask && mask!(data,maskkey) + + return WebSocketFragment(fin,rsv1,rsv2,rsv3,opcode,mask,payload_len,maskkey,data) end +""" + read(ws::WebSocket) +Read one non-control message from a WebSocket. Any control messages that are +read will be handled by the handle_control_frame function. This function will +not return until a full non-control message has been read. If the other side +doesn't ever complete its message, this function will never return. Only the +data (contents/body/payload) of the message will be returned from this +function. +""" +function Base.read(ws::WebSocket) + if !isopen(ws) + error("Attempt to read from closed WebSocket") + end + frame = read_frame(ws.socket) + # Handle control (non-data) messages. + if is_control_frame(frame) + # Don't return control frames; they're not interesting to users. + handle_control_frame(ws,frame) -function accept_hash(key) + # Recurse to return the next data frame. + return read(ws) + end + + # Handle data message that uses multiple fragments. + if !frame.is_last + return vcat(frame.data, read(ws)) + end + + return frame.data +end + +""" + WebSocket Handshake Procedure +`generate_websocket_key(key)` transforms a websocket client key into the server's accept +value. This is done in three steps: +1. Concatenate key with magic string from RFC. +2. SHA1 hash the resulting base64 string. +3. Encode the resulting number in base64. +This function then returns the string of the base64-encoded value. +""" +function generate_websocket_key(key) hashkey = "$(key)258EAFA5-E914-47DA-95CA-C5AB0DC85B11" return base64encode(digest(MD_SHA1, hashkey)) end +# """ +# Responds to a WebSocket handshake request. +# Checks for required headers and subprotocols; sends Response(400) if they're missing or bad. Otherwise, transforms client key into accept value, and sends Reponse(101). +# Function returns true for accepted handshakes. +# """ +# function websocket_handshake(request,client) +# if !haskey(request.headers, "Sec-WebSocket-Key") +# Base.write(client.sock, Response(400)) +# return false +# end +# if get(request.headers, "Sec-WebSocket-Version", "13") != "13" +# response = Response(400) +# response.headers["Sec-WebSocket-Version"] = "13" +# Base.write(client.sock, response) +# return false +# end + +# key = request.headers["Sec-WebSocket-Key"] +# if length(decode(Base64,key)) != 16 # Key must be 16 bytes +# Base.write(client.sock, Response(400)) +# return false +# end +# resp_key = generate_websocket_key(key) + +# response = Response(101) +# response.headers["Upgrade"] = "websocket" +# response.headers["Connection"] = "Upgrade" +# response.headers["Sec-WebSocket-Accept"] = resp_key + +# if haskey(request.headers, "Sec-WebSocket-Protocol") +# if hasprotocol(request.headers["Sec-WebSocket-Protocol"]) +# response.headers["Sec-WebSocket-Protocol"] = request.headers["Sec-WebSocket-Protocol"] +# else +# Base.write(client.sock, Response(400)) +# return false +# end +# end + +# Base.write(client.sock, response) +# return true +# end + +function mask!(data, mask=rand(UInt8, 4)) + for i in 1:length(data) + data[i] = data[i] ⊻ mask[((i-1) % 4)+1] + end + return mask +end function open(f::Function, url; binary=false, verbose=false, kw...) - key = base64encode(rand(UInt8, 16)) + key = HTTP.base64encode(rand(UInt8, 16)) headers = [ "Upgrade" => "websocket", @@ -96,7 +473,7 @@ function open(f::Function, url; binary=false, verbose=false, kw...) HTTP.open("GET", url, headers; reuse_limit=0, verbose=verbose ? 2 : 0, kw...) do http - startread(http) + HTTP.startread(http) status = http.message.status if status != 101 @@ -105,208 +482,66 @@ function open(f::Function, url; binary=false, verbose=false, kw...) check_upgrade(http) - if header(http, "Sec-WebSocket-Accept") != accept_hash(key) + if HTTP.header(http, "Sec-WebSocket-Accept") != generate_websocket_key(key) throw(WebSocketError(0, "Invalid Sec-WebSocket-Accept\n" * "$(http.message)")) end - io = ConnectionPool.getrawstream(http) - f(WebSocket(io; binary=binary)) + io = HTTP.ConnectionPool.getrawstream(http) + f(WebSocket(io,false)) end end +# mutable struct WebSocket +# id::Int +# socket::TCPSock +# state::ReadyState -function listen(f::Function, - host::String="localhost", port::UInt16=UInt16(8081); - binary=false, verbose=false) - - HTTP.listen(host, port; verbose=verbose) do http - upgrade(f, http; binary=binary) - end -end +# function WebSocket(id::Int,socket::TCPSock) +# init_socket(socket) +# new(id, socket, CONNECTED) +# end +# end +# function listen(f::Function, host::String="localhost", port::UInt16=UInt16(8081); binary=false, verbose=false) +# HTTP.listen(host, port; verbose=verbose) do http +# upgrade(f, http; binary=binary) +# end +# end function upgrade(f::Function, http::HTTP.Stream; binary=false) check_upgrade(http) - if !hasheader(http, "Sec-WebSocket-Version", "13") - throw(WebSocketError(0, "Expected \"Sec-WebSocket-Version: 13\"!\n" * - "$(http.message)")) + if !HTTP.hasheader(http, "Sec-WebSocket-Version", "13") + throw(WebSocketError(0, "Expected \"Sec-WebSocket-Version: 13\"!\n$(http.message)")) end - setstatus(http, 101) - setheader(http, "Upgrade" => "websocket") - setheader(http, "Connection" => "Upgrade") - key = header(http, "Sec-WebSocket-Key") - setheader(http, "Sec-WebSocket-Accept" => accept_hash(key)) + HTTP.setstatus(http, 101) + HTTP.setheader(http, "Upgrade" => "websocket") + HTTP.setheader(http, "Connection" => "Upgrade") + key = HTTP.header(http, "Sec-WebSocket-Key") + HTTP.setheader(http, "Sec-WebSocket-Accept" => generate_websocket_key(key)) - startwrite(http) + HTTP.startwrite(http) - io = ConnectionPool.getrawstream(http) - f(WebSocket(io; binary=binary, server=true)) + io = HTTP.ConnectionPool.getrawstream(http) + f(WebSocket(io, true)) end - - -# Sending Frames - - -function Base.unsafe_write(ws::WebSocket, p::Ptr{UInt8}, n::UInt) - return wswrite(ws, unsafe_wrap(Array, p, n)) -end - - -function Base.write(ws::WebSocket, x1, x2, xs...) - local n::Int = 0 - n += wswrite(ws, ws.frame_type, x1) - xs = (x2, xs...) - l = length(xs) - for i in 1:l - n += wswrite(ws, i == l ? WS_FINAL : WS_CONTINUATION, xs[i]) - end - return n -end - - -function IOExtras.closewrite(ws::WebSocket) - @require !ws.txclosed - opcode = WS_FINAL | WS_CLOSE - @debug 1 "WebSocket ⬅️ $(WebSocketHeader(opcode, 0x00))" - write(ws.io, opcode, 0x00) - ws.txclosed = true -end - - -wslength(l) = l < 0x7E ? (UInt8(l), UInt8[]) : - l <= 0xFFFF ? (0x7E, reinterpret(UInt8, [hton(UInt16(l))])) : - (0x7F, reinterpret(UInt8, [hton(UInt64(l))])) - - -wswrite(ws::WebSocket, x) = wswrite(ws, WS_FINAL | ws.frame_type, x) - -wswrite(ws::WebSocket, opcode::UInt8, x) = wswrite(ws, opcode, Vector{UInt8}(x)) - -function wswrite(ws::WebSocket, opcode::UInt8, bytes::Vector{UInt8}) - - n = length(bytes) - len, extended_len = wslength(n) - if ws.server - mask = UInt8[] - ws.txpayload = bytes - else - len |= WS_MASK - mask = mask!(ws.txpayload, bytes, n) - end - - @debug 1 "WebSocket ⬅️ $(WebSocketHeader(opcode, len, extended_len, mask))" - write(ws.io, opcode, len, extended_len, mask) - - @debug 2 " ⬅️ $(ws.txpayload[1:n])" - unsafe_write(ws.io, pointer(ws.txpayload), n) -end - - -function mask!(to, from, l, mask=rand(UInt8, 4)) - if length(to) < l - resize!(to, l) - end - for i in 1:l - to[i] = from[i] ⊻ mask[((i-1) % 4)+1] - end - return mask -end - - -function Base.close(ws::WebSocket) - if !ws.txclosed - closewrite(ws) - end - while !ws.rxclosed - readframe(ws) - end -end - - -Base.isopen(ws::WebSocket) = !ws.rxclosed - - - -# Receiving Frames - -Base.eof(ws::WebSocket) = eof(ws.io) - -Base.readavailable(ws::WebSocket) = collect(readframe(ws)) - - -function readheader(io::IO) - b = UInt8[0,0] - read!(io, b) - len = b[2] & ~WS_MASK - WebSocketHeader( - b[1] & 0x0F, - b[1] & WS_FINAL > 0, - len == 0x7F ? UInt(ntoh(read(io, UInt64))) : - len == 0x7E ? UInt(ntoh(read(io, UInt16))) : UInt(len), - b[2] & WS_MASK > 0, - b[2] & WS_MASK > 0 ? read(io, UInt32) : UInt32(0)) -end - - -function readframe(ws::WebSocket) - h = readheader(ws.io) - @debug 1 "WebSocket ➡️ $h" - - if h.length > 0 - if length(ws.rxpayload) < h.length - resize!(ws.rxpayload, h.length) - end - unsafe_read(ws.io, pointer(ws.rxpayload), h.length) - @debug 2 " ➡️ \"$(String(ws.rxpayload[1:h.length]))\"" +function check_upgrade(http) + if !HTTP.hasheader(http, "Upgrade", "websocket") + throw(WebSocketError(0, "Expected \"Upgrade: websocket\"!\n$(http.message)")) end - if h.opcode == WS_CLOSE - ws.rxclosed = true - if h.length >= 2 - status = UInt16(ws.rxpayload[1]) << 8 | ws.rxpayload[2] - if status != 1000 - message = String(ws.rxpayload[3:h.length]) - throw(WebSocketError(status, message)) - end - end - return UInt8[] - elseif h.opcode == WS_PING - write(ws.io, [WS_PONG, 0x00]) - wswrite(ws, WS_FINAL | WS_PONG, ws.rxpayload) - return readframe(ws) - else - l = Int(h.length) - if h.hasmask - mask!(ws.rxpayload, ws.rxpayload, l, reinterpret(UInt8, [h.mask])) - end - return view(ws.rxpayload, 1:l) + if !HTTP.hasheader(http, "Connection", "upgrade") + throw(WebSocketError(0, "Expected \"Connection: upgrade\"!\n$(http.message)")) end end -function WebSocketHeader(bytes...) - io = IOBuffer() - write(io, bytes...) - seek(io, 0) - return readheader(io) -end - -function Base.show(io::IO, h::WebSocketHeader) - print(io, "WebSocketHeader(", - h.opcode == WS_CONTINUATION ? "CONTINUATION" : - h.opcode == WS_TEXT ? "TEXT" : - h.opcode == WS_BINARY ? "BINARY" : - h.opcode == WS_CLOSE ? "CLOSE" : - h.opcode == WS_PING ? "PING" : - h.opcode == WS_PONG ? "PONG" : h.opcode, - h.final ? " | FINAL, " : ", ", - h.length > 0 ? "$(Int(h.length))-byte payload" : "", - h.hasmask ? ", mask = $(hex(h.mask))" : "", - ")") +function is_upgrade(r::HTTP.Message) + (r isa HTTP.Request && r.method == "GET" || r.status == 101) && + HTTP.hasheader(r, "Connection", "upgrade") && + HTTP.hasheader(r, "Upgrade", "websocket") end - end # module WebSockets diff --git a/test/WebSockets.jl b/test/WebSockets.jl index 303e9310e..dc163bcca 100644 --- a/test/WebSockets.jl +++ b/test/WebSockets.jl @@ -1,28 +1,53 @@ using HTTP -using HTTP.Test +using HTTP.WebSockets using HTTP.IOExtras +using Base.Test -@testset "WebSockets" begin +import HTTP.WebSockets: CONNECTED, CLOSING, CLOSED -for s in ["ws", "wss"] +@testset "WebSockets" begin - HTTP.WebSockets.open("$s://echo.websocket.org") do io - write(io, HTTP.bytes("Foo")) - @test !eof(io) - @test String(readavailable(io)) == "Foo" +info("Testing ws...") +WebSockets.open("ws://echo.websocket.org") do ws + write(ws, "Foo") + @test String(read(ws)) == "Foo" - write(io, HTTP.bytes("Hello")) - write(io, " There") - write(io, " World", "!") - closewrite(io) + close(ws) +end +sleep(1) - buf = IOBuffer() - write(buf, io) - @test String(take!(buf)) == "Hello There World!" +info("Testing wss...") +WebSockets.open("wss://echo.websocket.org") do ws + write(ws, "Foo") + @test String(read(ws)) == "Foo" - close(io) + close(ws) +end +sleep(1) + +p = UInt16(8000) +@async HTTP.listen("127.0.0.1",p) do http + if WebSockets.is_upgrade(http.message) + WebSockets.upgrade(http) do ws + while ws.state == CONNECTED + data = String(read(ws)) + write(ws,data) + end + end end +end + +sleep(2) + +info("Testing local server...") +WebSockets.open("ws://127.0.0.1:$(p)") do ws + write(ws, "Foo") + @test String(read(ws)) == "Foo" + + write(ws, "Bar") + @test String(read(ws)) == "Bar" + close(ws) end end # testset