ws-client

  1. ws-client
    1. Description
    2. Requirements
    3. API
      1. Establishing a WebSocket connection
      2. Sending and receiving messages
      3. Closing a WebSocket connection
      4. Low-level interface
      5. The permessage-deflate extension
      6. Handling exceptions
    4. Examples
    5. Author
    6. Repository
    7. License
    8. Version History

Description

An implementation of the client side of the WebSocket Protocol (RFC6455), including the permessage-deflate extension (RFC7692). It passes all of the Autobahn Testsuite compliance tests.

Requirements

External dependencies:

Uses tcp6 if it is imported together with this extension.

API

Establishing a WebSocket connection
[record] ws-connection
[procedure] (ws-connect STRING #!optional (list WS-EXTENSION ...) (list SYMBOL ...)) -> (or WS-CONNECTION RECORD)

The state of a WebSocket connection is stored in a record of type ws-connection.

The procedure ws-connect attempts to establish a WebSocket connection to a server. The argument STRING should be a valid WebSocket URI.

If the opening handshake for the WebSocket protocol succeeds, this procedure returns a ws-connection record. If it receives a HTTP response from the server with status code other than 101, it instead returns a record containing the HTTP response (see intarweb). The user should call is-ws-connection? on the result of ws-connect to check if this has happened and, in the case of an unsuccessful handshake, process the HTTP response accordingly, e.g. by following a redirect.

To use WebSockets over TLS, supply a URI with scheme wss.

The procedure ws-connect optionally accepts a list of extensions the client hopes to use. The only extension currently supported by this library is permessage-deflate; see the relevant section.

The second optional argument is a list of flags (symbols). The only one currently supported is 'strip-host, which strips the hostname, port, and scheme from the method line of the client handshake. Some non-compliant implementations seem to require this.

Sending and receiving messages
[record] ws-message
[procedure] (message-type WS-MESSAGE) -> SYMBOL
[procedure] (message-data* WS-MESSAGE) -> U8VECTOR
[procedure] (message-frames WS-MESSAGE) -> U8VECTOR

WebSocket messages are represented with the record type ws-message.

For a message m, The procedure (message-type m) yields the type of m as specified by the opcode of its first frame, which is either 'text or 'binary.

If m is a message obtained with recv-message, (message-frames m) is a list of frames from which m has been assembled, after any per-frame transforms defined by extensions have been applied. In the absence of extensions which apply per-message transforms, (message-data* m) is the concatenation of the contents of each (frame-payload-data f) for f in (message-frames m).

[procedure] (message-size WS-MESSAGE) -> INTEGER

Equivalent to (u8vector-length (message-data* WS-MESSAGE)).

[procedure] (message-data WS-MESSAGE) -> STRING or BLOB

The procedure (message-data m) yields (message-data* m) converted to the corresponding type depending on (message-type m).

[procedure] (recv-message WS-CONNECTION) -> WS-MESSAGE or FALSE

Receives a message from the server. The procedure quietly processes any control frames encountered while assembling the message.

If it encounters a connection-close frame, it closes the connection and returns #f instead. Once this has happened, the application should not invoke recv-message again on the same connection.

[procedure] (send-message WS-CONNECTION WS-MESSAGE)

Sends a message to the server. Currently only useful for echoing received messages; see instead send-text-message and send-binary-message below.

[procedure] (send-text-message WS-CONNECTION STRING)
[procedure] (send-binary-message WS-CONNECTION BLOB)

Sends a text (resp. binary) message to (resp. from) the server.

[procedure] (recv-message-loop WS-CONNECTION HANDLER)

Receives messages from the server and calls the procedure HANDLER on each received message, until the connection is closed. The procedure responds to (or sends, e.g. in the case of closure due to a protocol error) connection close frames so that the connection is closed cleanly.

Closing a WebSocket connection
[procedure] (ws-close WS-CONNECTION SYMBOL)

Sends a frame with optype 'connection-close to the server, with payload the close code corresponding to the reason given by SYMBOL (see reason->close-code).

Once the application has called ws-close with a connection, it should not send any further data through the connection, though it may still receive messages until the server sends a connection close frame in return.

Low-level interface

An interface is provided for when an application wishes to interact with the connection on the level of individual WebSocket frames.

Care should be taken if these procedures are used in conjunction with the message-level interface: for example, if the first frame of a fragmented message has been consumed using recv-frame, invoking recv-message results in an error on the next frame since there is nothing to continue.

[record] ws-frame
[procedure] (frame-fin WS-FRAME) -> BOOLEAN
[procedure] (frame-rsv WS-FRAME) -> INTEGER
[procedure] (frame-opcode WS-FRAME) -> INTEGER
[procedure] (frame-optype WS-FRAME) -> SYMBOL
[procedure] (frame-mask? WS-FRAME) -> BOOLEAN
[procedure] (frame-payload-length WS-FRAME) -> INTEGER
[procedure] (frame-payload-data WS-FRAME) -> U8VECTOR

WebSocket messages are represented with the record type ws-frame.

For a frame f:

[procedure] (frame-rsv-bit WS-FRAME N) -> BOOLEAN

Is #t iff the bitwise-and of N with the result of frame-rsv is nonzero. For example, (frame-rsv-bit f 4) is #t iff the RSV1 bit is set on the frame f.

[procedure] (recv-frame WS-CONNECTION) -> WS-FRAME
[procedure] (send-frame WS-CONNECTION WS-FRAME)

Receives (resp. sends) a frame from (resp. to) the server.

[procedure] (opcode->optype INTEGER) -> SYMBOL
[procedure] (optype->opcode SYMBOL) -> INTEGER

Maps between opcodes and optypes:

opcode optype
#x0 'continuation
#x1 'text
#x2 'binary
#x8 'connection-close
#x9 'ping
#xa 'pong

Signals a composite condition of kind 'websocket 'exn if an unrecognised optype or opcode is supplied. The application is expected to redefine this procedure if it wishes to make use of reserved opcodes.

[procedure] (reason->close-code SYMBOL) -> U8VECTOR
[procedure] (close-code->reason INTEGER) -> SYMBOL

Maps between reasons for closing a connection and the corresponding close codes:

code (integer) code (u8vector) reason
1000 #u8(3 232) 'normal-closure
1001 #u8(3 233) 'going-away
1002 #u8(3 234) 'protocol-error
1003 #u8(3 235) 'unsupported-data
1005 #u8(3 237) 'no-status-rcvd
1006 #u8(3 238) 'abnormal-closure
1007 #u8(3 239) 'invalid-frame-payload-data
1008 #u8(3 240) 'policy-violation
1009 #u8(3 241) 'message-too-big
1010 #u8(3 242) 'mandatory-ext
1011 #u8(3 243) 'internal-server-error
1015 #u8(3 247) 'tls-handshake

The procedure reason->close-code returns the close code as a u8vector rather than an integer; this is so that the value can be included into a frame payload without conversion.

Signals a composite condition of kind 'websocket 'exn if an unrecognised close code or reason is supplied. The application is expected to redefine this procedure if it wishes to make use of reserved close codes. Note that not all of the close codes listed here are expected to occur in a connection close frame.

The permessage-deflate extension

This library provides an implementation of the permessage-deflate extension, using zlib for compression/decompression.

[procedure] (permessage-deflate PARAMETERS) -> WS-EXTENSION

To offer to use this extension during the opening handshake, supply (list (permessage-deflate PARAMETERS)) as an optional argument to ws-connect.

The argument PARAMETERS should be a list in which each item is an alist, each of whose items is in turn a pairs of strings (PARAMETER . VALUE) or a pair of form (PARAMETER . #t). These specify the parameters to be advertised during the opening handshake.

For example, to ask the server to use an LZ77 sliding window of length no greater than 1024, but to fall back to permessage-deflate without this parameter if the server does not support it, one might specify

'((("server_max_window_bits" . "10")) ())

so that the Sec-WebSocket-Extensions header in the client opening handshake will contain the string

 permessage-deflate;server_max_window_bits=10,permessage-deflate

Handling exceptions

When a procedure encounters a situation which should result in the WebSocket connection being failed, it signals a composite condition of kind 'websocket 'fail, with the following properties:

reason
a symbol representing the reason for failing the connection, and
message
a string describing the exception.

The application may choose to catch and handle these exceptions, such as by calling ws-close with the corresponding arguments (see reason->close-code). The procedure recv-message-loop does this automatically.

For cases where it does not make sense to attempt to close the connection (for example if the underlying TCP connection fails, or if an error results from user input instead of a problem with data received from the server) a condition of kind 'websocket 'exn is signalled instead, with the following properties:

message
a string describing the exception.

Examples

A client which connects to localhost port 9001 without TLS, and echoes back every text message it receives from the server:

(import ws-client)

(let ((conn (ws-connect "ws://localhost:9001")))
  (recv-message-loop conn
    (lambda (m)
      (if (eq? 'text (message-type m))
          (send-text-message conn (message-data m))))))

See the examples folder in the source repository for more examples.

Author

Lo̍h Ka-tsùn

Repository

https://github.com/lohkatsun/ws-client-egg

License

BSD

Version History

1.0
0.2