Class: Lutaml::Model::MappingRule

Inherits:
Object
  • Object
show all
Includes:
DeepDupable
Defined in:
lib/lutaml/model/mapping/mapping_rule.rb

Direct Known Subclasses

KeyValue::MappingRule, Xml::MappingRule

Constant Summary collapse

ALLOWED_OPTIONS =
{
  render_nil: %i[
    omit
    as_nil
    as_blank
    as_empty
  ],
  render_empty: %i[
    omit
    as_empty
    as_blank
    as_nil
  ],
}.freeze
EMPTY_TRANSFORMERS =

Frozen empty array for the common case of no transformers

[].freeze

Instance Attribute Summary collapse

Instance Method Summary collapse

Constructor Details

#initialize(name, to:, to_instance: nil, as_attribute: nil, render_nil: false, render_default: false, render_empty: false, treat_nil: :nil, treat_empty: :empty, treat_omitted: :nil, with: {}, attribute: false, delegate: nil, root_mappings: nil, polymorphic: {}, polymorphic_map: {}, transform: {}, value_map: {}) ⇒ MappingRule

Returns a new instance of MappingRule.



48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
# File 'lib/lutaml/model/mapping/mapping_rule.rb', line 48

def initialize(
  name,
  to:,
  to_instance: nil,
  as_attribute: nil,
  render_nil: false,
  render_default: false,
  render_empty: false,
  treat_nil: :nil,
  treat_empty: :empty,
  treat_omitted: :nil,
  with: {},
  attribute: false,
  delegate: nil,
  root_mappings: nil,
  polymorphic: {},
  polymorphic_map: {},
  transform: {},
  value_map: {}
)
  @name = name
  @to = to
  @to_instance = to_instance
  @as_attribute = as_attribute
  @render_nil = render_nil
  @render_default = render_default
  @render_empty = render_empty
  @treat_nil = treat_nil
  @treat_empty = treat_empty
  @treat_omitted = treat_omitted
  @custom_methods = with
  @attribute = attribute
  @delegate = delegate
  @root_mappings = root_mappings
  @polymorphic = polymorphic
  @polymorphic_map = polymorphic_map
  @transform = transform

  # Cache whether this rule needs the full deserialize chain.
  # Over 95% of rules are "simple" (no custom method, no delegate).
  @needs_full_deserialize = has_custom_method_for_deserialization? || !!delegate

  # Only calculate default_value_map if value_map is not fully provided
  if value_map.empty? || !value_map[:from] || !value_map[:to]
    # Build value_map by starting with defaults from render_nil/render_empty,
    # then overlaying user-provided value_map entries on top.
    # User-provided value_map entries take PRECEDENCE over computed defaults.
    # This ensures that value_map: { to: { empty: :empty } } overrides
    # the default render_empty: false → :omitted behavior.
    vm = {
      from: (value_map[:from] || {}).dup,
      to: (value_map[:to] || {}).dup,
    }
    defaults = default_value_map
    vm[:from] = defaults[:from].merge(vm[:from])
    vm[:to] = defaults[:to].merge(vm[:to])
    @value_map = vm
  else
    # Complete value_map provided (e.g., from deep_dup), use it directly.
    @value_map = value_map
  end

  # Freeze value_map and its inner hashes — they are never mutated after
  # construction, and downstream code reads them on every serialization.
  @value_map[:from].freeze
  @value_map[:to].freeze
  @value_map.freeze
end

Instance Attribute Details

#as_attributeObject (readonly)

Returns the value of attribute as_attribute.



6
7
8
# File 'lib/lutaml/model/mapping/mapping_rule.rb', line 6

def as_attribute
  @as_attribute
end

#attributeObject (readonly) Also known as: attribute?

Returns the value of attribute attribute.



6
7
8
# File 'lib/lutaml/model/mapping/mapping_rule.rb', line 6

def attribute
  @attribute
end

#custom_methodsObject (readonly)

Returns the value of attribute custom_methods.



6
7
8
# File 'lib/lutaml/model/mapping/mapping_rule.rb', line 6

def custom_methods
  @custom_methods
end

#delegateObject (readonly)

Returns the value of attribute delegate.



6
7
8
# File 'lib/lutaml/model/mapping/mapping_rule.rb', line 6

def delegate
  @delegate
end

#formatObject (readonly)

Returns the value of attribute format.



6
7
8
# File 'lib/lutaml/model/mapping/mapping_rule.rb', line 6

def format
  @format
end

#nameObject (readonly) Also known as: from

Returns the value of attribute name.



6
7
8
# File 'lib/lutaml/model/mapping/mapping_rule.rb', line 6

def name
  @name
end

#polymorphicObject (readonly)

Returns the value of attribute polymorphic.



6
7
8
# File 'lib/lutaml/model/mapping/mapping_rule.rb', line 6

def polymorphic
  @polymorphic
end

#polymorphic_mapObject (readonly)

Returns the value of attribute polymorphic_map.



6
7
8
# File 'lib/lutaml/model/mapping/mapping_rule.rb', line 6

def polymorphic_map
  @polymorphic_map
end

#render_defaultObject (readonly) Also known as: render_default?

Returns the value of attribute render_default.



6
7
8
# File 'lib/lutaml/model/mapping/mapping_rule.rb', line 6

def render_default
  @render_default
end

#render_emptyObject (readonly)

Returns the value of attribute render_empty.



6
7
8
# File 'lib/lutaml/model/mapping/mapping_rule.rb', line 6

def render_empty
  @render_empty
end

#render_nilObject (readonly)

Returns the value of attribute render_nil.



6
7
8
# File 'lib/lutaml/model/mapping/mapping_rule.rb', line 6

def render_nil
  @render_nil
end

#toObject (readonly)

Returns the value of attribute to.



6
7
8
# File 'lib/lutaml/model/mapping/mapping_rule.rb', line 6

def to
  @to
end

#to_instanceObject (readonly)

Returns the value of attribute to_instance.



6
7
8
# File 'lib/lutaml/model/mapping/mapping_rule.rb', line 6

def to_instance
  @to_instance
end

#transformObject (readonly)

Returns the value of attribute transform.



6
7
8
# File 'lib/lutaml/model/mapping/mapping_rule.rb', line 6

def transform
  @transform
end

#treat_emptyObject (readonly)

Returns the value of attribute treat_empty.



6
7
8
# File 'lib/lutaml/model/mapping/mapping_rule.rb', line 6

def treat_empty
  @treat_empty
end

#treat_nilObject (readonly)

Returns the value of attribute treat_nil.



6
7
8
# File 'lib/lutaml/model/mapping/mapping_rule.rb', line 6

def treat_nil
  @treat_nil
end

#treat_omittedObject (readonly)

Returns the value of attribute treat_omitted.



6
7
8
# File 'lib/lutaml/model/mapping/mapping_rule.rb', line 6

def treat_omitted
  @treat_omitted
end

Instance Method Details

#can_transform_to?(attribute, format) ⇒ Boolean

Returns:

  • (Boolean)


374
375
376
# File 'lib/lutaml/model/mapping/mapping_rule.rb', line 374

def can_transform_to?(attribute, format)
  get_transformers(attribute).any? { |t| t.can_transform?(:to, format) }
end

#deep_dupObject

Raises:

  • (NotImplementedError)


322
323
324
# File 'lib/lutaml/model/mapping/mapping_rule.rb', line 322

def deep_dup
  raise NotImplementedError, "Subclasses must implement `deep_dup`."
end

#default_value_map(options = {}) ⇒ Object



117
118
119
120
121
122
123
124
125
126
127
128
129
130
# File 'lib/lutaml/model/mapping/mapping_rule.rb', line 117

def default_value_map(options = {})
  render_nil_as = render_as(:render_nil, :omitted, options)
  render_empty_as = render_as(:render_empty, :empty, options)

  treat_nil_as = treat_as(:treat_nil, :nil, options)
  treat_empty_as = treat_as(:treat_empty, :empty, options)
  treat_omitted_as = treat_as(:treat_omitted, :nil, options)

  {
    from: { omitted: treat_omitted_as, nil: treat_nil_as,
            empty: treat_empty_as },
    to: { omitted: :omitted, nil: render_nil_as, empty: render_empty_as },
  }
end

#deserialize(model, value, attributes, mapper_class = nil) ⇒ Object



288
289
290
291
292
293
294
295
296
# File 'lib/lutaml/model/mapping/mapping_rule.rb', line 288

def deserialize(model, value, attributes, mapper_class = nil)
  if @needs_full_deserialize
    handle_custom_method(model, value, mapper_class) ||
      handle_delegate(model, value, attributes) ||
      handle_transform_method(model, value, attributes)
  else
    handle_transform_method(model, value, attributes)
  end
end

#eql?(other) ⇒ Boolean Also known as: ==

Returns:

  • (Boolean)


314
315
316
317
318
319
# File 'lib/lutaml/model/mapping/mapping_rule.rb', line 314

def eql?(other)
  other.class == self.class &&
    instance_variables.all? do |var|
      instance_variable_get(var) == other.instance_variable_get(var)
    end
end

#get_transformers(attribute) ⇒ Object



381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
# File 'lib/lutaml/model/mapping/mapping_rule.rb', line 381

def get_transformers(attribute)
  # Fast path: most rules have no transforms at all
  rule_transform = transform
  attr_transform = attribute&.transform

  if !rule_transform && !attr_transform
    return EMPTY_TRANSFORMERS
  end

  # Build transformer list only when needed
  transformers = []
  transformers << rule_transform if rule_transform
  transformers << attr_transform if attr_transform
  transformers.select! { |t| t.is_a?(Class) }
  transformers.freeze
end

#has_custom_method_for_deserialization?Boolean

Returns:

  • (Boolean)


302
303
304
# File 'lib/lutaml/model/mapping/mapping_rule.rb', line 302

def has_custom_method_for_deserialization?
  !custom_methods.empty? && custom_methods[:from]
end

#has_custom_method_for_serialization?Boolean

Returns:

  • (Boolean)


298
299
300
# File 'lib/lutaml/model/mapping/mapping_rule.rb', line 298

def has_custom_method_for_serialization?
  !custom_methods.empty? && custom_methods[:to]
end

#has_items?(value) ⇒ Boolean

Check if value is a non-empty collection

Returns:

  • (Boolean)


209
210
211
212
213
214
# File 'lib/lutaml/model/mapping/mapping_rule.rb', line 209

def has_items?(value)
  return false if value.nil? || Utils.uninitialized?(value)
  return false unless value.is_a?(String) || value.is_a?(Array) || value.is_a?(Hash)

  !value.empty?
end

#multiple_mappings?Boolean

Returns:

  • (Boolean)


306
307
308
# File 'lib/lutaml/model/mapping/mapping_rule.rb', line 306

def multiple_mappings?
  name.is_a?(Array)
end

#mutated_collection?(value, instance) ⇒ Boolean

Returns:

  • (Boolean)


199
200
201
202
203
204
205
206
# File 'lib/lutaml/model/mapping/mapping_rule.rb', line 199

def mutated_collection?(value, instance)
  return false if value.nil? || Utils.uninitialized?(value)
  return false unless value.is_a?(Array) || value.is_a?(Lutaml::Model::Collection)
  return false if value.empty? # Empty collection is still default

  # If it's a non-empty collection and marked as using_default, it was mutated
  instance.is_a?(Lutaml::Model::Serialize) && instance.using_default?(to)
end

#polymorphic_mapping?Boolean

Returns:

  • (Boolean)


247
248
249
# File 'lib/lutaml/model/mapping/mapping_rule.rb', line 247

def polymorphic_mapping?
  polymorphic_map && !polymorphic_map.empty?
end

#raw_mapping?Boolean

Returns:

  • (Boolean)


310
311
312
# File 'lib/lutaml/model/mapping/mapping_rule.rb', line 310

def raw_mapping?
  name == Constants::RAW_MAPPING_KEY
end

#raw_value_mapObject

Raw value_map hash access (for compiled rules). Use value_map(key, options) for individual lookups with overrides.



328
329
330
# File 'lib/lutaml/model/mapping/mapping_rule.rb', line 328

def raw_value_map
  @value_map
end

#render?(value, instance = nil, options = {}) ⇒ Boolean

Returns:

  • (Boolean)


160
161
162
163
164
165
166
167
168
169
170
171
172
173
# File 'lib/lutaml/model/mapping/mapping_rule.rb', line 160

def render?(value, instance = nil, options = {})
  if invalid_value?(value, options)
    false
  # FIXED: Check if collection was mutated after initialization
  # A non-empty collection initialized with default [] should render if mutated
  # This handles the case where collection is mutated with << or custom methods
  elsif mutated_collection?(value, instance)
    true
  elsif instance.is_a?(Lutaml::Model::Serialize) && instance.using_default?(to)
    render_default? || RenderPolicy.derived_attribute_for?(instance, to)
  else
    true
  end
end

#render_as(key, default_value, options = {}) ⇒ Object



132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
# File 'lib/lutaml/model/mapping/mapping_rule.rb', line 132

def render_as(key, default_value, options = {})
  value = public_send(key)
  value = options[key] if value.nil?

  if value == true
    key.to_s.split("_").last.to_sym
  elsif value == false
    :omitted
  elsif value
    {
      as_empty: :empty,
      as_blank: :blank,
      as_nil: :nil,
      omit: :omitted,
    }[value]
  else
    default_value
  end
end

#render_empty?(options = {}) ⇒ Boolean

Returns:

  • (Boolean)


227
228
229
# File 'lib/lutaml/model/mapping/mapping_rule.rb', line 227

def render_empty?(options = {})
  value_map(:to, options)[:empty] != :omitted
end

#render_nil?(options = {}) ⇒ Boolean

Returns:

  • (Boolean)


223
224
225
# File 'lib/lutaml/model/mapping/mapping_rule.rb', line 223

def render_nil?(options = {})
  value_map(:to, options)[:nil] != :omitted
end

#render_omitted?(options = {}) ⇒ Boolean

Returns:

  • (Boolean)


231
232
233
# File 'lib/lutaml/model/mapping/mapping_rule.rb', line 231

def render_omitted?(options = {})
  value_map(:to, options)[:omitted] != :omitted
end

#render_value_for(value) ⇒ Object



187
188
189
190
191
192
193
194
195
196
197
# File 'lib/lutaml/model/mapping/mapping_rule.rb', line 187

def render_value_for(value)
  if value.nil?
    value_for_option(value_map(:to)[:nil])
  elsif Utils.empty?(value)
    value_for_option(value_map(:to)[:empty], value)
  elsif Utils.uninitialized?(value)
    value_for_option(value_map(:to)[:omitted])
  else
    value
  end
end

#serialize(model, parent = nil, doc = nil) ⇒ Object



267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
# File 'lib/lutaml/model/mapping/mapping_rule.rb', line 267

def serialize(model, parent = nil, doc = nil)
  if custom_methods[:to]
    model.send(custom_methods[:to], model, parent, doc)
  else
    value = to_value_for(model)

    # Handle Reference types - extract the key for serialization
    # This ensures references serialize as their key, not the resolved object
    if value.is_a?(Lutaml::Model::Type::Reference)
      value = value.key
    elsif value.is_a?(Array)
      # Handle collection of references
      value = value.map do |v|
        v.is_a?(Lutaml::Model::Type::Reference) ? v.key : v
      end
    end

    value
  end
end

#serialize_attribute(model, element, doc) ⇒ Object



251
252
253
254
255
# File 'lib/lutaml/model/mapping/mapping_rule.rb', line 251

def serialize_attribute(model, element, doc)
  if custom_methods[:to]
    model.send(custom_methods[:to], model, element, doc)
  end
end

#to_value_for(model) ⇒ Object



257
258
259
260
261
262
263
264
265
# File 'lib/lutaml/model/mapping/mapping_rule.rb', line 257

def to_value_for(model)
  if delegate
    model.public_send(delegate).public_send(to)
  else
    return if to.nil?

    model.public_send(to)
  end
end

#transform_value(attribute, value, read_method, format) ⇒ Object



349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
# File 'lib/lutaml/model/mapping/mapping_rule.rb', line 349

def transform_value(attribute, value, read_method, format)
  # Fast path: no transforms at all (covers 95%+ of rules)
  # transform defaults to {}, so check for actual content, not just truthiness
  has_rule_transform = transform.is_a?(Class)
  has_attr_transform = attribute&.transform.is_a?(Class)
  return value unless has_rule_transform || has_attr_transform

  transformers = get_transformers(attribute)
  transformers = transformers.reverse if read_method == :to

  return value if transformers.empty? || transformers.none? do |t|
    t.can_transform?(read_method, format)
  end

  # Apply transformers in sequence
  transformers.reduce(value) do |v, transformer|
    if transformer.is_a?(Class) && transformer < Lutaml::Model::ValueTransformer
      # Call class method directly: NameTransformer.from(value, :json)
    else
      # Hash/proc transformer
    end
    transformer.public_send(read_method, v, format)
  end
end

#treat?(value) ⇒ Boolean

Returns:

  • (Boolean)


175
176
177
178
179
180
181
182
183
184
185
# File 'lib/lutaml/model/mapping/mapping_rule.rb', line 175

def treat?(value)
  if value.nil?
    treat_nil?
  elsif Utils.uninitialized?(value)
    treat_omitted?
  elsif Utils.empty?(value)
    treat_empty?
  else
    true
  end
end

#treat_as(key, default_value, options = {}) ⇒ Object



152
153
154
# File 'lib/lutaml/model/mapping/mapping_rule.rb', line 152

def treat_as(key, default_value, options = {})
  public_send(key) || options[key] || default_value
end

#treat_empty?(options = {}) ⇒ Boolean

Returns:

  • (Boolean)


239
240
241
# File 'lib/lutaml/model/mapping/mapping_rule.rb', line 239

def treat_empty?(options = {})
  value_map(:from, options)[:empty] != :omitted
end

#treat_nil?(options = {}) ⇒ Boolean

Returns:

  • (Boolean)


235
236
237
# File 'lib/lutaml/model/mapping/mapping_rule.rb', line 235

def treat_nil?(options = {})
  value_map(:from, options)[:nil] != :omitted
end

#treat_omitted?(options = {}) ⇒ Boolean

Returns:

  • (Boolean)


243
244
245
# File 'lib/lutaml/model/mapping/mapping_rule.rb', line 243

def treat_omitted?(options = {})
  value_map(:from, options)[:omitted] != :omitted
end

#value_for_option(option, empty_value = nil) ⇒ Object



216
217
218
219
220
221
# File 'lib/lutaml/model/mapping/mapping_rule.rb', line 216

def value_for_option(option, empty_value = nil)
  return nil if option == :nil
  return empty_value || "" if option == :empty

  Lutaml::Model::UninitializedClass.instance
end

#value_map(key, options = {}) ⇒ Object



332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
# File 'lib/lutaml/model/mapping/mapping_rule.rb', line 332

def value_map(key, options = {})
  # Fast path: when no overrides, return cached value directly
  # Callers MUST NOT mutate the returned hash
  if !options[:nil] && !options[:empty] && !options[:omitted]
    return @value_map[key]
  end

  # Slow path: merge overrides (only when options have actual values)
  overrides = {
    nil: options[:nil],
    empty: options[:empty],
    omitted: options[:omitted],
  }.compact

  @value_map[key].merge(overrides)
end