Rubame is a simple Ruby websocket game server. I was originally using EventMachine for a private project, but it became apparent that EM may not be the most suitable tool. EM is designed (as I understand it) for servers that will handle lots of low duration connections, whereas my project will involve only a small handful (between 1 and 20, likely) that are long lasting. However, the primary concern I had was how EM takes over the thread. In order to have game logic run, it seemed the options were to create a new thread, or use periodic timers to have a game loop called.
Neither seemed ideal, and I wanted to have more control in order to better manage cpu usage. Rubame allows me to run the network loop when it suits me, allowing the game logic/engine to be the primary tool.
Rubame makes use of WebSocket Ruby to handle the websocket protocol, and the standard ruby sockets libraries for the actual network connections.
I am by no means a Ruby expert, so I expect there to be many ways I could improve this. I also expect there are ways I have done this that will make someone proficient in Ruby cringe. If you have such comments, please email me. I also expect to improve the code as my project progresses, and my needs grow. However, I will keep improvements to only be those that are of general use, and not specific to my project.
require 'rubame'
server = Rubame::Server.new("0.0.0.0", 25252)
while true
server.run do |client|
client.onopen do
puts "Server reports: client open"
end
client.onmessage do |mess|
puts "Server reports: message received: #{mess}"
client.send "You sent me #{mess}"
end
client.onclose do
puts "Server reports: client closed"
end
end
end
A useful Chrome plugin to help test connections is this Simple Websocket Client.
Each time the loop is run, it checks with ruby IO to see if any sockets have any new data. It then reads up to (currently) 2000 bytes from that socket. If there is more data than that waiting, it will have to wait until the next time the loop is called. Once enough data has been received for a complete frame, that frame will be read and returned as a message, via client.onmessage as seen in the example.
- Binary data - Currently, Rubame only deals with text network traffic. This is not ideal, and so I hope to add the ability to send binary data, along with examples here showing how to communicate with a javascript client.
- Lazy send - Currently, Rubame sends a message as soon as the message is requested to be sent. Sometimes, a game may want to send some traffic whenever it is convenient, and some traffic immediately. I may add the ability to lazy send data.
Please Note: The lazy send code currently is broken, so do not use it!
The RSpec is quite incomplete for now. I expect to add more tests later.
(The MIT License)
Copyright © 2013 Mark Saward
Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the ‘Software’), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions:
The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software.
THE SOFTWARE IS PROVIDED ‘AS IS’, WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.