Skip to content

Commit

Permalink
Merge nil values by default
Browse files Browse the repository at this point in the history
Previously, if you merged in {a: nil}, it would not overwrite an
existing value.  In rubyconfig#13, this was found to be an undesirable default
behavior.

Now, we will merge nil values by default. You can retain the old
behavior via `config.merge_nil_values = false` in your Config initializer.

This required changes to deep merge, found in: danielsdeleo/deep_merge#33
This was released in deep_merge 1.2.0+.

Fixes rubyconfig#13
  • Loading branch information
jrafanie committed Feb 7, 2018
1 parent 0820c8d commit f1caeb1
Show file tree
Hide file tree
Showing 7 changed files with 89 additions and 5 deletions.
1 change: 1 addition & 0 deletions CHANGELOG.md
Original file line number Diff line number Diff line change
Expand Up @@ -2,6 +2,7 @@

## Unreleased

* **WARNING:** nil values will now overwrite an existing value. This change of behavior can be reverted via `config.merge_nil_values = false` in your Config initializer ([#196](https://github.com/railsconfig/config/pull/196))
* Make dry-validation dependency less strict allowing to use newer versions ([#183](https://github.com/railsconfig/config/pull/183))
* Fix `key?` and `has_key?`, which raise NoMethodError in non Rails environment, by using ActiveSupport `#delegate` implicitly ([#185](https://github.com/railsconfig/config/pull/185))
* Update `deep_merge` dependency to latest version (v1.2.1) ([#191](https://github.com/railsconfig/config/pull/191))
Expand Down
22 changes: 22 additions & 0 deletions README.md
Original file line number Diff line number Diff line change
Expand Up @@ -271,6 +271,28 @@ located at `config/initializers/config.rb`.
### Merge customization

* `overwrite_arrays` - overwrite arrays found in previously loaded settings file. Default: `true`
* `merge_nil_values` - overwrite a value with a nil value. Default: `true`.

```ruby
# merge_nil_values is true by default
c = Config.load_files("./spec/fixtures/development.yml") # => #<Config::Options size=2, ...>
c.size # => 2
c.merge!(size: nil) => #<Config::Options size=nil, ...>
c.size # => nil
```

```ruby
# To reject nil values when merging settings:
Config.setup do |config|
config.merge_nil_values = false
end
c = Config.load_files("./spec/fixtures/development.yml") # => #<Config::Options size=2, ...>
c.size # => 2
c.merge!(size: nil) => #<Config::Options size=nil, ...>
c.size # => 2
```

* `knockout_prefix` - ability to remove elements of the array set in earlier loaded settings file. Makes sense only when `overwrite_arrays = false`, otherwise array settings would be overwritten by default. Default: `nil`

Check [Deep Merge](https://github.com/danielsdeleo/deep_merge) for more details.
Expand Down
3 changes: 2 additions & 1 deletion lib/config.rb
Original file line number Diff line number Diff line change
Expand Up @@ -26,8 +26,9 @@ module Config
@@fail_on_missing = false

# deep_merge options
mattr_accessor :knockout_prefix, :overwrite_arrays
mattr_accessor :knockout_prefix, :merge_nil_values, :overwrite_arrays
@@knockout_prefix = nil
@@merge_nil_values = true
@@overwrite_arrays = true

def self.setup
Expand Down
1 change: 1 addition & 0 deletions lib/config/options.rb
Original file line number Diff line number Diff line change
Expand Up @@ -127,6 +127,7 @@ def merge!(hash)
current = to_hash
DeepMerge.deep_merge!(hash.dup,
current,
merge_nil_values: Config.merge_nil_values,
preserve_unmergeables: false,
overwrite_arrays: Config.overwrite_arrays)
marshal_load(__convert(current).marshal_dump)
Expand Down
5 changes: 5 additions & 0 deletions lib/generators/config/templates/config.rb
Original file line number Diff line number Diff line change
Expand Up @@ -6,6 +6,11 @@
#
# config.knockout_prefix = nil

# When merging nil values for settings, overwrite an existing value with nil.
# When set to `false`, the original value is retained.
#
# config.merge_nil_values = true

# Overwrite arrays found in previously loaded settings file. When set to `false`, arrays will be merged.
#
# config.overwrite_arrays = true
Expand Down
59 changes: 55 additions & 4 deletions spec/config_spec.rb
Original file line number Diff line number Diff line change
Expand Up @@ -198,7 +198,8 @@

context "Merging nested hash at runtime" do
let(:config) { Config.load_files("#{fixture_path}/deep_merge/config1.yml") }
let(:hash) { { :inner => { :something1 => 'changed1', :something3 => 'changed3' } } }
let(:hash) { { inner: { something1: 'changed1', something3: 'changed3' } } }
let(:hash_with_nil) { { inner: { something1: nil } } }

it 'should preserve first level keys' do
expect { config.merge!(hash) }.to_not change { config.keys }
Expand All @@ -210,11 +211,61 @@
end

it 'should add new nested key' do
expect { config.merge!(hash) }.to change { config.inner.something3 }.from(nil).to("changed3")
expect { config.merge!(hash) }
.to change { config.inner.something3 }.from(nil).to('changed3')
end

it 'should rewrite a merged value' do
expect { config.merge!(hash) }.to change { config.inner.something1 }.from('blah1').to('changed1')
expect { config.merge!(hash) }
.to change { config.inner.something1 }.from('blah1').to('changed1')
end

it 'should update a string to nil ' do
expect { config.merge!(hash_with_nil) }
.to change { config.inner.something1 }.from('blah1').to(nil)
end

it 'should update something nil to true' do
expect { config.merge!(inner: { somethingnil: true }) }
.to change { config.inner.somethingnil }.from(nil).to(true)
end

it 'should update something nil to false' do
expect { config.merge!(inner: { somethingnil: false }) }
.to change { config.inner.somethingnil }.from(nil).to(false)
end

it 'should update something false to true' do
expect { config.merge!(inner: { somethingfalse: true }) }
.to change { config.inner.somethingfalse }.from(false).to(true)
end

it 'should update something false to nil' do
expect { config.merge!(inner: { somethingfalse: nil }) }
.to change { config.inner.somethingfalse }.from(false).to(nil)
end

it 'should update something true to false' do
expect { config.merge!(inner: { somethingtrue: false }) }
.to change { config.inner.somethingtrue }.from(true).to(false)
end

it 'should update something true to nil' do
expect { config.merge!(inner: { somethingtrue: nil }) }
.to change { config.inner.somethingtrue }.from(true).to(nil)
end

context 'with Config.merge_nil_values = false' do
let(:config) do
Config.merge_nil_values = false
Config.load_files("#{fixture_path}/deep_merge/config1.yml")
end

it 'should not overwrite values with nil' do
old_value = config.inner.something1
config.merge!(hash_with_nil)
expect(config.inner.something1).to eq(old_value)
end
end
end

Expand Down Expand Up @@ -355,7 +406,7 @@
end

it 'should merge hashes from multiple configs' do
expect(config.inner.marshal_dump.keys.size).to eq(3)
expect(config.inner.marshal_dump.keys.size).to eq(6)
expect(config.inner2.inner2_inner.marshal_dump.keys.size).to eq(3)
end

Expand Down
3 changes: 3 additions & 0 deletions spec/fixtures/deep_merge/config1.yml
Original file line number Diff line number Diff line change
Expand Up @@ -3,6 +3,9 @@ server: google.com
inner:
something1: "blah1"
something2: "blah2"
somethingnil:
somethingfalse: false
somethingtrue: true

inner2:
inner2_inner:
Expand Down

0 comments on commit f1caeb1

Please sign in to comment.