module RelatonBib class HashConverter class << self # rubocop:disable Metrics/MethodLength, Metrics/AbcSize # @param args [Hash] # @param neated [TrueClas, FalseClass] default false # @return [Hash] def hash_to_bib(args, nested = false) return nil unless args.is_a?(Hash) ret = Marshal.load(Marshal.dump(symbolize(args))) # deep copy timestamp_hash(ret) unless nested title_hash_to_bib(ret) link_hash_to_bib(ret) language_hash_to_bib(ret) script_hash_to_bib(ret) dates_hash_to_bib(ret) docid_hash_to_bib(ret) version_hash_to_bib(ret) biblionote_hash_to_bib(ret) abstract_hash_to_bib(ret) formattedref_hash_to_bib(ret) docstatus_hash_to_bib(ret) contributors_hash_to_bib(ret) copyright_hash_to_bib(ret) relations_hash_to_bib(ret) series_hash_to_bib(ret) medium_hash_to_bib(ret) place_hash_to_bib(ret) extent_hash_to_bib(ret) accesslocation_hash_to_bib(ret) classification_hash_to_bib(ret) validity_hash_to_bib(ret) ret[:keyword] = array(ret[:keyword]) ret[:license] = array(ret[:license]) editorialgroup_hash_to_bib ret ics_hash_to_bib ret structuredidentifier_hash_to_bib ret ret end # rubocop:enable Metrics/MethodLength, Metrics/AbcSize def timestamp_hash(ret) ret[:fetched] ||= Date.today.to_s end def extent_hash_to_bib(ret) return unless ret[:extent] ret[:extent] = array(ret[:extent]) ret[:extent]&.each_with_index do |e, i| ret[:extent][i] = BibItemLocality.new(e[:type], e[:reference_from], e[:reference_to]) end end def title_hash_to_bib(ret) return unless ret[:title] ret[:title] = array(ret[:title]) .reduce(TypedTitleStringCollection.new) do |m, t| if t.is_a?(Hash) then m << t else m + TypedTitleString.from_string(t) end end end def language_hash_to_bib(ret) return unless ret[:language] ret[:language] = array(ret[:language]) end def script_hash_to_bib(ret) return unless ret[:script] ret[:script] = array(ret[:script]) end def abstract_hash_to_bib(ret) return unless ret[:abstract] ret[:abstract] = array(ret[:abstract]).map do |a| a.is_a?(String) ? FormattedString.new(content: a) : a end end def link_hash_to_bib(ret) return unless ret[:link] ret[:link] = array(ret[:link]) end def place_hash_to_bib(ret) return unless ret[:place] ret[:place] = array(ret[:place]).map do |pl| pl.is_a?(String) ? Place.new(name: pl) : Place.new(**pl) end end def accesslocation_hash_to_bib(ret) return unless ret[:accesslocation] ret[:accesslocation] = array(ret[:accesslocation]) end def dates_hash_to_bib(ret) # rubocop:disable Metrics/AbcSize return unless ret[:date] ret[:date] = array(ret[:date]) ret[:date].each_with_index do |d, i| # value is synonym of on: it is reserved word in YAML if d[:value] ret[:date][i][:on] ||= d[:value] ret[:date][i].delete(:value) end end end def docid_hash_to_bib(ret) # rubocop:disable Metrics/AbcSize return unless ret[:docid] ret[:docid] = array(ret[:docid]) ret[:docid]&.each_with_index do |id, i| type = id[:type] || id[:id].match(/^\w+(?=\s)/)&.to_s ret[:docid][i] = DocumentIdentifier.new(id: id[:id], type: type, scope: id[:scope]) end end def version_hash_to_bib(ret) return unless ret[:version] ret[:version][:draft] = array(ret[:version][:draft]) ret[:version] && ret[:version] = BibliographicItem::Version.new( ret[:version][:revision_date], ret[:version][:draft] ) end def biblionote_hash_to_bib(ret) # rubocop:disable Metrics/MethodLength,Metrics/AbcSize return unless ret[:biblionote] ret[:biblionote] = array(ret[:biblionote]) .reduce(BiblioNoteCollection.new([])) do |mem, n| mem << if n.is_a?(String) then BiblioNote.new content: n else BiblioNote.new(**n) end end end def formattedref_hash_to_bib(ret) ret[:formattedref] && ret[:formattedref] = formattedref(ret[:formattedref]) end def docstatus_hash_to_bib(ret) ret[:docstatus] && ret[:docstatus] = DocumentStatus.new( stage: stage(ret[:docstatus][:stage]), substage: stage(ret[:docstatus][:substage]), iteration: ret[:docstatus][:iteration], ) end # @param stg [Hash] # @return [RelatonBib::DocumentStatus::Stage] def stage(stg) return unless stg args = stg.is_a?(String) ? { value: stg } : stg DocumentStatus::Stage.new(**args) end def contributors_hash_to_bib(ret) # rubocop:disable Metrics/AbcSize, Metrics/CyclomaticComplexity, Metrics/MethodLength,Metrics/PerceivedComplexity return unless ret[:contributor] ret[:contributor] = array(ret[:contributor]) ret[:contributor]&.each_with_index do |c, i| roles = array(ret[:contributor][i][:role]).map do |r| if r.is_a? Hash { type: r[:type], description: array(r[:description]) } # elsif r.is_a? Array # { type: r[0], description: r.fetch(1) } else { type: r } end end ret[:contributor][i][:role] = roles ret[:contributor][i][:entity] = if c[:person] person_hash_to_bib(c[:person]) else org_hash_to_bib(c[:organization]) end ret[:contributor][i].delete(:person) ret[:contributor][i].delete(:organization) end end def org_hash_to_bib(org) return nil if org.nil? org[:identifier] = array(org[:identifier])&.map do |a| OrgIdentifier.new(a[:type], a[:id]) end org[:subdivision] = array(org[:subdivision]).map do |sd| LocalizedString.new sd end org end def person_hash_to_bib(person) Person.new( name: fullname_hash_to_bib(person), affiliation: affiliation_hash_to_bib(person), contact: contacts_hash_to_bib(person), identifier: person_identifiers_hash_to_bib(person), ) end def fullname_hash_to_bib(person) # rubocop:disable Metrics/AbcSize,Metrics/CyclomaticComplexity,Metrics/PerceivedComplexity n = person[:name] FullName.new( forename: array(n[:forename])&.map { |f| localname(f, person) }, initial: array(n[:initial])&.map { |f| localname(f, person) }, addition: array(n[:addition])&.map { |f| localname(f, person) }, prefix: array(n[:prefix])&.map { |f| localname(f, person) }, surname: localname(n[:surname], person), completename: localname(n[:completename], person), ) end def person_identifiers_hash_to_bib(person) array(person[:identifier])&.map do |a| PersonIdentifier.new(a[:type], a[:id]) end end def affiliation_hash_to_bib(person) # rubocop:disable Metrics/AbcSize, Metrics/MethodLength return [] unless person[:affiliation] array(person[:affiliation]).map do |a| a[:description] = array(a[:description])&.map do |d| cnt = if d.is_a?(Hash) { content: d[:content], language: d[:language], script: d[:script], format: d[:format] } else { content: d } end FormattedString.new(**cnt) end Affiliation.new( organization: Organization.new(**org_hash_to_bib(a[:organization])), description: a[:description], ) end end def contacts_hash_to_bib(person) # rubocop:disable Metrics/AbcSize, Metrics/MethodLength return [] unless person[:contact] array(person[:contact]).map do |a| if a[:city] || a[:country] RelatonBib::Address.new( street: Array(a[:street]), city: a[:city], postcode: a[:postcode], country: a[:country], state: a[:state] ) else RelatonBib::Contact.new(type: a[:type], value: a[:value]) end end end # @param ret [Hash] def copyright_hash_to_bib(ret) return unless ret[:copyright] ret[:copyright] = array(ret[:copyright]).map do |c| c[:owner] = array(c[:owner]) c end end # @param ret [Hash] def relations_hash_to_bib(ret) return unless ret[:relation] ret[:relation] = array(ret[:relation]) ret[:relation]&.each do |r| if r[:description] r[:description] = FormattedString.new(**r[:description]) end relation_bibitem_hash_to_bib(r) relation_locality_hash_to_bib(r) relation_source_locality_hash_to_bib(r) end end # @param rel [Hash] relation def relation_bibitem_hash_to_bib(rel) if rel[:bibitem] rel[:bibitem] = bib_item hash_to_bib(rel[:bibitem], true) else warn "[relaton-bib] bibitem missing: #{rel}" rel[:bibitem] = nil end end # @param item_hash [Hash] # @return [RelatonBib::BibliographicItem] def bib_item(item_hash) BibliographicItem.new(**item_hash) end # @param rel [Hash] relation # @return [RelatonBib::LocalityStack] def relation_locality_hash_to_bib(rel) # rubocop:disable Metrics/AbcSize, Metrics/MethodLength rel[:locality] = array(rel[:locality])&.map do |bl| ls = if bl[:locality_stack] array(bl[:locality_stack]).map do |l| Locality.new(l[:type], l[:reference_from], l[:reference_to]) end else l = Locality.new(bl[:type], bl[:reference_from], bl[:reference_to]) [l] end LocalityStack.new ls end end # @param rel [Hash] relation def relation_source_locality_hash_to_bib(rel) # rubocop:disable Metrics/AbcSize, Metrics/MethodLength rel[:source_locality] = array(rel[:source_locality])&.map do |sl| sls = if sl[:source_locality_stack] array(sl[:source_locality_stack]).map do |l| SourceLocality.new(l[:type], l[:reference_from], l[:reference_to]) end else l = SourceLocality.new(sl[:type], sl[:reference_from], sl[:reference_to]) [l] end SourceLocalityStack.new sls end end # @param ret [Hash] def series_hash_to_bib(ret) # rubocop:disable Metrics/AbcSize, Metrics/CyclomaticComplexity ret[:series] = array(ret[:series])&.map do |s| s[:formattedref] && s[:formattedref] = formattedref(s[:formattedref]) if s[:title] s[:title] = { content: s[:title] } unless s[:title].is_a?(Hash) s[:title] = typed_title_strig(s[:title]) end s[:abbreviation] && s[:abbreviation] = localizedstring(s[:abbreviation]) Series.new(**s) end end # @param title [Hash] # @return [RelatonBib::TypedTitleString] def typed_title_strig(title) TypedTitleString.new(**title) end # @param ret [Hash] def medium_hash_to_bib(ret) ret[:medium] = Medium.new(**ret[:medium]) if ret[:medium] end # @param ret [Hash] def classification_hash_to_bib(ret) if ret[:classification] ret[:classification] = array(ret[:classification]).map do |cls| Classification.new(**cls) end end end # @param ret [Hash] def validity_hash_to_bib(ret) return unless ret[:validity] b = parse_validity_time(ret[:validity], :begins) e = parse_validity_time(ret[:validity], :ends) r = parse_validity_time(ret[:validity], :revision) ret[:validity] = Validity.new(begins: b, ends: e, revision: r) end def parse_validity_time(val, period) t = val[period]&.to_s return unless t p = period == :ends ? -1 : 1 case t when /^\d{4}$/ Date.new(t.to_i, p, p).to_time when /^(?\d{4})-(?\d{1,2})$/ Date.new($~[:year].to_i, $~[:month].to_i, p).to_time else Time.parse t end end # @param ret [Hash] def editorialgroup_hash_to_bib(ret) return unless ret[:editorialgroup] technical_committee = array(ret[:editorialgroup]).map do |wg| TechnicalCommittee.new WorkGroup.new(**wg) end ret[:editorialgroup] = EditorialGroup.new technical_committee end # @param ret [Hash] def ics_hash_to_bib(ret) return unless ret[:ics] ret[:ics] = array(ret[:ics]).map { |ics| ICS.new(**ics) } end # @param ret [Hash] def structuredidentifier_hash_to_bib(ret) return unless ret[:structuredidentifier] sids = array(ret[:structuredidentifier]).map do |si| si[:agency] = array si[:agency] StructuredIdentifier.new(**si) end ret[:structuredidentifier] = StructuredIdentifierCollection.new sids end # @param ogj [Hash, Array, String] # @return [Hash, Array, String] def symbolize(obj) case obj when Hash obj.reduce({}) do |memo, (k, v)| memo[k.to_sym] = symbolize(v) memo end when Array then obj.reduce([]) { |memo, v| memo << symbolize(v) } else obj end end # @param arr [NilClass, Array, #is_a?] # @return [Array] def array(arr) return [] unless arr return [arr] unless arr.is_a?(Array) arr end # @param name [Hash, String, NilClass] # @param person [Hash] # @return [RelatonBib::LocalizedString] def localname(name, person) # rubocop:disable Metrics/CyclomaticComplexity,Metrics/AbcSize return nil if name.nil? lang = name[:language] if name.is_a?(Hash) lang ||= person[:name][:language] script = name[:script] if name.is_a?(Hash) script ||= person[:name][:script] n = name.is_a?(Hash) ? name[:content] : name RelatonBib::LocalizedString.new(n, lang, script) end # @param lst [Hash, Array] # @return [RelatonBib::LocalizedString] def localizedstring(lst) if lst.is_a?(Hash) RelatonBib::LocalizedString.new(lst[:content], lst[:language], lst[:script]) else RelatonBib::LocalizedString.new(lst) end end # @param frf [Hash, String] # @return [RelatonBib::FormattedRef] def formattedref(frf) if frf.is_a?(Hash) RelatonBib::FormattedRef.new(**frf) else RelatonBib::FormattedRef.new(content: frf) end end end end end