quantity 0.1.1 → 0.1.2
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.
- data/.gemspec +36 -0
- data/.gitignore +10 -0
- data/.yardopts +13 -0
- data/AUTHORS +1 -0
- data/Gemfile +4 -0
- data/README +4 -5
- data/README.md +217 -0
- data/Rakefile +48 -0
- data/VERSION +1 -1
- data/doc/.gitignore +2 -0
- data/lib/quantity.rb +10 -0
- data/lib/quantity/dimension.rb +8 -1
- data/lib/quantity/rails.rb +12 -0
- data/lib/quantity/unit.rb +17 -5
- data/lib/quantity/version.rb +1 -1
- data/quantity.gemspec +42 -0
- data/spec/dimension.spec +330 -0
- data/spec/quantity.spec +282 -0
- data/spec/systems.spec +31 -0
- data/spec/unit.spec +284 -0
- metadata +94 -24
data/.gemspec
ADDED
@@ -0,0 +1,36 @@
|
|
1
|
+
#!/usr/bin/env ruby -rubygems
|
2
|
+
# -*- encoding: utf-8 -*-
|
3
|
+
|
4
|
+
GEMSPEC = Gem::Specification.new do |gem|
|
5
|
+
gem.version = File.read('VERSION').chomp
|
6
|
+
gem.date = File.mtime('VERSION').strftime('%Y-%m-%d')
|
7
|
+
|
8
|
+
gem.name = 'quantity'
|
9
|
+
gem.homepage = 'http://quantity.rubyforge.org/'
|
10
|
+
gem.license = 'Public Domain' if gem.respond_to?(:license=)
|
11
|
+
gem.summary = 'Units and quantities for Ruby.'
|
12
|
+
gem.description = <<-EOF
|
13
|
+
Quantity provides first-class quantities, units, and base quantities in pure ruby.
|
14
|
+
Things like 1.meter / 1.second == 1 meter/second.
|
15
|
+
EOF
|
16
|
+
gem.rubyforge_project = 'quantity'
|
17
|
+
|
18
|
+
gem.authors = ['Ben Lavender', 'Arto Bendiken']
|
19
|
+
gem.email = 'blavender@gmail.com'
|
20
|
+
|
21
|
+
gem.platform = Gem::Platform::RUBY
|
22
|
+
gem.files = %w(AUTHORS README UNLICENSE VERSION) + Dir.glob('lib/**/*.rb')
|
23
|
+
gem.bindir = %q(bin)
|
24
|
+
gem.executables = %w()
|
25
|
+
gem.default_executable = gem.executables.first
|
26
|
+
gem.require_paths = %w(lib)
|
27
|
+
gem.extensions = %w()
|
28
|
+
gem.test_files = %w()
|
29
|
+
gem.has_rdoc = false
|
30
|
+
|
31
|
+
gem.required_ruby_version = '>= 1.8.2'
|
32
|
+
gem.requirements = []
|
33
|
+
gem.add_development_dependency 'rspec', '>= 1.2.9'
|
34
|
+
gem.add_development_dependency 'yard' , '>= 0.5.2'
|
35
|
+
gem.post_install_message = nil
|
36
|
+
end
|
data/.gitignore
ADDED
data/.yardopts
ADDED
data/AUTHORS
CHANGED
data/Gemfile
ADDED
data/README
CHANGED
@@ -13,7 +13,7 @@ Hopefully this readme will be all you need, but [there are yardocs](http://quant
|
|
13
13
|
newton = 1.meter * 1.kilogram / 1.second**2 #=> 1 meter*kilogram/second^2
|
14
14
|
newton.to_feet #=> 3.28083989501312 foot*kilogram/second^2
|
15
15
|
newton.convert(:feet) #=> 3.28083989501312 foot*kilogram/second^2
|
16
|
-
jerk_newton / 1.second
|
16
|
+
jerk_newton = newton / 1.second #=> 1 meter*kilogram/second^3
|
17
17
|
jerk_newton * 1.second == newton #=> true
|
18
18
|
|
19
19
|
mmcubed = 1.mm.cubed #=> 1 millimeter^3
|
@@ -32,9 +32,8 @@ dimensions in Ruby. Some terminology:
|
|
32
32
|
* Dimension: Some base quantity to be measured, such as 'length'
|
33
33
|
|
34
34
|
Quantities perform complete mathematical operations over their units,
|
35
|
-
including `+`, `-`, `\*`, `/`,
|
36
|
-
and the dimensions they measure are fully represented and support
|
37
|
-
`\*` and `/`.
|
35
|
+
including `+`, `-`, `\*`, `/`, `**`, `%`, `abs`, `divmod`, `<=>`, and negation. Units
|
36
|
+
and the dimensions they measure are fully represented and support `*` and `/`.
|
38
37
|
|
39
38
|
Quantity extends Numeric to allow easy creation of quantities, but there
|
40
39
|
are direct interfaces to the library as well.
|
@@ -210,7 +209,7 @@ Authors
|
|
210
209
|
|
211
210
|
* [Ben Lavender](mailto:blavender@gmail.com) - <http://bhuga.net/>
|
212
211
|
* [Arto Bendiken](mailto:arto.bendiken@gmail.com) - <http://ar.to/>
|
213
|
-
|
212
|
+
* [Matt Todd](mtodd@highgroove.com) - <http://maraby.org/>
|
214
213
|
License
|
215
214
|
-------
|
216
215
|
|
data/README.md
ADDED
@@ -0,0 +1,217 @@
|
|
1
|
+
Quantity.rb: Units and Quantities for Ruby
|
2
|
+
==========================================
|
3
|
+
Quantity.rb provides first-class support for units and quantities in Ruby.
|
4
|
+
The right abstractions for true quantity representation and complex conversions.
|
5
|
+
Hopefully this readme will be all you need, but [there are yardocs](http://quantity.rubyforge.org)
|
6
|
+
|
7
|
+
## Overview
|
8
|
+
require 'quantity/all'
|
9
|
+
1.meter #=> 1 meter
|
10
|
+
1.meter.to_feet #=> 3.28083... foot
|
11
|
+
c = 299792458.meters / 1.second #=> 299792458 meter/second
|
12
|
+
|
13
|
+
newton = 1.meter * 1.kilogram / 1.second**2 #=> 1 meter*kilogram/second^2
|
14
|
+
newton.to_feet #=> 3.28083989501312 foot*kilogram/second^2
|
15
|
+
newton.convert(:feet) #=> 3.28083989501312 foot*kilogram/second^2
|
16
|
+
jerk_newton = newton / 1.second #=> 1 meter*kilogram/second^3
|
17
|
+
jerk_newton * 1.second == newton #=> true
|
18
|
+
|
19
|
+
mmcubed = 1.mm.cubed #=> 1 millimeter^3
|
20
|
+
mmcubed * 1000 == 1.milliliter #=> true
|
21
|
+
|
22
|
+
[1.meter, 1.foot, 1.inch].sort #=> [1 inch, 1 foot, 1 meter]
|
23
|
+
|
24
|
+
m_to_f = Quantity::Unit.for(:meter).convert_proc(:feet)
|
25
|
+
m_to_f.call(1) #=> 3.28083... (or a Rational)
|
26
|
+
|
27
|
+
Quantity.rb provides full-featured support for quantities, units, and
|
28
|
+
dimensions in Ruby. Some terminology:
|
29
|
+
|
30
|
+
* Quantity: An amount of a unit, such as 12 meters.
|
31
|
+
* Unit: An amount of a given dimension to be measured, such as 'meter'
|
32
|
+
* Dimension: Some base quantity to be measured, such as 'length'
|
33
|
+
|
34
|
+
Quantities perform complete mathematical operations over their units,
|
35
|
+
including `+`, `-`, `\*`, `/`, `**`, `%`, `abs`, `divmod`, `<=>`, and negation. Units
|
36
|
+
and the dimensions they measure are fully represented and support `*` and `/`.
|
37
|
+
|
38
|
+
Quantity extends Numeric to allow easy creation of quantities, but there
|
39
|
+
are direct interfaces to the library as well.
|
40
|
+
|
41
|
+
1.meter == Quantity.new(1,Quantity::Unit.for(:meter))
|
42
|
+
1.meter.unit == Quantity::Unit.for(:meter)
|
43
|
+
1.meter.unit.dimension == Quantity::Dimension.for(:length)
|
44
|
+
|
45
|
+
See the units section for supported units, and how to add your own.
|
46
|
+
|
47
|
+
Quantities are first-class citizens which do a fair job of imitating
|
48
|
+
Numeric. Quantities support coerce, and can thus be used in almost
|
49
|
+
any situation a numeric can:
|
50
|
+
|
51
|
+
2.5 + 5.meters # => 7.5 meters
|
52
|
+
5 == 5.meters # => true
|
53
|
+
|
54
|
+
## Status and TODO
|
55
|
+
Quantity.rb is not ready for production use for some areas, but should be
|
56
|
+
fine for simple conversion use cases. If it breaks, please email the
|
57
|
+
author for a full refund.
|
58
|
+
|
59
|
+
Specifically broken in this version are some operations on named
|
60
|
+
higher dimensions:
|
61
|
+
|
62
|
+
1.liter / 1.second #=> should be 1 liter/second, but explodes
|
63
|
+
1.liter.convert(:'mm^3') / 1.second #=> 1000000.0 millimeter^3/second
|
64
|
+
|
65
|
+
If you just work with units derived from the base dimensions, there aren't
|
66
|
+
known bugs. Please add a spec if you find one.
|
67
|
+
|
68
|
+
### TODO
|
69
|
+
* Lots more units are planned.
|
70
|
+
* BigDecimal support a la Rational.
|
71
|
+
* Supporting lambdas for unit values
|
72
|
+
* BigDecimal/Rational compatible values for existing units
|
73
|
+
* Some DSL sugar for adding derived dimension units
|
74
|
+
|
75
|
+
## Units
|
76
|
+
Quantity.rb comes with a sizable collection of units, but still needs significant expansion.
|
77
|
+
|
78
|
+
A number of base unit sets exist:
|
79
|
+
require 'quantity/all' #=> load everything. uses US versions of foot, lb, etc
|
80
|
+
require 'quantity/systems/si' #=> load SI
|
81
|
+
require 'quantity/systems/us' #=> load US versions of foot, lb, etc
|
82
|
+
require 'quantity/systems/imperial' #=> load British versions of foot, lb, etc
|
83
|
+
require 'quantity/systems/information' #=> bits, bytes, and all that
|
84
|
+
require 'quantity/systems/enumerable' #=> countable things--dozen, score, etc
|
85
|
+
|
86
|
+
Note that US and Imperial conflict with each other. Loading both is unsupported.
|
87
|
+
|
88
|
+
Adding your own units is simple:
|
89
|
+
|
90
|
+
Quantity::Unit.add_unit :furlong, :length, 201168, :furlongs
|
91
|
+
1.furlong #=> 1 furlong
|
92
|
+
|
93
|
+
201168 represents 1 furlong in millimeters. Each base dimension, such as length, time,
|
94
|
+
current, temperature, etc, is represented by a reference unit, which is generally the
|
95
|
+
milli-version of the SI unit referencing that domain. [NIST](http://physics.nist.gov/cuu/Units/units.html)
|
96
|
+
has an explanation of how the SI system works, and how all units are actually derived from
|
97
|
+
very few.
|
98
|
+
|
99
|
+
All units for derived dimensions used the derived reference unit. For example, length
|
100
|
+
is referenced to millimeters, so each unit of length is defined in terms of them:
|
101
|
+
|
102
|
+
Quantity::Unit.add_unit :meter, :length, 1000
|
103
|
+
Quantity::Unit.add_unit :millimeter, :length, 1, :mm
|
104
|
+
|
105
|
+
Thus, the base unit for volume is 1 mm^3:
|
106
|
+
volume = Quantity::Dimension.add_dimension length**3, :volume
|
107
|
+
ml = Quantity::Dimension.add_unit :milliliter, :volume, 1000, :ml, :milliliters
|
108
|
+
1.mm**3 * 1000 == 1.milliliter #=> true
|
109
|
+
|
110
|
+
See the bugs section for some current issues using units defined on derived dimensions.
|
111
|
+
|
112
|
+
The full list of included base dimensions and their reference units:
|
113
|
+
* :length => :millimeter
|
114
|
+
* :time => :millisecond
|
115
|
+
* :current => :milliampere
|
116
|
+
* :luminosity => :millicandela
|
117
|
+
* :substance => :millimole
|
118
|
+
* :temperature => :millikelvin
|
119
|
+
* :mass => :milligram
|
120
|
+
* :information => :bit # use :megabytes and :mebibytes
|
121
|
+
* :quantity => :item # for countable quantities. units include 1.dozen, for example
|
122
|
+
* :currency => :dollar # These are not really implemented yet
|
123
|
+
|
124
|
+
To determine the base unit for a derived dimension, you can use Quantity.rb itself:
|
125
|
+
|
126
|
+
force = Quantity::Dimension.for(:force)
|
127
|
+
newton = 1.meter * 1.kilogram / 1.second**2
|
128
|
+
newton.measures == force #=> true
|
129
|
+
newton_value = newton.to_mm.to_mg.to_ms #=> 1000.0 millimeter*milligram/millisecond^2
|
130
|
+
|
131
|
+
Thus, a newton would be 1000 when added specifically:
|
132
|
+
|
133
|
+
Quantity::Unit.add_unit :newton, :force, 1000, :newtons
|
134
|
+
1.newton == newton #=> true
|
135
|
+
|
136
|
+
## Dimensions
|
137
|
+
A dimension is a measurable thing, often called a 'base quantity' in scientific literature,
|
138
|
+
but Quantity.rb specifically avoids that nomenclature, reserving 'quantity' for the class
|
139
|
+
representing a unit and a value. As always, [wikipedia has the answers.](http://en.wikipedia.org/wiki/Physical_quantity)
|
140
|
+
|
141
|
+
Dimensions are not very useful by themselves, but you can play with them
|
142
|
+
if you want.
|
143
|
+
|
144
|
+
length = Quantity::Dimension.for(:length)
|
145
|
+
time = Quantity::Dimension.for(:time)
|
146
|
+
speed = length / time
|
147
|
+
|
148
|
+
A number of dimensions are enabled by default (see dimension/base.rb).
|
149
|
+
|
150
|
+
A DSL of sorts is provided for declaring dimensions:
|
151
|
+
|
152
|
+
length = Quantity::Dimension.add_dimenson :length
|
153
|
+
area = Quantity::Dimension.add_dimension length**2, :area
|
154
|
+
|
155
|
+
length = Quantity::Dimension.for(:length)
|
156
|
+
area = Quantity::Dimension.for(:area)
|
157
|
+
area == length * length #=> true
|
158
|
+
|
159
|
+
Quantity::Dimension is extended with empty subclasses for some base dimensions,
|
160
|
+
so you can do pattern patching on the class:
|
161
|
+
|
162
|
+
case 1.meter.measures
|
163
|
+
when Quantity::Dimension::Length
|
164
|
+
puts "I am printed"
|
165
|
+
end
|
166
|
+
|
167
|
+
## I just want to convert things, this is all just too much
|
168
|
+
Quantity.rb provides you the ability to intuitively create the conversions
|
169
|
+
your application needs, and then bypass the rest of the library.
|
170
|
+
|
171
|
+
m_to_f = 1.meter.measures.convert_proc(:feet)
|
172
|
+
m_to_f.call(5) # => 5 meters in feet
|
173
|
+
|
174
|
+
This Proc object has been broken down into a single division; it no longer references
|
175
|
+
any units, dimensions, or quantities. It's hard to be faster in pure Ruby.
|
176
|
+
|
177
|
+
### On precision and speed
|
178
|
+
|
179
|
+
By default, whatever Numeric you are using will be the stored value for the
|
180
|
+
quantity.
|
181
|
+
|
182
|
+
5.meters
|
183
|
+
Rational(5).meters
|
184
|
+
5.0.meters
|
185
|
+
|
186
|
+
This value will be held. However, divisions are required for conversions,
|
187
|
+
and the default is to force values into floats.
|
188
|
+
|
189
|
+
If accuracy is required, just require 'rational'. If Rational is defined,
|
190
|
+
you'll get rationals instead of divided floats everywhere. In tests, this
|
191
|
+
is an order of magnitude slower.
|
192
|
+
|
193
|
+
## 'Why' and previous work
|
194
|
+
This is by no means the first unit conversion/quantity library for Ruby, but
|
195
|
+
none of the existing ones scratched my itch just right. My goal is that this will
|
196
|
+
be the last one I (and you) need. The abstractions go all the way down, and
|
197
|
+
any conceivable conversion or munging functionality should be buildable on top
|
198
|
+
of this.
|
199
|
+
|
200
|
+
Inspiration comes from:
|
201
|
+
|
202
|
+
* [Quanty](http://narray.rubyforge.org/quanty/quanty-en.html)
|
203
|
+
Why oh why did they involve yacc?
|
204
|
+
* [Ruby Units](http://ruby-units.rubyforge.org/ruby-units/)
|
205
|
+
* [Alchemist](http://github.com/toastyapps/alchemist)
|
206
|
+
|
207
|
+
Authors
|
208
|
+
-------
|
209
|
+
|
210
|
+
* [Ben Lavender](mailto:blavender@gmail.com) - <http://bhuga.net/>
|
211
|
+
* [Arto Bendiken](mailto:arto.bendiken@gmail.com) - <http://ar.to/>
|
212
|
+
* [Matt Todd](mtodd@highgroove.com) - <http://maraby.org/>
|
213
|
+
License
|
214
|
+
-------
|
215
|
+
|
216
|
+
Quantity.rb is free and unencumbered public domain software. For more
|
217
|
+
information, see <http://unlicense.org/> or the accompanying UNLICENSE file.
|
data/Rakefile
ADDED
@@ -0,0 +1,48 @@
|
|
1
|
+
#!/usr/bin/env ruby
|
2
|
+
$:.unshift(File.expand_path(File.join(File.dirname(__FILE__), 'lib')))
|
3
|
+
require 'quantity'
|
4
|
+
|
5
|
+
require "rubygems"
|
6
|
+
require "bundler"
|
7
|
+
Bundler::GemHelper.install_tasks
|
8
|
+
|
9
|
+
begin
|
10
|
+
Bundler.setup(:default, :development)
|
11
|
+
rescue Bundler::BundlerError => e
|
12
|
+
$stderr.puts e.message
|
13
|
+
$stderr.puts "Run `bundle install` to install missing gems"
|
14
|
+
exit e.status_code
|
15
|
+
end
|
16
|
+
|
17
|
+
require 'spec'
|
18
|
+
require 'spec/rake/spectask'
|
19
|
+
require 'yard'
|
20
|
+
|
21
|
+
desc "Run specs"
|
22
|
+
Spec::Rake::SpecTask.new('spec') do |t|
|
23
|
+
t.spec_files = FileList['spec/quantity.spec']
|
24
|
+
t.spec_opts = ["-cfn"]
|
25
|
+
end
|
26
|
+
|
27
|
+
desc "Run unit specs"
|
28
|
+
Spec::Rake::SpecTask.new('unit') do |t|
|
29
|
+
t.spec_files = FileList['spec/dimension.spec', 'spec/unit.spec', 'spec/systems.spec']
|
30
|
+
t.spec_opts = ["-cfn"]
|
31
|
+
end
|
32
|
+
|
33
|
+
desc "specs with backtrace"
|
34
|
+
Spec::Rake::SpecTask.new('tracespec') do |t|
|
35
|
+
t.spec_files = FileList['spec/quantity.spec']
|
36
|
+
t.spec_opts = ["-bcfn"]
|
37
|
+
end
|
38
|
+
|
39
|
+
desc "unit specs with backtrace"
|
40
|
+
Spec::Rake::SpecTask.new('traceunit') do |t|
|
41
|
+
t.spec_files = FileList['spec/dimension.spec', 'spec/unit.spec', 'spec/systems.spec']
|
42
|
+
t.spec_opts = ["-bcfn"]
|
43
|
+
end
|
44
|
+
|
45
|
+
desc "package yardocs"
|
46
|
+
YARD::Rake::YardocTask.new('yard') do |t|
|
47
|
+
# see .yardopts for the action
|
48
|
+
end
|
data/VERSION
CHANGED
@@ -1 +1 @@
|
|
1
|
-
0.1.
|
1
|
+
0.1.2
|
data/doc/.gitignore
ADDED
data/lib/quantity.rb
CHANGED
data/lib/quantity/dimension.rb
CHANGED
@@ -120,6 +120,13 @@ class Quantity
|
|
120
120
|
@name.to_s
|
121
121
|
end
|
122
122
|
|
123
|
+
# The reduced form of a named complex unit
|
124
|
+
# @example
|
125
|
+
# Quantity::Dimension.for(:force).reduced_name # => :'mass*length/time^2'
|
126
|
+
def reduced_name
|
127
|
+
string_form.to_sym
|
128
|
+
end
|
129
|
+
|
123
130
|
# Dimensional multiplication
|
124
131
|
# @param [Dimension] other
|
125
132
|
# @return [Dimension]
|
@@ -178,7 +185,7 @@ class Quantity
|
|
178
185
|
#
|
179
186
|
# @return [String]
|
180
187
|
def inspect
|
181
|
-
sprintf("#<%s
|
188
|
+
sprintf("#<%s:0x%s @name=%s reduced_name=%s>", self.class.name, self.__id__.to_s(16), @name, reduced_name)
|
182
189
|
end
|
183
190
|
|
184
191
|
|
@@ -0,0 +1,12 @@
|
|
1
|
+
ActiveSupport::Duration.class_eval do
|
2
|
+
def quantity
|
3
|
+
Quantity.new(@value, Quantity::Unit.for(:second))
|
4
|
+
end
|
5
|
+
|
6
|
+
def method_missing_with_quantity(method, *args, &block) #:nodoc:
|
7
|
+
return quantity.send(method, *args, &block) if quantity.respond_to?(method)
|
8
|
+
|
9
|
+
method_missing_without_quantity(method, *args, &block)
|
10
|
+
end
|
11
|
+
alias_method_chain :method_missing, :quantity
|
12
|
+
end
|
data/lib/quantity/unit.rb
CHANGED
@@ -76,6 +76,12 @@ class Quantity
|
|
76
76
|
self.class_eval(&block)
|
77
77
|
end
|
78
78
|
|
79
|
+
# Reset the world. Useful in testing.
|
80
|
+
# @private
|
81
|
+
def self.__reset!
|
82
|
+
@@units = {}
|
83
|
+
end
|
84
|
+
|
79
85
|
### Instance-level methods/vars
|
80
86
|
attr_reader :name, :value, :dimension, :aliases
|
81
87
|
|
@@ -85,6 +91,11 @@ class Quantity
|
|
85
91
|
[@name] + @aliases
|
86
92
|
end
|
87
93
|
|
94
|
+
# A reduced form of this unit
|
95
|
+
def reduced_name
|
96
|
+
to_string_form.to_sym
|
97
|
+
end
|
98
|
+
|
88
99
|
# Can this unit be converted into the target unit?
|
89
100
|
# @param [Symbol String Unit]
|
90
101
|
# @return [Boolean]
|
@@ -150,7 +161,8 @@ class Quantity
|
|
150
161
|
end
|
151
162
|
|
152
163
|
def inspect
|
153
|
-
|
164
|
+
sprintf('#<%s:0x%s @name=%s @value=%s @dimension=%s>', self.class.name,
|
165
|
+
self.__id__.to_s(16), @name.inspect, @value.inspect, @dimension.inspect)
|
154
166
|
end
|
155
167
|
|
156
168
|
def <=>(other)
|
@@ -311,10 +323,10 @@ class Quantity
|
|
311
323
|
string
|
312
324
|
end
|
313
325
|
|
314
|
-
#
|
315
|
-
|
316
|
-
|
317
|
-
@@units = {}
|
326
|
+
# A reduced form of this unit
|
327
|
+
def reduced_name
|
328
|
+
string_form.to_sym
|
318
329
|
end
|
330
|
+
|
319
331
|
end
|
320
332
|
end
|