mobileminer-adapter 0.0.1

Sign up to get free protection for your applications and to get access to all the features.
checksums.yaml ADDED
@@ -0,0 +1,7 @@
1
+ ---
2
+ SHA1:
3
+ metadata.gz: dfbb72406c793c0d4972624f21bf4fca85be60d2
4
+ data.tar.gz: bb0ea9e7d534e5b6c41faa5bf75ca0d541253efd
5
+ SHA512:
6
+ metadata.gz: 65771f065904050cc14c7676812f34e8c958932b4f2440452da91e95c2ad03a6bf1e47c34e3db97d404fcda0cf1950be24b8aeb564c5d5325febe502fb855aa7
7
+ data.tar.gz: cd866cd3e6551cd81ed56d7dde785557ce017776d6197419deaf4a84ab5560136f82ecea3973f39356715588a707102f788edbfb9385ca89690110149f09fcab
data/.gitignore ADDED
@@ -0,0 +1,19 @@
1
+ *.gem
2
+ *.rbc
3
+ .bundle
4
+ .config
5
+ .yardoc
6
+ Gemfile.lock
7
+ InstalledFiles
8
+ _yardoc
9
+ coverage
10
+ doc/
11
+ lib/bundler/man
12
+ pkg
13
+ rdoc
14
+ spec/reports
15
+ test/tmp
16
+ test/version_tmp
17
+ tmp
18
+ .idea
19
+ config.yaml
data/.ruby-version ADDED
@@ -0,0 +1 @@
1
+ 2.1.0
data/.travis.yml ADDED
@@ -0,0 +1,7 @@
1
+ rvm:
2
+ - 1.9.3
3
+ - 2.0.0
4
+ - 2.1.0
5
+
6
+ notifications:
7
+ disabled: true
data/Gemfile ADDED
@@ -0,0 +1,4 @@
1
+ source 'https://rubygems.org'
2
+
3
+ # Specify your gem's dependencies in mobileminer-adapter.gemspec
4
+ gemspec
data/LICENSE.txt ADDED
@@ -0,0 +1,22 @@
1
+ Copyright (c) 2014 Nick Veys
2
+
3
+ MIT License
4
+
5
+ Permission is hereby granted, free of charge, to any person obtaining
6
+ a copy of this software and associated documentation files (the
7
+ "Software"), to deal in the Software without restriction, including
8
+ without limitation the rights to use, copy, modify, merge, publish,
9
+ distribute, sublicense, and/or sell copies of the Software, and to
10
+ permit persons to whom the Software is furnished to do so, subject to
11
+ the following conditions:
12
+
13
+ The above copyright notice and this permission notice shall be
14
+ included in all copies or substantial portions of the Software.
15
+
16
+ THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
17
+ EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
18
+ MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
19
+ NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
20
+ LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
21
+ OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
22
+ WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
data/README.md ADDED
@@ -0,0 +1,29 @@
1
+ # Mobileminer::Adapter [![Build Status](https://travis-ci.org/code-lever/mobileminer-adapter-gem.png)](https://travis-ci.org/code-lever/mobileminer-adapter-gem) [![Dependency Status](https://gemnasium.com/code-lever/mobileminer-adapter-gem.png)](https://gemnasium.com/code-lever/mobileminer-adapter-gem) [![Code Climate](https://codeclimate.com/github/code-lever/mobileminer-adapter-gem.png)](https://codeclimate.com/github/code-lever/mobileminer-adapter-gem)
2
+
3
+ TODO: Write a gem description
4
+
5
+ ## Installation
6
+
7
+ Add this line to your application's Gemfile:
8
+
9
+ gem 'mobileminer-adapter'
10
+
11
+ And then execute:
12
+
13
+ $ bundle
14
+
15
+ Or install it yourself as:
16
+
17
+ $ gem install mobileminer-adapter
18
+
19
+ ## Usage
20
+
21
+ TODO: Write usage instructions here
22
+
23
+ ## Contributing
24
+
25
+ 1. Fork it ( http://github.com/<my-github-username>/mobileminer-adapter/fork )
26
+ 2. Create your feature branch (`git checkout -b my-new-feature`)
27
+ 3. Commit your changes (`git commit -am 'Add some feature'`)
28
+ 4. Push to the branch (`git push origin my-new-feature`)
29
+ 5. Create new Pull Request
data/Rakefile ADDED
@@ -0,0 +1,6 @@
1
+ #!/usr/bin/env rake
2
+ require 'bundler/gem_tasks'
3
+ require 'rspec/core/rake_task'
4
+
5
+ RSpec::Core::RakeTask.new('spec')
6
+ task :default => :spec
@@ -0,0 +1,73 @@
1
+ #!/usr/bin/env ruby
2
+
3
+ require 'awesome_print'
4
+ require 'mobileminer/adapter'
5
+ require 'cgminer/api'
6
+ require 'logger'
7
+ require 'optparse'
8
+ require 'yaml'
9
+
10
+ me = File.basename(__FILE__)
11
+
12
+ log = Logger.new(STDOUT)
13
+ log.level = Logger::WARN
14
+
15
+ options = {}
16
+ optparse = OptionParser.new do |opts|
17
+ opts.banner = "Usage: #{me} [options]"
18
+ opts.on('-s', '--sample [FILE]', String, 'Write a sample configuration file') do |file|
19
+ file ||= 'config.yaml'
20
+ File.open(file, 'w') do |f|
21
+ f.write <<EOF
22
+ # Set the email address and application key to your own.
23
+ application-key: REPLACE_WITH_YOUR_APPLICATION_KEY
24
+ email-address: REPLACE_WITH_YOUR_EMAIL_ADDRESS
25
+
26
+ # Add the miners you wish to monitor with this agent.
27
+ # port is optional, it defaults to 4028.
28
+ miners:
29
+ 'Unnamed Miner 1':
30
+ host: miner1_hostname
31
+ 'Unnamed Miner 2':
32
+ host: miner2_hostname
33
+ port: 4029
34
+ EOF
35
+ end
36
+ puts "Sample configuration written to #{file}"
37
+ exit
38
+ end
39
+ opts.on('-c', '--config FILE', String, 'YAML configuration file') do |c|
40
+ options[:config] = c
41
+ end
42
+ opts.on('-d', '--debug LEVEL', [:debug, :info], 'Debug level (debug, info)') do |dbg|
43
+ log.level = { debug: Logger::DEBUG, info: Logger::INFO }.fetch(dbg, Logger::WARN)
44
+ end
45
+ opts.on('-v', '--version', 'Print application version') do
46
+ puts "#{me} v#{Mobileminer::Adapter::VERSION}"
47
+ exit
48
+ end
49
+ end.parse!
50
+
51
+ fail 'Configuration file is required' unless options[:config]
52
+
53
+ begin
54
+ config = YAML.load(open(options[:config]).read)
55
+ rescue => e
56
+ puts 'An error occurred parsing configuration file'
57
+ fail
58
+ end
59
+
60
+ key = options.fetch(:key, config['application-key'])
61
+ fail 'Application key is required' if key.nil?
62
+
63
+ email = options.fetch(:key, config['email-address'])
64
+ fail 'Email address is required' if email.nil?
65
+
66
+ # build a hash-o-miner clients
67
+ miners = config['miners'].map do |miner|
68
+ client = CGMiner::API::Client.new(miner[1]['host'], miner[1]['port'] || 4028)
69
+ { name: miner[0], client: client }
70
+ end
71
+
72
+ monitor = Mobileminer::Adapter::Monitor.new(email, key, miners, log)
73
+ monitor.run
@@ -0,0 +1,4 @@
1
+ require 'mobileminer/adapter/api'
2
+ require 'mobileminer/adapter/fetcher'
3
+ require 'mobileminer/adapter/monitor'
4
+ require 'mobileminer/adapter/version'
@@ -0,0 +1,39 @@
1
+ require 'httparty'
2
+
3
+ module Mobileminer
4
+ module Adapter
5
+
6
+ class API
7
+
8
+ @@API_KEY = '64OVS1JhOcDOTn'
9
+
10
+ include HTTParty
11
+ base_uri 'https://mobileminer.azurewebsites.net/api'
12
+
13
+ def initialize(email, key, logger)
14
+ @email = email
15
+ @key = key
16
+ @log = logger
17
+ end
18
+
19
+ def statistics(machine, body)
20
+ command("/MiningStatisticsInput#{auth_string}&machineName=#{machine}", body)
21
+ end
22
+
23
+ private
24
+
25
+ def auth_string
26
+ "?emailAddress=#{@email}&applicationKey=#{@key}&apiKey=#{@@API_KEY}"
27
+ end
28
+
29
+ def command(command, body)
30
+ command = URI.encode(command)
31
+ body = body.to_json
32
+ @log.debug("Posting command: #{command}, body: #{body}")
33
+ self.class.post(command, body: body, :headers => { 'Content-Type' => 'application/json' })
34
+ end
35
+
36
+ end
37
+
38
+ end
39
+ end
@@ -0,0 +1,119 @@
1
+ module Mobileminer
2
+ module Adapter
3
+
4
+ class Fetcher
5
+
6
+ def initialize(client, name, logger)
7
+ @client = client
8
+ @name = name
9
+ @log = logger
10
+ end
11
+
12
+ def get_update
13
+ devs = @client.devs
14
+ details = @client.devdetails
15
+ pools = @client.pools
16
+
17
+ # XXX check status replies on each command?
18
+
19
+ updates = []
20
+ devs.body.each do |dev|
21
+ update = host_info
22
+ if dev.has_key? 'GPU'
23
+ update['Kind'] = update['Name'] = 'GPU'
24
+ update['FullName'] = find_device_details(details.body, :gpu, dev['GPU'])['Model']
25
+ update.merge!(gpu_info(dev))
26
+ elsif dev.has_key? 'ASC'
27
+ update['Kind'] = update['Name'] = 'ASC'
28
+ update['FullName'] = find_device_details(details.body, :asc, dev['ASC'])['Model']
29
+ update.merge!(asic_info(dev))
30
+ elsif dev.has_key? 'PGA'
31
+ update['Kind'] = update['Name'] = 'PGA'
32
+ update['FullName'] = find_device_details(details.body, :pga, dev['PGA'])['Model']
33
+ update.merge!(fpga_info(dev))
34
+ else
35
+ @log.warn("Skipped unknown device: #{dev}")
36
+ end
37
+
38
+ active_pool = pools.body.select { |pool| pool['Stratum Active'] }.first
39
+ update['PoolIndex'] = active_pool['POOL']
40
+ update['PoolName'] = active_pool['URL']
41
+
42
+ updates << update
43
+ end
44
+
45
+ @log.debug("Built update: #{updates}")
46
+ updates
47
+ end
48
+
49
+ private
50
+
51
+ def find_device_details(details, type, index)
52
+ details.select { |detail| detail['Name'] == type.to_s.upcase }[index]
53
+ end
54
+
55
+ def find_mhash_current(blob)
56
+ blob[blob.keys.select { |k| /MHS \ds/.match(k) }.first] || 0
57
+ end
58
+
59
+ def host_info
60
+ {
61
+ 'MinerName' => 'mobileminer-adapter-gem',
62
+ 'CoinSymbol' => 'LTC',
63
+ 'CoinName' => 'Litecoin',
64
+ 'Algorithm' => 'scrypt',
65
+ }
66
+ end
67
+
68
+ def device_info(device)
69
+ {
70
+ 'Temperature' => device['Temperature'],
71
+ 'Enabled' => device['Enabled'] == 'Y',
72
+ 'Status' => device['Status'],
73
+ 'AverageHashrate' => device['MHS av'].to_f * 1000.0,
74
+ 'CurrentHashrate' => find_mhash_current(device) * 1000.0,
75
+ 'AcceptedShares' => device['Accepted'],
76
+ 'RejectedShares' => device['Rejected'],
77
+ 'HardwareErrors' => device['Hardware Errors'],
78
+ 'Utility' => device['Utility'],
79
+ 'RejectedSharesPercent' => device['Device Rejected%'],
80
+ }
81
+ end
82
+
83
+ def asic_info(device)
84
+ {
85
+ 'Index' => device['ASC'],
86
+ }.merge(device_info(device))
87
+ end
88
+
89
+ def fpga_info(device)
90
+ {
91
+ 'Index' => device['PGA'],
92
+ }.merge(device_info(device))
93
+ end
94
+
95
+ def gpu_info(device)
96
+ {
97
+ 'Index' => device['GPU'],
98
+ 'FanSpeed' => device['Fan Speed'],
99
+ 'FanPercent' => device['Fan Percent'],
100
+ 'GpuClock' => device['GPU Clock'],
101
+ 'MemoryClock' => device['Memory Clock'],
102
+ 'GpuVoltage' => device['GPU Voltage'],
103
+ 'GpuActivity' => device['GPU Activity'],
104
+ 'PowerTune' => device['Powertune'],
105
+ 'Intensity' => device['Intensity'],
106
+ }.merge(device_info(device))
107
+ end
108
+
109
+ def pool_info(pool)
110
+ {
111
+ 'PoolIndex' => pool['POOL'],
112
+ 'PoolName' => pool['URL'],
113
+ }
114
+ end
115
+
116
+ end
117
+
118
+ end
119
+ end
@@ -0,0 +1,70 @@
1
+ require 'json'
2
+
3
+ module Mobileminer
4
+ module Adapter
5
+
6
+ class Monitor
7
+
8
+ def initialize(email, key, miners, logger)
9
+ @email = email
10
+ @key = key
11
+ @miners = miners.map do |m|
12
+ fetcher = Mobileminer::Adapter::Fetcher.new(m[:client], m[:name], logger)
13
+ { fetcher: fetcher }.merge(m)
14
+ end
15
+ @log = logger
16
+ end
17
+
18
+ def run
19
+ api = Mobileminer::Adapter::API.new(@email, @key, @log)
20
+
21
+ @log.info("Monitoring #{@miners.length} miners")
22
+ loop do
23
+ @miners.each do |miner|
24
+ @log.debug("Beginning miner: #{miner}")
25
+
26
+ begin
27
+ update = miner[:fetcher].get_update
28
+ rescue Timeout::Error => e
29
+ @log.warn("Timeout::Error building update for #{miner[:name]} (#{e})")
30
+ next
31
+ # XXX put something in the update to indicate it barfed?
32
+ rescue SystemCallError => e
33
+ @log.error("SystemCallError building update for #{miner[:name]} (#{e})")
34
+ next
35
+ # XXX put something in the update to indicate it barfed?
36
+ rescue Exception => e
37
+ @log.error("Exception building update for #{miner[:name]} (#{e})")
38
+ next
39
+ # XXX put something in the update to indicate it barfed?
40
+ end
41
+
42
+ @log.info("Submitting update for #{miner[:name]}")
43
+ begin
44
+ tries ||= 3
45
+ response = api.statistics(miner[:name], update)
46
+ rescue Exception => e
47
+ @log.error("Exception submitting updates (#{e})")
48
+ unless (tries -= 1).zero?
49
+ @log.error('Retrying...')
50
+ retry
51
+ else
52
+ @log.error('Giving up for this update...')
53
+ end
54
+ else
55
+ if 201 == response.code
56
+ @log.debug('Successfully submitted update')
57
+ else
58
+ @log.error('Error submitting updates, check API key!')
59
+ end
60
+ end
61
+ end
62
+
63
+ sleep 60
64
+ end
65
+ end
66
+
67
+ end
68
+
69
+ end
70
+ end
@@ -0,0 +1,5 @@
1
+ module Mobileminer
2
+ module Adapter
3
+ VERSION = '0.0.1'
4
+ end
5
+ end
@@ -0,0 +1,34 @@
1
+ # coding: utf-8
2
+ lib = File.expand_path('../lib', __FILE__)
3
+ $LOAD_PATH.unshift(lib) unless $LOAD_PATH.include?(lib)
4
+ require 'mobileminer/adapter/version'
5
+
6
+ Gem::Specification.new do |spec|
7
+ spec.name = 'mobileminer-adapter'
8
+ spec.version = Mobileminer::Adapter::VERSION
9
+ spec.authors = ['Nick Veys']
10
+ spec.email = ['nick@codelever.com']
11
+ spec.description = %q{Ruby gem to report miner information to MobileMiner.}
12
+ spec.summary = %q{MobileMiner reporting gem.}
13
+ spec.homepage = 'http://github.com/code-lever/mobileminer-adapter-gem.git'
14
+ spec.license = 'MIT'
15
+
16
+ spec.files = `git ls-files -z`.split("\x0")
17
+ spec.executables = spec.files.grep(%r{^bin/}) { |f| File.basename(f) }
18
+ spec.test_files = spec.files.grep(%r{^(test|spec|features)/})
19
+ spec.require_paths = ['lib']
20
+
21
+ spec.required_ruby_version = '>= 1.9.3'
22
+
23
+ spec.add_development_dependency 'bundler', '~> 1.5'
24
+ spec.add_development_dependency 'ci_reporter', '= 1.8.4'
25
+ spec.add_development_dependency 'rake', '~> 10.0'
26
+ spec.add_development_dependency 'rspec', '~> 2.13'
27
+ spec.add_development_dependency 'simplecov'
28
+ spec.add_development_dependency 'simplecov-gem-adapter'
29
+ spec.add_development_dependency 'simplecov-rcov'
30
+
31
+ spec.add_dependency 'awesome_print', '~> 1.2'
32
+ spec.add_dependency 'cgminer-api', '~> 0.1'
33
+ spec.add_dependency 'httparty', '~> 0.12'
34
+ end
metadata ADDED
@@ -0,0 +1,199 @@
1
+ --- !ruby/object:Gem::Specification
2
+ name: mobileminer-adapter
3
+ version: !ruby/object:Gem::Version
4
+ version: 0.0.1
5
+ platform: ruby
6
+ authors:
7
+ - Nick Veys
8
+ autorequire:
9
+ bindir: bin
10
+ cert_chain: []
11
+ date: 2014-02-19 00:00:00.000000000 Z
12
+ dependencies:
13
+ - !ruby/object:Gem::Dependency
14
+ name: bundler
15
+ requirement: !ruby/object:Gem::Requirement
16
+ requirements:
17
+ - - "~>"
18
+ - !ruby/object:Gem::Version
19
+ version: '1.5'
20
+ type: :development
21
+ prerelease: false
22
+ version_requirements: !ruby/object:Gem::Requirement
23
+ requirements:
24
+ - - "~>"
25
+ - !ruby/object:Gem::Version
26
+ version: '1.5'
27
+ - !ruby/object:Gem::Dependency
28
+ name: ci_reporter
29
+ requirement: !ruby/object:Gem::Requirement
30
+ requirements:
31
+ - - '='
32
+ - !ruby/object:Gem::Version
33
+ version: 1.8.4
34
+ type: :development
35
+ prerelease: false
36
+ version_requirements: !ruby/object:Gem::Requirement
37
+ requirements:
38
+ - - '='
39
+ - !ruby/object:Gem::Version
40
+ version: 1.8.4
41
+ - !ruby/object:Gem::Dependency
42
+ name: rake
43
+ requirement: !ruby/object:Gem::Requirement
44
+ requirements:
45
+ - - "~>"
46
+ - !ruby/object:Gem::Version
47
+ version: '10.0'
48
+ type: :development
49
+ prerelease: false
50
+ version_requirements: !ruby/object:Gem::Requirement
51
+ requirements:
52
+ - - "~>"
53
+ - !ruby/object:Gem::Version
54
+ version: '10.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: '2.13'
62
+ type: :development
63
+ prerelease: false
64
+ version_requirements: !ruby/object:Gem::Requirement
65
+ requirements:
66
+ - - "~>"
67
+ - !ruby/object:Gem::Version
68
+ version: '2.13'
69
+ - !ruby/object:Gem::Dependency
70
+ name: simplecov
71
+ requirement: !ruby/object:Gem::Requirement
72
+ requirements:
73
+ - - ">="
74
+ - !ruby/object:Gem::Version
75
+ version: '0'
76
+ type: :development
77
+ prerelease: false
78
+ version_requirements: !ruby/object:Gem::Requirement
79
+ requirements:
80
+ - - ">="
81
+ - !ruby/object:Gem::Version
82
+ version: '0'
83
+ - !ruby/object:Gem::Dependency
84
+ name: simplecov-gem-adapter
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
+ - !ruby/object:Gem::Dependency
98
+ name: simplecov-rcov
99
+ requirement: !ruby/object:Gem::Requirement
100
+ requirements:
101
+ - - ">="
102
+ - !ruby/object:Gem::Version
103
+ version: '0'
104
+ type: :development
105
+ prerelease: false
106
+ version_requirements: !ruby/object:Gem::Requirement
107
+ requirements:
108
+ - - ">="
109
+ - !ruby/object:Gem::Version
110
+ version: '0'
111
+ - !ruby/object:Gem::Dependency
112
+ name: awesome_print
113
+ requirement: !ruby/object:Gem::Requirement
114
+ requirements:
115
+ - - "~>"
116
+ - !ruby/object:Gem::Version
117
+ version: '1.2'
118
+ type: :runtime
119
+ prerelease: false
120
+ version_requirements: !ruby/object:Gem::Requirement
121
+ requirements:
122
+ - - "~>"
123
+ - !ruby/object:Gem::Version
124
+ version: '1.2'
125
+ - !ruby/object:Gem::Dependency
126
+ name: cgminer-api
127
+ requirement: !ruby/object:Gem::Requirement
128
+ requirements:
129
+ - - "~>"
130
+ - !ruby/object:Gem::Version
131
+ version: '0.1'
132
+ type: :runtime
133
+ prerelease: false
134
+ version_requirements: !ruby/object:Gem::Requirement
135
+ requirements:
136
+ - - "~>"
137
+ - !ruby/object:Gem::Version
138
+ version: '0.1'
139
+ - !ruby/object:Gem::Dependency
140
+ name: httparty
141
+ requirement: !ruby/object:Gem::Requirement
142
+ requirements:
143
+ - - "~>"
144
+ - !ruby/object:Gem::Version
145
+ version: '0.12'
146
+ type: :runtime
147
+ prerelease: false
148
+ version_requirements: !ruby/object:Gem::Requirement
149
+ requirements:
150
+ - - "~>"
151
+ - !ruby/object:Gem::Version
152
+ version: '0.12'
153
+ description: Ruby gem to report miner information to MobileMiner.
154
+ email:
155
+ - nick@codelever.com
156
+ executables:
157
+ - mobileminer-adapter
158
+ extensions: []
159
+ extra_rdoc_files: []
160
+ files:
161
+ - ".gitignore"
162
+ - ".ruby-version"
163
+ - ".travis.yml"
164
+ - Gemfile
165
+ - LICENSE.txt
166
+ - README.md
167
+ - Rakefile
168
+ - bin/mobileminer-adapter
169
+ - lib/mobileminer/adapter.rb
170
+ - lib/mobileminer/adapter/api.rb
171
+ - lib/mobileminer/adapter/fetcher.rb
172
+ - lib/mobileminer/adapter/monitor.rb
173
+ - lib/mobileminer/adapter/version.rb
174
+ - mobileminer-adapter.gemspec
175
+ homepage: http://github.com/code-lever/mobileminer-adapter-gem.git
176
+ licenses:
177
+ - MIT
178
+ metadata: {}
179
+ post_install_message:
180
+ rdoc_options: []
181
+ require_paths:
182
+ - lib
183
+ required_ruby_version: !ruby/object:Gem::Requirement
184
+ requirements:
185
+ - - ">="
186
+ - !ruby/object:Gem::Version
187
+ version: 1.9.3
188
+ required_rubygems_version: !ruby/object:Gem::Requirement
189
+ requirements:
190
+ - - ">="
191
+ - !ruby/object:Gem::Version
192
+ version: '0'
193
+ requirements: []
194
+ rubyforge_project:
195
+ rubygems_version: 2.2.1
196
+ signing_key:
197
+ specification_version: 4
198
+ summary: MobileMiner reporting gem.
199
+ test_files: []