simple_rpc

RPC Server and Client for Crystal. Implements msgpack-rpc protocol.
1.3.1 released
kostya/simple_rpc
65 3
Kostya M

simple_rpc

Build Status

Remote Procedure Call Server and Client for Crystal. Implements msgpack-rpc protocall. Designed to be reliable and stable (catch every possible protocall/socket errors). It also quite performant: benchmark reach 201660 rps in pool mode (single server core, single client core).

Installation

Add this to your application's shard.yml:

dependencies:
  simple_rpc:
    github: kostya/simple_rpc

Usage

require "simple_rpc"

# Example run server and client.

class MyRpc
  # When including SimpleRpc::Proto, all public instance methods inside class,
  # would be exposed to external rpc call.
  # Each method should define type for each argument, and also return type.
  # (Types of arguments should supports MessagePack::Serializable).
  # Instance of this class created on server for each call.
  include SimpleRpc::Proto

  def bla(x : Int32, y : String) : Float64
    x * y.to_f
  end
end

spawn do
  # running RPC server on 9000 port in background fiber
  MyRpc::Server.new("127.0.0.1", 9000).run
end

# wait until server up
sleep 0.1

# create rpc client
client = MyRpc::Client.new("127.0.0.1", 9000)
result = client.bla!(3, "5.5") # here can raise SimpleRpc::Errors
p result # => 16.5

When client code have no access to server proto, you can call raw requests:

require "simple_rpc"

client = SimpleRpc::Client.new("127.0.0.1", 9000)
result = client.request!(Float64, :bla, 3, "5.5") # here can raise SimpleRpc::Errors
p result # => 16.5

When you dont want to raises on problems, you can check result by yourself:

require "simple_rpc"

client = SimpleRpc::Client.new("127.0.0.1", 9000)
result = client.request(Float64, :bla, 3, "5.5") # here can raise SimpleRpc::Errors
if result.ok?
  p result.value! # => 16.5
else
  p result.message!
end

If you dont know what return type is, use MessagePack::Any:

require "simple_rpc"

client = SimpleRpc::Client.new("127.0.0.1", 9000)
result = client.request!(MessagePack::Any, :bla, 3, "5.5")
p result.as_f + 1 # => 17.5

If you want to exchange complex data types, you should include MessagePack::Serializable to your data

require "simple_rpc"

record MyResult, a : Int32, b : String { include MessagePack::Serializable }

class MyRequest
  include MessagePack::Serializable

  property a : Int32
  property b : Hash(String, String)?

  @[MessagePack::Field(ignore: true)]
  property c : Int32?
end

class MyRpc 
  include SimpleRpc::Proto

  def doit(req : MyRequest) : MyResult
    # ...
  end
end

Example calling from Ruby, with gem msgpack-rpc

require 'msgpack/rpc'

client = MessagePack::RPC::Client.new('127.0.0.1', 9000)
result = client.call(:bla, 3, "5.5")
p result # => 16.5
simple_rpc:
  github: kostya/simple_rpc
  version: ~> 1.3.1
License MIT
Crystal 0.27.0

Authors

Dependencies 2

  • msgpack >=0.13.1
    {'github' => 'crystal-community/msgpack-crystal', 'version' => '>=0.13.1'}
  • pool
    {'github' => 'ysbaddaden/pool'}

Development Dependencies 0

Dependents 0

Last synced .
search fire star recently