# frozen_string_literal: true require 'spec_helper' describe Hashdiff do it 'is able to diff two empty hashes' do diff = described_class.diff({}, {}) diff.should == [] end it 'is able to diff an hash with an empty hash' do a = { 'a' => 3, 'b' => 2 } b = {} diff = described_class.diff(a, b) expect(diff).to eq([['-', 'a', 3], ['-', 'b', 2]]) diff = described_class.diff(b, a) diff.should == [['+', 'a', 3], ['+', 'b', 2]] end it 'is able to diff two equal hashes' do diff = described_class.diff({ 'a' => 2, 'b' => 2 }, 'a' => 2, 'b' => 2) diff.should == [] end it 'is able to diff two equal hashes with mixed key types' do a = { 'a' => 1, :b => 1 } diff = described_class.diff(a, a) diff.should == [] end it 'is able to diff if mixed key types are removed' do a = { 'a' => 1, :b => 1 } b = {} diff = described_class.diff(a, b) diff.should == [['-', 'a', 1], ['-', 'b', 1]] end it 'is able to diff if mixed key types are added' do a = { 'a' => 1, :b => 1 } b = {} diff = described_class.diff(b, a) diff.should == [['+', 'a', 1], ['+', 'b', 1]] end it 'is able to diff two hashes with equivalent numerics, when strict is false' do diff = described_class.diff({ 'a' => 2.0, 'b' => 2 }, { 'a' => 2, 'b' => 2.0 }, strict: false) diff.should == [] end context 'with the ignore_keys option' do a = { a: 1, b: { d: 2, a: 3 }, c: 4 } b = { a: 2, b: { d: 2, a: 7 }, c: 5 } it 'ignores a single key' do diff = described_class.diff(a, b, ignore_keys: :a) diff.should == [['~', 'c', 4, 5]] end it 'ignores an array of keys' do diff = described_class.diff(a, b, ignore_keys: %i[a c]) diff.should == [] end end it 'ignores string vs symbol differences, when indifferent is true' do diff = described_class.diff({ 'a' => 2, :b => 2 }, { :a => 2, 'b' => 2, :c => 3 }, indifferent: true) diff.should == [['+', 'c', 3]] end it 'is able to diff changes in hash value' do diff = described_class.diff({ 'a' => 2, 'b' => 3, 'c' => ' hello' }, 'a' => 2, 'b' => 4, 'c' => 'hello') diff.should == [['~', 'b', 3, 4], ['~', 'c', ' hello', 'hello']] end it 'is able to diff changes in hash value which is array' do diff = described_class.diff({ 'a' => 2, 'b' => [1, 2, 3] }, 'a' => 2, 'b' => [1, 3, 4]) diff.should == [['-', 'b[1]', 2], ['+', 'b[2]', 4]] end it 'is able to diff changes in hash value which is hash' do diff = described_class.diff({ 'a' => { 'x' => 2, 'y' => 3, 'z' => 4 }, 'b' => { 'x' => 3, 'z' => 45 } }, 'a' => { 'y' => 3 }, 'b' => { 'y' => 3, 'z' => 30 }) diff.should == [['-', 'a.x', 2], ['-', 'a.z', 4], ['-', 'b.x', 3], ['~', 'b.z', 45, 30], ['+', 'b.y', 3]] end it 'is able to best diff similar objects in array' do diff = described_class.best_diff({ 'a' => [{ 'x' => 2, 'y' => 3, 'z' => 4 }, { 'x' => 11, 'y' => 22, 'z' => 33 }], 'b' => { 'x' => 3, 'z' => 45 } }, 'a' => [{ 'y' => 3 }, { 'x' => 11, 'z' => 33 }], 'b' => { 'y' => 22 }) diff.should == [['-', 'a[0].x', 2], ['-', 'a[0].z', 4], ['-', 'a[1].y', 22], ['-', 'b.x', 3], ['-', 'b.z', 45], ['+', 'b.y', 22]] end it 'is able to diff addition of key value pair' do a = { 'a' => 3, 'c' => 11, 'd' => 45, 'e' => 100, 'f' => 200 } b = { 'a' => 3, 'c' => 11, 'd' => 45, 'e' => 100, 'f' => 200, 'g' => 300 } diff = described_class.diff(a, b) expect(diff).to eq([['+', 'g', 300]]) diff = described_class.diff(b, a) diff.should == [['-', 'g', 300]] end it 'is able to diff value type changes' do a = { 'a' => 3 } b = { 'a' => { 'a1' => 1, 'a2' => 2 } } diff = described_class.diff(a, b) expect(diff).to eq([['~', 'a', 3, { 'a1' => 1, 'a2' => 2 }]]) diff = described_class.diff(b, a) diff.should == [['~', 'a', { 'a1' => 1, 'a2' => 2 }, 3]] end it 'is able to diff value changes: array <=> []' do a = { 'a' => 1, 'b' => [1, 2] } b = { 'a' => 1, 'b' => [] } diff = described_class.diff(a, b) diff.should == [['-', 'b[1]', 2], ['-', 'b[0]', 1]] end it 'is able to diff value changes: array <=> nil' do a = { 'a' => 1, 'b' => [1, 2] } b = { 'a' => 1, 'b' => nil } diff = described_class.diff(a, b) diff.should == [['~', 'b', [1, 2], nil]] end it 'is able to diff value chagnes: remove array completely' do a = { 'a' => 1, 'b' => [1, 2] } b = { 'a' => 1 } diff = described_class.diff(a, b) diff.should == [['-', 'b', [1, 2]]] end it 'is able to diff value changes: remove whole hash' do a = { 'a' => 1, 'b' => { 'b1' => 1, 'b2' => 2 } } b = { 'a' => 1 } diff = described_class.diff(a, b) diff.should == [['-', 'b', { 'b1' => 1, 'b2' => 2 }]] end it 'is able to diff value changes: hash <=> {}' do a = { 'a' => 1, 'b' => { 'b1' => 1, 'b2' => 2 } } b = { 'a' => 1, 'b' => {} } diff = described_class.diff(a, b) diff.should == [['-', 'b.b1', 1], ['-', 'b.b2', 2]] end it 'is able to diff value changes: hash <=> nil' do a = { 'a' => 1, 'b' => { 'b1' => 1, 'b2' => 2 } } b = { 'a' => 1, 'b' => nil } diff = described_class.diff(a, b) diff.should == [['~', 'b', { 'b1' => 1, 'b2' => 2 }, nil]] end it 'is able to diff similar objects in array' do a = [{ 'a' => 1, 'b' => 2, 'c' => 3, 'd' => 4, 'e' => 5 }, 3] b = [1, { 'a' => 1, 'b' => 2, 'c' => 3, 'e' => 5 }] diff = described_class.diff(a, b) diff.should == [['-', '[0].d', 4], ['+', '[0]', 1], ['-', '[2]', 3]] end it 'is able to diff similar & equal objects in array' do a = [{ 'a' => 1, 'b' => 2, 'c' => 3, 'd' => 4, 'e' => 5 }, { 'x' => 5, 'y' => 6, 'z' => 3 }, 3] b = [{ 'a' => 1, 'b' => 2, 'c' => 3, 'e' => 5 }, 3] diff = described_class.diff(a, b) diff.should == [['-', '[0].d', 4], ['-', '[1]', { 'x' => 5, 'y' => 6, 'z' => 3 }]] end it 'uses custom delimiter when provided' do a = [{ 'a' => 1, 'b' => 2, 'c' => 3, 'd' => 4, 'e' => 5 }, { 'x' => 5, 'y' => 6, 'z' => 3 }, 3] b = [{ 'a' => 1, 'b' => 2, 'c' => 3, 'e' => 5 }, 3] diff = described_class.diff(a, b, similarity: 0.8, delimiter: "\t") diff.should == [['-', "[0]\td", 4], ['-', '[1]', { 'x' => 5, 'y' => 6, 'z' => 3 }]] end context 'when :numeric_tolerance requested' do it 'is able to diff changes in hash value' do a = { 'a' => 0.558, 'b' => 0.0, 'c' => 0.65, 'd' => 'fin' } b = { 'a' => 0.557, 'b' => 'hats', 'c' => 0.67, 'd' => 'fin' } diff = described_class.diff(a, b, numeric_tolerance: 0.01) expect(diff).to eq([['~', 'b', 0.0, 'hats'], ['~', 'c', 0.65, 0.67]]) diff = described_class.diff(b, a, numeric_tolerance: 0.01) diff.should == [['~', 'b', 'hats', 0.0], ['~', 'c', 0.67, 0.65]] end it 'is able to diff changes in nested values' do a = { 'a' => { 'x' => 0.4, 'y' => 0.338 }, 'b' => [13, 68.03] } b = { 'a' => { 'x' => 0.6, 'y' => 0.341 }, 'b' => [14, 68.025] } diff = described_class.diff(a, b, numeric_tolerance: 0.01) expect(diff).to eq([['~', 'a.x', 0.4, 0.6], ['-', 'b[0]', 13], ['+', 'b[0]', 14]]) diff = described_class.diff(b, a, numeric_tolerance: 0.01) diff.should == [['~', 'a.x', 0.6, 0.4], ['-', 'b[0]', 14], ['+', 'b[0]', 13]] end end context 'when :strip requested' do it 'strips strings before comparing' do a = { 'a' => ' foo', 'b' => 'fizz buzz' } b = { 'a' => 'foo', 'b' => 'fizzbuzz' } diff = described_class.diff(a, b, strip: true) diff.should == [['~', 'b', 'fizz buzz', 'fizzbuzz']] end it 'strips nested strings before comparing' do a = { 'a' => { 'x' => ' foo' }, 'b' => ['fizz buzz', 'nerf'] } b = { 'a' => { 'x' => 'foo' }, 'b' => %w[fizzbuzz nerf] } diff = described_class.diff(a, b, strip: true) diff.should == [['-', 'b[0]', 'fizz buzz'], ['+', 'b[0]', 'fizzbuzz']] end end context 'when :case_insensitive requested' do it 'strips strings before comparing' do a = { 'a' => 'Foo', 'b' => 'fizz buzz' } b = { 'a' => 'foo', 'b' => 'fizzBuzz' } diff = described_class.diff(a, b, case_insensitive: true) diff.should == [['~', 'b', 'fizz buzz', 'fizzBuzz']] end it 'ignores case on nested strings before comparing' do a = { 'a' => { 'x' => 'Foo' }, 'b' => ['fizz buzz', 'nerf'] } b = { 'a' => { 'x' => 'foo' }, 'b' => %w[fizzbuzz nerf] } diff = described_class.diff(a, b, case_insensitive: true) diff.should == [['-', 'b[0]', 'fizz buzz'], ['+', 'b[0]', 'fizzbuzz']] end end context 'when both :strip and :numeric_tolerance requested' do it 'applies filters to proper object types' do a = { 'a' => ' foo', 'b' => 35, 'c' => 'bar', 'd' => 'baz' } b = { 'a' => 'foo', 'b' => 35.005, 'c' => 'bar', 'd' => 18.5 } diff = described_class.diff(a, b, strict: false, numeric_tolerance: 0.01, strip: true) diff.should == [['~', 'd', 'baz', 18.5]] end end context 'when both :strip and :case_insensitive requested' do it 'applies both filters to strings' do a = { 'a' => ' Foo', 'b' => 'fizz buzz' } b = { 'a' => 'foo', 'b' => 'fizzBuzz' } diff = described_class.diff(a, b, case_insensitive: true, strip: true) diff.should == [['~', 'b', 'fizz buzz', 'fizzBuzz']] end end context 'with custom comparison' do let(:a) { { 'a' => 'car', 'b' => 'boat', 'c' => 'plane' } } let(:b) { { 'a' => 'bus', 'b' => 'truck', 'c' => ' plan' } } it 'compares using proc specified in block' do diff = described_class.diff(a, b) do |prefix, obj1, obj2| case prefix when /a|b|c/ obj1.length == obj2.length end end diff.should == [['~', 'b', 'boat', 'truck']] end it 'yields added keys' do x = { 'a' => 'car', 'b' => 'boat' } y = { 'a' => 'car' } diff = described_class.diff(x, y) do |prefix, _obj1, _obj2| case prefix when /b/ true end end diff.should == [] end it 'compares with both proc and :strip when both provided' do diff = described_class.diff(a, b, strip: true) do |prefix, obj1, obj2| case prefix when 'a' obj1.length == obj2.length end end diff.should == [['~', 'b', 'boat', 'truck'], ['~', 'c', 'plane', ' plan']] end it 'compares nested arrays using proc specified in block' do a = { a: 'car', b: %w[boat plane] } b = { a: 'bus', b: ['truck', ' plan'] } diff = described_class.diff(a, b) do |path, obj1, obj2| case path when 'b[*]' obj1.length == obj2.length end end expect(diff).to eq [['~', 'a', 'car', 'bus'], ['~', 'b[1]', 'plane', ' plan'], ['-', 'b[0]', 'boat'], ['+', 'b[0]', 'truck']] end end context 'when :array_path is true' do it 'returns the diff path in an array rather than a string' do x = { 'a' => 'foo' } y = { 'a' => 'bar' } diff = described_class.diff(x, y, array_path: true) diff.should == [['~', ['a'], 'foo', 'bar']] end it 'shows array indexes in paths' do x = { 'a' => [0, 1, 2] } y = { 'a' => [0, 1, 2, 3] } diff = described_class.diff(x, y, array_path: true) diff.should == [['+', ['a', 3], 3]] end it 'shows differences with string and symbol keys' do x = { 'a' => 'foo' } y = { a: 'bar' } diff = described_class.diff(x, y, array_path: true) diff.should == [['-', ['a'], 'foo'], ['+', [:a], 'bar']] end it 'supports other key types' do time = Time.now x = { time => 'foo' } y = { 0 => 'bar' } diff = described_class.diff(x, y, array_path: true) diff.should == [['-', [time], 'foo'], ['+', [0], 'bar']] end end context 'when :use_lcs is false' do it 'shows items in an array as changed' do x = %i[a b] y = %i[c d] diff = described_class.diff(x, y, use_lcs: false) diff.should == [['~', '[0]', :a, :c], ['~', '[1]', :b, :d]] end it 'shows additions to arrays' do x = { a: [0] } y = { a: [0, 1] } diff = described_class.diff(x, y, use_lcs: false) diff.should == [['+', 'a[1]', 1]] end it 'shows changes to nested arrays' do x = { a: [[0, 1]] } y = { a: [[1, 2]] } diff = described_class.diff(x, y, use_lcs: false) diff.should == [['~', 'a[0][0]', 0, 1], ['~', 'a[0][1]', 1, 2]] end end end