mirror of
https://github.com/discourse/discourse.git
synced 2025-01-19 03:12:46 +08:00
806255b3c4
introduce a couple of custom validators fix minor discrepancies in tests copy I18n error message keys to default location clean up validation invocation move some responsibilities out of validator into class
53 lines
1.4 KiB
Ruby
53 lines
1.4 KiB
Ruby
#
|
|
# Given a string, tell us whether or not is acceptable.
|
|
#
|
|
class TextSentinel
|
|
|
|
attr_accessor :text
|
|
|
|
def initialize(text, opts=nil)
|
|
@opts = opts || {}
|
|
@text = text.to_s.encode('UTF-8', invalid: :replace, undef: :replace, replace: '')
|
|
end
|
|
|
|
def self.non_symbols_regexp
|
|
/[\ -\/\[-\`\:-\@\{-\~]/m
|
|
end
|
|
|
|
def self.body_sentinel(text)
|
|
TextSentinel.new(text, min_entropy: SiteSetting.body_min_entropy)
|
|
end
|
|
|
|
def self.title_sentinel(text)
|
|
TextSentinel.new(text,
|
|
min_entropy: SiteSetting.title_min_entropy,
|
|
max_word_length: SiteSetting.max_word_length)
|
|
end
|
|
|
|
# Entropy is a number of how many unique characters the string needs.
|
|
def entropy
|
|
@entropy ||= @text.to_s.strip.split('').uniq.size
|
|
end
|
|
|
|
def valid?
|
|
# Blank strings are not valid
|
|
@text.present? &&
|
|
|
|
# Minimum entropy if entropy check required
|
|
(@opts[:min_entropy].blank? || (entropy >= @opts[:min_entropy])) &&
|
|
|
|
# At least some non-symbol characters
|
|
# (We don't have a comprehensive list of symbols, but this will eliminate some noise)
|
|
(@text.gsub(TextSentinel.non_symbols_regexp, '').size > 0) &&
|
|
|
|
# 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] ) &&
|
|
|
|
# We don't allow all upper case content in english
|
|
not((@text =~ /[A-Z]+/) && (@text == @text.upcase)) &&
|
|
|
|
true
|
|
end
|
|
|
|
end
|