mirror of
https://github.com/ditkrg/validates_timeliness.git
synced 2026-01-25 15:22:58 +00:00
Compare commits
69 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
54ae49b8a0 | ||
|
|
2fef0a23d6 | ||
|
|
12f6d2a467 | ||
|
|
913bd4ccfa | ||
|
|
f1b8925a36 | ||
|
|
f983dcc9dc | ||
|
|
7be1539cee | ||
|
|
99ae8fe7d7 | ||
|
|
00ce472d3e | ||
|
|
0d2c7ce554 | ||
|
|
3d798697e1 | ||
|
|
dc0fdc0340 | ||
|
|
dd3b6b5514 | ||
|
|
609fafe7bb | ||
|
|
df9677f5bf | ||
|
|
b463b4356a | ||
|
|
7dd579b0e0 | ||
|
|
5becd7886b | ||
|
|
2225c747e1 | ||
|
|
a1dfbf5d7d | ||
|
|
02fbdc6028 | ||
|
|
4fe22458d3 | ||
|
|
8c698be4c4 | ||
|
|
091e7ecfa0 | ||
|
|
8168bcbd3a | ||
|
|
973bbfa82c | ||
|
|
62557e7e04 | ||
|
|
fd73c4eccd | ||
|
|
7bcdea1738 | ||
|
|
8898b8686c | ||
|
|
aad2db8662 | ||
|
|
8e08cbf6e4 | ||
|
|
8dd607975b | ||
|
|
b6acda539f | ||
|
|
4aa20bb002 | ||
|
|
df12d6dad0 | ||
|
|
f11255a7a3 | ||
|
|
a6a3dff4d4 | ||
|
|
ff594d5873 | ||
|
|
8c1ba048df | ||
|
|
48fd72ae5b | ||
|
|
43f49076fb | ||
|
|
64a7440de4 | ||
|
|
2702ec3266 | ||
|
|
20549c7acd | ||
|
|
4acd0180f9 | ||
|
|
3f4615e053 | ||
|
|
84316bc73a | ||
|
|
5be45b00db | ||
|
|
f8aeeca0a9 | ||
|
|
f5e2deeb73 | ||
|
|
acd10f7b64 | ||
|
|
68b2579ca0 | ||
|
|
7a784a6c54 | ||
|
|
907fd3e439 | ||
|
|
e44e9d2f11 | ||
|
|
86b7bc4829 | ||
|
|
34824bbbab | ||
|
|
11ae94cba6 | ||
|
|
473da0fa5e | ||
|
|
e7b503ae4f | ||
|
|
af9d6cdba4 | ||
|
|
9a99c47323 | ||
|
|
fbc2867197 | ||
|
|
969edfcb83 | ||
|
|
1faf0efa4b | ||
|
|
2b53d713bc | ||
|
|
ac90ab96cd | ||
|
|
b4c1a39343 |
2
.gitignore
vendored
2
.gitignore
vendored
@@ -1,3 +1,5 @@
|
||||
pkg/
|
||||
.bundle/
|
||||
.rvmrc
|
||||
Gemfile.lock
|
||||
gemfiles/*.lock
|
||||
|
||||
27
Appraisals
Normal file
27
Appraisals
Normal file
@@ -0,0 +1,27 @@
|
||||
appraise "rails_3_0" do
|
||||
gem "rails", "~> 3.0.0"
|
||||
end
|
||||
|
||||
appraise "rails_3_1" do
|
||||
gem "rails", "~> 3.1.0"
|
||||
end
|
||||
|
||||
appraise "rails_3_2" do
|
||||
gem "rails", "~> 3.2.0"
|
||||
end
|
||||
|
||||
appraise "mongoid_2_1" do
|
||||
gem "mongoid", "~> 2.1.0"
|
||||
end
|
||||
|
||||
appraise "mongoid_2_2" do
|
||||
gem "mongoid", "~> 2.2.0"
|
||||
end
|
||||
|
||||
appraise "mongoid_2_3" do
|
||||
gem "mongoid", "~> 2.3.0"
|
||||
end
|
||||
|
||||
appraise "mongoid_2_4" do
|
||||
gem "mongoid", "~> 2.4.0"
|
||||
end
|
||||
@@ -1,3 +1,39 @@
|
||||
= 3.0.15 [2015-12-29]
|
||||
* Fixes mongoid 3 support and removes mongoid 2 support(johnnyshields)
|
||||
* Some documentation/comments tidying
|
||||
* Some general tidying up
|
||||
|
||||
= 3.0.14 [2012-08-23]
|
||||
* Fix for using validates :timeliness => {} form to correctly add attributes to timeliness validated attributes.
|
||||
|
||||
= 3.0.13 [2012-08-21]
|
||||
* Fix ActiveRecord issues with using plugin parser by using old way of caching values.
|
||||
* Allow any ActiveRecord non-column attribute to be validated
|
||||
|
||||
= 3.0.12 [2012-06-23]
|
||||
* Fix load order issue when relying on Railtie to load ActiveRecord extension
|
||||
|
||||
= 3.0.11 [2012-04-01]
|
||||
* Change dependency on Timeliness version due to a broken release
|
||||
|
||||
= 3.0.10 [2012-03-26]
|
||||
* Fix for ActiveRecord shim and validation with :allow_blank => true in AR 3.1+. Fixes issue#52.
|
||||
|
||||
= 3.0.9 [2012-03-26]
|
||||
* ActiveRecord 3.1+ suport
|
||||
* Fixes for multiparameter extension with empty date values (thanks @mogox, @Sharagoz)
|
||||
|
||||
= 3.0.8 [2011-12-24]
|
||||
* Remove deprecated InstanceMethods module when using AS::Concern (carlosantoniodasilva)
|
||||
* Update Mongoid shim for v2.3 compatability.
|
||||
|
||||
= 3.0.7 [2011-09-21]
|
||||
* Fix ActiveRecord before_type_cast extension for non-dirty attributes.
|
||||
* Don't override AR before_type_cast for >= 3.1.0 which now has it's own implementation for date/time attributes.
|
||||
* Fix DateTimeSelect extension to convert params to integers (#45)
|
||||
* Add #change method to DateTimeSelect extension (@trusche, #45)
|
||||
* Cleanup Mongoid shim.
|
||||
|
||||
= 3.0.6 [2011-05-09]
|
||||
* Fix for AR type conversion for date columns when using plugin parser.
|
||||
* Add timeliness_type_cast_code for ORM specific type casting after parsing.
|
||||
|
||||
15
Gemfile
15
Gemfile
@@ -2,15 +2,20 @@ source 'http://rubygems.org'
|
||||
|
||||
gemspec
|
||||
|
||||
gem 'ZenTest'
|
||||
gem 'rails', '3.0.7'
|
||||
gem 'rspec', '~> 2.4'
|
||||
gem 'rspec-rails', '~> 2.4'
|
||||
gem 'rails', '~> 3.2.6'
|
||||
gem 'rspec', '~> 2.8'
|
||||
gem 'rspec-rails', '~> 2.8'
|
||||
gem 'timecop'
|
||||
gem 'rspec_tag_matchers'
|
||||
gem 'ruby-debug', :platforms => [:ruby_18, :jruby]
|
||||
gem 'debugger', :platforms => [:ruby_19]
|
||||
gem 'appraisal'
|
||||
gem 'sqlite3'
|
||||
|
||||
group :mongoid do
|
||||
gem 'mongoid', '2.0.0.rc.8'
|
||||
gem 'mongoid', '>= 3.0'
|
||||
gem 'bson_ext'
|
||||
gem 'system_timer', :platforms => [:ruby_18]
|
||||
end
|
||||
|
||||
group :active_record do
|
||||
|
||||
118
Gemfile.lock
118
Gemfile.lock
@@ -1,118 +0,0 @@
|
||||
PATH
|
||||
remote: .
|
||||
specs:
|
||||
validates_timeliness (3.0.6)
|
||||
timeliness (~> 0.3.3)
|
||||
|
||||
GEM
|
||||
remote: http://rubygems.org/
|
||||
specs:
|
||||
ZenTest (4.5.0)
|
||||
abstract (1.0.0)
|
||||
actionmailer (3.0.7)
|
||||
actionpack (= 3.0.7)
|
||||
mail (~> 2.2.15)
|
||||
actionpack (3.0.7)
|
||||
activemodel (= 3.0.7)
|
||||
activesupport (= 3.0.7)
|
||||
builder (~> 2.1.2)
|
||||
erubis (~> 2.6.6)
|
||||
i18n (~> 0.5.0)
|
||||
rack (~> 1.2.1)
|
||||
rack-mount (~> 0.6.14)
|
||||
rack-test (~> 0.5.7)
|
||||
tzinfo (~> 0.3.23)
|
||||
activemodel (3.0.7)
|
||||
activesupport (= 3.0.7)
|
||||
builder (~> 2.1.2)
|
||||
i18n (~> 0.5.0)
|
||||
activerecord (3.0.7)
|
||||
activemodel (= 3.0.7)
|
||||
activesupport (= 3.0.7)
|
||||
arel (~> 2.0.2)
|
||||
tzinfo (~> 0.3.23)
|
||||
activeresource (3.0.7)
|
||||
activemodel (= 3.0.7)
|
||||
activesupport (= 3.0.7)
|
||||
activesupport (3.0.7)
|
||||
arel (2.0.9)
|
||||
bson (1.3.0)
|
||||
builder (2.1.2)
|
||||
diff-lcs (1.1.2)
|
||||
erubis (2.6.6)
|
||||
abstract (>= 1.0.0)
|
||||
i18n (0.5.0)
|
||||
mail (2.2.19)
|
||||
activesupport (>= 2.3.6)
|
||||
i18n (>= 0.4.0)
|
||||
mime-types (~> 1.16)
|
||||
treetop (~> 1.4.8)
|
||||
mime-types (1.16)
|
||||
mongo (1.3.0)
|
||||
bson (>= 1.3.0)
|
||||
mongoid (2.0.0.rc.8)
|
||||
activemodel (~> 3.0)
|
||||
mongo (~> 1.2)
|
||||
tzinfo (~> 0.3.22)
|
||||
will_paginate (~> 3.0.pre)
|
||||
nokogiri (1.4.4)
|
||||
polyglot (0.3.1)
|
||||
rack (1.2.2)
|
||||
rack-mount (0.6.14)
|
||||
rack (>= 1.0.0)
|
||||
rack-test (0.5.7)
|
||||
rack (>= 1.0)
|
||||
rails (3.0.7)
|
||||
actionmailer (= 3.0.7)
|
||||
actionpack (= 3.0.7)
|
||||
activerecord (= 3.0.7)
|
||||
activeresource (= 3.0.7)
|
||||
activesupport (= 3.0.7)
|
||||
bundler (~> 1.0)
|
||||
railties (= 3.0.7)
|
||||
railties (3.0.7)
|
||||
actionpack (= 3.0.7)
|
||||
activesupport (= 3.0.7)
|
||||
rake (>= 0.8.7)
|
||||
thor (~> 0.14.4)
|
||||
rake (0.8.7)
|
||||
rspec (2.5.0)
|
||||
rspec-core (~> 2.5.0)
|
||||
rspec-expectations (~> 2.5.0)
|
||||
rspec-mocks (~> 2.5.0)
|
||||
rspec-core (2.5.1)
|
||||
rspec-expectations (2.5.0)
|
||||
diff-lcs (~> 1.1.2)
|
||||
rspec-mocks (2.5.0)
|
||||
rspec-rails (2.5.0)
|
||||
actionpack (~> 3.0)
|
||||
activesupport (~> 3.0)
|
||||
railties (~> 3.0)
|
||||
rspec (~> 2.5.0)
|
||||
rspec_tag_matchers (1.0.0)
|
||||
nokogiri (>= 1.4.0)
|
||||
rspec-rails (>= 1.2.6)
|
||||
sqlite3 (1.3.3)
|
||||
sqlite3-ruby (1.3.3)
|
||||
sqlite3 (>= 1.3.3)
|
||||
thor (0.14.6)
|
||||
timecop (0.3.5)
|
||||
timeliness (0.3.3)
|
||||
treetop (1.4.9)
|
||||
polyglot (>= 0.3.1)
|
||||
tzinfo (0.3.27)
|
||||
will_paginate (3.0.pre2)
|
||||
|
||||
PLATFORMS
|
||||
ruby
|
||||
|
||||
DEPENDENCIES
|
||||
ZenTest
|
||||
mongoid (= 2.0.0.rc.8)
|
||||
rails (= 3.0.7)
|
||||
rspec (~> 2.4)
|
||||
rspec-rails (~> 2.4)
|
||||
rspec_tag_matchers
|
||||
sqlite3-ruby
|
||||
timecop
|
||||
validates_timeliness!
|
||||
32
README.rdoc
32
README.rdoc
@@ -1,7 +1,7 @@
|
||||
= ValidatesTimeliness
|
||||
|
||||
* Source: http://github.com/adzap/validates_timeliness
|
||||
* Bugs: http://github.com/adzap/validates_timeliness/issues
|
||||
* Issues: http://github.com/adzap/validates_timeliness/issues
|
||||
|
||||
== Description
|
||||
|
||||
@@ -18,24 +18,19 @@ If you a looking for the old version for Rails 2.x go here[http://github.com/adz
|
||||
|
||||
* Only Rails date/time validation plugin offering complete validation (See ORM/ODM support)
|
||||
|
||||
* Adds extensions to fix Rails date/time select issues (See Extensions)
|
||||
|
||||
* Uses extensible date/time parser (Using {timeliness gem}[http://github.com/adzap/timeliness]. See Plugin Parser)
|
||||
|
||||
* Adds extensions to fix Rails date/time select issues (See Extensions)
|
||||
|
||||
* Supports I18n for the error messages
|
||||
|
||||
* Supports Ruby 1.8.x, 1.9.x and Rubinius.
|
||||
* Supports all the Rubies (that any sane person would be using in production).
|
||||
|
||||
|
||||
== Installation
|
||||
|
||||
As plugin (from master)
|
||||
|
||||
rails plugin install git://github.com/adzap/validates_timeliness.git
|
||||
|
||||
As gem
|
||||
|
||||
# in Gemfile
|
||||
gem 'validates_timeliness', '~> 3.0.2'
|
||||
gem 'validates_timeliness', '~> 3.0'
|
||||
|
||||
# Run bundler
|
||||
$ bundle install
|
||||
@@ -44,7 +39,7 @@ Then run
|
||||
|
||||
$ rails generate validates_timeliness:install
|
||||
|
||||
This creates configuration initializer and locale files. In the initializer, you there are a number of config
|
||||
This creates configuration initializer and locale files. In the initializer, there are a number of config
|
||||
options to customize the plugin.
|
||||
|
||||
NOTE: You may wish to enable the plugin parser and the extensions to start. Please read those sections first.
|
||||
@@ -60,7 +55,10 @@ NOTE: You may wish to enable the plugin parser and the extensions to start. Plea
|
||||
validates_datetime :finish_time, :after => :start_time # Method symbol
|
||||
|
||||
validates_date :booked_at, :on => :create, :on_or_after => :today # See Restriction Shorthand.
|
||||
validates_time :booked_at, :between => ['9.00am', '5:00pm']
|
||||
|
||||
validates_time :booked_at, :between => ['9:00am', '5:00pm'] # On or after 9:00AM and on or before 5:00PM
|
||||
validates_time :booked_at, :between => '9:00am'..'5:00pm' # The same as previous example
|
||||
validates_time :booked_at, :between => '9:00am'...'5:00pm' # On or after 9:00AM and strictly before 5:00PM
|
||||
|
||||
validates_time :breakfast_time, :on_or_after => '6:00am',
|
||||
:on_or_after_message => 'must be after opening time',
|
||||
@@ -79,7 +77,7 @@ validation method
|
||||
validates :date_of_birth, :timeliness => {:on_or_before => lambda { Date.current }, :type => :date}
|
||||
end
|
||||
|
||||
# or even on a specific record, per ActiveModel API.
|
||||
or even on a specific record, per ActiveModel API.
|
||||
|
||||
@person.validates_date :date_of_birth, :on_or_before => lambda { Date.current }
|
||||
|
||||
@@ -178,8 +176,8 @@ You can also use validation options for custom error messages. The following opt
|
||||
:after_message
|
||||
:on_or_after_message
|
||||
|
||||
Note: There is no :between_message option. The between error message should be defined using the
|
||||
:on_or_before and :on_or_after messages.
|
||||
Note: There is no :between_message option. The between error message should be defined using the :on_or_after and :on_or_before
|
||||
(:before in case when :between argument is a Range with excluded high value, see Examples) messages.
|
||||
|
||||
It is highly recommended you use the I18n system for error messages.
|
||||
|
||||
@@ -298,4 +296,4 @@ To see the generous people who have contributed code, take a look at the {contri
|
||||
|
||||
== License
|
||||
|
||||
Copyright (c) 2008-2010 Adam Meehan, released under the MIT license
|
||||
Copyright (c) 2008 Adam Meehan, released under the MIT license
|
||||
|
||||
6
Rakefile
6
Rakefile
@@ -1,7 +1,11 @@
|
||||
require 'bundler'
|
||||
require 'bundler/setup'
|
||||
|
||||
require 'appraisal'
|
||||
|
||||
Bundler::GemHelper.install_tasks
|
||||
|
||||
require 'rake/rdoctask'
|
||||
require 'rdoc/task'
|
||||
require 'rspec/core/rake_task'
|
||||
|
||||
desc "Run specs"
|
||||
|
||||
16
gemfiles/mongoid_2_1.gemfile
Normal file
16
gemfiles/mongoid_2_1.gemfile
Normal file
@@ -0,0 +1,16 @@
|
||||
# This file was generated by Appraisal
|
||||
|
||||
source "http://rubygems.org"
|
||||
|
||||
gem "rails", "~> 3.2.6"
|
||||
gem "rspec", "~> 2.8"
|
||||
gem "rspec-rails", "~> 2.8"
|
||||
gem "timecop"
|
||||
gem "rspec_tag_matchers"
|
||||
gem "ruby-debug", :platforms=>[:ruby_18, :jruby]
|
||||
gem "debugger", :platforms=>[:ruby_19]
|
||||
gem "appraisal"
|
||||
gem "sqlite3"
|
||||
gem "mongoid", "~> 2.1.0"
|
||||
|
||||
gemspec :path=>"../"
|
||||
16
gemfiles/mongoid_2_2.gemfile
Normal file
16
gemfiles/mongoid_2_2.gemfile
Normal file
@@ -0,0 +1,16 @@
|
||||
# This file was generated by Appraisal
|
||||
|
||||
source "http://rubygems.org"
|
||||
|
||||
gem "rails", "~> 3.2.6"
|
||||
gem "rspec", "~> 2.8"
|
||||
gem "rspec-rails", "~> 2.8"
|
||||
gem "timecop"
|
||||
gem "rspec_tag_matchers"
|
||||
gem "ruby-debug", :platforms=>[:ruby_18, :jruby]
|
||||
gem "debugger", :platforms=>[:ruby_19]
|
||||
gem "appraisal"
|
||||
gem "sqlite3"
|
||||
gem "mongoid", "~> 2.2.0"
|
||||
|
||||
gemspec :path=>"../"
|
||||
16
gemfiles/mongoid_2_3.gemfile
Normal file
16
gemfiles/mongoid_2_3.gemfile
Normal file
@@ -0,0 +1,16 @@
|
||||
# This file was generated by Appraisal
|
||||
|
||||
source "http://rubygems.org"
|
||||
|
||||
gem "rails", "~> 3.2.6"
|
||||
gem "rspec", "~> 2.8"
|
||||
gem "rspec-rails", "~> 2.8"
|
||||
gem "timecop"
|
||||
gem "rspec_tag_matchers"
|
||||
gem "ruby-debug", :platforms=>[:ruby_18, :jruby]
|
||||
gem "debugger", :platforms=>[:ruby_19]
|
||||
gem "appraisal"
|
||||
gem "sqlite3"
|
||||
gem "mongoid", "~> 2.3.0"
|
||||
|
||||
gemspec :path=>"../"
|
||||
16
gemfiles/mongoid_2_4.gemfile
Normal file
16
gemfiles/mongoid_2_4.gemfile
Normal file
@@ -0,0 +1,16 @@
|
||||
# This file was generated by Appraisal
|
||||
|
||||
source "http://rubygems.org"
|
||||
|
||||
gem "rails", "~> 3.2.6"
|
||||
gem "rspec", "~> 2.8"
|
||||
gem "rspec-rails", "~> 2.8"
|
||||
gem "timecop"
|
||||
gem "rspec_tag_matchers"
|
||||
gem "ruby-debug", :platforms=>[:ruby_18, :jruby]
|
||||
gem "debugger", :platforms=>[:ruby_19]
|
||||
gem "appraisal"
|
||||
gem "sqlite3"
|
||||
gem "mongoid", "~> 2.4.0"
|
||||
|
||||
gemspec :path=>"../"
|
||||
15
gemfiles/rails_3_0.gemfile
Normal file
15
gemfiles/rails_3_0.gemfile
Normal file
@@ -0,0 +1,15 @@
|
||||
# This file was generated by Appraisal
|
||||
|
||||
source "http://rubygems.org"
|
||||
|
||||
gem "rspec", "~> 2.8"
|
||||
gem "rspec-rails", "~> 2.8"
|
||||
gem "timecop"
|
||||
gem "rspec_tag_matchers"
|
||||
gem "ruby-debug", :platforms=>[:ruby_18, :jruby]
|
||||
gem "debugger", :platforms=>[:ruby_19]
|
||||
gem "appraisal"
|
||||
gem "sqlite3"
|
||||
gem "rails", "~> 3.0.0"
|
||||
|
||||
gemspec :path=>"../"
|
||||
15
gemfiles/rails_3_1.gemfile
Normal file
15
gemfiles/rails_3_1.gemfile
Normal file
@@ -0,0 +1,15 @@
|
||||
# This file was generated by Appraisal
|
||||
|
||||
source "http://rubygems.org"
|
||||
|
||||
gem "rspec", "~> 2.8"
|
||||
gem "rspec-rails", "~> 2.8"
|
||||
gem "timecop"
|
||||
gem "rspec_tag_matchers"
|
||||
gem "ruby-debug", :platforms=>[:ruby_18, :jruby]
|
||||
gem "debugger", :platforms=>[:ruby_19]
|
||||
gem "appraisal"
|
||||
gem "sqlite3"
|
||||
gem "rails", "~> 3.1.0"
|
||||
|
||||
gemspec :path=>"../"
|
||||
15
gemfiles/rails_3_2.gemfile
Normal file
15
gemfiles/rails_3_2.gemfile
Normal file
@@ -0,0 +1,15 @@
|
||||
# This file was generated by Appraisal
|
||||
|
||||
source "http://rubygems.org"
|
||||
|
||||
gem "rspec", "~> 2.8"
|
||||
gem "rspec-rails", "~> 2.8"
|
||||
gem "timecop"
|
||||
gem "rspec_tag_matchers"
|
||||
gem "ruby-debug", :platforms=>[:ruby_18, :jruby]
|
||||
gem "debugger", :platforms=>[:ruby_19]
|
||||
gem "appraisal"
|
||||
gem "sqlite3"
|
||||
gem "rails", "~> 3.2.0"
|
||||
|
||||
gemspec :path=>"../"
|
||||
@@ -3,7 +3,6 @@ module ValidatesTimeliness
|
||||
class InstallGenerator < Rails::Generators::Base
|
||||
desc "Copy ValidatesTimeliness default files"
|
||||
source_root File.expand_path('../templates', __FILE__)
|
||||
class_option :template_engine
|
||||
|
||||
def copy_initializers
|
||||
copy_file 'validates_timeliness.rb', 'config/initializers/validates_timeliness.rb'
|
||||
|
||||
@@ -32,7 +32,7 @@ ValidatesTimeliness.setup do |config|
|
||||
# Remove one or more formats making them invalid. e.g. remove_formats(:date, 'dd/mm/yyy')
|
||||
# config.parser.remove_formats()
|
||||
#
|
||||
# Change the amiguous year threshold when parsing a 2 digit year
|
||||
# Change the ambiguous year threshold when parsing a 2 digit year
|
||||
# config.parser.ambiguous_year_threshold = 30
|
||||
#
|
||||
# Treat ambiguous dates, such as 01/02/1950, as a Non-US date.
|
||||
|
||||
@@ -24,26 +24,24 @@ module ValidatesTimeliness
|
||||
|
||||
class << self
|
||||
delegate :default_timezone, :default_timezone=, :dummy_date_for_time_type, :dummy_date_for_time_type=, :to => Timeliness
|
||||
|
||||
attr_accessor :extend_orms, :ignore_restriction_errors, :restriction_shorthand_symbols, :use_plugin_parser
|
||||
end
|
||||
|
||||
# Extend ORM/ODMs for full support (:active_record, :mongoid).
|
||||
mattr_accessor :extend_orms
|
||||
@@extend_orms = []
|
||||
self.extend_orms = []
|
||||
|
||||
# Ignore errors when restriction options are evaluated
|
||||
mattr_accessor :ignore_restriction_errors
|
||||
@@ignore_restriction_errors = false
|
||||
self.ignore_restriction_errors = false
|
||||
|
||||
# Shorthand time and date symbols for restrictions
|
||||
mattr_accessor :restriction_shorthand_symbols
|
||||
@@restriction_shorthand_symbols = {
|
||||
self.restriction_shorthand_symbols = {
|
||||
:now => lambda { Time.current },
|
||||
:today => lambda { Date.current }
|
||||
}
|
||||
|
||||
# Use the plugin date/time parser which is stricter and extensible
|
||||
mattr_accessor :use_plugin_parser
|
||||
@@use_plugin_parser = false
|
||||
self.use_plugin_parser = false
|
||||
|
||||
# Default timezone
|
||||
self.default_timezone = :utc
|
||||
@@ -51,15 +49,17 @@ module ValidatesTimeliness
|
||||
# Set the dummy date part for a time type values.
|
||||
self.dummy_date_for_time_type = [ 2000, 1, 1 ]
|
||||
|
||||
def self.parser
|
||||
Timeliness
|
||||
end
|
||||
|
||||
# Setup method for plugin configuration
|
||||
def self.setup
|
||||
yield self
|
||||
load_orms
|
||||
end
|
||||
|
||||
def self.load_orms
|
||||
extend_orms.each {|orm| require "validates_timeliness/orm/#{orm}" }
|
||||
end
|
||||
|
||||
def self.parser; Timeliness end
|
||||
end
|
||||
|
||||
require 'validates_timeliness/conversion'
|
||||
|
||||
@@ -20,24 +20,34 @@ module ValidatesTimeliness
|
||||
:datetime
|
||||
end
|
||||
|
||||
def undefine_attribute_methods
|
||||
super
|
||||
undefine_timeliness_attribute_methods
|
||||
end
|
||||
|
||||
protected
|
||||
|
||||
def define_timeliness_methods(before_type_cast=false)
|
||||
return if timeliness_validated_attributes.blank?
|
||||
timeliness_validated_attributes.each do |attr_name|
|
||||
define_attribute_timeliness_methods(attr_name, before_type_cast)
|
||||
end
|
||||
end
|
||||
|
||||
def define_attribute_timeliness_methods(attr_name, before_type_cast=false)
|
||||
define_timeliness_write_method(attr_name)
|
||||
define_timeliness_before_type_cast_method(attr_name) if before_type_cast
|
||||
end
|
||||
end
|
||||
|
||||
def define_timeliness_write_method(attr_name)
|
||||
method_body, line = <<-EOV, __LINE__ + 1
|
||||
def #{attr_name}=(value)
|
||||
original_value = value
|
||||
@timeliness_cache ||= {}
|
||||
@timeliness_cache["#{attr_name}"] = value
|
||||
|
||||
@timeliness_cache["#{attr_name}"] = original_value
|
||||
#{ "if value.is_a?(String)\n#{timeliness_type_cast_code(attr_name, 'value')}\nend" if ValidatesTimeliness.use_plugin_parser }
|
||||
super
|
||||
|
||||
super(value)
|
||||
end
|
||||
EOV
|
||||
generated_timeliness_methods.module_eval(method_body, __FILE__, line)
|
||||
@@ -46,7 +56,7 @@ module ValidatesTimeliness
|
||||
def define_timeliness_before_type_cast_method(attr_name)
|
||||
method_body, line = <<-EOV, __LINE__ + 1
|
||||
def #{attr_name}_before_type_cast
|
||||
_timeliness_raw_value_for('#{attr_name}')
|
||||
_timeliness_raw_value_for('#{attr_name}') || @attributes['#{attr_name}']
|
||||
end
|
||||
EOV
|
||||
generated_timeliness_methods.module_eval(method_body, __FILE__, line)
|
||||
@@ -63,17 +73,20 @@ module ValidatesTimeliness
|
||||
def generated_timeliness_methods
|
||||
@generated_timeliness_methods ||= Module.new.tap { |m| include(m) }
|
||||
end
|
||||
|
||||
def undefine_timeliness_attribute_methods
|
||||
generated_timeliness_methods.module_eval do
|
||||
instance_methods.each { |m| undef_method(m) }
|
||||
end
|
||||
end
|
||||
end
|
||||
|
||||
module InstanceMethods
|
||||
def _timeliness_raw_value_for(attr_name)
|
||||
@timeliness_cache && @timeliness_cache[attr_name.to_s]
|
||||
@timeliness_cache && @timeliness_cache[attr_name]
|
||||
end
|
||||
|
||||
def _clear_timeliness_cache
|
||||
@timeliness_cache = {}
|
||||
end
|
||||
end
|
||||
|
||||
end
|
||||
end
|
||||
|
||||
@@ -10,35 +10,52 @@ module ValidatesTimeliness
|
||||
|
||||
included do
|
||||
alias_method_chain :datetime_selector, :timeliness
|
||||
alias_method_chain :value, :timeliness
|
||||
end
|
||||
|
||||
module InstanceMethods
|
||||
class TimelinessDateTime
|
||||
attr_accessor :year, :month, :day, :hour, :min, :sec
|
||||
|
||||
TimelinessDateTime = Struct.new(:year, :month, :day, :hour, :min, :sec)
|
||||
def initialize(year, month, day, hour, min, sec)
|
||||
@year, @month, @day, @hour, @min, @sec = year, month, day, hour, min, sec
|
||||
end
|
||||
|
||||
# adapted from activesupport/lib/active_support/core_ext/date_time/calculations.rb, line 36 (3.0.7)
|
||||
def change(options)
|
||||
TimelinessDateTime.new(
|
||||
options[:year] || year,
|
||||
options[:month] || month,
|
||||
options[:day] || day,
|
||||
options[:hour] || hour,
|
||||
options[:min] || (options[:hour] ? 0 : min),
|
||||
options[:sec] || ((options[:hour] || options[:min]) ? 0 : sec)
|
||||
)
|
||||
end
|
||||
end
|
||||
|
||||
def datetime_selector_with_timeliness(*args)
|
||||
@timeliness_date_or_time_tag = true
|
||||
datetime_selector_without_timeliness(*args)
|
||||
end
|
||||
|
||||
def value(object)
|
||||
def value_with_timeliness(object)
|
||||
unless @timeliness_date_or_time_tag && @template_object.params[@object_name]
|
||||
return super
|
||||
return value_without_timeliness(object)
|
||||
end
|
||||
|
||||
@template_object.params[@object_name]
|
||||
|
||||
pairs = @template_object.params[@object_name].select {|k,v| k =~ /^#{@method_name}\(/ }
|
||||
return super if pairs.empty?
|
||||
return value_without_timeliness(object) if pairs.empty?
|
||||
|
||||
values = [nil] * 6
|
||||
pairs.map do |(param, value)|
|
||||
position = param.scan(/\(([0-9]*).*\)/).first.first
|
||||
values[position.to_i-1] = value
|
||||
position = param.scan(/\((\d+)\w+\)/).first.first
|
||||
values[position.to_i-1] = value.to_i
|
||||
end
|
||||
|
||||
TimelinessDateTime.new(*values)
|
||||
end
|
||||
end
|
||||
|
||||
end
|
||||
end
|
||||
end
|
||||
|
||||
@@ -8,7 +8,8 @@ module ValidatesTimeliness
|
||||
|
||||
included do
|
||||
alias_method_chain :instantiate_time_object, :timeliness
|
||||
alias_method_chain :execute_callstack_for_multiparameter_attributes, :timeliness
|
||||
alias_method :execute_callstack_for_multiparameter_attributes, :execute_callstack_for_multiparameter_attributes_with_timeliness
|
||||
alias_method :read_value_from_parameter, :read_value_from_parameter_with_timeliness
|
||||
end
|
||||
|
||||
private
|
||||
@@ -20,42 +21,52 @@ module ValidatesTimeliness
|
||||
end
|
||||
|
||||
def instantiate_time_object_with_timeliness(name, values)
|
||||
if Date.valid_civil?(*values[0..2])
|
||||
validate_multiparameter_date_values(values) {
|
||||
instantiate_time_object_without_timeliness(name, values)
|
||||
}
|
||||
end
|
||||
|
||||
def instantiate_date_object(name, values)
|
||||
validate_multiparameter_date_values(values) {
|
||||
Date.new(*values)
|
||||
}
|
||||
end
|
||||
|
||||
# Yield if date values are valid
|
||||
def validate_multiparameter_date_values(values)
|
||||
if values[0..2].all?{ |v| v.present? } && Date.valid_civil?(*values[0..2])
|
||||
yield
|
||||
else
|
||||
invalid_multiparameter_date_or_time_as_string(values)
|
||||
end
|
||||
end
|
||||
|
||||
def instantiate_date_object(name, values)
|
||||
values = values.map { |v| v.nil? ? 1 : v }
|
||||
Date.new(*values)
|
||||
rescue ArgumentError => ex
|
||||
invalid_multiparameter_date_or_time_as_string(values)
|
||||
def read_value_from_parameter_with_timeliness(name, values_from_param)
|
||||
klass = (self.class.reflect_on_aggregation(name.to_sym) || column_for_attribute(name)).klass
|
||||
values = values_from_param.is_a?(Hash) ? values_from_param.to_a.sort_by(&:first).map(&:last) : values_from_param
|
||||
|
||||
if values.empty? || values.all?{ |v| v.nil? }
|
||||
nil
|
||||
elsif klass == Time
|
||||
instantiate_time_object(name, values)
|
||||
elsif klass == Date
|
||||
instantiate_date_object(name, values)
|
||||
else
|
||||
if respond_to?(:read_other_parameter_value)
|
||||
read_date_parameter_value(name, values_from_param)
|
||||
else
|
||||
klass.new(*values)
|
||||
end
|
||||
end
|
||||
end
|
||||
|
||||
def execute_callstack_for_multiparameter_attributes_with_timeliness(callstack)
|
||||
errors = []
|
||||
callstack.each do |name, values_with_empty_parameters|
|
||||
begin
|
||||
klass = (self.class.reflect_on_aggregation(name.to_sym) || column_for_attribute(name)).klass
|
||||
values = values_with_empty_parameters.reject { |v| v.nil? }
|
||||
|
||||
if values.empty?
|
||||
send(name + "=", nil)
|
||||
else
|
||||
|
||||
value = if Time == klass
|
||||
instantiate_time_object(name, values)
|
||||
elsif Date == klass
|
||||
instantiate_date_object(name, values_with_empty_parameters)
|
||||
else
|
||||
klass.new(*values)
|
||||
end
|
||||
|
||||
send(name + "=", value)
|
||||
end
|
||||
send(name + "=", read_value_from_parameter(name, values_with_empty_parameters))
|
||||
rescue => ex
|
||||
values = values_with_empty_parameters.is_a?(Hash) ? values_with_empty_parameters.values : values_with_empty_parameters
|
||||
errors << ActiveRecord::AttributeAssignmentError.new("error on assignment #{values.inspect} to #{name}", ex, name)
|
||||
end
|
||||
end
|
||||
|
||||
@@ -15,12 +15,7 @@ module ActiveModel
|
||||
end
|
||||
|
||||
def timeliness_validation_for(attr_names, type)
|
||||
options = _merge_attributes(attr_names).merge(:type => type)
|
||||
if respond_to?(:timeliness_validated_attributes)
|
||||
self.timeliness_validated_attributes ||= []
|
||||
self.timeliness_validated_attributes += (attr_names - self.timeliness_validated_attributes)
|
||||
end
|
||||
validates_with TimelinessValidator, options
|
||||
validates_with TimelinessValidator, _merge_attributes(attr_names).merge(:type => type)
|
||||
end
|
||||
end
|
||||
|
||||
|
||||
@@ -4,37 +4,44 @@ module ValidatesTimeliness
|
||||
extend ActiveSupport::Concern
|
||||
|
||||
module ClassMethods
|
||||
def define_attribute_methods
|
||||
super
|
||||
# Define write method and before_type_cast method
|
||||
define_timeliness_methods(true)
|
||||
end
|
||||
public
|
||||
|
||||
def timeliness_attribute_timezone_aware?(attr_name)
|
||||
attr_name = attr_name.to_s
|
||||
create_time_zone_conversion_attribute?(attr_name, columns_hash[attr_name])
|
||||
create_time_zone_conversion_attribute?(attr_name, timeliness_column_for_attribute(attr_name))
|
||||
end
|
||||
|
||||
def timeliness_attribute_type(attr_name)
|
||||
columns_hash[attr_name.to_s].type
|
||||
timeliness_column_for_attribute(attr_name).type
|
||||
end
|
||||
|
||||
def timeliness_column_for_attribute(attr_name)
|
||||
columns_hash.fetch(attr_name.to_s) do |attr_name|
|
||||
validation_type = _validators[attr_name.to_sym].find {|v| v.kind == :timeliness }.type
|
||||
::ActiveRecord::ConnectionAdapters::Column.new(attr_name, nil, validation_type.to_s)
|
||||
end
|
||||
end
|
||||
|
||||
def define_attribute_methods
|
||||
super.tap do |attribute_methods_generated|
|
||||
define_timeliness_methods true
|
||||
end
|
||||
end
|
||||
|
||||
protected
|
||||
|
||||
def timeliness_type_cast_code(attr_name, var_name)
|
||||
type = timeliness_attribute_type(attr_name)
|
||||
|
||||
<<-END
|
||||
#{super}
|
||||
#{var_name} = #{var_name}.to_date if #{var_name} && :#{type} == :date
|
||||
END
|
||||
method_body = super
|
||||
method_body << "\n#{var_name} = #{var_name}.to_date if #{var_name}" if type == :date
|
||||
method_body
|
||||
end
|
||||
end
|
||||
|
||||
module InstanceMethods
|
||||
def reload(*args)
|
||||
_clear_timeliness_cache
|
||||
super
|
||||
end
|
||||
end
|
||||
|
||||
end
|
||||
end
|
||||
|
||||
@@ -7,6 +7,8 @@ module ValidatesTimeliness
|
||||
# field value in Mongoid. Parser will return nil rather than error.
|
||||
|
||||
module ClassMethods
|
||||
public
|
||||
|
||||
# Mongoid has no bulk attribute method definition hook. It defines
|
||||
# them with each field definition. So we likewise define them after
|
||||
# each validation is defined.
|
||||
@@ -16,42 +18,32 @@ module ValidatesTimeliness
|
||||
attr_names.each { |attr_name| define_timeliness_write_method(attr_name) }
|
||||
end
|
||||
|
||||
def timeliness_attribute_type(attr_name)
|
||||
{
|
||||
Date => :date,
|
||||
Time => :time,
|
||||
DateTime => :datetime
|
||||
}[fields[database_field_name(attr_name)].type] || :datetime
|
||||
end
|
||||
|
||||
protected
|
||||
|
||||
def timeliness_type_cast_code(attr_name, var_name)
|
||||
type = timeliness_attribute_type(attr_name)
|
||||
|
||||
"#{var_name} = Timeliness::Parser.parse(value, :#{type})"
|
||||
end
|
||||
|
||||
def timeliness_attribute_type(attr_name)
|
||||
{
|
||||
Date => :date,
|
||||
Time => :datetime,
|
||||
DateTime => :datetime
|
||||
}[fields[attr_name.to_s].type] || :datetime
|
||||
end
|
||||
end
|
||||
|
||||
def reload(*args)
|
||||
_clear_timeliness_cache
|
||||
super
|
||||
end
|
||||
end
|
||||
end
|
||||
end
|
||||
|
||||
module Mongoid::Document
|
||||
# Due to how Mongoid misuses ActiveSupport::Concern,
|
||||
# the only way to override a core component method is
|
||||
# using an append_features hook.
|
||||
#
|
||||
module TimelinessConcern
|
||||
def append_features(base)
|
||||
super
|
||||
base.send :include, ValidatesTimeliness::AttributeMethods
|
||||
base.send :include, ValidatesTimeliness::ORM::Mongoid
|
||||
end
|
||||
end
|
||||
extend TimelinessConcern
|
||||
|
||||
def reload_with_timeliness
|
||||
_clear_timeliness_cache
|
||||
reload_without_timeliness
|
||||
end
|
||||
alias_method_chain :reload, :timeliness
|
||||
include ValidatesTimeliness::AttributeMethods
|
||||
include ValidatesTimeliness::ORM::Mongoid
|
||||
end
|
||||
|
||||
@@ -3,7 +3,8 @@ module ValidatesTimeliness
|
||||
initializer "validates_timeliness.initialize_active_record", :after => 'active_record.initialize_timezone' do
|
||||
ActiveSupport.on_load(:active_record) do
|
||||
ValidatesTimeliness.default_timezone = ActiveRecord::Base.default_timezone
|
||||
ValidatesTimeliness.extend_orms = [ :active_record ]
|
||||
ValidatesTimeliness.extend_orms << :active_record
|
||||
ValidatesTimeliness.load_orms
|
||||
end
|
||||
end
|
||||
|
||||
|
||||
@@ -1,3 +1,4 @@
|
||||
require 'active_model'
|
||||
require 'active_model/validator'
|
||||
|
||||
module ValidatesTimeliness
|
||||
@@ -22,19 +23,35 @@ module ValidatesTimeliness
|
||||
|
||||
RESTRICTION_ERROR_MESSAGE = "Error occurred validating %s for %s restriction:\n%s"
|
||||
|
||||
def self.kind
|
||||
:timeliness
|
||||
end
|
||||
|
||||
def initialize(options)
|
||||
@type = options.delete(:type) || :datetime
|
||||
@allow_nil, @allow_blank = options.delete(:allow_nil), options.delete(:allow_blank)
|
||||
|
||||
if range = options.delete(:between)
|
||||
raise ArgumentError, ":between must be a Range or an Array" unless range.is_a?(Range) || range.is_a?(Array)
|
||||
options[:on_or_after], options[:on_or_before] = range.first, range.last
|
||||
options[:on_or_after] = range.first
|
||||
if range.is_a?(Range) && range.exclude_end?
|
||||
options[:before] = range.last
|
||||
else
|
||||
options[:on_or_before] = range.last
|
||||
end
|
||||
end
|
||||
|
||||
@restrictions_to_check = RESTRICTIONS.keys & options.keys
|
||||
super
|
||||
end
|
||||
|
||||
def setup(model)
|
||||
if model.respond_to?(:timeliness_validated_attributes)
|
||||
model.timeliness_validated_attributes ||= []
|
||||
model.timeliness_validated_attributes |= @attributes
|
||||
end
|
||||
end
|
||||
|
||||
def validate_each(record, attr_name, value)
|
||||
raw_value = attribute_raw_value(record, attr_name) || value
|
||||
return if (@allow_nil && raw_value.nil?) || (@allow_blank && raw_value.blank?)
|
||||
@@ -77,7 +94,7 @@ module ValidatesTimeliness
|
||||
|
||||
def attribute_raw_value(record, attr_name)
|
||||
record.respond_to?(:_timeliness_raw_value_for) &&
|
||||
record._timeliness_raw_value_for(attr_name)
|
||||
record._timeliness_raw_value_for(attr_name.to_s)
|
||||
end
|
||||
|
||||
def timezone_aware?(record, attr_name)
|
||||
|
||||
@@ -1,3 +1,3 @@
|
||||
module ValidatesTimeliness
|
||||
VERSION = '3.0.6'
|
||||
VERSION = '3.0.15'
|
||||
end
|
||||
|
||||
@@ -49,9 +49,7 @@ class Person
|
||||
attribute :birth_date, :date
|
||||
attribute :birth_time, :time
|
||||
attribute :birth_datetime, :datetime
|
||||
validates_date :birth_date
|
||||
validates_time :birth_time
|
||||
validates_datetime :birth_datetime
|
||||
|
||||
define_attribute_methods model_attributes.keys
|
||||
end
|
||||
|
||||
@@ -59,6 +57,7 @@ class PersonWithShim < Person
|
||||
include TestModelShim
|
||||
end
|
||||
|
||||
ActiveRecord::Base.default_timezone = :utc
|
||||
ActiveRecord::Base.time_zone_aware_attributes = true
|
||||
ActiveRecord::Base.establish_connection({:adapter => 'sqlite3', :database => ':memory:'})
|
||||
ActiveRecord::Migration.verbose = false
|
||||
@@ -73,12 +72,10 @@ ActiveRecord::Schema.define(:version => 1) do
|
||||
end
|
||||
|
||||
class Employee < ActiveRecord::Base
|
||||
validates_date :birth_date
|
||||
validates_time :birth_time
|
||||
validates_datetime :birth_datetime
|
||||
define_attribute_methods
|
||||
|
||||
attr_accessor :redefined_birth_date_called
|
||||
validates_date :birth_date, :allow_nil => true
|
||||
validates_date :birth_time, :allow_nil => true
|
||||
validates_date :birth_datetime, :allow_nil => true
|
||||
|
||||
def birth_date=(value)
|
||||
self.redefined_birth_date_called = true
|
||||
@@ -86,17 +83,17 @@ class Employee < ActiveRecord::Base
|
||||
end
|
||||
end
|
||||
|
||||
Rspec.configure do |c|
|
||||
RSpec.configure do |c|
|
||||
c.mock_with :rspec
|
||||
c.include(RspecTagMatchers)
|
||||
c.include(ModelHelpers)
|
||||
c.include(ConfigHelper)
|
||||
c.before do
|
||||
Person.reset_callbacks(:validate)
|
||||
PersonWithShim.timeliness_validated_attributes = []
|
||||
Person._validators.clear
|
||||
Employee.reset_callbacks(:validate)
|
||||
Employee.timeliness_validated_attributes = []
|
||||
Employee._validators.clear
|
||||
reset_validation_setup_for(Person)
|
||||
reset_validation_setup_for(PersonWithShim)
|
||||
end
|
||||
|
||||
c.filter_run_excluding :active_record => lambda {|version|
|
||||
!(::ActiveRecord::VERSION::STRING.to_s =~ /^#{version.to_s}/)
|
||||
}
|
||||
end
|
||||
|
||||
@@ -10,6 +10,16 @@ module ConfigHelper
|
||||
ValidatesTimeliness.send(:"#{preference_name}=", old_value)
|
||||
end
|
||||
|
||||
def reset_validation_setup_for(model_class)
|
||||
model_class.reset_callbacks(:validate)
|
||||
model_class._validators.clear
|
||||
model_class.timeliness_validated_attributes = [] if model_class.respond_to?(:timeliness_validated_attributes)
|
||||
model_class.undefine_attribute_methods
|
||||
# This is a hack to avoid a disabled super method error message after an undef
|
||||
model_class.instance_variable_set(:@generated_attribute_methods, nil)
|
||||
model_class.instance_variable_set(:@generated_timeliness_methods, nil)
|
||||
end
|
||||
|
||||
module ClassMethods
|
||||
def with_config(preference_name, temporary_value)
|
||||
original_config_value = ValidatesTimeliness.send(preference_name)
|
||||
|
||||
@@ -31,7 +31,7 @@ module TestModel
|
||||
end
|
||||
|
||||
def initialize(attributes = nil)
|
||||
@attributes = self.class.model_attributes.inject({}) do |hash, column|
|
||||
@attributes = self.class.model_attributes.keys.inject({}) do |hash, column|
|
||||
hash[column.to_s] = nil
|
||||
hash
|
||||
end
|
||||
@@ -39,7 +39,7 @@ module TestModel
|
||||
end
|
||||
|
||||
def attributes
|
||||
@attributes.keys
|
||||
@attributes
|
||||
end
|
||||
|
||||
def attributes=(new_attributes={})
|
||||
@@ -49,14 +49,12 @@ module TestModel
|
||||
end
|
||||
|
||||
def method_missing(method_id, *args, &block)
|
||||
if !self.class.attribute_methods_generated?
|
||||
self.class.define_attribute_methods self.class.model_attributes.keys.map(&:to_s)
|
||||
method_name = method_id.to_s
|
||||
if match_attribute_method?(method_id.to_s)
|
||||
self.class.define_attribute_methods self.class.model_attributes.keys
|
||||
send(method_id, *args, &block)
|
||||
else
|
||||
super
|
||||
end
|
||||
end
|
||||
|
||||
end
|
||||
|
||||
|
||||
@@ -31,7 +31,7 @@ describe ValidatesTimeliness::AttributeMethods do
|
||||
it 'should cache attribute raw value' do
|
||||
r = PersonWithCache.new
|
||||
r.birth_datetime = date_string = '2010-01-01'
|
||||
r._timeliness_raw_value_for(:birth_datetime).should == date_string
|
||||
r._timeliness_raw_value_for('birth_datetime').should == date_string
|
||||
end
|
||||
|
||||
it 'should not overwrite user defined methods' do
|
||||
@@ -40,6 +40,21 @@ describe ValidatesTimeliness::AttributeMethods do
|
||||
e.redefined_birth_date_called.should be_true
|
||||
end
|
||||
|
||||
it 'should be undefined if model class has dynamic attribute methods reset' do
|
||||
# Force method definitions
|
||||
PersonWithShim.validates_date :birth_date
|
||||
r = PersonWithShim.new
|
||||
r.birth_date = Time.now
|
||||
|
||||
write_method = RUBY_VERSION < '1.9' ? 'birth_date=' : :birth_date=
|
||||
|
||||
PersonWithShim.send(:generated_timeliness_methods).instance_methods.should include(write_method)
|
||||
|
||||
PersonWithShim.undefine_attribute_methods
|
||||
|
||||
PersonWithShim.send(:generated_timeliness_methods).instance_methods.should_not include(write_method)
|
||||
end
|
||||
|
||||
context "with plugin parser" do
|
||||
with_config(:use_plugin_parser, true)
|
||||
|
||||
|
||||
@@ -14,12 +14,12 @@ describe ValidatesTimeliness::Extensions::DateTimeSelect do
|
||||
describe "datetime_select" do
|
||||
it "should use param values when attribute is nil" do
|
||||
@params["person"] = {
|
||||
"birth_datetime(1i)" => 2009,
|
||||
"birth_datetime(2i)" => 2,
|
||||
"birth_datetime(3i)" => 29,
|
||||
"birth_datetime(4i)" => 12,
|
||||
"birth_datetime(5i)" => 13,
|
||||
"birth_datetime(6i)" => 14,
|
||||
"birth_datetime(1i)" => '2009',
|
||||
"birth_datetime(2i)" => '2',
|
||||
"birth_datetime(3i)" => '29',
|
||||
"birth_datetime(4i)" => '12',
|
||||
"birth_datetime(5i)" => '13',
|
||||
"birth_datetime(6i)" => '14',
|
||||
}
|
||||
person.birth_datetime = nil
|
||||
@output = datetime_select(:person, :birth_datetime, :include_blank => true, :include_seconds => true)
|
||||
@@ -28,12 +28,12 @@ describe ValidatesTimeliness::Extensions::DateTimeSelect do
|
||||
|
||||
it "should override object values and use params if present" do
|
||||
@params["person"] = {
|
||||
"birth_datetime(1i)" => 2009,
|
||||
"birth_datetime(2i)" => 2,
|
||||
"birth_datetime(3i)" => 29,
|
||||
"birth_datetime(4i)" => 12,
|
||||
"birth_datetime(5i)" => 13,
|
||||
"birth_datetime(6i)" => 14,
|
||||
"birth_datetime(1i)" => '2009',
|
||||
"birth_datetime(2i)" => '2',
|
||||
"birth_datetime(3i)" => '29',
|
||||
"birth_datetime(4i)" => '12',
|
||||
"birth_datetime(5i)" => '13',
|
||||
"birth_datetime(6i)" => '14',
|
||||
}
|
||||
person.birth_datetime = "2010-01-01 15:16:17"
|
||||
@output = datetime_select(:person, :birth_datetime, :include_blank => true, :include_seconds => true)
|
||||
@@ -63,44 +63,56 @@ describe ValidatesTimeliness::Extensions::DateTimeSelect do
|
||||
describe "date_select" do
|
||||
it "should use param values when attribute is nil" do
|
||||
@params["person"] = {
|
||||
"birth_date(1i)" => 2009,
|
||||
"birth_date(2i)" => 2,
|
||||
"birth_date(3i)" => 29,
|
||||
"birth_date(1i)" => '2009',
|
||||
"birth_date(2i)" => '2',
|
||||
"birth_date(3i)" => '29',
|
||||
}
|
||||
person.birth_date = nil
|
||||
@output = date_select(:person, :birth_date, :include_blank => true, :include_seconds => true)
|
||||
@output = date_select(:person, :birth_date, :include_blank => true)
|
||||
should_have_datetime_selected(:birth_date, :year => 2009, :month => 'February', :day => 29)
|
||||
end
|
||||
|
||||
it "should override object values and use params if present" do
|
||||
@params["person"] = {
|
||||
"birth_date(1i)" => 2009,
|
||||
"birth_date(2i)" => 2,
|
||||
"birth_date(3i)" => 29,
|
||||
"birth_date(1i)" => '2009',
|
||||
"birth_date(2i)" => '2',
|
||||
"birth_date(3i)" => '29',
|
||||
}
|
||||
person.birth_date = "2009-03-01"
|
||||
@output = date_select(:person, :birth_date, :include_blank => true, :include_seconds => true)
|
||||
@output = date_select(:person, :birth_date, :include_blank => true)
|
||||
should_have_datetime_selected(:birth_date, :year => 2009, :month => 'February', :day => 29)
|
||||
end
|
||||
|
||||
it "should select attribute values from object if no params" do
|
||||
person.birth_date = "2009-01-02"
|
||||
@output = date_select(:person, :birth_date, :include_blank => true, :include_seconds => true)
|
||||
@output = date_select(:person, :birth_date, :include_blank => true)
|
||||
should_have_datetime_selected(:birth_date, :year => 2009, :month => 'January', :day => 2)
|
||||
end
|
||||
|
||||
it "should select attribute values if params does not contain attribute params" do
|
||||
person.birth_date = "2009-01-02"
|
||||
@params["person"] = { }
|
||||
@output = date_select(:person, :birth_date, :include_blank => true, :include_seconds => true)
|
||||
@output = date_select(:person, :birth_date, :include_blank => true)
|
||||
should_have_datetime_selected(:birth_date, :year => 2009, :month => 'January', :day => 2)
|
||||
end
|
||||
|
||||
it "should not select values when attribute value is nil and has no param values" do
|
||||
person.birth_date = nil
|
||||
@output = date_select(:person, :birth_date, :include_blank => true, :include_seconds => true)
|
||||
@output = date_select(:person, :birth_date, :include_blank => true)
|
||||
should_not_have_datetime_selected(:birth_time, :year, :month, :day)
|
||||
end
|
||||
|
||||
it "should allow the day part to be discarded" do
|
||||
@params["person"] = {
|
||||
"birth_date(1i)" => '2009',
|
||||
"birth_date(2i)" => '2',
|
||||
}
|
||||
|
||||
@output = date_select(:person, :birth_date, :include_blank => true, :discard_day => true)
|
||||
should_have_datetime_selected(:birth_date, :year => 2009, :month => 'February')
|
||||
should_not_have_datetime_selected(:birth_time, :day)
|
||||
@output.should have_tag("input[id=person_birth_date_3i][type=hidden][value='1']")
|
||||
end
|
||||
end
|
||||
|
||||
describe "time_select" do
|
||||
@@ -110,12 +122,12 @@ describe ValidatesTimeliness::Extensions::DateTimeSelect do
|
||||
|
||||
it "should use param values when attribute is nil" do
|
||||
@params["person"] = {
|
||||
"birth_time(1i)" => 2000,
|
||||
"birth_time(2i)" => 1,
|
||||
"birth_time(3i)" => 1,
|
||||
"birth_time(4i)" => 12,
|
||||
"birth_time(5i)" => 13,
|
||||
"birth_time(6i)" => 14,
|
||||
"birth_time(1i)" => '2000',
|
||||
"birth_time(2i)" => '1',
|
||||
"birth_time(3i)" => '1',
|
||||
"birth_time(4i)" => '12',
|
||||
"birth_time(5i)" => '13',
|
||||
"birth_time(6i)" => '14',
|
||||
}
|
||||
person.birth_time = nil
|
||||
@output = time_select(:person, :birth_time, :include_blank => true, :include_seconds => true)
|
||||
|
||||
@@ -1,33 +1,44 @@
|
||||
require 'spec_helper'
|
||||
|
||||
describe ValidatesTimeliness::Extensions::MultiparameterHandler do
|
||||
let(:employee) { Employee.new }
|
||||
|
||||
context "time column" do
|
||||
it 'should return string value for invalid date portion' do
|
||||
multiparameter_attribute(:birth_datetime, [2000, 2, 31, 12, 0, 0])
|
||||
employee.birth_datetime_before_type_cast.should == '2000-02-31 12:00:00'
|
||||
it 'should assign a string value for invalid date portion' do
|
||||
employee = record_with_multiparameter_attribute(:birth_datetime, [2000, 2, 31, 12, 0, 0])
|
||||
employee.birth_datetime_before_type_cast.should eq '2000-02-31 12:00:00'
|
||||
end
|
||||
|
||||
it 'should return Time value for valid datetimes' do
|
||||
multiparameter_attribute(:birth_datetime, [2000, 2, 28, 12, 0, 0])
|
||||
employee.birth_datetime_before_type_cast.should be_kind_of(Time)
|
||||
it 'should assign a Time value for valid datetimes' do
|
||||
employee = record_with_multiparameter_attribute(:birth_datetime, [2000, 2, 28, 12, 0, 0])
|
||||
employee.birth_datetime_before_type_cast.should eq Time.zone.local(2000, 2, 28, 12, 0, 0)
|
||||
end
|
||||
|
||||
it 'should assign a string value for incomplete time' do
|
||||
employee = record_with_multiparameter_attribute(:birth_datetime, [2000, nil, nil])
|
||||
employee.birth_datetime_before_type_cast.should eq '2000-00-00'
|
||||
end
|
||||
end
|
||||
|
||||
context "date column" do
|
||||
it 'should return string value for invalid date' do
|
||||
multiparameter_attribute(:birth_date, [2000, 2, 31])
|
||||
employee.birth_date_before_type_cast.should == '2000-02-31'
|
||||
it 'should assign a string value for invalid date' do
|
||||
employee = record_with_multiparameter_attribute(:birth_date, [2000, 2, 31])
|
||||
employee.birth_date_before_type_cast.should eq '2000-02-31'
|
||||
end
|
||||
|
||||
it 'should return Date value for valid date' do
|
||||
multiparameter_attribute(:birth_date, [2000, 2, 28])
|
||||
employee.birth_date_before_type_cast.should be_kind_of(Date)
|
||||
it 'should assign a Date value for valid date' do
|
||||
employee = record_with_multiparameter_attribute(:birth_date, [2000, 2, 28])
|
||||
employee.birth_date_before_type_cast.should eq Date.new(2000, 2, 28)
|
||||
end
|
||||
|
||||
it 'should assign a string value for incomplete date' do
|
||||
employee = record_with_multiparameter_attribute(:birth_date, [2000, nil, nil])
|
||||
employee.birth_date_before_type_cast.should eq '2000-00-00'
|
||||
end
|
||||
end
|
||||
|
||||
def multiparameter_attribute(name, values)
|
||||
employee.send(:execute_callstack_for_multiparameter_attributes, name.to_s => values)
|
||||
def record_with_multiparameter_attribute(name, values)
|
||||
hash = {}
|
||||
values.each_with_index {|value, index| hash["#{name}(#{index+1}i)"] = value.to_s }
|
||||
Employee.new(hash)
|
||||
end
|
||||
end
|
||||
|
||||
@@ -1,6 +1,8 @@
|
||||
require 'spec_helper'
|
||||
|
||||
describe ValidatesTimeliness, 'HelperMethods' do
|
||||
let(:record) { Person.new }
|
||||
|
||||
it 'should define class validation methods' do
|
||||
Person.should respond_to(:validates_date)
|
||||
Person.should respond_to(:validates_time)
|
||||
@@ -8,14 +10,21 @@ describe ValidatesTimeliness, 'HelperMethods' do
|
||||
end
|
||||
|
||||
it 'should define instance validation methods' do
|
||||
Person.new.should respond_to(:validates_date)
|
||||
Person.new.should respond_to(:validates_time)
|
||||
Person.new.should respond_to(:validates_datetime)
|
||||
record.should respond_to(:validates_date)
|
||||
record.should respond_to(:validates_time)
|
||||
record.should respond_to(:validates_datetime)
|
||||
end
|
||||
|
||||
it 'should validate instance when validation method called' do
|
||||
r = Person.new
|
||||
r.validates_date :birth_date
|
||||
r.errors[:birth_date].should_not be_empty
|
||||
it 'should validate instance using class validation defined' do
|
||||
Person.validates_date :birth_date
|
||||
record.valid?
|
||||
|
||||
record.errors[:birth_date].should_not be_empty
|
||||
end
|
||||
|
||||
it 'should validate instance using instance valiation method' do
|
||||
record.validates_date :birth_date
|
||||
|
||||
record.errors[:birth_date].should_not be_empty
|
||||
end
|
||||
end
|
||||
|
||||
@@ -3,6 +3,8 @@ require 'spec_helper'
|
||||
describe ValidatesTimeliness, 'ActiveRecord' do
|
||||
|
||||
context "validation methods" do
|
||||
let(:record) { Employee.new }
|
||||
|
||||
it 'should be defined for the class' do
|
||||
ActiveRecord::Base.should respond_to(:validates_date)
|
||||
ActiveRecord::Base.should respond_to(:validates_time)
|
||||
@@ -10,91 +12,233 @@ describe ValidatesTimeliness, 'ActiveRecord' do
|
||||
end
|
||||
|
||||
it 'should defines for the instance' do
|
||||
Employee.new.should respond_to(:validates_date)
|
||||
Employee.new.should respond_to(:validates_time)
|
||||
Employee.new.should respond_to(:validates_datetime)
|
||||
record.should respond_to(:validates_date)
|
||||
record.should respond_to(:validates_time)
|
||||
record.should respond_to(:validates_datetime)
|
||||
end
|
||||
|
||||
it "should validate a valid value string" do
|
||||
record.birth_date = '2012-01-01'
|
||||
|
||||
record.valid?
|
||||
record.errors[:birth_date].should be_empty
|
||||
end
|
||||
|
||||
it "should validate a invalid value string" do
|
||||
record.birth_date = 'not a date'
|
||||
|
||||
record.valid?
|
||||
record.errors[:birth_date].should_not be_empty
|
||||
end
|
||||
|
||||
it "should validate a nil value" do
|
||||
record.birth_date = nil
|
||||
|
||||
record.valid?
|
||||
record.errors[:birth_date].should be_empty
|
||||
end
|
||||
end
|
||||
|
||||
it 'should determine type for attribute' do
|
||||
Employee.timeliness_attribute_type(:birth_date).should == :date
|
||||
Employee.timeliness_attribute_type(:birth_date).should eq :date
|
||||
end
|
||||
|
||||
context 'attribute timezone awareness' do
|
||||
let(:klass) {
|
||||
Class.new(ActiveRecord::Base) do
|
||||
self.table_name = 'employees'
|
||||
attr_accessor :some_date
|
||||
attr_accessor :some_time
|
||||
attr_accessor :some_datetime
|
||||
validates_date :some_date
|
||||
validates_time :some_time
|
||||
validates_datetime :some_datetime
|
||||
end
|
||||
}
|
||||
|
||||
context 'for column attribute' do
|
||||
it 'should be detected from column type' do
|
||||
klass.timeliness_attribute_timezone_aware?(:birth_date).should be_false
|
||||
klass.timeliness_attribute_timezone_aware?(:birth_time).should be_false
|
||||
klass.timeliness_attribute_timezone_aware?(:birth_datetime).should be_true
|
||||
end
|
||||
end
|
||||
|
||||
context 'for non-column attribute' do
|
||||
it 'should be detected from the validation type' do
|
||||
klass.timeliness_attribute_timezone_aware?(:some_date).should be_false
|
||||
klass.timeliness_attribute_timezone_aware?(:some_time).should be_false
|
||||
klass.timeliness_attribute_timezone_aware?(:some_datetime).should be_true
|
||||
end
|
||||
end
|
||||
end
|
||||
|
||||
context "attribute write method" do
|
||||
class EmployeeWithCache < ActiveRecord::Base
|
||||
set_table_name 'employees'
|
||||
validates_datetime :birth_datetime
|
||||
self.table_name = 'employees'
|
||||
validates_date :birth_date, :allow_blank => true
|
||||
validates_time :birth_time, :allow_blank => true
|
||||
validates_datetime :birth_datetime, :allow_blank => true
|
||||
end
|
||||
|
||||
it 'should cache attribute raw value' do
|
||||
r = EmployeeWithCache.new
|
||||
r.birth_datetime = date_string = '2010-01-01'
|
||||
r._timeliness_raw_value_for(:birth_datetime).should == date_string
|
||||
context 'value cache' do
|
||||
let(:record) { EmployeeWithCache.new }
|
||||
|
||||
context 'for datetime column' do
|
||||
it 'should store raw value' do
|
||||
record.birth_datetime = datetime_string = '2010-01-01 12:30'
|
||||
|
||||
record._timeliness_raw_value_for('birth_datetime').should eq datetime_string
|
||||
end
|
||||
end
|
||||
|
||||
context 'for date column' do
|
||||
it 'should store raw value' do
|
||||
record.birth_date = date_string = '2010-01-01'
|
||||
|
||||
record._timeliness_raw_value_for('birth_date').should eq date_string
|
||||
end
|
||||
end
|
||||
|
||||
context 'for time column' do
|
||||
it 'should store raw value' do
|
||||
record.birth_time = time_string = '12:12'
|
||||
|
||||
record._timeliness_raw_value_for('birth_time').should eq time_string
|
||||
end
|
||||
end
|
||||
end
|
||||
|
||||
context "with plugin parser" do
|
||||
with_config(:use_plugin_parser, true)
|
||||
let(:record) { EmployeeWithParser.new }
|
||||
|
||||
class EmployeeWithParser < ActiveRecord::Base
|
||||
set_table_name 'employees'
|
||||
validates_date :birth_date
|
||||
validates_datetime :birth_datetime
|
||||
end
|
||||
|
||||
it 'should parse a string value' do
|
||||
Timeliness::Parser.should_receive(:parse)
|
||||
r = EmployeeWithParser.new
|
||||
r.birth_date = '2010-01-01'
|
||||
self.table_name = 'employees'
|
||||
validates_date :birth_date, :allow_blank => true
|
||||
validates_time :birth_time, :allow_blank => true
|
||||
validates_datetime :birth_datetime, :allow_blank => true
|
||||
end
|
||||
|
||||
context "for a date column" do
|
||||
it 'should store a date value after parsing string' do
|
||||
r = EmployeeWithParser.new
|
||||
r.birth_date = '2010-01-01'
|
||||
it 'should parse a string value' do
|
||||
Timeliness::Parser.should_receive(:parse)
|
||||
|
||||
r.birth_date.should be_kind_of(Date)
|
||||
r.birth_date.should == Date.new(2010, 1, 1)
|
||||
record.birth_date = '2010-01-01'
|
||||
end
|
||||
|
||||
it 'should parse a invalid string value as nil' do
|
||||
Timeliness::Parser.should_receive(:parse)
|
||||
|
||||
record.birth_date = 'not valid'
|
||||
end
|
||||
|
||||
it 'should store a Date value after parsing string' do
|
||||
record.birth_date = '2010-01-01'
|
||||
|
||||
record.birth_date.should be_kind_of(Date)
|
||||
record.birth_date.should eq Date.new(2010, 1, 1)
|
||||
end
|
||||
end
|
||||
|
||||
context "for a time column" do
|
||||
it 'should parse a string value' do
|
||||
Timeliness::Parser.should_receive(:parse)
|
||||
|
||||
record.birth_time = '12:30'
|
||||
end
|
||||
|
||||
it 'should parse a invalid string value as nil' do
|
||||
Timeliness::Parser.should_receive(:parse)
|
||||
|
||||
record.birth_time = 'not valid'
|
||||
end
|
||||
|
||||
it 'should store a Time value after parsing string' do
|
||||
record.birth_time = '12:30'
|
||||
|
||||
record.birth_time.should be_kind_of(Time)
|
||||
record.birth_time.should eq Time.utc(2000, 1, 1, 12, 30)
|
||||
end
|
||||
end
|
||||
|
||||
context "for a datetime column" do
|
||||
with_config(:default_timezone, 'Australia/Melbourne')
|
||||
|
||||
it 'should parse string into Time value' do
|
||||
r = EmployeeWithParser.new
|
||||
r.birth_datetime = '2010-01-01 12:00'
|
||||
it 'should parse a string value' do
|
||||
Timeliness::Parser.should_receive(:parse)
|
||||
|
||||
r.birth_datetime.should be_kind_of(Time)
|
||||
record.birth_datetime = '2010-01-01 12:00'
|
||||
end
|
||||
|
||||
it 'should parse a invalid string value as nil' do
|
||||
Timeliness::Parser.should_receive(:parse)
|
||||
|
||||
record.birth_datetime = 'not valid'
|
||||
end
|
||||
|
||||
it 'should parse string into Time value' do
|
||||
record.birth_datetime = '2010-01-01 12:00'
|
||||
|
||||
record.birth_datetime.should be_kind_of(Time)
|
||||
end
|
||||
|
||||
it 'should parse string as current timezone' do
|
||||
r = EmployeeWithParser.new
|
||||
r.birth_datetime = '2010-06-01 12:00'
|
||||
record.birth_datetime = '2010-06-01 12:00'
|
||||
|
||||
r.birth_datetime.utc_offset.should == Time.zone.utc_offset
|
||||
record.birth_datetime.utc_offset.should eq Time.zone.utc_offset
|
||||
end
|
||||
end
|
||||
end
|
||||
end
|
||||
|
||||
context "cached value" do
|
||||
it 'should be cleared on reload' do
|
||||
r = Employee.create!
|
||||
r.birth_date = '2010-01-01'
|
||||
r.reload
|
||||
r._timeliness_raw_value_for(:birth_date).should be_nil
|
||||
context "reload" do
|
||||
it 'should clear cache value' do
|
||||
record = Employee.create!
|
||||
record.birth_date = '2010-01-01'
|
||||
|
||||
record.reload
|
||||
|
||||
record._timeliness_raw_value_for('birth_date').should be_nil
|
||||
end
|
||||
end
|
||||
|
||||
context "before_type_cast method" do
|
||||
let(:record) { Employee.new }
|
||||
|
||||
it 'should be defined on class if ORM supports it' do
|
||||
Employee.new.should respond_to(:birth_datetime_before_type_cast)
|
||||
record.should respond_to(:birth_datetime_before_type_cast)
|
||||
end
|
||||
|
||||
it 'should return original value' do
|
||||
r = Employee.new
|
||||
r.birth_datetime = date_string = '2010-01-01'
|
||||
r.birth_datetime_before_type_cast.should == date_string
|
||||
record.birth_datetime = date_string = '2010-01-01'
|
||||
|
||||
record.birth_datetime_before_type_cast.should eq date_string
|
||||
end
|
||||
|
||||
it 'should return attribute if no attribute assignment has been made' do
|
||||
datetime = Time.zone.local(2010,01,01)
|
||||
Employee.create(:birth_datetime => datetime)
|
||||
|
||||
record = Employee.last
|
||||
record.birth_datetime_before_type_cast.should match(/#{datetime.utc.to_s[0...-4]}/)
|
||||
end
|
||||
|
||||
context "with plugin parser" do
|
||||
with_config(:use_plugin_parser, true)
|
||||
|
||||
it 'should return original value' do
|
||||
record.birth_datetime = date_string = '2010-01-31'
|
||||
|
||||
record.birth_datetime_before_type_cast.should eq date_string
|
||||
end
|
||||
end
|
||||
|
||||
end
|
||||
|
||||
context "define_attribute_methods" do
|
||||
it "returns a falsy value if the attribute methods have already been generated" do
|
||||
Employee.define_attribute_methods.should be_false
|
||||
end
|
||||
end
|
||||
end
|
||||
|
||||
@@ -5,17 +5,17 @@ begin
|
||||
|
||||
require 'mongoid'
|
||||
require 'validates_timeliness/orm/mongoid'
|
||||
|
||||
Mongoid.configure do |config|
|
||||
name = "validates_timeliness_test"
|
||||
host = "localhost"
|
||||
config.master = Mongo::Connection.new.db(name)
|
||||
config.persist_in_safe_mode = false
|
||||
config.connect_to('validates_timeliness_test')
|
||||
end
|
||||
|
||||
describe ValidatesTimeliness, 'Mongoid' do
|
||||
after(:each) do
|
||||
Mongoid.purge!
|
||||
end
|
||||
|
||||
class Article
|
||||
::ValidatesTimeliness.use_plugin_parser = true
|
||||
include Mongoid::Document
|
||||
field :publish_date, :type => Date
|
||||
field :publish_time, :type => Time
|
||||
@@ -23,10 +23,11 @@ describe ValidatesTimeliness, 'Mongoid' do
|
||||
validates_date :publish_date, :allow_nil => true
|
||||
validates_time :publish_time, :allow_nil => true
|
||||
validates_datetime :publish_datetime, :allow_nil => true
|
||||
::ValidatesTimeliness.use_plugin_parser = false
|
||||
end
|
||||
|
||||
context "validation methods" do
|
||||
let(:record) { Article.new }
|
||||
|
||||
it 'should be defined on the class' do
|
||||
Article.should respond_to(:validates_date)
|
||||
Article.should respond_to(:validates_time)
|
||||
@@ -34,49 +35,124 @@ describe ValidatesTimeliness, 'Mongoid' do
|
||||
end
|
||||
|
||||
it 'should be defined on the instance' do
|
||||
Article.new.should respond_to(:validates_date)
|
||||
Article.new.should respond_to(:validates_time)
|
||||
Article.new.should respond_to(:validates_datetime)
|
||||
record.should respond_to(:validates_date)
|
||||
record.should respond_to(:validates_time)
|
||||
record.should respond_to(:validates_datetime)
|
||||
end
|
||||
|
||||
it "should validate a valid value string" do
|
||||
record.publish_date = '2012-01-01'
|
||||
|
||||
record.valid?
|
||||
record.errors[:publish_date].should be_empty
|
||||
end
|
||||
|
||||
it "should validate a nil value" do
|
||||
record.publish_date = nil
|
||||
|
||||
record.valid?
|
||||
record.errors[:publish_date].should be_empty
|
||||
end
|
||||
end
|
||||
|
||||
it 'should determine type for attribute' do
|
||||
Article.timeliness_attribute_type(:publish_date).should == :date
|
||||
Article.timeliness_attribute_type(:publish_time).should == :time
|
||||
Article.timeliness_attribute_type(:publish_datetime).should == :datetime
|
||||
end
|
||||
|
||||
context "attribute write method" do
|
||||
let(:record) { Article.new }
|
||||
|
||||
it 'should cache attribute raw value' do
|
||||
r = Article.new
|
||||
r.publish_datetime = date_string = '2010-01-01'
|
||||
r._timeliness_raw_value_for(:publish_datetime).should == date_string
|
||||
record.publish_datetime = date_string = '2010-01-01'
|
||||
|
||||
record._timeliness_raw_value_for('publish_datetime').should == date_string
|
||||
end
|
||||
|
||||
context "with plugin parser" do
|
||||
with_config(:use_plugin_parser, false)
|
||||
let(:record) { ArticleWithParser.new }
|
||||
|
||||
it 'should parse a string value' do
|
||||
Timeliness::Parser.should_receive(:parse)
|
||||
r = Article.new
|
||||
r.publish_date = '2010-01-01'
|
||||
class ArticleWithParser
|
||||
include Mongoid::Document
|
||||
field :publish_date, :type => Date
|
||||
field :publish_time, :type => Time
|
||||
field :publish_datetime, :type => DateTime
|
||||
|
||||
ValidatesTimeliness.use_plugin_parser = true
|
||||
validates_date :publish_date, :allow_nil => true
|
||||
validates_time :publish_time, :allow_nil => true
|
||||
validates_datetime :publish_datetime, :allow_nil => true
|
||||
ValidatesTimeliness.use_plugin_parser = false
|
||||
end
|
||||
|
||||
context "for a date column" do
|
||||
it 'should store a date value after parsing string' do
|
||||
r = Article.new
|
||||
r.publish_date = '2010-01-01'
|
||||
it 'should parse a string value' do
|
||||
Timeliness::Parser.should_receive(:parse)
|
||||
|
||||
r.publish_date.should be_kind_of(Date)
|
||||
r.publish_date.should == Date.new(2010, 1, 1)
|
||||
record.publish_date = '2010-01-01'
|
||||
end
|
||||
|
||||
it 'should parse a invalid string value as nil' do
|
||||
Timeliness::Parser.should_receive(:parse)
|
||||
|
||||
record.publish_date = 'not valid'
|
||||
end
|
||||
|
||||
it 'should store a Date value after parsing string' do
|
||||
record.publish_date = '2010-01-01'
|
||||
|
||||
record.publish_date.should be_kind_of(Date)
|
||||
record.publish_date.should eq Date.new(2010, 1, 1)
|
||||
end
|
||||
end
|
||||
|
||||
context "for a time column" do
|
||||
it 'should parse a string value' do
|
||||
Timeliness::Parser.should_receive(:parse)
|
||||
|
||||
record.publish_time = '12:30'
|
||||
end
|
||||
|
||||
it 'should parse a invalid string value as nil' do
|
||||
Timeliness::Parser.should_receive(:parse)
|
||||
|
||||
record.publish_time = 'not valid'
|
||||
end
|
||||
|
||||
it 'should store a Time value after parsing string' do
|
||||
record.publish_time = '12:30'
|
||||
|
||||
record.publish_time.should be_kind_of(Time)
|
||||
record.publish_time.should eq Time.utc(2000, 1, 1, 12, 30)
|
||||
end
|
||||
end
|
||||
|
||||
context "for a datetime column" do
|
||||
it 'should parse string into Time value' do
|
||||
r = Article.new
|
||||
r.publish_datetime = '2010-01-01 12:00'
|
||||
with_config(:default_timezone, 'Australia/Melbourne')
|
||||
|
||||
r.publish_datetime.should be_kind_of(Time)
|
||||
r.publish_datetime.should == Time.utc(2010,1,1,12,0)
|
||||
it 'should parse a string value' do
|
||||
Timeliness::Parser.should_receive(:parse)
|
||||
|
||||
record.publish_datetime = '2010-01-01 12:00'
|
||||
end
|
||||
|
||||
it 'should parse a invalid string value as nil' do
|
||||
Timeliness::Parser.should_receive(:parse)
|
||||
|
||||
record.publish_datetime = 'not valid'
|
||||
end
|
||||
|
||||
it 'should parse string into DateTime value' do
|
||||
record.publish_datetime = '2010-01-01 12:00'
|
||||
|
||||
record.publish_datetime.should be_kind_of(DateTime)
|
||||
end
|
||||
|
||||
it 'should parse string as current timezone' do
|
||||
record.publish_datetime = '2010-06-01 12:00'
|
||||
|
||||
record.publish_datetime.utc_offset.should eq Time.zone.utc_offset
|
||||
end
|
||||
end
|
||||
end
|
||||
@@ -84,22 +160,64 @@ describe ValidatesTimeliness, 'Mongoid' do
|
||||
|
||||
context "cached value" do
|
||||
it 'should be cleared on reload' do
|
||||
r = Article.create!
|
||||
r.publish_date = '2010-01-01'
|
||||
r.reload
|
||||
r._timeliness_raw_value_for(:publish_date).should be_nil
|
||||
record = Article.create!
|
||||
record.publish_date = '2010-01-01'
|
||||
record.reload
|
||||
record._timeliness_raw_value_for('publish_date').should be_nil
|
||||
end
|
||||
end
|
||||
|
||||
context "before_type_cast method" do
|
||||
it 'should not be defined if ORM does not support it' do
|
||||
Article.new.should_not respond_to(:publish_datetime_before_type_cast)
|
||||
let(:record){ Article.new }
|
||||
|
||||
it 'should be defined on class if ORM supports it' do
|
||||
record.should respond_to(:publish_datetime_before_type_cast)
|
||||
end
|
||||
|
||||
it 'should return original value' do
|
||||
record.publish_datetime = date_string = '2010-01-01'
|
||||
|
||||
record.publish_datetime_before_type_cast.should eq date_string
|
||||
end
|
||||
|
||||
it 'should return attribute if no attribute assignment has been made' do
|
||||
time = Time.zone.local(2010,01,01)
|
||||
Article.create(:publish_datetime => time)
|
||||
record = Article.last
|
||||
record.publish_datetime_before_type_cast.should eq time.to_datetime
|
||||
end
|
||||
|
||||
context "with plugin parser" do
|
||||
with_config(:use_plugin_parser, true)
|
||||
|
||||
it 'should return original value' do
|
||||
record.publish_datetime = date_string = '2010-01-31'
|
||||
record.publish_datetime_before_type_cast.should eq date_string
|
||||
end
|
||||
end
|
||||
end
|
||||
|
||||
context "with aliased fields" do
|
||||
class ArticleWithAliasedFields
|
||||
include Mongoid::Document
|
||||
field :pd, as: :publish_date, :type => Date
|
||||
field :pt, as: :publish_time, :type => Time
|
||||
field :pdt, as: :publish_datetime, :type => DateTime
|
||||
validates_date :publish_date, :allow_nil => true
|
||||
validates_time :publish_time, :allow_nil => true
|
||||
validates_datetime :publish_datetime, :allow_nil => true
|
||||
end
|
||||
|
||||
it 'should determine type for attribute' do
|
||||
ArticleWithAliasedFields.timeliness_attribute_type(:publish_date).should == :date
|
||||
ArticleWithAliasedFields.timeliness_attribute_type(:publish_time).should == :time
|
||||
ArticleWithAliasedFields.timeliness_attribute_type(:publish_datetime).should == :datetime
|
||||
end
|
||||
end
|
||||
end
|
||||
|
||||
rescue LoadError
|
||||
puts "Mongoid specs skipped. Mongoid not installed"
|
||||
rescue StandardError
|
||||
puts "Mongoid specs skipped. MongoDB connection failed."
|
||||
rescue StandardError => e
|
||||
puts "Mongoid specs skipped. MongoDB connection failed with error: #{e.message}"
|
||||
end
|
||||
|
||||
@@ -1,8 +1,6 @@
|
||||
require 'spec_helper'
|
||||
|
||||
describe ValidatesTimeliness::Validator do
|
||||
NIL = [nil]
|
||||
|
||||
before do
|
||||
Timecop.freeze(Time.local_time(2010, 1, 1, 0, 0, 0))
|
||||
end
|
||||
@@ -19,6 +17,14 @@ describe ValidatesTimeliness::Validator do
|
||||
Person.validates :birth_datetime, :timeliness => {:is_at => Time.mktime(2010,1,1)}
|
||||
Person.validators.first.type.should == :datetime
|
||||
end
|
||||
|
||||
it 'should add attribute to timeliness attributes set' do
|
||||
PersonWithShim.timeliness_validated_attributes.should_not include(:birth_time)
|
||||
|
||||
PersonWithShim.validates :birth_time, :timeliness => {:is_at => "12:30"}
|
||||
|
||||
PersonWithShim.timeliness_validated_attributes.should include(:birth_time)
|
||||
end
|
||||
end
|
||||
|
||||
it 'should not be valid for value which not valid date or time value' do
|
||||
@@ -38,13 +44,24 @@ describe ValidatesTimeliness::Validator do
|
||||
describe ":allow_nil option" do
|
||||
it 'should not allow nil by default' do
|
||||
Person.validates_date :birth_date
|
||||
invalid!(:birth_date, NIL, 'is not a valid date')
|
||||
invalid!(:birth_date, [nil], 'is not a valid date')
|
||||
valid!(:birth_date, Date.today)
|
||||
end
|
||||
|
||||
it 'should allow nil when true' do
|
||||
Person.validates_date :birth_date, :allow_nil => true
|
||||
valid!(:birth_date, NIL)
|
||||
valid!(:birth_date, [nil])
|
||||
end
|
||||
|
||||
context "with raw value cache" do
|
||||
it "should not be valid with an invalid format" do
|
||||
PersonWithShim.validates_date :birth_date, :allow_nil => true
|
||||
|
||||
p = PersonWithShim.new
|
||||
p.birth_date = 'bogus'
|
||||
|
||||
p.should_not be_valid
|
||||
end
|
||||
end
|
||||
end
|
||||
|
||||
@@ -59,6 +76,17 @@ describe ValidatesTimeliness::Validator do
|
||||
Person.validates_date :birth_date, :allow_blank => true
|
||||
valid!(:birth_date, '')
|
||||
end
|
||||
|
||||
context "with raw value cache" do
|
||||
it "should not be valid with an invalid format" do
|
||||
PersonWithShim.validates_date :birth_date, :allow_blank => true
|
||||
|
||||
p = PersonWithShim.new
|
||||
p.birth_date = 'bogus'
|
||||
|
||||
p.should_not be_valid
|
||||
end
|
||||
end
|
||||
end
|
||||
|
||||
describe ":between option" do
|
||||
@@ -87,6 +115,19 @@ describe ValidatesTimeliness::Validator do
|
||||
valid!(:birth_date, on_or_before)
|
||||
end
|
||||
end
|
||||
|
||||
describe "range with excluded end value" do
|
||||
it 'should be split option into :on_or_after and :before values' do
|
||||
on_or_after, before = Date.new(2010,1,1), Date.new(2010,1,3)
|
||||
Person.validates_date :birth_date, :between => on_or_after...before
|
||||
Person.validators.first.options[:on_or_after].should == on_or_after
|
||||
Person.validators.first.options[:before].should == before
|
||||
invalid!(:birth_date, on_or_after - 1, "must be on or after 2010-01-01")
|
||||
invalid!(:birth_date, before, "must be before 2010-01-03")
|
||||
valid!(:birth_date, on_or_after)
|
||||
valid!(:birth_date, before - 1)
|
||||
end
|
||||
end
|
||||
end
|
||||
|
||||
describe ":ignore_usec option" do
|
||||
@@ -138,7 +179,7 @@ describe ValidatesTimeliness::Validator do
|
||||
it "should be added when ignore_restriction_errors is false" do
|
||||
with_config(:ignore_restriction_errors, false) do
|
||||
person.valid?
|
||||
person.errors[:birth_date].first.should match("Error occurred validating birth_date for :is_at restriction")
|
||||
person.errors[:birth_date].first.should match("Error occurred validating birth_date")
|
||||
end
|
||||
end
|
||||
|
||||
|
||||
@@ -12,9 +12,9 @@ Gem::Specification.new do |s|
|
||||
s.homepage = %q{http://github.com/adzap/validates_timeliness}
|
||||
|
||||
s.require_paths = ["lib"]
|
||||
s.files = `git ls-files`.split("\n") - %w{ .gitignore .rspec Gemfile Gemfile.lock autotest/discover.rb }
|
||||
s.files = `git ls-files`.split("\n") - %w{ .gitignore .rspec Gemfile Gemfile.lock autotest/discover.rb Appraisals Travis.yml } - Dir['gemsfiles/*']
|
||||
s.test_files = `git ls-files -- {test,spec,features}/*`.split("\n")
|
||||
s.extra_rdoc_files = ["README.rdoc", "CHANGELOG.rdoc", "LICENSE"]
|
||||
|
||||
s.add_runtime_dependency(%q<timeliness>, ["~> 0.3.3"])
|
||||
s.add_runtime_dependency(%q<timeliness>, ["~> 0.3.7"])
|
||||
end
|
||||
|
||||
Reference in New Issue
Block a user