Как проверить дату в рельсах?

Я хочу подтвердить дату в моей модели в Ruby on Rails, однако значения дня, месяца и года уже конвертируются в неправильную дату к моменту, когда они достигают моей модели.

Например, если я вхожу 31 февраля 2009 года в мой взгляд, когда я использую Model.new(params[:model]) в своем контроллере, он преобразует его в "3 марта 2009", который моя модель затем видит как действительную дату, но это неверно.

Я хотел бы иметь возможность выполнить эту проверку в моей модели. Есть ли способ, которым я могу, или я об этом совершенно неправильно?

Я нашел это "" Проверка даты", в котором обсуждалась проблема, но она никогда не была разрешена.

Ответ 1

Я предполагаю, что вы используете помощник date_select для создания тегов для даты. Другой способ, которым вы могли бы это сделать, - использовать помощник формы формы для полей day, month, year. Подобно этому (например, я использовал поле date_at created):

<%= f.select :month, (1..12).to_a, selected: @user.created_at.month %>
<%= f.select :day, (1..31).to_a, selected: @user.created_at.day %>
<%= f.select :year, ((Time.now.year - 20)..Time.now.year).to_a, selected: @user.created_at.year %>

И в модели вы подтверждаете дату:

attr_accessor :month, :day, :year
validate :validate_created_at

private

def convert_created_at
  begin
    self.created_at = Date.civil(self.year.to_i, self.month.to_i, self.day.to_i)
  rescue ArgumentError
    false
  end
end

def validate_created_at
  errors.add("Created at date", "is invalid.") unless convert_created_at
end

Если вы ищете решение для плагинов, я бы посмотрел плагин validates_timeliness. Он работает так (со страницы github):

class Person < ActiveRecord::Base
  validates_date :date_of_birth, on_or_before: lambda { Date.current }
  # or
  validates :date_of_birth, timeliness: { on_or_before: lambda { Date.current }, type: :date }
end 

Список доступных методов проверки:

validates_date     - validate value as date
validates_time     - validate value as time only i.e. '12:20pm'
validates_datetime - validate value as a full date and time
validates          - use the :timeliness key and set the type in the hash.

Ответ 2

Использование хронического драгоценного камня:

class MyModel < ActiveRecord::Base
  validate :valid_date?

  def valid_date?
    unless Chronic.parse(from_date)
      errors.add(:from_date, "is missing or invalid")
    end
  end

end

Ответ 3

Если вы хотите совместимость с Rails 3 или Ruby 1.9, попробуйте date_validator gem.

Ответ 4

Активная запись предоставляет атрибуты _before_type_cast, которые содержат исходные данные атрибута перед приведением типов. Это может быть полезно для возврата сообщений об ошибках с заранее заданными значениями или просто для проверки, которые невозможны после приведения типов.

Я бы уклонился от предложения Daniel Von Fange о том, чтобы переопределить аксессуар, потому что выполнение проверки в аксессуаре слегка изменило контракт на доступ. В этой ситуации активная запись имеет функцию явно. Используйте его.

Ответ 5

Поскольку вам нужно обработать строку даты до того, как она будет преобразована в дату в вашей модели, я бы переопределил аксессуар для этого поля

Скажем, ваше поле даты published_date. Добавьте это в объект модели:

def published_date=(value)
    # do sanity checking here
    # then hand it back to rails to convert and store
    self.write_attribute(:published_date, value) 
end

Ответ 6

Здесь не хронический ответ.

class Pimping < ActiveRecord::Base

validate :valid_date?

def valid_date?
  if scheduled_on.present?
    unless scheduled_on.is_a?(Time)
      errors.add(:scheduled_on, "Is an invalid date.")
    end
  end
end

Ответ 7

Вы можете проверить дату и время (например, где-то в вашем контроллере с доступом к вашим параметрам, если вы пользуетесь пользовательскими выборами)...

# Set parameters
year = params[:date][:year].to_i
month = params[:date][:month].to_i
mday = params[:date][:mday].to_i
hour = params[:date][:hour].to_i
minute = params[:date][:minute].to_i

# Validate date, time and hour
valid_date    = Date.valid_date? year, month, mday
valid_hour    = (0..23).to_a.include? hour
valid_minute  = (0..59).to_a.include? minute
valid_time    = valid_hour && valid_minute

# Check if parameters are valid and generate appropriate date
if valid_date && valid_time
  second = 0
  offset = '0'
  DateTime.civil(year, month, mday, hour, minute, second, offset)
else
  # Some fallback if you want like ...
  DateTime.current.utc
end

Ответ 8

Немного позже, но благодаря Как проверить дату в рельсах?" Мне удалось написать этот валидатор, надеюсь, что кому-то полезен:

Внутри model.rb

validate :date_field_must_be_a_date_or_blank

# If your field is called :date_field, use :date_field_before_type_cast
def date_field_must_be_a_date_or_blank
  date_field_before_type_cast.to_date
rescue ArgumentError
  errors.add(:birthday, :invalid)
end