resque-analytics 0.5.0

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: 7de5a9dd7d1aec1589887da6071dc7b988f71c3a
4
+ data.tar.gz: 22e1b5663baa1e2eeac7c2e858291899a59e26bf
5
+ SHA512:
6
+ metadata.gz: 87b121b68b72695be1cc095cfaa7d3a7b0854c5f793d42bab36f2e3453833c9101d7e11684dde7ec368bf1f2c210a0ab11f70e85bae348b0fdc9002c27bb1032
7
+ data.tar.gz: 6c81ccd7035efa82023b90befea4e927239b758bff7e4779298a9d3b4baecbfb83b02b1428419bbc31d607d23c1342b7c4fb28492488c9a45e62b24bdf3a254e
data/.document ADDED
@@ -0,0 +1,2 @@
1
+ lib/**/*.rb
2
+ LICENSE.txt
data/Gemfile ADDED
@@ -0,0 +1,10 @@
1
+ source 'http://rubygems.org'
2
+
3
+ gem 'resque', "1.25.1"
4
+ gem 'googlecharts', "1.6.8"
5
+
6
+ group :development do
7
+ gem 'bundler'
8
+ gem 'jeweler'
9
+ gem 'byebug'
10
+ end
data/Gemfile.lock ADDED
@@ -0,0 +1,83 @@
1
+ GEM
2
+ remote: http://rubygems.org/
3
+ specs:
4
+ addressable (2.3.6)
5
+ builder (3.2.2)
6
+ byebug (2.7.0)
7
+ columnize (~> 0.3)
8
+ debugger-linecache (~> 1.2)
9
+ columnize (0.3.6)
10
+ debugger-linecache (1.2.0)
11
+ descendants_tracker (0.0.4)
12
+ thread_safe (~> 0.3, >= 0.3.1)
13
+ faraday (0.9.0)
14
+ multipart-post (>= 1.2, < 3)
15
+ git (1.2.6)
16
+ github_api (0.11.3)
17
+ addressable (~> 2.3)
18
+ descendants_tracker (~> 0.0.1)
19
+ faraday (~> 0.8, < 0.10)
20
+ hashie (>= 1.2)
21
+ multi_json (>= 1.7.5, < 2.0)
22
+ nokogiri (~> 1.6.0)
23
+ oauth2
24
+ googlecharts (1.6.8)
25
+ hashie (3.0.0)
26
+ highline (1.6.21)
27
+ jeweler (2.0.1)
28
+ builder
29
+ bundler (>= 1.0)
30
+ git (>= 1.2.5)
31
+ github_api
32
+ highline (>= 1.6.15)
33
+ nokogiri (>= 1.5.10)
34
+ rake
35
+ rdoc
36
+ json (1.8.1)
37
+ jwt (1.0.0)
38
+ mini_portile (0.6.0)
39
+ mono_logger (1.1.0)
40
+ multi_json (1.10.1)
41
+ multi_xml (0.5.5)
42
+ multipart-post (2.0.0)
43
+ nokogiri (1.6.2.1)
44
+ mini_portile (= 0.6.0)
45
+ oauth2 (0.9.4)
46
+ faraday (>= 0.8, < 0.10)
47
+ jwt (~> 1.0)
48
+ multi_json (~> 1.3)
49
+ multi_xml (~> 0.5)
50
+ rack (~> 1.2)
51
+ rack (1.5.2)
52
+ rack-protection (1.5.3)
53
+ rack
54
+ rake (10.3.2)
55
+ rdoc (4.1.1)
56
+ json (~> 1.4)
57
+ redis (3.0.7)
58
+ redis-namespace (1.4.1)
59
+ redis (~> 3.0.4)
60
+ resque (1.25.1)
61
+ mono_logger (~> 1.0)
62
+ multi_json (~> 1.0)
63
+ redis-namespace (~> 1.2)
64
+ sinatra (>= 0.9.2)
65
+ vegas (~> 0.1.2)
66
+ sinatra (1.4.5)
67
+ rack (~> 1.4)
68
+ rack-protection (~> 1.4)
69
+ tilt (~> 1.3, >= 1.3.4)
70
+ thread_safe (0.3.4)
71
+ tilt (1.4.1)
72
+ vegas (0.1.11)
73
+ rack (>= 1.0.0)
74
+
75
+ PLATFORMS
76
+ ruby
77
+
78
+ DEPENDENCIES
79
+ bundler
80
+ byebug
81
+ googlecharts (= 1.6.8)
82
+ jeweler
83
+ resque (= 1.25.1)
data/LICENSE ADDED
@@ -0,0 +1,21 @@
1
+ The MIT License (MIT)
2
+
3
+ Copyright (c) 2014 Nir Tzur
4
+
5
+ Permission is hereby granted, free of charge, to any person obtaining a copy
6
+ of this software and associated documentation files (the "Software"), to deal
7
+ in the Software without restriction, including without limitation the rights
8
+ to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
9
+ copies of the Software, and to permit persons to whom the Software is
10
+ furnished to do so, subject to the following conditions:
11
+
12
+ The above copyright notice and this permission notice shall be included in all
13
+ copies or substantial portions of the Software.
14
+
15
+ THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
16
+ IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
17
+ FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
18
+ AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
19
+ LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
20
+ OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
21
+ SOFTWARE.
data/README.rdoc ADDED
@@ -0,0 +1,56 @@
1
+ = resque-analytics
2
+
3
+ Resque jobs analytics and key performance indicators over time
4
+
5
+ This gem keeps statistics per worker for the last 90 days, to be later shown as over time graphs.
6
+ For each worker, the gem keeps the following key performance indicators:
7
+ * Number of times it executed successfuly per day
8
+ * Number of times is failed per day
9
+ * Total run time per day
10
+
11
+ == Installation
12
+
13
+ Requires resque '~> 1.25.1', and googlecharts '~> 1.6.8'
14
+
15
+
16
+ Add the following to your gemfile:
17
+
18
+ gem 'resque-analytics', require: 'resque-analytics/server'
19
+
20
+ == Usage
21
+
22
+ Extend Resque workers with:
23
+
24
+ class Worker
25
+ extend Resque::Plugins::Analytics
26
+
27
+ @queue = :queue
28
+ def self.perform(*args)
29
+ # ..
30
+ end
31
+ end
32
+
33
+ On the Resque screens, you will find a new tab called "Analytics".
34
+
35
+ == TODO
36
+
37
+ * Add tests
38
+ * Add queued time to total run time analytics
39
+
40
+ == Contributing to resque-analytics
41
+
42
+ * Check out the latest master to make sure the feature hasn't been implemented or the bug hasn't been fixed yet
43
+ * Check out the issue tracker to make sure someone already hasn't requested it and/or contributed it
44
+ * Fork the project
45
+ * Start a feature/bugfix branch
46
+ * Commit and push until you are happy with your contribution
47
+ * Make sure to add tests for it. This is important so I don't break it in a future version unintentionally.
48
+ * Please try not to mess with the Rakefile, version, or history. If you want to have your own version, or is otherwise necessary, that is fine, but please isolate to its own commit so I can cherry-pick around it.
49
+
50
+ == Contributers
51
+
52
+ * {nirtzur}[https://github.com/nirtzur]
53
+
54
+ == Copyright
55
+
56
+ Copyright (c) 2014 Nir Tzur. See LICENSE for further details.
data/Rakefile ADDED
@@ -0,0 +1,51 @@
1
+ # encoding: utf-8
2
+
3
+ require 'rubygems'
4
+ require 'bundler'
5
+ begin
6
+ Bundler.setup(:default, :development)
7
+ rescue Bundler::BundlerError => e
8
+ $stderr.puts e.message
9
+ $stderr.puts "Run `bundle install` to install missing gems"
10
+ exit e.status_code
11
+ end
12
+ require 'rake'
13
+
14
+ require 'jeweler'
15
+ Jeweler::Tasks.new do |gem|
16
+ # gem is a Gem::Specification... see http://guides.rubygems.org/specification-reference/ for more options
17
+ gem.name = "resque-analytics"
18
+ gem.homepage = "http://github.com/nirtzur/resque-analytics"
19
+ gem.license = "MIT"
20
+ gem.summary = %Q{Resque Job Analytics}
21
+ gem.description = %Q{Shows Resque jobs key performance indciators over time}
22
+ gem.email = "nir.tzur@samanage.com"
23
+ gem.authors = ["Nir Tzur"]
24
+ # dependencies defined in Gemfile
25
+ end
26
+ Jeweler::RubygemsDotOrgTasks.new
27
+
28
+ require 'rake/testtask'
29
+ Rake::TestTask.new(:test) do |test|
30
+ test.libs << 'lib' << 'test'
31
+ test.pattern = 'test/**/test_*.rb'
32
+ test.verbose = true
33
+ end
34
+
35
+ desc "Code coverage detail"
36
+ task :simplecov do
37
+ ENV['COVERAGE'] = "true"
38
+ Rake::Task['test'].execute
39
+ end
40
+
41
+ task :default => :test
42
+
43
+ require 'rdoc/task'
44
+ Rake::RDocTask.new do |rdoc|
45
+ version = File.exist?('VERSION') ? File.read('VERSION') : ""
46
+
47
+ rdoc.rdoc_dir = 'rdoc'
48
+ rdoc.title = "resque-analytics #{version}"
49
+ rdoc.rdoc_files.include('README*')
50
+ rdoc.rdoc_files.include('lib/**/*.rb')
51
+ end
data/VERSION ADDED
@@ -0,0 +1 @@
1
+ 0.5.0
@@ -0,0 +1,38 @@
1
+ require 'resque'
2
+
3
+ module Resque
4
+ module Plugins
5
+ module Analytics
6
+
7
+ FAILED = "failed"
8
+ PERFORMED = "performed"
9
+ TOTAL_TIME = "total_time"
10
+ # WAIT_TIME = "wait_time"
11
+
12
+ def key(kpi)
13
+ date = Time.now.strftime("%y_%m_%d")
14
+ "analytics:#{kpi}:#{self.name}:#{date}"
15
+ end
16
+
17
+ def around_perform_analytics(*args)
18
+ start = Time.now
19
+ yield
20
+ total_time = Time.now - start
21
+
22
+ Resque.redis.lpush(key(TOTAL_TIME), total_time)
23
+ Resque.redis.expire(key(TOTAL_TIME), 90 * 24 * 60)
24
+ end
25
+
26
+ def after_perform_analytics(*args)
27
+ Resque.redis.incr(key(PERFORMED))
28
+ Resque.redis.expire(key(PERFORMED), 90 * 24 * 60)
29
+ end
30
+
31
+ def on_failure_analytics(e,*args)
32
+ Resque.redis.incr(key(FAILED))
33
+ Resque.redis.expire(key(FAILED), 90 * 24 * 60)
34
+ end
35
+
36
+ end
37
+ end
38
+ end
@@ -0,0 +1,37 @@
1
+ <h1>Resque Analytics</h1>
2
+
3
+ <p class="intro">
4
+ Various KPIs related to Resque jobs
5
+ </p>
6
+
7
+ <% @data.keys.each do |job| %>
8
+ <h2><%= job %></h2>
9
+ <% performed = chart_data(@data, job, Resque::Plugins::Analytics::PERFORMED) %>
10
+ <% failed = chart_data(@data, job, Resque::Plugins::Analytics::FAILED) %>
11
+ <%= Gchart.bar(
12
+ data: [performed, failed],
13
+ title: "Number of jobs",
14
+ format: "image_tag",
15
+ size: "1000x300",
16
+ axis_with_labels: ['x,y'],
17
+ bar_width_and_spacing: '6',
18
+ axis_range: [[-90,0,-5], [0, performed.max + failed.max ]],
19
+ legend: [Resque::Plugins::Analytics::PERFORMED, Resque::Plugins::Analytics::FAILED],
20
+ legend_position: 'top',
21
+ bar_colors: ['000000', '0088FF']
22
+ ) %>
23
+
24
+ <% run_time = chart_data(@data, job, Resque::Plugins::Analytics::TOTAL_TIME) %>
25
+ <%= Gchart.bar(
26
+ data: [run_time],
27
+ title: "Run time",
28
+ format: "image_tag",
29
+ size: "1000x300",
30
+ axis_with_labels: ['x,y'],
31
+ bar_width_and_spacing: '6',
32
+ axis_range: [[-90,0,-5], [0, run_time.max ]],
33
+ legend: [Resque::Plugins::Analytics::TOTAL_TIME],
34
+ legend_position: 'top'
35
+ ) %>
36
+
37
+ <% end %>
@@ -0,0 +1,58 @@
1
+ require 'resque/server'
2
+ require 'resque/plugins/analytics'
3
+
4
+ module Resque
5
+ module Plugins
6
+ module Analytics
7
+ module Server
8
+ VIEW_PATH = File.join(File.dirname(__FILE__), 'server', 'views')
9
+
10
+ module Helpers
11
+ def counters_for(job, kpi)
12
+ kpi_keys = Resque.redis.keys("analytics:#{kpi}:#{job}:*").sort { |a,b| a <=> b }
13
+ kpi_keys.inject({}) { |res, key| res[key.split(':').last] = key_value(key); res}
14
+ end
15
+
16
+ def measured_jobs
17
+ Resque.redis.keys("analytics*").map { |key| key.split(':')[2] }.uniq
18
+ end
19
+
20
+ def key_value(key)
21
+ if Resque.redis.type(key) == "string"
22
+ Resque.redis.get(key)
23
+ else
24
+ Resque.redis.lrange(key, 0, -1).map(&:to_f).sum
25
+ end
26
+ end
27
+
28
+ def legend_keys
29
+ @legend_keys ||= (-90..0).map { |number| number.days.from_now.strftime("%y_%m_%d")}
30
+ end
31
+
32
+ def chart_data(data, job, kpi)
33
+ legend_keys.map { |key| @data[job][kpi][key].to_i || 0 }
34
+ end
35
+ end
36
+
37
+ class << self
38
+ def registered(app)
39
+ app.get '/analytics' do
40
+ @data = measured_jobs.inject({}) { |res, job|
41
+ res[job] = {}
42
+ [PERFORMED, FAILED, TOTAL_TIME].each { |kpi| res[job][kpi] = counters_for(job, kpi) }
43
+ res
44
+ }
45
+ erb(File.read(File.join(VIEW_PATH, 'analytics.erb')))
46
+ end
47
+
48
+ app.tabs << "Analytics"
49
+
50
+ app.helpers(Helpers)
51
+ end
52
+ end
53
+ end
54
+ end
55
+ end
56
+ end
57
+
58
+ Resque::Server.register Resque::Plugins::Analytics::Server
@@ -0,0 +1 @@
1
+ require 'resque/plugins/analytics'
@@ -0,0 +1,63 @@
1
+ # Generated by jeweler
2
+ # DO NOT EDIT THIS FILE DIRECTLY
3
+ # Instead, edit Jeweler::Tasks in Rakefile, and run 'rake gemspec'
4
+ # -*- encoding: utf-8 -*-
5
+ # stub: resque-analytics 0.5.0 ruby lib
6
+
7
+ Gem::Specification.new do |s|
8
+ s.name = "resque-analytics"
9
+ s.version = "0.5.0"
10
+
11
+ s.required_rubygems_version = Gem::Requirement.new(">= 0") if s.respond_to? :required_rubygems_version=
12
+ s.require_paths = ["lib"]
13
+ s.authors = ["Nir Tzur"]
14
+ s.date = "2014-06-08"
15
+ s.description = "Shows Resque jobs key performance indciators over time"
16
+ s.email = "nir.tzur@samanage.com"
17
+ s.extra_rdoc_files = [
18
+ "LICENSE",
19
+ "README.rdoc"
20
+ ]
21
+ s.files = [
22
+ ".document",
23
+ "Gemfile",
24
+ "Gemfile.lock",
25
+ "LICENSE",
26
+ "Rakefile",
27
+ "VERSION",
28
+ "lib/resque-analytics.rb",
29
+ "lib/resque-analytics/server.rb",
30
+ "lib/resque-analytics/server/views/analytics.erb",
31
+ "lib/resque/plugins/analytics.rb",
32
+ "resque-analytics.gemspec"
33
+ ]
34
+ s.homepage = "http://github.com/nirtzur/resque-analytics"
35
+ s.licenses = ["MIT"]
36
+ s.rubygems_version = "2.2.2"
37
+ s.summary = "Resque Job Analytics"
38
+
39
+ if s.respond_to? :specification_version then
40
+ s.specification_version = 4
41
+
42
+ if Gem::Version.new(Gem::VERSION) >= Gem::Version.new('1.2.0') then
43
+ s.add_runtime_dependency(%q<resque>, ["= 1.25.1"])
44
+ s.add_runtime_dependency(%q<googlecharts>, ["= 1.6.8"])
45
+ s.add_development_dependency(%q<bundler>, [">= 0"])
46
+ s.add_development_dependency(%q<jeweler>, [">= 0"])
47
+ s.add_development_dependency(%q<byebug>, [">= 0"])
48
+ else
49
+ s.add_dependency(%q<resque>, ["= 1.25.1"])
50
+ s.add_dependency(%q<googlecharts>, ["= 1.6.8"])
51
+ s.add_dependency(%q<bundler>, [">= 0"])
52
+ s.add_dependency(%q<jeweler>, [">= 0"])
53
+ s.add_dependency(%q<byebug>, [">= 0"])
54
+ end
55
+ else
56
+ s.add_dependency(%q<resque>, ["= 1.25.1"])
57
+ s.add_dependency(%q<googlecharts>, ["= 1.6.8"])
58
+ s.add_dependency(%q<bundler>, [">= 0"])
59
+ s.add_dependency(%q<jeweler>, [">= 0"])
60
+ s.add_dependency(%q<byebug>, [">= 0"])
61
+ end
62
+ end
63
+
metadata ADDED
@@ -0,0 +1,127 @@
1
+ --- !ruby/object:Gem::Specification
2
+ name: resque-analytics
3
+ version: !ruby/object:Gem::Version
4
+ version: 0.5.0
5
+ platform: ruby
6
+ authors:
7
+ - Nir Tzur
8
+ autorequire:
9
+ bindir: bin
10
+ cert_chain: []
11
+ date: 2014-06-08 00:00:00.000000000 Z
12
+ dependencies:
13
+ - !ruby/object:Gem::Dependency
14
+ name: resque
15
+ requirement: !ruby/object:Gem::Requirement
16
+ requirements:
17
+ - - '='
18
+ - !ruby/object:Gem::Version
19
+ version: 1.25.1
20
+ type: :runtime
21
+ prerelease: false
22
+ version_requirements: !ruby/object:Gem::Requirement
23
+ requirements:
24
+ - - '='
25
+ - !ruby/object:Gem::Version
26
+ version: 1.25.1
27
+ - !ruby/object:Gem::Dependency
28
+ name: googlecharts
29
+ requirement: !ruby/object:Gem::Requirement
30
+ requirements:
31
+ - - '='
32
+ - !ruby/object:Gem::Version
33
+ version: 1.6.8
34
+ type: :runtime
35
+ prerelease: false
36
+ version_requirements: !ruby/object:Gem::Requirement
37
+ requirements:
38
+ - - '='
39
+ - !ruby/object:Gem::Version
40
+ version: 1.6.8
41
+ - !ruby/object:Gem::Dependency
42
+ name: bundler
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: jeweler
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
+ - !ruby/object:Gem::Dependency
70
+ name: byebug
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
+ description: Shows Resque jobs key performance indciators over time
84
+ email: nir.tzur@samanage.com
85
+ executables: []
86
+ extensions: []
87
+ extra_rdoc_files:
88
+ - LICENSE
89
+ - README.rdoc
90
+ files:
91
+ - .document
92
+ - Gemfile
93
+ - Gemfile.lock
94
+ - LICENSE
95
+ - Rakefile
96
+ - VERSION
97
+ - lib/resque-analytics.rb
98
+ - lib/resque-analytics/server.rb
99
+ - lib/resque-analytics/server/views/analytics.erb
100
+ - lib/resque/plugins/analytics.rb
101
+ - resque-analytics.gemspec
102
+ - README.rdoc
103
+ homepage: http://github.com/nirtzur/resque-analytics
104
+ licenses:
105
+ - MIT
106
+ metadata: {}
107
+ post_install_message:
108
+ rdoc_options: []
109
+ require_paths:
110
+ - lib
111
+ required_ruby_version: !ruby/object:Gem::Requirement
112
+ requirements:
113
+ - - '>='
114
+ - !ruby/object:Gem::Version
115
+ version: '0'
116
+ required_rubygems_version: !ruby/object:Gem::Requirement
117
+ requirements:
118
+ - - '>='
119
+ - !ruby/object:Gem::Version
120
+ version: '0'
121
+ requirements: []
122
+ rubyforge_project:
123
+ rubygems_version: 2.1.11
124
+ signing_key:
125
+ specification_version: 4
126
+ summary: Resque Job Analytics
127
+ test_files: []