enumstats 0.0.1

Sign up to get free protection for your applications and to get access to all the features.
data/.gitignore ADDED
@@ -0,0 +1,17 @@
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
data/Gemfile ADDED
@@ -0,0 +1,4 @@
1
+ source 'https://rubygems.org'
2
+
3
+ # Specify your gem's dependencies in enumstats.gemspec
4
+ gemspec
data/LICENSE ADDED
@@ -0,0 +1,22 @@
1
+ Copyright (c) 2012 Nicholas E. Rabenau
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
+ # Enumstats
2
+
3
+ TODO: Write a gem description
4
+
5
+ ## Installation
6
+
7
+ Add this line to your application's Gemfile:
8
+
9
+ gem 'enumstats'
10
+
11
+ And then execute:
12
+
13
+ $ bundle
14
+
15
+ Or install it yourself as:
16
+
17
+ $ gem install enumstats
18
+
19
+ ## Usage
20
+
21
+ TODO: Write usage instructions here
22
+
23
+ ## Contributing
24
+
25
+ 1. Fork it
26
+ 2. Create your feature branch (`git checkout -b my-new-feature`)
27
+ 3. Commit your changes (`git commit -am 'Added 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,10 @@
1
+ #!/usr/bin/env rake
2
+ require "bundler/gem_tasks"
3
+ require 'rake/testtask'
4
+
5
+ Rake::TestTask.new do |test|
6
+ test.libs << 'lib' << 'test' << 'test/unit'
7
+ test.pattern = 'test/unit/test_*.rb'
8
+ end
9
+
10
+ task :default => :test
data/enumstats.gemspec ADDED
@@ -0,0 +1,19 @@
1
+ # -*- encoding: utf-8 -*-
2
+ require File.expand_path('../lib/enumstats/version', __FILE__)
3
+
4
+ Gem::Specification.new do |gem|
5
+ gem.authors = ["Nicholas E. Rabenau"]
6
+ gem.email = ["nerab@gmx.net"]
7
+ gem.description = %q{Adds simple stats like average, variance and standard deviation to Enumerable}
8
+ gem.summary = %q{Stats for Enumerables}
9
+ gem.homepage = ""
10
+
11
+ gem.files = `git ls-files`.split($\)
12
+ gem.executables = gem.files.grep(%r{^bin/}).map{ |f| File.basename(f) }
13
+ gem.test_files = gem.files.grep(%r{^(test|spec|features)/})
14
+ gem.name = "enumstats"
15
+ gem.require_paths = ["lib"]
16
+ gem.version = Enumstats::VERSION
17
+
18
+ gem.add_development_dependency 'rake'
19
+ end
@@ -0,0 +1,49 @@
1
+ module Enumerable
2
+ def sum
3
+ inject(0){|acc, i| acc + (block_given? ? yield(i) : i)}
4
+ end
5
+
6
+ def average(&block)
7
+ sum(&block) / count.to_f
8
+ end
9
+
10
+ alias :mean :average
11
+
12
+ def median
13
+ return nil if empty?
14
+
15
+ mid = count / 2
16
+ sorted = to_a.sort
17
+
18
+ if count.odd?
19
+ (block_given? ? yield(sorted[mid]) : sorted[mid])
20
+ else
21
+ if block_given?
22
+ (yield(sorted[mid - 1]) + yield(sorted[mid])).to_f / 2.0
23
+ else
24
+ (sorted[mid - 1] + sorted[mid]).to_f / 2.0
25
+ end
26
+ end
27
+ end
28
+
29
+ def variance(&block)
30
+ return 0 if empty?
31
+ (sum_sq(&block) - ((sum(&block) ** 2) / count)) / (count - 1)
32
+ end
33
+
34
+ def standard_deviation(&block)
35
+ var = variance(&block)
36
+
37
+ if (count > 1 && var != 0)
38
+ Math.sqrt(var)
39
+ else
40
+ 0.0
41
+ end
42
+ end
43
+
44
+ private
45
+
46
+ def sum_sq
47
+ self.inject(0){|acc, i| acc + (block_given? ? yield(i) : i) ** 2}
48
+ end
49
+ end
@@ -0,0 +1,3 @@
1
+ module Enumstats
2
+ VERSION = "0.0.1"
3
+ end
data/lib/enumstats.rb ADDED
@@ -0,0 +1,2 @@
1
+ require "enumstats/version"
2
+ require "enumstats/enumerable"
data/test/helper.rb ADDED
@@ -0,0 +1,2 @@
1
+ require 'minitest/autorun'
2
+ require 'enumstats'
@@ -0,0 +1,47 @@
1
+ require 'helper'
2
+
3
+ class TestEnumerableStats < MiniTest::Unit::TestCase
4
+ def test_stats_example1
5
+ # test data example1 from http://web.cs.wpi.edu/~claypool/misc/stats/stats.txt
6
+ samples = [5.0, 10.0, 15.0]
7
+ assert_in_delta(10.0, samples.mean, 0.001)
8
+ assert_in_delta(25.0, samples.variance, 0.0001)
9
+ assert_in_delta(5.0, samples.standard_deviation, 0.00001)
10
+ assert_in_delta(30.0, samples.sum, 0.001)
11
+ assert_in_delta(5.0, samples.min, 0.001)
12
+ assert_in_delta(15.0, samples.max, 0.001)
13
+ end
14
+
15
+ def test_stats_example2_f2
16
+ # test data example1 from http://web.cs.wpi.edu/~claypool/misc/stats/stats.txt
17
+ samples = [20.0, 40.0, 31.0]
18
+
19
+ assert_in_delta(30.333333333333, samples.mean, 0.001)
20
+ assert_in_delta(100.333333333333, samples.variance, 0.0001)
21
+ assert_in_delta(10.016652800878, samples.standard_deviation, 0.00001)
22
+ assert_in_delta(91.000000000000, samples.sum, 0.001)
23
+ assert_in_delta(20.000000000000, samples.min, 0.001)
24
+ assert_in_delta(40.000000000000, samples.max, 0.001)
25
+ end
26
+
27
+ def test_stats_example2_f4
28
+ # test data example1 from http://web.cs.wpi.edu/~claypool/misc/stats/stats.txt
29
+ samples = [10.2, 19.3, 15.4]
30
+
31
+ assert_in_delta(14.966666666667, samples.mean, 0.001)
32
+ assert_in_delta(20.843333333333, samples.variance, 0.0001)
33
+ assert_in_delta(4.565449959570, samples.standard_deviation, 0.00001)
34
+ assert_in_delta(44.900000000000, samples.sum, 0.001)
35
+ assert_in_delta(10.200000000000, samples.min, 0.001)
36
+ assert_in_delta(19.300000000000, samples.max, 0.001)
37
+ end
38
+
39
+ def test_median
40
+ # test data from http://stackoverflow.com/a/11332640
41
+ samples = [0, 1, 2, 3, 4, 5, 6, 7, 8, 9, -1, -2, -3, -4, -5, -6, -7, -8, -9]
42
+ assert_in_delta(0, samples.median, 0.0001)
43
+
44
+ samples = [0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 17, 18]
45
+ assert_in_delta(9, samples.median, 0.0001)
46
+ end
47
+ end
@@ -0,0 +1,61 @@
1
+ require 'helper'
2
+
3
+ class Person
4
+ attr_accessor :age
5
+
6
+ def initialize(age)
7
+ @age = age
8
+ end
9
+
10
+ def <=>(other)
11
+ @age <=> other.age
12
+ end
13
+ end
14
+
15
+ class TestEnumerableStatsWithBlock < MiniTest::Unit::TestCase
16
+ def test_stats_example1
17
+ # test data example1 from http://web.cs.wpi.edu/~claypool/misc/stats/stats.txt
18
+ samples = [5.0, 10.0, 15.0].map{|f| Person.new(f)}
19
+ assert_in_delta(10.0, samples.mean{|p| p.age}, 1e-6)
20
+ assert_in_delta(25.0, samples.variance{|p| p.age}, 1e-6)
21
+ assert_in_delta(5.0, samples.standard_deviation{|p| p.age}, 1e-6)
22
+ assert_in_delta(30.0, samples.sum{|p| p.age}, 1e-6)
23
+
24
+ # not really testing out impl here, but the original tests this, too
25
+ assert_in_delta(5.0, samples.map{|p| p.age}.min, 1e-6)
26
+ assert_in_delta(15.0, samples.map{|p| p.age}.max, 1e-6)
27
+ end
28
+
29
+ def test_stats_example2_f2
30
+ # test data example2 from http://web.cs.wpi.edu/~claypool/misc/stats/stats.txt
31
+ samples = [20.0, 40.0, 31.0].map{|f| Person.new(f)}
32
+
33
+ assert_in_delta(30.333333333333, samples.mean{|p| p.age}, 1e-6)
34
+ assert_in_delta(100.333333333333, samples.variance{|p| p.age}, 1e-6)
35
+ assert_in_delta(10.016652800878, samples.standard_deviation{|p| p.age}, 1e-6)
36
+ assert_in_delta(91.000000000000, samples.sum{|p| p.age}, 1e-6)
37
+ assert_in_delta(20.000000000000, samples.map{|p| p.age}.min, 1e-6)
38
+ assert_in_delta(40.000000000000, samples.map{|p| p.age}.max, 1e-6)
39
+ end
40
+
41
+ def test_stats_example2_f4
42
+ # test data example2 from http://web.cs.wpi.edu/~claypool/misc/stats/stats.txt
43
+ samples = [10.2, 19.3, 15.4].map{|f| Person.new(f)}
44
+
45
+ assert_in_delta(14.966666666667, samples.mean{|p| p.age}, 1e-6)
46
+ assert_in_delta(20.843333333333, samples.variance{|p| p.age}, 1e-6)
47
+ assert_in_delta(4.565449959570, samples.standard_deviation{|p| p.age}, 1e-6)
48
+ assert_in_delta(44.900000000000, samples.sum{|p| p.age}, 1e-6)
49
+ assert_in_delta(10.200000000000, samples.map{|p| p.age}.min, 1e-6)
50
+ assert_in_delta(19.300000000000, samples.map{|p| p.age}.max, 1e-6)
51
+ end
52
+
53
+ def test_median
54
+ # test data from http://stackoverflow.com/a/11332640
55
+ samples = [0, 1, 2, 3, 4, 5, 6, 7, 8, 9, -1, -2, -3, -4, -5, -6, -7, -8, -9].map{|f| Person.new(f)}
56
+ assert_in_delta(0, samples.median{|p| p.age}, 1e-6)
57
+
58
+ samples = [0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 17, 18].map{|f| Person.new(f)}
59
+ assert_in_delta(9, samples.median{|p| p.age}, 1e-6)
60
+ end
61
+ end
metadata ADDED
@@ -0,0 +1,76 @@
1
+ --- !ruby/object:Gem::Specification
2
+ name: enumstats
3
+ version: !ruby/object:Gem::Version
4
+ version: 0.0.1
5
+ prerelease:
6
+ platform: ruby
7
+ authors:
8
+ - Nicholas E. Rabenau
9
+ autorequire:
10
+ bindir: bin
11
+ cert_chain: []
12
+ date: 2012-08-06 00:00:00.000000000 Z
13
+ dependencies:
14
+ - !ruby/object:Gem::Dependency
15
+ name: rake
16
+ requirement: !ruby/object:Gem::Requirement
17
+ none: false
18
+ requirements:
19
+ - - ! '>='
20
+ - !ruby/object:Gem::Version
21
+ version: '0'
22
+ type: :development
23
+ prerelease: false
24
+ version_requirements: !ruby/object:Gem::Requirement
25
+ none: false
26
+ requirements:
27
+ - - ! '>='
28
+ - !ruby/object:Gem::Version
29
+ version: '0'
30
+ description: Adds simple stats like average, variance and standard deviation to Enumerable
31
+ email:
32
+ - nerab@gmx.net
33
+ executables: []
34
+ extensions: []
35
+ extra_rdoc_files: []
36
+ files:
37
+ - .gitignore
38
+ - Gemfile
39
+ - LICENSE
40
+ - README.md
41
+ - Rakefile
42
+ - enumstats.gemspec
43
+ - lib/enumstats.rb
44
+ - lib/enumstats/enumerable.rb
45
+ - lib/enumstats/version.rb
46
+ - test/helper.rb
47
+ - test/unit/test_enumerable_stats.rb
48
+ - test/unit/test_enumerable_stats_block.rb
49
+ homepage: ''
50
+ licenses: []
51
+ post_install_message:
52
+ rdoc_options: []
53
+ require_paths:
54
+ - lib
55
+ required_ruby_version: !ruby/object:Gem::Requirement
56
+ none: false
57
+ requirements:
58
+ - - ! '>='
59
+ - !ruby/object:Gem::Version
60
+ version: '0'
61
+ required_rubygems_version: !ruby/object:Gem::Requirement
62
+ none: false
63
+ requirements:
64
+ - - ! '>='
65
+ - !ruby/object:Gem::Version
66
+ version: '0'
67
+ requirements: []
68
+ rubyforge_project:
69
+ rubygems_version: 1.8.24
70
+ signing_key:
71
+ specification_version: 3
72
+ summary: Stats for Enumerables
73
+ test_files:
74
+ - test/helper.rb
75
+ - test/unit/test_enumerable_stats.rb
76
+ - test/unit/test_enumerable_stats_block.rb