BigW Consortium Gitlab

file_size_validator.rb 2.25 KB
Newer Older
gitlabhq committed
1
class FileSizeValidator < ActiveModel::EachValidator
2 3
  MESSAGES = { is: :wrong_size, minimum: :size_too_small, maximum: :size_too_big }.freeze
  CHECKS   = { is: :==, minimum: :>=, maximum: :<= }.freeze
gitlabhq committed
4

5 6
  DEFAULT_TOKENIZER = -> (value) { value.split(//) }.freeze
  RESERVED_OPTIONS  = [:minimum, :maximum, :within, :is, :tokenizer, :too_short, :too_long].freeze
gitlabhq committed
7 8 9 10

  def initialize(options)
    if range = (options.delete(:in) || options.delete(:within))
      raise ArgumentError, ":in and :within must be a Range" unless range.is_a?(Range)
11

gitlabhq committed
12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28
      options[:minimum], options[:maximum] = range.begin, range.end
      options[:maximum] -= 1 if range.exclude_end?
    end

    super
  end

  def check_validity!
    keys = CHECKS.keys & options.keys

    if keys.empty?
      raise ArgumentError, 'Range unspecified. Specify the :within, :maximum, :minimum, or :is option.'
    end

    keys.each do |key|
      value = options[key]

29 30
      unless (value.is_a?(Integer) && value >= 0) || value.is_a?(Symbol)
        raise ArgumentError, ":#{key} must be a nonnegative Integer or symbol"
gitlabhq committed
31 32 33 34 35
      end
    end
  end

  def validate_each(record, attribute, value)
Douwe Maan committed
36
    raise(ArgumentError, "A CarrierWave::Uploader::Base object was expected") unless value.is_a? CarrierWave::Uploader::Base
Nihad Abbasov committed
37

Douwe Maan committed
38
    value = (options[:tokenizer] || DEFAULT_TOKENIZER).call(value) if value.is_a?(String)
gitlabhq committed
39 40 41 42

    CHECKS.each do |key, validity_check|
      next unless check_value = options[key]

43 44 45 46 47
      check_value =
        case check_value
        when Integer
          check_value
        when Symbol
48
          record.public_send(check_value) # rubocop:disable GitlabSecurity/PublicSend
49 50
        end

gitlabhq committed
51 52 53
      value ||= [] if key == :maximum

      value_size = value.size
54
      next if value_size.public_send(validity_check, check_value) # rubocop:disable GitlabSecurity/PublicSend
gitlabhq committed
55 56 57 58 59 60 61 62 63 64

      errors_options = options.except(*RESERVED_OPTIONS)
      errors_options[:file_size] = help.number_to_human_size check_value

      default_message = options[MESSAGES[key]]
      errors_options[:message] ||= default_message if default_message

      record.errors.add(attribute, MESSAGES[key], errors_options)
    end
  end
Nihad Abbasov committed
65

gitlabhq committed
66 67 68 69 70 71 72 73 74
  def help
    Helper.instance
  end

  class Helper
    include Singleton
    include ActionView::Helpers::NumberHelper
  end
end