Created
August 6, 2014 18:57
-
-
Save Heath101/043bc9a5cc5296d9a063 to your computer and use it in GitHub Desktop.
Underscore Accessor
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
require 'minitest/autorun' | |
class Hash | |
def method_missing( symbol, *args ) | |
if symbol.to_s.start_with?('_') | |
if symbol.to_s.end_with?('=') | |
key = symbol.to_s.sub('_', '').chop | |
if self.has_key?(key.to_sym) | |
self[key.to_sym] = args.first | |
elsif self.has_key?(key) | |
self[key] = args.first | |
else | |
self[key.to_sym] = args.first | |
end | |
else | |
key = symbol.to_s.sub('_', '') | |
if self.has_key?(key.to_sym) | |
self[key.to_sym] | |
else | |
self[key] | |
end | |
end | |
end | |
end | |
end | |
describe 'Underscore accessor' do | |
describe 'returning correct values' do | |
it 'returns nil for empty hash' do | |
assert_equal(nil, {}._foo) | |
end | |
it 'returns correct value for existing keys' do | |
assert_equal('bar', {foo: 'bar'}._foo) | |
assert_equal('bar', {'foo' => 'bar'}._foo) | |
assert_equal('sym', {foo: 'sym', 'foo' => 'str'}._foo) | |
end | |
end | |
describe 'setting keys' do | |
it 'returns value when setting non-existent key' do | |
assert_equal('bar', {}._foo = 'bar') | |
end | |
it 'sets value for non-existent key, as a symbol key' do | |
test = {} | |
test._foo = 'bar' | |
assert_equal('bar', test[:foo]) | |
assert_equal(nil, test['foo']) | |
end | |
it 'sets value for symbol key' do | |
test = {foo: 'bar'} | |
test._foo = 'sym' | |
assert_equal('sym', test[:foo]) | |
end | |
it 'sets value for a string key' do | |
test = {'fuu' => 'baz'} | |
test._fuu = 'str' | |
assert_equal('str', test['fuu']) | |
end | |
it 'sets only symbol value when key exists as both symbol and string' do | |
test = {foo: 'bar', 'foo' => 'baz'} | |
test._foo = 'sym' | |
assert_equal('sym', test[:foo]) | |
assert_equal('baz', test['foo']) | |
end | |
end | |
describe 'edge cases' do | |
it 'retrieves value when value is false' do | |
assert_equal(false, {:enabled=>false, "enabled"=>true}._enabled) | |
end | |
it 'sets value when initial value is nil' do | |
test = {:enabled=>nil, "enabled"=>true} | |
test._enabled = false | |
assert_equal(false, test[:enabled]) | |
end | |
end | |
end |
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment
I don't like the method name 'msg_for' because it doesn't really say what it does to method.