2013-02-07 09:09:31 +08:00
|
|
|
#
|
2013-04-10 17:00:50 +08:00
|
|
|
# Given a string, tell us whether or not is acceptable.
|
2013-02-07 09:09:31 +08:00
|
|
|
#
|
|
|
|
class TextSentinel
|
|
|
|
|
|
|
|
attr_accessor :text
|
|
|
|
|
2013-04-10 17:00:50 +08:00
|
|
|
def initialize(text, opts=nil)
|
|
|
|
@opts = opts || {}
|
2013-05-23 12:52:12 +08:00
|
|
|
@text = text.to_s.encode('UTF-8', invalid: :replace, undef: :replace, replace: '')
|
2013-04-10 17:00:50 +08:00
|
|
|
end
|
|
|
|
|
2013-06-13 16:18:17 +08:00
|
|
|
def self.body_sentinel(text, opts={})
|
|
|
|
entropy = SiteSetting.body_min_entropy
|
|
|
|
if opts[:private_message]
|
|
|
|
scale_entropy = SiteSetting.min_private_message_post_length.to_f / SiteSetting.min_post_length.to_f
|
|
|
|
entropy = (entropy * scale_entropy).to_i
|
|
|
|
end
|
|
|
|
TextSentinel.new(text, min_entropy: entropy)
|
2013-02-07 09:09:31 +08:00
|
|
|
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.
|
2013-06-08 02:47:07 +08:00
|
|
|
# Non-ASCII characters are weighted heavier since they contain more "information"
|
2013-02-07 09:09:31 +08:00
|
|
|
def entropy
|
2013-06-08 02:47:07 +08:00
|
|
|
chars = @text.to_s.strip.split('')
|
|
|
|
@entropy ||= chars.pack('M*'*chars.size).gsub("\n",'').split('=').uniq.size
|
2013-02-07 09:09:31 +08:00
|
|
|
end
|
|
|
|
|
2013-02-26 00:42:20 +08:00
|
|
|
def valid?
|
2013-05-23 12:52:12 +08:00
|
|
|
@text.present? &&
|
2013-05-24 01:48:37 +08:00
|
|
|
seems_meaningful? &&
|
|
|
|
seems_pronounceable? &&
|
|
|
|
seems_unpretentious? &&
|
2013-06-18 13:49:10 +08:00
|
|
|
seems_quiet?
|
2013-05-24 01:48:37 +08:00
|
|
|
end
|
|
|
|
|
|
|
|
private
|
2013-02-07 09:09:31 +08:00
|
|
|
|
2013-05-24 01:48:37 +08:00
|
|
|
def symbols_regex
|
|
|
|
/[\ -\/\[-\`\:-\@\{-\~]/m
|
|
|
|
end
|
2013-02-07 09:09:31 +08:00
|
|
|
|
2013-05-24 01:48:37 +08:00
|
|
|
def seems_meaningful?
|
2013-05-23 12:52:12 +08:00
|
|
|
# Minimum entropy if entropy check required
|
2013-05-24 01:48:37 +08:00
|
|
|
@opts[:min_entropy].blank? || (entropy >= @opts[:min_entropy])
|
|
|
|
end
|
2013-02-07 09:09:31 +08:00
|
|
|
|
2013-05-24 01:48:37 +08:00
|
|
|
def seems_pronounceable?
|
2013-05-23 12:52:12 +08:00
|
|
|
# At least some non-symbol characters
|
|
|
|
# (We don't have a comprehensive list of symbols, but this will eliminate some noise)
|
2013-05-24 01:48:37 +08:00
|
|
|
@text.gsub(symbols_regex, '').size > 0
|
|
|
|
end
|
2013-02-07 09:09:31 +08:00
|
|
|
|
2013-05-24 01:48:37 +08:00
|
|
|
def seems_unpretentious?
|
2013-05-23 12:52:12 +08:00
|
|
|
# Don't allow super long words if there is a word length maximum
|
2013-05-27 22:56:55 +08:00
|
|
|
@opts[:max_word_length].blank? || @text.split(/\s/).map(&:size).max <= @opts[:max_word_length]
|
2013-05-24 01:48:37 +08:00
|
|
|
end
|
2013-02-07 09:09:31 +08:00
|
|
|
|
|
|
|
|
2013-05-24 01:48:37 +08:00
|
|
|
def seems_quiet?
|
2013-02-15 05:09:57 +08:00
|
|
|
# We don't allow all upper case content in english
|
2013-06-18 00:22:23 +08:00
|
|
|
not((@text =~ /[A-Z]+/) && !(@text =~ /[^[:ascii:]]/) && (@text == @text.upcase))
|
2013-02-07 09:09:31 +08:00
|
|
|
end
|
|
|
|
|
|
|
|
end
|