# encoding: utf-8 require 'fast-stemmer' require 'highscore' require 'htmlentities' module Ebooks module NLP # We deliberately limit our punctuation handling to stuff we can do consistently # It'll just be a part of another token if we don't split it out, and that's fine PUNCTUATION = ".?!," # Lazy-load NLP libraries and resources # Some of this stuff is pretty heavy and we don't necessarily need # to be using it all of the time # Lazily loads an array of stopwords # Stopwords are common words that should often be ignored # @return [Array] def self.stopwords @stopwords ||= File.exists?('stopwords.txt') ? File.read('stopwords.txt').split : [] end # Lazily loads an array of known English nouns # @return [Array] def self.nouns @nouns ||= File.read(File.join(DATA_PATH, 'nouns.txt')).split end # Lazily loads an array of known English adjectives # @return [Array] def self.adjectives @adjectives ||= File.read(File.join(DATA_PATH, 'adjectives.txt')).split end # Lazily load part-of-speech tagging library # This can determine whether a word is being used as a noun/adjective/verb # @return [EngTagger] def self.tagger require 'engtagger' @tagger ||= EngTagger.new end # Lazily load HTML entity decoder # @return [HTMLEntities] def self.htmlentities @htmlentities ||= HTMLEntities.new end ### Utility functions # Normalize some strange unicode punctuation variants # @param text [String] # @return [String] def self.normalize(text) htmlentities.decode text.gsub('“', '"').gsub('”', '"').gsub('’', "'").gsub('…', '...') end # Split text into sentences # We use ad hoc approach because fancy libraries do not deal # especially well with tweet formatting, and we can fake solving # the quote problem during generation # @param text [String] # @return [Array] def self.sentences(text) text.split(/\n+|(?<=[.?!])\s+/) end # Split a sentence into word-level tokens # As above, this is ad hoc because tokenization libraries # do not behave well wrt. things like emoticons and timestamps # @param sentence [String] # @return [Array] def self.tokenize(sentence) regex = /\s+|(?<=[#{PUNCTUATION}]\s)(?=[a-zA-Z])|(?<=[a-zA-Z])(?=[#{PUNCTUATION}]+\s)/ sentence.split(regex) end # Get the 'stem' form of a word e.g. 'cats' -> 'cat' # @param word [String] # @return [String] def self.stem(word) Stemmer::stem_word(word.downcase) end # Use highscore gem to find interesting keywords in a corpus # @param text [String] # @return [Highscore::Keywords] def self.keywords(text) # Preprocess to remove stopwords (highscore's blacklist is v. slow) text = NLP.tokenize(text).reject { |t| stopword?(t) }.join(' ') text = Highscore::Content.new(text) text.configure do #set :multiplier, 2 #set :upper_case, 3 #set :long_words, 2 #set :long_words_threshold, 15 #set :vowels, 1 # => default: 0 = not considered #set :consonants, 5 # => default: 0 = not considered #set :ignore_case, true # => default: false set :word_pattern, /(? default: /\w+/ #set :stemming, true # => default: false end text.keywords end # Builds a proper sentence from a list of tikis # @param tikis [Array] # @param tokens [Array] # @return [String] def self.reconstruct(tikis, tokens) text = "" last_token = nil tikis.each do |tiki| next if tiki == INTERIM token = tokens[tiki] text += ' ' if last_token && space_between?(last_token, token) text += token last_token = token end text end # Determine if we need to insert a space between two tokens # @param token1 [String] # @param token2 [String] # @return [Boolean] def self.space_between?(token1, token2) p1 = self.punctuation?(token1) p2 = self.punctuation?(token2) if p1 && p2 # "foo?!" false elsif !p1 && p2 # "foo." false elsif p1 && !p2 # "foo. rah" true else # "foo rah" true end end # Is this token comprised of punctuation? # @param token [String] # @return [Boolean] def self.punctuation?(token) (token.chars.to_set - PUNCTUATION.chars.to_set).empty? end # Is this token a stopword? # @param token [String] # @return [Boolean] def self.stopword?(token) @stopword_set ||= stopwords.map(&:downcase).to_set @stopword_set.include?(token.downcase) end # Determine if a sample of text contains unmatched brackets or quotes # This is one of the more frequent and noticeable failure modes for # the generator; we can just tell it to retry # @param text [String] # @return [Boolean] def self.unmatched_enclosers?(text) enclosers = ['**', '""', '()', '[]', '``', "''"] enclosers.each do |pair| starter = Regexp.new('(\W|^)' + Regexp.escape(pair[0]) + '\S') ender = Regexp.new('\S' + Regexp.escape(pair[1]) + '(\W|$)') opened = 0 tokenize(text).each do |token| opened += 1 if token.match(starter) opened -= 1 if token.match(ender) return true if opened < 0 # Too many ends! end return true if opened != 0 # Mismatch somewhere. end false end # Determine if a2 is a subsequence of a1 # @param a1 [Array] # @param a2 [Array] # @return [Boolean] def self.subseq?(a1, a2) !a1.each_index.find do |i| a1[i...i+a2.length] == a2 end.nil? end end end