sneakers_packer 0.1.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.
checksums.yaml ADDED
@@ -0,0 +1,7 @@
1
+ ---
2
+ SHA1:
3
+ metadata.gz: 3eb01e938f800fb40861378799200f937c6742da
4
+ data.tar.gz: 82a8e039d1e3e3da2302e12e03e2ca183113c1b7
5
+ SHA512:
6
+ metadata.gz: 2039e09ee76c8823f97b0b3fef012c56d3c74d6870b3b3afeb3b8134de5c38c97e21b4793707abe3c6ed010ab19781e56594e6e73ea1835133cf6b7132ae73c9
7
+ data.tar.gz: 2953841eb9bfc66421b30290ae2012e1c171446dadd942154809a8150b5cfbe6e06d2337d12b438d349a3f0e283fac957657c469009fdbbc84e74afaefe9e8c4
data/.gitignore ADDED
@@ -0,0 +1,9 @@
1
+ /.bundle/
2
+ /.yardoc
3
+ /Gemfile.lock
4
+ /_yardoc/
5
+ /coverage/
6
+ /doc/
7
+ /pkg/
8
+ /spec/reports/
9
+ /tmp/
data/.travis.yml ADDED
@@ -0,0 +1,6 @@
1
+ language: ruby
2
+ rvm:
3
+ - 2.1.5
4
+ - 2.2.3
5
+ - 2.3.0
6
+ before_install: gem install bundler -v 1.11.2
@@ -0,0 +1,49 @@
1
+ # Contributor Code of Conduct
2
+
3
+ As contributors and maintainers of this project, and in the interest of
4
+ fostering an open and welcoming community, we pledge to respect all people who
5
+ contribute through reporting issues, posting feature requests, updating
6
+ documentation, submitting pull requests or patches, and other activities.
7
+
8
+ We are committed to making participation in this project a harassment-free
9
+ experience for everyone, regardless of level of experience, gender, gender
10
+ identity and expression, sexual orientation, disability, personal appearance,
11
+ body size, race, ethnicity, age, religion, or nationality.
12
+
13
+ Examples of unacceptable behavior by participants include:
14
+
15
+ * The use of sexualized language or imagery
16
+ * Personal attacks
17
+ * Trolling or insulting/derogatory comments
18
+ * Public or private harassment
19
+ * Publishing other's private information, such as physical or electronic
20
+ addresses, without explicit permission
21
+ * Other unethical or unprofessional conduct
22
+
23
+ Project maintainers have the right and responsibility to remove, edit, or
24
+ reject comments, commits, code, wiki edits, issues, and other contributions
25
+ that are not aligned to this Code of Conduct, or to ban temporarily or
26
+ permanently any contributor for other behaviors that they deem inappropriate,
27
+ threatening, offensive, or harmful.
28
+
29
+ By adopting this Code of Conduct, project maintainers commit themselves to
30
+ fairly and consistently applying these principles to every aspect of managing
31
+ this project. Project maintainers who do not follow or enforce the Code of
32
+ Conduct may be permanently removed from the project team.
33
+
34
+ This code of conduct applies both within project spaces and in public spaces
35
+ when an individual is representing the project or its community.
36
+
37
+ Instances of abusive, harassing, or otherwise unacceptable behavior may be
38
+ reported by contacting a project maintainer at vincent@boohee.com. All
39
+ complaints will be reviewed and investigated and will result in a response that
40
+ is deemed necessary and appropriate to the circumstances. Maintainers are
41
+ obligated to maintain confidentiality with regard to the reporter of an
42
+ incident.
43
+
44
+ This Code of Conduct is adapted from the [Contributor Covenant][homepage],
45
+ version 1.3.0, available at
46
+ [http://contributor-covenant.org/version/1/3/0/][version]
47
+
48
+ [homepage]: http://contributor-covenant.org
49
+ [version]: http://contributor-covenant.org/version/1/3/0/
data/Gemfile ADDED
@@ -0,0 +1,4 @@
1
+ source 'https://rubygems.org'
2
+
3
+ # Specify your gem's dependencies in sneakers_packer.gemspec
4
+ gemspec
data/README.md ADDED
@@ -0,0 +1,134 @@
1
+ # SneakersPacker
2
+
3
+ SneakersPacker is a gem for using sneakers to 3 message communication patterns job message, broadcast and RPC(remote procedure call).
4
+
5
+ ## Installation
6
+
7
+ Intall the `sneakers' gem first. see [sneakers](https://github.com/jondot/sneakers)
8
+
9
+ Then add this line to your application's Gemfile:
10
+
11
+ ```ruby
12
+ gem 'sneakers_packer'
13
+ ```
14
+
15
+ And then execute:
16
+
17
+ $ bundle
18
+
19
+ Or install it yourself as:
20
+
21
+ $ gem install sneakers_packer
22
+
23
+ ## Usage
24
+
25
+ ### Configuation
26
+
27
+ SneakersPacker uses most of sneakers configuration for simpleness.
28
+ There are `app_name` and `rpc_timeout` should be set for SneakersPacker.
29
+
30
+ Append below to `config/initializers/sneakers.rb`.
31
+
32
+ ```
33
+ SneakersPacker.configure do |conf|
34
+ conf.rpc_timeout = 3 # rpc client timeout. default is 5 seconds.
35
+ conf.app_name = "sneakers_test" # rpc client or server app's name. default is 'unknown'
36
+ end
37
+ ```
38
+
39
+ ### API usage examples
40
+
41
+ - Job Message
42
+
43
+ **Client**
44
+
45
+ `SneakersPacker.publish("demo", "hello world")`
46
+
47
+ **Server**
48
+
49
+ ```ruby
50
+ class DemoWorker
51
+ include SneakersPacker::CommonWorker
52
+ from_queue :demo
53
+
54
+ def call(data)
55
+ puts "data is #{data}"
56
+ # do something...
57
+ end
58
+ end
59
+ ```
60
+
61
+ - Broadcast
62
+
63
+ **Client**
64
+
65
+ It is same with Job Message
66
+
67
+ `SneakersPacker.publish("demo.suprise", "hello world")`
68
+
69
+ **Server**
70
+
71
+ It is almost same with Job Message except that one routing_key with multiple queues.
72
+
73
+ ```ruby
74
+ class OneWorker
75
+ include SneakersPacker::CommonWorker
76
+
77
+ from_queue :one_name, routing_key: "demo.suprise"
78
+
79
+ def call(data)
80
+ puts "one: #{data}"
81
+ # do something...
82
+ end
83
+ end
84
+ ```
85
+
86
+ ```
87
+ class OtherWorker
88
+ include SneakersPacker::CommonWorker
89
+
90
+ from_queue :other_name, routing_key: "demo.suprise"
91
+
92
+ def call(data)
93
+ puts "other: #{data}"
94
+ end
95
+ end
96
+ ```
97
+
98
+ - RPC
99
+
100
+ **Client**
101
+
102
+ remote call with default timeouit. default is 5 seconds.
103
+ `SneakersPacker.remote_call("rpc_server", 10)`
104
+
105
+ remote call with custom timeouit.
106
+ `SneakersPacker.remote_call("rpc_server", 12, timeout: 2)`
107
+
108
+ **Server**
109
+
110
+ ```ruby
111
+ class RpcServerWorker
112
+ include SneakersPacker::RpcWorker
113
+
114
+ from_queue :rpc_server
115
+
116
+ # return value of call will be result of remote procedure call
117
+ def call(data)
118
+ data.to_i ** 3
119
+ end
120
+ end
121
+ ```
122
+
123
+ **See the gem doc or source code for accurate detail**
124
+
125
+ ## Development
126
+
127
+ After checking out the repo, run `bin/setup` to install dependencies. Then, run `rake test` to run the tests. You can also run `bin/console` for an interactive prompt that will allow you to experiment.
128
+
129
+ To install this gem onto your local machine, run `bundle exec rake install`. To release a new version, update the version number in `version.rb`, and then run `bundle exec rake release`, which will create a git tag for the version, push git commits and tags, and push the `.gem` file to [rubygems.org](https://rubygems.org).
130
+
131
+ ## Contributing
132
+
133
+ Bug reports and pull requests are welcome on GitHub at https://github.com/xiewenwei/sneakers_packer. This project is intended to be a safe, welcoming space for collaboration, and contributors are expected to adhere to the [Contributor Covenant](http://contributor-covenant.org) code of conduct.
134
+
data/Rakefile ADDED
@@ -0,0 +1,10 @@
1
+ require "bundler/gem_tasks"
2
+ require "rake/testtask"
3
+
4
+ Rake::TestTask.new(:test) do |t|
5
+ t.libs << "test"
6
+ t.libs << "lib"
7
+ t.test_files = FileList['test/**/*_test.rb']
8
+ end
9
+
10
+ task :default => :spec
data/bin/console ADDED
@@ -0,0 +1,14 @@
1
+ #!/usr/bin/env ruby
2
+
3
+ require "bundler/setup"
4
+ require "sneakers_packer"
5
+
6
+ # You can add fixtures and/or initialization code here to make experimenting
7
+ # with your gem easier. You can also use a different console, if you like.
8
+
9
+ # (If you use this, don't forget to add pry to your Gemfile!)
10
+ # require "pry"
11
+ # Pry.start
12
+
13
+ require "irb"
14
+ IRB.start
data/bin/setup ADDED
@@ -0,0 +1,8 @@
1
+ #!/usr/bin/env bash
2
+ set -euo pipefail
3
+ IFS=$'\n\t'
4
+ set -vx
5
+
6
+ bundle install
7
+
8
+ # Do any other automated setup that you need to do here
@@ -0,0 +1,21 @@
1
+ module SneakersPacker
2
+ module CommonWorker
3
+ def self.included(klass)
4
+ klass.class_eval do
5
+ include ::Sneakers::Worker
6
+ end
7
+ end
8
+
9
+ def packer
10
+ SneakersPacker.message_packer
11
+ end
12
+
13
+ def work(message)
14
+ #puts "get #{message}"
15
+ request_data, from = packer.unpack_request message
16
+ #puts "call from #{from}"
17
+ call request_data
18
+ ack!
19
+ end
20
+ end
21
+ end
@@ -0,0 +1,21 @@
1
+ module SneakersPacker
2
+ class Configuration
3
+ attr_accessor :app_name, :rpc_timeout
4
+
5
+ # default timeout for remote call. unit is seconds
6
+ DEFAULT_RPC_TIMEOUT = 5
7
+
8
+ def initialize
9
+ @app_name = 'unknown'
10
+ @rpc_timeout = DEFAULT_RPC_TIMEOUT
11
+ end
12
+ end
13
+
14
+ def self.conf
15
+ @conf ||= Configuration.new
16
+ end
17
+
18
+ def self.configure(&block)
19
+ block.call self.conf
20
+ end
21
+ end
@@ -0,0 +1,72 @@
1
+ require 'socket'
2
+ require 'multi_json'
3
+
4
+ module SneakersPacker
5
+ class MessagePacker
6
+ attr_reader :app_name, :host_name
7
+
8
+ def initialize(app_name = nil)
9
+ @app_name = app_name || 'unknown'
10
+ @host_name = Socket.gethostname
11
+ end
12
+
13
+ # Pack request data with standart json format
14
+ # It should include from_info and data
15
+ # @param data payload for request
16
+ # @return [string] json string body
17
+ # @example
18
+ # param data = 12
19
+ # return {"from" : "one boohee-tiger 6354", "data" : 12 }
20
+ # @example
21
+ # param data = ["ok", name: "vincent"]
22
+ # return "{\"from\" : \"one boohee-tiger 6354\", \"data\" : [\"1\", {\"name\" : \"vincent\"}]}"
23
+
24
+ def pack_request(data)
25
+ MultiJson.dump "data" => data, "from" => from_info
26
+ end
27
+
28
+ # Unpack request data which is standart json format
29
+ # It should include status, message(optional) and data
30
+ # @param body response raw data
31
+ # @return hash
32
+ # @example
33
+ # param message = "{\"from\":\"boohee\", \"data\":12}"
34
+ # return array [12, "boohee"]
35
+
36
+ def unpack_request(message)
37
+ hash = unpack_message(message)
38
+ [hash["data"], hash["from"]]
39
+ end
40
+
41
+ def pack_response(data, status)
42
+ MultiJson.dump "data" => data, "from" => from_info, "status" => status
43
+ end
44
+
45
+ # Unpack response data which is standart json format
46
+ # It should include status, message(optional) and data
47
+ # @param body response raw data
48
+ # @return hash
49
+ # @example
50
+ # param message = "{\"status\":200, \"data\":2}"
51
+ # return array [2, 200]
52
+
53
+ def unpack_response(message)
54
+ hash = unpack_message(message)
55
+ [hash["data"], hash["from"], hash["status"]]
56
+ end
57
+
58
+ private
59
+
60
+ def from_info
61
+ "#{@app_name} #{@host_name} #{Process.pid}"
62
+ end
63
+
64
+ def pack_data(data)
65
+ MultiJson.dump data
66
+ end
67
+
68
+ def unpack_message(message)
69
+ MultiJson.load message
70
+ end
71
+ end
72
+ end
@@ -0,0 +1,100 @@
1
+ module SneakersPacker
2
+ class RpcClient
3
+
4
+ attr_reader :reply_queue
5
+ attr_accessor :response, :call_id
6
+ attr_reader :lock, :condition
7
+
8
+ def initialize(publisher)
9
+ @publisher = publisher
10
+ channel, exchange = fetch_channel_and_exchange
11
+ @consumer = build_reply_queue(channel, exchange)
12
+ end
13
+
14
+ # call remote service via rabbitmq rpc
15
+ # @param name route_key for service
16
+ # @param message
17
+ # @param options{timeout} [int] timeout. seconds. optional
18
+ # @return result of service
19
+ # @raise RemoteCallTimeoutError if timeout
20
+ def call(name, message, options = {})
21
+ self.call_id = SecureRandom.uuid
22
+ self.response = nil
23
+
24
+ ensure_reply_queue!
25
+
26
+ @exchange.publish(message.to_s,
27
+ routing_key: name.to_s,
28
+ correlation_id: call_id,
29
+ reply_to: @reply_queue.name)
30
+
31
+ timeout = (options[:timeout] || SneakersPacker.conf.rpc_timeout).to_i
32
+
33
+ lock.synchronize { condition.wait(lock, timeout) }
34
+
35
+ if response
36
+ response
37
+ else
38
+ raise RemoteCallTimeoutError.new("远程调用超时")
39
+ end
40
+ end
41
+
42
+ private
43
+
44
+ def ensure_reply_queue!
45
+ reconnected = false
46
+ channel = nil
47
+ exchange = nil
48
+
49
+ @publisher.instance_eval do
50
+ # ensure_connection connection first
51
+ @mutex.synchronize do
52
+ unless connected?
53
+ ensure_connection!
54
+ reconnected = true
55
+ channel = @channel
56
+ exchange = @exchange
57
+ end
58
+ end
59
+ end
60
+
61
+ # rebuid reply_queue when reconnecting occur
62
+ if reconnected
63
+ @consumer = build_reply_queue(channel, exchange)
64
+ end
65
+ end
66
+
67
+ def build_reply_queue(channel, exchange)
68
+ @channel, @exchange = channel, exchange
69
+
70
+ @reply_queue = channel.queue("", exclusive: true)
71
+ @reply_queue.bind(exchange, routing_key: @reply_queue.name)
72
+
73
+ @lock = Mutex.new
74
+ @condition = ConditionVariable.new
75
+ that = self
76
+
77
+ @reply_queue.subscribe(manual_ack: false) do |delivery_info, properties, payload|
78
+ if properties[:correlation_id] == that.call_id
79
+ that.response = payload
80
+ that.lock.synchronize { that.condition.signal }
81
+ end
82
+ end
83
+ end
84
+
85
+ # hack seankers publisher to get channel and exchange
86
+ def fetch_channel_and_exchange
87
+ ret = nil
88
+
89
+ @publisher.instance_eval do
90
+ # ensure_connection connection first
91
+ @mutex.synchronize do
92
+ ensure_connection! unless connected?
93
+ end
94
+ ret = [@channel, @exchange]
95
+ end
96
+
97
+ ret
98
+ end
99
+ end
100
+ end
@@ -0,0 +1,32 @@
1
+ module SneakersPacker
2
+ module RpcWorker
3
+ def self.included(klass)
4
+ klass.class_eval do
5
+ include ::Sneakers::Worker
6
+ end
7
+ end
8
+
9
+ def packer
10
+ SneakersPacker.message_packer
11
+ end
12
+
13
+ def work_with_params(message, delivery_info, metadata)
14
+ #puts "get #{message}"
15
+ request_data, from = packer.unpack_request message
16
+ #puts "call from #{from}"
17
+
18
+ #begin
19
+ response_data = call request_data
20
+ result = packer.pack_response response_data, 200
21
+ # rescue Exception => ex
22
+ # # caught exeption
23
+ # result = packer.pack_response nil, 500
24
+ # ensure
25
+ publish(result, to_queue: metadata.reply_to, correlation_id: metadata.correlation_id)
26
+
27
+ ack!
28
+ #end
29
+ end
30
+
31
+ end
32
+ end
@@ -0,0 +1,3 @@
1
+ module SneakersPacker
2
+ VERSION = "0.1.1"
3
+ end
@@ -0,0 +1,44 @@
1
+ require "sneakers"
2
+ require "sneakers_packer/version"
3
+ require "sneakers_packer/configuration"
4
+ require "sneakers_packer/message_packer"
5
+ require "sneakers_packer/common_worker"
6
+ require "sneakers_packer/rpc_worker"
7
+ require "sneakers_packer/rpc_client"
8
+
9
+ module SneakersPacker
10
+ class RemoteCallTimeoutError < Exception; end
11
+
12
+ # sender message to sneaker exchange
13
+ # @param name route_key for message
14
+ # @param data
15
+ def self.publish(name, data)
16
+ message = message_packer.pack_request(data)
17
+
18
+ publisher.publish message, to_queue: name
19
+ end
20
+
21
+ # call remote service via rabbitmq rpc
22
+ # @param name route_key for service
23
+ # @param data
24
+ # @param options{timeout} [int] timeout. seconds. optional
25
+ # @return result of service
26
+ # @raise RemoteCallTimeoutError if timeout
27
+ #
28
+ def self.remote_call(name, data, options = {})
29
+ @client ||= RpcClient.new(publisher)
30
+ message = message_packer.pack_request(data)
31
+ response = @client.call name, message, options
32
+ response_data, from, status = message_packer.unpack_response(response)
33
+ response_data
34
+ end
35
+
36
+ def self.publisher
37
+ @publisher ||= ::Sneakers::Publisher.new
38
+ end
39
+
40
+ # message_packer is a singleton object
41
+ def self.message_packer
42
+ @message_packer ||= MessagePacker.new(self.conf.app_name)
43
+ end
44
+ end
@@ -0,0 +1,28 @@
1
+ # coding: utf-8
2
+ lib = File.expand_path('../lib', __FILE__)
3
+ $LOAD_PATH.unshift(lib) unless $LOAD_PATH.include?(lib)
4
+ require 'sneakers_packer/version'
5
+
6
+ Gem::Specification.new do |spec|
7
+ spec.name = "sneakers_packer"
8
+ spec.version = SneakersPacker::VERSION
9
+ spec.authors = ["vincent"]
10
+ spec.email = ["vincent@boohee.com"]
11
+
12
+ spec.summary = %q{SneakersPacker is a gem for using sneakers to 3 message communication patterns job message, broadcast and RPC(remote procedure call).}
13
+ spec.description = %q{SneakersPacker is a gem for using sneakers to 3 message communication patterns job message, broadcast and RPC(remote procedure call).}
14
+ spec.homepage = "https://github.com/xiewenwei/sneakers_packer"
15
+
16
+ spec.files = `git ls-files -z`.split("\x0").reject { |f| f.match(%r{^(test|spec|features)/}) }
17
+ spec.bindir = "exe"
18
+ spec.executables = spec.files.grep(%r{^exe/}) { |f| File.basename(f) }
19
+ spec.require_paths = ["lib"]
20
+
21
+ spec.add_dependency "sneakers", "~> 2.3.5"
22
+ spec.add_dependency "multi_json"
23
+
24
+ spec.add_development_dependency "bundler", "~> 1.11"
25
+ spec.add_development_dependency "rake", "~> 10.0"
26
+ spec.add_development_dependency "minitest", "~> 5.0"
27
+ spec.add_development_dependency "mocha"
28
+ end
metadata ADDED
@@ -0,0 +1,145 @@
1
+ --- !ruby/object:Gem::Specification
2
+ name: sneakers_packer
3
+ version: !ruby/object:Gem::Version
4
+ version: 0.1.1
5
+ platform: ruby
6
+ authors:
7
+ - vincent
8
+ autorequire:
9
+ bindir: exe
10
+ cert_chain: []
11
+ date: 2016-01-18 00:00:00.000000000 Z
12
+ dependencies:
13
+ - !ruby/object:Gem::Dependency
14
+ name: sneakers
15
+ requirement: !ruby/object:Gem::Requirement
16
+ requirements:
17
+ - - "~>"
18
+ - !ruby/object:Gem::Version
19
+ version: 2.3.5
20
+ type: :runtime
21
+ prerelease: false
22
+ version_requirements: !ruby/object:Gem::Requirement
23
+ requirements:
24
+ - - "~>"
25
+ - !ruby/object:Gem::Version
26
+ version: 2.3.5
27
+ - !ruby/object:Gem::Dependency
28
+ name: multi_json
29
+ requirement: !ruby/object:Gem::Requirement
30
+ requirements:
31
+ - - ">="
32
+ - !ruby/object:Gem::Version
33
+ version: '0'
34
+ type: :runtime
35
+ prerelease: false
36
+ version_requirements: !ruby/object:Gem::Requirement
37
+ requirements:
38
+ - - ">="
39
+ - !ruby/object:Gem::Version
40
+ version: '0'
41
+ - !ruby/object:Gem::Dependency
42
+ name: bundler
43
+ requirement: !ruby/object:Gem::Requirement
44
+ requirements:
45
+ - - "~>"
46
+ - !ruby/object:Gem::Version
47
+ version: '1.11'
48
+ type: :development
49
+ prerelease: false
50
+ version_requirements: !ruby/object:Gem::Requirement
51
+ requirements:
52
+ - - "~>"
53
+ - !ruby/object:Gem::Version
54
+ version: '1.11'
55
+ - !ruby/object:Gem::Dependency
56
+ name: rake
57
+ requirement: !ruby/object:Gem::Requirement
58
+ requirements:
59
+ - - "~>"
60
+ - !ruby/object:Gem::Version
61
+ version: '10.0'
62
+ type: :development
63
+ prerelease: false
64
+ version_requirements: !ruby/object:Gem::Requirement
65
+ requirements:
66
+ - - "~>"
67
+ - !ruby/object:Gem::Version
68
+ version: '10.0'
69
+ - !ruby/object:Gem::Dependency
70
+ name: minitest
71
+ requirement: !ruby/object:Gem::Requirement
72
+ requirements:
73
+ - - "~>"
74
+ - !ruby/object:Gem::Version
75
+ version: '5.0'
76
+ type: :development
77
+ prerelease: false
78
+ version_requirements: !ruby/object:Gem::Requirement
79
+ requirements:
80
+ - - "~>"
81
+ - !ruby/object:Gem::Version
82
+ version: '5.0'
83
+ - !ruby/object:Gem::Dependency
84
+ name: mocha
85
+ requirement: !ruby/object:Gem::Requirement
86
+ requirements:
87
+ - - ">="
88
+ - !ruby/object:Gem::Version
89
+ version: '0'
90
+ type: :development
91
+ prerelease: false
92
+ version_requirements: !ruby/object:Gem::Requirement
93
+ requirements:
94
+ - - ">="
95
+ - !ruby/object:Gem::Version
96
+ version: '0'
97
+ description: SneakersPacker is a gem for using sneakers to 3 message communication
98
+ patterns job message, broadcast and RPC(remote procedure call).
99
+ email:
100
+ - vincent@boohee.com
101
+ executables: []
102
+ extensions: []
103
+ extra_rdoc_files: []
104
+ files:
105
+ - ".gitignore"
106
+ - ".travis.yml"
107
+ - CODE_OF_CONDUCT.md
108
+ - Gemfile
109
+ - README.md
110
+ - Rakefile
111
+ - bin/console
112
+ - bin/setup
113
+ - lib/sneakers_packer.rb
114
+ - lib/sneakers_packer/common_worker.rb
115
+ - lib/sneakers_packer/configuration.rb
116
+ - lib/sneakers_packer/message_packer.rb
117
+ - lib/sneakers_packer/rpc_client.rb
118
+ - lib/sneakers_packer/rpc_worker.rb
119
+ - lib/sneakers_packer/version.rb
120
+ - sneakers_packer.gemspec
121
+ homepage: https://github.com/xiewenwei/sneakers_packer
122
+ licenses: []
123
+ metadata: {}
124
+ post_install_message:
125
+ rdoc_options: []
126
+ require_paths:
127
+ - lib
128
+ required_ruby_version: !ruby/object:Gem::Requirement
129
+ requirements:
130
+ - - ">="
131
+ - !ruby/object:Gem::Version
132
+ version: '0'
133
+ required_rubygems_version: !ruby/object:Gem::Requirement
134
+ requirements:
135
+ - - ">="
136
+ - !ruby/object:Gem::Version
137
+ version: '0'
138
+ requirements: []
139
+ rubyforge_project:
140
+ rubygems_version: 2.5.1
141
+ signing_key:
142
+ specification_version: 4
143
+ summary: SneakersPacker is a gem for using sneakers to 3 message communication patterns
144
+ job message, broadcast and RPC(remote procedure call).
145
+ test_files: []