Class: Lutaml::Model::MappingRule
- Inherits:
-
Object
- Object
- Lutaml::Model::MappingRule
- Includes:
- DeepDupable
- Defined in:
- lib/lutaml/model/mapping/mapping_rule.rb
Direct Known Subclasses
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
-
#as_attribute ⇒ Object
readonly
Returns the value of attribute as_attribute.
-
#attribute ⇒ Object
(also: #attribute?)
readonly
Returns the value of attribute attribute.
-
#custom_methods ⇒ Object
readonly
Returns the value of attribute custom_methods.
-
#delegate ⇒ Object
readonly
Returns the value of attribute delegate.
-
#format ⇒ Object
readonly
Returns the value of attribute format.
-
#name ⇒ Object
(also: #from)
readonly
Returns the value of attribute name.
-
#polymorphic ⇒ Object
readonly
Returns the value of attribute polymorphic.
-
#polymorphic_map ⇒ Object
readonly
Returns the value of attribute polymorphic_map.
-
#render_default ⇒ Object
(also: #render_default?)
readonly
Returns the value of attribute render_default.
-
#render_empty ⇒ Object
readonly
Returns the value of attribute render_empty.
-
#render_nil ⇒ Object
readonly
Returns the value of attribute render_nil.
-
#to ⇒ Object
readonly
Returns the value of attribute to.
-
#to_instance ⇒ Object
readonly
Returns the value of attribute to_instance.
-
#transform ⇒ Object
readonly
Returns the value of attribute transform.
-
#treat_empty ⇒ Object
readonly
Returns the value of attribute treat_empty.
-
#treat_nil ⇒ Object
readonly
Returns the value of attribute treat_nil.
-
#treat_omitted ⇒ Object
readonly
Returns the value of attribute treat_omitted.
Instance Method Summary collapse
- #can_transform_to?(attribute, format) ⇒ Boolean
- #deep_dup ⇒ Object
- #default_value_map(options = {}) ⇒ Object
- #deserialize(model, value, attributes, mapper_class = nil) ⇒ Object
- #eql?(other) ⇒ Boolean (also: #==)
- #get_transformers(attribute) ⇒ Object
- #has_custom_method_for_deserialization? ⇒ Boolean
- #has_custom_method_for_serialization? ⇒ Boolean
-
#has_items?(value) ⇒ Boolean
Check if value is a non-empty collection.
-
#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
constructor
A new instance of MappingRule.
- #multiple_mappings? ⇒ Boolean
- #mutated_collection?(value, instance) ⇒ Boolean
- #polymorphic_mapping? ⇒ Boolean
- #raw_mapping? ⇒ Boolean
-
#raw_value_map ⇒ Object
Raw value_map hash access (for compiled rules).
- #render?(value, instance = nil, options = {}) ⇒ Boolean
- #render_as(key, default_value, options = {}) ⇒ Object
- #render_empty?(options = {}) ⇒ Boolean
- #render_nil?(options = {}) ⇒ Boolean
- #render_omitted?(options = {}) ⇒ Boolean
- #render_value_for(value) ⇒ Object
- #serialize(model, parent = nil, doc = nil) ⇒ Object
- #serialize_attribute(model, element, doc) ⇒ Object
- #to_value_for(model) ⇒ Object
- #transform_value(attribute, value, read_method, format) ⇒ Object
- #treat?(value) ⇒ Boolean
- #treat_as(key, default_value, options = {}) ⇒ Object
- #treat_empty?(options = {}) ⇒ Boolean
- #treat_nil?(options = {}) ⇒ Boolean
- #treat_omitted?(options = {}) ⇒ Boolean
- #value_for_option(option, empty_value = nil) ⇒ Object
- #value_map(key, options = {}) ⇒ Object
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_attribute ⇒ Object (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 |
#attribute ⇒ Object (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_methods ⇒ Object (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 |
#delegate ⇒ Object (readonly)
Returns the value of attribute delegate.
6 7 8 |
# File 'lib/lutaml/model/mapping/mapping_rule.rb', line 6 def delegate @delegate end |
#format ⇒ Object (readonly)
Returns the value of attribute format.
6 7 8 |
# File 'lib/lutaml/model/mapping/mapping_rule.rb', line 6 def format @format end |
#name ⇒ Object (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 |
#polymorphic ⇒ Object (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_map ⇒ Object (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_default ⇒ Object (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_empty ⇒ Object (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_nil ⇒ Object (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 |
#to ⇒ Object (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_instance ⇒ Object (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 |
#transform ⇒ Object (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_empty ⇒ Object (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_nil ⇒ Object (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_omitted ⇒ Object (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
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_dup ⇒ Object
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( = {}) render_nil_as = render_as(:render_nil, :omitted, ) render_empty_as = render_as(:render_empty, :empty, ) treat_nil_as = treat_as(:treat_nil, :nil, ) treat_empty_as = treat_as(:treat_empty, :empty, ) treat_omitted_as = treat_as(:treat_omitted, :nil, ) { 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: ==
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
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
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
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
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
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
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
310 311 312 |
# File 'lib/lutaml/model/mapping/mapping_rule.rb', line 310 def raw_mapping? name == Constants::RAW_MAPPING_KEY end |
#raw_value_map ⇒ Object
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
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, = {}) if invalid_value?(value, ) 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, = {}) value = public_send(key) value = [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
227 228 229 |
# File 'lib/lutaml/model/mapping/mapping_rule.rb', line 227 def render_empty?( = {}) value_map(:to, )[:empty] != :omitted end |
#render_nil?(options = {}) ⇒ Boolean
223 224 225 |
# File 'lib/lutaml/model/mapping/mapping_rule.rb', line 223 def render_nil?( = {}) value_map(:to, )[:nil] != :omitted end |
#render_omitted?(options = {}) ⇒ Boolean
231 232 233 |
# File 'lib/lutaml/model/mapping/mapping_rule.rb', line 231 def render_omitted?( = {}) value_map(:to, )[: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
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, = {}) public_send(key) || [key] || default_value end |
#treat_empty?(options = {}) ⇒ Boolean
239 240 241 |
# File 'lib/lutaml/model/mapping/mapping_rule.rb', line 239 def treat_empty?( = {}) value_map(:from, )[:empty] != :omitted end |
#treat_nil?(options = {}) ⇒ Boolean
235 236 237 |
# File 'lib/lutaml/model/mapping/mapping_rule.rb', line 235 def treat_nil?( = {}) value_map(:from, )[:nil] != :omitted end |
#treat_omitted?(options = {}) ⇒ Boolean
243 244 245 |
# File 'lib/lutaml/model/mapping/mapping_rule.rb', line 243 def treat_omitted?( = {}) value_map(:from, )[: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, = {}) # Fast path: when no overrides, return cached value directly # Callers MUST NOT mutate the returned hash if ![:nil] && ![:empty] && ![:omitted] return @value_map[key] end # Slow path: merge overrides (only when options have actual values) overrides = { nil: [:nil], empty: [:empty], omitted: [:omitted], }.compact @value_map[key].merge(overrides) end |