mirror of
https://github.com/ditkrg/validates_timeliness.git
synced 2026-01-25 15:22:58 +00:00
Compare commits
83 Commits
v3.0.15
...
4-0-stable
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
3269312ae2 | ||
|
|
d9b81b79a4 | ||
|
|
8a85da19e2 | ||
|
|
f2cd9aca17 | ||
|
|
5354f603ff | ||
|
|
6193410b55 | ||
|
|
93b8b1a70b | ||
|
|
e531c8f8ef | ||
|
|
658deca1c8 | ||
|
|
a6d617e77d | ||
|
|
101bb5d5f7 | ||
|
|
5e6e5222dc | ||
|
|
c81ec5d604 | ||
|
|
f3f3d01db7 | ||
|
|
acd9fc13e4 | ||
|
|
5743d87bc7 | ||
|
|
e0790bca9b | ||
|
|
39f698feb2 | ||
|
|
bd39aef4fb | ||
|
|
a3431bc91a | ||
|
|
fda194584a | ||
|
|
a05f091a42 | ||
|
|
faf708e3be | ||
|
|
38bb74844d | ||
|
|
e275b63203 | ||
|
|
e73e0eb30f | ||
|
|
43554d8bf0 | ||
|
|
da473b7eea | ||
|
|
fc2af73656 | ||
|
|
936c853fdd | ||
|
|
f27324a404 | ||
|
|
4221bf7709 | ||
|
|
b51a4544ff | ||
|
|
285bc769ba | ||
|
|
a0f2759c7a | ||
|
|
cc57b1b758 | ||
|
|
d30cebfc7b | ||
|
|
a96c24268d | ||
|
|
13fcc32a83 | ||
|
|
e8a96fe9b5 | ||
|
|
f9009995fc | ||
|
|
5419a1cc42 | ||
|
|
9dd2e87087 | ||
|
|
8a04deebfa | ||
|
|
e9c9914c4f | ||
|
|
0219c3850d | ||
|
|
35caf3638e | ||
|
|
153051730b | ||
|
|
5c1406e9b0 | ||
|
|
32f92cc885 | ||
|
|
cb3ac8d445 | ||
|
|
4e5f5fa695 | ||
|
|
321834b5ff | ||
|
|
ff131415f7 | ||
|
|
ad43eb31d6 | ||
|
|
0f782a4ed5 | ||
|
|
d57c64dcc2 | ||
|
|
244f398cc4 | ||
|
|
0feeb746cd | ||
|
|
6f23e40c25 | ||
|
|
773e816a4d | ||
|
|
80c9df448d | ||
|
|
b20b25940e | ||
|
|
51bc0d7d05 | ||
|
|
c7d87e3fe6 | ||
|
|
4dcebde1bc | ||
|
|
1a869bfb1f | ||
|
|
22ab7f4455 | ||
|
|
57379fd78a | ||
|
|
2424acc075 | ||
|
|
d883758046 | ||
|
|
2bcb3ae4dd | ||
|
|
0444bdc650 | ||
|
|
feb508f171 | ||
|
|
24f2752c2e | ||
|
|
afa8dee6ab | ||
|
|
08b16f155c | ||
|
|
1ddf0f1b0f | ||
|
|
6560a09ad2 | ||
|
|
d0f2158ae5 | ||
|
|
bb830baea2 | ||
|
|
4d48c3578b | ||
|
|
7d37882327 |
1
.gitignore
vendored
1
.gitignore
vendored
@@ -3,3 +3,4 @@ pkg/
|
|||||||
.rvmrc
|
.rvmrc
|
||||||
Gemfile.lock
|
Gemfile.lock
|
||||||
gemfiles/*.lock
|
gemfiles/*.lock
|
||||||
|
.byebug_history
|
||||||
|
|||||||
4
.rspec
4
.rspec
@@ -1,2 +1,4 @@
|
|||||||
--format nested
|
--format documentation
|
||||||
--color
|
--color
|
||||||
|
--require spec_helper
|
||||||
|
--require byebug
|
||||||
|
|||||||
24
.travis.yml
Normal file
24
.travis.yml
Normal file
@@ -0,0 +1,24 @@
|
|||||||
|
language: ruby
|
||||||
|
before_install:
|
||||||
|
- gem uninstall -v '>= 2' -i $(rvm gemdir)@global -ax bundler || true
|
||||||
|
- gem install bundler -v '< 2'
|
||||||
|
before_script:
|
||||||
|
- bundle install
|
||||||
|
cache: bundler
|
||||||
|
bundler_args: --verbose
|
||||||
|
|
||||||
|
matrix:
|
||||||
|
include:
|
||||||
|
- rvm: "2.4.6"
|
||||||
|
gemfile: gemfiles/rails_4_2.gemfile
|
||||||
|
- rvm: "2.5.5"
|
||||||
|
gemfile: gemfiles/rails_4_2.gemfile
|
||||||
|
|
||||||
|
script: 'bundle exec rspec'
|
||||||
|
|
||||||
|
notifications:
|
||||||
|
email:
|
||||||
|
recipients:
|
||||||
|
- adam.meehan@gmail.com
|
||||||
|
on_failure: change
|
||||||
|
on_success: never
|
||||||
28
Appraisals
28
Appraisals
@@ -1,27 +1,11 @@
|
|||||||
appraise "rails_3_0" do
|
appraise "rails_4_0" do
|
||||||
gem "rails", "~> 3.0.0"
|
gem "rails", "~> 4.0.13"
|
||||||
end
|
end
|
||||||
|
|
||||||
appraise "rails_3_1" do
|
appraise "rails_4_1" do
|
||||||
gem "rails", "~> 3.1.0"
|
gem "rails", "~> 4.1.14"
|
||||||
end
|
end
|
||||||
|
|
||||||
appraise "rails_3_2" do
|
appraise "rails_4_2" do
|
||||||
gem "rails", "~> 3.2.0"
|
gem "rails", "~> 4.2.11"
|
||||||
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
|
end
|
||||||
|
|||||||
@@ -1,3 +1,29 @@
|
|||||||
|
= 4.1.1 [2019-08-06]
|
||||||
|
* Add initializer to ensure Timeliness default ambigiuous date handling config
|
||||||
|
in Timeliness v0.4.1+ is set correctly when using `use_us_formats` or
|
||||||
|
`use_euro_formats` switcher to set default.
|
||||||
|
* Removed build support for Ruby 2.3 and Rails 4.0 and 4.1 to EOL official
|
||||||
|
support for those.
|
||||||
|
|
||||||
|
= 4.1.0 [2019-06-11]
|
||||||
|
* Relaxed Timeliness dependency version to >= 0.3.10 and < 1, which allows
|
||||||
|
version 0.4 with threadsafety fix for use_us_formats and use_euro_formats
|
||||||
|
hot switching in a request.
|
||||||
|
|
||||||
|
= 4.0.2 [2016-01-07]
|
||||||
|
* Fix undefine_generated_methods ivar guard setting to false
|
||||||
|
|
||||||
|
= 4.0.1 [2016-01-06]
|
||||||
|
* Fix undefine_generated_methods thread locking bug
|
||||||
|
* Created an ActiveModel ORM, for manual require if using without any full blown ORM
|
||||||
|
|
||||||
|
= 4.0.0 [2015-12-29]
|
||||||
|
* Extracted mongoid support into https://github.com/adzap/validates_timeliness-mongoid which is broken (not supported anymore).
|
||||||
|
* Fixed Rails 4.0, 4.1 and 4.2 compatability issues
|
||||||
|
* Upgrade specs to RSpec 3
|
||||||
|
* Added travis config
|
||||||
|
* Huge thanks to @johncarney for keeping it alive with his fork (https://github.com/johncarney/validates_timeliness)
|
||||||
|
|
||||||
= 3.0.15 [2015-12-29]
|
= 3.0.15 [2015-12-29]
|
||||||
* Fixes mongoid 3 support and removes mongoid 2 support(johnnyshields)
|
* Fixes mongoid 3 support and removes mongoid 2 support(johnnyshields)
|
||||||
* Some documentation/comments tidying
|
* Some documentation/comments tidying
|
||||||
|
|||||||
26
Gemfile
26
Gemfile
@@ -1,23 +1,13 @@
|
|||||||
source 'http://rubygems.org'
|
source 'https://rubygems.org'
|
||||||
|
|
||||||
gemspec
|
gemspec
|
||||||
|
|
||||||
gem 'rails', '~> 3.2.6'
|
gem 'rake'
|
||||||
gem 'rspec', '~> 2.8'
|
gem 'rails', '~> 4.2.11.1'
|
||||||
gem 'rspec-rails', '~> 2.8'
|
gem 'rspec'
|
||||||
|
gem 'rspec-rails', '~> 3.7'
|
||||||
gem 'timecop'
|
gem 'timecop'
|
||||||
gem 'rspec_tag_matchers'
|
gem 'byebug'
|
||||||
gem 'ruby-debug', :platforms => [:ruby_18, :jruby]
|
|
||||||
gem 'debugger', :platforms => [:ruby_19]
|
|
||||||
gem 'appraisal'
|
gem 'appraisal'
|
||||||
gem 'sqlite3'
|
gem 'sqlite3', '~> 1.3.13'
|
||||||
|
gem 'nokogiri', '~> 1.8'
|
||||||
group :mongoid do
|
|
||||||
gem 'mongoid', '>= 3.0'
|
|
||||||
gem 'bson_ext'
|
|
||||||
gem 'system_timer', :platforms => [:ruby_18]
|
|
||||||
end
|
|
||||||
|
|
||||||
group :active_record do
|
|
||||||
gem 'sqlite3-ruby', :require => 'sqlite3'
|
|
||||||
end
|
|
||||||
|
|||||||
13
README.rdoc
13
README.rdoc
@@ -5,9 +5,10 @@
|
|||||||
|
|
||||||
== Description
|
== Description
|
||||||
|
|
||||||
Complete validation of dates, times and datetimes for Rails 3.x and ActiveModel.
|
Complete validation of dates, times and datetimes for Rails 4.2.x and ActiveModel. Rails 4.0.x and 4.1.x may
|
||||||
|
still work but official support has ended.
|
||||||
|
|
||||||
If you a looking for the old version for Rails 2.x go here[http://github.com/adzap/validates_timeliness/tree/v2.3].
|
If you a looking for the old version for Rails 3.x go here[http://github.com/adzap/validates_timeliness/tree/v3.x].
|
||||||
|
|
||||||
|
|
||||||
== Features
|
== Features
|
||||||
@@ -22,7 +23,7 @@ If you a looking for the old version for Rails 2.x go here[http://github.com/adz
|
|||||||
|
|
||||||
* Adds extensions to fix Rails date/time select issues (See Extensions)
|
* Adds extensions to fix Rails date/time select issues (See Extensions)
|
||||||
|
|
||||||
* Supports I18n for the error messages
|
* Supports I18n for the error messages. For multi-language support try {timeliness-i18n gem}[https://github.com/pedrofurtado/timeliness-i18n].
|
||||||
|
|
||||||
* Supports all the Rubies (that any sane person would be using in production).
|
* Supports all the Rubies (that any sane person would be using in production).
|
||||||
|
|
||||||
@@ -30,7 +31,7 @@ If you a looking for the old version for Rails 2.x go here[http://github.com/adz
|
|||||||
== Installation
|
== Installation
|
||||||
|
|
||||||
# in Gemfile
|
# in Gemfile
|
||||||
gem 'validates_timeliness', '~> 3.0'
|
gem 'validates_timeliness', '~> 4.1'
|
||||||
|
|
||||||
# Run bundler
|
# Run bundler
|
||||||
$ bundle install
|
$ bundle install
|
||||||
@@ -137,8 +138,8 @@ like so
|
|||||||
|
|
||||||
ValidatesTimeliness.setup do |config|
|
ValidatesTimeliness.setup do |config|
|
||||||
|
|
||||||
# Extend ORM/ODMs for full support (:active_record, :mongoid).
|
# Extend ORM/ODMs for full support (:active_record).
|
||||||
config.extend_orms = [ :mongoid ]
|
config.extend_orms = [ :active_record ]
|
||||||
|
|
||||||
end
|
end
|
||||||
|
|
||||||
|
|||||||
@@ -1 +0,0 @@
|
|||||||
Autotest.add_discovery { "rspec2" }
|
|
||||||
@@ -1,16 +0,0 @@
|
|||||||
# 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=>"../"
|
|
||||||
@@ -1,16 +0,0 @@
|
|||||||
# 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=>"../"
|
|
||||||
@@ -1,16 +0,0 @@
|
|||||||
# 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=>"../"
|
|
||||||
@@ -1,16 +0,0 @@
|
|||||||
# 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=>"../"
|
|
||||||
@@ -1,15 +0,0 @@
|
|||||||
# 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=>"../"
|
|
||||||
@@ -1,15 +0,0 @@
|
|||||||
# 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=>"../"
|
|
||||||
@@ -1,15 +0,0 @@
|
|||||||
# 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=>"../"
|
|
||||||
14
gemfiles/rails_4_2.gemfile
Normal file
14
gemfiles/rails_4_2.gemfile
Normal file
@@ -0,0 +1,14 @@
|
|||||||
|
# This file was generated by Appraisal
|
||||||
|
|
||||||
|
source "https://rubygems.org"
|
||||||
|
|
||||||
|
gem "rails", "~> 4.2.8"
|
||||||
|
gem "rspec", "~> 3.6.0"
|
||||||
|
gem "rspec-rails", "~> 3.6.0"
|
||||||
|
gem "timecop"
|
||||||
|
gem "byebug"
|
||||||
|
gem "appraisal"
|
||||||
|
gem "sqlite3", "~> 1.3.0"
|
||||||
|
gem "nokogiri", "1.6.7"
|
||||||
|
|
||||||
|
gemspec path: "../"
|
||||||
@@ -1,6 +1,6 @@
|
|||||||
ValidatesTimeliness.setup do |config|
|
ValidatesTimeliness.setup do |config|
|
||||||
# Extend ORM/ODMs for full support (:active_record, :mongoid).
|
# Extend ORM/ODMs for full support (:active_record included).
|
||||||
# config.extend_orms = [ :active_record ]
|
config.extend_orms = [ :active_record ]
|
||||||
#
|
#
|
||||||
# Default timezone
|
# Default timezone
|
||||||
# config.default_timezone = :utc
|
# config.default_timezone = :utc
|
||||||
|
|||||||
@@ -28,7 +28,7 @@ module ValidatesTimeliness
|
|||||||
attr_accessor :extend_orms, :ignore_restriction_errors, :restriction_shorthand_symbols, :use_plugin_parser
|
attr_accessor :extend_orms, :ignore_restriction_errors, :restriction_shorthand_symbols, :use_plugin_parser
|
||||||
end
|
end
|
||||||
|
|
||||||
# Extend ORM/ODMs for full support (:active_record, :mongoid).
|
# Extend ORM/ODMs for full support (:active_record).
|
||||||
self.extend_orms = []
|
self.extend_orms = []
|
||||||
|
|
||||||
# Ignore errors when restriction options are evaluated
|
# Ignore errors when restriction options are evaluated
|
||||||
|
|||||||
@@ -20,13 +20,6 @@ module ValidatesTimeliness
|
|||||||
:datetime
|
:datetime
|
||||||
end
|
end
|
||||||
|
|
||||||
def undefine_attribute_methods
|
|
||||||
super
|
|
||||||
undefine_timeliness_attribute_methods
|
|
||||||
end
|
|
||||||
|
|
||||||
protected
|
|
||||||
|
|
||||||
def define_timeliness_methods(before_type_cast=false)
|
def define_timeliness_methods(before_type_cast=false)
|
||||||
return if timeliness_validated_attributes.blank?
|
return if timeliness_validated_attributes.blank?
|
||||||
timeliness_validated_attributes.each do |attr_name|
|
timeliness_validated_attributes.each do |attr_name|
|
||||||
@@ -34,55 +27,58 @@ module ValidatesTimeliness
|
|||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
|
def generated_timeliness_methods
|
||||||
|
@generated_timeliness_methods ||= Module.new { |m|
|
||||||
|
extend Mutex_m
|
||||||
|
}.tap { |mod| include mod }
|
||||||
|
end
|
||||||
|
|
||||||
|
def undefine_timeliness_attribute_methods
|
||||||
|
generated_timeliness_methods.module_eval do
|
||||||
|
instance_methods.each { |m| undef_method(m) }
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
protected
|
||||||
|
|
||||||
def define_attribute_timeliness_methods(attr_name, before_type_cast=false)
|
def define_attribute_timeliness_methods(attr_name, before_type_cast=false)
|
||||||
define_timeliness_write_method(attr_name)
|
define_timeliness_write_method(attr_name)
|
||||||
define_timeliness_before_type_cast_method(attr_name) if before_type_cast
|
define_timeliness_before_type_cast_method(attr_name) if before_type_cast
|
||||||
end
|
end
|
||||||
|
|
||||||
def define_timeliness_write_method(attr_name)
|
def define_timeliness_write_method(attr_name)
|
||||||
method_body, line = <<-EOV, __LINE__ + 1
|
generated_timeliness_methods.module_eval <<-STR, __FILE__, __LINE__ + 1
|
||||||
def #{attr_name}=(value)
|
def #{attr_name}=(value)
|
||||||
original_value = value
|
write_timeliness_attribute('#{attr_name}', value)
|
||||||
@timeliness_cache ||= {}
|
|
||||||
@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(value)
|
|
||||||
end
|
end
|
||||||
EOV
|
STR
|
||||||
generated_timeliness_methods.module_eval(method_body, __FILE__, line)
|
|
||||||
end
|
end
|
||||||
|
|
||||||
def define_timeliness_before_type_cast_method(attr_name)
|
def define_timeliness_before_type_cast_method(attr_name)
|
||||||
method_body, line = <<-EOV, __LINE__ + 1
|
generated_timeliness_methods.module_eval <<-STR, __FILE__, __LINE__ + 1
|
||||||
def #{attr_name}_before_type_cast
|
def #{attr_name}_before_type_cast
|
||||||
_timeliness_raw_value_for('#{attr_name}') || @attributes['#{attr_name}']
|
read_timeliness_attribute_before_type_cast('#{attr_name}')
|
||||||
end
|
end
|
||||||
EOV
|
STR
|
||||||
generated_timeliness_methods.module_eval(method_body, __FILE__, line)
|
|
||||||
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
|
|
||||||
@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
|
||||||
end
|
end
|
||||||
|
|
||||||
def _timeliness_raw_value_for(attr_name)
|
def write_timeliness_attribute(attr_name, value)
|
||||||
@timeliness_cache && @timeliness_cache[attr_name]
|
@timeliness_cache ||= {}
|
||||||
|
@timeliness_cache[attr_name] = value
|
||||||
|
|
||||||
|
if ValidatesTimeliness.use_plugin_parser
|
||||||
|
type = self.class.timeliness_attribute_type(attr_name)
|
||||||
|
timezone = :current if self.class.timeliness_attribute_timezone_aware?(attr_name)
|
||||||
|
value = Timeliness::Parser.parse(value, type, :zone => timezone)
|
||||||
|
value = value.to_date if value && type == :date
|
||||||
|
end
|
||||||
|
|
||||||
|
@attributes[attr_name] = value
|
||||||
|
end
|
||||||
|
|
||||||
|
def read_timeliness_attribute_before_type_cast(attr_name)
|
||||||
|
@timeliness_cache && @timeliness_cache[attr_name] || @attributes[attr_name]
|
||||||
end
|
end
|
||||||
|
|
||||||
def _clear_timeliness_cache
|
def _clear_timeliness_cache
|
||||||
|
|||||||
@@ -12,6 +12,8 @@ module ValidatesTimeliness
|
|||||||
value.to_date
|
value.to_date
|
||||||
when :datetime
|
when :datetime
|
||||||
value.is_a?(Time) ? value : value.to_time
|
value.is_a?(Time) ? value : value.to_time
|
||||||
|
else
|
||||||
|
value
|
||||||
end
|
end
|
||||||
if options[:ignore_usec] && value.is_a?(Time)
|
if options[:ignore_usec] && value.is_a?(Time)
|
||||||
Timeliness::Parser.make_time(Array(value).reverse[4..9], (:current if @timezone_aware))
|
Timeliness::Parser.make_time(Array(value).reverse[4..9], (:current if @timezone_aware))
|
||||||
|
|||||||
@@ -1,14 +1,13 @@
|
|||||||
module ValidatesTimeliness
|
module ValidatesTimeliness
|
||||||
module Extensions
|
module Extensions
|
||||||
autoload :DateTimeSelect, 'validates_timeliness/extensions/date_time_select'
|
autoload :DateTimeSelect, 'validates_timeliness/extensions/date_time_select'
|
||||||
autoload :MultiparameterHandler, 'validates_timeliness/extensions/multiparameter_handler'
|
|
||||||
end
|
end
|
||||||
|
|
||||||
def self.enable_date_time_select_extension!
|
def self.enable_date_time_select_extension!
|
||||||
::ActionView::Helpers::InstanceTag.send(:include, ValidatesTimeliness::Extensions::DateTimeSelect)
|
::ActionView::Helpers::Tags::DateSelect.send(:include, ValidatesTimeliness::Extensions::DateTimeSelect)
|
||||||
end
|
end
|
||||||
|
|
||||||
def self.enable_multiparameter_extension!
|
def self.enable_multiparameter_extension!
|
||||||
::ActiveRecord::Base.send(:include, ValidatesTimeliness::Extensions::MultiparameterHandler)
|
require 'validates_timeliness/extensions/multiparameter_handler'
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|||||||
@@ -9,7 +9,6 @@ module ValidatesTimeliness
|
|||||||
# It's a minor usability improvement which is rarely an issue for the user.
|
# It's a minor usability improvement which is rarely an issue for the user.
|
||||||
|
|
||||||
included do
|
included do
|
||||||
alias_method_chain :datetime_selector, :timeliness
|
|
||||||
alias_method_chain :value, :timeliness
|
alias_method_chain :value, :timeliness
|
||||||
end
|
end
|
||||||
|
|
||||||
@@ -33,15 +32,8 @@ module ValidatesTimeliness
|
|||||||
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)
|
def value_with_timeliness(object)
|
||||||
unless @timeliness_date_or_time_tag && @template_object.params[@object_name]
|
return value_without_timeliness(object) unless @template_object.params[@object_name]
|
||||||
return value_without_timeliness(object)
|
|
||||||
end
|
|
||||||
|
|
||||||
@template_object.params[@object_name]
|
@template_object.params[@object_name]
|
||||||
|
|
||||||
@@ -56,6 +48,7 @@ module ValidatesTimeliness
|
|||||||
|
|
||||||
TimelinessDateTime.new(*values)
|
TimelinessDateTime.new(*values)
|
||||||
end
|
end
|
||||||
|
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|||||||
@@ -1,80 +1,74 @@
|
|||||||
module ValidatesTimeliness
|
ActiveRecord::AttributeAssignment::MultiparameterAttribute.class_eval do
|
||||||
module Extensions
|
private
|
||||||
module MultiparameterHandler
|
|
||||||
extend ActiveSupport::Concern
|
|
||||||
|
|
||||||
# Stricter handling of date and time values from multiparameter
|
|
||||||
# assignment from the date/time select view helpers
|
|
||||||
|
|
||||||
included do
|
|
||||||
alias_method_chain :instantiate_time_object, :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
|
|
||||||
|
|
||||||
def invalid_multiparameter_date_or_time_as_string(values)
|
|
||||||
value = [values[0], *values[1..2].map {|s| s.to_s.rjust(2,"0")} ].join("-")
|
|
||||||
value += ' ' + values[3..5].map {|s| s.to_s.rjust(2, "0") }.join(":") unless values[3..5].empty?
|
|
||||||
value
|
|
||||||
end
|
|
||||||
|
|
||||||
def instantiate_time_object_with_timeliness(name, values)
|
|
||||||
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 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
|
|
||||||
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
|
|
||||||
unless errors.empty?
|
|
||||||
raise ActiveRecord::MultiparameterAssignmentErrors.new(errors), "#{errors.size} error(s) on assignment of multiparameter attributes"
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
|
# Yield if date values are valid
|
||||||
|
def validate_multiparameter_date_values(set_values)
|
||||||
|
if set_values[0..2].all?{ |v| v.present? } && Date.valid_civil?(*set_values[0..2])
|
||||||
|
yield
|
||||||
|
else
|
||||||
|
invalid_multiparameter_date_or_time_as_string(set_values)
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
|
def invalid_multiparameter_date_or_time_as_string(values)
|
||||||
|
value = [values[0], *values[1..2].map {|s| s.to_s.rjust(2,"0")} ].join("-")
|
||||||
|
value += ' ' + values[3..5].map {|s| s.to_s.rjust(2, "0") }.join(":") unless values[3..5].empty?
|
||||||
|
value
|
||||||
|
end
|
||||||
|
|
||||||
|
def instantiate_time_object(set_values)
|
||||||
|
raise if set_values.any?(&:nil?)
|
||||||
|
|
||||||
|
validate_multiparameter_date_values(set_values) {
|
||||||
|
set_values = set_values.map {|v| v.is_a?(String) ? v.strip : v }
|
||||||
|
|
||||||
|
if object.class.send(:create_time_zone_conversion_attribute?, name, cast_type_or_column)
|
||||||
|
Time.zone.local(*set_values)
|
||||||
|
else
|
||||||
|
Time.send(object.class.default_timezone, *set_values)
|
||||||
|
end
|
||||||
|
}
|
||||||
|
rescue
|
||||||
|
invalid_multiparameter_date_or_time_as_string(set_values)
|
||||||
|
end
|
||||||
|
|
||||||
|
def read_time
|
||||||
|
# If column is a :time (and not :date or :timestamp) there is no need to validate if
|
||||||
|
# there are year/month/day fields
|
||||||
|
if cast_type_or_column.type == :time
|
||||||
|
# if the column is a time set the values to their defaults as January 1, 1970, but only if they're nil
|
||||||
|
{ 1 => 1970, 2 => 1, 3 => 1 }.each do |key,value|
|
||||||
|
values[key] ||= value
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
max_position = extract_max_param(6)
|
||||||
|
set_values = values.values_at(*(1..max_position))
|
||||||
|
|
||||||
|
instantiate_time_object(set_values)
|
||||||
|
end
|
||||||
|
|
||||||
|
def read_date
|
||||||
|
set_values = values.values_at(1,2,3).map {|v| v.is_a?(String) ? v.strip : v }
|
||||||
|
|
||||||
|
if set_values.any? { |v| v.is_a?(String) }
|
||||||
|
Timeliness.parse(set_values.join('-'), :date).try(:to_date) or raise TypeError
|
||||||
|
else
|
||||||
|
Date.new(*set_values)
|
||||||
|
end
|
||||||
|
rescue TypeError, ArgumentError, NoMethodError => ex # if Date.new raises an exception on an invalid date
|
||||||
|
# Date.new with nil values throws NoMethodError
|
||||||
|
raise ex if ex.is_a?(NoMethodError) && ex.message !~ /undefined method `div' for/
|
||||||
|
invalid_multiparameter_date_or_time_as_string(set_values)
|
||||||
|
end
|
||||||
|
|
||||||
|
# Cast type is v4.2 and column before
|
||||||
|
def cast_type_or_column
|
||||||
|
@cast_type || @column
|
||||||
|
end
|
||||||
|
|
||||||
|
def timezone_conversion_attribute?
|
||||||
|
object.class.send(:create_time_zone_conversion_attribute?, name, column)
|
||||||
|
end
|
||||||
|
|
||||||
end
|
end
|
||||||
|
|||||||
@@ -14,8 +14,14 @@ module ActiveModel
|
|||||||
timeliness_validation_for attr_names, :datetime
|
timeliness_validation_for attr_names, :datetime
|
||||||
end
|
end
|
||||||
|
|
||||||
def timeliness_validation_for(attr_names, type)
|
def validates_timeliness_of(*attr_names)
|
||||||
validates_with TimelinessValidator, _merge_attributes(attr_names).merge(:type => type)
|
timeliness_validation_for attr_names
|
||||||
|
end
|
||||||
|
|
||||||
|
def timeliness_validation_for(attr_names, type=nil)
|
||||||
|
options = _merge_attributes(attr_names)
|
||||||
|
options.update(:type => type) if type
|
||||||
|
validates_with TimelinessValidator, options
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
|
|||||||
20
lib/validates_timeliness/orm/active_model.rb
Normal file
20
lib/validates_timeliness/orm/active_model.rb
Normal file
@@ -0,0 +1,20 @@
|
|||||||
|
module ValidatesTimeliness
|
||||||
|
module ORM
|
||||||
|
module ActiveModel
|
||||||
|
extend ActiveSupport::Concern
|
||||||
|
|
||||||
|
module ClassMethods
|
||||||
|
public
|
||||||
|
|
||||||
|
def define_attribute_methods(*attr_names)
|
||||||
|
super.tap { define_timeliness_methods}
|
||||||
|
end
|
||||||
|
|
||||||
|
def undefine_attribute_methods
|
||||||
|
super.tap { undefine_timeliness_attribute_methods }
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
@@ -14,28 +14,74 @@ module ValidatesTimeliness
|
|||||||
timeliness_column_for_attribute(attr_name).type
|
timeliness_column_for_attribute(attr_name).type
|
||||||
end
|
end
|
||||||
|
|
||||||
def timeliness_column_for_attribute(attr_name)
|
if ::ActiveModel.version >= Gem::Version.new('4.2')
|
||||||
columns_hash.fetch(attr_name.to_s) do |attr_name|
|
def timeliness_column_for_attribute(attr_name)
|
||||||
validation_type = _validators[attr_name.to_sym].find {|v| v.kind == :timeliness }.type
|
columns_hash.fetch(attr_name.to_s) do |key|
|
||||||
::ActiveRecord::ConnectionAdapters::Column.new(attr_name, nil, validation_type.to_s)
|
validation_type = _validators[key.to_sym].find {|v| v.kind == :timeliness }.type.to_s
|
||||||
|
::ActiveRecord::ConnectionAdapters::Column.new(key, nil, lookup_cast_type(validation_type), validation_type)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
def lookup_cast_type(sql_type)
|
||||||
|
case sql_type
|
||||||
|
when 'datetime' then ::ActiveRecord::Type::DateTime.new
|
||||||
|
when 'date' then ::ActiveRecord::Type::Date.new
|
||||||
|
when 'time' then ::ActiveRecord::Type::Time.new
|
||||||
|
end
|
||||||
|
end
|
||||||
|
else
|
||||||
|
def timeliness_column_for_attribute(attr_name)
|
||||||
|
columns_hash.fetch(attr_name.to_s) do |key|
|
||||||
|
validation_type = _validators[key.to_sym].find {|v| v.kind == :timeliness }.type.to_s
|
||||||
|
::ActiveRecord::ConnectionAdapters::Column.new(key, nil, validation_type)
|
||||||
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
def define_attribute_methods
|
def define_attribute_methods
|
||||||
super.tap do |attribute_methods_generated|
|
super.tap {
|
||||||
define_timeliness_methods true
|
generated_timeliness_methods.synchronize do
|
||||||
end
|
return if @timeliness_methods_generated
|
||||||
|
define_timeliness_methods true
|
||||||
|
@timeliness_methods_generated = true
|
||||||
|
end
|
||||||
|
}
|
||||||
end
|
end
|
||||||
|
|
||||||
protected
|
def undefine_attribute_methods
|
||||||
|
super.tap {
|
||||||
def timeliness_type_cast_code(attr_name, var_name)
|
generated_timeliness_methods.synchronize do
|
||||||
type = timeliness_attribute_type(attr_name)
|
return unless @timeliness_methods_generated
|
||||||
|
undefine_timeliness_attribute_methods
|
||||||
method_body = super
|
@timeliness_methods_generated = false
|
||||||
method_body << "\n#{var_name} = #{var_name}.to_date if #{var_name}" if type == :date
|
end
|
||||||
method_body
|
}
|
||||||
end
|
end
|
||||||
|
# Override to overwrite methods in ActiveRecord attribute method module because in AR 4+
|
||||||
|
# there is curious code which calls the method directly from the generated methods module
|
||||||
|
# via bind inside method_missing. This means our method in the formerly custom timeliness
|
||||||
|
# methods module was never reached.
|
||||||
|
def generated_timeliness_methods
|
||||||
|
generated_attribute_methods
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
def write_timeliness_attribute(attr_name, value)
|
||||||
|
@timeliness_cache ||= {}
|
||||||
|
@timeliness_cache[attr_name] = value
|
||||||
|
|
||||||
|
if ValidatesTimeliness.use_plugin_parser
|
||||||
|
type = self.class.timeliness_attribute_type(attr_name)
|
||||||
|
timezone = :current if self.class.timeliness_attribute_timezone_aware?(attr_name)
|
||||||
|
value = Timeliness::Parser.parse(value, type, :zone => timezone)
|
||||||
|
value = value.to_date if value && type == :date
|
||||||
|
end
|
||||||
|
|
||||||
|
write_attribute(attr_name, value)
|
||||||
|
end
|
||||||
|
|
||||||
|
def read_timeliness_attribute_before_type_cast(attr_name)
|
||||||
|
@timeliness_cache && @timeliness_cache[attr_name] || read_attribute_before_type_cast(attr_name)
|
||||||
end
|
end
|
||||||
|
|
||||||
def reload(*args)
|
def reload(*args)
|
||||||
|
|||||||
@@ -1,49 +0,0 @@
|
|||||||
module ValidatesTimeliness
|
|
||||||
module ORM
|
|
||||||
module Mongoid
|
|
||||||
extend ActiveSupport::Concern
|
|
||||||
# You need define the fields before you define the validations.
|
|
||||||
# It is best to use the plugin parser to avoid errors on a bad
|
|
||||||
# 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.
|
|
||||||
#
|
|
||||||
def timeliness_validation_for(attr_names, type)
|
|
||||||
super
|
|
||||||
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
|
|
||||||
end
|
|
||||||
|
|
||||||
def reload(*args)
|
|
||||||
_clear_timeliness_cache
|
|
||||||
super
|
|
||||||
end
|
|
||||||
end
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
module Mongoid::Document
|
|
||||||
include ValidatesTimeliness::AttributeMethods
|
|
||||||
include ValidatesTimeliness::ORM::Mongoid
|
|
||||||
end
|
|
||||||
@@ -11,5 +11,13 @@ module ValidatesTimeliness
|
|||||||
initializer "validates_timeliness.initialize_restriction_errors" do
|
initializer "validates_timeliness.initialize_restriction_errors" do
|
||||||
ValidatesTimeliness.ignore_restriction_errors = !Rails.env.test?
|
ValidatesTimeliness.ignore_restriction_errors = !Rails.env.test?
|
||||||
end
|
end
|
||||||
|
|
||||||
|
initializer "validates_timeliness.initialize_timeliness_ambiguous_date_format", :after => 'load_config_initializers' do
|
||||||
|
if Timeliness.respond_to?(:ambiguous_date_format) # i.e. v0.4+
|
||||||
|
# Set default for each new thread if you have changed the default using
|
||||||
|
# the format switching methods.
|
||||||
|
Timeliness.configuration.ambiguous_date_format = Timeliness::Definitions.current_date_format
|
||||||
|
end
|
||||||
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|||||||
@@ -5,7 +5,7 @@ module ValidatesTimeliness
|
|||||||
class Validator < ActiveModel::EachValidator
|
class Validator < ActiveModel::EachValidator
|
||||||
include Conversion
|
include Conversion
|
||||||
|
|
||||||
attr_reader :type
|
attr_reader :type, :attributes
|
||||||
|
|
||||||
RESTRICTIONS = {
|
RESTRICTIONS = {
|
||||||
:is_at => :==,
|
:is_at => :==,
|
||||||
@@ -42,16 +42,24 @@ module ValidatesTimeliness
|
|||||||
end
|
end
|
||||||
|
|
||||||
@restrictions_to_check = RESTRICTIONS.keys & options.keys
|
@restrictions_to_check = RESTRICTIONS.keys & options.keys
|
||||||
|
|
||||||
super
|
super
|
||||||
|
|
||||||
|
setup_timeliness_validated_attributes(options[:class]) if options[:class]
|
||||||
end
|
end
|
||||||
|
|
||||||
def setup(model)
|
def setup_timeliness_validated_attributes(model)
|
||||||
if model.respond_to?(:timeliness_validated_attributes)
|
if model.respond_to?(:timeliness_validated_attributes)
|
||||||
model.timeliness_validated_attributes ||= []
|
model.timeliness_validated_attributes ||= []
|
||||||
model.timeliness_validated_attributes |= @attributes
|
model.timeliness_validated_attributes |= attributes
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
|
# Rails 4.0 compatibility for old #setup method with class as arg
|
||||||
|
if Gem::Version.new(ActiveModel::VERSION::STRING) <= Gem::Version.new('4.1')
|
||||||
|
alias_method(:setup, :setup_timeliness_validated_attributes)
|
||||||
|
end
|
||||||
|
|
||||||
def validate_each(record, attr_name, value)
|
def validate_each(record, attr_name, value)
|
||||||
raw_value = attribute_raw_value(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?)
|
return if (@allow_nil && raw_value.nil?) || (@allow_blank && raw_value.blank?)
|
||||||
@@ -83,7 +91,7 @@ module ValidatesTimeliness
|
|||||||
|
|
||||||
def add_error(record, attr_name, message, value=nil)
|
def add_error(record, attr_name, message, value=nil)
|
||||||
value = format_error_value(value) if value
|
value = format_error_value(value) if value
|
||||||
message_options = { :message => options[:"#{message}_message"], :restriction => value }
|
message_options = { :message => options.fetch(:"#{message}_message", options[:message]), :restriction => value }
|
||||||
record.errors.add(attr_name, message, message_options)
|
record.errors.add(attr_name, message, message_options)
|
||||||
end
|
end
|
||||||
|
|
||||||
@@ -93,8 +101,8 @@ module ValidatesTimeliness
|
|||||||
end
|
end
|
||||||
|
|
||||||
def attribute_raw_value(record, attr_name)
|
def attribute_raw_value(record, attr_name)
|
||||||
record.respond_to?(:_timeliness_raw_value_for) &&
|
record.respond_to?(:read_timeliness_attribute_before_type_cast) &&
|
||||||
record._timeliness_raw_value_for(attr_name.to_s)
|
record.read_timeliness_attribute_before_type_cast(attr_name.to_s)
|
||||||
end
|
end
|
||||||
|
|
||||||
def timezone_aware?(record, attr_name)
|
def timezone_aware?(record, attr_name)
|
||||||
|
|||||||
@@ -1,3 +1,3 @@
|
|||||||
module ValidatesTimeliness
|
module ValidatesTimeliness
|
||||||
VERSION = '3.0.15'
|
VERSION = '4.1.1'
|
||||||
end
|
end
|
||||||
|
|||||||
@@ -5,13 +5,16 @@ require 'active_model/validations'
|
|||||||
require 'active_record'
|
require 'active_record'
|
||||||
require 'action_view'
|
require 'action_view'
|
||||||
require 'timecop'
|
require 'timecop'
|
||||||
require 'rspec_tag_matchers'
|
|
||||||
|
|
||||||
require 'validates_timeliness'
|
require 'validates_timeliness'
|
||||||
|
require 'validates_timeliness/orm/active_model'
|
||||||
|
|
||||||
|
require 'rails/railtie'
|
||||||
|
|
||||||
require 'support/test_model'
|
require 'support/test_model'
|
||||||
require 'support/model_helpers'
|
require 'support/model_helpers'
|
||||||
require 'support/config_helper'
|
require 'support/config_helper'
|
||||||
|
require 'support/tag_matcher'
|
||||||
|
|
||||||
ValidatesTimeliness.setup do |c|
|
ValidatesTimeliness.setup do |c|
|
||||||
c.extend_orms = [ :active_record ]
|
c.extend_orms = [ :active_record ]
|
||||||
@@ -24,19 +27,15 @@ Time.zone = 'Australia/Melbourne'
|
|||||||
|
|
||||||
LOCALE_PATH = File.expand_path(File.dirname(__FILE__) + '/../lib/generators/validates_timeliness/templates/en.yml')
|
LOCALE_PATH = File.expand_path(File.dirname(__FILE__) + '/../lib/generators/validates_timeliness/templates/en.yml')
|
||||||
I18n.load_path.unshift(LOCALE_PATH)
|
I18n.load_path.unshift(LOCALE_PATH)
|
||||||
|
I18n.available_locales = ['en', 'es']
|
||||||
|
|
||||||
# Extend TestModel as you would another ORM/ODM module
|
# Extend TestModel as you would another ORM/ODM module
|
||||||
module TestModelShim
|
module TestModelShim
|
||||||
extend ActiveSupport::Concern
|
extend ActiveSupport::Concern
|
||||||
include ValidatesTimeliness::AttributeMethods
|
include ValidatesTimeliness::AttributeMethods
|
||||||
|
include ValidatesTimeliness::ORM::ActiveModel
|
||||||
|
|
||||||
module ClassMethods
|
module ClassMethods
|
||||||
# Hook method for attribute method generation
|
|
||||||
def define_attribute_methods(attr_names)
|
|
||||||
super
|
|
||||||
define_timeliness_methods
|
|
||||||
end
|
|
||||||
|
|
||||||
# Hook into native time zone handling check, if any
|
# Hook into native time zone handling check, if any
|
||||||
def timeliness_attribute_timezone_aware?(attr_name)
|
def timeliness_attribute_timezone_aware?(attr_name)
|
||||||
false
|
false
|
||||||
@@ -74,8 +73,8 @@ end
|
|||||||
class Employee < ActiveRecord::Base
|
class Employee < ActiveRecord::Base
|
||||||
attr_accessor :redefined_birth_date_called
|
attr_accessor :redefined_birth_date_called
|
||||||
validates_date :birth_date, :allow_nil => true
|
validates_date :birth_date, :allow_nil => true
|
||||||
validates_date :birth_time, :allow_nil => true
|
validates_time :birth_time, :allow_nil => true
|
||||||
validates_date :birth_datetime, :allow_nil => true
|
validates_datetime :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
|
||||||
@@ -85,7 +84,7 @@ end
|
|||||||
|
|
||||||
RSpec.configure do |c|
|
RSpec.configure do |c|
|
||||||
c.mock_with :rspec
|
c.mock_with :rspec
|
||||||
c.include(RspecTagMatchers)
|
c.include(TagMatcher)
|
||||||
c.include(ModelHelpers)
|
c.include(ModelHelpers)
|
||||||
c.include(ConfigHelper)
|
c.include(ConfigHelper)
|
||||||
c.before do
|
c.before do
|
||||||
|
|||||||
@@ -3,15 +3,15 @@ module ModelHelpers
|
|||||||
# Some test helpers from Rails source
|
# Some test helpers from Rails source
|
||||||
def invalid!(attr_name, values, error = nil)
|
def invalid!(attr_name, values, error = nil)
|
||||||
with_each_person_value(attr_name, values) do |record, value|
|
with_each_person_value(attr_name, values) do |record, value|
|
||||||
record.should be_invalid
|
expect(record).to be_invalid
|
||||||
record.errors[attr_name].size.should >= 1
|
expect(record.errors[attr_name].size).to be >= 1
|
||||||
record.errors[attr_name].first.should == error if error
|
expect(record.errors[attr_name].first).to eq(error) if error
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
def valid!(attr_name, values)
|
def valid!(attr_name, values)
|
||||||
with_each_person_value(attr_name, values) do |record, value|
|
with_each_person_value(attr_name, values) do |record, value|
|
||||||
record.should be_valid
|
expect(record).to be_valid
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
|
|||||||
35
spec/support/tag_matcher.rb
Normal file
35
spec/support/tag_matcher.rb
Normal file
@@ -0,0 +1,35 @@
|
|||||||
|
require 'nokogiri'
|
||||||
|
|
||||||
|
module TagMatcher
|
||||||
|
extend RSpec::Matchers::DSL
|
||||||
|
|
||||||
|
matcher :have_tag do |selector|
|
||||||
|
match do |subject|
|
||||||
|
matches = doc(subject).search(selector)
|
||||||
|
|
||||||
|
if @inner_text
|
||||||
|
matches = matches.select { |element| element.inner_text == @inner_text }
|
||||||
|
end
|
||||||
|
|
||||||
|
matches.any?
|
||||||
|
end
|
||||||
|
|
||||||
|
chain :with_inner_text do |inner_text|
|
||||||
|
@inner_text = inner_text
|
||||||
|
end
|
||||||
|
|
||||||
|
private
|
||||||
|
|
||||||
|
def body(subject)
|
||||||
|
if subject.respond_to?(:body)
|
||||||
|
subject.body
|
||||||
|
else
|
||||||
|
subject.to_s
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
def doc(subject)
|
||||||
|
@doc ||= Nokogiri::HTML(body(subject))
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
@@ -5,7 +5,6 @@ module TestModel
|
|||||||
include ActiveModel::AttributeMethods
|
include ActiveModel::AttributeMethods
|
||||||
|
|
||||||
included do
|
included do
|
||||||
attribute_method_suffix ""
|
|
||||||
attribute_method_suffix "="
|
attribute_method_suffix "="
|
||||||
cattr_accessor :model_attributes
|
cattr_accessor :model_attributes
|
||||||
end
|
end
|
||||||
|
|||||||
@@ -1,8 +1,6 @@
|
|||||||
require 'spec_helper'
|
RSpec.describe ValidatesTimeliness::AttributeMethods do
|
||||||
|
it 'should define read_timeliness_attribute_before_type_cast instance method' do
|
||||||
describe ValidatesTimeliness::AttributeMethods do
|
expect(PersonWithShim.new).to respond_to(:read_timeliness_attribute_before_type_cast)
|
||||||
it 'should define _timeliness_raw_value_for instance method' do
|
|
||||||
PersonWithShim.new.should respond_to(:_timeliness_raw_value_for)
|
|
||||||
end
|
end
|
||||||
|
|
||||||
describe ".timeliness_validated_attributes" do
|
describe ".timeliness_validated_attributes" do
|
||||||
@@ -12,7 +10,7 @@ describe ValidatesTimeliness::AttributeMethods do
|
|||||||
PersonWithShim.validates_time :birth_time
|
PersonWithShim.validates_time :birth_time
|
||||||
PersonWithShim.validates_datetime :birth_datetime
|
PersonWithShim.validates_datetime :birth_datetime
|
||||||
|
|
||||||
PersonWithShim.timeliness_validated_attributes.should == [ :birth_date, :birth_time, :birth_datetime ]
|
expect(PersonWithShim.timeliness_validated_attributes).to eq([ :birth_date, :birth_time, :birth_datetime ])
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
@@ -31,13 +29,13 @@ describe ValidatesTimeliness::AttributeMethods do
|
|||||||
it 'should cache attribute raw value' do
|
it 'should cache attribute raw value' do
|
||||||
r = PersonWithCache.new
|
r = PersonWithCache.new
|
||||||
r.birth_datetime = date_string = '2010-01-01'
|
r.birth_datetime = date_string = '2010-01-01'
|
||||||
r._timeliness_raw_value_for('birth_datetime').should == date_string
|
expect(r.read_timeliness_attribute_before_type_cast('birth_datetime')).to eq(date_string)
|
||||||
end
|
end
|
||||||
|
|
||||||
it 'should not overwrite user defined methods' do
|
it 'should not overwrite user defined methods' do
|
||||||
e = Employee.new
|
e = Employee.new
|
||||||
e.birth_date = '2010-01-01'
|
e.birth_date = '2010-01-01'
|
||||||
e.redefined_birth_date_called.should be_true
|
expect(e.redefined_birth_date_called).to be_truthy
|
||||||
end
|
end
|
||||||
|
|
||||||
it 'should be undefined if model class has dynamic attribute methods reset' do
|
it 'should be undefined if model class has dynamic attribute methods reset' do
|
||||||
@@ -46,13 +44,13 @@ describe ValidatesTimeliness::AttributeMethods do
|
|||||||
r = PersonWithShim.new
|
r = PersonWithShim.new
|
||||||
r.birth_date = Time.now
|
r.birth_date = Time.now
|
||||||
|
|
||||||
write_method = RUBY_VERSION < '1.9' ? 'birth_date=' : :birth_date=
|
write_method = :birth_date=
|
||||||
|
|
||||||
PersonWithShim.send(:generated_timeliness_methods).instance_methods.should include(write_method)
|
expect(PersonWithShim.send(:generated_timeliness_methods).instance_methods).to include(write_method)
|
||||||
|
|
||||||
PersonWithShim.undefine_attribute_methods
|
PersonWithShim.undefine_attribute_methods
|
||||||
|
|
||||||
PersonWithShim.send(:generated_timeliness_methods).instance_methods.should_not include(write_method)
|
expect(PersonWithShim.send(:generated_timeliness_methods).instance_methods).not_to include(write_method)
|
||||||
end
|
end
|
||||||
|
|
||||||
context "with plugin parser" do
|
context "with plugin parser" do
|
||||||
@@ -70,7 +68,7 @@ describe ValidatesTimeliness::AttributeMethods do
|
|||||||
end
|
end
|
||||||
|
|
||||||
it 'should parse a string value' do
|
it 'should parse a string value' do
|
||||||
Timeliness::Parser.should_receive(:parse)
|
expect(Timeliness::Parser).to receive(:parse)
|
||||||
r = PersonWithParser.new
|
r = PersonWithParser.new
|
||||||
r.birth_date = '2010-01-01'
|
r.birth_date = '2010-01-01'
|
||||||
end
|
end
|
||||||
@@ -80,7 +78,7 @@ describe ValidatesTimeliness::AttributeMethods 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
|
||||||
PersonWithShim.new.should_not respond_to(:birth_datetime_before_type_cast)
|
expect(PersonWithShim.new).not_to respond_to(:birth_datetime_before_type_cast)
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|||||||
@@ -1,6 +1,4 @@
|
|||||||
require 'spec_helper'
|
RSpec.describe ValidatesTimeliness::Conversion do
|
||||||
|
|
||||||
describe ValidatesTimeliness::Conversion do
|
|
||||||
include ValidatesTimeliness::Conversion
|
include ValidatesTimeliness::Conversion
|
||||||
|
|
||||||
let(:options) { Hash.new }
|
let(:options) { Hash.new }
|
||||||
@@ -12,68 +10,68 @@ describe ValidatesTimeliness::Conversion do
|
|||||||
describe "#type_cast_value" do
|
describe "#type_cast_value" do
|
||||||
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)
|
expect(type_cast_value(Date.new(2010, 1, 1), :date)).to eq(Date.new(2010, 1, 1))
|
||||||
end
|
end
|
||||||
|
|
||||||
it "should return date part of time value" do
|
it "should return date part of time value" do
|
||||||
type_cast_value(Time.mktime(2010, 1, 1, 0, 0, 0), :date).should == Date.new(2010, 1, 1)
|
expect(type_cast_value(Time.mktime(2010, 1, 1, 0, 0, 0), :date)).to eq(Date.new(2010, 1, 1))
|
||||||
end
|
end
|
||||||
|
|
||||||
it "should return date part of datetime value" do
|
it "should return date part of datetime value" do
|
||||||
type_cast_value(DateTime.new(2010, 1, 1, 0, 0, 0), :date).should == Date.new(2010, 1, 1)
|
expect(type_cast_value(DateTime.new(2010, 1, 1, 0, 0, 0), :date)).to eq(Date.new(2010, 1, 1))
|
||||||
end
|
end
|
||||||
|
|
||||||
it 'should return nil for invalid value types' do
|
it 'should return nil for invalid value types' do
|
||||||
type_cast_value(12, :date).should == nil
|
expect(type_cast_value(12, :date)).to eq(nil)
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
describe "for time type" do
|
describe "for time type" do
|
||||||
it "should return same value for time value matching dummy date part" do
|
it "should return same value for time value matching dummy date part" do
|
||||||
type_cast_value(Time.utc(2000, 1, 1, 0, 0, 0), :time).should == Time.utc(2000, 1, 1, 0, 0, 0)
|
expect(type_cast_value(Time.utc(2000, 1, 1, 0, 0, 0), :time)).to eq(Time.utc(2000, 1, 1, 0, 0, 0))
|
||||||
end
|
end
|
||||||
|
|
||||||
it "should return dummy time value with same time part for time value with different date" do
|
it "should return dummy time value with same time part for time value with different date" do
|
||||||
type_cast_value(Time.utc(2010, 1, 1, 0, 0, 0), :time).should == Time.utc(2000, 1, 1, 0, 0, 0)
|
expect(type_cast_value(Time.utc(2010, 1, 1, 0, 0, 0), :time)).to eq(Time.utc(2000, 1, 1, 0, 0, 0))
|
||||||
end
|
end
|
||||||
|
|
||||||
it "should return dummy time only for date value" do
|
it "should return dummy time only for date value" do
|
||||||
type_cast_value(Date.new(2010, 1, 1), :time).should == Time.utc(2000, 1, 1, 0, 0, 0)
|
expect(type_cast_value(Date.new(2010, 1, 1), :time)).to eq(Time.utc(2000, 1, 1, 0, 0, 0))
|
||||||
end
|
end
|
||||||
|
|
||||||
it "should return dummy date with time part for datetime value" do
|
it "should return dummy date with time part for datetime value" do
|
||||||
type_cast_value(DateTime.civil_from_format(:utc, 2010, 1, 1, 12, 34, 56), :time).should == Time.utc(2000, 1, 1, 12, 34, 56)
|
expect(type_cast_value(DateTime.civil_from_format(:utc, 2010, 1, 1, 12, 34, 56), :time)).to eq(Time.utc(2000, 1, 1, 12, 34, 56))
|
||||||
end
|
end
|
||||||
|
|
||||||
it 'should return nil for invalid value types' do
|
it 'should return nil for invalid value types' do
|
||||||
type_cast_value(12, :time).should == nil
|
expect(type_cast_value(12, :time)).to eq(nil)
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
describe "for datetime type" do
|
describe "for datetime type" do
|
||||||
it "should return Date as Time value" do
|
it "should return Date as Time value" do
|
||||||
type_cast_value(Date.new(2010, 1, 1), :datetime).should == Time.local_time(2010, 1, 1, 0, 0, 0)
|
expect(type_cast_value(Date.new(2010, 1, 1), :datetime)).to eq(Time.local(2010, 1, 1, 0, 0, 0))
|
||||||
end
|
end
|
||||||
|
|
||||||
it "should return same Time value" do
|
it "should return same Time value" do
|
||||||
value = Time.utc(2010, 1, 1, 12, 34, 56)
|
value = Time.utc(2010, 1, 1, 12, 34, 56)
|
||||||
type_cast_value(Time.utc(2010, 1, 1, 12, 34, 56), :datetime).should == value
|
expect(type_cast_value(Time.utc(2010, 1, 1, 12, 34, 56), :datetime)).to eq(value)
|
||||||
end
|
end
|
||||||
|
|
||||||
it "should return as Time with same component values" do
|
it "should return as Time with same component values" do
|
||||||
type_cast_value(DateTime.civil_from_format(:utc, 2010, 1, 1, 12, 34, 56), :datetime).should == Time.utc(2010, 1, 1, 12, 34, 56)
|
expect(type_cast_value(DateTime.civil_from_format(:utc, 2010, 1, 1, 12, 34, 56), :datetime)).to eq(Time.utc(2010, 1, 1, 12, 34, 56))
|
||||||
end
|
end
|
||||||
|
|
||||||
it "should return same Time in correct zone if timezone aware" do
|
it "should return same Time in correct zone if timezone aware" do
|
||||||
@timezone_aware = true
|
@timezone_aware = true
|
||||||
value = Time.utc(2010, 1, 1, 12, 34, 56)
|
value = Time.utc(2010, 1, 1, 12, 34, 56)
|
||||||
result = type_cast_value(value, :datetime)
|
result = type_cast_value(value, :datetime)
|
||||||
result.should == Time.zone.local(2010, 1, 1, 23, 34, 56)
|
expect(result).to eq(Time.zone.local(2010, 1, 1, 23, 34, 56))
|
||||||
result.zone.should == 'EST'
|
expect(result.zone).to eq('AEDT')
|
||||||
end
|
end
|
||||||
|
|
||||||
it 'should return nil for invalid value types' do
|
it 'should return nil for invalid value types' do
|
||||||
type_cast_value(12, :datetime).should == nil
|
expect(type_cast_value(12, :datetime)).to eq(nil)
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
@@ -82,41 +80,41 @@ describe ValidatesTimeliness::Conversion do
|
|||||||
|
|
||||||
it "should ignore usec on time values when evaluated" do
|
it "should ignore usec on time values when evaluated" do
|
||||||
value = Time.utc(2010, 1, 1, 12, 34, 56, 10000)
|
value = Time.utc(2010, 1, 1, 12, 34, 56, 10000)
|
||||||
type_cast_value(value, :datetime).should == Time.utc(2010, 1, 1, 12, 34, 56)
|
expect(type_cast_value(value, :datetime)).to eq(Time.utc(2010, 1, 1, 12, 34, 56))
|
||||||
end
|
end
|
||||||
|
|
||||||
it "should ignore usec and return time in correct zone if timezone aware" do
|
it "should ignore usec and return time in correct zone if timezone aware" do
|
||||||
@timezone_aware = true
|
@timezone_aware = true
|
||||||
value = Time.utc(2010, 1, 1, 12, 34, 56, 10000)
|
value = Time.utc(2010, 1, 1, 12, 34, 56, 10000)
|
||||||
result = type_cast_value(value, :datetime)
|
result = type_cast_value(value, :datetime)
|
||||||
result.should == Time.zone.local(2010, 1, 1, 23, 34, 56)
|
expect(result).to eq(Time.zone.local(2010, 1, 1, 23, 34, 56))
|
||||||
result.zone.should == 'EST'
|
expect(result.zone).to eq('AEDT')
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
describe "#dummy_time" do
|
describe "#dummy_time" do
|
||||||
it 'should return Time with dummy date values but same time components' do
|
it 'should return Time with dummy date values but same time components' do
|
||||||
dummy_time(Time.utc(2010, 11, 22, 12, 34, 56)).should == Time.utc(2000, 1, 1, 12, 34, 56)
|
expect(dummy_time(Time.utc(2010, 11, 22, 12, 34, 56))).to eq(Time.utc(2000, 1, 1, 12, 34, 56))
|
||||||
end
|
end
|
||||||
|
|
||||||
it 'should return same value for Time which already has dummy date values' do
|
it 'should return same value for Time which already has dummy date values' do
|
||||||
dummy_time(Time.utc(2000, 1, 1, 12, 34, 56)).should == Time.utc(2000, 1, 1, 12, 34, 56)
|
expect(dummy_time(Time.utc(2000, 1, 1, 12, 34, 56))).to eq(Time.utc(2000, 1, 1, 12, 34, 56))
|
||||||
end
|
end
|
||||||
|
|
||||||
it 'should return time component values shifted to current zone if timezone aware' do
|
it 'should return time component values shifted to current zone if timezone aware' do
|
||||||
@timezone_aware = true
|
@timezone_aware = true
|
||||||
dummy_time(Time.utc(2000, 1, 1, 12, 34, 56)).should == Time.zone.local(2000, 1, 1, 23, 34, 56)
|
expect(dummy_time(Time.utc(2000, 1, 1, 12, 34, 56))).to eq(Time.zone.local(2000, 1, 1, 23, 34, 56))
|
||||||
end
|
end
|
||||||
|
|
||||||
it 'should return base dummy time value for Date value' do
|
it 'should return base dummy time value for Date value' do
|
||||||
dummy_time(Date.new(2010, 11, 22)).should == Time.utc(2000, 1, 1, 0, 0, 0)
|
expect(dummy_time(Date.new(2010, 11, 22))).to eq(Time.utc(2000, 1, 1, 0, 0, 0))
|
||||||
end
|
end
|
||||||
|
|
||||||
describe "with custom dummy date" do
|
describe "with custom dummy date" do
|
||||||
it 'should return dummy time with custom dummy date' do
|
it 'should return dummy time with custom dummy date' do
|
||||||
with_config(:dummy_date_for_time_type, [2010, 1, 1] ) do
|
with_config(:dummy_date_for_time_type, [2010, 1, 1] ) do
|
||||||
dummy_time(Time.utc(1999, 11, 22, 12, 34, 56)).should == Time.utc(2010, 1, 1, 12, 34, 56)
|
expect(dummy_time(Time.utc(1999, 11, 22, 12, 34, 56))).to eq(Time.utc(2010, 1, 1, 12, 34, 56))
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
@@ -127,56 +125,56 @@ describe ValidatesTimeliness::Conversion do
|
|||||||
|
|
||||||
it 'should return Date object as is' do
|
it 'should return Date object as is' do
|
||||||
value = Date.new(2010,1,1)
|
value = Date.new(2010,1,1)
|
||||||
evaluate_option_value(value, person).should == value
|
expect(evaluate_option_value(value, person)).to eq(value)
|
||||||
end
|
end
|
||||||
|
|
||||||
it 'should return Time object as is' do
|
it 'should return Time object as is' do
|
||||||
value = Time.mktime(2010,1,1)
|
value = Time.mktime(2010,1,1)
|
||||||
evaluate_option_value(value, person).should == value
|
expect(evaluate_option_value(value, person)).to eq(value)
|
||||||
end
|
end
|
||||||
|
|
||||||
it 'should return DateTime object as is' do
|
it 'should return DateTime object as is' do
|
||||||
value = DateTime.new(2010,1,1,0,0,0)
|
value = DateTime.new(2010,1,1,0,0,0)
|
||||||
evaluate_option_value(value, person).should == value
|
expect(evaluate_option_value(value, person)).to eq(value)
|
||||||
end
|
end
|
||||||
|
|
||||||
it 'should return Time value returned from proc with 0 arity' do
|
it 'should return Time value returned from proc with 0 arity' do
|
||||||
value = Time.mktime(2010,1,1)
|
value = Time.mktime(2010,1,1)
|
||||||
evaluate_option_value(lambda { value }, person).should == value
|
expect(evaluate_option_value(lambda { value }, person)).to eq(value)
|
||||||
end
|
end
|
||||||
|
|
||||||
it 'should return Time value returned by record attribute call in proc arity of 1' do
|
it 'should return Time value returned by record attribute call in proc arity of 1' do
|
||||||
value = Time.mktime(2010,1,1)
|
value = Time.mktime(2010,1,1)
|
||||||
person.birth_time = value
|
person.birth_time = value
|
||||||
evaluate_option_value(lambda {|r| r.birth_time }, person).should == value
|
expect(evaluate_option_value(lambda {|r| r.birth_time }, person)).to eq(value)
|
||||||
end
|
end
|
||||||
|
|
||||||
it 'should return Time value for attribute method symbol which returns Time' do
|
it 'should return Time value for attribute method symbol which returns Time' do
|
||||||
value = Time.mktime(2010,1,1)
|
value = Time.mktime(2010,1,1)
|
||||||
person.birth_time = value
|
person.birth_time = value
|
||||||
evaluate_option_value(:birth_time, person).should == value
|
expect(evaluate_option_value(:birth_time, person)).to eq(value)
|
||||||
end
|
end
|
||||||
|
|
||||||
it 'should return Time value is default zone from string time value' do
|
it 'should return Time value is default zone from string time value' do
|
||||||
value = '2010-01-01 12:00:00'
|
value = '2010-01-01 12:00:00'
|
||||||
evaluate_option_value(value, person).should == Time.utc(2010,1,1,12,0,0)
|
expect(evaluate_option_value(value, person)).to eq(Time.utc(2010,1,1,12,0,0))
|
||||||
end
|
end
|
||||||
|
|
||||||
it 'should return Time value is current zone from string time value if timezone aware' do
|
it 'should return Time value is current zone from string time value if timezone aware' do
|
||||||
@timezone_aware = true
|
@timezone_aware = true
|
||||||
value = '2010-01-01 12:00:00'
|
value = '2010-01-01 12:00:00'
|
||||||
evaluate_option_value(value, person).should == Time.zone.local(2010,1,1,12,0,0)
|
expect(evaluate_option_value(value, person)).to eq(Time.zone.local(2010,1,1,12,0,0))
|
||||||
end
|
end
|
||||||
|
|
||||||
it 'should return Time value in default zone from proc which returns string time' do
|
it 'should return Time value in default zone from proc which returns string time' do
|
||||||
value = '2010-01-01 12:00:00'
|
value = '2010-01-01 12:00:00'
|
||||||
evaluate_option_value(lambda { value }, person).should == Time.utc(2010,1,1,12,0,0)
|
expect(evaluate_option_value(lambda { value }, person)).to eq(Time.utc(2010,1,1,12,0,0))
|
||||||
end
|
end
|
||||||
|
|
||||||
it 'should return Time value for attribute method symbol which returns string time value' do
|
it 'should return Time value for attribute method symbol which returns string time value' do
|
||||||
value = '2010-01-01 12:00:00'
|
value = '2010-01-01 12:00:00'
|
||||||
person.birth_time = value
|
person.birth_time = value
|
||||||
evaluate_option_value(:birth_time, person).should == Time.utc(2010,1,1,12,0,0)
|
expect(evaluate_option_value(:birth_time, person)).to eq(Time.local(2010,1,1,12,0,0))
|
||||||
end
|
end
|
||||||
|
|
||||||
context "restriction shorthand" do
|
context "restriction shorthand" do
|
||||||
@@ -185,17 +183,17 @@ describe ValidatesTimeliness::Conversion do
|
|||||||
end
|
end
|
||||||
|
|
||||||
it 'should evaluate :now as current time' do
|
it 'should evaluate :now as current time' do
|
||||||
evaluate_option_value(:now, person).should == Time.now
|
expect(evaluate_option_value(:now, person)).to eq(Time.now)
|
||||||
end
|
end
|
||||||
|
|
||||||
it 'should evaluate :today as current time' do
|
it 'should evaluate :today as current time' do
|
||||||
evaluate_option_value(:today, person).should == Date.today
|
expect(evaluate_option_value(:today, person)).to eq(Date.today)
|
||||||
end
|
end
|
||||||
|
|
||||||
it 'should not use shorthand if symbol if is record method' do
|
it 'should not use shorthand if symbol if is record method' do
|
||||||
time = 1.day.from_now
|
time = 1.day.from_now
|
||||||
person.stub!(:now).and_return(time)
|
allow(person).to receive(:now).and_return(time)
|
||||||
evaluate_option_value(:now, person).should == time
|
expect(evaluate_option_value(:now, person)).to eq(time)
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
@@ -205,7 +203,7 @@ describe ValidatesTimeliness::Conversion do
|
|||||||
with_config(:use_plugin_parser, true)
|
with_config(:use_plugin_parser, true)
|
||||||
|
|
||||||
it 'should use timeliness' do
|
it 'should use timeliness' do
|
||||||
Timeliness::Parser.should_receive(:parse)
|
expect(Timeliness::Parser).to receive(:parse)
|
||||||
parse('2000-01-01')
|
parse('2000-01-01')
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
@@ -215,20 +213,20 @@ describe ValidatesTimeliness::Conversion do
|
|||||||
|
|
||||||
it 'should use Time.zone.parse attribute is timezone aware' do
|
it 'should use Time.zone.parse attribute is timezone aware' do
|
||||||
@timezone_aware = true
|
@timezone_aware = true
|
||||||
Time.zone.should_receive(:parse)
|
expect(Time.zone).to receive(:parse)
|
||||||
parse('2000-01-01')
|
parse('2000-01-01')
|
||||||
end
|
end
|
||||||
|
|
||||||
it 'should use value#to_time if use_plugin_parser setting is false and attribute is not timezone aware' do
|
it 'should use value#to_time if use_plugin_parser setting is false and attribute is not timezone aware' do
|
||||||
@timezone_aware = false
|
@timezone_aware = false
|
||||||
value = '2000-01-01'
|
value = '2000-01-01'
|
||||||
value.should_receive(:to_time)
|
expect(value).to receive(:to_time)
|
||||||
parse(value)
|
parse(value)
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
it 'should return nil if value is nil' do
|
it 'should return nil if value is nil' do
|
||||||
parse(nil).should be_nil
|
expect(parse(nil)).to be_nil
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|||||||
@@ -1,6 +1,4 @@
|
|||||||
require 'spec_helper'
|
RSpec.describe 'ValidatesTimeliness::Extensions::DateTimeSelect' do
|
||||||
|
|
||||||
describe ValidatesTimeliness::Extensions::DateTimeSelect do
|
|
||||||
include ActionView::Helpers::DateHelper
|
include ActionView::Helpers::DateHelper
|
||||||
attr_reader :person, :params
|
attr_reader :person, :params
|
||||||
|
|
||||||
@@ -111,7 +109,7 @@ describe ValidatesTimeliness::Extensions::DateTimeSelect do
|
|||||||
@output = date_select(:person, :birth_date, :include_blank => true, :discard_day => true)
|
@output = date_select(:person, :birth_date, :include_blank => true, :discard_day => true)
|
||||||
should_have_datetime_selected(:birth_date, :year => 2009, :month => 'February')
|
should_have_datetime_selected(:birth_date, :year => 2009, :month => 'February')
|
||||||
should_not_have_datetime_selected(:birth_time, :day)
|
should_not_have_datetime_selected(:birth_time, :day)
|
||||||
@output.should have_tag("input[id=person_birth_date_3i][type=hidden][value='1']")
|
expect(@output).to have_tag("input[id=person_birth_date_3i][type=hidden][value='1']")
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
@@ -150,14 +148,15 @@ describe ValidatesTimeliness::Extensions::DateTimeSelect do
|
|||||||
def should_have_datetime_selected(field, datetime_hash)
|
def should_have_datetime_selected(field, datetime_hash)
|
||||||
datetime_hash.each do |key, value|
|
datetime_hash.each do |key, value|
|
||||||
index = {:year => 1, :month => 2, :day => 3, :hour => 4, :min => 5, :sec => 6}[key]
|
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)
|
expect(@output).to have_tag("select[id=person_#{field}_#{index}i] option[selected=selected]", value.to_s)
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
def should_not_have_datetime_selected(field, *attributes)
|
def should_not_have_datetime_selected(field, *attributes)
|
||||||
attributes.each do |attribute|
|
attributes.each do |attribute|
|
||||||
index = {:year => 1, :month => 2, :day => 3, :hour => 4, :min => 5, :sec => 6}[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]")
|
expect(@output).not_to have_tag("select[id=person_#{attribute}_#{index}i] option[selected=selected]")
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
end
|
end
|
||||||
|
|||||||
@@ -1,38 +1,36 @@
|
|||||||
require 'spec_helper'
|
RSpec.describe 'ValidatesTimeliness::Extensions::MultiparameterHandler' do
|
||||||
|
|
||||||
describe ValidatesTimeliness::Extensions::MultiparameterHandler do
|
|
||||||
|
|
||||||
context "time column" do
|
context "time column" do
|
||||||
it 'should assign a string value for invalid date portion' do
|
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 = 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'
|
expect(employee.birth_datetime_before_type_cast).to eq '2000-02-31 12:00:00'
|
||||||
end
|
end
|
||||||
|
|
||||||
it 'should assign a Time value for valid datetimes' do
|
it 'should assign a Time value for valid datetimes' do
|
||||||
employee = record_with_multiparameter_attribute(:birth_datetime, [2000, 2, 28, 12, 0, 0])
|
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)
|
expect(employee.birth_datetime_before_type_cast).to eq Time.zone.local(2000, 2, 28, 12, 0, 0)
|
||||||
end
|
end
|
||||||
|
|
||||||
it 'should assign a string value for incomplete time' do
|
it 'should assign a string value for incomplete time' do
|
||||||
employee = record_with_multiparameter_attribute(:birth_datetime, [2000, nil, nil])
|
employee = record_with_multiparameter_attribute(:birth_datetime, [2000, nil, nil])
|
||||||
employee.birth_datetime_before_type_cast.should eq '2000-00-00'
|
expect(employee.birth_datetime_before_type_cast).to eq '2000-00-00'
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
context "date column" do
|
context "date column" do
|
||||||
it 'should assign a string value for invalid date' do
|
it 'should assign a string value for invalid date' do
|
||||||
employee = record_with_multiparameter_attribute(:birth_date, [2000, 2, 31])
|
employee = record_with_multiparameter_attribute(:birth_date, [2000, 2, 31])
|
||||||
employee.birth_date_before_type_cast.should eq '2000-02-31'
|
expect(employee.birth_date_before_type_cast).to eq '2000-02-31'
|
||||||
end
|
end
|
||||||
|
|
||||||
it 'should assign a Date value for valid date' do
|
it 'should assign a Date value for valid date' do
|
||||||
employee = record_with_multiparameter_attribute(:birth_date, [2000, 2, 28])
|
employee = record_with_multiparameter_attribute(:birth_date, [2000, 2, 28])
|
||||||
employee.birth_date_before_type_cast.should eq Date.new(2000, 2, 28)
|
expect(employee.birth_date_before_type_cast).to eq Date.new(2000, 2, 28)
|
||||||
end
|
end
|
||||||
|
|
||||||
it 'should assign a string value for incomplete date' do
|
it 'should assign a string value for incomplete date' do
|
||||||
employee = record_with_multiparameter_attribute(:birth_date, [2000, nil, nil])
|
employee = record_with_multiparameter_attribute(:birth_date, [2000, nil, nil])
|
||||||
employee.birth_date_before_type_cast.should eq '2000-00-00'
|
expect(employee.birth_date_before_type_cast).to eq '2000-00-00'
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
@@ -41,4 +39,5 @@ describe ValidatesTimeliness::Extensions::MultiparameterHandler do
|
|||||||
values.each_with_index {|value, index| hash["#{name}(#{index+1}i)"] = value.to_s }
|
values.each_with_index {|value, index| hash["#{name}(#{index+1}i)"] = value.to_s }
|
||||||
Employee.new(hash)
|
Employee.new(hash)
|
||||||
end
|
end
|
||||||
|
|
||||||
end
|
end
|
||||||
|
|||||||
@@ -1,30 +1,28 @@
|
|||||||
require 'spec_helper'
|
RSpec.describe ValidatesTimeliness, 'HelperMethods' do
|
||||||
|
|
||||||
describe ValidatesTimeliness, 'HelperMethods' do
|
|
||||||
let(:record) { Person.new }
|
let(:record) { Person.new }
|
||||||
|
|
||||||
it 'should define class validation methods' do
|
it 'should define class validation methods' do
|
||||||
Person.should respond_to(:validates_date)
|
expect(Person).to respond_to(:validates_date)
|
||||||
Person.should respond_to(:validates_time)
|
expect(Person).to respond_to(:validates_time)
|
||||||
Person.should respond_to(:validates_datetime)
|
expect(Person).to respond_to(:validates_datetime)
|
||||||
end
|
end
|
||||||
|
|
||||||
it 'should define instance validation methods' do
|
it 'should define instance validation methods' do
|
||||||
record.should respond_to(:validates_date)
|
expect(record).to respond_to(:validates_date)
|
||||||
record.should respond_to(:validates_time)
|
expect(record).to respond_to(:validates_time)
|
||||||
record.should respond_to(:validates_datetime)
|
expect(record).to respond_to(:validates_datetime)
|
||||||
end
|
end
|
||||||
|
|
||||||
it 'should validate instance using class validation defined' do
|
it 'should validate instance using class validation defined' do
|
||||||
Person.validates_date :birth_date
|
Person.validates_date :birth_date
|
||||||
record.valid?
|
record.valid?
|
||||||
|
|
||||||
record.errors[:birth_date].should_not be_empty
|
expect(record.errors[:birth_date]).not_to be_empty
|
||||||
end
|
end
|
||||||
|
|
||||||
it 'should validate instance using instance valiation method' do
|
it 'should validate instance using instance valiation method' do
|
||||||
record.validates_date :birth_date
|
record.validates_date :birth_date
|
||||||
|
|
||||||
record.errors[:birth_date].should_not be_empty
|
expect(record.errors[:birth_date]).not_to be_empty
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|||||||
@@ -1,46 +1,44 @@
|
|||||||
require 'spec_helper'
|
RSpec.describe ValidatesTimeliness, 'ActiveRecord' do
|
||||||
|
|
||||||
describe ValidatesTimeliness, 'ActiveRecord' do
|
|
||||||
|
|
||||||
context "validation methods" do
|
context "validation methods" do
|
||||||
let(:record) { Employee.new }
|
let(:record) { Employee.new }
|
||||||
|
|
||||||
it 'should be defined for the class' do
|
it 'should be defined for the class' do
|
||||||
ActiveRecord::Base.should respond_to(:validates_date)
|
expect(ActiveRecord::Base).to respond_to(:validates_date)
|
||||||
ActiveRecord::Base.should respond_to(:validates_time)
|
expect(ActiveRecord::Base).to respond_to(:validates_time)
|
||||||
ActiveRecord::Base.should respond_to(:validates_datetime)
|
expect(ActiveRecord::Base).to respond_to(:validates_datetime)
|
||||||
end
|
end
|
||||||
|
|
||||||
it 'should defines for the instance' do
|
it 'should defines for the instance' do
|
||||||
record.should respond_to(:validates_date)
|
expect(record).to respond_to(:validates_date)
|
||||||
record.should respond_to(:validates_time)
|
expect(record).to respond_to(:validates_time)
|
||||||
record.should respond_to(:validates_datetime)
|
expect(record).to respond_to(:validates_datetime)
|
||||||
end
|
end
|
||||||
|
|
||||||
it "should validate a valid value string" do
|
it "should validate a valid value string" do
|
||||||
record.birth_date = '2012-01-01'
|
record.birth_date = '2012-01-01'
|
||||||
|
|
||||||
record.valid?
|
record.valid?
|
||||||
record.errors[:birth_date].should be_empty
|
expect(record.errors[:birth_date]).to be_empty
|
||||||
end
|
end
|
||||||
|
|
||||||
it "should validate a invalid value string" do
|
it "should validate a invalid value string" do
|
||||||
record.birth_date = 'not a date'
|
record.birth_date = 'not a date'
|
||||||
|
|
||||||
record.valid?
|
record.valid?
|
||||||
record.errors[:birth_date].should_not be_empty
|
expect(record.errors[:birth_date]).not_to be_empty
|
||||||
end
|
end
|
||||||
|
|
||||||
it "should validate a nil value" do
|
it "should validate a nil value" do
|
||||||
record.birth_date = nil
|
record.birth_date = nil
|
||||||
|
|
||||||
record.valid?
|
record.valid?
|
||||||
record.errors[:birth_date].should be_empty
|
expect(record.errors[:birth_date]).to be_empty
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
it 'should determine type for attribute' do
|
it 'should determine type for attribute' do
|
||||||
Employee.timeliness_attribute_type(:birth_date).should eq :date
|
expect(Employee.timeliness_attribute_type(:birth_date)).to eq :date
|
||||||
end
|
end
|
||||||
|
|
||||||
context 'attribute timezone awareness' do
|
context 'attribute timezone awareness' do
|
||||||
@@ -58,17 +56,17 @@ describe ValidatesTimeliness, 'ActiveRecord' do
|
|||||||
|
|
||||||
context 'for column attribute' do
|
context 'for column attribute' do
|
||||||
it 'should be detected from column type' do
|
it 'should be detected from column type' do
|
||||||
klass.timeliness_attribute_timezone_aware?(:birth_date).should be_false
|
expect(klass.timeliness_attribute_timezone_aware?(:birth_date)).to be_falsey
|
||||||
klass.timeliness_attribute_timezone_aware?(:birth_time).should be_false
|
expect(klass.timeliness_attribute_timezone_aware?(:birth_time)).to be_falsey
|
||||||
klass.timeliness_attribute_timezone_aware?(:birth_datetime).should be_true
|
expect(klass.timeliness_attribute_timezone_aware?(:birth_datetime)).to be_truthy
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
context 'for non-column attribute' do
|
context 'for non-column attribute' do
|
||||||
it 'should be detected from the validation type' do
|
it 'should be detected from the validation type' do
|
||||||
klass.timeliness_attribute_timezone_aware?(:some_date).should be_false
|
expect(klass.timeliness_attribute_timezone_aware?(:some_date)).to be_falsey
|
||||||
klass.timeliness_attribute_timezone_aware?(:some_time).should be_false
|
expect(klass.timeliness_attribute_timezone_aware?(:some_time)).to be_falsey
|
||||||
klass.timeliness_attribute_timezone_aware?(:some_datetime).should be_true
|
expect(klass.timeliness_attribute_timezone_aware?(:some_datetime)).to be_truthy
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
@@ -88,7 +86,7 @@ describe ValidatesTimeliness, 'ActiveRecord' do
|
|||||||
it 'should store raw value' do
|
it 'should store raw value' do
|
||||||
record.birth_datetime = datetime_string = '2010-01-01 12:30'
|
record.birth_datetime = datetime_string = '2010-01-01 12:30'
|
||||||
|
|
||||||
record._timeliness_raw_value_for('birth_datetime').should eq datetime_string
|
expect(record.read_timeliness_attribute_before_type_cast('birth_datetime')).to eq datetime_string
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
@@ -96,7 +94,7 @@ describe ValidatesTimeliness, 'ActiveRecord' do
|
|||||||
it 'should store raw value' do
|
it 'should store raw value' do
|
||||||
record.birth_date = date_string = '2010-01-01'
|
record.birth_date = date_string = '2010-01-01'
|
||||||
|
|
||||||
record._timeliness_raw_value_for('birth_date').should eq date_string
|
expect(record.read_timeliness_attribute_before_type_cast('birth_date')).to eq date_string
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
@@ -104,7 +102,7 @@ describe ValidatesTimeliness, 'ActiveRecord' do
|
|||||||
it 'should store raw value' do
|
it 'should store raw value' do
|
||||||
record.birth_time = time_string = '12:12'
|
record.birth_time = time_string = '12:12'
|
||||||
|
|
||||||
record._timeliness_raw_value_for('birth_time').should eq time_string
|
expect(record.read_timeliness_attribute_before_type_cast('birth_time')).to eq time_string
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
@@ -122,13 +120,13 @@ describe ValidatesTimeliness, 'ActiveRecord' do
|
|||||||
|
|
||||||
context "for a date column" do
|
context "for a date column" do
|
||||||
it 'should parse a string value' do
|
it 'should parse a string value' do
|
||||||
Timeliness::Parser.should_receive(:parse)
|
expect(Timeliness::Parser).to receive(:parse)
|
||||||
|
|
||||||
record.birth_date = '2010-01-01'
|
record.birth_date = '2010-01-01'
|
||||||
end
|
end
|
||||||
|
|
||||||
it 'should parse a invalid string value as nil' do
|
it 'should parse a invalid string value as nil' do
|
||||||
Timeliness::Parser.should_receive(:parse)
|
expect(Timeliness::Parser).to receive(:parse)
|
||||||
|
|
||||||
record.birth_date = 'not valid'
|
record.birth_date = 'not valid'
|
||||||
end
|
end
|
||||||
@@ -136,20 +134,20 @@ describe ValidatesTimeliness, 'ActiveRecord' do
|
|||||||
it 'should store a Date value after parsing string' do
|
it 'should store a Date value after parsing string' do
|
||||||
record.birth_date = '2010-01-01'
|
record.birth_date = '2010-01-01'
|
||||||
|
|
||||||
record.birth_date.should be_kind_of(Date)
|
expect(record.birth_date).to be_kind_of(Date)
|
||||||
record.birth_date.should eq Date.new(2010, 1, 1)
|
expect(record.birth_date).to eq Date.new(2010, 1, 1)
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
context "for a time column" do
|
context "for a time column" do
|
||||||
it 'should parse a string value' do
|
it 'should parse a string value' do
|
||||||
Timeliness::Parser.should_receive(:parse)
|
expect(Timeliness::Parser).to receive(:parse)
|
||||||
|
|
||||||
record.birth_time = '12:30'
|
record.birth_time = '12:30'
|
||||||
end
|
end
|
||||||
|
|
||||||
it 'should parse a invalid string value as nil' do
|
it 'should parse a invalid string value as nil' do
|
||||||
Timeliness::Parser.should_receive(:parse)
|
expect(Timeliness::Parser).to receive(:parse)
|
||||||
|
|
||||||
record.birth_time = 'not valid'
|
record.birth_time = 'not valid'
|
||||||
end
|
end
|
||||||
@@ -157,8 +155,8 @@ describe ValidatesTimeliness, 'ActiveRecord' do
|
|||||||
it 'should store a Time value after parsing string' do
|
it 'should store a Time value after parsing string' do
|
||||||
record.birth_time = '12:30'
|
record.birth_time = '12:30'
|
||||||
|
|
||||||
record.birth_time.should be_kind_of(Time)
|
expect(record.birth_time).to be_kind_of(Time)
|
||||||
record.birth_time.should eq Time.utc(2000, 1, 1, 12, 30)
|
expect(record.birth_time).to eq Time.utc(2000, 1, 1, 12, 30)
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
@@ -166,13 +164,13 @@ describe ValidatesTimeliness, 'ActiveRecord' do
|
|||||||
with_config(:default_timezone, 'Australia/Melbourne')
|
with_config(:default_timezone, 'Australia/Melbourne')
|
||||||
|
|
||||||
it 'should parse a string value' do
|
it 'should parse a string value' do
|
||||||
Timeliness::Parser.should_receive(:parse)
|
expect(Timeliness::Parser).to receive(:parse)
|
||||||
|
|
||||||
record.birth_datetime = '2010-01-01 12:00'
|
record.birth_datetime = '2010-01-01 12:00'
|
||||||
end
|
end
|
||||||
|
|
||||||
it 'should parse a invalid string value as nil' do
|
it 'should parse a invalid string value as nil' do
|
||||||
Timeliness::Parser.should_receive(:parse)
|
expect(Timeliness::Parser).to receive(:parse)
|
||||||
|
|
||||||
record.birth_datetime = 'not valid'
|
record.birth_datetime = 'not valid'
|
||||||
end
|
end
|
||||||
@@ -180,13 +178,13 @@ describe ValidatesTimeliness, 'ActiveRecord' do
|
|||||||
it 'should parse string into Time value' do
|
it 'should parse string into Time value' do
|
||||||
record.birth_datetime = '2010-01-01 12:00'
|
record.birth_datetime = '2010-01-01 12:00'
|
||||||
|
|
||||||
record.birth_datetime.should be_kind_of(Time)
|
expect(record.birth_datetime).to be_kind_of(Time)
|
||||||
end
|
end
|
||||||
|
|
||||||
it 'should parse string as current timezone' do
|
it 'should parse string as current timezone' do
|
||||||
record.birth_datetime = '2010-06-01 12:00'
|
record.birth_datetime = '2010-06-01 12:00'
|
||||||
|
|
||||||
record.birth_datetime.utc_offset.should eq Time.zone.utc_offset
|
expect(record.birth_datetime.utc_offset).to eq Time.zone.utc_offset
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
@@ -199,7 +197,7 @@ describe ValidatesTimeliness, 'ActiveRecord' do
|
|||||||
|
|
||||||
record.reload
|
record.reload
|
||||||
|
|
||||||
record._timeliness_raw_value_for('birth_date').should be_nil
|
expect(record.read_timeliness_attribute_before_type_cast('birth_date')).to be_nil
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
@@ -207,13 +205,13 @@ describe ValidatesTimeliness, 'ActiveRecord' do
|
|||||||
let(:record) { Employee.new }
|
let(:record) { Employee.new }
|
||||||
|
|
||||||
it 'should be defined on class if ORM supports it' do
|
it 'should be defined on class if ORM supports it' do
|
||||||
record.should respond_to(:birth_datetime_before_type_cast)
|
expect(record).to respond_to(:birth_datetime_before_type_cast)
|
||||||
end
|
end
|
||||||
|
|
||||||
it 'should return original value' do
|
it 'should return original value' do
|
||||||
record.birth_datetime = date_string = '2010-01-01'
|
record.birth_datetime = date_string = '2010-01-01'
|
||||||
|
|
||||||
record.birth_datetime_before_type_cast.should eq date_string
|
expect(record.birth_datetime_before_type_cast).to eq date_string
|
||||||
end
|
end
|
||||||
|
|
||||||
it 'should return attribute if no attribute assignment has been made' do
|
it 'should return attribute if no attribute assignment has been made' do
|
||||||
@@ -221,7 +219,7 @@ describe ValidatesTimeliness, 'ActiveRecord' do
|
|||||||
Employee.create(:birth_datetime => datetime)
|
Employee.create(:birth_datetime => datetime)
|
||||||
|
|
||||||
record = Employee.last
|
record = Employee.last
|
||||||
record.birth_datetime_before_type_cast.should match(/#{datetime.utc.to_s[0...-4]}/)
|
expect(record.birth_datetime_before_type_cast).to match(/#{datetime.utc.to_s[0...-4]}/)
|
||||||
end
|
end
|
||||||
|
|
||||||
context "with plugin parser" do
|
context "with plugin parser" do
|
||||||
@@ -230,7 +228,7 @@ describe ValidatesTimeliness, 'ActiveRecord' do
|
|||||||
it 'should return original value' do
|
it 'should return original value' do
|
||||||
record.birth_datetime = date_string = '2010-01-31'
|
record.birth_datetime = date_string = '2010-01-31'
|
||||||
|
|
||||||
record.birth_datetime_before_type_cast.should eq date_string
|
expect(record.birth_datetime_before_type_cast).to eq date_string
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
@@ -238,7 +236,19 @@ describe ValidatesTimeliness, 'ActiveRecord' do
|
|||||||
|
|
||||||
context "define_attribute_methods" do
|
context "define_attribute_methods" do
|
||||||
it "returns a falsy value if the attribute methods have already been generated" do
|
it "returns a falsy value if the attribute methods have already been generated" do
|
||||||
Employee.define_attribute_methods.should be_false
|
expect(Employee.define_attribute_methods).to be_falsey
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
context "undefine_attribute_methods" do
|
||||||
|
it "returns remove attribute methods that have already been generated" do
|
||||||
|
Employee.define_attribute_methods
|
||||||
|
|
||||||
|
expect(Employee.instance_methods).to include(:birth_datetime)
|
||||||
|
|
||||||
|
Employee.undefine_attribute_methods
|
||||||
|
|
||||||
|
expect(Employee.instance_methods).to_not include(:birth_datetime)
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|||||||
@@ -1,223 +0,0 @@
|
|||||||
require 'spec_helper'
|
|
||||||
|
|
||||||
# Try loading mongoid and connecting. Otherwise, abort and skip spec.
|
|
||||||
begin
|
|
||||||
|
|
||||||
require 'mongoid'
|
|
||||||
require 'validates_timeliness/orm/mongoid'
|
|
||||||
|
|
||||||
Mongoid.configure do |config|
|
|
||||||
config.connect_to('validates_timeliness_test')
|
|
||||||
end
|
|
||||||
|
|
||||||
describe ValidatesTimeliness, 'Mongoid' do
|
|
||||||
after(:each) do
|
|
||||||
Mongoid.purge!
|
|
||||||
end
|
|
||||||
|
|
||||||
class Article
|
|
||||||
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
|
|
||||||
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)
|
|
||||||
Article.should respond_to(:validates_datetime)
|
|
||||||
end
|
|
||||||
|
|
||||||
it 'should be defined on the instance' do
|
|
||||||
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
|
|
||||||
record.publish_datetime = date_string = '2010-01-01'
|
|
||||||
|
|
||||||
record._timeliness_raw_value_for('publish_datetime').should == date_string
|
|
||||||
end
|
|
||||||
|
|
||||||
context "with plugin parser" do
|
|
||||||
let(:record) { ArticleWithParser.new }
|
|
||||||
|
|
||||||
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 parse a string value' do
|
|
||||||
Timeliness::Parser.should_receive(:parse)
|
|
||||||
|
|
||||||
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
|
|
||||||
with_config(:default_timezone, 'Australia/Melbourne')
|
|
||||||
|
|
||||||
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
|
|
||||||
end
|
|
||||||
|
|
||||||
context "cached value" do
|
|
||||||
it 'should be cleared on reload' do
|
|
||||||
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
|
|
||||||
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 => e
|
|
||||||
puts "Mongoid specs skipped. MongoDB connection failed with error: #{e.message}"
|
|
||||||
end
|
|
||||||
22
spec/validates_timeliness/railtie_spec.rb
Normal file
22
spec/validates_timeliness/railtie_spec.rb
Normal file
@@ -0,0 +1,22 @@
|
|||||||
|
require 'validates_timeliness/railtie'
|
||||||
|
|
||||||
|
RSpec.describe ValidatesTimeliness::Railtie do
|
||||||
|
context "intializers" do
|
||||||
|
context "validates_timeliness.initialize_timeliness_ambiguous_date_format" do
|
||||||
|
it 'should set the timeliness default ambiguous date format from the current format' do
|
||||||
|
expect(Timeliness.configuration.ambiguous_date_format).to eq :us
|
||||||
|
ValidatesTimeliness.parser.use_euro_formats
|
||||||
|
|
||||||
|
initializer("validates_timeliness.initialize_timeliness_ambiguous_date_format").run
|
||||||
|
|
||||||
|
expect(Timeliness.configuration.ambiguous_date_format).to eq :euro
|
||||||
|
end
|
||||||
|
end if Timeliness.respond_to?(:ambiguous_date_format)
|
||||||
|
|
||||||
|
def initializer(name)
|
||||||
|
ValidatesTimeliness::Railtie.initializers.find { |i|
|
||||||
|
i.name == name
|
||||||
|
} || raise("Initializer #{name} not found")
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
@@ -1,6 +1,4 @@
|
|||||||
require 'spec_helper'
|
RSpec.describe ValidatesTimeliness::Validator, ":after option" do
|
||||||
|
|
||||||
describe ValidatesTimeliness::Validator, ":after option" do
|
|
||||||
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)
|
||||||
@@ -25,15 +23,15 @@ describe ValidatesTimeliness::Validator, ":after option" do
|
|||||||
end
|
end
|
||||||
|
|
||||||
it "should not be valid for same time as restriction" do
|
it "should not be valid for same time as restriction" do
|
||||||
invalid!(:birth_time, Time.local_time(2000, 1, 1, 12, 0, 0), 'must be after 12:00:00')
|
invalid!(:birth_time, Time.local(2000, 1, 1, 12, 0, 0), 'must be after 12:00:00')
|
||||||
end
|
end
|
||||||
|
|
||||||
it "should not be valid for time before restriction" do
|
it "should not be valid for time before restriction" do
|
||||||
invalid!(:birth_time, Time.local_time(2000, 1, 1, 11, 59, 59), 'must be after 12:00:00')
|
invalid!(:birth_time, Time.local(2000, 1, 1, 11, 59, 59), 'must be after 12:00:00')
|
||||||
end
|
end
|
||||||
|
|
||||||
it "should be valid for time after restriction" do
|
it "should be valid for time after restriction" do
|
||||||
valid!(:birth_time, Time.local_time(2000, 1, 1, 12, 00, 01))
|
valid!(:birth_time, Time.local(2000, 1, 1, 12, 00, 01))
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
|
|||||||
@@ -1,6 +1,4 @@
|
|||||||
require 'spec_helper'
|
RSpec.describe ValidatesTimeliness::Validator, ":before option" do
|
||||||
|
|
||||||
describe ValidatesTimeliness::Validator, ":before option" do
|
|
||||||
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)
|
||||||
@@ -25,15 +23,15 @@ describe ValidatesTimeliness::Validator, ":before option" do
|
|||||||
end
|
end
|
||||||
|
|
||||||
it "should not be valid for time after restriction" do
|
it "should not be valid for time after restriction" do
|
||||||
invalid!(:birth_time, Time.local_time(2000, 1, 1, 12, 00, 01), 'must be before 12:00:00')
|
invalid!(:birth_time, Time.local(2000, 1, 1, 12, 00, 01), 'must be before 12:00:00')
|
||||||
end
|
end
|
||||||
|
|
||||||
it "should not be valid for same time as restriction" do
|
it "should not be valid for same time as restriction" do
|
||||||
invalid!(:birth_time, Time.local_time(2000, 1, 1, 12, 0, 0), 'must be before 12:00:00')
|
invalid!(:birth_time, Time.local(2000, 1, 1, 12, 0, 0), 'must be before 12:00:00')
|
||||||
end
|
end
|
||||||
|
|
||||||
it "should be valid for time before restriction" do
|
it "should be valid for time before restriction" do
|
||||||
valid!(:birth_time, Time.local_time(2000, 1, 1, 11, 59, 59))
|
valid!(:birth_time, Time.local(2000, 1, 1, 11, 59, 59))
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
|
|||||||
@@ -1,8 +1,6 @@
|
|||||||
require 'spec_helper'
|
RSpec.describe ValidatesTimeliness::Validator, ":is_at option" do
|
||||||
|
|
||||||
describe ValidatesTimeliness::Validator, ":is_at option" do
|
|
||||||
before do
|
before do
|
||||||
Timecop.freeze(Time.local_time(2010, 1, 1, 0, 0, 0))
|
Timecop.freeze(Time.local(2010, 1, 1, 0, 0, 0))
|
||||||
end
|
end
|
||||||
|
|
||||||
describe "for date type" do
|
describe "for date type" do
|
||||||
@@ -29,15 +27,15 @@ describe ValidatesTimeliness::Validator, ":is_at option" do
|
|||||||
end
|
end
|
||||||
|
|
||||||
it "should not be valid for time before restriction" do
|
it "should not be valid for time before restriction" do
|
||||||
invalid!(:birth_time, Time.local_time(2000, 1, 1, 11, 59, 59), 'must be at 12:00:00')
|
invalid!(:birth_time, Time.local(2000, 1, 1, 11, 59, 59), 'must be at 12:00:00')
|
||||||
end
|
end
|
||||||
|
|
||||||
it "should not be valid for time after restriction" do
|
it "should not be valid for time after restriction" do
|
||||||
invalid!(:birth_time, Time.local_time(2000, 1, 1, 12, 00, 01), 'must be at 12:00:00')
|
invalid!(:birth_time, Time.local(2000, 1, 1, 12, 00, 01), 'must be at 12:00:00')
|
||||||
end
|
end
|
||||||
|
|
||||||
it "should be valid for same time as restriction" do
|
it "should be valid for same time as restriction" do
|
||||||
valid!(:birth_time, Time.local_time(2000, 1, 1, 12, 0, 0))
|
valid!(:birth_time, Time.local(2000, 1, 1, 12, 0, 0))
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
|
|||||||
@@ -1,6 +1,4 @@
|
|||||||
require 'spec_helper'
|
RSpec.describe ValidatesTimeliness::Validator, ":on_or_after option" do
|
||||||
|
|
||||||
describe ValidatesTimeliness::Validator, ":on_or_after option" do
|
|
||||||
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)
|
||||||
@@ -25,15 +23,15 @@ describe ValidatesTimeliness::Validator, ":on_or_after option" do
|
|||||||
end
|
end
|
||||||
|
|
||||||
it "should not be valid for time before restriction" do
|
it "should not be valid for time before restriction" do
|
||||||
invalid!(:birth_time, Time.local_time(2000, 1, 1, 11, 59, 59), 'must be on or after 12:00:00')
|
invalid!(:birth_time, Time.local(2000, 1, 1, 11, 59, 59), 'must be on or after 12:00:00')
|
||||||
end
|
end
|
||||||
|
|
||||||
it "should be valid for time after restriction" do
|
it "should be valid for time after restriction" do
|
||||||
valid!(:birth_time, Time.local_time(2000, 1, 1, 12, 00, 01))
|
valid!(:birth_time, Time.local(2000, 1, 1, 12, 00, 01))
|
||||||
end
|
end
|
||||||
|
|
||||||
it "should be valid for same time as restriction" do
|
it "should be valid for same time as restriction" do
|
||||||
valid!(:birth_time, Time.local_time(2000, 1, 1, 12, 0, 0))
|
valid!(:birth_time, Time.local(2000, 1, 1, 12, 0, 0))
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
|
|||||||
@@ -1,6 +1,4 @@
|
|||||||
require 'spec_helper'
|
RSpec.describe ValidatesTimeliness::Validator, ":on_or_before option" do
|
||||||
|
|
||||||
describe ValidatesTimeliness::Validator, ":on_or_before option" do
|
|
||||||
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)
|
||||||
@@ -25,15 +23,15 @@ describe ValidatesTimeliness::Validator, ":on_or_before option" do
|
|||||||
end
|
end
|
||||||
|
|
||||||
it "should not be valid for time after restriction" do
|
it "should not be valid for time after restriction" do
|
||||||
invalid!(:birth_time, Time.local_time(2000, 1, 1, 12, 00, 01), 'must be on or before 12:00:00')
|
invalid!(:birth_time, Time.local(2000, 1, 1, 12, 00, 01), 'must be on or before 12:00:00')
|
||||||
end
|
end
|
||||||
|
|
||||||
it "should be valid for time before restriction" do
|
it "should be valid for time before restriction" do
|
||||||
valid!(:birth_time, Time.local_time(2000, 1, 1, 11, 59, 59))
|
valid!(:birth_time, Time.local(2000, 1, 1, 11, 59, 59))
|
||||||
end
|
end
|
||||||
|
|
||||||
it "should be valid for same time as restriction" do
|
it "should be valid for same time as restriction" do
|
||||||
valid!(:birth_time, Time.local_time(2000, 1, 1, 12, 0, 0))
|
valid!(:birth_time, Time.local(2000, 1, 1, 12, 0, 0))
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
|
|||||||
@@ -1,29 +1,27 @@
|
|||||||
require 'spec_helper'
|
RSpec.describe ValidatesTimeliness::Validator do
|
||||||
|
|
||||||
describe ValidatesTimeliness::Validator do
|
|
||||||
before do
|
before do
|
||||||
Timecop.freeze(Time.local_time(2010, 1, 1, 0, 0, 0))
|
Timecop.freeze(Time.local(2010, 1, 1, 0, 0, 0))
|
||||||
end
|
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}
|
||||||
Person.validators.should have(1).kind_of(ActiveModel::Validations::TimelinessValidator)
|
expect(Person.validators.select { |v| v.is_a?(ActiveModel::Validations::TimelinessValidator) }.size).to eq(1)
|
||||||
invalid!(:birth_date, Date.new(2010,1,2))
|
invalid!(:birth_date, Date.new(2010,1,2))
|
||||||
valid!(:birth_date, Date.new(2010,1,1))
|
valid!(:birth_date, Date.new(2010,1,1))
|
||||||
end
|
end
|
||||||
|
|
||||||
it 'should use default to :datetime type' do
|
it 'should use default to :datetime type' do
|
||||||
Person.validates :birth_datetime, :timeliness => {:is_at => Time.mktime(2010,1,1)}
|
Person.validates :birth_datetime, :timeliness => {:is_at => Time.mktime(2010,1,1)}
|
||||||
Person.validators.first.type.should == :datetime
|
expect(Person.validators.first.type).to eq(:datetime)
|
||||||
end
|
end
|
||||||
|
|
||||||
it 'should add attribute to timeliness attributes set' do
|
it 'should add attribute to timeliness attributes set' do
|
||||||
PersonWithShim.timeliness_validated_attributes.should_not include(:birth_time)
|
expect(PersonWithShim.timeliness_validated_attributes).not_to include(:birth_time)
|
||||||
|
|
||||||
PersonWithShim.validates :birth_time, :timeliness => {:is_at => "12:30"}
|
PersonWithShim.validates :birth_time, :timeliness => {:is_at => "12:30"}
|
||||||
|
|
||||||
PersonWithShim.timeliness_validated_attributes.should include(:birth_time)
|
expect(PersonWithShim.timeliness_validated_attributes).to include(:birth_time)
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
@@ -35,10 +33,10 @@ describe ValidatesTimeliness::Validator do
|
|||||||
it 'should not be valid attribute is type cast to nil but raw value is non-nil invalid value' do
|
it 'should not be valid attribute is type cast to nil but raw value is non-nil invalid value' do
|
||||||
Person.validates_date :birth_date, :allow_nil => true
|
Person.validates_date :birth_date, :allow_nil => true
|
||||||
record = Person.new
|
record = Person.new
|
||||||
record.stub!(:birth_date).and_return(nil)
|
allow(record).to receive(:birth_date).and_return(nil)
|
||||||
record.stub!(:_timeliness_raw_value_for).and_return("Not a date")
|
allow(record).to receive(:read_timeliness_attribute_before_type_cast).and_return("Not a date")
|
||||||
record.should_not be_valid
|
expect(record).not_to be_valid
|
||||||
record.errors[:birth_date].first.should == 'is not a valid date'
|
expect(record.errors[:birth_date].first).to eq('is not a valid date')
|
||||||
end
|
end
|
||||||
|
|
||||||
describe ":allow_nil option" do
|
describe ":allow_nil option" do
|
||||||
@@ -60,7 +58,7 @@ describe ValidatesTimeliness::Validator do
|
|||||||
p = PersonWithShim.new
|
p = PersonWithShim.new
|
||||||
p.birth_date = 'bogus'
|
p.birth_date = 'bogus'
|
||||||
|
|
||||||
p.should_not be_valid
|
expect(p).not_to be_valid
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
@@ -84,18 +82,32 @@ describe ValidatesTimeliness::Validator do
|
|||||||
p = PersonWithShim.new
|
p = PersonWithShim.new
|
||||||
p.birth_date = 'bogus'
|
p.birth_date = 'bogus'
|
||||||
|
|
||||||
p.should_not be_valid
|
expect(p).not_to be_valid
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
|
describe ':message options' do
|
||||||
|
it 'should allow message option too' do
|
||||||
|
Person.validates_date :birth_date, on_or_after: :today, message: 'cannot be in past'
|
||||||
|
invalid!(:birth_date, Date.today - 5.days, 'cannot be in past')
|
||||||
|
valid!(:birth_date, Date.today)
|
||||||
|
end
|
||||||
|
|
||||||
|
it 'should first allow the defined message' do
|
||||||
|
Person.validates_date :birth_date, on_or_after: :today, on_or_after_message: 'cannot be in past', message: 'dummy message'
|
||||||
|
invalid!(:birth_date, Date.today - 5.days, 'cannot be in past')
|
||||||
|
valid!(:birth_date, Date.today)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
describe ":between option" do
|
describe ":between option" do
|
||||||
describe "array value" do
|
describe "array value" do
|
||||||
it 'should be split option into :on_or_after and :on_or_before values' do
|
it 'should be split option into :on_or_after and :on_or_before values' do
|
||||||
on_or_after, on_or_before = Date.new(2010,1,1), Date.new(2010,1,2)
|
on_or_after, on_or_before = Date.new(2010,1,1), Date.new(2010,1,2)
|
||||||
Person.validates_date :birth_date, :between => [on_or_after, on_or_before]
|
Person.validates_date :birth_date, :between => [on_or_after, on_or_before]
|
||||||
Person.validators.first.options[:on_or_after].should == on_or_after
|
expect(Person.validators.first.options[:on_or_after]).to eq(on_or_after)
|
||||||
Person.validators.first.options[:on_or_before].should == on_or_before
|
expect(Person.validators.first.options[:on_or_before]).to eq(on_or_before)
|
||||||
invalid!(:birth_date, on_or_after - 1, "must be on or after 2010-01-01")
|
invalid!(:birth_date, on_or_after - 1, "must be on or after 2010-01-01")
|
||||||
invalid!(:birth_date, on_or_before + 1, "must be on or before 2010-01-02")
|
invalid!(:birth_date, on_or_before + 1, "must be on or before 2010-01-02")
|
||||||
valid!(:birth_date, on_or_after)
|
valid!(:birth_date, on_or_after)
|
||||||
@@ -107,8 +119,8 @@ describe ValidatesTimeliness::Validator do
|
|||||||
it 'should be split option into :on_or_after and :on_or_before values' do
|
it 'should be split option into :on_or_after and :on_or_before values' do
|
||||||
on_or_after, on_or_before = Date.new(2010,1,1), Date.new(2010,1,2)
|
on_or_after, on_or_before = Date.new(2010,1,1), Date.new(2010,1,2)
|
||||||
Person.validates_date :birth_date, :between => on_or_after..on_or_before
|
Person.validates_date :birth_date, :between => on_or_after..on_or_before
|
||||||
Person.validators.first.options[:on_or_after].should == on_or_after
|
expect(Person.validators.first.options[:on_or_after]).to eq(on_or_after)
|
||||||
Person.validators.first.options[:on_or_before].should == on_or_before
|
expect(Person.validators.first.options[:on_or_before]).to eq(on_or_before)
|
||||||
invalid!(:birth_date, on_or_after - 1, "must be on or after 2010-01-01")
|
invalid!(:birth_date, on_or_after - 1, "must be on or after 2010-01-01")
|
||||||
invalid!(:birth_date, on_or_before + 1, "must be on or before 2010-01-02")
|
invalid!(:birth_date, on_or_before + 1, "must be on or before 2010-01-02")
|
||||||
valid!(:birth_date, on_or_after)
|
valid!(:birth_date, on_or_after)
|
||||||
@@ -120,8 +132,8 @@ describe ValidatesTimeliness::Validator do
|
|||||||
it 'should be split option into :on_or_after and :before values' 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)
|
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.validates_date :birth_date, :between => on_or_after...before
|
||||||
Person.validators.first.options[:on_or_after].should == on_or_after
|
expect(Person.validators.first.options[:on_or_after]).to eq(on_or_after)
|
||||||
Person.validators.first.options[:before].should == before
|
expect(Person.validators.first.options[:before]).to eq(before)
|
||||||
invalid!(:birth_date, on_or_after - 1, "must be on or after 2010-01-01")
|
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")
|
invalid!(:birth_date, before, "must be before 2010-01-03")
|
||||||
valid!(:birth_date, on_or_after)
|
valid!(:birth_date, on_or_after)
|
||||||
@@ -159,13 +171,13 @@ describe ValidatesTimeliness::Validator do
|
|||||||
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'
|
||||||
person.valid?
|
person.valid?
|
||||||
person.errors[:birth_date].should be_empty
|
expect(person.errors[:birth_date]).to be_empty
|
||||||
end
|
end
|
||||||
|
|
||||||
it "should not be valid when value does not match format" do
|
it "should not be valid when value does not match format" do
|
||||||
person.birth_date = '1913-12-11'
|
person.birth_date = '1913-12-11'
|
||||||
person.valid?
|
person.valid?
|
||||||
person.errors[:birth_date].should include('is not a valid date')
|
expect(person.errors[:birth_date]).to include('is not a valid date')
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
@@ -179,21 +191,21 @@ describe ValidatesTimeliness::Validator do
|
|||||||
it "should be added when ignore_restriction_errors is false" do
|
it "should be added when ignore_restriction_errors is false" do
|
||||||
with_config(:ignore_restriction_errors, false) do
|
with_config(:ignore_restriction_errors, false) do
|
||||||
person.valid?
|
person.valid?
|
||||||
person.errors[:birth_date].first.should match("Error occurred validating birth_date")
|
expect(person.errors[:birth_date].first).to match("Error occurred validating birth_date")
|
||||||
end
|
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
|
||||||
with_config(:ignore_restriction_errors, true) do
|
with_config(:ignore_restriction_errors, true) do
|
||||||
person.valid?
|
person.valid?
|
||||||
person.errors[:birth_date].should be_empty
|
expect(person.errors[:birth_date]).to be_empty
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
it 'should exit on first error' do
|
it 'should exit on first error' do
|
||||||
with_config(:ignore_restriction_errors, false) do
|
with_config(:ignore_restriction_errors, false) do
|
||||||
person.valid?
|
person.valid?
|
||||||
person.errors[:birth_date].should have(1).items
|
expect(person.errors[:birth_date].size).to eq(1)
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
@@ -202,17 +214,17 @@ describe ValidatesTimeliness::Validator do
|
|||||||
describe "default" do
|
describe "default" do
|
||||||
it 'should format date error value as yyyy-mm-dd' do
|
it 'should format date error value as yyyy-mm-dd' do
|
||||||
validator = ValidatesTimeliness::Validator.new(:attributes => [:birth_date], :type => :date)
|
validator = ValidatesTimeliness::Validator.new(:attributes => [:birth_date], :type => :date)
|
||||||
validator.format_error_value(Date.new(2010,1,1)).should == '2010-01-01'
|
expect(validator.format_error_value(Date.new(2010,1,1))).to eq('2010-01-01')
|
||||||
end
|
end
|
||||||
|
|
||||||
it 'should format time error value as hh:nn:ss' do
|
it 'should format time error value as hh:nn:ss' do
|
||||||
validator = ValidatesTimeliness::Validator.new(:attributes => [:birth_time], :type => :time)
|
validator = ValidatesTimeliness::Validator.new(:attributes => [:birth_time], :type => :time)
|
||||||
validator.format_error_value(Time.mktime(2010,1,1,12,34,56)).should == '12:34:56'
|
expect(validator.format_error_value(Time.mktime(2010,1,1,12,34,56))).to eq('12:34:56')
|
||||||
end
|
end
|
||||||
|
|
||||||
it 'should format datetime error value as yyyy-mm-dd hh:nn:ss' do
|
it 'should format datetime error value as yyyy-mm-dd hh:nn:ss' do
|
||||||
validator = ValidatesTimeliness::Validator.new(:attributes => [:birth_datetime], :type => :datetime)
|
validator = ValidatesTimeliness::Validator.new(:attributes => [:birth_datetime], :type => :datetime)
|
||||||
validator.format_error_value(Time.mktime(2010,1,1,12,34,56)).should == '2010-01-01 12:34:56'
|
expect(validator.format_error_value(Time.mktime(2010,1,1,12,34,56))).to eq('2010-01-01 12:34:56')
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
@@ -223,7 +235,7 @@ describe ValidatesTimeliness::Validator do
|
|||||||
|
|
||||||
it 'should use the default format for the type' do
|
it 'should use the default format for the type' do
|
||||||
validator = ValidatesTimeliness::Validator.new(:attributes => [:birth_date], :type => :date)
|
validator = ValidatesTimeliness::Validator.new(:attributes => [:birth_date], :type => :date)
|
||||||
validator.format_error_value(Date.new(2010,1,1)).should == '2010-01-01'
|
expect(validator.format_error_value(Date.new(2010,1,1))).to eq('2010-01-01')
|
||||||
end
|
end
|
||||||
|
|
||||||
after :all do
|
after :all do
|
||||||
|
|||||||
@@ -1,41 +1,39 @@
|
|||||||
require 'spec_helper'
|
RSpec.describe ValidatesTimeliness do
|
||||||
|
|
||||||
describe ValidatesTimeliness do
|
|
||||||
|
|
||||||
it 'should alias use_euro_formats to remove_us_formats on Timeliness gem' do
|
it 'should alias use_euro_formats to remove_us_formats on Timeliness gem' do
|
||||||
Timeliness.should respond_to(:remove_us_formats)
|
expect(Timeliness).to respond_to(:remove_us_formats)
|
||||||
end
|
end
|
||||||
|
|
||||||
it 'should alias to date_for_time_type to dummy_date_for_time_type on Timeliness gem' do
|
it 'should alias to date_for_time_type to dummy_date_for_time_type on Timeliness gem' do
|
||||||
Timeliness.should respond_to(:dummy_date_for_time_type)
|
expect(Timeliness).to respond_to(:dummy_date_for_time_type)
|
||||||
end
|
end
|
||||||
|
|
||||||
describe "config" do
|
describe "config" do
|
||||||
it 'should delegate default_timezone to Timeliness gem' do
|
it 'should delegate default_timezone to Timeliness gem' do
|
||||||
Timeliness.should_receive(:default_timezone=)
|
expect(Timeliness).to receive(:default_timezone=)
|
||||||
ValidatesTimeliness.default_timezone = :utc
|
ValidatesTimeliness.default_timezone = :utc
|
||||||
end
|
end
|
||||||
|
|
||||||
it 'should delegate dummy_date_for_time_type to Timeliness gem' do
|
it 'should delegate dummy_date_for_time_type to Timeliness gem' do
|
||||||
Timeliness.should_receive(:dummy_date_for_time_type)
|
expect(Timeliness).to receive(:dummy_date_for_time_type)
|
||||||
Timeliness.should_receive(:dummy_date_for_time_type=)
|
expect(Timeliness).to receive(:dummy_date_for_time_type=)
|
||||||
array = ValidatesTimeliness.dummy_date_for_time_type
|
array = ValidatesTimeliness.dummy_date_for_time_type
|
||||||
ValidatesTimeliness.dummy_date_for_time_type = array
|
ValidatesTimeliness.dummy_date_for_time_type = array
|
||||||
end
|
end
|
||||||
|
|
||||||
context "parser" do
|
context "parser" do
|
||||||
it 'should delegate add_formats to Timeliness gem' do
|
it 'should delegate add_formats to Timeliness gem' do
|
||||||
Timeliness.should_receive(:add_formats)
|
expect(Timeliness).to receive(:add_formats)
|
||||||
ValidatesTimeliness.parser.add_formats
|
ValidatesTimeliness.parser.add_formats
|
||||||
end
|
end
|
||||||
|
|
||||||
it 'should delegate remove_formats to Timeliness gem' do
|
it 'should delegate remove_formats to Timeliness gem' do
|
||||||
Timeliness.should_receive(:remove_formats)
|
expect(Timeliness).to receive(:remove_formats)
|
||||||
ValidatesTimeliness.parser.remove_formats
|
ValidatesTimeliness.parser.remove_formats
|
||||||
end
|
end
|
||||||
|
|
||||||
it 'should delegate remove_us_formats to Timeliness gem' do
|
it 'should delegate remove_us_formats to Timeliness gem' do
|
||||||
Timeliness.should_receive(:remove_us_formats)
|
expect(Timeliness).to receive(:remove_us_formats)
|
||||||
ValidatesTimeliness.parser.remove_us_formats
|
ValidatesTimeliness.parser.remove_us_formats
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|||||||
@@ -10,11 +10,12 @@ Gem::Specification.new do |s|
|
|||||||
s.description = %q{Adds validation methods to ActiveModel for validating dates and times. Works with multiple ORMS.}
|
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.email = %q{adam.meehan@gmail.com}
|
||||||
s.homepage = %q{http://github.com/adzap/validates_timeliness}
|
s.homepage = %q{http://github.com/adzap/validates_timeliness}
|
||||||
|
s.license = "MIT"
|
||||||
|
|
||||||
s.require_paths = ["lib"]
|
s.require_paths = ["lib"]
|
||||||
s.files = `git ls-files`.split("\n") - %w{ .gitignore .rspec Gemfile Gemfile.lock autotest/discover.rb Appraisals Travis.yml } - Dir['gemsfiles/*']
|
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.test_files = `git ls-files -- {test,spec,features}/*`.split("\n")
|
||||||
s.extra_rdoc_files = ["README.rdoc", "CHANGELOG.rdoc", "LICENSE"]
|
s.extra_rdoc_files = ["README.rdoc", "CHANGELOG.rdoc", "LICENSE"]
|
||||||
|
|
||||||
s.add_runtime_dependency(%q<timeliness>, ["~> 0.3.7"])
|
s.add_runtime_dependency(%q<timeliness>, [">= 0.3.10", "< 1"])
|
||||||
end
|
end
|
||||||
|
|||||||
Reference in New Issue
Block a user