unitwise 2.0.0 → 2.1.0
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 +4 -4
- data/.ruby-version +1 -1
- data/.travis.yml +6 -10
- data/CHANGELOG.md +18 -0
- data/Gemfile +2 -2
- data/README.md +33 -3
- data/data/base_unit.yaml +2 -2
- data/data/derived_unit.yaml +60 -22
- data/data/prefix.yaml +1 -1
- data/lib/unitwise.rb +13 -1
- data/lib/unitwise/atom.rb +31 -2
- data/lib/unitwise/errors.rb +5 -2
- data/lib/unitwise/expression/decomposer.rb +21 -9
- data/lib/unitwise/expression/matcher.rb +3 -8
- data/lib/unitwise/expression/parser.rb +11 -4
- data/lib/unitwise/version.rb +1 -1
- data/test/test_helper.rb +1 -2
- data/test/unitwise/atom_test.rb +38 -2
- data/test/unitwise/expression/matcher_test.rb +3 -3
- data/test/unitwise_test.rb +21 -1
- metadata +3 -3
    
        checksums.yaml
    CHANGED
    
    | @@ -1,7 +1,7 @@ | |
| 1 1 | 
             
            ---
         | 
| 2 2 | 
             
            SHA1:
         | 
| 3 | 
            -
              metadata.gz:  | 
| 4 | 
            -
              data.tar.gz:  | 
| 3 | 
            +
              metadata.gz: fb8605b862b95bff79423e03fabf502e89a39440
         | 
| 4 | 
            +
              data.tar.gz: 83bf9041753023411557a9f9b22d84d6d61cecfc
         | 
| 5 5 | 
             
            SHA512:
         | 
| 6 | 
            -
              metadata.gz:  | 
| 7 | 
            -
              data.tar.gz:  | 
| 6 | 
            +
              metadata.gz: c202998427bdbd04aef777b310afbca4bbfd1866bc2d5405054864f28b17d9e723ea0885eba6bed4ee981b9078ed99c3ab10c6f0f98ebc6ef73fe5eac94713d6
         | 
| 7 | 
            +
              data.tar.gz: 806a06c5c4fcd80cd2ec3be0c6e723b2e7ce1b9b621340d9db194a782f08356594ea7816b8dec3d024245283404555559c5635b4c8fe05633a71aa0d043fd4e3
         | 
    
        data/.ruby-version
    CHANGED
    
    | @@ -1 +1 @@ | |
| 1 | 
            -
            2. | 
| 1 | 
            +
            2.4.0
         | 
    
        data/.travis.yml
    CHANGED
    
    | @@ -3,18 +3,14 @@ cache: bundler | |
| 3 3 | 
             
            bundler_args: --without yard pry
         | 
| 4 4 | 
             
            sudo: false
         | 
| 5 5 | 
             
            rvm:
         | 
| 6 | 
            -
              - 1. | 
| 7 | 
            -
              - 2. | 
| 8 | 
            -
              - 2. | 
| 9 | 
            -
              - 2. | 
| 6 | 
            +
              - 2.1.10
         | 
| 7 | 
            +
              - 2.2.7
         | 
| 8 | 
            +
              - 2.3.4
         | 
| 9 | 
            +
              - 2.4.1
         | 
| 10 | 
            +
              - jruby-9.1.8.0
         | 
| 10 11 | 
             
              - ruby-head
         | 
| 11 | 
            -
              -  | 
| 12 | 
            +
              - jruby-head
         | 
| 12 13 | 
             
            matrix:
         | 
| 13 | 
            -
              include:
         | 
| 14 | 
            -
                - rvm: jruby-19mode
         | 
| 15 | 
            -
                  env: JRUBY_OPTS="$JRUBY_OPTS --debug"
         | 
| 16 | 
            -
                - rvm: jruby-head
         | 
| 17 | 
            -
                  env: JRUBY_OPTS="$JRUBY_OPTS --debug"
         | 
| 18 14 | 
             
              allow_failures:
         | 
| 19 15 | 
             
                - rvm: ruby-head
         | 
| 20 16 | 
             
                - rvm: jruby-head
         | 
    
        data/CHANGELOG.md
    CHANGED
    
    | @@ -5,6 +5,24 @@ version 1.0.0. | |
| 5 5 |  | 
| 6 6 | 
             
            Unitwise uses semantic versioning.
         | 
| 7 7 |  | 
| 8 | 
            +
            ## Unreleased
         | 
| 9 | 
            +
             | 
| 10 | 
            +
            ## 2.1.0 - 2017-04-28
         | 
| 11 | 
            +
             | 
| 12 | 
            +
            ### Removed
         | 
| 13 | 
            +
             | 
| 14 | 
            +
            - Support for Ruby MRI 1.9.3, MRI 2.0, and Rubinius
         | 
| 15 | 
            +
             | 
| 16 | 
            +
            ### Added
         | 
| 17 | 
            +
             | 
| 18 | 
            +
            - `Unitwise.register` is a new method that allows user defined units
         | 
| 19 | 
            +
            - Support for MRI 2.3 and 2.4
         | 
| 20 | 
            +
             | 
| 21 | 
            +
            ### Changed
         | 
| 22 | 
            +
             | 
| 23 | 
            +
            - Unit data refreshed from latest UCUM spec. Most notably, some metric atoms
         | 
| 24 | 
            +
              names have seen case changes.
         | 
| 25 | 
            +
             | 
| 8 26 | 
             
            ## 2.0.0 - 2015-09-13
         | 
| 9 27 |  | 
| 10 28 | 
             
            ### Fixed
         | 
    
        data/Gemfile
    CHANGED
    
    | @@ -2,9 +2,9 @@ source 'https://rubygems.org' | |
| 2 2 |  | 
| 3 3 | 
             
            gem 'coveralls',  '~> 0.7'
         | 
| 4 4 | 
             
            if RUBY_VERSION >= '2.2.0'
         | 
| 5 | 
            -
              gem 'bigdecimal', ' | 
| 5 | 
            +
              gem 'bigdecimal', '>= 1.2.6', :platform => :mri
         | 
| 6 6 | 
             
            else
         | 
| 7 | 
            -
              gem 'bigdecimal', ' | 
| 7 | 
            +
              gem 'bigdecimal', '<= 1.2.5', :platform => :mri
         | 
| 8 8 | 
             
            end
         | 
| 9 9 |  | 
| 10 10 | 
             
            gemspec
         | 
    
        data/README.md
    CHANGED
    
    | @@ -237,17 +237,47 @@ Unitwise(1, "meter/s")      # Does not work, mixed designations (name and primar | |
| 237 237 | 
             
            Unitwise(1, "meter") / Unitwise(1, "s") # Also works
         | 
| 238 238 | 
             
            ```
         | 
| 239 239 |  | 
| 240 | 
            +
             | 
| 241 | 
            +
            ### Adding custom units
         | 
| 242 | 
            +
             | 
| 243 | 
            +
            While UCUM's list of units is rather exhaustive, there may still be occasions
         | 
| 244 | 
            +
            where you need custom or uncommon measurements. You can add them yourself
         | 
| 245 | 
            +
            with `Unitwise.register`, which will allow you to convert to or from the new
         | 
| 246 | 
            +
            unit.
         | 
| 247 | 
            +
             | 
| 248 | 
            +
            For example, if your app needed to pour "3 fingers" of bourbon, you could
         | 
| 249 | 
            +
            register an atom for that:
         | 
| 250 | 
            +
             | 
| 251 | 
            +
            ```ruby
         | 
| 252 | 
            +
            Unitwise.register(
         | 
| 253 | 
            +
              names: ["finger", "fingers"],
         | 
| 254 | 
            +
              symbol: "🥃",
         | 
| 255 | 
            +
              primary_code: "fng",
         | 
| 256 | 
            +
              secondary_code: "fng",
         | 
| 257 | 
            +
              scale: {
         | 
| 258 | 
            +
                value: 1.0,
         | 
| 259 | 
            +
                unit_code: '[foz_us]'
         | 
| 260 | 
            +
              },
         | 
| 261 | 
            +
              property: 'fluid volume'
         | 
| 262 | 
            +
            )
         | 
| 263 | 
            +
             | 
| 264 | 
            +
            Unitwise(1, "gallon").to_fingers
         | 
| 265 | 
            +
            # => #<Unitwise::Measurement value=0.153721590464621430998E3 unit=fingers>
         | 
| 266 | 
            +
             | 
| 267 | 
            +
            Unitwise(1, "🥃").to_cup
         | 
| 268 | 
            +
            # => #<Unitwise::Measurement value=0.125 unit=cup>
         | 
| 269 | 
            +
            ```
         | 
| 270 | 
            +
             | 
| 240 271 | 
             
            ## Supported Ruby Versions
         | 
| 241 272 |  | 
| 242 273 | 
             
            This library aims to support and is tested against the following Ruby
         | 
| 243 274 | 
             
            implementations:
         | 
| 244 275 |  | 
| 245 | 
            -
            * Ruby 1.9.3
         | 
| 246 | 
            -
            * Ruby 2.0.0
         | 
| 247 276 | 
             
            * Ruby 2.1
         | 
| 248 277 | 
             
            * Ruby 2.2
         | 
| 278 | 
            +
            * Ruby 2.3
         | 
| 279 | 
            +
            * Ruby 2.4
         | 
| 249 280 | 
             
            * [JRuby](http://jruby.org/)
         | 
| 250 | 
            -
            * [Rubinius](http://rubini.us/)
         | 
| 251 281 |  | 
| 252 282 | 
             
            If something doesn't work on one of these versions, it's a bug.
         | 
| 253 283 |  | 
    
        data/data/base_unit.yaml
    CHANGED
    
    | @@ -23,13 +23,13 @@ | |
| 23 23 | 
             
              :secondary_code: RAD
         | 
| 24 24 | 
             
              :property: plane angle
         | 
| 25 25 | 
             
              :dim: A
         | 
| 26 | 
            -
            - :names:  | 
| 26 | 
            +
            - :names: kelvin
         | 
| 27 27 | 
             
              :symbol: K
         | 
| 28 28 | 
             
              :primary_code: K
         | 
| 29 29 | 
             
              :secondary_code: K
         | 
| 30 30 | 
             
              :property: temperature
         | 
| 31 31 | 
             
              :dim: C
         | 
| 32 | 
            -
            - :names:  | 
| 32 | 
            +
            - :names: coulomb
         | 
| 33 33 | 
             
              :symbol: C
         | 
| 34 34 | 
             
              :primary_code: C
         | 
| 35 35 | 
             
              :secondary_code: C
         | 
    
        data/data/derived_unit.yaml
    CHANGED
    
    | @@ -24,7 +24,7 @@ | |
| 24 24 | 
             
              :special: false
         | 
| 25 25 | 
             
              :arbitrary: false
         | 
| 26 26 | 
             
            - :names: the number pi
         | 
| 27 | 
            -
              :symbol:  | 
| 27 | 
            +
              :symbol: π
         | 
| 28 28 | 
             
              :primary_code: "[pi]"
         | 
| 29 29 | 
             
              :secondary_code: "[PI]"
         | 
| 30 30 | 
             
              :scale:
         | 
| @@ -119,7 +119,7 @@ | |
| 119 119 | 
             
              :metric: true
         | 
| 120 120 | 
             
              :special: false
         | 
| 121 121 | 
             
              :arbitrary: false
         | 
| 122 | 
            -
            - :names:  | 
| 122 | 
            +
            - :names: hertz
         | 
| 123 123 | 
             
              :symbol: Hz
         | 
| 124 124 | 
             
              :primary_code: Hz
         | 
| 125 125 | 
             
              :secondary_code: HZ
         | 
| @@ -131,7 +131,7 @@ | |
| 131 131 | 
             
              :metric: true
         | 
| 132 132 | 
             
              :special: false
         | 
| 133 133 | 
             
              :arbitrary: false
         | 
| 134 | 
            -
            - :names:  | 
| 134 | 
            +
            - :names: newton
         | 
| 135 135 | 
             
              :symbol: N
         | 
| 136 136 | 
             
              :primary_code: N
         | 
| 137 137 | 
             
              :secondary_code: N
         | 
| @@ -143,7 +143,7 @@ | |
| 143 143 | 
             
              :metric: true
         | 
| 144 144 | 
             
              :special: false
         | 
| 145 145 | 
             
              :arbitrary: false
         | 
| 146 | 
            -
            - :names:  | 
| 146 | 
            +
            - :names: pascal
         | 
| 147 147 | 
             
              :symbol: Pa
         | 
| 148 148 | 
             
              :primary_code: Pa
         | 
| 149 149 | 
             
              :secondary_code: PAL
         | 
| @@ -155,7 +155,7 @@ | |
| 155 155 | 
             
              :metric: true
         | 
| 156 156 | 
             
              :special: false
         | 
| 157 157 | 
             
              :arbitrary: false
         | 
| 158 | 
            -
            - :names:  | 
| 158 | 
            +
            - :names: joule
         | 
| 159 159 | 
             
              :symbol: J
         | 
| 160 160 | 
             
              :primary_code: J
         | 
| 161 161 | 
             
              :secondary_code: J
         | 
| @@ -167,7 +167,7 @@ | |
| 167 167 | 
             
              :metric: true
         | 
| 168 168 | 
             
              :special: false
         | 
| 169 169 | 
             
              :arbitrary: false
         | 
| 170 | 
            -
            - :names:  | 
| 170 | 
            +
            - :names: watt
         | 
| 171 171 | 
             
              :symbol: W
         | 
| 172 172 | 
             
              :primary_code: W
         | 
| 173 173 | 
             
              :secondary_code: W
         | 
| @@ -179,7 +179,7 @@ | |
| 179 179 | 
             
              :metric: true
         | 
| 180 180 | 
             
              :special: false
         | 
| 181 181 | 
             
              :arbitrary: false
         | 
| 182 | 
            -
            - :names:  | 
| 182 | 
            +
            - :names: ampère
         | 
| 183 183 | 
             
              :symbol: A
         | 
| 184 184 | 
             
              :primary_code: A
         | 
| 185 185 | 
             
              :secondary_code: A
         | 
| @@ -191,7 +191,7 @@ | |
| 191 191 | 
             
              :metric: true
         | 
| 192 192 | 
             
              :special: false
         | 
| 193 193 | 
             
              :arbitrary: false
         | 
| 194 | 
            -
            - :names:  | 
| 194 | 
            +
            - :names: volt
         | 
| 195 195 | 
             
              :symbol: V
         | 
| 196 196 | 
             
              :primary_code: V
         | 
| 197 197 | 
             
              :secondary_code: V
         | 
| @@ -203,7 +203,7 @@ | |
| 203 203 | 
             
              :metric: true
         | 
| 204 204 | 
             
              :special: false
         | 
| 205 205 | 
             
              :arbitrary: false
         | 
| 206 | 
            -
            - :names:  | 
| 206 | 
            +
            - :names: farad
         | 
| 207 207 | 
             
              :symbol: F
         | 
| 208 208 | 
             
              :primary_code: F
         | 
| 209 209 | 
             
              :secondary_code: F
         | 
| @@ -215,8 +215,8 @@ | |
| 215 215 | 
             
              :metric: true
         | 
| 216 216 | 
             
              :special: false
         | 
| 217 217 | 
             
              :arbitrary: false
         | 
| 218 | 
            -
            - :names:  | 
| 219 | 
            -
              :symbol:  | 
| 218 | 
            +
            - :names: ohm
         | 
| 219 | 
            +
              :symbol: Ω
         | 
| 220 220 | 
             
              :primary_code: Ohm
         | 
| 221 221 | 
             
              :secondary_code: OHM
         | 
| 222 222 | 
             
              :scale:
         | 
| @@ -227,7 +227,7 @@ | |
| 227 227 | 
             
              :metric: true
         | 
| 228 228 | 
             
              :special: false
         | 
| 229 229 | 
             
              :arbitrary: false
         | 
| 230 | 
            -
            - :names:  | 
| 230 | 
            +
            - :names: siemens
         | 
| 231 231 | 
             
              :symbol: S
         | 
| 232 232 | 
             
              :primary_code: S
         | 
| 233 233 | 
             
              :secondary_code: SIE
         | 
| @@ -239,7 +239,7 @@ | |
| 239 239 | 
             
              :metric: true
         | 
| 240 240 | 
             
              :special: false
         | 
| 241 241 | 
             
              :arbitrary: false
         | 
| 242 | 
            -
            - :names:  | 
| 242 | 
            +
            - :names: weber
         | 
| 243 243 | 
             
              :symbol: Wb
         | 
| 244 244 | 
             
              :primary_code: Wb
         | 
| 245 245 | 
             
              :secondary_code: WB
         | 
| @@ -264,7 +264,7 @@ | |
| 264 264 | 
             
              :metric: true
         | 
| 265 265 | 
             
              :special: true
         | 
| 266 266 | 
             
              :arbitrary: false
         | 
| 267 | 
            -
            - :names:  | 
| 267 | 
            +
            - :names: tesla
         | 
| 268 268 | 
             
              :symbol: T
         | 
| 269 269 | 
             
              :primary_code: T
         | 
| 270 270 | 
             
              :secondary_code: T
         | 
| @@ -276,7 +276,7 @@ | |
| 276 276 | 
             
              :metric: true
         | 
| 277 277 | 
             
              :special: false
         | 
| 278 278 | 
             
              :arbitrary: false
         | 
| 279 | 
            -
            - :names:  | 
| 279 | 
            +
            - :names: henry
         | 
| 280 280 | 
             
              :symbol: H
         | 
| 281 281 | 
             
              :primary_code: H
         | 
| 282 282 | 
             
              :secondary_code: H
         | 
| @@ -312,7 +312,7 @@ | |
| 312 312 | 
             
              :metric: true
         | 
| 313 313 | 
             
              :special: false
         | 
| 314 314 | 
             
              :arbitrary: false
         | 
| 315 | 
            -
            - :names:  | 
| 315 | 
            +
            - :names: becquerel
         | 
| 316 316 | 
             
              :symbol: Bq
         | 
| 317 317 | 
             
              :primary_code: Bq
         | 
| 318 318 | 
             
              :secondary_code: BQ
         | 
| @@ -324,7 +324,7 @@ | |
| 324 324 | 
             
              :metric: true
         | 
| 325 325 | 
             
              :special: false
         | 
| 326 326 | 
             
              :arbitrary: false
         | 
| 327 | 
            -
            - :names:  | 
| 327 | 
            +
            - :names: gray
         | 
| 328 328 | 
             
              :symbol: Gy
         | 
| 329 329 | 
             
              :primary_code: Gy
         | 
| 330 330 | 
             
              :secondary_code: GY
         | 
| @@ -336,7 +336,7 @@ | |
| 336 336 | 
             
              :metric: true
         | 
| 337 337 | 
             
              :special: false
         | 
| 338 338 | 
             
              :arbitrary: false
         | 
| 339 | 
            -
            - :names:  | 
| 339 | 
            +
            - :names: sievert
         | 
| 340 340 | 
             
              :symbol: Sv
         | 
| 341 341 | 
             
              :primary_code: Sv
         | 
| 342 342 | 
             
              :secondary_code: SV
         | 
| @@ -666,7 +666,7 @@ | |
| 666 666 | 
             
              :primary_code: "[h]"
         | 
| 667 667 | 
             
              :secondary_code: "[H]"
         | 
| 668 668 | 
             
              :scale:
         | 
| 669 | 
            -
                :value: 6.6260755e- | 
| 669 | 
            +
                :value: 6.6260755e-34
         | 
| 670 670 | 
             
                :unit_code: J.s
         | 
| 671 671 | 
             
              :classification: const
         | 
| 672 672 | 
             
              :property: action
         | 
| @@ -1069,7 +1069,7 @@ | |
| 1069 1069 | 
             
              :metric: false
         | 
| 1070 1070 | 
             
              :special: false
         | 
| 1071 1071 | 
             
              :arbitrary: false
         | 
| 1072 | 
            -
            - :names:  | 
| 1072 | 
            +
            - :names: mile
         | 
| 1073 1073 | 
             
              :symbol: mi
         | 
| 1074 1074 | 
             
              :primary_code: "[mi_i]"
         | 
| 1075 1075 | 
             
              :secondary_code: "[MI_I]"
         | 
| @@ -2244,6 +2244,19 @@ | |
| 2244 2244 | 
             
              :metric: false
         | 
| 2245 2245 | 
             
              :special: false
         | 
| 2246 2246 | 
             
              :arbitrary: false
         | 
| 2247 | 
            +
            - :names: degree Réaumur
         | 
| 2248 | 
            +
              :symbol: "°Ré"
         | 
| 2249 | 
            +
              :primary_code: "[degRe]"
         | 
| 2250 | 
            +
              :secondary_code: "[degRe]"
         | 
| 2251 | 
            +
              :scale:
         | 
| 2252 | 
            +
                :function_code: degre
         | 
| 2253 | 
            +
                :value: 5.0
         | 
| 2254 | 
            +
                :unit_code: K/4
         | 
| 2255 | 
            +
              :classification: heat
         | 
| 2256 | 
            +
              :property: temperature
         | 
| 2257 | 
            +
              :metric: false
         | 
| 2258 | 
            +
              :special: true
         | 
| 2259 | 
            +
              :arbitrary: false
         | 
| 2247 2260 | 
             
            - :names: calorie at 15 °C
         | 
| 2248 2261 | 
             
              :symbol: cal<sub>15°C</sub>
         | 
| 2249 2262 | 
             
              :primary_code: cal_[15]
         | 
| @@ -3138,6 +3151,19 @@ | |
| 3138 3151 | 
             
              :metric: false
         | 
| 3139 3152 | 
             
              :special: false
         | 
| 3140 3153 | 
             
              :arbitrary: true
         | 
| 3154 | 
            +
            - :names: index of reactivity
         | 
| 3155 | 
            +
              :symbol: IR
         | 
| 3156 | 
            +
              :primary_code: "[IR]"
         | 
| 3157 | 
            +
              :secondary_code: "[IR]"
         | 
| 3158 | 
            +
              :scale:
         | 
| 3159 | 
            +
                :value: 1.0
         | 
| 3160 | 
            +
                :unit_code: '1'
         | 
| 3161 | 
            +
              :classification: chemical
         | 
| 3162 | 
            +
              :property: amount of an allergen callibrated through in-vivo testing using the Stallergenes®
         | 
| 3163 | 
            +
                method.
         | 
| 3164 | 
            +
              :metric: false
         | 
| 3165 | 
            +
              :special: false
         | 
| 3166 | 
            +
              :arbitrary: true
         | 
| 3141 3167 | 
             
            - :names: bioequivalent allergen unit
         | 
| 3142 3168 | 
             
              :symbol: BAU
         | 
| 3143 3169 | 
             
              :primary_code: "[BAU]"
         | 
| @@ -3372,8 +3398,8 @@ | |
| 3372 3398 | 
             
              :metric: true
         | 
| 3373 3399 | 
             
              :special: false
         | 
| 3374 3400 | 
             
              :arbitrary: false
         | 
| 3375 | 
            -
            - :names:  | 
| 3376 | 
            -
              :symbol:  | 
| 3401 | 
            +
            - :names: Ångström
         | 
| 3402 | 
            +
              :symbol: Å
         | 
| 3377 3403 | 
             
              :primary_code: Ao
         | 
| 3378 3404 | 
             
              :secondary_code: AO
         | 
| 3379 3405 | 
             
              :scale:
         | 
| @@ -3491,6 +3517,18 @@ | |
| 3491 3517 | 
             
              :metric: false
         | 
| 3492 3518 | 
             
              :special: false
         | 
| 3493 3519 | 
             
              :arbitrary: false
         | 
| 3520 | 
            +
            - :names: meter per square seconds per square root of hertz
         | 
| 3521 | 
            +
              :primary_code: "[m/s2/Hz^(1/2)]"
         | 
| 3522 | 
            +
              :secondary_code: "[M/S2/HZ^(1/2)]"
         | 
| 3523 | 
            +
              :scale:
         | 
| 3524 | 
            +
                :function_code: sqrt
         | 
| 3525 | 
            +
                :value: 1.0
         | 
| 3526 | 
            +
                :unit_code: m2/s4/Hz
         | 
| 3527 | 
            +
              :classification: misc
         | 
| 3528 | 
            +
              :property: amplitude spectral density
         | 
| 3529 | 
            +
              :metric: false
         | 
| 3530 | 
            +
              :special: true
         | 
| 3531 | 
            +
              :arbitrary: false
         | 
| 3494 3532 | 
             
            - :names: bit
         | 
| 3495 3533 | 
             
              :symbol: bit<sub>s</sub>
         | 
| 3496 3534 | 
             
              :primary_code: bit_s
         | 
    
        data/data/prefix.yaml
    CHANGED
    
    
    
        data/lib/unitwise.rb
    CHANGED
    
    | @@ -43,7 +43,19 @@ module Unitwise | |
| 43 43 | 
             
                  false
         | 
| 44 44 | 
             
                end
         | 
| 45 45 | 
             
              end
         | 
| 46 | 
            -
             | 
| 46 | 
            +
             | 
| 47 | 
            +
              # Add additional atoms. Useful for registering uncommon or custom units.
         | 
| 48 | 
            +
              # @param properties [Hash] Properties of the atom
         | 
| 49 | 
            +
              # @return [Unitwise::Atom] The newly created atom
         | 
| 50 | 
            +
              # @raise [Unitwise::DefinitionError]
         | 
| 51 | 
            +
              def self.register(atom_hash)
         | 
| 52 | 
            +
                atom = Unitwise::Atom.new(atom_hash)
         | 
| 53 | 
            +
                atom.validate!
         | 
| 54 | 
            +
                Unitwise::Atom.all.push(atom)
         | 
| 55 | 
            +
                Unitwise::Expression::Decomposer.send(:reset)
         | 
| 56 | 
            +
                atom
         | 
| 57 | 
            +
              end
         | 
| 58 | 
            +
             | 
| 47 59 | 
             
              # The system path for the installed gem
         | 
| 48 60 | 
             
              # @api private
         | 
| 49 61 | 
             
              def self.path
         | 
    
        data/lib/unitwise/atom.rb
    CHANGED
    
    | @@ -7,13 +7,13 @@ module Unitwise | |
| 7 7 | 
             
                include Compatible
         | 
| 8 8 |  | 
| 9 9 | 
             
                class << self
         | 
| 10 | 
            -
                  # Array of hashes representing atom properties.
         | 
| 10 | 
            +
                  # Array of hashes representing default atom properties.
         | 
| 11 11 | 
             
                  # @api private
         | 
| 12 12 | 
             
                  def data
         | 
| 13 13 | 
             
                    @data ||= data_files.map { |file| YAML.load(File.open file) }.flatten
         | 
| 14 14 | 
             
                  end
         | 
| 15 15 |  | 
| 16 | 
            -
                  # Data files containing atom data
         | 
| 16 | 
            +
                  # Data files containing default atom data
         | 
| 17 17 | 
             
                  # @api private
         | 
| 18 18 | 
             
                  def data_files
         | 
| 19 19 | 
             
                    %w(base_unit derived_unit).map { |type| Unitwise.data_file type }
         | 
| @@ -117,5 +117,34 @@ module Unitwise | |
| 117 117 | 
             
                  base? ? [Term.new(:atom_code => primary_code)] : scale.root_terms
         | 
| 118 118 | 
             
                end
         | 
| 119 119 | 
             
                memoize :root_terms
         | 
| 120 | 
            +
             | 
| 121 | 
            +
             | 
| 122 | 
            +
                # A basic validator for atoms. It checks for the bare minimum properties
         | 
| 123 | 
            +
                # and that it's scalar and magnitude can be resolved. Note that this method
         | 
| 124 | 
            +
                # requires the units it depends on to already exist, so it is not used
         | 
| 125 | 
            +
                # when loading the initial data from UCUM.
         | 
| 126 | 
            +
                # @return [true] returns true if the atom is valid
         | 
| 127 | 
            +
                # @raise [Unitwise::DefinitionError]
         | 
| 128 | 
            +
                def validate!
         | 
| 129 | 
            +
                  missing_properties = %i{primary_code names scale}.select do |prop|
         | 
| 130 | 
            +
                    val = liner_get(prop)
         | 
| 131 | 
            +
                    val.nil? || (val.respond_to?(:empty) && val.empty?)
         | 
| 132 | 
            +
                  end
         | 
| 133 | 
            +
             | 
| 134 | 
            +
                  if !missing_properties.empty?
         | 
| 135 | 
            +
                    missing_list = missing_properties.join(',')
         | 
| 136 | 
            +
                    raise Unitwise::DefinitionError,
         | 
| 137 | 
            +
                      "Atom has missing properties: #{missing_list}."
         | 
| 138 | 
            +
                  end
         | 
| 139 | 
            +
             | 
| 140 | 
            +
                  msg = "Atom definition could not be resolved. Ensure that it is a base " \
         | 
| 141 | 
            +
                        "unit or is defined relative to existing units."
         | 
| 142 | 
            +
             | 
| 143 | 
            +
                  begin
         | 
| 144 | 
            +
                    !scalar.nil? && !magnitude.nil? || raise(Unitwise::DefinitionError, msg)
         | 
| 145 | 
            +
                  rescue Unitwise::ExpressionError
         | 
| 146 | 
            +
                    raise Unitwise::DefinitionError,  msg
         | 
| 147 | 
            +
                  end
         | 
| 148 | 
            +
                end
         | 
| 120 149 | 
             
              end
         | 
| 121 150 | 
             
            end
         | 
    
        data/lib/unitwise/errors.rb
    CHANGED
    
    
| @@ -5,13 +5,7 @@ module Unitwise | |
| 5 5 | 
             
                # of a string, as well as caching the results.
         | 
| 6 6 | 
             
                class Decomposer
         | 
| 7 7 |  | 
| 8 | 
            -
                  MODES = [:primary_code, :secondary_code, :names, :slugs, :symbol]
         | 
| 9 | 
            -
             | 
| 10 | 
            -
                  PARSERS = MODES.reduce({}) do |hash, mode|
         | 
| 11 | 
            -
                    hash[mode] = Parser.new(mode); hash
         | 
| 12 | 
            -
                  end
         | 
| 13 | 
            -
             | 
| 14 | 
            -
                  TRANSFORMER = Transformer.new
         | 
| 8 | 
            +
                  MODES = [:primary_code, :secondary_code, :names, :slugs, :symbol].freeze
         | 
| 15 9 |  | 
| 16 10 | 
             
                  class << self
         | 
| 17 11 |  | 
| @@ -25,6 +19,16 @@ module Unitwise | |
| 25 19 | 
             
                      end
         | 
| 26 20 | 
             
                    end
         | 
| 27 21 |  | 
| 22 | 
            +
                    def parsers
         | 
| 23 | 
            +
                      @parsers ||= MODES.reduce({}) do |hash, mode|
         | 
| 24 | 
            +
                        hash[mode] = Parser.new(mode); hash
         | 
| 25 | 
            +
                      end
         | 
| 26 | 
            +
                    end
         | 
| 27 | 
            +
             | 
| 28 | 
            +
                    def transformer
         | 
| 29 | 
            +
                      @transformer = Transformer.new
         | 
| 30 | 
            +
                    end
         | 
| 31 | 
            +
             | 
| 28 32 | 
             
                    private
         | 
| 29 33 |  | 
| 30 34 | 
             
                    # A simple cache to prevent re-decomposing the same units
         | 
| @@ -32,6 +36,14 @@ module Unitwise | |
| 32 36 | 
             
                    def cache
         | 
| 33 37 | 
             
                      @cache ||= {}
         | 
| 34 38 | 
             
                    end
         | 
| 39 | 
            +
             | 
| 40 | 
            +
                    # Reset memoized data. Allows rebuilding of parsers, transformers, and
         | 
| 41 | 
            +
                    # the cache after list of atoms has been modified.
         | 
| 42 | 
            +
                    def reset
         | 
| 43 | 
            +
                      @parsers = nil
         | 
| 44 | 
            +
                      @transformer = nil
         | 
| 45 | 
            +
                      @cache = nil
         | 
| 46 | 
            +
                    end
         | 
| 35 47 | 
             
                  end
         | 
| 36 48 |  | 
| 37 49 | 
             
                  attr_reader :expression, :mode
         | 
| @@ -44,7 +56,7 @@ module Unitwise | |
| 44 56 | 
             
                  end
         | 
| 45 57 |  | 
| 46 58 | 
             
                  def parse
         | 
| 47 | 
            -
                     | 
| 59 | 
            +
                    self.class.parsers.reduce(nil) do |_, (mode, parser)|
         | 
| 48 60 | 
             
                      parsed = parser.parse(expression) rescue next
         | 
| 49 61 | 
             
                      @mode = mode
         | 
| 50 62 | 
             
                      break parsed
         | 
| @@ -52,7 +64,7 @@ module Unitwise | |
| 52 64 | 
             
                  end
         | 
| 53 65 |  | 
| 54 66 | 
             
                  def transform
         | 
| 55 | 
            -
                    @transform ||=  | 
| 67 | 
            +
                    @transform ||= self.class.transformer.apply(parse, :mode => mode)
         | 
| 56 68 | 
             
                  end
         | 
| 57 69 |  | 
| 58 70 | 
             
                  def terms
         | 
| @@ -5,19 +5,15 @@ module Unitwise | |
| 5 5 | 
             
                class Matcher
         | 
| 6 6 | 
             
                  class << self
         | 
| 7 7 | 
             
                    def atom(mode)
         | 
| 8 | 
            -
                       | 
| 9 | 
            -
                      @atom[mode] ||= new(Atom.all, mode).alternative
         | 
| 8 | 
            +
                      new(Atom.all, mode).alternative
         | 
| 10 9 | 
             
                    end
         | 
| 11 10 |  | 
| 12 11 | 
             
                    def metric_atom(mode)
         | 
| 13 | 
            -
                       | 
| 14 | 
            -
                      @metric_atom[mode] ||=
         | 
| 15 | 
            -
                        new(Atom.all.select(&:metric?), mode).alternative
         | 
| 12 | 
            +
                      new(Atom.all.select(&:metric?), mode).alternative
         | 
| 16 13 | 
             
                    end
         | 
| 17 14 |  | 
| 18 15 | 
             
                    def prefix(mode)
         | 
| 19 | 
            -
                       | 
| 20 | 
            -
                      @prefix[mode] ||= new(Prefix.all, mode).alternative
         | 
| 16 | 
            +
                      new(Prefix.all, mode).alternative
         | 
| 21 17 | 
             
                    end
         | 
| 22 18 | 
             
                  end
         | 
| 23 19 |  | 
| @@ -41,7 +37,6 @@ module Unitwise | |
| 41 37 | 
             
                  def alternative
         | 
| 42 38 | 
             
                    Parslet::Atoms::Alternative.new(*matchers)
         | 
| 43 39 | 
             
                  end
         | 
| 44 | 
            -
             | 
| 45 40 | 
             
                end
         | 
| 46 41 | 
             
              end
         | 
| 47 42 | 
             
            end
         | 
| @@ -5,14 +5,21 @@ module Unitwise | |
| 5 5 | 
             
                class Parser < Parslet::Parser
         | 
| 6 6 | 
             
                  attr_reader :key
         | 
| 7 7 | 
             
                  def initialize(key = :primary_code)
         | 
| 8 | 
            -
                    @key | 
| 8 | 
            +
                    @key                 = key
         | 
| 9 | 
            +
                    @atom_matcher        = Matcher.atom(key)
         | 
| 10 | 
            +
                    @metric_atom_matcher = Matcher.metric_atom(key)
         | 
| 11 | 
            +
                    @prefix_matcher      = Matcher.prefix(key)
         | 
| 9 12 | 
             
                  end
         | 
| 10 13 |  | 
| 14 | 
            +
                  private
         | 
| 15 | 
            +
             | 
| 16 | 
            +
                  attr_reader :atom_matcher, :metric_atom_matcher, :prefix_matcher
         | 
| 17 | 
            +
             | 
| 11 18 | 
             
                  root :expression
         | 
| 12 19 |  | 
| 13 | 
            -
                  rule (:atom) {  | 
| 14 | 
            -
                  rule (:metric_atom) {  | 
| 15 | 
            -
                  rule (:prefix) {  | 
| 20 | 
            +
                  rule (:atom) { atom_matcher.as(:atom_code) }
         | 
| 21 | 
            +
                  rule (:metric_atom) { metric_atom_matcher.as(:atom_code) }
         | 
| 22 | 
            +
                  rule (:prefix) { prefix_matcher.as(:prefix_code) }
         | 
| 16 23 |  | 
| 17 24 | 
             
                  rule (:simpleton) do
         | 
| 18 25 | 
             
                    (prefix.as(:prefix) >> metric_atom.as(:atom) | atom.as(:atom))
         | 
    
        data/lib/unitwise/version.rb
    CHANGED
    
    
    
        data/test/test_helper.rb
    CHANGED
    
    | @@ -9,9 +9,8 @@ require 'minitest/pride' | |
| 9 9 | 
             
            require 'unitwise'
         | 
| 10 10 |  | 
| 11 11 | 
             
            module Minitest::Assertions
         | 
| 12 | 
            -
              def assert_almost_equal(expected, actual)
         | 
| 12 | 
            +
              def assert_almost_equal(expected, actual, range=0.0001)
         | 
| 13 13 | 
             
                message = "Expected #{actual} to be almost equal to #{expected}"
         | 
| 14 | 
            -
                range   = 0.00001
         | 
| 15 14 | 
             
                assert expected + range > actual && expected - range < actual, message
         | 
| 16 15 | 
             
              end
         | 
| 17 16 | 
             
            end
         | 
    
        data/test/unitwise/atom_test.rb
    CHANGED
    
    | @@ -31,7 +31,7 @@ describe Unitwise::Atom do | |
| 31 31 | 
             
              let(:joule)   { Unitwise::Atom.find("J")}
         | 
| 32 32 | 
             
              describe "#scale" do
         | 
| 33 33 | 
             
                it "must be nil for base atoms" do
         | 
| 34 | 
            -
                  second.scale. | 
| 34 | 
            +
                  second.scale.must_be_nil
         | 
| 35 35 | 
             
                end
         | 
| 36 36 | 
             
                it "sould be a Scale object for derived atoms" do
         | 
| 37 37 | 
             
                  yard.scale.must_be_instance_of Unitwise::Scale
         | 
| @@ -126,4 +126,40 @@ describe Unitwise::Atom do | |
| 126 126 | 
             
                  second.frozen?.must_equal true
         | 
| 127 127 | 
             
                end
         | 
| 128 128 | 
             
              end
         | 
| 129 | 
            -
             | 
| 129 | 
            +
             | 
| 130 | 
            +
              describe "validate!" do
         | 
| 131 | 
            +
                it "returns true for a valid atom" do
         | 
| 132 | 
            +
                  atom = Unitwise::Atom.new(
         | 
| 133 | 
            +
                    primary_code: "warp",
         | 
| 134 | 
            +
                    secondary_code: "[warp]",
         | 
| 135 | 
            +
                    names: ["Warp", "Warp Factor"],
         | 
| 136 | 
            +
                    scale: {
         | 
| 137 | 
            +
                      value: 1,
         | 
| 138 | 
            +
                      unit_code: "[c]"
         | 
| 139 | 
            +
                    }
         | 
| 140 | 
            +
                  )
         | 
| 141 | 
            +
             | 
| 142 | 
            +
                  atom.validate!.must_equal true
         | 
| 143 | 
            +
                end
         | 
| 144 | 
            +
             | 
| 145 | 
            +
                it "returns an error for an atom with missing properties" do
         | 
| 146 | 
            +
                  atom = Unitwise::Atom.new(names: "hot dog")
         | 
| 147 | 
            +
             | 
| 148 | 
            +
                  assert_raises { atom.validate! }
         | 
| 149 | 
            +
                end
         | 
| 150 | 
            +
             | 
| 151 | 
            +
                it "returns an error for an atom that doesn't resolve" do
         | 
| 152 | 
            +
                  atom = Unitwise::Atom.new(
         | 
| 153 | 
            +
                    primary_code: "feels",
         | 
| 154 | 
            +
                    secondary_code: "FEELS",
         | 
| 155 | 
            +
                    names: ["feels"],
         | 
| 156 | 
            +
                    scale: {
         | 
| 157 | 
            +
                      value: 1,
         | 
| 158 | 
            +
                      unit_code: "hearts"
         | 
| 159 | 
            +
                    }
         | 
| 160 | 
            +
                  )
         | 
| 161 | 
            +
             | 
| 162 | 
            +
                  assert_raises { atom.validate! }
         | 
| 163 | 
            +
                end
         | 
| 164 | 
            +
              end
         | 
| 165 | 
            +
            end
         | 
| @@ -15,8 +15,8 @@ describe Unitwise::Expression::Matcher do | |
| 15 15 | 
             
                it "must be an Alternative list of names" do
         | 
| 16 16 | 
             
                  subject.must_be_instance_of Parslet::Atoms::Alternative
         | 
| 17 17 | 
             
                end
         | 
| 18 | 
            -
                it "must parse ' | 
| 19 | 
            -
                  subject.parse(' | 
| 18 | 
            +
                it "must parse 'joule'" do
         | 
| 19 | 
            +
                  subject.parse('joule').must_equal('joule')
         | 
| 20 20 | 
             
                end
         | 
| 21 21 | 
             
              end
         | 
| 22 22 |  | 
| @@ -39,4 +39,4 @@ describe Unitwise::Expression::Matcher do | |
| 39 39 | 
             
                  subject.parse('h').must_equal('h')
         | 
| 40 40 | 
             
                end
         | 
| 41 41 | 
             
              end
         | 
| 42 | 
            -
            end
         | 
| 42 | 
            +
            end
         | 
    
        data/test/unitwise_test.rb
    CHANGED
    
    | @@ -28,8 +28,28 @@ describe Unitwise do | |
| 28 28 | 
             
                end
         | 
| 29 29 | 
             
              end
         | 
| 30 30 |  | 
| 31 | 
            +
              describe 'register' do
         | 
| 32 | 
            +
                it 'should allow custom units to be registered' do
         | 
| 33 | 
            +
                  josh = {
         | 
| 34 | 
            +
                    names: ["Josh W Lewis", "joshwlewis"],
         | 
| 35 | 
            +
                    symbol: "JWL",
         | 
| 36 | 
            +
                    primary_code: "jwl",
         | 
| 37 | 
            +
                    secondary_code: "jwl",
         | 
| 38 | 
            +
                    scale: {
         | 
| 39 | 
            +
                      value: 71.875,
         | 
| 40 | 
            +
                      unit_code: "[in_i]"
         | 
| 41 | 
            +
                    }
         | 
| 42 | 
            +
                  }
         | 
| 43 | 
            +
             | 
| 44 | 
            +
                  Unitwise.register(josh)
         | 
| 45 | 
            +
             | 
| 46 | 
            +
                  joshes = Unitwise(1, 'mile').to_jwl
         | 
| 47 | 
            +
             | 
| 48 | 
            +
                  joshes.to_i.must_equal(881)
         | 
| 49 | 
            +
                end
         | 
| 50 | 
            +
              end
         | 
| 51 | 
            +
             | 
| 31 52 | 
             
              it "should have a path" do
         | 
| 32 53 | 
             
                Unitwise.path.must_match(/unitwise$/)
         | 
| 33 54 | 
             
              end
         | 
| 34 | 
            -
             | 
| 35 55 | 
             
            end
         | 
    
        metadata
    CHANGED
    
    | @@ -1,14 +1,14 @@ | |
| 1 1 | 
             
            --- !ruby/object:Gem::Specification
         | 
| 2 2 | 
             
            name: unitwise
         | 
| 3 3 | 
             
            version: !ruby/object:Gem::Version
         | 
| 4 | 
            -
              version: 2. | 
| 4 | 
            +
              version: 2.1.0
         | 
| 5 5 | 
             
            platform: ruby
         | 
| 6 6 | 
             
            authors:
         | 
| 7 7 | 
             
            - Josh Lewis
         | 
| 8 8 | 
             
            autorequire: 
         | 
| 9 9 | 
             
            bindir: bin
         | 
| 10 10 | 
             
            cert_chain: []
         | 
| 11 | 
            -
            date:  | 
| 11 | 
            +
            date: 2017-04-28 00:00:00.000000000 Z
         | 
| 12 12 | 
             
            dependencies:
         | 
| 13 13 | 
             
            - !ruby/object:Gem::Dependency
         | 
| 14 14 | 
             
              name: liner
         | 
| @@ -217,7 +217,7 @@ required_rubygems_version: !ruby/object:Gem::Requirement | |
| 217 217 | 
             
                  version: '0'
         | 
| 218 218 | 
             
            requirements: []
         | 
| 219 219 | 
             
            rubyforge_project: 
         | 
| 220 | 
            -
            rubygems_version: 2. | 
| 220 | 
            +
            rubygems_version: 2.6.8
         | 
| 221 221 | 
             
            signing_key: 
         | 
| 222 222 | 
             
            specification_version: 4
         | 
| 223 223 | 
             
            summary: Convert between and perform mathematical operations on physical quantities
         |