discourse/lib/text_sentinel.rb

66 lines
1.6 KiB
Ruby
Raw Normal View History

#
2013-04-10 17:00:50 +08:00
# Given a string, tell us whether or not is acceptable.
#
class TextSentinel
attr_accessor :text
2013-04-10 17:00:50 +08:00
def initialize(text, opts=nil)
@opts = opts || {}
@text = text.to_s.encode('UTF-8', invalid: :replace, undef: :replace, replace: '')
2013-04-10 17:00:50 +08:00
end
def self.body_sentinel(text)
TextSentinel.new(text, min_entropy: SiteSetting.body_min_entropy)
end
2013-02-09 05:55:40 +08:00
def self.title_sentinel(text)
2013-02-26 00:42:20 +08:00
TextSentinel.new(text,
2013-02-09 05:55:40 +08:00
min_entropy: SiteSetting.title_min_entropy,
2013-04-10 17:00:50 +08:00
max_word_length: SiteSetting.max_word_length)
2013-02-09 05:55:40 +08:00
end
2013-02-26 00:42:20 +08:00
# Entropy is a number of how many unique characters the string needs.
def entropy
@entropy ||= @text.to_s.strip.split('').uniq.size
end
2013-02-26 00:42:20 +08:00
def valid?
@text.present? &&
seems_meaningful? &&
seems_pronounceable? &&
seems_unpretentious? &&
seems_quiet? &&
true
end
private
def symbols_regex
/[\ -\/\[-\`\:-\@\{-\~]/m
end
def seems_meaningful?
# Minimum entropy if entropy check required
@opts[:min_entropy].blank? || (entropy >= @opts[:min_entropy])
end
def seems_pronounceable?
# At least some non-symbol characters
# (We don't have a comprehensive list of symbols, but this will eliminate some noise)
@text.gsub(symbols_regex, '').size > 0
end
def seems_unpretentious?
# Don't allow super long words if there is a word length maximum
@opts[:max_word_length].blank? || (@text.split(/\W/).map(&:size).max <= @opts[:max_word_length])
end
def seems_quiet?
# We don't allow all upper case content in english
not((@text =~ /[A-Z]+/) && (@text == @text.upcase))
end
end