# frozen_string_literal: true
module Html2rss
##
# Provides a namespace for item extractors.
module ItemExtractors
##
# The Error class to be thrown when an unknown extractor name is requested.
class UnknownExtractorName < StandardError; end
##
# Maps the extractor name to the class implementing the extractor.
#
# The key is the name to use in the feed config.
NAME_TO_CLASS = {
attribute: Attribute,
href: Href,
html: Html,
static: Static,
text: Text
}.freeze
##
# Maps the extractor class to its corresponding options class.
ITEM_OPTION_CLASSES = Hash.new do |hash, klass|
hash[klass] = klass.const_get(:Options)
end
DEFAULT_EXTRACTOR = :text
##
# Retrieves an element from Nokogiri XML based on the selector.
#
# @param xml [Nokogiri::XML::Document]
# @param selector [String, nil]
# @return [Nokogiri::XML::ElementSet] selected XML elements
def self.element(xml, selector)
selector ? xml.css(selector) : xml
end
##
# Creates an instance of the requested item extractor.
#
# @param attribute_options [Hash]
# Should contain at least `:extractor` (the name) and required options for that extractor.
# @param xml [Nokogiri::XML::Document]
# @return [Object] instance of the specified item extractor class
def self.item_extractor_factory(attribute_options, xml)
extractor_name = attribute_options[:extractor]&.to_sym || DEFAULT_EXTRACTOR
extractor_class = find_extractor_class(extractor_name)
options_instance = build_options_instance(extractor_class, attribute_options)
create_extractor_instance(extractor_class, xml, options_instance)
end
##
# Finds the extractor class based on the name.
#
# @param extractor_name [Symbol] the name of the extractor
# @return [Class] the class implementing the extractor
# @raise [UnknownExtractorName] if the extractor class is not found
def self.find_extractor_class(extractor_name)
NAME_TO_CLASS[extractor_name] || raise(UnknownExtractorName,
"Unknown extractor name '#{extractor_name}' requested in NAME_TO_CLASS")
end
##
# Builds the options instance for the extractor class.
#
# @param extractor_class [Class] the class implementing the extractor
# @param attribute_options [Hash] the attribute options
# @return [Object] an instance of the options class for the extractor
def self.build_options_instance(extractor_class, attribute_options)
options = attribute_options.slice(*extractor_class::Options.members)
ITEM_OPTION_CLASSES[extractor_class].new(options)
end
##
# Creates an instance of the extractor class.
#
# @param extractor_class [Class] the class implementing the extractor
# @param xml [Nokogiri::XML::Document] the XML document
# @param options_instance [Object] the options instance
# @return [Object] an instance of the extractor class
def self.create_extractor_instance(extractor_class, xml, options_instance)
extractor_class.new(xml, options_instance)
end
end
end