Module: Lutaml::Model::Serialize::FormatConversion
- Included in:
- ClassMethods
- Defined in:
- lib/lutaml/model/serialize/format_conversion.rb
Overview
Handles format conversion methods for Serialize::ClassMethods
Extracted from serialize.rb to improve code organization. Provides methods for serializing/deserializing between formats.
Instance Method Summary collapse
-
#array_passthrough_format?(format) ⇒ Boolean
Whether this format+model combination requires the parsed array to pass through to the transformer as a whole (not split per element).
-
#as(format, instance, options = {}) ⇒ Object
Convert a model instance to format-specific data structure.
-
#default_mappings(format) ⇒ Mapping
Generate default mappings for a format.
-
#format_error_types ⇒ Array<Class>
Get list of error types that can be raised during format parsing.
-
#from(format, data, options = {}) ⇒ Object
Deserialize from a format.
-
#key_value(&block) ⇒ Object
Define key-value mappings for multiple formats.
-
#mappings_for(format, register = nil) ⇒ Mapping?
Get resolved mapping for a format.
-
#of(format, doc, options = {}) ⇒ Object
Create a model instance from a parsed document.
-
#post_process_mapping(_format) ⇒ Object
Hook for format-specific post-processing after mapping DSL evaluation.
-
#pre_deserialize_hook(_format, _register) ⇒ Object
Hook for format-specific pre-deserialization logic.
-
#pre_serialize_hook(_format, _register) ⇒ Object
Hook for format-specific pre-serialization logic.
-
#prepare_to_options(_format, _instance, options) ⇒ Hash
Hook for format-specific options preparation before serialization.
-
#process_mapping(format, *_args) ⇒ Object
Process mapping DSL for a format.
-
#rdf(&block) ⇒ Object
Define RDF mappings for multiple formats (Turtle, JSON-LD, etc.).
-
#reset_format_error_types_cache! ⇒ Object
Reset cached error types (for test isolation).
-
#to(format, instance, options = {}) ⇒ String
Serialize a model instance to a format.
-
#validate_document(_format, _doc, _options, _register) ⇒ Object
Hook for format-specific document validation.
Instance Method Details
#array_passthrough_format?(format) ⇒ Boolean
Whether this format+model combination requires the parsed array to pass through to the transformer as a whole (not split per element). YAMLS with sequence definitions needs the full document array.
170 171 172 173 174 175 176 177 178 179 |
# File 'lib/lutaml/model/serialize/format_conversion.rb', line 170 def array_passthrough_format?(format) return true if format == :jsonl if format == :yamls mapping = mappings[format] return true if mapping.respond_to?(:yamls_sequence) && mapping.yamls_sequence end false end |
#as(format, instance, options = {}) ⇒ Object
Convert a model instance to format-specific data structure
230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 |
# File 'lib/lutaml/model/serialize/format_conversion.rb', line 230 def as(format, instance, = {}) if instance.is_a?(Array) return instance.map { |item| public_send(:"as_#{format}", item) } end unless instance.is_a?(model) msg = "argument is a '#{instance.class}' but should be a '#{model}'" raise Lutaml::Model::IncorrectModelError, msg end # Resolve imports at the start of serialization register = [:register] || Lutaml::Model::Config.default_register # Hook for format-specific pre-serialization (e.g., XML mapping import resolution) pre_serialize_hook(format, register) # Recursively resolve child model imports ensure_child_imports_resolved!(register) transformer = Lutaml::Model::Config.transformer_for(format) transformer.model_to_data(self, instance, format, ) end |
#default_mappings(format) ⇒ Mapping
Generate default mappings for a format
316 317 318 319 320 321 322 323 324 325 326 327 328 329 330 331 332 |
# File 'lib/lutaml/model/serialize/format_conversion.rb', line 316 def default_mappings(format) klass = ::Lutaml::Model::Config.mappings_class_for(format) mappings = klass.new mappings.tap do |mapping| attributes&.each_key do |name| mapping.map_element( name.to_s, to: name, ) end # DO NOT auto-generate root element for XML # Models without an explicit xml block should be type-only models # If a root element is needed, declare it explicitly in xml block end end |
#format_error_types ⇒ Array<Class>
Get list of error types that can be raised during format parsing. Core errors are always included; format-specific errors come from FormatRegistry registrations.
Performance: Cached base types + lazy TOML lookup. Tomlib::ParseError is lazily loaded, so we check for it on each call rather than caching a stale nil reference.
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 116 117 118 119 120 121 122 123 124 |
# File 'lib/lutaml/model/serialize/format_conversion.rb', line 88 def format_error_types @format_error_types_base ||= begin errors = [ Lutaml::Model::RuntimeCompatibility.safe_constantize("Psych::SyntaxError"), Lutaml::Model::RuntimeCompatibility.safe_constantize("JSON::ParserError"), NoMethodError, TypeError, ArgumentError, ] # Collect format-specific error types from FormatRegistry compatibility = Lutaml::Model::RuntimeCompatibility FormatRegistry.all.each_value do |info| next unless info[:error_types] info[:error_types].each do |error_class| cls = if error_class.is_a?(String) compatibility.safe_constantize(error_class) else error_class end errors << cls end end errors.compact.freeze end # Legacy TOML error types are lazy, so check them on each call. compatibility = Lutaml::Model::RuntimeCompatibility toml_errors = compatibility.safe_constantize("TomlRB::ParseError") toml_errors = Array(toml_errors) tomllib_err = compatibility.safe_constantize("Tomlib::ParseError") toml_errors << tomllib_err if tomllib_err @format_error_types_base + toml_errors end |
#from(format, data, options = {}) ⇒ Object
Deserialize from a format
47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 |
# File 'lib/lutaml/model/serialize/format_conversion.rb', line 47 def from(format, data, = {}) Instrumentation.instrument(:from, model: name, format: format) do adapter = resolve_adapter(format, .delete(:adapter)) raise Lutaml::Model::FormatAdapterNotSpecifiedError.new(format) if adapter.nil? # Resolve imports at the entry point of deserialization register = [:register] || Lutaml::Model::Config.default_register # Hook for format-specific pre-deserialization (e.g., XML mapping import resolution) pre_deserialize_hook(format, register) # Recursively resolve child model imports # This ensures the entire model tree is finalized before parsing ensure_child_imports_resolved!(register) doc = adapter.parse(data, ) send("of_#{format}", doc, ) end rescue *format_error_types => e raise Lutaml::Model::InvalidFormatError.new(format, e.) end |
#key_value(&block) ⇒ Object
Define key-value mappings for multiple formats. Uses FormatRegistry to discover key-value formats dynamically, falling back to Config::KEY_VALUE_FORMATS for bootstrap.
267 268 269 270 271 272 273 274 275 276 277 278 279 |
# File 'lib/lutaml/model/serialize/format_conversion.rb', line 267 def key_value(&block) formats = if FormatRegistry.formats.any? FormatRegistry.key_value_formats else Lutaml::Model::Config::KEY_VALUE_FORMATS end formats.each do |format| mappings[format] ||= Lutaml::KeyValue::Mapping.new(format) mappings[format].instance_eval(&block) mappings[format].finalize(self) end end |
#mappings_for(format, register = nil) ⇒ Mapping?
Get resolved mapping for a format
Delegates to TransformationRegistry for centralized caching (Single Source of Truth - Phase 11.5).
Register resolution: If the caller passes a parent register (e.g., :default) but this class declares its own ‘lutaml_default_register`, the child’s register takes precedence. This ensures mappings are resolved in the correct context for cross-register embedding.
305 306 307 308 309 310 |
# File 'lib/lutaml/model/serialize/format_conversion.rb', line 305 def mappings_for(format, register = nil) resolved_register = Lutaml::Model::Register.resolve_for_child(self, register) TransformationRegistry.instance.get_or_build_mapping(self, format, resolved_register) end |
#of(format, doc, options = {}) ⇒ Object
Create a model instance from a parsed document
137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 |
# File 'lib/lutaml/model/serialize/format_conversion.rb', line 137 def of(format, doc, = {}) if doc.is_a?(Array) && !array_passthrough_format?(format) return doc.map { |item| send(:"of_#{format}", item) } end register = extract_register_id([:register]) # Hook for format-specific document validation (e.g., XML root/encoding/doctype) validate_document(format, doc, , register) [:register] = register transformer = Lutaml::Model::Config.transformer_for(format) transformer.data_to_model(self, doc, format, ) end |
#post_process_mapping(_format) ⇒ Object
Hook for format-specific post-processing after mapping DSL evaluation. XML overrides this to call check_sort_configs!.
37 38 39 |
# File 'lib/lutaml/model/serialize/format_conversion.rb', line 37 def post_process_mapping(_format) # No-op by default; XML overrides via prepend end |
#pre_deserialize_hook(_format, _register) ⇒ Object
Hook for format-specific pre-deserialization logic. XML overrides to resolve XML mapping imports.
75 76 77 |
# File 'lib/lutaml/model/serialize/format_conversion.rb', line 75 def pre_deserialize_hook(_format, _register) # No-op by default; XML overrides via prepend end |
#pre_serialize_hook(_format, _register) ⇒ Object
Hook for format-specific pre-serialization logic. XML overrides to resolve XML mapping imports.
258 259 260 |
# File 'lib/lutaml/model/serialize/format_conversion.rb', line 258 def pre_serialize_hook(_format, _register) # No-op by default; XML overrides via prepend end |
#prepare_to_options(_format, _instance, options) ⇒ Hash
Hook for format-specific options preparation before serialization. XML overrides to handle prefix, namespace overrides, declaration plan.
220 221 222 |
# File 'lib/lutaml/model/serialize/format_conversion.rb', line 220 def (_format, _instance, ) end |
#process_mapping(format, *_args) ⇒ Object
Process mapping DSL for a format
16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 |
# File 'lib/lutaml/model/serialize/format_conversion.rb', line 16 def process_mapping(format, *_args, &) klass = ::Lutaml::Model::Config.mappings_class_for(format) existing = mappings[format] mappings[format] = if existing.nil? || !existing.is_a?(klass) klass.new else existing end mappings[format].instance_eval(&) if mappings[format].respond_to?(:finalize) mappings[format].finalize(self) end post_process_mapping(format) end |
#rdf(&block) ⇒ Object
Define RDF mappings for multiple formats (Turtle, JSON-LD, etc.). Discovers RDF formats dynamically from FormatRegistry.
285 286 287 288 289 290 |
# File 'lib/lutaml/model/serialize/format_conversion.rb', line 285 def rdf(&block) Lutaml::Model::FormatRegistry.rdf_formats.each do |format| mappings[format] = Lutaml::Rdf::Mapping.new mappings[format].instance_eval(&block) end end |
#reset_format_error_types_cache! ⇒ Object
Reset cached error types (for test isolation)
127 128 129 |
# File 'lib/lutaml/model/serialize/format_conversion.rb', line 127 def reset_format_error_types_cache! @format_error_types_base = nil end |
#to(format, instance, options = {}) ⇒ String
Serialize a model instance to a format
194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 |
# File 'lib/lutaml/model/serialize/format_conversion.rb', line 194 def to(format, instance, = {}) Instrumentation.instrument(:to, model: name, format: format) do adapter_override = .is_a?(Hash) && .delete(:adapter) if adapter_override && .is_a?(Hash) [:_adapter_override] = true end value = public_send(:"as_#{format}", instance, ) adapter = resolve_adapter(format, adapter_override) # Hook for format-specific options preparation (e.g., XML prefix/namespace/declaration) = (format, instance, ) adapter.new(value, register: [:register]).public_send( :"to_#{format}", ) end end |
#validate_document(_format, _doc, _options, _register) ⇒ Object
Hook for format-specific document validation. XML overrides to validate root mapping and extract encoding/doctype.
160 161 162 |
# File 'lib/lutaml/model/serialize/format_conversion.rb', line 160 def validate_document(_format, _doc, , _register) # No-op by default; XML overrides via prepend end |