# frozen_string_literal: true require "relaton_bib/typed_uri" require "relaton_bib/document_identifier" require "relaton_bib/copyright_association" require "relaton_bib/formatted_string" require "relaton_bib/contribution_info" require "relaton_bib/bibliographic_date" require "relaton_bib/series" require "relaton_bib/document_status" require "relaton_bib/organization" require "relaton_bib/document_relation_collection" require "relaton_bib/typed_title_string" require "relaton_bib/formatted_ref" require "relaton_bib/medium" require "relaton_bib/classification" require "relaton_bib/validity" require "relaton_bib/document_relation" require "relaton_bib/bib_item_locality" require "relaton_bib/xml_parser" require "relaton_bib/bibtex_parser" require "relaton_bib/biblio_note" require "relaton_bib/biblio_version" require "relaton_bib/workers_pool" require "relaton_bib/hash_converter" require "relaton_bib/place" module RelatonBib # Bibliographic item class BibliographicItem include RelatonBib TYPES = %W[article book booklet conference manual proceedings presentation thesis techreport standard unpublished map electronic\sresource audiovisual film video broadcast graphic_work music patent inbook incollection inproceedings journal].freeze # @return [String, NilClass] attr_reader :id # @return [Array] attr_reader :title # @return [Array] attr_reader :link # @return [String, NilClass] attr_reader :type # @return [Array] attr_reader :docidentifier # @return [String, NilClass] docnumber attr_reader :docnumber # @return [Array] attr_accessor :date # @return [Array] attr_reader :contributor # @return [String, NillClass] attr_reader :edition # @return [RelatonBib::BibliongraphicItem::Version, NilClass] attr_reader :version # @return [Array] attr_reader :biblionote # @return [Array] language Iso639 code attr_reader :language # @return [Array] script Iso15924 code attr_reader :script # @return [RelatonBib::FormattedRef, NilClass] attr_reader :formattedref # @!attribute [r] abstract # @return [Array] # @return [RelatonBib::DocumentStatus, NilClass] attr_reader :status # @return [RelatonBib::CopyrightAssociation, NilClass] attr_reader :copyright # @return [RelatonBib::DocRelationCollection] attr_reader :relation # @return [Array] attr_reader :series # @return [RelatonBib::Medium, NilClass] attr_reader :medium # @return [Array] attr_reader :place # @return [Array] attr_reader :extent # @return [Array] attr_reader :accesslocation, :license # @return [Array] attr_reader :classification # @return [RelatonBib:Validity, NilClass] attr_reader :validity # @return [Date] attr_reader :fetched # @return [Array] attr_reader :keyword # rubocop:disable Metrics/MethodLength, Metrics/AbcSize # rubocop:disable Metrics/CyclomaticComplexity, Metrics/PerceivedComplexity # @param id [String, NilClass] # @param title [Array] # @param formattedref [RelatonBib::FormattedRef, NilClass] # @param type [String, NilClass] # @param docid [Array] # @param docnumber [String, NilClass] # @param language [Arra] # @param script [Array] # @param docstatus [RelatonBib::DocumentStatus, NilClass] # @param edition [String, NilClass] # @param version [RelatonBib::BibliographicItem::Version, NilClass] # @param biblionote [Array] # @param series [Array] # @param medium [RelatonBib::Medium, NilClas] # @param place [Array] # @param extent [Array] # @param accesslocation [Array] # @param classification [Array] # @param validity [RelatonBib:Validity, NilClass] # @param fetched [Date, NilClass] default nil # @param keyword [Array] # # @param copyright [Hash, RelatonBib::CopyrightAssociation, NilClass] # @option copyright [Hash, RelatonBib::ContributionInfo] :owner # @option copyright [String] :form # @option copyright [String, NilClass] :to # # @param date [Array] # @option date [String] :type # @option date [String] :from # @option date [String] :to # # @param contributor [Array] # @option contributor [RealtonBib::Organization, RelatonBib::Person] # @option contributor [String] :type # @option contributor [String] :from # @option contributor [String] :to # @option contributor [String] :abbreviation # @option contributor [Array>>] :role # # @param abstract [Array] # @option abstract [String] :content # @option abstract [String] :language # @option abstract [String] :script # @option abstract [String] :type # # @param relation [Array] # @option relation [String] :type # @option relation [RelatonBib::BibliographicItem, RelatonIso::IsoBibliographicItem] :bibitem # @option relation [Array] :bib_locality # # @param link [Array] # @option link [String] :type # @option link [String] :content def initialize(**args) if args[:type] && !TYPES.include?(args[:type]) warn %{[relaton-bib] document type "#{args[:type]}" is invalid.} end @title = (args[:title] || []).map do |t| t.is_a?(Hash) ? TypedTitleString.new(t) : t end @date = (args[:date] || []).map do |d| d.is_a?(Hash) ? BibliographicDate.new(d) : d end @contributor = (args[:contributor] || []).map do |c| if c.is_a? Hash e = c[:entity].is_a?(Hash) ? Organization.new(c[:entity]) : c[:entity] ContributionInfo.new(entity: e, role: c[:role]) else c end end @abstract = (args[:abstract] || []).map do |a| a.is_a?(Hash) ? FormattedString.new(a) : a end if args[:copyright] @copyright = if args[:copyright].is_a?(Hash) CopyrightAssociation.new args[:copyright] else args[:copyright] end end @docidentifier = args[:docid] || [] @formattedref = args[:formattedref] if title.empty? @id = args[:id] || makeid(nil, false) @type = args[:type] @docnumber = args[:docnumber] @edition = args[:edition] @version = args[:version] @biblionote = args.fetch :biblionote, [] @language = args.fetch :language, [] @script = args.fetch :script, [] @status = args[:docstatus] @relation = DocRelationCollection.new(args[:relation] || []) @link = args.fetch(:link, []).map { |s| s.is_a?(Hash) ? TypedUri.new(s) : s } @series = args.fetch :series, [] @medium = args[:medium] @place = args.fetch(:place, []).map { |pl| pl.is_a?(String) ? Place.new(name: pl) : pl } @extent = args[:extent] || [] @accesslocation = args.fetch :accesslocation, [] @classification = args.fetch :classification, [] @validity = args[:validity] @fetched = args.fetch :fetched, nil # , Date.today # we should pass the fetched arg from scrappers @keyword = (args[:keyword] || []).map { |kw| LocalizedString.new(kw) } @license = args.fetch :license, [] end # rubocop:enable Metrics/MethodLength, Metrics/AbcSize # rubocop:enable Metrics/CyclomaticComplexity, Metrics/PerceivedComplexity # @param lang [String] language code Iso639 # @return [RelatonBib::FormattedString, Array] def abstract(lang: nil) if lang @abstract.detect { |a| a.language.include? lang } else @abstract end end def makeid(id, attribute) return nil if attribute && !@id_attribute id ||= @docidentifier.reject { |i| i.type == "DOI" }[0] return unless id # contribs = publishers.map { |p| p&.entity&.abbreviation }.join '/' # idstr = "#{contribs}#{delim}#{id.project_number}" # idstr = id.project_number.to_s idstr = id.id.gsub(/:/, "-").gsub /\s/, "" # if id.part_number&.size&.positive? then idstr += "-#{id.part_number}" idstr.strip end # @return [String] def shortref(identifier, **opts) pubdate = date.select { |d| d.type == "published" } year = if opts[:no_year] || pubdate.empty? then "" else ":" + pubdate&.first&.on&.year.to_s end year += ": All Parts" if opts[:all_parts] || @all_parts "#{makeid(identifier, false)}#{year}" end # @param builder [Nokogiri::XML::Builder, NillClass] (nil) # @return [String] def to_xml(builder = nil, **opts, &block) if builder render_xml builder, **opts, &block else Nokogiri::XML::Builder.new(encoding: "UTF-8") do |xml| render_xml xml, **opts, &block end.doc.root.to_xml end end # @return [Hash] def to_hash hash = {} hash["id"] = id if id hash["title"] = single_element_array(title) if title&.any? hash["link"] = single_element_array(link) if link&.any? hash["type"] = type if type hash["docid"] = single_element_array(docidentifier) if docidentifier&.any? hash["docnumber"] = docnumber if docnumber hash["date"] = single_element_array(date) if date&.any? hash["contributor"] = single_element_array(contributor) if contributor&.any? hash["edition"] = edition if edition hash["version"] = version.to_hash if version hash["revdate"] = revdate if revdate hash["biblionote"] = single_element_array(biblionote) if biblionote&.any? hash["language"] = single_element_array(language) if language&.any? hash["script"] = single_element_array(script) if script&.any? hash["formattedref"] = formattedref.to_hash if formattedref hash["abstract"] = single_element_array(abstract) if abstract&.any? hash["docstatus"] = status.to_hash if status hash["copyright"] = copyright.to_hash if copyright hash["relation"] = single_element_array(relation) if relation&.any? hash["series"] = single_element_array(series) if series&.any? hash["medium"] = medium.to_hash if medium hash["place"] = single_element_array(place) if place&.any? hash["extent"] = single_element_array(extent) if extent&.any? hash["accesslocation"] = single_element_array(accesslocation) if accesslocation&.any? hash["classification"] = single_element_array(classification) if classification&.any? hash["validity"] = validity.to_hash if validity hash["fetched"] = fetched.to_s if fetched hash["keyword"] = single_element_array(keyword) if keyword&.any? hash["license"] = single_element_array(license) if license&.any? hash end # @param bibtex [BibTeX::Bibliography, NilClass] # @return [String] def to_bibtex(bibtex = nil) item = BibTeX::Entry.new item.type = bibtex_type item.key = id bibtex_title item item.edition = edition if edition bibtex_author item bibtex_contributor item item.address = place.first.name if place.any? bibtex_note item bibtex_relation item bibtex_extent item bibtex_date item bibtex_series item bibtex_classification item item.keywords = keyword.map(&:content).join(", ") if keyword.any? bibtex_docidentifier item item.timestamp = fetched.to_s if fetched bibtex_link item bibtex ||= BibTeX::Bibliography.new bibtex << item bibtex.to_s end # If revision_date exists then returns it else returns published date or nil # @return [String, NilClass] def revdate @revdate ||= if version&.revision_date version.revision_date else date.detect { |d| d.type == "published" }&.on&.to_s end end private # @return [String] def bibtex_title(item) title.each do |t| case t.type when "main" then item.tile = t.title.content end end end # @return [String] def bibtex_type case type when "standard", nil then "misc" else type end end # @param [BibTeX::Entry] def bibtex_author(item) authors = contributor.select do |c| c.entity.is_a?(Person) && c.role.map(&:type).include?("author") end.map &:entity return unless authors.any? item.author = authors.map do |a| if a.name.surname "#{a.name.surname}, #{a.name.forename.map(&:to_s).join(" ")}" else a.name.completename.to_s end end.join " and " end # @param [BibTeX::Entry] def bibtex_contributor(item) contributor.each do |c| rls = c.role.map(&:type) if rls.include?("publisher") then item.publisher = c.entity.name elsif rls.include?("distributor") case type when "techreport" then item.institution = c.entity.name when "inproceedings", "conference", "manual", "proceedings" item.organization = c.entity.name when "mastersthesis", "phdthesis" then item.school = c.entity.name end end end end # @param [BibTeX::Entry] def bibtex_note(item) biblionote.each do |n| case n.type when "annote" then item.annote = n.content when "howpublished" then item.howpublished = n.content when "comment" then item.comment = n.content when "tableOfContents" then item.content = n.content when nil then item.note = n.content end end end # @param [BibTeX::Entry] def bibtex_relation(item) rel = relation.detect { |r| r.type == "partOf" } item.booktitle = rel.bibitem.title.detect { |t| t.type == "main" }.title.content if rel end # @param [BibTeX::Entry] def bibtex_extent(item) extent.each do |e| case e.type when "chapter" then item.chapter = e.reference_from when "page" value = e.reference_from value += "-#{e.reference_to}" if e.reference_to item.pages = value when "volume" then item.volume = e.reference_from end end end # @param [BibTeX::Entry] def bibtex_date(item) date.each do |d| case d.type when "published" item.year = d.on.year item.month = d.on.month when "accessed" then item.urldate = d.on.to_s end end end # @param [BibTeX::Entry] def bibtex_series(item) series.each do |s| case s.type when "journal" item.journal = s.title.title item.number = s.number if s.number when nil then item.series = s.title.title end end end # @param [BibTeX::Entry] def bibtex_classification(item) classification.each do |c| case c.type when "type" then item["type"] = c.value # when "keyword" then item.keywords = c.value when "mendeley" then item["mendeley-tags"] = c.value end end end # @param [BibTeX::Entry] def bibtex_docidentifier(item) docidentifier.each do |i| case i.type when "isbn" then item.isbn = i.id when "lccn" then item.lccn = i.id when "issn" then item.issn = i.id end end end # @param [BibTeX::Entry] def bibtex_link(item) link.each do |l| case l.type when "doi" then item.doi = l.content when "file" then item.file2 = l.content when "src" then item.url = l.content end end end # rubocop:disable Metrics/AbcSize, Metrics/MethodLength # rubocop:disable Metrics/CyclomaticComplexity, Metrics/PerceivedComplexity # rubocop:disable Style/NestedParenthesizedCalls, Metrics/BlockLength # @param builder [Nokogiri::XML::Builder] # @return [String] def render_xml(builder, **opts) root = opts[:bibdata] ? :bibdata : :bibitem xml = builder.send(root) do builder.fetched fetched if fetched title.each { |t| builder.title { t.to_xml builder } } formattedref&.to_xml builder link.each { |s| s.to_xml builder } docidentifier.each { |di| di.to_xml builder } builder.docnumber docnumber if docnumber date.each { |d| d.to_xml builder, **opts } contributor.each do |c| builder.contributor do c.role.each { |r| r.to_xml builder } c.to_xml builder end end builder.edition edition if edition version&.to_xml builder biblionote.each { |n| n.to_xml builder } language.each { |l| builder.language l } script.each { |s| builder.script s } abstract.each { |a| builder.abstract { a.to_xml(builder) } } status&.to_xml builder copyright&.to_xml builder relation.each { |r| r.to_xml builder, **opts } series.each { |s| s.to_xml builder } medium&.to_xml builder place.each { |pl| pl.to_xml builder } extent.each { |e| builder.extent { e.to_xml builder } } accesslocation.each { |al| builder.accesslocation al } license.each { |l| builder.license l } classification.each { |cls| cls.to_xml builder } keyword.each { |kw| builder.keyword { kw.to_xml(builder) } } validity&.to_xml builder if block_given? yield builder end end xml[:id] = id if id && !opts[:bibdata] && !opts[:embedded] xml[:type] = type if type xml end # rubocop:enable Metrics/AbcSize, Metrics/MethodLength # rubocop:enable Metrics/CyclomaticComplexity, Metrics/PerceivedComplexity # rubocop:enable Style/NestedParenthesizedCalls, Metrics/BlockLength end end