Compare commits

..

32 Commits

Author SHA1 Message Date
Adam Meehan
acd10f7b64 v3.0.8 2011-12-24 17:50:22 +11:00
Adam Meehan
68b2579ca0 Merge pull request #61 from carlosantoniodasilva/rails-3.2-deprecation-warnings
Remove deprecated InstanceMethods module when using AS::Concern
2011-12-22 23:00:25 -08:00
Carlos Antonio da Silva
7a784a6c54 Remove InstanceMethods module when using AS::Concern, as it was deprecated in 3.2 2011-12-20 20:52:33 -02:00
Adam Meehan
907fd3e439 Fix for Mongoid shim for reload which was nicely moved into a module to allow super 2011-12-10 17:46:35 +11:00
Adam Meehan
e44e9d2f11 v3.0.7 2011-09-21 21:41:01 +10:00
Adam Meehan
86b7bc4829 Fix before_type_cast for non-dirty attributes (issue #52).
Only use before_type_cast implementation if earlier than 3.1.0 which has
it's own working version for date/time fields now.
2011-09-21 21:28:53 +10:00
Adam Meehan
34824bbbab Specs against rails 3.1.0 2011-09-18 18:56:06 +10:00
Adam Meehan
11ae94cba6 Cleanup DateTimeSelect extension specs 2011-09-18 18:53:38 +10:00
Adam Meehan
473da0fa5e Fix DateTimeSelect extension for Rails 3.1.0 2011-09-18 18:37:01 +10:00
Adam Meehan
e7b503ae4f Fix DateTimeSelect extension to use integers from param values.
Change extension spec to use string params.
2011-09-18 18:34:23 +10:00
Adam Meehan
af9d6cdba4 Fix brokena and brittle tests 2011-09-18 18:33:30 +10:00
Adam Meehan
9a99c47323 Can use simple Mongoid inclusion now that fixed Concern usage is released (a while ago) 2011-09-18 18:33:29 +10:00
Adam Meehan
fbc2867197 Remove Gemfile.lock from source and ignore 2011-09-18 18:33:29 +10:00
Adam Meehan
969edfcb83 Helper method spec 2011-09-18 18:33:29 +10:00
Adam Meehan
1faf0efa4b Spec helper validate method setup in test models not needed 2011-09-18 18:33:29 +10:00
Adam Meehan
2b53d713bc Merge pull request #45 from trusche/master
Add support for :discard_day option in DateTimeSelect extension
2011-09-18 01:32:08 -07:00
Thilo Rusche
ac90ab96cd fixed integer vs string expectation 2011-05-13 17:47:59 +01:00
Thilo Rusche
b4c1a39343 added support for :discard_day in form helpers 2011-05-13 17:13:11 +01:00
Adam Meehan
7886132114 version 3.0.5 2011-05-09 17:34:09 +10:00
Adam Meehan
c3b1ddffbb Readme typo 2011-05-09 07:45:38 +10:00
Adam Meehan
ac98978441 Better timeliness_type_cast_code 2011-05-09 07:43:41 +10:00
Adam Meehan
ea7c9ec7be Type caste to Date for date columns in AR after parsing string in attribute writer 2011-04-27 07:37:26 +10:00
Adam Meehan
e3dea5aebd use activemodel scoped class name in helper method to remove need for .kind override 2011-02-24 21:12:49 +11:00
Adam Meehan
9d654b1122 update to rails 3.0.4 2011-02-24 07:59:01 +11:00
Adam Meehan
2822e368e3 Exit restriction validation on first restriction error 2011-02-24 07:58:48 +11:00
Adam Meehan
2244d5e74d Use bundle gem tasks and move gemspec from Rakefile the .gemspec 2011-02-24 07:44:39 +11:00
Adam Meehan
1c70af4dba simplify data select spec 2011-02-23 20:05:09 +11:00
Adam Meehan
dac62060df version 3.0.5 2011-01-29 17:42:42 +11:00
Adam Meehan
4efa91f3b0 some Rakefile cleanup and simplification 2011-01-29 17:39:00 +11:00
Adam Meehan
38457ec334 change spec config setup to use class method with_config 2011-01-29 16:33:37 +11:00
Adam Meehan
3793ef2ed4 fix for validation nil value with :format option plugin parser disabled (issue 34) 2011-01-29 16:07:56 +11:00
Adam Meehan
983e80f239 move some files into spec/support and include ModelHelpers in config 2011-01-28 19:36:02 +11:00
31 changed files with 404 additions and 483 deletions

1
.gitignore vendored
View File

@@ -1,3 +1,4 @@
pkg/ pkg/
.bundle/ .bundle/
.rvmrc .rvmrc
Gemfile.lock

View File

@@ -1,3 +1,21 @@
= 3.0.8 [2011-12-24]
* Remove deprecated InstanceMethods module when using AS::Concern
* 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.
= 3.0.5 [2011-01-29]
* Fix for Conversion#parse when given nil value (closes issue #34)
= 3.0.4 [2011-01-22] = 3.0.4 [2011-01-22]
* Fix :between option which was being ignored (ebeigarts) * Fix :between option which was being ignored (ebeigarts)
* Use class_attribute to remove deprecated class_inheritable_accessor * Use class_attribute to remove deprecated class_inheritable_accessor

13
Gemfile
View File

@@ -2,15 +2,18 @@ source 'http://rubygems.org'
gemspec gemspec
gem 'ZenTest' gem 'rails', '3.1.0'
gem 'rails', '3.0.3' gem 'rspec', '~> 2.6'
gem 'rspec', '~> 2.4' gem 'rspec-rails', '~> 2.6'
gem 'rspec-rails', '~> 2.4'
gem 'timecop' gem 'timecop'
gem 'rspec_tag_matchers' gem 'rspec_tag_matchers'
gem 'ruby-debug', :platforms => [:ruby_18, :jruby]
gem 'ruby-debug19', :platforms => [:ruby_19]
group :mongoid do group :mongoid do
gem 'mongoid', '2.0.0.rc.6' gem 'mongoid', '~> 2.3.0'
gem 'bson_ext'
gem 'system_timer', :platforms => [:ruby_18]
end end
group :active_record do group :active_record do

View File

@@ -1,119 +0,0 @@
PATH
remote: .
specs:
validates_timeliness (3.0.4)
timeliness (~> 0.3.2)
GEM
remote: http://rubygems.org/
specs:
ZenTest (4.4.2)
abstract (1.0.0)
actionmailer (3.0.3)
actionpack (= 3.0.3)
mail (~> 2.2.9)
actionpack (3.0.3)
activemodel (= 3.0.3)
activesupport (= 3.0.3)
builder (~> 2.1.2)
erubis (~> 2.6.6)
i18n (~> 0.4)
rack (~> 1.2.1)
rack-mount (~> 0.6.13)
rack-test (~> 0.5.6)
tzinfo (~> 0.3.23)
activemodel (3.0.3)
activesupport (= 3.0.3)
builder (~> 2.1.2)
i18n (~> 0.4)
activerecord (3.0.3)
activemodel (= 3.0.3)
activesupport (= 3.0.3)
arel (~> 2.0.2)
tzinfo (~> 0.3.23)
activeresource (3.0.3)
activemodel (= 3.0.3)
activesupport (= 3.0.3)
activesupport (3.0.3)
arel (2.0.7)
bson (1.2.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.14)
activesupport (>= 2.3.6)
i18n (>= 0.4.0)
mime-types (~> 1.16)
treetop (~> 1.4.8)
mime-types (1.16)
mongo (1.2.0)
bson (>= 1.2.0)
mongoid (2.0.0.rc.6)
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.1)
rack-mount (0.6.13)
rack (>= 1.0.0)
rack-test (0.5.7)
rack (>= 1.0)
rails (3.0.3)
actionmailer (= 3.0.3)
actionpack (= 3.0.3)
activerecord (= 3.0.3)
activeresource (= 3.0.3)
activesupport (= 3.0.3)
bundler (~> 1.0)
railties (= 3.0.3)
railties (3.0.3)
actionpack (= 3.0.3)
activesupport (= 3.0.3)
rake (>= 0.8.7)
thor (~> 0.14.4)
rake (0.8.7)
rspec (2.4.0)
rspec-core (~> 2.4.0)
rspec-expectations (~> 2.4.0)
rspec-mocks (~> 2.4.0)
rspec-core (2.4.0)
rspec-expectations (2.4.0)
diff-lcs (~> 1.1.2)
rspec-mocks (2.4.0)
rspec-rails (2.4.1)
actionpack (~> 3.0)
activesupport (~> 3.0)
railties (~> 3.0)
rspec (~> 2.4.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.24)
will_paginate (3.0.pre2)
PLATFORMS
ruby
DEPENDENCIES
ZenTest
mongoid (= 2.0.0.rc.6)
rails (= 3.0.3)
rspec (~> 2.4)
rspec-rails (~> 2.4)
rspec_tag_matchers
sqlite3-ruby
timecop
timeliness (~> 0.3.2)
validates_timeliness!

View File

@@ -54,7 +54,7 @@ NOTE: You may wish to enable the plugin parser and the extensions to start. Plea
validates_datetime :occurred_at validates_datetime :occurred_at
validates_date :date_of_birth :before => lambda { 18.years.ago }, validates_date :date_of_birth, :before => lambda { 18.years.ago },
:before_message => "must be at least 18 years old" :before_message => "must be at least 18 years old"
validates_datetime :finish_time, :after => :start_time # Method symbol validates_datetime :finish_time, :after => :start_time # Method symbol

View File

@@ -1,37 +1,11 @@
require 'rubygems' require 'bundler'
Bundler::GemHelper.install_tasks
require 'rake/rdoctask' require 'rake/rdoctask'
require 'rake/gempackagetask'
require 'rubygems/specification'
require 'rspec/core/rake_task' require 'rspec/core/rake_task'
require 'lib/validates_timeliness/version'
GEM_NAME = "validates_timeliness"
GEM_VERSION = ValidatesTimeliness::VERSION
spec = Gem::Specification.new do |s|
s.name = GEM_NAME
s.version = GEM_VERSION
s.platform = Gem::Platform::RUBY
s.rubyforge_project = "validates_timeliness"
s.has_rdoc = true
s.extra_rdoc_files = ["README.rdoc", "CHANGELOG.rdoc", "LICENSE"]
s.summary = %q{Date and time validation plugin for Rails which allows custom formats}
s.description = s.summary
s.author = "Adam Meehan"
s.email = "adam.meehan@gmail.com"
s.homepage = "http://github.com/adzap/validates_timeliness"
s.require_path = 'lib'
s.files = %w(validates_timeliness.gemspec LICENSE CHANGELOG.rdoc README.rdoc Rakefile) + Dir.glob("{lib,spec}/**/*")
s.add_runtime_dependency 'timeliness', '~> 0.3.2'
end
desc 'Default: run specs.'
task :default => :spec
desc "Run specs" desc "Run specs"
RSpec::Core::RakeTask.new do |t| RSpec::Core::RakeTask.new(:spec)
t.pattern = "./spec/**/*_spec.rb" # don't need this, it's default.
end
desc "Generate code coverage" desc "Generate code coverage"
RSpec::Core::RakeTask.new(:coverage) do |t| RSpec::Core::RakeTask.new(:coverage) do |t|
@@ -48,18 +22,5 @@ Rake::RDocTask.new(:rdoc) do |rdoc|
rdoc.rdoc_files.include('lib/**/*.rb') rdoc.rdoc_files.include('lib/**/*.rb')
end end
Rake::GemPackageTask.new(spec) do |pkg| desc 'Default: run specs.'
pkg.gem_spec = spec task :default => :spec
end
desc "Install the gem locally"
task :install => [:package] do
sh %{gem install pkg/#{GEM_NAME}-#{GEM_VERSION}}
end
desc "Create a gemspec file"
task :make_spec do
File.open("#{GEM_NAME}.gemspec", "w") do |file|
file.puts spec.to_ruby
end
end

View File

@@ -31,14 +31,12 @@ module ValidatesTimeliness
end end
def define_timeliness_write_method(attr_name) def define_timeliness_write_method(attr_name)
type = timeliness_attribute_type(attr_name)
timezone_aware = timeliness_attribute_timezone_aware?(attr_name)
method_body, line = <<-EOV, __LINE__ + 1 method_body, line = <<-EOV, __LINE__ + 1
def #{attr_name}=(value) def #{attr_name}=(value)
@timeliness_cache ||= {} @timeliness_cache ||= {}
@timeliness_cache["#{attr_name}"] = value @timeliness_cache["#{attr_name}"] = value
#{ "value = Timeliness::Parser.parse(value, :#{type}, :zone => (:current if #{timezone_aware})) if value.is_a?(String)" if ValidatesTimeliness.use_plugin_parser }
#{ "if value.is_a?(String)\n#{timeliness_type_cast_code(attr_name, 'value')}\nend" if ValidatesTimeliness.use_plugin_parser }
super super
end end
EOV EOV
@@ -48,26 +46,31 @@ module ValidatesTimeliness
def define_timeliness_before_type_cast_method(attr_name) def define_timeliness_before_type_cast_method(attr_name)
method_body, line = <<-EOV, __LINE__ + 1 method_body, line = <<-EOV, __LINE__ + 1
def #{attr_name}_before_type_cast def #{attr_name}_before_type_cast
_timeliness_raw_value_for('#{attr_name}') _timeliness_raw_value_for('#{attr_name}') || @attributes['#{attr_name}']
end end
EOV EOV
generated_timeliness_methods.module_eval(method_body, __FILE__, line) generated_timeliness_methods.module_eval(method_body, __FILE__, line)
end end
def timeliness_type_cast_code(attr_name, var_name)
type = timeliness_attribute_type(attr_name)
timezone_aware = timeliness_attribute_timezone_aware?(attr_name)
timezone = :current if timezone_aware
"#{var_name} = Timeliness::Parser.parse(#{var_name}, :#{type}, :zone => #{timezone.inspect})"
end
def generated_timeliness_methods def generated_timeliness_methods
@generated_timeliness_methods ||= Module.new.tap { |m| include(m) } @generated_timeliness_methods ||= Module.new.tap { |m| include(m) }
end end
end end
module InstanceMethods def _timeliness_raw_value_for(attr_name)
def _timeliness_raw_value_for(attr_name) @timeliness_cache && @timeliness_cache[attr_name.to_s]
@timeliness_cache && @timeliness_cache[attr_name.to_s]
end
def _clear_timeliness_cache
@timeliness_cache = {}
end
end end
def _clear_timeliness_cache
@timeliness_cache = {}
end
end end
end end

View File

@@ -56,6 +56,7 @@ module ValidatesTimeliness
end end
def parse(value) def parse(value)
return nil if value.nil?
if ValidatesTimeliness.use_plugin_parser if ValidatesTimeliness.use_plugin_parser
Timeliness::Parser.parse(value, @type, :zone => (:current if @timezone_aware), :format => options[:format], :strict => false) Timeliness::Parser.parse(value, @type, :zone => (:current if @timezone_aware), :format => options[:format], :strict => false)
else else

View File

@@ -10,35 +10,52 @@ module ValidatesTimeliness
included do included do
alias_method_chain :datetime_selector, :timeliness alias_method_chain :datetime_selector, :timeliness
alias_method_chain :value, :timeliness
end 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
def datetime_selector_with_timeliness(*args)
@timeliness_date_or_time_tag = true
datetime_selector_without_timeliness(*args)
end end
def value(object) # adapted from activesupport/lib/active_support/core_ext/date_time/calculations.rb, line 36 (3.0.7)
unless @timeliness_date_or_time_tag && @template_object.params[@object_name] def change(options)
return super TimelinessDateTime.new(
end options[:year] || year,
options[:month] || month,
pairs = @template_object.params[@object_name].select {|k,v| k =~ /^#{@method_name}\(/ } options[:day] || day,
return super if pairs.empty? options[:hour] || hour,
options[:min] || (options[:hour] ? 0 : min),
values = [nil] * 6 options[:sec] || ((options[:hour] || options[:min]) ? 0 : sec)
pairs.map do |(param, value)| )
position = param.scan(/\(([0-9]*).*\)/).first.first
values[position.to_i-1] = value
end
TimelinessDateTime.new(*values)
end end
end end
def datetime_selector_with_timeliness(*args)
@timeliness_date_or_time_tag = true
datetime_selector_without_timeliness(*args)
end
def value_with_timeliness(object)
unless @timeliness_date_or_time_tag && @template_object.params[@object_name]
return value_without_timeliness(object)
end
@template_object.params[@object_name]
pairs = @template_object.params[@object_name].select {|k,v| k =~ /^#{@method_name}\(/ }
return value_without_timeliness(object) if pairs.empty?
values = [nil] * 6
pairs.map do |(param, 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
end end

View File

@@ -20,7 +20,7 @@ module ActiveModel
self.timeliness_validated_attributes ||= [] self.timeliness_validated_attributes ||= []
self.timeliness_validated_attributes += (attr_names - self.timeliness_validated_attributes) self.timeliness_validated_attributes += (attr_names - self.timeliness_validated_attributes)
end end
validates_with ValidatesTimeliness::Validator, options validates_with TimelinessValidator, options
end end
end end

View File

@@ -7,7 +7,8 @@ module ValidatesTimeliness
def define_attribute_methods def define_attribute_methods
super super
# Define write method and before_type_cast method # Define write method and before_type_cast method
define_timeliness_methods(true) use_before_type_cast = ::ActiveRecord::VERSION::STRING < '3.1.0'
define_timeliness_methods(use_before_type_cast)
end end
def timeliness_attribute_timezone_aware?(attr_name) def timeliness_attribute_timezone_aware?(attr_name)
@@ -18,15 +19,21 @@ module ValidatesTimeliness
def timeliness_attribute_type(attr_name) def timeliness_attribute_type(attr_name)
columns_hash[attr_name.to_s].type columns_hash[attr_name.to_s].type
end end
end
module InstanceMethods def timeliness_type_cast_code(attr_name, var_name)
def reload(*args) type = timeliness_attribute_type(attr_name)
_clear_timeliness_cache
super <<-END
#{super}
#{var_name} = #{var_name}.to_date if #{var_name} && :#{type} == :date
END
end end
end end
def reload(*args)
_clear_timeliness_cache
super
end
end end
end end
end end

View File

@@ -16,17 +16,10 @@ module ValidatesTimeliness
attr_names.each { |attr_name| define_timeliness_write_method(attr_name) } attr_names.each { |attr_name| define_timeliness_write_method(attr_name) }
end end
def define_timeliness_write_method(attr_name) def timeliness_type_cast_code(attr_name, var_name)
type = timeliness_attribute_type(attr_name) type = timeliness_attribute_type(attr_name)
method_body, line = <<-EOV, __LINE__ + 1
def #{attr_name}=(value) "#{var_name} = Timeliness::Parser.parse(value, :#{type})"
@timeliness_cache ||= {}
@timeliness_cache["#{attr_name}"] = value
#{ "value = Timeliness::Parser.parse(value, :#{type}) if value.is_a?(String)" if ValidatesTimeliness.use_plugin_parser }
write_attribute(:#{attr_name}, value)
end
EOV
class_eval(method_body, __FILE__, line)
end end
def timeliness_attribute_type(attr_name) def timeliness_attribute_type(attr_name)
@@ -38,27 +31,24 @@ module ValidatesTimeliness
end end
end end
def reload
_clear_timeliness_cache
super
end
end end
end end
end end
module Mongoid::Document module Mongoid::Document
# Due to how Mongoid misuses ActiveSupport::Concern, include ValidatesTimeliness::AttributeMethods
# the only way to override a core component method is include ValidatesTimeliness::ORM::Mongoid
# 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 # Pre-2.3 reload
_clear_timeliness_cache if instance_methods.include?('reload')
reload_without_timeliness def reload_with_timeliness
_clear_timeliness_cache
reload_without_timeliness
end
alias_method_chain :reload, :timeliness
end end
alias_method_chain :reload, :timeliness
end end

View File

@@ -20,9 +20,7 @@ module ValidatesTimeliness
:datetime => '%Y-%m-%d %H:%M:%S' :datetime => '%Y-%m-%d %H:%M:%S'
}.freeze }.freeze
def self.kind RESTRICTION_ERROR_MESSAGE = "Error occurred validating %s for %s restriction:\n%s"
:timeliness
end
def initialize(options) def initialize(options)
@type = options.delete(:type) || :datetime @type = options.delete(:type) || :datetime
@@ -45,17 +43,22 @@ module ValidatesTimeliness
value = parse(raw_value) if value.is_a?(String) || options[:format] value = parse(raw_value) if value.is_a?(String) || options[:format]
value = type_cast_value(value, @type) value = type_cast_value(value, @type)
return add_error(record, attr_name, :"invalid_#{@type}") if value.blank? add_error(record, attr_name, :"invalid_#{@type}") and return if value.blank?
validate_restrictions(record, attr_name, value)
end
def validate_restrictions(record, attr_name, value)
@restrictions_to_check.each do |restriction| @restrictions_to_check.each do |restriction|
begin begin
restriction_value = type_cast_value(evaluate_option_value(options[restriction], record), @type) restriction_value = type_cast_value(evaluate_option_value(options[restriction], record), @type)
unless value.send(RESTRICTIONS[restriction], restriction_value) unless value.send(RESTRICTIONS[restriction], restriction_value)
return add_error(record, attr_name, restriction, restriction_value) add_error(record, attr_name, restriction, restriction_value) and break
end end
rescue => e rescue => e
unless ValidatesTimeliness.ignore_restriction_errors unless ValidatesTimeliness.ignore_restriction_errors
add_error(record, attr_name, "Error occurred validating #{attr_name} for #{restriction.inspect} restriction:\n#{e.message}") message = RESTRICTION_ERROR_MESSAGE % [ attr_name, restriction.inspect, e.message ]
add_error(record, attr_name, message) and break
end end
end end
end end

View File

@@ -1,3 +1,3 @@
module ValidatesTimeliness module ValidatesTimeliness
VERSION = '3.0.4' VERSION = '3.0.8'
end end

View File

@@ -6,10 +6,12 @@ require 'active_record'
require 'action_view' require 'action_view'
require 'timecop' require 'timecop'
require 'rspec_tag_matchers' require 'rspec_tag_matchers'
require 'model_helpers'
require 'validates_timeliness' require 'validates_timeliness'
require 'test_model'
require 'support/test_model'
require 'support/model_helpers'
require 'support/config_helper'
ValidatesTimeliness.setup do |c| ValidatesTimeliness.setup do |c|
c.extend_orms = [ :active_record ] c.extend_orms = [ :active_record ]
@@ -47,9 +49,7 @@ class Person
attribute :birth_date, :date attribute :birth_date, :date
attribute :birth_time, :time attribute :birth_time, :time
attribute :birth_datetime, :datetime attribute :birth_datetime, :datetime
validates_date :birth_date
validates_time :birth_time
validates_datetime :birth_datetime
define_attribute_methods model_attributes.keys define_attribute_methods model_attributes.keys
end end
@@ -71,12 +71,10 @@ ActiveRecord::Schema.define(:version => 1) do
end end
class Employee < ActiveRecord::Base 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 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) def birth_date=(value)
self.redefined_birth_date_called = true self.redefined_birth_date_called = true
@@ -84,15 +82,14 @@ class Employee < ActiveRecord::Base
end end
end end
Rspec.configure do |c| RSpec.configure do |c|
c.mock_with :rspec c.mock_with :rspec
c.include(RspecTagMatchers) c.include(RspecTagMatchers)
c.include(ModelHelpers)
c.include(ConfigHelper)
c.before do c.before do
Person.reset_callbacks(:validate) Person.reset_callbacks(:validate)
PersonWithShim.timeliness_validated_attributes = [] PersonWithShim.timeliness_validated_attributes = []
Person._validators.clear Person._validators.clear
Employee.reset_callbacks(:validate)
Employee.timeliness_validated_attributes = []
Employee._validators.clear
end end
end end

View File

@@ -0,0 +1,26 @@
module ConfigHelper
extend ActiveSupport::Concern
# Justin French tip
def with_config(preference_name, temporary_value)
old_value = ValidatesTimeliness.send(preference_name)
ValidatesTimeliness.send(:"#{preference_name}=", temporary_value)
yield
ensure
ValidatesTimeliness.send(:"#{preference_name}=", old_value)
end
module ClassMethods
def with_config(preference_name, temporary_value)
original_config_value = ValidatesTimeliness.send(preference_name)
before(:all) do
ValidatesTimeliness.send(:"#{preference_name}=", temporary_value)
end
after(:all) do
ValidatesTimeliness.send(:"#{preference_name}=", original_config_value)
end
end
end
end

View File

@@ -31,7 +31,7 @@ module TestModel
end end
def initialize(attributes = nil) 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[column.to_s] = nil
hash hash
end end
@@ -39,7 +39,7 @@ module TestModel
end end
def attributes def attributes
@attributes.keys @attributes
end end
def attributes=(new_attributes={}) def attributes=(new_attributes={})
@@ -49,14 +49,12 @@ module TestModel
end end
def method_missing(method_id, *args, &block) def method_missing(method_id, *args, &block)
if !self.class.attribute_methods_generated? if match_attribute_method?(method_id.to_s)
self.class.define_attribute_methods self.class.model_attributes.keys.map(&:to_s) self.class.define_attribute_methods self.class.model_attributes.keys
method_name = method_id.to_s
send(method_id, *args, &block) send(method_id, *args, &block)
else else
super super
end end
end end
end end

View File

@@ -41,6 +41,8 @@ describe ValidatesTimeliness::AttributeMethods do
end end
context "with plugin parser" do context "with plugin parser" do
with_config(:use_plugin_parser, true)
class PersonWithParser class PersonWithParser
include TestModel include TestModel
include TestModelShim include TestModelShim
@@ -52,25 +54,12 @@ describe ValidatesTimeliness::AttributeMethods do
validates_datetime :birth_datetime validates_datetime :birth_datetime
end end
before :all do
ValidatesTimeliness.use_plugin_parser = true
end
it 'should parse a string value' do it 'should parse a string value' do
Timeliness::Parser.should_receive(:parse) Timeliness::Parser.should_receive(:parse)
r = PersonWithParser.new r = PersonWithParser.new
r.birth_date = '2010-01-01' r.birth_date = '2010-01-01'
end end
it 'should parse string as current timezone' do
r = PersonWithParser.new
r.birth_datetime = '2010-01-01 12:00'
r.birth_datetime.zone == Time.zone.name
end
after :all do
ValidatesTimeliness.use_plugin_parser = false
end
end end
end end

View File

@@ -3,13 +3,13 @@ require 'spec_helper'
describe ValidatesTimeliness::Conversion do describe ValidatesTimeliness::Conversion do
include ValidatesTimeliness::Conversion include ValidatesTimeliness::Conversion
let(:options) { Hash.new }
before do before do
Timecop.freeze(Time.mktime(2010, 1, 1, 0, 0, 0)) Timecop.freeze(Time.mktime(2010, 1, 1, 0, 0, 0))
end end
describe "#type_cast_value" do describe "#type_cast_value" do
let(:options) { Hash.new }
describe "for date type" do describe "for date type" do
it "should return same value for date value" do it "should return same value for date value" do
type_cast_value(Date.new(2010, 1, 1), :date).should == Date.new(2010, 1, 1) type_cast_value(Date.new(2010, 1, 1), :date).should == Date.new(2010, 1, 1)
@@ -114,17 +114,10 @@ describe ValidatesTimeliness::Conversion do
end end
describe "with custom dummy date" do describe "with custom dummy date" do
before do
@original_dummy_date = ValidatesTimeliness.dummy_date_for_time_type
ValidatesTimeliness.dummy_date_for_time_type = [2010, 1, 1]
end
it 'should return dummy time with custom dummy date' do it 'should return dummy time with custom dummy date' do
dummy_time(Time.utc(1999, 11, 22, 12, 34, 56)).should == Time.utc(2010, 1, 1, 12, 34, 56) with_config(:dummy_date_for_time_type, [2010, 1, 1] ) do
end dummy_time(Time.utc(1999, 11, 22, 12, 34, 56)).should == Time.utc(2010, 1, 1, 12, 34, 56)
end
after do
ValidatesTimeliness.dummy_date_for_time_type = @original_dummy_date
end end
end end
end end
@@ -206,4 +199,36 @@ describe ValidatesTimeliness::Conversion do
end end
end end
end end
describe "#parse" do
context "use_plugin_parser setting is true" do
with_config(:use_plugin_parser, true)
it 'should use timeliness' do
Timeliness::Parser.should_receive(:parse)
parse('2000-01-01')
end
end
context "use_plugin_parser setting is false" do
with_config(:use_plugin_parser, false)
it 'should use Time.zone.parse attribute is timezone aware' do
@timezone_aware = true
Time.zone.should_receive(:parse)
parse('2000-01-01')
end
it 'should use value#to_time if use_plugin_parser setting is false and attribute is not timezone aware' do
@timezone_aware = false
value = '2000-01-01'
value.should_receive(:to_time)
parse(value)
end
end
it 'should return nil if value is nil' do
parse(nil).should be_nil
end
end
end end

View File

@@ -4,9 +4,7 @@ describe ValidatesTimeliness::Extensions::DateTimeSelect do
include ActionView::Helpers::DateHelper include ActionView::Helpers::DateHelper
attr_reader :person, :params attr_reader :person, :params
before :all do with_config(:use_plugin_parser, true)
ValidatesTimeliness.use_plugin_parser = true
end
before do before do
@person = Person.new @person = Person.new
@@ -15,128 +13,105 @@ describe ValidatesTimeliness::Extensions::DateTimeSelect do
describe "datetime_select" do describe "datetime_select" do
it "should use param values when attribute is nil" do it "should use param values when attribute is nil" do
params["person"] = { @params["person"] = {
"birth_datetime(1i)" => 2009, "birth_datetime(1i)" => '2009',
"birth_datetime(2i)" => 2, "birth_datetime(2i)" => '2',
"birth_datetime(3i)" => 29, "birth_datetime(3i)" => '29',
"birth_datetime(4i)" => 12, "birth_datetime(4i)" => '12',
"birth_datetime(5i)" => 13, "birth_datetime(5i)" => '13',
"birth_datetime(6i)" => 14, "birth_datetime(6i)" => '14',
} }
person.birth_datetime = nil person.birth_datetime = nil
output = datetime_select(:person, :birth_datetime, :include_blank => true, :include_seconds => true) @output = datetime_select(:person, :birth_datetime, :include_blank => true, :include_seconds => true)
output.should have_tag('select[id=person_birth_datetime_1i] option[selected=selected]', '2009') should_have_datetime_selected(:birth_datetime, :year => 2009, :month => 'February', :day => 29, :hour => 12, :min => 13, :sec => 14)
output.should have_tag('select[id=person_birth_datetime_2i] option[selected=selected]', 'February')
output.should have_tag('select[id=person_birth_datetime_3i] option[selected=selected]', '29')
output.should have_tag('select[id=person_birth_datetime_4i] option[selected=selected]', '12')
output.should have_tag('select[id=person_birth_datetime_5i] option[selected=selected]', '13')
output.should have_tag('select[id=person_birth_datetime_6i] option[selected=selected]', '14')
end end
it "should override object values and use params if present" do it "should override object values and use params if present" do
params["person"] = { @params["person"] = {
"birth_datetime(1i)" => 2009, "birth_datetime(1i)" => '2009',
"birth_datetime(2i)" => 2, "birth_datetime(2i)" => '2',
"birth_datetime(3i)" => 29, "birth_datetime(3i)" => '29',
"birth_datetime(4i)" => 12, "birth_datetime(4i)" => '12',
"birth_datetime(5i)" => 13, "birth_datetime(5i)" => '13',
"birth_datetime(6i)" => 14, "birth_datetime(6i)" => '14',
} }
person.birth_datetime = "2010-01-01 15:16:17" person.birth_datetime = "2010-01-01 15:16:17"
output = datetime_select(:person, :birth_datetime, :include_blank => true, :include_seconds => true) @output = datetime_select(:person, :birth_datetime, :include_blank => true, :include_seconds => true)
output.should have_tag('select[id=person_birth_datetime_1i] option[selected=selected]', '2009') should_have_datetime_selected(:birth_datetime, :year => 2009, :month => 'February', :day => 29, :hour => 12, :min => 13, :sec => 14)
output.should have_tag('select[id=person_birth_datetime_2i] option[selected=selected]', 'February')
output.should have_tag('select[id=person_birth_datetime_3i] option[selected=selected]', '29')
output.should have_tag('select[id=person_birth_datetime_4i] option[selected=selected]', '12')
output.should have_tag('select[id=person_birth_datetime_5i] option[selected=selected]', '13')
output.should have_tag('select[id=person_birth_datetime_6i] option[selected=selected]', '14')
end end
it "should use attribute values from object if no params" do it "should use attribute values from object if no params" do
person.birth_datetime = "2009-01-02 12:13:14" person.birth_datetime = "2009-01-02 12:13:14"
output = datetime_select(:person, :birth_datetime, :include_blank => true, :include_seconds => true) @output = datetime_select(:person, :birth_datetime, :include_blank => true, :include_seconds => true)
output.should have_tag('select[id=person_birth_datetime_1i] option[selected=selected]', '2009') should_have_datetime_selected(:birth_datetime, :year => 2009, :month => 'January', :day => 2, :hour => 12, :min => 13, :sec => 14)
output.should have_tag('select[id=person_birth_datetime_2i] option[selected=selected]', 'January')
output.should have_tag('select[id=person_birth_datetime_3i] option[selected=selected]', '2')
output.should have_tag('select[id=person_birth_datetime_4i] option[selected=selected]', '12')
output.should have_tag('select[id=person_birth_datetime_5i] option[selected=selected]', '13')
output.should have_tag('select[id=person_birth_datetime_6i] option[selected=selected]', '14')
end end
it "should use attribute values if params does not contain attribute params" do it "should use attribute values if params does not contain attribute params" do
person.birth_datetime = "2009-01-02 12:13:14" person.birth_datetime = "2009-01-02 12:13:14"
params["person"] = { } @params["person"] = { }
output = datetime_select(:person, :birth_datetime, :include_blank => true, :include_seconds => true) @output = datetime_select(:person, :birth_datetime, :include_blank => true, :include_seconds => true)
output.should have_tag('select[id=person_birth_datetime_1i] option[selected=selected]', '2009') should_have_datetime_selected(:birth_datetime, :year => 2009, :month => 'January', :day => 2, :hour => 12, :min => 13, :sec => 14)
output.should have_tag('select[id=person_birth_datetime_2i] option[selected=selected]', 'January')
output.should have_tag('select[id=person_birth_datetime_3i] option[selected=selected]', '2')
output.should have_tag('select[id=person_birth_datetime_4i] option[selected=selected]', '12')
output.should have_tag('select[id=person_birth_datetime_5i] option[selected=selected]', '13')
output.should have_tag('select[id=person_birth_datetime_6i] option[selected=selected]', '14')
end end
it "should not select values when attribute value is nil and has no param values" do it "should not select values when attribute value is nil and has no param values" do
person.birth_datetime = nil person.birth_datetime = nil
output = datetime_select(:person, :birth_datetime, :include_blank => true, :include_seconds => true) @output = datetime_select(:person, :birth_datetime, :include_blank => true, :include_seconds => true)
output.should_not have_tag('select[id=person_birth_datetime_1i] option[selected=selected]') should_not_have_datetime_selected(:birth_datetime, :year, :month, :day, :hour, :min, :sec)
output.should_not have_tag('select[id=person_birth_datetime_2i] option[selected=selected]')
output.should_not have_tag('select[id=person_birth_datetime_3i] option[selected=selected]')
output.should_not have_tag('select[id=person_birth_datetime_4i] option[selected=selected]')
output.should_not have_tag('select[id=person_birth_datetime_5i] option[selected=selected]')
output.should_not have_tag('select[id=person_birth_datetime_6i] option[selected=selected]')
end end
end end
describe "date_select" do describe "date_select" do
it "should use param values when attribute is nil" do it "should use param values when attribute is nil" do
params["person"] = { @params["person"] = {
"birth_date(1i)" => 2009, "birth_date(1i)" => '2009',
"birth_date(2i)" => 2, "birth_date(2i)" => '2',
"birth_date(3i)" => 29, "birth_date(3i)" => '29',
} }
person.birth_date = nil 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)
output.should have_tag('select[id=person_birth_date_1i] option[selected=selected]', '2009') should_have_datetime_selected(:birth_date, :year => 2009, :month => 'February', :day => 29)
output.should have_tag('select[id=person_birth_date_2i] option[selected=selected]', 'February')
output.should have_tag('select[id=person_birth_date_3i] option[selected=selected]', '29')
end end
it "should override object values and use params if present" do it "should override object values and use params if present" do
params["person"] = { @params["person"] = {
"birth_date(1i)" => 2009, "birth_date(1i)" => '2009',
"birth_date(2i)" => 2, "birth_date(2i)" => '2',
"birth_date(3i)" => 29, "birth_date(3i)" => '29',
} }
person.birth_date = "2009-03-01" 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)
output.should have_tag('select[id=person_birth_date_1i] option[selected=selected]', '2009') should_have_datetime_selected(:birth_date, :year => 2009, :month => 'February', :day => 29)
output.should have_tag('select[id=person_birth_date_2i] option[selected=selected]', 'February')
output.should have_tag('select[id=person_birth_date_3i] option[selected=selected]', '29')
end end
it "should select attribute values from object if no params" do it "should select attribute values from object if no params" do
person.birth_date = "2009-01-02" 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)
output.should have_tag('select[id=person_birth_date_1i] option[selected=selected]', '2009') should_have_datetime_selected(:birth_date, :year => 2009, :month => 'January', :day => 2)
output.should have_tag('select[id=person_birth_date_2i] option[selected=selected]', 'January')
output.should have_tag('select[id=person_birth_date_3i] option[selected=selected]', '2')
end end
it "should select attribute values if params does not contain attribute params" do it "should select attribute values if params does not contain attribute params" do
person.birth_date = "2009-01-02" person.birth_date = "2009-01-02"
params["person"] = { } @params["person"] = { }
output = date_select(:person, :birth_date, :include_blank => true, :include_seconds => true) @output = date_select(:person, :birth_date, :include_blank => true)
output.should have_tag('select[id=person_birth_date_1i] option[selected=selected]', '2009') should_have_datetime_selected(:birth_date, :year => 2009, :month => 'January', :day => 2)
output.should have_tag('select[id=person_birth_date_2i] option[selected=selected]', 'January')
output.should have_tag('select[id=person_birth_date_3i] option[selected=selected]', '2')
end end
it "should not select values when attribute value is nil and has no param values" do it "should not select values when attribute value is nil and has no param values" do
person.birth_date = nil 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)
output.should_not have_tag('select[id=person_birth_date_1i] option[selected=selected]') should_not_have_datetime_selected(:birth_time, :year, :month, :day)
output.should_not have_tag('select[id=person_birth_date_2i] option[selected=selected]') end
output.should_not have_tag('select[id=person_birth_date_3i] option[selected=selected]')
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
end end
@@ -146,39 +121,43 @@ describe ValidatesTimeliness::Extensions::DateTimeSelect do
end end
it "should use param values when attribute is nil" do it "should use param values when attribute is nil" do
params["person"] = { @params["person"] = {
"birth_time(1i)" => 2000, "birth_time(1i)" => '2000',
"birth_time(2i)" => 1, "birth_time(2i)" => '1',
"birth_time(3i)" => 1, "birth_time(3i)" => '1',
"birth_time(4i)" => 12, "birth_time(4i)" => '12',
"birth_time(5i)" => 13, "birth_time(5i)" => '13',
"birth_time(6i)" => 14, "birth_time(6i)" => '14',
} }
person.birth_time = nil person.birth_time = nil
output = time_select(:person, :birth_time, :include_blank => true, :include_seconds => true) @output = time_select(:person, :birth_time, :include_blank => true, :include_seconds => true)
output.should have_tag('select[id=person_birth_time_4i] option[selected=selected]', '12') should_have_datetime_selected(:birth_time, :hour => 12, :min => 13, :sec => 14)
output.should have_tag('select[id=person_birth_time_5i] option[selected=selected]', '13')
output.should have_tag('select[id=person_birth_time_6i] option[selected=selected]', '14')
end end
it "should select attribute values from object if no params" do it "should select attribute values from object if no params" do
person.birth_time = "2000-01-01 12:13:14" person.birth_time = "2000-01-01 12:13:14"
output = time_select(:person, :birth_time, :include_blank => true, :include_seconds => true) @output = time_select(:person, :birth_time, :include_blank => true, :include_seconds => true)
output.should have_tag('select[id=person_birth_time_4i] option[selected=selected]', '12') should_have_datetime_selected(:birth_time, :hour => 12, :min => 13, :sec => 14)
output.should have_tag('select[id=person_birth_time_5i] option[selected=selected]', '13')
output.should have_tag('select[id=person_birth_time_6i] option[selected=selected]', '14')
end end
it "should not select values when attribute value is nil and has no param values" do it "should not select values when attribute value is nil and has no param values" do
person.birth_time = nil person.birth_time = nil
output = time_select(:person, :birth_time, :include_blank => true, :include_seconds => true) @output = time_select(:person, :birth_time, :include_blank => true, :include_seconds => true)
output.should_not have_tag('select[id=person_birth_time_4i] option[selected=selected]') should_not_have_datetime_selected(:birth_time, :hour, :min, :sec)
output.should_not have_tag('select[id=person_birth_time_5i] option[selected=selected]')
output.should_not have_tag('select[id=person_birth_time_6i] option[selected=selected]')
end end
end end
after :all do def should_have_datetime_selected(field, datetime_hash)
ValidatesTimeliness.use_plugin_parser = false datetime_hash.each do |key, value|
index = {:year => 1, :month => 2, :day => 3, :hour => 4, :min => 5, :sec => 6}[key]
@output.should have_tag("select[id=person_#{field}_#{index}i] option[selected=selected]", value.to_s)
end
end
def should_not_have_datetime_selected(field, *attributes)
attributes.each do |attribute|
index = {:year => 1, :month => 2, :day => 3, :hour => 4, :min => 5, :sec => 6}[attribute]
@output.should_not have_tag("select[id=person_#{attribute}_#{index}i] option[selected=selected]")
end
end end
end end

View File

@@ -13,9 +13,18 @@ describe ValidatesTimeliness, 'HelperMethods' do
Person.new.should respond_to(:validates_datetime) Person.new.should respond_to(:validates_datetime)
end end
it 'should validate instance when validation method called' do it 'should validate instance using class validation defined' do
Person.validates_date :birth_date
r = Person.new
r.valid?
r.errors[:birth_date].should_not be_empty
end
it 'should validate instance using instance valiation method' do
r = Person.new r = Person.new
r.validates_date :birth_date r.validates_date :birth_date
r.errors[:birth_date].should_not be_empty r.errors[:birth_date].should_not be_empty
end end
end end

View File

@@ -33,31 +33,46 @@ describe ValidatesTimeliness, 'ActiveRecord' do
end end
context "with plugin parser" do context "with plugin parser" do
with_config(:use_plugin_parser, true)
class EmployeeWithParser < ActiveRecord::Base class EmployeeWithParser < ActiveRecord::Base
set_table_name 'employees' set_table_name 'employees'
validates_date :birth_date validates_date :birth_date
validates_datetime :birth_datetime validates_datetime :birth_datetime
end end
before :all do
ValidatesTimeliness.use_plugin_parser = true
end
it 'should parse a string value' do it 'should parse a string value' do
Timeliness::Parser.should_receive(:parse) Timeliness::Parser.should_receive(:parse)
r = EmployeeWithParser.new r = EmployeeWithParser.new
r.birth_date = '2010-01-01' r.birth_date = '2010-01-01'
end end
it 'should parse string as current timezone' do context "for a date column" do
r = EmployeeWithParser.new it 'should store a date value after parsing string' do
r.birth_datetime = '2010-06-01 12:00' r = EmployeeWithParser.new
r.birth_datetime.utc_offset.should == 10.hours r.birth_date = '2010-01-01'
r.birth_date.should be_kind_of(Date)
r.birth_date.should == Date.new(2010, 1, 1)
end
end end
after :all do context "for a datetime column" do
Time.zone = 'Australia/Melbourne' with_config(:default_timezone, 'Australia/Melbourne')
ValidatesTimeliness.use_plugin_parser = false
it 'should parse string into Time value' do
r = EmployeeWithParser.new
r.birth_datetime = '2010-01-01 12:00'
r.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'
r.birth_datetime.utc_offset.should == Time.zone.utc_offset
end
end end
end end
end end
@@ -67,6 +82,7 @@ describe ValidatesTimeliness, 'ActiveRecord' do
r = Employee.create! r = Employee.create!
r.birth_date = '2010-01-01' r.birth_date = '2010-01-01'
r.reload r.reload
r._timeliness_raw_value_for(:birth_date).should be_nil r._timeliness_raw_value_for(:birth_date).should be_nil
end end
end end
@@ -79,7 +95,16 @@ describe ValidatesTimeliness, 'ActiveRecord' do
it 'should return original value' do it 'should return original value' do
r = Employee.new r = Employee.new
r.birth_datetime = date_string = '2010-01-01' r.birth_datetime = date_string = '2010-01-01'
r.birth_datetime_before_type_cast.should == date_string r.birth_datetime_before_type_cast.should == date_string
end 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)
r = Employee.last
r.birth_datetime_before_type_cast.should match(/2010-01-01 00:00:00/)
end
end end
end end

View File

@@ -12,20 +12,20 @@ Mongoid.configure do |config|
config.persist_in_safe_mode = false config.persist_in_safe_mode = false
end end
class Article
::ValidatesTimeliness.use_plugin_parser = true
include Mongoid::Document
field :publish_date, :type => Date
field :publish_time, :type => Time
field :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
::ValidatesTimeliness.use_plugin_parser = false
end
describe ValidatesTimeliness, 'Mongoid' do describe ValidatesTimeliness, 'Mongoid' do
class Article
include Mongoid::Document
ValidatesTimeliness.use_plugin_parser = true
field :publish_date, :type => Date
field :publish_time, :type => Time
field :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
ValidatesTimeliness.use_plugin_parser = false
end
context "validation methods" do context "validation methods" do
it 'should be defined on the class' do it 'should be defined on the class' do
Article.should respond_to(:validates_date) Article.should respond_to(:validates_date)
@@ -52,9 +52,7 @@ describe ValidatesTimeliness, 'Mongoid' do
end end
context "with plugin parser" do context "with plugin parser" do
before :all do with_config(:use_plugin_parser, true)
ValidatesTimeliness.use_plugin_parser = true
end
it 'should parse a string value' do it 'should parse a string value' do
Timeliness::Parser.should_receive(:parse) Timeliness::Parser.should_receive(:parse)
@@ -62,14 +60,32 @@ describe ValidatesTimeliness, 'Mongoid' do
r.publish_date = '2010-01-01' r.publish_date = '2010-01-01'
end end
it 'should parse string into Time value' do it 'should parse an invalid value as nil' do
Timeliness::Parser.should_receive(:parse)
r = Article.new r = Article.new
r.publish_datetime = '2010-01-01 12:00' r.publish_date = 'bad value'
r.publish_datetime.should == Time.utc(2010,1,1,12,0)
r.publish_date.should be_nil
end end
after :all do context "for a date column" do
ValidatesTimeliness.use_plugin_parser = false it 'should store a Date value after parsing string' do
r = Article.new
r.publish_date = '2010-01-01'
r.publish_date.should be_kind_of(Date)
r.publish_date.should == Date.new(2010, 1, 1)
end
end
context "for a datetime column" do
it 'should parse string into DateTime value' do
r = Article.new
r.publish_datetime = '2010-01-01 12:00'
r.publish_datetime.should be_kind_of(DateTime)
r.publish_datetime.should == DateTime.new(2010,1,1,12,0)
end
end end
end end
end end
@@ -85,13 +101,13 @@ describe ValidatesTimeliness, 'Mongoid' do
context "before_type_cast method" do context "before_type_cast method" do
it 'should not be defined if ORM does not support it' do it 'should not be defined if ORM does not support it' do
Article.new.should_not respond_to(:birth_datetime_before_type_cast) Article.new.should_not respond_to(:publish_datetime_before_type_cast)
end end
end end
end end
rescue LoadError rescue LoadError
puts "Mongoid specs skipped. Mongoid not installed" puts "Mongoid specs skipped. Mongoid not installed"
rescue StandardError rescue StandardError => e
puts "Mongoid specs skipped. MongoDB connection failed." puts "Mongoid specs skipped. MongoDB connection failed with error: #{e.message}"
end end

View File

@@ -1,8 +1,6 @@
require 'spec_helper' require 'spec_helper'
describe ValidatesTimeliness::Validator, ":after option" do describe ValidatesTimeliness::Validator, ":after option" do
include ModelHelpers
describe "for date type" do describe "for date type" do
before do before do
Person.validates_date :birth_date, :after => Date.new(2010, 1, 1) Person.validates_date :birth_date, :after => Date.new(2010, 1, 1)

View File

@@ -1,8 +1,6 @@
require 'spec_helper' require 'spec_helper'
describe ValidatesTimeliness::Validator, ":before option" do describe ValidatesTimeliness::Validator, ":before option" do
include ModelHelpers
describe "for date type" do describe "for date type" do
before do before do
Person.validates_date :birth_date, :before => Date.new(2010, 1, 1) Person.validates_date :birth_date, :before => Date.new(2010, 1, 1)

View File

@@ -1,8 +1,6 @@
require 'spec_helper' require 'spec_helper'
describe ValidatesTimeliness::Validator, ":is_at option" do describe ValidatesTimeliness::Validator, ":is_at option" do
include ModelHelpers
before do before do
Timecop.freeze(Time.local_time(2010, 1, 1, 0, 0, 0)) Timecop.freeze(Time.local_time(2010, 1, 1, 0, 0, 0))
end end

View File

@@ -1,8 +1,6 @@
require 'spec_helper' require 'spec_helper'
describe ValidatesTimeliness::Validator, ":on_or_after option" do describe ValidatesTimeliness::Validator, ":on_or_after option" do
include ModelHelpers
describe "for date type" do describe "for date type" do
before do before do
Person.validates_date :birth_date, :on_or_after => Date.new(2010, 1, 1) Person.validates_date :birth_date, :on_or_after => Date.new(2010, 1, 1)

View File

@@ -1,8 +1,6 @@
require 'spec_helper' require 'spec_helper'
describe ValidatesTimeliness::Validator, ":on_or_before option" do describe ValidatesTimeliness::Validator, ":on_or_before option" do
include ModelHelpers
describe "for date type" do describe "for date type" do
before do before do
Person.validates_date :birth_date, :on_or_before => Date.new(2010, 1, 1) Person.validates_date :birth_date, :on_or_before => Date.new(2010, 1, 1)

View File

@@ -1,17 +1,12 @@
require 'spec_helper' require 'spec_helper'
describe ValidatesTimeliness::Validator do describe ValidatesTimeliness::Validator do
include ModelHelpers
NIL = [nil] NIL = [nil]
before do before do
Timecop.freeze(Time.local_time(2010, 1, 1, 0, 0, 0)) Timecop.freeze(Time.local_time(2010, 1, 1, 0, 0, 0))
end end
it 'should return validator kind as :timeliness' do
ValidatesTimeliness::Validator.kind.should == :timeliness
end
describe "Model.validates with :timeliness option" do describe "Model.validates with :timeliness option" do
it 'should use plugin validator class' do it 'should use plugin validator class' do
Person.validates :birth_date, :timeliness => {:is_at => Date.new(2010,1,1), :type => :date} Person.validates :birth_date, :timeliness => {:is_at => Date.new(2010,1,1), :type => :date}
@@ -118,9 +113,7 @@ describe ValidatesTimeliness::Validator do
let(:person) { PersonWithFormatOption.new } let(:person) { PersonWithFormatOption.new }
before(:all) do with_config(:use_plugin_parser, true)
ValidatesTimeliness.use_plugin_parser = true
end
it "should be valid when value matches format" do it "should be valid when value matches format" do
person.birth_date = '11-12-1913' person.birth_date = '11-12-1913'
@@ -133,33 +126,34 @@ describe ValidatesTimeliness::Validator do
person.valid? person.valid?
person.errors[:birth_date].should include('is not a valid date') person.errors[:birth_date].should include('is not a valid date')
end end
after(:all) do
ValidatesTimeliness.use_plugin_parser = false
end
end end
describe "restriction value errors" do describe "restriction value errors" do
let(:person) { Person.new(:birth_date => Date.today) } let(:person) { Person.new(:birth_date => Date.today) }
before do before do
Person.validates_time :birth_date, :is_at => lambda { raise } Person.validates_time :birth_date, :is_at => lambda { raise }, :before => lambda { raise }
end end
it "should be added when ignore_restriction_errors is false" do it "should be added when ignore_restriction_errors is false" do
ValidatesTimeliness.ignore_restriction_errors = false with_config(:ignore_restriction_errors, false) do
person.valid? 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 end
it "should not be added when ignore_restriction_errors is true" do it "should not be added when ignore_restriction_errors is true" do
ValidatesTimeliness.ignore_restriction_errors = true with_config(:ignore_restriction_errors, true) do
person.valid? person.valid?
person.errors[:birth_date].should be_empty person.errors[:birth_date].should be_empty
end
end end
after :all do it 'should exit on first error' do
ValidatesTimeliness.ignore_restriction_errors = false with_config(:ignore_restriction_errors, false) do
person.valid?
person.errors[:birth_date].should have(1).items
end
end end
end end

View File

@@ -1,32 +1,20 @@
# -*- encoding: utf-8 -*- # -*- encoding: utf-8 -*-
$:.push File.expand_path("../lib", __FILE__)
require "validates_timeliness/version"
Gem::Specification.new do |s| Gem::Specification.new do |s|
s.name = %q{validates_timeliness} s.name = "validates_timeliness"
s.version = "3.0.4" s.version = ValidatesTimeliness::VERSION
s.authors = ["Adam Meehan"]
s.summary = %q{Date and time validation plugin for Rails which allows custom formats}
s.description = %q{Adds validation methods to ActiveModel for validating dates and times. Works with multiple ORMS.}
s.email = %q{adam.meehan@gmail.com}
s.homepage = %q{http://github.com/adzap/validates_timeliness}
s.required_rubygems_version = Gem::Requirement.new(">= 0") if s.respond_to? :required_rubygems_version= s.require_paths = ["lib"]
s.authors = ["Adam Meehan"] s.files = `git ls-files`.split("\n") - %w{ .gitignore .rspec Gemfile Gemfile.lock autotest/discover.rb }
s.date = %q{2011-01-22} s.test_files = `git ls-files -- {test,spec,features}/*`.split("\n")
s.description = %q{Date and time validation plugin for Rails which allows custom formats}
s.email = %q{adam.meehan@gmail.com}
s.extra_rdoc_files = ["README.rdoc", "CHANGELOG.rdoc", "LICENSE"] s.extra_rdoc_files = ["README.rdoc", "CHANGELOG.rdoc", "LICENSE"]
s.files = ["validates_timeliness.gemspec", "LICENSE", "CHANGELOG.rdoc", "README.rdoc", "Rakefile", "lib/generators", "lib/generators/validates_timeliness", "lib/generators/validates_timeliness/install_generator.rb", "lib/generators/validates_timeliness/templates", "lib/generators/validates_timeliness/templates/en.yml", "lib/generators/validates_timeliness/templates/validates_timeliness.rb", "lib/validates_timeliness", "lib/validates_timeliness/attribute_methods.rb", "lib/validates_timeliness/conversion.rb", "lib/validates_timeliness/extensions", "lib/validates_timeliness/extensions/date_time_select.rb", "lib/validates_timeliness/extensions/multiparameter_handler.rb", "lib/validates_timeliness/extensions.rb", "lib/validates_timeliness/helper_methods.rb", "lib/validates_timeliness/orm", "lib/validates_timeliness/orm/active_record.rb", "lib/validates_timeliness/orm/mongoid.rb", "lib/validates_timeliness/railtie.rb", "lib/validates_timeliness/validator.rb", "lib/validates_timeliness/version.rb", "lib/validates_timeliness.rb", "spec/model_helpers.rb", "spec/spec_helper.rb", "spec/test_model.rb", "spec/validates_timeliness", "spec/validates_timeliness/attribute_methods_spec.rb", "spec/validates_timeliness/conversion_spec.rb", "spec/validates_timeliness/extensions", "spec/validates_timeliness/extensions/date_time_select_spec.rb", "spec/validates_timeliness/extensions/multiparameter_handler_spec.rb", "spec/validates_timeliness/helper_methods_spec.rb", "spec/validates_timeliness/orm", "spec/validates_timeliness/orm/active_record_spec.rb", "spec/validates_timeliness/orm/mongoid_spec.rb", "spec/validates_timeliness/validator", "spec/validates_timeliness/validator/after_spec.rb", "spec/validates_timeliness/validator/before_spec.rb", "spec/validates_timeliness/validator/is_at_spec.rb", "spec/validates_timeliness/validator/on_or_after_spec.rb", "spec/validates_timeliness/validator/on_or_before_spec.rb", "spec/validates_timeliness/validator_spec.rb", "spec/validates_timeliness_spec.rb"]
s.homepage = %q{http://github.com/adzap/validates_timeliness}
s.require_paths = ["lib"]
s.rubyforge_project = %q{validates_timeliness}
s.rubygems_version = %q{1.3.7}
s.summary = %q{Date and time validation plugin for Rails which allows custom formats}
if s.respond_to? :specification_version then s.add_runtime_dependency(%q<timeliness>, ["~> 0.3.4"])
current_version = Gem::Specification::CURRENT_SPECIFICATION_VERSION
s.specification_version = 3
if Gem::Version.new(Gem::VERSION) >= Gem::Version.new('1.2.0') then
s.add_runtime_dependency(%q<timeliness>, ["~> 0.3.2"])
else
s.add_dependency(%q<timeliness>, ["~> 0.3.2"])
end
else
s.add_dependency(%q<timeliness>, ["~> 0.3.2"])
end
end end