super_diff 0.11.0 → 0.12.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.
Potentially problematic release.
This version of super_diff might be problematic. Click here for more details.
- checksums.yaml +4 -4
- data/README.md +19 -166
- data/lib/super_diff/active_record/differs/active_record_relation.rb +1 -1
- data/lib/super_diff/active_record/inspection_tree_builders/active_record_model.rb +57 -0
- data/lib/super_diff/active_record/inspection_tree_builders/active_record_relation.rb +34 -0
- data/lib/super_diff/active_record/inspection_tree_builders.rb +14 -0
- data/lib/super_diff/active_record/monkey_patches.rb +6 -3
- data/lib/super_diff/active_record/object_inspection.rb +16 -4
- data/lib/super_diff/active_record/operation_tree_builders/active_record_model.rb +6 -2
- data/lib/super_diff/active_record/operation_tree_builders/active_record_relation.rb +1 -1
- data/lib/super_diff/active_record/operation_tree_flatteners/active_record_relation.rb +1 -1
- data/lib/super_diff/active_record/operation_trees/active_record_relation.rb +1 -1
- data/lib/super_diff/active_record.rb +12 -16
- data/lib/super_diff/active_support/differs/hash_with_indifferent_access.rb +1 -1
- data/lib/super_diff/active_support/inspection_tree_builders/hash_with_indifferent_access.rb +44 -0
- data/lib/super_diff/active_support/inspection_tree_builders/ordered_options.rb +44 -0
- data/lib/super_diff/active_support/inspection_tree_builders.rb +14 -0
- data/lib/super_diff/active_support/object_inspection.rb +16 -4
- data/lib/super_diff/active_support/operation_tree_builders/hash_with_indifferent_access.rb +1 -1
- data/lib/super_diff/active_support/operation_tree_flatteners/hash_with_indifferent_access.rb +1 -1
- data/lib/super_diff/active_support/operation_trees/hash_with_indifferent_access.rb +1 -1
- data/lib/super_diff/active_support.rb +11 -15
- data/lib/super_diff/basic/diff_formatters/collection.rb +135 -0
- data/lib/super_diff/basic/diff_formatters/multiline_string.rb +34 -0
- data/lib/super_diff/basic/diff_formatters.rb +11 -0
- data/lib/super_diff/basic/differs/array.rb +17 -0
- data/lib/super_diff/basic/differs/custom_object.rb +19 -0
- data/lib/super_diff/basic/differs/date_like.rb +17 -0
- data/lib/super_diff/basic/differs/default_object.rb +24 -0
- data/lib/super_diff/basic/differs/hash.rb +17 -0
- data/lib/super_diff/basic/differs/multiline_string.rb +18 -0
- data/lib/super_diff/basic/differs/time_like.rb +17 -0
- data/lib/super_diff/basic/differs.rb +24 -0
- data/lib/super_diff/{object_inspection → basic}/inspection_tree_builders/array.rb +3 -3
- data/lib/super_diff/{object_inspection → basic}/inspection_tree_builders/custom_object.rb +3 -3
- data/lib/super_diff/{object_inspection → basic}/inspection_tree_builders/date_like.rb +3 -3
- data/lib/super_diff/{object_inspection → basic}/inspection_tree_builders/default_object.rb +5 -7
- data/lib/super_diff/{object_inspection → basic}/inspection_tree_builders/hash.rb +3 -3
- data/lib/super_diff/{object_inspection → basic}/inspection_tree_builders/primitive.rb +3 -3
- data/lib/super_diff/{object_inspection → basic}/inspection_tree_builders/time_like.rb +3 -3
- data/lib/super_diff/basic/inspection_tree_builders.rb +20 -0
- data/lib/super_diff/basic/operation_tree_builders/array.rb +111 -0
- data/lib/super_diff/basic/operation_tree_builders/custom_object.rb +42 -0
- data/lib/super_diff/basic/operation_tree_builders/date_like.rb +17 -0
- data/lib/super_diff/basic/operation_tree_builders/default_object.rb +117 -0
- data/lib/super_diff/basic/operation_tree_builders/hash.rb +222 -0
- data/lib/super_diff/basic/operation_tree_builders/multiline_string.rb +90 -0
- data/lib/super_diff/basic/operation_tree_builders/time_like.rb +26 -0
- data/lib/super_diff/basic/operation_tree_builders.rb +34 -0
- data/lib/super_diff/basic/operation_tree_flatteners/array.rb +17 -0
- data/lib/super_diff/basic/operation_tree_flatteners/collection.rb +140 -0
- data/lib/super_diff/basic/operation_tree_flatteners/custom_object.rb +30 -0
- data/lib/super_diff/basic/operation_tree_flatteners/default_object.rb +32 -0
- data/lib/super_diff/basic/operation_tree_flatteners/hash.rb +35 -0
- data/lib/super_diff/basic/operation_tree_flatteners/multiline_string.rb +20 -0
- data/lib/super_diff/basic/operation_tree_flatteners.rb +24 -0
- data/lib/super_diff/basic/operation_trees/array.rb +17 -0
- data/lib/super_diff/basic/operation_trees/custom_object.rb +17 -0
- data/lib/super_diff/basic/operation_trees/default_object.rb +42 -0
- data/lib/super_diff/basic/operation_trees/hash.rb +17 -0
- data/lib/super_diff/basic/operation_trees/multiline_string.rb +17 -0
- data/lib/super_diff/basic/operation_trees.rb +25 -0
- data/lib/super_diff/basic.rb +48 -0
- data/lib/super_diff/{differs/base.rb → core/abstract_differ.rb} +2 -2
- data/lib/super_diff/core/abstract_inspection_tree_builder.rb +26 -0
- data/lib/super_diff/{operation_trees/base.rb → core/abstract_operation_tree.rb} +6 -2
- data/lib/super_diff/{operation_tree_builders/base.rb → core/abstract_operation_tree_builder.rb} +4 -8
- data/lib/super_diff/{operation_tree_flatteners/base.rb → core/abstract_operation_tree_flattener.rb} +2 -2
- data/lib/super_diff/{operations → core}/binary_operation.rb +1 -1
- data/lib/super_diff/core/colorized_document_extensions.rb +20 -0
- data/lib/super_diff/core/configuration.rb +192 -0
- data/lib/super_diff/core/differ_dispatcher.rb +33 -0
- data/lib/super_diff/core/gem_version.rb +47 -0
- data/lib/super_diff/core/helpers.rb +88 -0
- data/lib/super_diff/core/implementation_checks.rb +21 -0
- data/lib/super_diff/{object_inspection → core}/inspection_tree.rb +7 -6
- data/lib/super_diff/core/inspection_tree_builder_dispatcher.rb +23 -0
- data/lib/super_diff/{object_inspection/nodes → core/inspection_tree_nodes}/as_lines_when_rendering_to_lines.rb +9 -3
- data/lib/super_diff/{object_inspection/nodes → core/inspection_tree_nodes}/as_prefix_when_rendering_to_lines.rb +3 -3
- data/lib/super_diff/{object_inspection/nodes → core/inspection_tree_nodes}/as_prelude_when_rendering_to_lines.rb +3 -3
- data/lib/super_diff/{object_inspection/nodes → core/inspection_tree_nodes}/as_single_line.rb +3 -3
- data/lib/super_diff/{object_inspection/nodes → core/inspection_tree_nodes}/base.rb +2 -2
- data/lib/super_diff/{object_inspection/nodes → core/inspection_tree_nodes}/inspection.rb +7 -11
- data/lib/super_diff/{object_inspection/nodes → core/inspection_tree_nodes}/nesting.rb +2 -2
- data/lib/super_diff/{object_inspection/nodes → core/inspection_tree_nodes}/only_when.rb +2 -2
- data/lib/super_diff/{object_inspection/nodes → core/inspection_tree_nodes}/text.rb +2 -2
- data/lib/super_diff/{object_inspection/nodes → core/inspection_tree_nodes}/when_empty.rb +2 -2
- data/lib/super_diff/{object_inspection/nodes → core/inspection_tree_nodes}/when_non_empty.rb +2 -2
- data/lib/super_diff/{object_inspection/nodes → core/inspection_tree_nodes}/when_rendering_to_lines.rb +2 -2
- data/lib/super_diff/{object_inspection/nodes → core/inspection_tree_nodes}/when_rendering_to_string.rb +2 -2
- data/lib/super_diff/core/inspection_tree_nodes.rb +55 -0
- data/lib/super_diff/core/line.rb +85 -0
- data/lib/super_diff/{errors → core}/no_differ_available_error.rb +1 -1
- data/lib/super_diff/core/no_inspection_tree_builder_available_error.rb +21 -0
- data/lib/super_diff/core/no_operation_tree_available_error.rb +20 -0
- data/lib/super_diff/core/no_operation_tree_builder_available_error.rb +24 -0
- data/lib/super_diff/{operation_tree_builders/main.rb → core/operation_tree_builder_dispatcher.rb} +9 -13
- data/lib/super_diff/core/operation_tree_finder.rb +27 -0
- data/lib/super_diff/core/prefix_for_next_inspection_tree_node.rb +6 -0
- data/lib/super_diff/core/prelude_for_next_inspection_tree_node.rb +6 -0
- data/lib/super_diff/core/recursion_guard.rb +52 -0
- data/lib/super_diff/core/tiered_lines.rb +6 -0
- data/lib/super_diff/core/tiered_lines_elider.rb +472 -0
- data/lib/super_diff/core/tiered_lines_formatter.rb +77 -0
- data/lib/super_diff/{operations → core}/unary_operation.rb +1 -1
- data/lib/super_diff/core.rb +69 -0
- data/lib/super_diff/differs.rb +19 -12
- data/lib/super_diff/equality_matchers/array.rb +3 -3
- data/lib/super_diff/equality_matchers/default.rb +8 -3
- data/lib/super_diff/equality_matchers/hash.rb +3 -3
- data/lib/super_diff/equality_matchers/multiline_string.rb +3 -3
- data/lib/super_diff/equality_matchers/primitive.rb +3 -3
- data/lib/super_diff/equality_matchers/singleline_string.rb +2 -2
- data/lib/super_diff/errors.rb +12 -8
- data/lib/super_diff/object_inspection.rb +63 -14
- data/lib/super_diff/operation_tree_builders.rb +19 -15
- data/lib/super_diff/operation_tree_flatteners.rb +19 -16
- data/lib/super_diff/operation_trees.rb +19 -9
- data/lib/super_diff/operations.rb +12 -2
- data/lib/super_diff/rspec/augmented_matcher.rb +1 -1
- data/lib/super_diff/rspec/differ.rb +4 -5
- data/lib/super_diff/rspec/differs/collection_containing_exactly.rb +1 -1
- data/lib/super_diff/rspec/differs/collection_including.rb +1 -1
- data/lib/super_diff/rspec/differs/hash_including.rb +1 -1
- data/lib/super_diff/rspec/differs/object_having_attributes.rb +1 -1
- data/lib/super_diff/rspec/inspection_tree_builders/collection_containing_exactly.rb +34 -0
- data/lib/super_diff/rspec/inspection_tree_builders/collection_including.rb +40 -0
- data/lib/super_diff/rspec/inspection_tree_builders/double.rb +100 -0
- data/lib/super_diff/rspec/inspection_tree_builders/generic_describable_matcher.rb +17 -0
- data/lib/super_diff/rspec/inspection_tree_builders/hash_including.rb +40 -0
- data/lib/super_diff/rspec/inspection_tree_builders/instance_of.rb +25 -0
- data/lib/super_diff/rspec/inspection_tree_builders/kind_of.rb +25 -0
- data/lib/super_diff/rspec/inspection_tree_builders/object_having_attributes.rb +34 -0
- data/lib/super_diff/rspec/inspection_tree_builders/primitive.rb +9 -0
- data/lib/super_diff/rspec/inspection_tree_builders/value_within.rb +30 -0
- data/lib/super_diff/rspec/inspection_tree_builders.rb +40 -0
- data/lib/super_diff/rspec/object_inspection.rb +14 -4
- data/lib/super_diff/rspec/operation_tree_builders/collection_containing_exactly.rb +4 -4
- data/lib/super_diff/rspec/operation_tree_builders/collection_including.rb +1 -1
- data/lib/super_diff/rspec/operation_tree_builders/hash_including.rb +1 -1
- data/lib/super_diff/rspec/operation_tree_builders/object_having_attributes.rb +2 -2
- data/lib/super_diff/rspec.rb +20 -18
- data/lib/super_diff/version.rb +1 -1
- data/lib/super_diff.rb +69 -21
- data/spec/examples.txt +704 -543
- data/spec/support/integration/helpers.rb +4 -1
- data/spec/support/integration/matchers/produce_output_when_run_matcher.rb +1 -1
- data/spec/support/models/active_record/person.rb +8 -1
- data/spec/support/shared_examples/active_record.rb +5 -5
- data/spec/support/unit/helpers.rb +12 -1
- data/spec/support/unit/matchers/be_deprecated_in_favor_of.rb +39 -0
- data/spec/unit/active_record/object_inspection_spec.rb +5 -5
- data/spec/unit/{operation_tree_flatteners → basic/operation_tree_flatteners}/array_spec.rb +8 -8
- data/spec/unit/{operation_tree_flatteners → basic/operation_tree_flatteners}/custom_object_spec.rb +9 -9
- data/spec/unit/{operation_tree_flatteners → basic/operation_tree_flatteners}/default_object_spec.rb +9 -9
- data/spec/unit/{operation_tree_flatteners → basic/operation_tree_flatteners}/hash_spec.rb +8 -8
- data/spec/unit/{operation_tree_flatteners → basic/operation_tree_flatteners}/multiline_string_spec.rb +4 -4
- data/spec/unit/{helpers_spec.rb → core/helpers_spec.rb} +2 -2
- data/spec/unit/{tiered_lines_elider_spec.rb → core/tiered_lines_elider_spec.rb} +2 -2
- data/spec/unit/{tiered_lines_formatter_spec.rb → core/tiered_lines_formatter_spec.rb} +1 -1
- data/spec/unit/deprecations_spec.rb +176 -0
- data/spec/unit/equality_matchers/main_spec.rb +5 -5
- data/super_diff.gemspec +6 -0
- metadata +127 -115
- data/lib/super_diff/active_record/object_inspection/inspection_tree_builders/active_record_model.rb +0 -51
- data/lib/super_diff/active_record/object_inspection/inspection_tree_builders/active_record_relation.rb +0 -36
- data/lib/super_diff/active_record/object_inspection/inspection_tree_builders.rb +0 -16
- data/lib/super_diff/active_support/object_inspection/inspection_tree_builders/hash_with_indifferent_access.rb +0 -46
- data/lib/super_diff/active_support/object_inspection/inspection_tree_builders/ordered_options.rb +0 -46
- data/lib/super_diff/active_support/object_inspection/inspection_tree_builders.rb +0 -16
- data/lib/super_diff/colorized_document_extensions.rb +0 -18
- data/lib/super_diff/configuration.rb +0 -149
- data/lib/super_diff/diff_formatters/collection.rb +0 -132
- data/lib/super_diff/diff_formatters/multiline_string.rb +0 -31
- data/lib/super_diff/differs/array.rb +0 -15
- data/lib/super_diff/differs/custom_object.rb +0 -17
- data/lib/super_diff/differs/date_like.rb +0 -15
- data/lib/super_diff/differs/default_object.rb +0 -19
- data/lib/super_diff/differs/defaults.rb +0 -13
- data/lib/super_diff/differs/empty.rb +0 -13
- data/lib/super_diff/differs/hash.rb +0 -15
- data/lib/super_diff/differs/main.rb +0 -31
- data/lib/super_diff/differs/multiline_string.rb +0 -16
- data/lib/super_diff/differs/time_like.rb +0 -15
- data/lib/super_diff/gem_version.rb +0 -45
- data/lib/super_diff/helpers.rb +0 -86
- data/lib/super_diff/implementation_checks.rb +0 -19
- data/lib/super_diff/line.rb +0 -83
- data/lib/super_diff/object_inspection/inspection_tree_builders/base.rb +0 -27
- data/lib/super_diff/object_inspection/inspection_tree_builders/defaults.rb +0 -15
- data/lib/super_diff/object_inspection/inspection_tree_builders/main.rb +0 -30
- data/lib/super_diff/object_inspection/inspection_tree_builders.rb +0 -48
- data/lib/super_diff/object_inspection/nodes.rb +0 -50
- data/lib/super_diff/object_inspection/prefix_for_next_node.rb +0 -6
- data/lib/super_diff/object_inspection/prelude_for_next_node.rb +0 -6
- data/lib/super_diff/operation_tree_builders/array.rb +0 -107
- data/lib/super_diff/operation_tree_builders/custom_object.rb +0 -40
- data/lib/super_diff/operation_tree_builders/date_like.rb +0 -15
- data/lib/super_diff/operation_tree_builders/default_object.rb +0 -119
- data/lib/super_diff/operation_tree_builders/defaults.rb +0 -5
- data/lib/super_diff/operation_tree_builders/hash.rb +0 -218
- data/lib/super_diff/operation_tree_builders/multiline_string.rb +0 -86
- data/lib/super_diff/operation_tree_builders/time_like.rb +0 -24
- data/lib/super_diff/operation_tree_flatteners/array.rb +0 -15
- data/lib/super_diff/operation_tree_flatteners/collection.rb +0 -136
- data/lib/super_diff/operation_tree_flatteners/custom_object.rb +0 -28
- data/lib/super_diff/operation_tree_flatteners/default_object.rb +0 -31
- data/lib/super_diff/operation_tree_flatteners/hash.rb +0 -33
- data/lib/super_diff/operation_tree_flatteners/multiline_string.rb +0 -18
- data/lib/super_diff/operation_trees/array.rb +0 -15
- data/lib/super_diff/operation_trees/custom_object.rb +0 -15
- data/lib/super_diff/operation_trees/default_object.rb +0 -40
- data/lib/super_diff/operation_trees/defaults.rb +0 -5
- data/lib/super_diff/operation_trees/hash.rb +0 -15
- data/lib/super_diff/operation_trees/main.rb +0 -35
- data/lib/super_diff/operation_trees/multiline_string.rb +0 -15
- data/lib/super_diff/recursion_guard.rb +0 -50
- data/lib/super_diff/rspec/object_inspection/inspection_tree_builders/collection_containing_exactly.rb +0 -36
- data/lib/super_diff/rspec/object_inspection/inspection_tree_builders/collection_including.rb +0 -42
- data/lib/super_diff/rspec/object_inspection/inspection_tree_builders/double.rb +0 -102
- data/lib/super_diff/rspec/object_inspection/inspection_tree_builders/generic_describable_matcher.rb +0 -19
- data/lib/super_diff/rspec/object_inspection/inspection_tree_builders/hash_including.rb +0 -42
- data/lib/super_diff/rspec/object_inspection/inspection_tree_builders/instance_of.rb +0 -27
- data/lib/super_diff/rspec/object_inspection/inspection_tree_builders/kind_of.rb +0 -27
- data/lib/super_diff/rspec/object_inspection/inspection_tree_builders/object_having_attributes.rb +0 -36
- data/lib/super_diff/rspec/object_inspection/inspection_tree_builders/primitive.rb +0 -10
- data/lib/super_diff/rspec/object_inspection/inspection_tree_builders/value_within.rb +0 -32
- data/lib/super_diff/rspec/object_inspection/inspection_tree_builders.rb +0 -48
- data/lib/super_diff/tiered_lines.rb +0 -4
- data/lib/super_diff/tiered_lines_elider.rb +0 -462
- data/lib/super_diff/tiered_lines_formatter.rb +0 -75
| @@ -1,119 +0,0 @@ | |
| 1 | 
            -
            module SuperDiff
         | 
| 2 | 
            -
              module OperationTreeBuilders
         | 
| 3 | 
            -
                class DefaultObject < Base
         | 
| 4 | 
            -
                  def self.applies_to?(_expected, _actual)
         | 
| 5 | 
            -
                    true
         | 
| 6 | 
            -
                  end
         | 
| 7 | 
            -
             | 
| 8 | 
            -
                  def initialize(*args)
         | 
| 9 | 
            -
                    super(*args)
         | 
| 10 | 
            -
             | 
| 11 | 
            -
                    establish_expected_and_actual_attributes
         | 
| 12 | 
            -
                  end
         | 
| 13 | 
            -
             | 
| 14 | 
            -
                  protected
         | 
| 15 | 
            -
             | 
| 16 | 
            -
                  def unary_operations
         | 
| 17 | 
            -
                    attribute_names.reduce([]) do |operations, name|
         | 
| 18 | 
            -
                      possibly_add_noop_operation_to(operations, name)
         | 
| 19 | 
            -
                      possibly_add_delete_operation_to(operations, name)
         | 
| 20 | 
            -
                      possibly_add_insert_operation_to(operations, name)
         | 
| 21 | 
            -
                      operations
         | 
| 22 | 
            -
                    end
         | 
| 23 | 
            -
                  end
         | 
| 24 | 
            -
             | 
| 25 | 
            -
                  def build_operation_tree
         | 
| 26 | 
            -
                    # XXX This assumes that `expected` and `actual` are the same
         | 
| 27 | 
            -
                    # TODO: Does this need to be find_operation_tree_for?
         | 
| 28 | 
            -
                    OperationTrees::DefaultObject.new([], underlying_object: actual)
         | 
| 29 | 
            -
                  end
         | 
| 30 | 
            -
             | 
| 31 | 
            -
                  def find_operation_tree_for(value)
         | 
| 32 | 
            -
                    OperationTrees::Main.call(value)
         | 
| 33 | 
            -
                  end
         | 
| 34 | 
            -
             | 
| 35 | 
            -
                  def attribute_names
         | 
| 36 | 
            -
                    (
         | 
| 37 | 
            -
                      expected.instance_variables.sort & actual.instance_variables.sort
         | 
| 38 | 
            -
                    ).map { |variable_name| variable_name[1..-1] }
         | 
| 39 | 
            -
                  end
         | 
| 40 | 
            -
             | 
| 41 | 
            -
                  private
         | 
| 42 | 
            -
             | 
| 43 | 
            -
                  attr_reader :expected_attributes, :actual_attributes
         | 
| 44 | 
            -
             | 
| 45 | 
            -
                  def establish_expected_and_actual_attributes
         | 
| 46 | 
            -
                    @expected_attributes =
         | 
| 47 | 
            -
                      attribute_names.reduce({}) do |hash, name|
         | 
| 48 | 
            -
                        hash.merge(name => expected.instance_variable_get("@#{name}"))
         | 
| 49 | 
            -
                      end
         | 
| 50 | 
            -
             | 
| 51 | 
            -
                    @actual_attributes =
         | 
| 52 | 
            -
                      attribute_names.reduce({}) do |hash, name|
         | 
| 53 | 
            -
                        hash.merge(name => actual.instance_variable_get("@#{name}"))
         | 
| 54 | 
            -
                      end
         | 
| 55 | 
            -
                  end
         | 
| 56 | 
            -
             | 
| 57 | 
            -
                  def possibly_add_noop_operation_to(operations, attribute_name)
         | 
| 58 | 
            -
                    if should_add_noop_operation?(attribute_name)
         | 
| 59 | 
            -
                      operations << Operations::UnaryOperation.new(
         | 
| 60 | 
            -
                        name: :noop,
         | 
| 61 | 
            -
                        collection: actual_attributes,
         | 
| 62 | 
            -
                        key: attribute_name,
         | 
| 63 | 
            -
                        index: attribute_names.index(attribute_name),
         | 
| 64 | 
            -
                        value: actual_attributes[attribute_name]
         | 
| 65 | 
            -
                      )
         | 
| 66 | 
            -
                    end
         | 
| 67 | 
            -
                  end
         | 
| 68 | 
            -
             | 
| 69 | 
            -
                  def should_add_noop_operation?(attribute_name)
         | 
| 70 | 
            -
                    expected_attributes.include?(attribute_name) &&
         | 
| 71 | 
            -
                      actual_attributes.include?(attribute_name) &&
         | 
| 72 | 
            -
                      expected_attributes[attribute_name] ==
         | 
| 73 | 
            -
                        actual_attributes[attribute_name]
         | 
| 74 | 
            -
                  end
         | 
| 75 | 
            -
             | 
| 76 | 
            -
                  def possibly_add_delete_operation_to(operations, attribute_name)
         | 
| 77 | 
            -
                    if should_add_delete_operation?(attribute_name)
         | 
| 78 | 
            -
                      operations << Operations::UnaryOperation.new(
         | 
| 79 | 
            -
                        name: :delete,
         | 
| 80 | 
            -
                        collection: expected_attributes,
         | 
| 81 | 
            -
                        key: attribute_name,
         | 
| 82 | 
            -
                        index: attribute_names.index(attribute_name),
         | 
| 83 | 
            -
                        value: expected_attributes[attribute_name]
         | 
| 84 | 
            -
                      )
         | 
| 85 | 
            -
                    end
         | 
| 86 | 
            -
                  end
         | 
| 87 | 
            -
             | 
| 88 | 
            -
                  def should_add_delete_operation?(attribute_name)
         | 
| 89 | 
            -
                    expected_attributes.include?(attribute_name) &&
         | 
| 90 | 
            -
                      (
         | 
| 91 | 
            -
                        !actual_attributes.include?(attribute_name) ||
         | 
| 92 | 
            -
                          expected_attributes[attribute_name] !=
         | 
| 93 | 
            -
                            actual_attributes[attribute_name]
         | 
| 94 | 
            -
                      )
         | 
| 95 | 
            -
                  end
         | 
| 96 | 
            -
             | 
| 97 | 
            -
                  def possibly_add_insert_operation_to(operations, attribute_name)
         | 
| 98 | 
            -
                    if should_add_insert_operation?(attribute_name)
         | 
| 99 | 
            -
                      operations << Operations::UnaryOperation.new(
         | 
| 100 | 
            -
                        name: :insert,
         | 
| 101 | 
            -
                        collection: actual_attributes,
         | 
| 102 | 
            -
                        key: attribute_name,
         | 
| 103 | 
            -
                        index: attribute_names.index(attribute_name),
         | 
| 104 | 
            -
                        value: actual_attributes[attribute_name]
         | 
| 105 | 
            -
                      )
         | 
| 106 | 
            -
                    end
         | 
| 107 | 
            -
                  end
         | 
| 108 | 
            -
             | 
| 109 | 
            -
                  def should_add_insert_operation?(attribute_name)
         | 
| 110 | 
            -
                    !expected_attributes.include?(attribute_name) ||
         | 
| 111 | 
            -
                      (
         | 
| 112 | 
            -
                        actual_attributes.include?(attribute_name) &&
         | 
| 113 | 
            -
                          expected_attributes[attribute_name] !=
         | 
| 114 | 
            -
                            actual_attributes[attribute_name]
         | 
| 115 | 
            -
                      )
         | 
| 116 | 
            -
                  end
         | 
| 117 | 
            -
                end
         | 
| 118 | 
            -
              end
         | 
| 119 | 
            -
            end
         | 
| @@ -1,218 +0,0 @@ | |
| 1 | 
            -
            module SuperDiff
         | 
| 2 | 
            -
              module OperationTreeBuilders
         | 
| 3 | 
            -
                class Hash < Base
         | 
| 4 | 
            -
                  def self.applies_to?(expected, actual)
         | 
| 5 | 
            -
                    expected.is_a?(::Hash) && actual.is_a?(::Hash)
         | 
| 6 | 
            -
                  end
         | 
| 7 | 
            -
             | 
| 8 | 
            -
                  protected
         | 
| 9 | 
            -
             | 
| 10 | 
            -
                  def unary_operations
         | 
| 11 | 
            -
                    unary_operations_using_variant_of_patience_algorithm
         | 
| 12 | 
            -
                  end
         | 
| 13 | 
            -
             | 
| 14 | 
            -
                  def build_operation_tree
         | 
| 15 | 
            -
                    OperationTrees::Hash.new([])
         | 
| 16 | 
            -
                  end
         | 
| 17 | 
            -
             | 
| 18 | 
            -
                  private
         | 
| 19 | 
            -
             | 
| 20 | 
            -
                  def unary_operations_using_variant_of_patience_algorithm
         | 
| 21 | 
            -
                    operations = []
         | 
| 22 | 
            -
                    aks, eks = actual.keys, expected.keys
         | 
| 23 | 
            -
                    previous_ei, ei = nil, 0
         | 
| 24 | 
            -
                    ai = 0
         | 
| 25 | 
            -
             | 
| 26 | 
            -
                    # When diffing a hash, we're more interested in the 'actual' version
         | 
| 27 | 
            -
                    # than the 'expected' version, because that's the ultimate truth.
         | 
| 28 | 
            -
                    # Therefore, the diff is presented from the perspective of the 'actual'
         | 
| 29 | 
            -
                    # hash, and we start off by looping over it.
         | 
| 30 | 
            -
                    while ai < aks.size
         | 
| 31 | 
            -
                      ak = aks[ai]
         | 
| 32 | 
            -
                      av, ev = actual[ak], expected[ak]
         | 
| 33 | 
            -
                      # While we iterate over 'actual' in order, we jump all over
         | 
| 34 | 
            -
                      # 'expected', trying to match up its keys with the keys in 'actual' as
         | 
| 35 | 
            -
                      # much as possible.
         | 
| 36 | 
            -
                      ei = eks.index(ak)
         | 
| 37 | 
            -
             | 
| 38 | 
            -
                      if should_add_noop_operation?(ak)
         | 
| 39 | 
            -
                        # (If we're here, it probably means that the key we're pointing to
         | 
| 40 | 
            -
                        # in the 'actual' and 'expected' hashes have the same value.)
         | 
| 41 | 
            -
             | 
| 42 | 
            -
                        if ei && previous_ei && (ei - previous_ei) > 1
         | 
| 43 | 
            -
                          # If we've jumped from one operation in the 'expected' hash to
         | 
| 44 | 
            -
                          # another operation later in 'expected' (due to the fact that the
         | 
| 45 | 
            -
                          # 'expected' hash is in a different order than 'actual'), collect
         | 
| 46 | 
            -
                          # any delete operations in between and add them to our operations
         | 
| 47 | 
            -
                          # array as deletes before adding the noop. If we don't do this
         | 
| 48 | 
            -
                          # now, then those deletes will disappear. (Again, we are mainly
         | 
| 49 | 
            -
                          # iterating over 'actual', so this is the only way to catch all of
         | 
| 50 | 
            -
                          # the keys in 'expected'.)
         | 
| 51 | 
            -
                          (previous_ei + 1).upto(ei - 1) do |ei2|
         | 
| 52 | 
            -
                            ek = eks[ei2]
         | 
| 53 | 
            -
                            ev2, av2 = expected[ek], actual[ek]
         | 
| 54 | 
            -
             | 
| 55 | 
            -
                            if (
         | 
| 56 | 
            -
                                 (!actual.include?(ek) || ev2 != av2) &&
         | 
| 57 | 
            -
                                   operations.none? do |operation|
         | 
| 58 | 
            -
                                     %i[delete noop].include?(operation.name) &&
         | 
| 59 | 
            -
                                       operation.key == ek
         | 
| 60 | 
            -
                                   end
         | 
| 61 | 
            -
                               )
         | 
| 62 | 
            -
                              operations << Operations::UnaryOperation.new(
         | 
| 63 | 
            -
                                name: :delete,
         | 
| 64 | 
            -
                                collection: expected,
         | 
| 65 | 
            -
                                key: ek,
         | 
| 66 | 
            -
                                value: ev2,
         | 
| 67 | 
            -
                                index: ei2
         | 
| 68 | 
            -
                              )
         | 
| 69 | 
            -
                            end
         | 
| 70 | 
            -
                          end
         | 
| 71 | 
            -
                        end
         | 
| 72 | 
            -
             | 
| 73 | 
            -
                        operations << Operations::UnaryOperation.new(
         | 
| 74 | 
            -
                          name: :noop,
         | 
| 75 | 
            -
                          collection: actual,
         | 
| 76 | 
            -
                          key: ak,
         | 
| 77 | 
            -
                          value: av,
         | 
| 78 | 
            -
                          index: ai
         | 
| 79 | 
            -
                        )
         | 
| 80 | 
            -
                      else
         | 
| 81 | 
            -
                        # (If we're here, it probably means that the key in 'actual' isn't
         | 
| 82 | 
            -
                        # present in 'expected' or the values don't match.)
         | 
| 83 | 
            -
             | 
| 84 | 
            -
                        if (
         | 
| 85 | 
            -
                             (operations.empty? || operations.last.name == :noop) &&
         | 
| 86 | 
            -
                               (ai == 0 || eks.include?(aks[ai - 1]))
         | 
| 87 | 
            -
                           )
         | 
| 88 | 
            -
                          # If we go from a match in the last iteration to a missing or
         | 
| 89 | 
            -
                          # extra key in this one, or we're at the first key in 'actual' and
         | 
| 90 | 
            -
                          # it's missing or extra, look for deletes in the 'expected' hash
         | 
| 91 | 
            -
                          # and add them to our list of operations before we add the
         | 
| 92 | 
            -
                          # inserts. In most cases we will accomplish this by backtracking a
         | 
| 93 | 
            -
                          # bit to the key in 'expected' that matched the key in 'actual' we
         | 
| 94 | 
            -
                          # processed in the previous iteration (or just the first key in
         | 
| 95 | 
            -
                          # 'expected' if this is the first key in 'actual'), and then
         | 
| 96 | 
            -
                          # iterating from there through 'expected' until we reach the end
         | 
| 97 | 
            -
                          # or we hit some other condition (see below).
         | 
| 98 | 
            -
             | 
| 99 | 
            -
                          start_index =
         | 
| 100 | 
            -
                            if ai > 0
         | 
| 101 | 
            -
                              eks.index(aks[ai - 1]) + 1
         | 
| 102 | 
            -
                            else
         | 
| 103 | 
            -
                              0
         | 
| 104 | 
            -
                            end
         | 
| 105 | 
            -
             | 
| 106 | 
            -
                          start_index.upto(eks.size - 1) do |ei2|
         | 
| 107 | 
            -
                            ek = eks[ei2]
         | 
| 108 | 
            -
                            ev, av2 = expected[ek], actual[ek]
         | 
| 109 | 
            -
             | 
| 110 | 
            -
                            if actual.include?(ek) && ev == av2
         | 
| 111 | 
            -
                              # If the key in 'expected' we've landed on happens to be a
         | 
| 112 | 
            -
                              # match in 'actual', then stop, because it's going to be
         | 
| 113 | 
            -
                              # handled in some future iteration of the 'actual' loop.
         | 
| 114 | 
            -
                              break
         | 
| 115 | 
            -
                            elsif (
         | 
| 116 | 
            -
                                  aks[ai + 1..-1].any? do |k|
         | 
| 117 | 
            -
                                    expected.include?(k) && expected[k] != actual[k]
         | 
| 118 | 
            -
                                  end
         | 
| 119 | 
            -
                                )
         | 
| 120 | 
            -
                              # While we backtracked a bit to iterate over 'expected', we
         | 
| 121 | 
            -
                              # now have to look ahead. If we will end up encountering a
         | 
| 122 | 
            -
                              # insert that matches this delete later, stop and go back to
         | 
| 123 | 
            -
                              # iterating over 'actual'. This is because the delete we would
         | 
| 124 | 
            -
                              # have added now will be added later when we encounter the
         | 
| 125 | 
            -
                              # associated insert, so we don't want to add it twice.
         | 
| 126 | 
            -
                              break
         | 
| 127 | 
            -
                            else
         | 
| 128 | 
            -
                              operations << Operations::UnaryOperation.new(
         | 
| 129 | 
            -
                                name: :delete,
         | 
| 130 | 
            -
                                collection: expected,
         | 
| 131 | 
            -
                                key: ek,
         | 
| 132 | 
            -
                                value: ev,
         | 
| 133 | 
            -
                                index: ei2
         | 
| 134 | 
            -
                              )
         | 
| 135 | 
            -
                            end
         | 
| 136 | 
            -
             | 
| 137 | 
            -
                            if ek == ak && ev != av
         | 
| 138 | 
            -
                              # If we're pointing to the same key in 'expected' as in
         | 
| 139 | 
            -
                              # 'actual', but with different values, go ahead and add an
         | 
| 140 | 
            -
                              # insert now to accompany the delete added above. That way
         | 
| 141 | 
            -
                              # they appear together, which will be easier to read.
         | 
| 142 | 
            -
                              operations << Operations::UnaryOperation.new(
         | 
| 143 | 
            -
                                name: :insert,
         | 
| 144 | 
            -
                                collection: actual,
         | 
| 145 | 
            -
                                key: ak,
         | 
| 146 | 
            -
                                value: av,
         | 
| 147 | 
            -
                                index: ai
         | 
| 148 | 
            -
                              )
         | 
| 149 | 
            -
                            end
         | 
| 150 | 
            -
                          end
         | 
| 151 | 
            -
                        end
         | 
| 152 | 
            -
             | 
| 153 | 
            -
                        if (
         | 
| 154 | 
            -
                             expected.include?(ak) && ev != av &&
         | 
| 155 | 
            -
                               operations.none? { |op| op.name == :delete && op.key == ak }
         | 
| 156 | 
            -
                           )
         | 
| 157 | 
            -
                          # If we're here, it means that we didn't encounter any delete
         | 
| 158 | 
            -
                          # operations above for whatever reason and so we need to add a
         | 
| 159 | 
            -
                          # delete to represent the fact that the value for this key has
         | 
| 160 | 
            -
                          # changed.
         | 
| 161 | 
            -
                          operations << Operations::UnaryOperation.new(
         | 
| 162 | 
            -
                            name: :delete,
         | 
| 163 | 
            -
                            collection: expected,
         | 
| 164 | 
            -
                            key: ak,
         | 
| 165 | 
            -
                            value: expected[ak],
         | 
| 166 | 
            -
                            index: ei
         | 
| 167 | 
            -
                          )
         | 
| 168 | 
            -
                        end
         | 
| 169 | 
            -
             | 
| 170 | 
            -
                        if operations.none? { |op| op.name == :insert && op.key == ak }
         | 
| 171 | 
            -
                          # If we're here, it means that we didn't encounter any insert
         | 
| 172 | 
            -
                          # operations above. Since we already handled delete, the only
         | 
| 173 | 
            -
                          # alternative is that this key must not exist in 'expected', so
         | 
| 174 | 
            -
                          # we need to add an insert.
         | 
| 175 | 
            -
                          operations << Operations::UnaryOperation.new(
         | 
| 176 | 
            -
                            name: :insert,
         | 
| 177 | 
            -
                            collection: actual,
         | 
| 178 | 
            -
                            key: ak,
         | 
| 179 | 
            -
                            value: av,
         | 
| 180 | 
            -
                            index: ai
         | 
| 181 | 
            -
                          )
         | 
| 182 | 
            -
                        end
         | 
| 183 | 
            -
                      end
         | 
| 184 | 
            -
             | 
| 185 | 
            -
                      ai += 1
         | 
| 186 | 
            -
                      previous_ei = ei
         | 
| 187 | 
            -
                    end
         | 
| 188 | 
            -
             | 
| 189 | 
            -
                    # The last thing to do is this: if there are keys in 'expected' that
         | 
| 190 | 
            -
                    # aren't in 'actual', and they aren't associated with any inserts to
         | 
| 191 | 
            -
                    # where they would have been added above, tack those deletes onto the
         | 
| 192 | 
            -
                    # end of our operations array.
         | 
| 193 | 
            -
                    (eks - aks - operations.map(&:key)).each do |ek|
         | 
| 194 | 
            -
                      ei = eks.index(ek)
         | 
| 195 | 
            -
                      ev = expected[ek]
         | 
| 196 | 
            -
             | 
| 197 | 
            -
                      operations << Operations::UnaryOperation.new(
         | 
| 198 | 
            -
                        name: :delete,
         | 
| 199 | 
            -
                        collection: expected,
         | 
| 200 | 
            -
                        key: ek,
         | 
| 201 | 
            -
                        value: ev,
         | 
| 202 | 
            -
                        index: ei
         | 
| 203 | 
            -
                      )
         | 
| 204 | 
            -
                    end
         | 
| 205 | 
            -
             | 
| 206 | 
            -
                    operations
         | 
| 207 | 
            -
                  end
         | 
| 208 | 
            -
             | 
| 209 | 
            -
                  def should_add_noop_operation?(key)
         | 
| 210 | 
            -
                    expected.include?(key) && expected[key] == actual[key]
         | 
| 211 | 
            -
                  end
         | 
| 212 | 
            -
             | 
| 213 | 
            -
                  def all_keys
         | 
| 214 | 
            -
                    actual.keys | expected.keys
         | 
| 215 | 
            -
                  end
         | 
| 216 | 
            -
                end
         | 
| 217 | 
            -
              end
         | 
| 218 | 
            -
            end
         | 
| @@ -1,86 +0,0 @@ | |
| 1 | 
            -
            module SuperDiff
         | 
| 2 | 
            -
              module OperationTreeBuilders
         | 
| 3 | 
            -
                class MultilineString < Base
         | 
| 4 | 
            -
                  def self.applies_to?(expected, actual)
         | 
| 5 | 
            -
                    expected.is_a?(::String) && actual.is_a?(::String) &&
         | 
| 6 | 
            -
                      (expected.include?("\n") || actual.include?("\n"))
         | 
| 7 | 
            -
                  end
         | 
| 8 | 
            -
             | 
| 9 | 
            -
                  def initialize(*args)
         | 
| 10 | 
            -
                    super(*args)
         | 
| 11 | 
            -
             | 
| 12 | 
            -
                    @original_expected = @expected
         | 
| 13 | 
            -
                    @original_actual = @actual
         | 
| 14 | 
            -
                    @expected = split_into_lines(@expected)
         | 
| 15 | 
            -
                    @actual = split_into_lines(@actual)
         | 
| 16 | 
            -
                    @sequence_matcher = PatienceDiff::SequenceMatcher.new
         | 
| 17 | 
            -
                  end
         | 
| 18 | 
            -
             | 
| 19 | 
            -
                  protected
         | 
| 20 | 
            -
             | 
| 21 | 
            -
                  def unary_operations
         | 
| 22 | 
            -
                    opcodes.flat_map do |code, a_start, a_end, b_start, b_end|
         | 
| 23 | 
            -
                      if code == :delete
         | 
| 24 | 
            -
                        add_delete_operations(a_start..a_end)
         | 
| 25 | 
            -
                      elsif code == :insert
         | 
| 26 | 
            -
                        add_insert_operations(b_start..b_end)
         | 
| 27 | 
            -
                      else
         | 
| 28 | 
            -
                        add_noop_operations(b_start..b_end)
         | 
| 29 | 
            -
                      end
         | 
| 30 | 
            -
                    end
         | 
| 31 | 
            -
                  end
         | 
| 32 | 
            -
             | 
| 33 | 
            -
                  def build_operation_tree
         | 
| 34 | 
            -
                    OperationTrees::MultilineString.new([])
         | 
| 35 | 
            -
                  end
         | 
| 36 | 
            -
             | 
| 37 | 
            -
                  private
         | 
| 38 | 
            -
             | 
| 39 | 
            -
                  attr_reader :sequence_matcher, :original_expected, :original_actual
         | 
| 40 | 
            -
             | 
| 41 | 
            -
                  def split_into_lines(string)
         | 
| 42 | 
            -
                    string.scan(/.+(?:\r|\n|\r\n|\Z)/)
         | 
| 43 | 
            -
                  end
         | 
| 44 | 
            -
             | 
| 45 | 
            -
                  def opcodes
         | 
| 46 | 
            -
                    sequence_matcher.diff_opcodes(expected, actual)
         | 
| 47 | 
            -
                  end
         | 
| 48 | 
            -
             | 
| 49 | 
            -
                  def add_delete_operations(indices)
         | 
| 50 | 
            -
                    indices.map do |index|
         | 
| 51 | 
            -
                      Operations::UnaryOperation.new(
         | 
| 52 | 
            -
                        name: :delete,
         | 
| 53 | 
            -
                        collection: expected,
         | 
| 54 | 
            -
                        key: index,
         | 
| 55 | 
            -
                        index: index,
         | 
| 56 | 
            -
                        value: expected[index]
         | 
| 57 | 
            -
                      )
         | 
| 58 | 
            -
                    end
         | 
| 59 | 
            -
                  end
         | 
| 60 | 
            -
             | 
| 61 | 
            -
                  def add_insert_operations(indices)
         | 
| 62 | 
            -
                    indices.map do |index|
         | 
| 63 | 
            -
                      Operations::UnaryOperation.new(
         | 
| 64 | 
            -
                        name: :insert,
         | 
| 65 | 
            -
                        collection: actual,
         | 
| 66 | 
            -
                        key: index,
         | 
| 67 | 
            -
                        index: index,
         | 
| 68 | 
            -
                        value: actual[index]
         | 
| 69 | 
            -
                      )
         | 
| 70 | 
            -
                    end
         | 
| 71 | 
            -
                  end
         | 
| 72 | 
            -
             | 
| 73 | 
            -
                  def add_noop_operations(indices)
         | 
| 74 | 
            -
                    indices.map do |index|
         | 
| 75 | 
            -
                      Operations::UnaryOperation.new(
         | 
| 76 | 
            -
                        name: :noop,
         | 
| 77 | 
            -
                        collection: actual,
         | 
| 78 | 
            -
                        key: index,
         | 
| 79 | 
            -
                        index: index,
         | 
| 80 | 
            -
                        value: actual[index]
         | 
| 81 | 
            -
                      )
         | 
| 82 | 
            -
                    end
         | 
| 83 | 
            -
                  end
         | 
| 84 | 
            -
                end
         | 
| 85 | 
            -
              end
         | 
| 86 | 
            -
            end
         | 
| @@ -1,24 +0,0 @@ | |
| 1 | 
            -
            module SuperDiff
         | 
| 2 | 
            -
              module OperationTreeBuilders
         | 
| 3 | 
            -
                class TimeLike < CustomObject
         | 
| 4 | 
            -
                  def self.applies_to?(expected, actual)
         | 
| 5 | 
            -
                    SuperDiff.time_like?(expected) && SuperDiff.time_like?(actual)
         | 
| 6 | 
            -
                  end
         | 
| 7 | 
            -
             | 
| 8 | 
            -
                  protected
         | 
| 9 | 
            -
             | 
| 10 | 
            -
                  def attribute_names
         | 
| 11 | 
            -
                    base = %w[year month day hour min sec subsec zone utc_offset]
         | 
| 12 | 
            -
             | 
| 13 | 
            -
                    # If timezones are different, also show a normalized timestamp at the
         | 
| 14 | 
            -
                    # end of the diff to help visualize why they are different moments in
         | 
| 15 | 
            -
                    # time.
         | 
| 16 | 
            -
                    if actual.zone != expected.zone
         | 
| 17 | 
            -
                      base + ["utc"]
         | 
| 18 | 
            -
                    else
         | 
| 19 | 
            -
                      base
         | 
| 20 | 
            -
                    end
         | 
| 21 | 
            -
                  end
         | 
| 22 | 
            -
                end
         | 
| 23 | 
            -
              end
         | 
| 24 | 
            -
            end
         | 
| @@ -1,136 +0,0 @@ | |
| 1 | 
            -
            module SuperDiff
         | 
| 2 | 
            -
              module OperationTreeFlatteners
         | 
| 3 | 
            -
                class Collection < Base
         | 
| 4 | 
            -
                  protected
         | 
| 5 | 
            -
             | 
| 6 | 
            -
                  def build_tiered_lines
         | 
| 7 | 
            -
                    [
         | 
| 8 | 
            -
                      Line.new(
         | 
| 9 | 
            -
                        type: :noop,
         | 
| 10 | 
            -
                        indentation_level: indentation_level,
         | 
| 11 | 
            -
                        value: open_token,
         | 
| 12 | 
            -
                        collection_bookend: :open
         | 
| 13 | 
            -
                      ),
         | 
| 14 | 
            -
                      *inner_lines,
         | 
| 15 | 
            -
                      Line.new(
         | 
| 16 | 
            -
                        type: :noop,
         | 
| 17 | 
            -
                        indentation_level: indentation_level,
         | 
| 18 | 
            -
                        value: close_token,
         | 
| 19 | 
            -
                        collection_bookend: :close
         | 
| 20 | 
            -
                      )
         | 
| 21 | 
            -
                    ]
         | 
| 22 | 
            -
                  end
         | 
| 23 | 
            -
             | 
| 24 | 
            -
                  def inner_lines
         | 
| 25 | 
            -
                    @_inner_lines ||=
         | 
| 26 | 
            -
                      operation_tree.flat_map do |operation|
         | 
| 27 | 
            -
                        lines =
         | 
| 28 | 
            -
                          if operation.name == :change
         | 
| 29 | 
            -
                            build_lines_for_change_operation(operation)
         | 
| 30 | 
            -
                          else
         | 
| 31 | 
            -
                            build_lines_for_non_change_operation(operation)
         | 
| 32 | 
            -
                          end
         | 
| 33 | 
            -
             | 
| 34 | 
            -
                        maybe_add_prefix_at_beginning_of_lines(
         | 
| 35 | 
            -
                          maybe_add_comma_at_end_of_lines(lines, operation),
         | 
| 36 | 
            -
                          operation
         | 
| 37 | 
            -
                        )
         | 
| 38 | 
            -
                      end
         | 
| 39 | 
            -
                  end
         | 
| 40 | 
            -
             | 
| 41 | 
            -
                  def maybe_add_prefix_at_beginning_of_lines(lines, operation)
         | 
| 42 | 
            -
                    if add_prefix_at_beginning_of_lines?(operation)
         | 
| 43 | 
            -
                      add_prefix_at_beginning_of_lines(lines, operation)
         | 
| 44 | 
            -
                    else
         | 
| 45 | 
            -
                      lines
         | 
| 46 | 
            -
                    end
         | 
| 47 | 
            -
                  end
         | 
| 48 | 
            -
             | 
| 49 | 
            -
                  def add_prefix_at_beginning_of_lines?(operation)
         | 
| 50 | 
            -
                    !!item_prefix_for(operation)
         | 
| 51 | 
            -
                  end
         | 
| 52 | 
            -
             | 
| 53 | 
            -
                  def add_prefix_at_beginning_of_lines(lines, operation)
         | 
| 54 | 
            -
                    [lines[0].prefixed_with(item_prefix_for(operation))] + lines[1..-1]
         | 
| 55 | 
            -
                  end
         | 
| 56 | 
            -
             | 
| 57 | 
            -
                  def maybe_add_comma_at_end_of_lines(lines, operation)
         | 
| 58 | 
            -
                    if last_item_in_collection?(operation)
         | 
| 59 | 
            -
                      lines
         | 
| 60 | 
            -
                    else
         | 
| 61 | 
            -
                      add_comma_at_end_of_lines(lines)
         | 
| 62 | 
            -
                    end
         | 
| 63 | 
            -
                  end
         | 
| 64 | 
            -
             | 
| 65 | 
            -
                  def last_item_in_collection?(operation)
         | 
| 66 | 
            -
                    if operation.name == :change
         | 
| 67 | 
            -
                      operation.left_index == operation.left_collection.size - 1 &&
         | 
| 68 | 
            -
                        operation.right_index == operation.right_collection.size - 1
         | 
| 69 | 
            -
                    else
         | 
| 70 | 
            -
                      operation.index == operation.collection.size - 1
         | 
| 71 | 
            -
                    end
         | 
| 72 | 
            -
                  end
         | 
| 73 | 
            -
             | 
| 74 | 
            -
                  def add_comma_at_end_of_lines(lines)
         | 
| 75 | 
            -
                    lines[0..-2] + [lines[-1].with_comma]
         | 
| 76 | 
            -
                  end
         | 
| 77 | 
            -
             | 
| 78 | 
            -
                  def build_lines_for_change_operation(operation)
         | 
| 79 | 
            -
                    SuperDiff::RecursionGuard.guarding_recursion_of(
         | 
| 80 | 
            -
                      operation.left_collection,
         | 
| 81 | 
            -
                      operation.right_collection
         | 
| 82 | 
            -
                    ) do |already_seen|
         | 
| 83 | 
            -
                      if already_seen
         | 
| 84 | 
            -
                        raise InfiniteRecursionError
         | 
| 85 | 
            -
                      else
         | 
| 86 | 
            -
                        operation.children.flatten(indentation_level: indentation_level + 1)
         | 
| 87 | 
            -
                      end
         | 
| 88 | 
            -
                    end
         | 
| 89 | 
            -
                  end
         | 
| 90 | 
            -
             | 
| 91 | 
            -
                  def build_lines_for_non_change_operation(operation)
         | 
| 92 | 
            -
                    indentation_level = @indentation_level + 1
         | 
| 93 | 
            -
             | 
| 94 | 
            -
                    if recursive_operation?(operation)
         | 
| 95 | 
            -
                      [
         | 
| 96 | 
            -
                        Line.new(
         | 
| 97 | 
            -
                          type: operation.name,
         | 
| 98 | 
            -
                          indentation_level: indentation_level,
         | 
| 99 | 
            -
                          value: SuperDiff::RecursionGuard::PLACEHOLDER
         | 
| 100 | 
            -
                        )
         | 
| 101 | 
            -
                      ]
         | 
| 102 | 
            -
                    else
         | 
| 103 | 
            -
                      build_lines_from_inspection_of(
         | 
| 104 | 
            -
                        operation.value,
         | 
| 105 | 
            -
                        type: operation.name,
         | 
| 106 | 
            -
                        indentation_level: indentation_level
         | 
| 107 | 
            -
                      )
         | 
| 108 | 
            -
                    end
         | 
| 109 | 
            -
                  end
         | 
| 110 | 
            -
             | 
| 111 | 
            -
                  def recursive_operation?(operation)
         | 
| 112 | 
            -
                    operation.value.equal?(operation.collection) ||
         | 
| 113 | 
            -
                      SuperDiff::RecursionGuard.already_seen?(operation.value)
         | 
| 114 | 
            -
                  end
         | 
| 115 | 
            -
             | 
| 116 | 
            -
                  def item_prefix_for(_operation)
         | 
| 117 | 
            -
                    ""
         | 
| 118 | 
            -
                  end
         | 
| 119 | 
            -
             | 
| 120 | 
            -
                  def build_lines_from_inspection_of(value, type:, indentation_level:)
         | 
| 121 | 
            -
                    SuperDiff.inspect_object(
         | 
| 122 | 
            -
                      value,
         | 
| 123 | 
            -
                      as_lines: true,
         | 
| 124 | 
            -
                      type: type,
         | 
| 125 | 
            -
                      indentation_level: indentation_level
         | 
| 126 | 
            -
                    )
         | 
| 127 | 
            -
                  end
         | 
| 128 | 
            -
             | 
| 129 | 
            -
                  class InfiniteRecursionError < StandardError
         | 
| 130 | 
            -
                    def initialize(_message = nil)
         | 
| 131 | 
            -
                      super("Unhandled recursive data structure encountered!")
         | 
| 132 | 
            -
                    end
         | 
| 133 | 
            -
                  end
         | 
| 134 | 
            -
                end
         | 
| 135 | 
            -
              end
         | 
| 136 | 
            -
            end
         | 
| @@ -1,28 +0,0 @@ | |
| 1 | 
            -
            module SuperDiff
         | 
| 2 | 
            -
              module OperationTreeFlatteners
         | 
| 3 | 
            -
                class CustomObject < Collection
         | 
| 4 | 
            -
                  protected
         | 
| 5 | 
            -
             | 
| 6 | 
            -
                  def open_token
         | 
| 7 | 
            -
                    "#<%<class>s {" % { class: operation_tree.underlying_object.class }
         | 
| 8 | 
            -
                  end
         | 
| 9 | 
            -
             | 
| 10 | 
            -
                  def close_token
         | 
| 11 | 
            -
                    "}>"
         | 
| 12 | 
            -
                  end
         | 
| 13 | 
            -
             | 
| 14 | 
            -
                  def item_prefix_for(operation)
         | 
| 15 | 
            -
                    key =
         | 
| 16 | 
            -
                      # Note: We could have used the right_key here too, they're both the
         | 
| 17 | 
            -
                      # same keys
         | 
| 18 | 
            -
                      if operation.respond_to?(:left_key)
         | 
| 19 | 
            -
                        operation.left_key
         | 
| 20 | 
            -
                      else
         | 
| 21 | 
            -
                        operation.key
         | 
| 22 | 
            -
                      end
         | 
| 23 | 
            -
             | 
| 24 | 
            -
                    "#{key}: "
         | 
| 25 | 
            -
                  end
         | 
| 26 | 
            -
                end
         | 
| 27 | 
            -
              end
         | 
| 28 | 
            -
            end
         | 
| @@ -1,31 +0,0 @@ | |
| 1 | 
            -
            module SuperDiff
         | 
| 2 | 
            -
              module OperationTreeFlatteners
         | 
| 3 | 
            -
                class DefaultObject < Collection
         | 
| 4 | 
            -
                  protected
         | 
| 5 | 
            -
             | 
| 6 | 
            -
                  def open_token
         | 
| 7 | 
            -
                    "#<#{operation_tree.underlying_object.class.name}:" +
         | 
| 8 | 
            -
                      SuperDiff::Helpers.object_address_for(
         | 
| 9 | 
            -
                        operation_tree.underlying_object
         | 
| 10 | 
            -
                      ) + " {"
         | 
| 11 | 
            -
                  end
         | 
| 12 | 
            -
             | 
| 13 | 
            -
                  def close_token
         | 
| 14 | 
            -
                    "}>"
         | 
| 15 | 
            -
                  end
         | 
| 16 | 
            -
             | 
| 17 | 
            -
                  def item_prefix_for(operation)
         | 
| 18 | 
            -
                    key =
         | 
| 19 | 
            -
                      # Note: We could have used the right_key here too, they're both the
         | 
| 20 | 
            -
                      # same keys
         | 
| 21 | 
            -
                      if operation.respond_to?(:left_key)
         | 
| 22 | 
            -
                        operation.left_key
         | 
| 23 | 
            -
                      else
         | 
| 24 | 
            -
                        operation.key
         | 
| 25 | 
            -
                      end
         | 
| 26 | 
            -
             | 
| 27 | 
            -
                    "@#{key}="
         | 
| 28 | 
            -
                  end
         | 
| 29 | 
            -
                end
         | 
| 30 | 
            -
              end
         | 
| 31 | 
            -
            end
         | 
| @@ -1,33 +0,0 @@ | |
| 1 | 
            -
            module SuperDiff
         | 
| 2 | 
            -
              module OperationTreeFlatteners
         | 
| 3 | 
            -
                class Hash < Collection
         | 
| 4 | 
            -
                  protected
         | 
| 5 | 
            -
             | 
| 6 | 
            -
                  def open_token
         | 
| 7 | 
            -
                    "{"
         | 
| 8 | 
            -
                  end
         | 
| 9 | 
            -
             | 
| 10 | 
            -
                  def close_token
         | 
| 11 | 
            -
                    "}"
         | 
| 12 | 
            -
                  end
         | 
| 13 | 
            -
             | 
| 14 | 
            -
                  def item_prefix_for(operation)
         | 
| 15 | 
            -
                    key = key_for(operation)
         | 
| 16 | 
            -
             | 
| 17 | 
            -
                    format_keys_as_kwargs? ? "#{key}: " : "#{key.inspect} => "
         | 
| 18 | 
            -
                  end
         | 
| 19 | 
            -
             | 
| 20 | 
            -
                  private
         | 
| 21 | 
            -
             | 
| 22 | 
            -
                  def format_keys_as_kwargs?
         | 
| 23 | 
            -
                    operation_tree.all? { |operation| key_for(operation).is_a?(Symbol) }
         | 
| 24 | 
            -
                  end
         | 
| 25 | 
            -
             | 
| 26 | 
            -
                  def key_for(operation)
         | 
| 27 | 
            -
                    # Note: We could have used the right_key here too, they're both the
         | 
| 28 | 
            -
                    # same keys
         | 
| 29 | 
            -
                    operation.respond_to?(:left_key) ? operation.left_key : operation.key
         | 
| 30 | 
            -
                  end
         | 
| 31 | 
            -
                end
         | 
| 32 | 
            -
              end
         | 
| 33 | 
            -
            end
         |