gorsse 0.0.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -0,0 +1,7 @@
1
+ ---
2
+ SHA1:
3
+ metadata.gz: bffb89b9e63f2abe4eea6f3b91f65c4f8b08d24d
4
+ data.tar.gz: fee3dec259477b40b7cf65dfe5b1b53ad6bbe364
5
+ SHA512:
6
+ metadata.gz: 4c1aa5c596190dc4c21ccfd30c5d99ca9a0431d47955de8191d6fbe0600c134c340c18e402040a9d59b726733585f2e7278eee12f08636c724fd664e9e04d9f7
7
+ data.tar.gz: 6229d81e78d340d42cd3a9b51a4818b55abe557036d5c7021123e9d02d53cb4c6c001d96fb04a26ee698708d9bf3fae75b102a41061e58ca658a82ed9f4d1cc9
data/Gemfile ADDED
@@ -0,0 +1,3 @@
1
+ source 'https://rubygems.org'
2
+
3
+ gemspec
data/LICENCE ADDED
@@ -0,0 +1,165 @@
1
+ GNU LESSER GENERAL PUBLIC LICENSE
2
+ Version 3, 29 June 2007
3
+
4
+ Copyright (C) 2007 Free Software Foundation, Inc. <http://fsf.org/>
5
+ Everyone is permitted to copy and distribute verbatim copies
6
+ of this license document, but changing it is not allowed.
7
+
8
+
9
+ This version of the GNU Lesser General Public License incorporates
10
+ the terms and conditions of version 3 of the GNU General Public
11
+ License, supplemented by the additional permissions listed below.
12
+
13
+ 0. Additional Definitions.
14
+
15
+ As used herein, "this License" refers to version 3 of the GNU Lesser
16
+ General Public License, and the "GNU GPL" refers to version 3 of the GNU
17
+ General Public License.
18
+
19
+ "The Library" refers to a covered work governed by this License,
20
+ other than an Application or a Combined Work as defined below.
21
+
22
+ An "Application" is any work that makes use of an interface provided
23
+ by the Library, but which is not otherwise based on the Library.
24
+ Defining a subclass of a class defined by the Library is deemed a mode
25
+ of using an interface provided by the Library.
26
+
27
+ A "Combined Work" is a work produced by combining or linking an
28
+ Application with the Library. The particular version of the Library
29
+ with which the Combined Work was made is also called the "Linked
30
+ Version".
31
+
32
+ The "Minimal Corresponding Source" for a Combined Work means the
33
+ Corresponding Source for the Combined Work, excluding any source code
34
+ for portions of the Combined Work that, considered in isolation, are
35
+ based on the Application, and not on the Linked Version.
36
+
37
+ The "Corresponding Application Code" for a Combined Work means the
38
+ object code and/or source code for the Application, including any data
39
+ and utility programs needed for reproducing the Combined Work from the
40
+ Application, but excluding the System Libraries of the Combined Work.
41
+
42
+ 1. Exception to Section 3 of the GNU GPL.
43
+
44
+ You may convey a covered work under sections 3 and 4 of this License
45
+ without being bound by section 3 of the GNU GPL.
46
+
47
+ 2. Conveying Modified Versions.
48
+
49
+ If you modify a copy of the Library, and, in your modifications, a
50
+ facility refers to a function or data to be supplied by an Application
51
+ that uses the facility (other than as an argument passed when the
52
+ facility is invoked), then you may convey a copy of the modified
53
+ version:
54
+
55
+ a) under this License, provided that you make a good faith effort to
56
+ ensure that, in the event an Application does not supply the
57
+ function or data, the facility still operates, and performs
58
+ whatever part of its purpose remains meaningful, or
59
+
60
+ b) under the GNU GPL, with none of the additional permissions of
61
+ this License applicable to that copy.
62
+
63
+ 3. Object Code Incorporating Material from Library Header Files.
64
+
65
+ The object code form of an Application may incorporate material from
66
+ a header file that is part of the Library. You may convey such object
67
+ code under terms of your choice, provided that, if the incorporated
68
+ material is not limited to numerical parameters, data structure
69
+ layouts and accessors, or small macros, inline functions and templates
70
+ (ten or fewer lines in length), you do both of the following:
71
+
72
+ a) Give prominent notice with each copy of the object code that the
73
+ Library is used in it and that the Library and its use are
74
+ covered by this License.
75
+
76
+ b) Accompany the object code with a copy of the GNU GPL and this license
77
+ document.
78
+
79
+ 4. Combined Works.
80
+
81
+ You may convey a Combined Work under terms of your choice that,
82
+ taken together, effectively do not restrict modification of the
83
+ portions of the Library contained in the Combined Work and reverse
84
+ engineering for debugging such modifications, if you also do each of
85
+ the following:
86
+
87
+ a) Give prominent notice with each copy of the Combined Work that
88
+ the Library is used in it and that the Library and its use are
89
+ covered by this License.
90
+
91
+ b) Accompany the Combined Work with a copy of the GNU GPL and this license
92
+ document.
93
+
94
+ c) For a Combined Work that displays copyright notices during
95
+ execution, include the copyright notice for the Library among
96
+ these notices, as well as a reference directing the user to the
97
+ copies of the GNU GPL and this license document.
98
+
99
+ d) Do one of the following:
100
+
101
+ 0) Convey the Minimal Corresponding Source under the terms of this
102
+ License, and the Corresponding Application Code in a form
103
+ suitable for, and under terms that permit, the user to
104
+ recombine or relink the Application with a modified version of
105
+ the Linked Version to produce a modified Combined Work, in the
106
+ manner specified by section 6 of the GNU GPL for conveying
107
+ Corresponding Source.
108
+
109
+ 1) Use a suitable shared library mechanism for linking with the
110
+ Library. A suitable mechanism is one that (a) uses at run time
111
+ a copy of the Library already present on the user's computer
112
+ system, and (b) will operate properly with a modified version
113
+ of the Library that is interface-compatible with the Linked
114
+ Version.
115
+
116
+ e) Provide Installation Information, but only if you would otherwise
117
+ be required to provide such information under section 6 of the
118
+ GNU GPL, and only to the extent that such information is
119
+ necessary to install and execute a modified version of the
120
+ Combined Work produced by recombining or relinking the
121
+ Application with a modified version of the Linked Version. (If
122
+ you use option 4d0, the Installation Information must accompany
123
+ the Minimal Corresponding Source and Corresponding Application
124
+ Code. If you use option 4d1, you must provide the Installation
125
+ Information in the manner specified by section 6 of the GNU GPL
126
+ for conveying Corresponding Source.)
127
+
128
+ 5. Combined Libraries.
129
+
130
+ You may place library facilities that are a work based on the
131
+ Library side by side in a single library together with other library
132
+ facilities that are not Applications and are not covered by this
133
+ License, and convey such a combined library under terms of your
134
+ choice, if you do both of the following:
135
+
136
+ a) Accompany the combined library with a copy of the same work based
137
+ on the Library, uncombined with any other library facilities,
138
+ conveyed under the terms of this License.
139
+
140
+ b) Give prominent notice with the combined library that part of it
141
+ is a work based on the Library, and explaining where to find the
142
+ accompanying uncombined form of the same work.
143
+
144
+ 6. Revised Versions of the GNU Lesser General Public License.
145
+
146
+ The Free Software Foundation may publish revised and/or new versions
147
+ of the GNU Lesser General Public License from time to time. Such new
148
+ versions will be similar in spirit to the present version, but may
149
+ differ in detail to address new problems or concerns.
150
+
151
+ Each version is given a distinguishing version number. If the
152
+ Library as you received it specifies that a certain numbered version
153
+ of the GNU Lesser General Public License "or any later version"
154
+ applies to it, you have the option of following the terms and
155
+ conditions either of that published version or of any later version
156
+ published by the Free Software Foundation. If the Library as you
157
+ received it does not specify a version number of the GNU Lesser
158
+ General Public License, you may choose any version of the GNU Lesser
159
+ General Public License ever published by the Free Software Foundation.
160
+
161
+ If the Library as you received it specifies that a proxy can decide
162
+ whether future versions of the GNU Lesser General Public License shall
163
+ apply, that proxy's public statement of acceptance of any version is
164
+ permanent authorization for you to choose that version for the
165
+ Library.
@@ -0,0 +1,219 @@
1
+ # Gorsse (Go Ruby SSE)
2
+
3
+ <a href="http://twitter.com/nicoolas25"><img src="http://www.pairprogramwith.me/assets/badge.svg" style="height:40px" title="We can pair on this!" /></a>
4
+
5
+ This is an proof of concept for an SSE system that will support many
6
+ users without compromising a Ruby server that isn't that good to handle
7
+ a lot of alive connections.
8
+
9
+ The typical flow is:
10
+
11
+ 1. The server povides an application
12
+ 2. The client needs realtime informations from the server
13
+ 3. The client establish a SSE connection to the application
14
+ 4. The server send the informations via the SSE connection
15
+ 5. The client is happy to have real-time updates
16
+
17
+ If your application get too many client, your server will have to handle
18
+ as much as connections. Since Ruby isn't very well suited for parallelism,
19
+ the existing webservers forks and/or threads the application to handle
20
+ multiple requests at a time. To serve two clients exactly at the same time,
21
+ you have to run two instances of your application. If those two client need
22
+ to keep the connection alive then your application is down to the rest of
23
+ the world...
24
+
25
+ This may be not the case on all the Ruby implementation and for all
26
+ webservers, see the alternative section for more details about it.
27
+
28
+ ## Usage examples
29
+
30
+ This project can be useful when you're using:
31
+
32
+ * chat systems,
33
+ * asynchronous job notifications,
34
+ * live monitoring,
35
+ * et cetera
36
+
37
+
38
+ ## Architecture
39
+
40
+ The whole idea is to keep your Ruby application as it is, adding one
41
+ or two other elements to your existing architecture.
42
+
43
+ ### The connection handler
44
+
45
+ When you run the connection handler, it will accept HTTP requests and
46
+ stream the response to the client. It will maintain the connection to the
47
+ client open.
48
+
49
+ The connection handler will also listen events from your application. When
50
+ a such event is received, it is forwarded to the relevent clients.
51
+
52
+ The connection handler lives in a separate project. It's written in a
53
+ language that have a more advanced concurrency and parallelism model than
54
+ Ruby.
55
+
56
+ See the documentation of the connection handler to configure it.
57
+
58
+ ### The callback receiver (optionnal)
59
+
60
+ This element receive callback from the connection handler. Those
61
+ callbacks can trigger business Ruby code depending on your needs.
62
+
63
+ If you don't need the callback receiver, you should disable them in
64
+ the connection handler configuration. In this case, you dont have to
65
+ run this component at all.
66
+
67
+ ### Communication
68
+
69
+ ZeroMQ allows the communication between:
70
+
71
+ * your application, that send event to the connection handler,
72
+ * the connection handler, that receive those events via SSE and
73
+ * the callback receiver.
74
+
75
+ ## Integration
76
+
77
+ This is probably the part that you've been waiting...
78
+
79
+ ### Configuration
80
+
81
+ First of all, your application must require 'gorsse' and configure it
82
+ properly:
83
+
84
+ ~~~ruby
85
+ require 'gorsse'
86
+
87
+ Gorsse.configure do |config|
88
+ config.receiver = 'tcp://127.0.0.1:4567'
89
+ config.handler = 'tcp://127.0.0.1:4568'
90
+ end
91
+ ~~~
92
+
93
+ The addresses are directly passed to ZeroMQ, feel free to take advantage
94
+ of it.
95
+
96
+ The receiver line isn't required unless you use the callback receiver.
97
+
98
+ ### Protocols & scopes
99
+
100
+ Gorsse is using two concepts to create SSE channels where it is possible
101
+ to publish informations.
102
+
103
+ *Protocols* match a specific communication pattern from your server to
104
+ your clients. This is an example of a two empty protocol:
105
+
106
+ ~~~ruby
107
+ class PostFeed < Gorsse::Protocol ; end
108
+ class ChatFeed < Gorsse::Protocol ; end
109
+ ~~~
110
+
111
+ *Scopes* are like an instance of the protocol, it's narrowing it. For
112
+ instance, if your application is a blog provider SaaS then you'll have
113
+ one `PostFeed` scope per blog. Or, if your application is a chat server
114
+ then you'll have one `ChatFeed` scope per channel.
115
+
116
+ ~~~ruby
117
+ endpoint = ChatFeed.new('room42')
118
+ ~~~
119
+
120
+ You can see the protocol and scope as an endpoint like `/ChatFeed/room42`.
121
+
122
+ ### Sending events
123
+
124
+ From the previous example, you can publish a message to all the connected
125
+ clients with the following code:
126
+
127
+ ~~~ruby
128
+ class Message < Struct.new(:author, :content)
129
+ # This is required to be send as an event by Gorsse.
130
+ # @return String
131
+ def to_sse
132
+ "#{author}|#{content}"
133
+ end
134
+ end
135
+
136
+ message = Message.new('Bob', 'Hello!')
137
+ endpoint.signal(message)
138
+ ~~~
139
+
140
+ The previous code will generate the following lines in the SSE stream:
141
+
142
+ ~~~
143
+ event: Message
144
+ data: Bob|Hello!
145
+  
146
+ ~~~
147
+
148
+ ### Private messages
149
+
150
+ You can achieve private messages with scopes.
151
+
152
+ There is also a notion of Client that allows you to do something like this:
153
+
154
+ ~~~ruby
155
+ client = Gorsse::Client.new('124df54b0')
156
+ message = Message.new('Alice', 'Goodbye...')
157
+ endpoint.signal(message, target: client)
158
+ ~~~
159
+
160
+ *This section should be completed.*
161
+
162
+ The missing piece here is that we need something to pass the client identity
163
+ from the server to the client then to the connection handler...
164
+
165
+ ## Installation
166
+
167
+ Add the classic line to your Gemfile:
168
+
169
+ ~~~
170
+ gem 'gorsse'
171
+ ~~~
172
+
173
+ Compile your own binaries for the connection handler (see the dedicated README
174
+ in the `gorsse_server` directory). You can also ask me for a binary version.
175
+
176
+ ## Alternative
177
+
178
+ There are many other solutions out there that are aiming the same goal.
179
+ You can try to fix the server with Goliath or you can rely on external
180
+ components with Faye.
181
+
182
+ There is a good article about [SSE in Ruby][ruby-sse] with Goliath and
183
+ EventMachine. The implementation now relies on EM channels that are
184
+ something native in Go. The backend of Gorsse could have been written as
185
+ described here.
186
+
187
+ I don't think webservers like Puma are a viable alternative even if it
188
+ is using threads to handle requests. I'll be glad to have your feedback
189
+ on it.
190
+
191
+ Of course all of this project is tied to the Ruby world. You can have
192
+ it all for free with other platforms like Meteor...
193
+
194
+ ## TODOs
195
+
196
+ * Documentation about proxying via nginx
197
+ * Tests both for the Go code and the Ruby code
198
+ * Benchmarking versus the alternatives
199
+ * Any other useful stuff that I didn't think of...
200
+
201
+ ## Contributions
202
+
203
+ They're welcome! Just know that this is a toy project for me yet.
204
+ Even if I'll be glad to receive contributions and advices about it
205
+ I may not be able to provide a excellent level of support.
206
+
207
+ ## Licence
208
+
209
+ This program is free software: you can redistribute it and/or modify
210
+ it under the terms of the GNU Lesser General Public License as
211
+ published by the Free Software Foundation, either version 3 of the
212
+ License, or (at your option) any later version.
213
+
214
+ This program is distributed in the hope that it will be useful,
215
+ but WITHOUT ANY WARRANTY; without even the implied warranty of
216
+ MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
217
+ GNU General Public License for more details.
218
+
219
+ [ruby-sse]: http://robots.thoughtbot.com/chat-example-app-using-server-sent-events
@@ -0,0 +1,41 @@
1
+ module Gorsse
2
+ {
3
+ :Client => 'client',
4
+ :Command => 'command',
5
+ :Config => 'config',
6
+ :Connection => 'connection',
7
+ :Event => 'event',
8
+ :Protocol => 'protocol',
9
+ :VERSION => 'version',
10
+ }.each { |mod, file| autoload mod, "gorsse/#{file}" }
11
+
12
+ def self.configure(&block)
13
+ block.call(config)
14
+ end
15
+
16
+ def self.config
17
+ @config ||= Config.new
18
+ end
19
+
20
+ def self.close_connections
21
+ @conn && @conn.close
22
+ @receiver_conn && @receiver_conn.close
23
+ Connection::ZCTX.terminate
24
+ end
25
+
26
+ def self.conn
27
+ @conn ||= Connection.new(config.handler, mode: :push, method: :connect)
28
+ end
29
+
30
+ def self.receiver_conn
31
+ @receiver_conn ||= Connection.new(config.receiver, mode: :pull, method: :bind)
32
+ end
33
+
34
+ def self.start_receiver_loop!
35
+ loop do
36
+ message = receiver_conn.receive
37
+ command = Command.new(message)
38
+ command.run!
39
+ end
40
+ end
41
+ end
@@ -0,0 +1,13 @@
1
+ module Gorsse
2
+ class Client
3
+ attr_reader :uid
4
+
5
+ def initialize(uid)
6
+ @uid = uid
7
+ end
8
+
9
+ def eql?(other)
10
+ other.is_a?(Client) && uid == other.uid
11
+ end
12
+ end
13
+ end
@@ -0,0 +1,16 @@
1
+ require 'json'
2
+
3
+ module Gorsse
4
+ class Command
5
+ def initialize(command)
6
+ @command = JSON.parse(command)
7
+ end
8
+
9
+ def run!
10
+ protocol_class = Object.const_get(@command['Protocol'])
11
+ protocol = protocol_class.new(@command['Scope'])
12
+ client = Client.new(@command['Client'])
13
+ protocol.__send__(@command['Callback'], client)
14
+ end
15
+ end
16
+ end
@@ -0,0 +1,5 @@
1
+ module Gorsse
2
+ class Config
3
+ attr_accessor :receiver, :handler
4
+ end
5
+ end
@@ -0,0 +1,40 @@
1
+ require 'ffi-rzmq'
2
+
3
+ module Gorsse
4
+ # This class hide the ZMQ library but it is closely tied to it.
5
+ class Connection
6
+ ZCTX = ZMQ::Context.new(1)
7
+
8
+ CONNECTION_MODE = {
9
+ rep: ZMQ::REP,
10
+ req: ZMQ::REQ,
11
+ pub: ZMQ::PUB,
12
+ sub: ZMQ::SUB,
13
+ pull: ZMQ::PULL,
14
+ push: ZMQ::PUSH,
15
+ }
16
+
17
+ # Open a connection to an URL. You can act as a server or as
18
+ # a client by tunning the 'server' parameter.
19
+ def initialize(url, mode: :push, method: :connect)
20
+ @zmq_socket = ZCTX.socket(CONNECTION_MODE[mode])
21
+ @zmq_socket.__send__(method, url)
22
+ puts '%s with %s to %s' % [method, mode, url]
23
+ end
24
+
25
+ def send(string, flags: 0)
26
+ puts 'Sending: %s' % string
27
+ @zmq_socket.send_string(string, flags)
28
+ end
29
+
30
+ def receive(flags: 0)
31
+ message = ''
32
+ @zmq_socket.recv_string(message, flags)
33
+ message
34
+ end
35
+
36
+ def close
37
+ @zmq_socket.close
38
+ end
39
+ end
40
+ end
@@ -0,0 +1,41 @@
1
+ require 'json'
2
+
3
+ module Gorsse
4
+ class Event
5
+ def initialize(protocol, target, entity)
6
+ @protocol = protocol
7
+ @target = target
8
+ @entity = entity
9
+ end
10
+
11
+ def send!
12
+ json = JSON.generate(msg)
13
+ Gorsse.conn.send(json)
14
+ end
15
+
16
+ private
17
+
18
+ def msg
19
+ hash = {
20
+ 'proto' => sse_class_for(@protocol),
21
+ 'scope' => @protocol.scope,
22
+ 'client' => @target.kind_of?(Client) ? @target.uid : 'all',
23
+ }
24
+
25
+ if @entity.respond_to?(:to_sse)
26
+ hash['title'] = sse_class_for(@entity)
27
+ hash['content'] = @entity.to_sse
28
+ else
29
+ hash['title'] = @entity.to_s
30
+ hash['content'] = ''
31
+ end
32
+
33
+ hash
34
+ end
35
+
36
+ def sse_class_for(instance)
37
+ klass = instance.class
38
+ klass.respond_to?(:sse_name) ? klass.sse_name : klass.name
39
+ end
40
+ end
41
+ end
@@ -0,0 +1,23 @@
1
+ module Gorsse
2
+ class Protocol
3
+ attr_reader :scope
4
+
5
+ def initialize(scope)
6
+ @scope = scope
7
+ end
8
+
9
+ # Send a message to the client through the Gorsse server.
10
+ def signal(entity, target: :all)
11
+ event = Event.new(self, target, entity)
12
+ event.send!
13
+ end
14
+
15
+ def after_connect(client_id)
16
+ # Do nothing. Override in subclasses.
17
+ end
18
+
19
+ def eql?(other)
20
+ self.class == other.class && self.scope == other.scope
21
+ end
22
+ end
23
+ end
@@ -0,0 +1,3 @@
1
+ module Gorsse
2
+ VERSION = "0.0.1"
3
+ end
@@ -0,0 +1,51 @@
1
+ require 'test_helper.rb'
2
+
3
+ describe Gorsse::Event do
4
+ let(:protocol_class) { stub_const('Protocol', Class.new(Gorsse::Protocol)) }
5
+ let(:event) { Gorsse::Event.new(protocol, target, entity) }
6
+
7
+ describe '#send!' do
8
+ before { allow(Gorsse.conn).to receive(:send).and_return(nil) }
9
+
10
+ subject { event.send! }
11
+
12
+ let(:scope) { 'scope' }
13
+ let(:protocol) { protocol_class.new(scope) }
14
+ let(:target) { :all }
15
+ let(:entity) { 'no_data' }
16
+
17
+ it 'sends json event data though the Gorsse.conn connection' do
18
+ expected_json = %Q({"proto":"Protocol","scope":"scope","client":"all","title":"no_data","content":""})
19
+ expect(Gorsse.conn).to receive(:send).with(expected_json)
20
+ subject
21
+ end
22
+
23
+ context 'when the target is a client' do
24
+ let(:target) { Gorsse::Client.new('1234') }
25
+
26
+ it 'sends the client uid in the "client" fields' do
27
+ expected_json = %Q({"proto":"Protocol","scope":"scope","client":"1234","title":"no_data","content":""})
28
+ expect(Gorsse.conn).to receive(:send).with(expected_json)
29
+ subject
30
+ end
31
+ end
32
+
33
+ context 'when the entity respond to the "to_sse" method' do
34
+ let(:entity) { double( class: double( name: 'Entity' ) ) }
35
+
36
+ it 'sends the entity class name as the "title" field' do
37
+ expected_json = %Q({"proto":"Protocol","scope":"scope","client":"all","title":"Entity","content":""})
38
+ allow(entity).to receive(:to_sse).and_return('')
39
+ expect(Gorsse.conn).to receive(:send).with(expected_json)
40
+ subject
41
+ end
42
+
43
+ it 'sends the result of the "to_sse" call as the "content" field' do
44
+ expected_json = %Q({"proto":"Protocol","scope":"scope","client":"all","title":"Entity","content":"data"})
45
+ allow(entity).to receive(:to_sse).and_return('data')
46
+ expect(Gorsse.conn).to receive(:send).with(expected_json)
47
+ subject
48
+ end
49
+ end
50
+ end
51
+ end
@@ -0,0 +1,14 @@
1
+ require 'gorsse'
2
+
3
+ Gorsse.configure do |config|
4
+ config.receiver = 'tcp://127.0.0.1:4567'
5
+ config.handler = 'tcp://127.0.0.1:4568'
6
+ end
7
+
8
+ class FakeConnection ; end
9
+
10
+ RSpec.configure do |config|
11
+ config.before(:example) do
12
+ allow(Gorsse).to receive(:conn).and_return(FakeConnection.new)
13
+ end
14
+ end
metadata ADDED
@@ -0,0 +1,115 @@
1
+ --- !ruby/object:Gem::Specification
2
+ name: gorsse
3
+ version: !ruby/object:Gem::Version
4
+ version: 0.0.1
5
+ platform: ruby
6
+ authors:
7
+ - Nicolas ZERMATI
8
+ autorequire:
9
+ bindir: bin
10
+ cert_chain: []
11
+ date: 2014-07-21 00:00:00.000000000 Z
12
+ dependencies:
13
+ - !ruby/object:Gem::Dependency
14
+ name: ffi-rzmq
15
+ requirement: !ruby/object:Gem::Requirement
16
+ requirements:
17
+ - - ">="
18
+ - !ruby/object:Gem::Version
19
+ version: '0'
20
+ type: :runtime
21
+ prerelease: false
22
+ version_requirements: !ruby/object:Gem::Requirement
23
+ requirements:
24
+ - - ">="
25
+ - !ruby/object:Gem::Version
26
+ version: '0'
27
+ - !ruby/object:Gem::Dependency
28
+ name: bundler
29
+ requirement: !ruby/object:Gem::Requirement
30
+ requirements:
31
+ - - "~>"
32
+ - !ruby/object:Gem::Version
33
+ version: '1.6'
34
+ type: :development
35
+ prerelease: false
36
+ version_requirements: !ruby/object:Gem::Requirement
37
+ requirements:
38
+ - - "~>"
39
+ - !ruby/object:Gem::Version
40
+ version: '1.6'
41
+ - !ruby/object:Gem::Dependency
42
+ name: rake
43
+ requirement: !ruby/object:Gem::Requirement
44
+ requirements:
45
+ - - ">="
46
+ - !ruby/object:Gem::Version
47
+ version: '0'
48
+ type: :development
49
+ prerelease: false
50
+ version_requirements: !ruby/object:Gem::Requirement
51
+ requirements:
52
+ - - ">="
53
+ - !ruby/object:Gem::Version
54
+ version: '0'
55
+ - !ruby/object:Gem::Dependency
56
+ name: rspec
57
+ requirement: !ruby/object:Gem::Requirement
58
+ requirements:
59
+ - - ">="
60
+ - !ruby/object:Gem::Version
61
+ version: '0'
62
+ type: :development
63
+ prerelease: false
64
+ version_requirements: !ruby/object:Gem::Requirement
65
+ requirements:
66
+ - - ">="
67
+ - !ruby/object:Gem::Version
68
+ version: '0'
69
+ description:
70
+ email:
71
+ - nicoolas25@gmail.com
72
+ executables: []
73
+ extensions: []
74
+ extra_rdoc_files: []
75
+ files:
76
+ - Gemfile
77
+ - LICENCE
78
+ - README.md
79
+ - lib/gorsse.rb
80
+ - lib/gorsse/client.rb
81
+ - lib/gorsse/command.rb
82
+ - lib/gorsse/config.rb
83
+ - lib/gorsse/connection.rb
84
+ - lib/gorsse/event.rb
85
+ - lib/gorsse/protocol.rb
86
+ - lib/gorsse/version.rb
87
+ - spec/gorsse/event_spec.rb
88
+ - spec/test_helper.rb
89
+ homepage: ''
90
+ licenses:
91
+ - LGPL
92
+ metadata: {}
93
+ post_install_message:
94
+ rdoc_options: []
95
+ require_paths:
96
+ - lib
97
+ required_ruby_version: !ruby/object:Gem::Requirement
98
+ requirements:
99
+ - - ">="
100
+ - !ruby/object:Gem::Version
101
+ version: '0'
102
+ required_rubygems_version: !ruby/object:Gem::Requirement
103
+ requirements:
104
+ - - ">="
105
+ - !ruby/object:Gem::Version
106
+ version: '0'
107
+ requirements: []
108
+ rubyforge_project:
109
+ rubygems_version: 2.2.2
110
+ signing_key:
111
+ specification_version: 4
112
+ summary: Have nice and fast SSE capabilities with any Ruby webserver.
113
+ test_files:
114
+ - spec/gorsse/event_spec.rb
115
+ - spec/test_helper.rb