# encoding: utf-8 require_relative 'monkeypatches' require_relative '../utils/logging' # @author Alexei Matyushkin module Qipowl::Bowlers # Base class for all the parsers. # # Technically it may be instantiated, but that’s meaningless. # Main operation method for it and all the descendants is # {#parse}. It sequentially executes following # private methods: # # - {#defreeze} # - {#roast} # - {#serveup} # # Normally the developer does not need to interfere the {#roast} # method which proceeds the input string. To prepare the input # for +evaluation+ one overwrites {#defreeze}, for some afterwork # the {#serveup} method is here. # # Descendants are supposed to overwrite {#method_missing} for some # custom processing and introduce DSL methods, which will be executed # by `eval` inside the {#roast} method. # # Instance variables `:in` and `:out` are here to gain an access to # the last interpreted input string and the result of evaluation # respectively. # class Bowler include TypoLogging attr_reader :in, :out # Internal constant for joining/splitting the strings during processing. # Override on your own risk. I can′t imagine why you would need to do so. SEPARATOR = $, || ' ' def execute str @out = (serveup roast defreeze @in = str) end # Everything is a DSL, remember? # # @return true def respond_to?(method) true end # Everything is a DSL, remember? Even constants. # @todo This fails to do with DSLing words, beginning with capitals :-( # # @return the constant name as is def self.const_missing name raise "There was CONST [#{name}] met. Stupid programming error." name end # If somebody needs to interfere the standard processing, # she supposed to introduce `special_handler` method. The descendants # will be processed before standard operation (which in fact simply # collects words within an array one by one.) def method_missing method, *args, &block method, *args = special_handler(method, *args, &block) \ if self.private_methods.include?(:special_handler) [method, args].flatten end # Adds new +entity+ in the section specified. # E. g., call to # # add_spice :linewide, :°, :deg, :degrees # # in HTML implementation adds a support for specifying something like: # # ° 15 # ° 30 # ° 45 # # which is to be converted to the following: # # # 15 # 30 # 45 # # # @param [Symbol] section the section (it must be one of {Mapping.SPICES}) to add new key to # @param [Symbol] key the name for the key # @param [Symbol] value the value # @param [Symbol] enclosure_value optional value to be added for the key into enclosures section def add_entity section, key, value, enclosure_value = null if (tags = self.class.const_get("#{section.upcase}_TAGS")) key = key.bowl.to_sym tags[key] = value.to_sym self.class.const_get("ENCLOSURES_TAGS")[key] = enclosure_value.to_sym if enclosure_value self.class.class_eval %Q{ alias_method :#{key}, :∀_#{section} } # unless self.class.instance_methods(true).include?(key.bowl) else logger.warn "Trying to add key “#{key}” in an invalid section “#{section}”. Ignoring…" end end # Removes key from both {Mapping.SPICES} and {Mapping.SALT}. See {#add_spice} # # @param [Symbol] key the key to be removed def remove_entity key %w(block alone magnet grip regular).each { |section| next unless send :"∃_#{section}_tag", key.to_sym self.class.const_get("#{section.upcase}_TAGS").delete key self.class.const_get("ENCLOSURES_TAGS").delete key self.class.class_eval %Q{ remove_method :#{key.bowl} } } end protected %w(block alone magnet grip regular).each { |section| define_method "∀_#{section}".to_sym, ->(*args) { raise "Default method for #{section} (“#{self.class.name.gsub(/_\d+\Z/, '')}#∀_#{section}”) MUST be defined" } } def defreeze str str end def roast str (split str).reverse.map { |dish| @yielded = [] rest = begin eval(dish.strip.carriage) rescue Exception => e msg = e.message.dup logger.error '='*78 logger.error "Could not roast dish [#{msg.force_encoding(Encoding::UTF_8)}].\nWill return as is… Dish follows:" logger.error '-'*78 logger.error dish logger.error '='*78 [*dish] end harvest(nil, orphan([*rest].join(SEPARATOR))) # FIXME Check if this is correct in all the cases @yielded.pop(@yielded.size).reverse.join(SEPARATOR) }.reverse.join($/).uncarriage.un␚ify.unspacefy.unbowl end def serveup str str.gsub(/⌦./, '').gsub(/.⌫/, '') end protected # The handler of the last “orphaned” text block in the input string. # # E.g.: # # Here goes a quite significant list: # # • line item 1 #  • nested li1 #  • nested li 2 # • line item 2 # # While all line items are operated by `•` method, the top sentence # is orphaned (has no prepnding DSL method to be called on.) # Since we still need to handle it somehow, the {#orphan} method is here. # # @param [String] str the string to be operated “bu default rule” # @return [String] the processed input (in derivatives, here it returns the untouched input string itself) def orphan str str end # The handler for harvesting partial result. # # Processing sometimes calls this method, designating the meaningful # part of input text is done and should be yielded. E.g. when the # block of code is processed: # # Λ ruby # @mapping[:inplace].each { |tag, htmltag| # do_smth tag, htmltag # } # Λ # # After we have this part of input processed, it should be considered # “done.” So block processors call {#harvest} to store processed parts. # # @param [Symbol] callee of this method. Qipowl hardly relies on method namings and sometimes we may need to know if the call was made by, say, lineitem DSL (`•`), not datalist (`▷`). # @param [String] str string to yield # # @return nil def harvest callee, str @yielded << str unless str.vacant? nil end private # Prepares blocks in the input for the execution def block str result = str.dup self.class::BLOCK_TAGS.each { |tag, value| result.gsub!(/(#{tag})\s*(\S*\s*|$)(.*?)(#{tag}|\Z)/m) { %Q{ #{$1}('#{$2.strip}', '#{$3.carriage}') } } } result end # Prepares customs in the input for the execution def custom str result = str.unbowl.dup self.class::CUSTOM_TAGS.each { |tag, value| result.gsub!(/#{tag}/, value) } result.bowl end # Prepares grips in the input for the execution # FIX