require_relative "xml_mapping_rule" module Lutaml module Model class XmlMapping attr_reader :root_element, :namespace_uri, :namespace_prefix, :mixed_content, :ordered def initialize @elements = {} @attributes = {} @content_mapping = nil @mixed_content = false end alias mixed_content? mixed_content alias ordered? ordered def root(name, mixed: false, ordered: false) @root_element = name @mixed_content = mixed @ordered = ordered || mixed # mixed contenet will always be ordered end def prefixed_root if namespace_uri && namespace_prefix "#{namespace_prefix}:#{root_element}" else root_element end end def namespace(uri, prefix = nil) @namespace_uri = uri @namespace_prefix = prefix end # rubocop:disable Metrics/ParameterLists def map_element( name, to: nil, render_nil: false, with: {}, delegate: nil, namespace: (namespace_set = false nil), prefix: (prefix_set = false nil) ) validate!(name, to, with) rule = XmlMappingRule.new( name, to: to, render_nil: render_nil, with: with, delegate: delegate, namespace: namespace, default_namespace: namespace_uri, prefix: prefix, namespace_set: namespace_set != false, prefix_set: prefix_set != false, ) @elements[rule.namespaced_name] = rule end def map_attribute( name, to: nil, render_nil: false, with: {}, delegate: nil, namespace: (namespace_set = false nil), prefix: (prefix_set = false nil) ) validate!(name, to, with) rule = XmlMappingRule.new( name, to: to, render_nil: render_nil, with: with, delegate: delegate, namespace: namespace, prefix: prefix, attribute: true, default_namespace: namespace_uri, namespace_set: namespace_set != false, prefix_set: prefix_set != false, ) @attributes[rule.namespaced_name] = rule end # rubocop:enable Metrics/ParameterLists def map_content( to: nil, render_nil: false, with: {}, delegate: nil, mixed: false ) validate!("content", to, with) @content_mapping = XmlMappingRule.new( nil, to: to, render_nil: render_nil, with: with, delegate: delegate, mixed_content: mixed, ) end def validate!(key, to, with) if to.nil? && with.empty? msg = ":to or :with argument is required for mapping '#{key}'" raise IncorrectMappingArgumentsError.new(msg) end if !with.empty? && (with[:from].nil? || with[:to].nil?) msg = ":with argument for mapping '#{key}' requires :to and :from keys" raise IncorrectMappingArgumentsError.new(msg) end end def elements @elements.values end def attributes @attributes.values end def content_mapping @content_mapping end def mappings elements + attributes + [content_mapping].compact end def element(name) elements.detect do |rule| name == rule.to end end def attribute(name) attributes.detect do |rule| name == rule.to end end def find_by_name(name) if name.to_s == "text" content_mapping else mappings.detect do |rule| rule.name == name.to_s || rule.name == name.to_sym end end end def deep_dup self.class.new.tap do |xml_mapping| xml_mapping.root(@root_element.dup, mixed: @mixed_content, ordered: @ordered) xml_mapping.namespace(@namespace_uri.dup, @namespace_prefix.dup) xml_mapping.instance_variable_set(:@attributes, dup_mappings(@attributes)) xml_mapping.instance_variable_set(:@elements, dup_mappings(@elements)) xml_mapping.instance_variable_set(:@content_mapping, @content_mapping&.deep_dup) end end def dup_mappings(mappings) new_mappings = {} mappings.each do |key, mapping_rule| new_mappings[key] = mapping_rule.deep_dup end new_mappings end end end end