Railsで日付を検証するにはどうすればよいですか?


92

Ruby on Railsでモデルの日付を検証したいのですが、モデルに到達するまでに、日、月、年の値が既に誤った日付に変換されています。

たとえば、ビューで2009年2月31日と入力した場合Model.new(params[:model])、コントローラーで使用すると、それは「2009年3月3日」に変換されます。モデルはそれを有効な日付として認識しますが、これは正しくありません。

私のモデルでこの検証を行えるようにしたいと思います。私ができる方法はありますか、それとも完全に間違っていますか?

問題を説明するこの「日付の検証」を見つけましたが、解決されませんでした。


4
この質問は現在、Railsの日付検証に関するGoogleの上位の結果の1つです。私がしたように、最初のパスでそれを見逃すかもしれません:選択された答えの重要な部分は、validates_timeliness gemです。他の場所でvalidates_timelinessについて知ったので、残りの答えも読みませんでした。そのgemは、カスタムコードを記述しなくても、必要なすべてを実行します。
Jason Swett、

回答:


61

date_selectヘルパーを使用して日付のタグを生成していると思います。別の方法としては、日、月、年のフィールドに選択フォームヘルパーを使用する方法があります。このように(私が使用した例はcreated_at日付フィールドです):

<%= 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.

提案された解決策は私には機能せず、Rails 2.3.5を使用しています
Roger

私はそれをよりクリーンに書き直し、Rails 2.3.5に対してテストしましたが、これは私にとってはうまくいきます。
Jack Chu

こんにちはジャック、私はあなたの解決策を試しました、それは私にattr_accessible:day、:month、:yearを追加するのに役立ちます。これは私には意味がありません...何か考えがあればありがとう!
benoitr 2011年

Rails 3では、github.com / adzap / validates_timelinessを使用しています。
Jason Swett、2012

validates_timelinessのプラグイン/宝石を使用するのは本当にいいです。魅力的に機能し、コードを小さく保ちます。先端をありがとう。
mjnissim 2012

20

慢性的な宝石の使用:

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
この特定の例ではChronic.parse(from_date_before_type_cast)、文字列値の入力を取得するために使用する必要がありました。そうしないとto_date、フィールドがフィールドであるため、Railsは文字列をタイプキャストしますdatetime
カルヴィン


11

アクティブレコードは_before_type_cast、型キャストの前に生の属性データを含む属性を提供します。これは、型キャスト前の値を含むエラーメッセージを返す場合や、型キャスト後には実行できない検証を行う場合に便利です。

アクセサーで検証を行うとアクセサーコントラクトがわずかに変更されるため、ダニエルフォンファンジェによるアクセサーのオーバーライドの提案は避けます。Active Recordには、この状況に明確に対応する機能があります。これを使って。


4

モデルで日付に変換される前に日付文字列を処理する必要があるため、そのフィールドのアクセサーをオーバーライドします

あなたの日付フィールドがであるとしましょう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

これをこの目的に使用するかどうかはわかりませんが、素晴らしい提案です。
Dan Rosenstark、2009

3

ここで少し遅れましたが、「レールで日付を検証するにはどうすればよいですか?」のおかげで、このバリデーターをなんとか書けました。誰かに役立つと思います

あなたの中 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

1

これが非慢性的な答えです。

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

1
これは常にfalseを返します。- "1/1/2013".is_a?(Date)日付はユーザー入力から取得されるため、文字列として供給されます。最初に日付として解析する必要がありますか?
モハマド2013

正しい。Date.parse("1/1/2013").is_a?(Date)ですが、まったく解析されない場合は、おそらく日付でもないことがわかります。
旅行

1
引数エラーを救済する必要があります...ああ、それが文字列を自動的に解析しただけなら素晴らしいでしょう...まあ、そのための宝石があります。
モハマド2013

0

そのようにして日付と時刻を検証できます(カスタム選択を使用している場合は、パラメーターにアクセスできるコントローラー内のどこかのメソッドで)...

# 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

-3

validates_date_timeプラグインを試しましたか?


このリンクで質問に答えることができますが、回答の重要な部分をここに含め、参照用のリンクを提供することをお勧めします。リンクされたページが変更されると、リンクのみの回答が無効になる可能性があります。
ピナル14年

私は自分の答えが好きです。二人は同意します。
Ryan Duffield、2014年

4
@Pinalリンクは本当に死んでいます。
ウェインコンラッド
弊社のサイトを使用することにより、あなたは弊社のクッキーポリシーおよびプライバシーポリシーを読み、理解したものとみなされます。
Licensed under cc by-sa 3.0 with attribution required.