cl-messagepack-rpc

https://github.com/adolenc/cl-messagepack-rpc.git

git clone 'https://github.com/adolenc/cl-messagepack-rpc.git'

(ql:quickload :cl-messagepack-rpc)
5

cl-messagepack-rpc

Build Status Coverage Status Quicklisp dist MIT licensed

A Common Lisp library implementing the MessagePack-RPC specification, which uses MessagePack serialization format to achieve efficient remote procedure calls (RPCs).

Library uses cl-messagepack and cl-async under the hood. Currently only the client side functionality is fully supported, but some server-side features (such as registering callbacks for sessions) are also implemented. Library supports connecting to the server via TCP sockets, named pipes or via user-specified streams (e.g. standard input and output streams).

Installing

Package is available through Quicklisp, so simply evaluating

* (ql:quickload :cl-messagepack-rpc)

from your REPL should properly load all the dependencies and cl-messagepack-rpc itself.

In order to get cl-async working, you will however also need libuv1-dev package, which you can install using your distribution's package manager, or by manually compiling libuv.

Usage

This library tries to follow the official specification as closely as possible. For a quick taste:

(ql:quickload :cl-messagepack-rpc)

(defparameter *client* (make-instance 'mrpc:client :file "/path/to/named/pipe"))
(mrpc:call *client* "echo" "Hello server!")
;=> "Hello server!"

(defparameter *future* (mrpc:call-async *client* "execute_some_long_task"))
;=> *future*

(mrpc:request *client* "add" 3 2) ; mrpc:request is an alias for mrpc:call
;=> 5

(mrpc:join *future*) ; calling join on future also returns its result (or throws an error)
;=> "Done with the execution of some long task!"

(mrpc:notify *client* "client_finished")
;=> T

Exported symbols

client (session)

Main class used to connect to an already running server. You can specify which means of transport the server uses via make-instance call: common-lisp (defparameter *client* (make-instance 'client :host "127.0.0.1" :port 1985)) ; to connect via TCP (defparameter *client2* (make-instance 'client :file "/path/to/named/pipe")) ; to connect via named pipe (defparameter *client3* (make-instance 'client)) ; to connect via standard input/output (defparameter *client4* (make-instance 'client :input-stream *standard-input* :output-stream *output-stream*)) ; *client4* is same as *client3*, but note that you can specify any (binary) input and output streams

If remote server uses extended types, you can specify that by passing a specification list via :extended-types argument to #'make-instance. For example, to translate the use case from cl-messagepack's readme, you would use: common-lisp (defparameter *client* (make-instance 'client :host "127.0.0.1" :port 1985) :extended-types '(:numeric 0 Buffer Window Tabpage)) Objects of extended type can be tested for equality using #'eq.

register-callback (session method callback)

Register a CALLBACK with name METHOD for some SESSION. Callback should be something callable: common-lisp (register-callback *client* "add" #'(lambda (a b) (+ a b)))

remove-callback (session method)

Remove a previously registered callback with name METHOD from SESSION. common-lisp (remove-callback *client* "add")

call-async (session method &rest params) ⇒ future

Use SESSION to call METHOD with PARAMS and immediately return control to the caller, returning a future object. common-lisp (call-async *client* "server_side_add" 1 2 3) ;=> #<FUTURE {100962B8F3}>

call (session method &rest params)

Invoke CALL-ASYNC on the passed arguments, and call JOIN on the returned future. common-lisp (call *client* "server_side_add" 1 2 3) ;=> 6

request (session method &rest params)

Alias for CALL.

notify (session method &rest params)

Use SESSION to call METHOD with PARAMS, immediately returning control to the caller. This call completely ignores server responses. common-lisp (notify *client* "do_something") ;=> T

future

Class used to hold responses from the server. You should not need to create future objects by hand.

join (future)

Block execution until FUTURE has a result from the server. Then either return a result, or throw an error, depending on how the server responded. ```common-lisp (let ((future (call-async client “add” 3 2))) ; do something… (join future)) ;⇒ 5

(let ((future (call-async client “add” 3 “some string”))) ; do something… (join future)) ;⇒ ERROR: unexpected types of arguments. ```

Running unit tests

Because server-side support is not yet implemented, tests use a python based server. So in order to test the library, first start the python server:

$ python t/server.py

and then evaluate

* (asdf:test-system :cl-messagepack-rpc)

in the REPL to run the actual tests.

Support

The library is developed and tested with sbcl under Debian GNU/Linux, but should work everywhere cl-async does.

License

Copyright (c) 2016 Andrej Dolenc

Licensed under the MIT License.