Compare commits

...

190 Commits

Author SHA1 Message Date
Adam Meehan
16221ac092
Merge pull request #190 from AleBL/patch-1
Update README.rdoc
2020-08-19 10:18:45 +10:00
Alessandro Barros
c0c42edd3f
Update README.rdoc
change to new syntax for keys
2020-08-18 17:56:15 -03:00
Adam Meehan
f8b91e9cea v5.0.0.beta2 2020-07-12 16:26:54 +10:00
Adam Meehan
7fa4d85ee3 Change load_config_initializers to sym in Railtie 2020-07-04 17:07:28 +10:00
Adam Meehan
70307293c6 defaults Rails to 5.2.x 2020-07-04 17:06:39 +10:00
Adam Meehan
f42e905cd1
Merge pull request #189 from VSPPedro/update-readme
readme version bump
2020-06-30 09:33:37 +10:00
Pedro Paiva
35eb50aa40 readme version bump 2020-06-29 19:44:40 -03:00
Adam Meehan
3134072f01 v5.0.0.beta1 2020-01-06 15:48:51 +11:00
Adam Meehan
797ba48036 default rails to 5.1.x 2020-01-06 15:47:34 +11:00
Adam Meehan
f8e6480f58 use ActiveSupport Array.wrap 2019-08-20 10:38:38 -10:00
Adam Meehan
3835b2b161 v5.0.0.alpha5 2019-08-08 10:34:27 +10:00
Adam Meehan
00d038bc6f Merge branch 'master' of github.com:adzap/validates_timeliness 2019-08-06 15:17:34 +10:00
Adam Meehan
a3d0182b5e
Merge pull request #186 from timdiggins/patch-1
Add in v4.1.0 details into Changelog
2019-08-06 14:55:45 +10:00
Adam Meehan
46296f914f ensure timeliness initializer is after initializer files 2019-08-03 12:56:08 +10:00
Adam Meehan
4447361743 hash syntax 2019-08-03 12:38:47 +10:00
Adam Meehan
4523138c3c Add Rails initializer to set Timeliness.ambiguous_date_format for Timeliness v0.4+ 2019-08-03 12:38:39 +10:00
Tim Diggins
72807b87a7
Add in v4.1.0 details into Changelog
v4.1.0 details were only available on the v4.1.0 tag. this brings those back in.
2019-08-01 15:30:42 +01:00
Adam Meehan
9daf12c4a1 README typo of Rails version support 2019-06-14 09:59:39 +10:00
Adam Meehan
2a80683683
Merge pull request #183 from kamille-gz/use_AS_on_load_hook
use ActiveSupport.on_load hook for extend ActiveRecord
2019-03-09 17:16:11 +11:00
kamille-321
3a2882be75 use ActiveSupport on_load hook for extend ActiveRecord 2019-03-09 01:14:08 +09:00
Adam Meehan
bf1c808846 fix appraisals 2019-02-09 12:23:57 +11:00
Adam Meehan
d0fcf754ec Update build to ruby 2.5.3 and fix sqlite 2019-02-09 12:15:58 +11:00
Adam Meehan
7233ff66dd v5.0.0.alpha4 2019-02-09 12:00:31 +11:00
Adam Meehan
f0ba09f278 relax timeliness dependency, assuming semver 2019-02-09 11:57:31 +11:00
Adam Meehan
80ee285b3a bump minimum timeliness version 2019-02-03 11:58:11 +11:00
Adam Meehan
cf20576253 readme version bump 2018-07-31 09:08:42 +10:00
Adam Meehan
f39fbb0ad2 v5.0.0.alpha3
Bump with fixed file permissions only.
2018-07-31 09:07:09 +10:00
Adam Meehan
67106b0212 bump gem version in readme 2018-07-02 21:13:24 +10:00
Adam Meehan
bc6d4fe5fc v5.0.0.alpha2 2018-07-02 21:12:12 +10:00
Adam Meehan
70335cd0ba Use prepend for Type overrides /ht @jasl closes #179 2018-07-02 21:08:47 +10:00
Adam Meehan
af406cdedc remove old sqlite-ruby gem. Thanks @jasl 2018-07-02 21:08:47 +10:00
Adam Meehan
837de0d212 v5.0.0.alpha1 2018-07-02 21:08:47 +10:00
Adam Meehan
116930d633
Merge pull request #178 from xMartin/patch-1
update README with actually working 5.x gem version
2018-06-28 09:43:40 +10:00
xMartin
f025d58073
update README with actually working 5.x gem version 2018-06-27 11:22:40 +02:00
Adam Meehan
e645a938b6 remove skipped specs which no longer apply 2018-06-06 13:34:20 -10:00
Adam Meehan
3bc583b200 add Rails 5.1 and 5.2 appraisals 2018-06-06 13:33:44 -10:00
Adam Meehan
4a824e94cd add breaking change for multiparam value format 2018-06-06 13:32:40 -10:00
Adam Meehan
093e33fbed Move conversion module methods in Converter class
Encapsulate conversion helper methods
2018-05-19 16:28:35 +10:00
Adam Meehan
e5bb096161 Move method generation to ActiveModel only and use type system in AR
The ActiveModel type system with extensions in ActiveRecord provide
us a lot of convenience. Some general attribute code is now moved to
ActiveModel only as the AR types provide raw value capturing
(before_type_cast) and type classes to handle type specific string
value parsing if enabled.

In my view, we need to go futher and strip out more from ActiveModel
extension but at least we should have compatibility at the moment.
2018-05-19 15:22:23 +10:00
Adam Meehan
4582c96c42 Time only values are zone aware from AR 5.0 2018-05-18 09:27:16 +10:00
Adam Meehan
38899712c5 update gemfile 2018-05-18 09:14:26 +10:00
Adam Meehan
1c2f4c1b05 turn off timezone aware check specs for now 2018-05-18 09:13:46 +10:00
Adam Meehan
402a6b6e3e Change multiparamter extension to use AR type classes
Multiparamter extension used to do a lot of crappy work to ensure that
the invalid Date dates were not accepted, but also to store a
before_type_cast value in cases when it was invalid. We no longer need
to handle the before_type_cast value as ActiveRecord does that nicely.

Now we just ensure values to cast with a invalid date parts is not
accepted.
2018-05-18 08:58:00 +10:00
Adam Meehan
a1f42fce28 Remove old version checks 2018-05-16 13:12:30 +10:00
Adam Meehan
076714d648 update hash syntax in spec 2018-05-16 12:54:32 +10:00
Adam Meehan
a40a758848 Update CHANGELOG 2018-05-16 12:47:18 +10:00
Adam Meehan
941055c16e Restore usage of DateTime substitute value object
Remove initialize override. Not sure why what was added.
2018-05-16 09:27:27 +10:00
Adam Meehan
4fef42ac24 Rails 5.x compatible Tag value method signature 2018-05-16 09:06:36 +10:00
Adam Meehan
84b1885b6b cannot use POSITION in actionview to map value key 2018-05-15 19:59:59 +10:00
Adam Meehan
90501a8e56 Tag value method takes objects 2018-05-15 19:27:15 +10:00
Adam Meehan
8dc191a4cd Update TextModel method_missing with current ActiveModel methods 2018-05-15 19:26:54 +10:00
Adam Meehan
7d006bd3bd not a fix but stops stack recursion problem 2018-05-15 17:22:43 +10:00
Adam Meehan
973090df85
Merge pull request #167 from AquisTech/fix-issue166-alias_method_chain_not_working_in_rails5
Fix issue#166 alias_method_chain not working in Rails 5
2018-05-15 17:09:07 +10:00
Adam Meehan
596bc87ffd fix rails 5 gemfile 2018-05-15 16:54:53 +10:00
Adam Meehan
5f8604e393 update README with travis badge and 4.x branch 2018-05-15 16:49:38 +10:00
Adam Meehan
a20bfd31ee remove old rails 4 gem and add rails 5.0 2018-05-15 16:45:25 +10:00
AquisTech
c9ca900abf Fix issue#166 alias_method_chain not working in Rails 5 2018-05-14 19:52:24 +05:30
Adam Meehan
a88ae0829c MultiparameterAttribute does not exist anymore 2018-05-13 21:46:26 +10:00
Adam Meehan
6cc47d84e8 setting up for rails 5 work 2018-05-13 21:45:06 +10:00
Adam Meehan
a05f091a42 drop rails 4.0 and 4.1 both EOLed 2018-05-13 21:30:20 +10:00
Adam Meehan
faf708e3be force nokigiri version up 2018-05-13 21:25:31 +10:00
Adam Meehan
38bb74844d need travis bundler fix 2018-05-13 21:15:11 +10:00
Adam Meehan
e275b63203 Merge branch 'master' of github.com:adzap/validates_timeliness 2018-05-13 21:01:36 +10:00
Adam Meehan
e73e0eb30f change travis command to rspec bin 2018-05-13 21:01:12 +10:00
Adam Meehan
43554d8bf0
Merge pull request #152 from aditya-kapoor/allow-generic-msg
Allow the validator to take in the `message` option too.
2018-05-13 20:50:08 +10:00
Adam Meehan
da473b7eea
Update validator.rb
tweak message key lookup
2018-05-13 20:49:25 +10:00
Adam Meehan
fc2af73656 update travis rubies and appraisal rails 2018-05-13 20:46:53 +10:00
Adam Meehan
936c853fdd Merge branch 'master' of github.com:adzap/validates_timeliness 2018-05-13 20:39:32 +10:00
Adam Meehan
f27324a404 relax gemfile gem version 2018-05-13 20:38:23 +10:00
Adam Meehan
4221bf7709
Merge pull request #161 from pedrofurtado/patch-1
Update README.rdoc with the suggestion of a gem with i18n translations
2018-05-13 11:54:55 +10:00
Pedro Felipe de Azevedo Furtado
b51a4544ff
Update README.rdoc 2018-05-10 13:12:52 -03:00
pedrofurtado
285bc769ba
Update README.rdoc 2017-12-29 13:45:39 -02:00
Adam Meehan
a0f2759c7a Merge pull request #160 from reiz/patch-1
Add license info to the gemspec.
2017-10-03 09:42:27 +11:00
Robert Reiz
cc57b1b758 Add license info to the gemspec. 2017-10-02 16:16:36 +02:00
Adam Meehan
d30cebfc7b Update gemfiles 2017-05-14 20:43:39 +10:00
Adam Meehan
a96c24268d Update travis ruby versions 2017-05-14 20:33:33 +10:00
Adam Meehan
13fcc32a83 Test model had wrong validation method 2017-05-14 20:12:36 +10:00
Adam Meehan
e8a96fe9b5 Set ruby version in gemfile 2017-05-14 20:12:14 +10:00
Adam Meehan
f9009995fc Update rspec and rails versions 2017-05-14 19:44:13 +10:00
Adam Meehan
5419a1cc42 Merge branch 'master' of github.com:adzap/validates_timeliness 2017-05-14 19:43:14 +10:00
Adam Meehan
9dd2e87087 Merge pull request #154 from yxf/master
Fixed version checking of ActiveModel
2017-05-14 19:35:49 +10:00
yxf
8a04deebfa Fixed version checking of ActiveModel 2017-05-14 17:07:24 +08:00
Aditya Kapoor
e9c9914c4f Allow the validator to take in the message option too. This is useful in those scenarios where we have a set of generic keys which we need to show.
This would eliminate the need to keeping multiple copies for the default keys set defined by gem.

    The following seems more apt than the later one:

    ```
    validates_date :start_at, on_or_after: :today, message: 'should not be in past'
    ```

    ```
    validates_date :start_at, on_or_after: :today, on_or_after_message: 'should not be in past'
    ```
2017-03-22 13:21:59 +05:30
Adam Meehan
0219c3850d minimum timeliness 0.3.8 2017-03-07 20:33:46 +11:00
Adam Meehan
35caf3638e Fix undefine_attribute_methods spec 2016-01-07 20:50:17 +11:00
Adam Meehan
153051730b Add latest changes 2016-01-07 08:29:49 +11:00
Adam Meehan
5c1406e9b0 v4.0.2 2016-01-07 08:26:19 +11:00
Adam Meehan
32f92cc885 Set @timeliness_methods_generated = false on undefined 2016-01-07 08:25:09 +11:00
Adam Meehan
cb3ac8d445 v4.0.1 2016-01-06 22:20:44 +11:00
Adam Meehan
4e5f5fa695 Return value if not of known type in type cast 2016-01-06 22:20:24 +11:00
Adam Meehan
321834b5ff Fix thread lock on undefine
Move active model methods into active model ORM module to be included
manually if not using AR or other ORM
2016-01-06 22:18:54 +11:00
Adam Meehan
ff131415f7 Fix shadowing 2016-01-06 16:36:25 +11:00
Adam Meehan
ad43eb31d6 Return date if date type 2016-01-06 16:35:01 +11:00
Adam Meehan
0f782a4ed5 Latest rails gems in appraisals 2015-12-30 12:07:06 +11:00
Adam Meehan
d57c64dcc2 Getting with the RSpec times 2015-12-30 12:06:42 +11:00
Adam Meehan
244f398cc4 travis gem install failure on 2.2.0 trying 2.2.3 2015-12-29 18:02:18 +11:00
Adam Meehan
0feeb746cd Fix zone in test 2015-12-29 17:57:54 +11:00
Adam Meehan
6f23e40c25 Remove ruby 2.0 from travis 2015-12-29 17:51:54 +11:00
Adam Meehan
773e816a4d Use latest rspec 2015-12-29 17:51:42 +11:00
Adam Meehan
80c9df448d v3.x changelog 2015-12-29 16:30:46 +11:00
Adam Meehan
54ae49b8a0 Update changelog 2015-12-29 16:27:59 +11:00
Adam Meehan
b20b25940e Update readme to reflect v4.x only 2015-12-29 16:16:08 +11:00
Adam Meehan
51bc0d7d05 Wrong gemfile names in travis 2015-12-29 16:12:37 +11:00
Adam Meehan
c7d87e3fe6 byebug ignore history 2015-12-29 16:11:23 +11:00
Adam Meehan
4dcebde1bc Not using autotest anymore 2015-12-29 16:11:13 +11:00
Adam Meehan
1a869bfb1f Was a random change by my back to rvm 2015-12-29 16:05:35 +11:00
Adam Meehan
22ab7f4455 v4.0.0 2015-12-29 16:01:57 +11:00
Adam Meehan
57379fd78a Just rake in travis i guess 2015-12-29 16:01:30 +11:00
Adam Meehan
2424acc075 Generate default config with AR orm enabled 2015-12-29 15:58:42 +11:00
Adam Meehan
d883758046 Trying travis setup 2015-12-29 15:56:59 +11:00
Adam Meehan
2bcb3ae4dd update appraisal gemfiles 2015-12-29 15:55:16 +11:00
Adam Meehan
0444bdc650 Datetime select extension fixes with help from @johncarney's fork 2015-12-29 15:49:18 +11:00
Adam Meehan
feb508f171 Validator 4.x compat 2015-12-29 14:57:56 +11:00
Adam Meehan
24f2752c2e Multiparam extension 4.x changes 2015-12-29 14:57:38 +11:00
Adam Meehan
afa8dee6ab Rails 4.x attribute assignment compatability 2015-12-29 14:56:58 +11:00
Adam Meehan
08b16f155c Rspec 3 syntax 2015-12-29 14:55:03 +11:00
Adam Meehan
1ddf0f1b0f Remove all mongoid stuff and put in validates_timeliness-mongoid 2015-12-28 14:24:24 +11:00
Adam Meehan
6560a09ad2 Change Time.local_time to Time.local 2015-12-28 13:47:11 +11:00
Adam Meehan
2fef0a23d6 v3.0.15 2015-11-10 18:50:02 +11:00
Adam Meehan
d0f2158ae5 Update Gemfile 2015-11-10 18:49:03 +11:00
Adam Meehan
bb830baea2 Rails 4.0 and 4.1 compat for #setup method removal 2015-11-10 17:52:34 +11:00
Adam Meehan
4d48c3578b Merge pull request #128 from monkbroc/patch-1
Generate attribute methods once instead of on each object creation
2015-09-25 09:13:02 +10:00
Julien Vanier
7d37882327 Generate attribute methods once instead of on each object creation
This is a memory and performance improvement change.

The attribute methods for timeliness are generated at each object creation instead of once, at first creation like AR attribute methods. Since creating those methods uses `module_eval` it's quite slow when it's done each `Model.new`.

For more details on the performance improvement, see the same change on John Carney's fork: https://github.com/johncarney/validates_timeliness/pull/6
2015-09-24 14:58:12 -04:00
Adam Meehan
12f6d2a467 Merge pull request #104 from johnnyshields/mongoid-3
Update to Mongoid 3
2015-09-12 17:23:20 +10:00
Adam Meehan
913bd4ccfa Merge pull request #123 from obfuscoder/master
Fix typo in comment of initializer template
2015-09-12 09:51:44 +10:00
Obfuscoder
f1b8925a36 Fix typo in comment of initializer template 2014-11-02 20:59:50 +01:00
Adam Meehan
f983dcc9dc Merge pull request #109 from RudyOnRails/patch-1
typo
2013-11-07 09:45:10 -08:00
Kevin Musiorski
7be1539cee typo 2013-11-06 13:59:17 -06:00
Johnny Shields
99ae8fe7d7 Update for Mongoid 3:
- Now requires Mongoid 3.0 as minimum version (this is the only version to support Rails 3.2)
- Upgrade specs/helpers for Mongoid 3
- Support Mongoid aliased field names (e.g. field :foo, as: bar, type: String). Includes spec
- Support before_type_cast on Mongoid. No code change was required for this, only a spec change.
- There was one mongoid spec that was previously marked pending. I've found that it now works on Mongoid 3.
- Removed some Mongoid 2.3 and earlier hacks, which simplifies the Mongoid ORM code.
2013-07-24 11:05:01 +09:00
Adam Meehan
00ce472d3e Merge pull request #99 from softace/fixing_build
Fix for running tests in non-Australian timezones
2013-06-16 15:22:33 -07:00
Jarl Friis
0d2c7ce554 Fix for running tests in non-Australian timezones 2013-04-30 19:56:19 +02:00
Adam Meehan
3d798697e1 Merge pull request #98 from 907th/master
Range with excluded end passed to :between option should be split into :on_or_after and :before options
2013-04-30 04:45:54 -07:00
Alexey Chernenkov
dc0fdc0340 Range with excluded end passed to :between option should be split into :on_or_after and :before options 2013-04-30 15:37:44 +06:00
Adam Meehan
dd3b6b5514 Merge pull request #95 from will-ob/fix/require-active-model
Require 'active_model'
2013-04-25 22:01:51 -07:00
Will O'Brien
609fafe7bb Require 'active_model'
Apparently classes are lazily required when using autoload. Prompted by
'uninitialized constant ActiveModel::Validations'
2013-04-25 22:42:28 -04:00
Adam Meehan
df9677f5bf timeliness minimum dep 0.3.7 2012-10-15 20:37:05 +11:00
Adam Meehan
b463b4356a Update rdoc require 2012-10-15 08:46:11 +11:00
Adam Meehan
7dd579b0e0 Get rid of this silly constant for nil array value 2012-10-15 08:45:57 +11:00
Adam Meehan
5becd7886b Removing silly class variables from config module 2012-09-14 18:44:35 +10:00
Adam Meehan
2225c747e1 DRYing up some specs 2012-08-26 16:38:00 +10:00
Adam Meehan
a1dfbf5d7d v3.0.14 2012-08-23 18:40:47 +10:00
Adam Meehan
02fbdc6028 Fix for validates :timeliness form to add attributes to plugin set 2012-08-23 18:38:33 +10:00
Adam Meehan
4fe22458d3 Add mongoid appraisals 2012-08-23 18:37:38 +10:00
Adam Meehan
8c698be4c4 v3.0.13 2012-08-21 10:56:46 +10:00
Adam Meehan
091e7ecfa0 Tweak AR specs 2012-08-21 10:53:54 +10:00
Adam Meehan
8168bcbd3a Ignore more lock files 2012-08-19 22:59:40 +10:00
Adam Meehan
973bbfa82c Moving back to plugin cache for ActiveRecord
This simplifies the code a lot and fixes the issues with date and time
colummns when using the plugin parser.

Add appraisals for all rails 3 versions
2012-08-19 22:57:19 +10:00
Adam Meehan
62557e7e04 Allow any validated attribute to pass timezone aware check in AR 2012-08-09 12:53:29 +10:00
Adam Meehan
fd73c4eccd README touch ups 2012-08-09 11:55:53 +10:00
Adam Meehan
7bcdea1738 RSpec config cleanup 2012-07-13 21:43:29 +10:00
Adam Meehan
8898b8686c v3.0.12 2012-06-23 19:06:38 +10:00
Adam Meehan
aad2db8662 Remove unused cargo culted generator method 2012-06-23 18:58:33 +10:00
Adam Meehan
8e08cbf6e4 Fix load order issue for AR extension using Railite 2012-06-23 18:58:01 +10:00
Adam Meehan
8dd607975b v3.0.11 2012-04-01 16:48:53 +10:00
Adam Meehan
b6acda539f v3.0.10 2012-03-29 13:45:40 +11:00
Adam Meehan
4aa20bb002 Stricter method signature for _timeliness_raw_value_for to take string
Add basic validation specs to each ORM shim
2012-03-29 13:00:28 +11:00
Adam Meehan
df12d6dad0 v3.0.9 2012-03-26 22:00:12 +11:00
Adam Meehan
f11255a7a3 Fix multiparameter extension to not allow partial dates as per ActiveRecord implementation. 2012-03-26 20:22:09 +11:00
Adam Meehan
a6a3dff4d4 Revert "Move AR railtie hook outside of initializer to avoid late inclusion of shim"
This reverts commit 48fd72ae5b.
2012-02-24 11:52:58 +11:00
Adam Meehan
ff594d5873 Tighten up mongoid shim for reload override in 1.8.x and 1.9.x 2012-02-06 22:48:29 +11:00
Adam Meehan
8c1ba048df Refactor timeliness method definition 2012-02-06 22:47:08 +11:00
Adam Meehan
48fd72ae5b Move AR railtie hook outside of initializer to avoid late inclusion of shim 2012-02-06 22:45:01 +11:00
Adam Meehan
43f49076fb Changed multiparameter implementation to be more like AR 3.1 but backwards compatible
Specs improved to not make direct call to multiparameter method.
2012-02-02 09:00:14 +11:00
Chris Griego
64a7440de4 Fix multiparameter extension to work with ActiveRecord 3.1 2012-02-02 08:54:39 +11:00
Chris Griego
2702ec3266 The ActiveRecord extension now returns the original return value for .define_attribute_methods
This helps other extensions, like PermalinkFu, which depend on the return value
2012-02-02 08:44:22 +11:00
Adam Meehan
20549c7acd Setup bundler in the rake file 2012-02-01 21:30:57 +11:00
Adam Meehan
4acd0180f9 Add specs for allow_nil and allow_blank with attribute value cache active 2012-02-01 21:25:42 +11:00
Adam Meehan
3f4615e053 scope a spec to AR version 2012-02-01 21:25:05 +11:00
Adam Meehan
84316bc73a Refactor AR shim for v3.1 checks 2012-02-01 21:24:38 +11:00
Adam Meehan
5be45b00db Hook into undefine_attributes_methods to remove timeliness methods as well 2012-02-01 20:38:05 +11:00
Adam Meehan
f8aeeca0a9 A fix for ActiveRecord 3.1.x and plugin parser not working together (issue #52) 2012-01-26 22:44:01 +11:00
Adam Meehan
f5e2deeb73 Add credit in changelog 2011-12-24 17:53:07 +11:00
Adam Meehan
acd10f7b64 v3.0.8 2011-12-24 17:50:22 +11:00
Adam Meehan
68b2579ca0 Merge pull request #61 from carlosantoniodasilva/rails-3.2-deprecation-warnings
Remove deprecated InstanceMethods module when using AS::Concern
2011-12-22 23:00:25 -08:00
Carlos Antonio da Silva
7a784a6c54 Remove InstanceMethods module when using AS::Concern, as it was deprecated in 3.2 2011-12-20 20:52:33 -02:00
Adam Meehan
907fd3e439 Fix for Mongoid shim for reload which was nicely moved into a module to allow super 2011-12-10 17:46:35 +11:00
Adam Meehan
e44e9d2f11 v3.0.7 2011-09-21 21:41:01 +10:00
Adam Meehan
86b7bc4829 Fix before_type_cast for non-dirty attributes (issue #52).
Only use before_type_cast implementation if earlier than 3.1.0 which has
it's own working version for date/time fields now.
2011-09-21 21:28:53 +10:00
Adam Meehan
34824bbbab Specs against rails 3.1.0 2011-09-18 18:56:06 +10:00
Adam Meehan
11ae94cba6 Cleanup DateTimeSelect extension specs 2011-09-18 18:53:38 +10:00
Adam Meehan
473da0fa5e Fix DateTimeSelect extension for Rails 3.1.0 2011-09-18 18:37:01 +10:00
Adam Meehan
e7b503ae4f Fix DateTimeSelect extension to use integers from param values.
Change extension spec to use string params.
2011-09-18 18:34:23 +10:00
Adam Meehan
af9d6cdba4 Fix brokena and brittle tests 2011-09-18 18:33:30 +10:00
Adam Meehan
9a99c47323 Can use simple Mongoid inclusion now that fixed Concern usage is released (a while ago) 2011-09-18 18:33:29 +10:00
Adam Meehan
fbc2867197 Remove Gemfile.lock from source and ignore 2011-09-18 18:33:29 +10:00
Adam Meehan
969edfcb83 Helper method spec 2011-09-18 18:33:29 +10:00
Adam Meehan
1faf0efa4b Spec helper validate method setup in test models not needed 2011-09-18 18:33:29 +10:00
Adam Meehan
2b53d713bc Merge pull request #45 from trusche/master
Add support for :discard_day option in DateTimeSelect extension
2011-09-18 01:32:08 -07:00
Thilo Rusche
ac90ab96cd fixed integer vs string expectation 2011-05-13 17:47:59 +01:00
Thilo Rusche
b4c1a39343 added support for :discard_day in form helpers 2011-05-13 17:13:11 +01:00
49 changed files with 1052 additions and 903 deletions

3
.gitignore vendored
View File

@ -1,3 +1,6 @@
pkg/ pkg/
.bundle/ .bundle/
.rvmrc .rvmrc
Gemfile.lock
gemfiles/*.lock
.byebug_history

4
.rspec
View File

@ -1,2 +1,4 @@
--format nested --format documentation
--color --color
--require spec_helper
--require byebug

20
.travis.yml Normal file
View File

@ -0,0 +1,20 @@
language: ruby
before_install: gem install bundler
cache: bundler
gemfile:
- gemfiles/rails_5_0.gemfile
- gemfiles/rails_5_1.gemfile
- gemfiles/rails_5_2.gemfile
rvm:
- "2.5.3"
script: 'bundle exec rspec'
notifications:
email:
recipients:
- adam.meehan@gmail.com
on_failure: change
on_success: never

11
Appraisals Normal file
View File

@ -0,0 +1,11 @@
appraise "rails_5_0" do
gem "rails", "~> 5.0.0"
end
appraise "rails_5_1" do
gem "rails", "~> 5.1.0"
end
appraise "rails_5_2" do
gem "rails", "~> 5.2.0"
end

View File

@ -1,3 +1,71 @@
= [UNRELEASED]
* Fix DateTimeSelect extension support (AquisTech)
* Relaxed Timeliness dependency version which allows for >= 0.4.0 with
threadsafety fix for use_us_formats and use_euro_formats for hot switching
in a request.
* Add initializer to ensure Timeliness v0.4+ ambiguous date config is set
correctly when using `use_euro_formats` or `remove_use_formats'.
Breaking Changes
* Update Multiparameter extension to use ActiveRecord type classes with multiparameter handling
which stores a hash of multiparamter values as the value before type cast, no longer a mushed datetime string
* Removed all custom plugin attribute methods and method overrides in favour using ActiveModel type system
= 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]
* Fixes mongoid 3 support and removes mongoid 2 support(johnnyshields)
* Some documentation/comments tidying
* Some general tidying up
= 3.0.14 [2012-08-23]
* Fix for using validates :timeliness => {} form to correctly add attributes to timeliness validated attributes.
= 3.0.13 [2012-08-21]
* Fix ActiveRecord issues with using plugin parser by using old way of caching values.
* Allow any ActiveRecord non-column attribute to be validated
= 3.0.12 [2012-06-23]
* Fix load order issue when relying on Railtie to load ActiveRecord extension
= 3.0.11 [2012-04-01]
* Change dependency on Timeliness version due to a broken release
= 3.0.10 [2012-03-26]
* Fix for ActiveRecord shim and validation with :allow_blank => true in AR 3.1+. Fixes issue#52.
= 3.0.9 [2012-03-26]
* ActiveRecord 3.1+ suport
* Fixes for multiparameter extension with empty date values (thanks @mogox, @Sharagoz)
= 3.0.8 [2011-12-24]
* Remove deprecated InstanceMethods module when using AS::Concern (carlosantoniodasilva)
* Update Mongoid shim for v2.3 compatability.
= 3.0.7 [2011-09-21]
* Fix ActiveRecord before_type_cast extension for non-dirty attributes.
* Don't override AR before_type_cast for >= 3.1.0 which now has it's own implementation for date/time attributes.
* Fix DateTimeSelect extension to convert params to integers (#45)
* Add #change method to DateTimeSelect extension (@trusche, #45)
* Cleanup Mongoid shim.
= 3.0.6 [2011-05-09] = 3.0.6 [2011-05-09]
* Fix for AR type conversion for date columns when using plugin parser. * Fix for AR type conversion for date columns when using plugin parser.
* Add timeliness_type_cast_code for ORM specific type casting after parsing. * Add timeliness_type_cast_code for ORM specific type casting after parsing.

20
Gemfile
View File

@ -2,17 +2,11 @@ source 'http://rubygems.org'
gemspec gemspec
gem 'ZenTest' gem 'rails', '~> 5.2.4'
gem 'rails', '3.0.7' gem 'rspec'
gem 'rspec', '~> 2.4' gem 'rspec-rails', '~> 3.7'
gem 'rspec-rails', '~> 2.4'
gem 'timecop' gem 'timecop'
gem 'rspec_tag_matchers' gem 'byebug'
gem 'appraisal'
group :mongoid do gem 'sqlite3', '~> 1.3.6'
gem 'mongoid', '2.0.0.rc.8' gem 'nokogiri', '~> 1.8'
end
group :active_record do
gem 'sqlite3-ruby', :require => 'sqlite3'
end

View File

@ -1,118 +0,0 @@
PATH
remote: .
specs:
validates_timeliness (3.0.6)
timeliness (~> 0.3.3)
GEM
remote: http://rubygems.org/
specs:
ZenTest (4.5.0)
abstract (1.0.0)
actionmailer (3.0.7)
actionpack (= 3.0.7)
mail (~> 2.2.15)
actionpack (3.0.7)
activemodel (= 3.0.7)
activesupport (= 3.0.7)
builder (~> 2.1.2)
erubis (~> 2.6.6)
i18n (~> 0.5.0)
rack (~> 1.2.1)
rack-mount (~> 0.6.14)
rack-test (~> 0.5.7)
tzinfo (~> 0.3.23)
activemodel (3.0.7)
activesupport (= 3.0.7)
builder (~> 2.1.2)
i18n (~> 0.5.0)
activerecord (3.0.7)
activemodel (= 3.0.7)
activesupport (= 3.0.7)
arel (~> 2.0.2)
tzinfo (~> 0.3.23)
activeresource (3.0.7)
activemodel (= 3.0.7)
activesupport (= 3.0.7)
activesupport (3.0.7)
arel (2.0.9)
bson (1.3.0)
builder (2.1.2)
diff-lcs (1.1.2)
erubis (2.6.6)
abstract (>= 1.0.0)
i18n (0.5.0)
mail (2.2.19)
activesupport (>= 2.3.6)
i18n (>= 0.4.0)
mime-types (~> 1.16)
treetop (~> 1.4.8)
mime-types (1.16)
mongo (1.3.0)
bson (>= 1.3.0)
mongoid (2.0.0.rc.8)
activemodel (~> 3.0)
mongo (~> 1.2)
tzinfo (~> 0.3.22)
will_paginate (~> 3.0.pre)
nokogiri (1.4.4)
polyglot (0.3.1)
rack (1.2.2)
rack-mount (0.6.14)
rack (>= 1.0.0)
rack-test (0.5.7)
rack (>= 1.0)
rails (3.0.7)
actionmailer (= 3.0.7)
actionpack (= 3.0.7)
activerecord (= 3.0.7)
activeresource (= 3.0.7)
activesupport (= 3.0.7)
bundler (~> 1.0)
railties (= 3.0.7)
railties (3.0.7)
actionpack (= 3.0.7)
activesupport (= 3.0.7)
rake (>= 0.8.7)
thor (~> 0.14.4)
rake (0.8.7)
rspec (2.5.0)
rspec-core (~> 2.5.0)
rspec-expectations (~> 2.5.0)
rspec-mocks (~> 2.5.0)
rspec-core (2.5.1)
rspec-expectations (2.5.0)
diff-lcs (~> 1.1.2)
rspec-mocks (2.5.0)
rspec-rails (2.5.0)
actionpack (~> 3.0)
activesupport (~> 3.0)
railties (~> 3.0)
rspec (~> 2.5.0)
rspec_tag_matchers (1.0.0)
nokogiri (>= 1.4.0)
rspec-rails (>= 1.2.6)
sqlite3 (1.3.3)
sqlite3-ruby (1.3.3)
sqlite3 (>= 1.3.3)
thor (0.14.6)
timecop (0.3.5)
timeliness (0.3.3)
treetop (1.4.9)
polyglot (>= 0.3.1)
tzinfo (0.3.27)
will_paginate (3.0.pre2)
PLATFORMS
ruby
DEPENDENCIES
ZenTest
mongoid (= 2.0.0.rc.8)
rails (= 3.0.7)
rspec (~> 2.4)
rspec-rails (~> 2.4)
rspec_tag_matchers
sqlite3-ruby
timecop
validates_timeliness!

View File

@ -1,13 +1,13 @@
= ValidatesTimeliness = ValidatesTimeliness {<img src="https://travis-ci.org/adzap/validates_timeliness.svg?branch=master" alt="Build Status" />}[https://travis-ci.org/adzap/validates_timeliness]
* Source: http://github.com/adzap/validates_timeliness * Source: http://github.com/adzap/validates_timeliness
* Bugs: http://github.com/adzap/validates_timeliness/issues * Issues: http://github.com/adzap/validates_timeliness/issues
== Description == Description
Complete validation of dates, times and datetimes for Rails 3.x and ActiveModel. Complete validation of dates, times and datetimes for Rails 5.x and ActiveModel.
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 4.x go here [https://github.com/adzap/validates_timeliness/tree/4-0-stable].
== Features == Features
@ -18,24 +18,19 @@ If you a looking for the old version for Rails 2.x go here[http://github.com/adz
* Only Rails date/time validation plugin offering complete validation (See ORM/ODM support) * Only Rails date/time validation plugin offering complete validation (See ORM/ODM support)
* Adds extensions to fix Rails date/time select issues (See Extensions)
* Uses extensible date/time parser (Using {timeliness gem}[http://github.com/adzap/timeliness]. See Plugin Parser) * Uses extensible date/time parser (Using {timeliness gem}[http://github.com/adzap/timeliness]. See Plugin Parser)
* Supports I18n for the error messages * Adds extensions to fix Rails date/time select issues (See Extensions)
* 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 Ruby 1.8.x, 1.9.x and Rubinius.
== Installation == Installation
As plugin (from master)
rails plugin install git://github.com/adzap/validates_timeliness.git
As gem
# in Gemfile # in Gemfile
gem 'validates_timeliness', '~> 3.0.2' gem 'validates_timeliness', '~> 5.0.0.beta1'
# Run bundler # Run bundler
$ bundle install $ bundle install
@ -44,7 +39,7 @@ Then run
$ rails generate validates_timeliness:install $ rails generate validates_timeliness:install
This creates configuration initializer and locale files. In the initializer, you there are a number of config This creates configuration initializer and locale files. In the initializer, there are a number of config
options to customize the plugin. options to customize the plugin.
NOTE: You may wish to enable the plugin parser and the extensions to start. Please read those sections first. NOTE: You may wish to enable the plugin parser and the extensions to start. Please read those sections first.
@ -54,18 +49,21 @@ NOTE: You may wish to enable the plugin parser and the extensions to start. Plea
validates_datetime :occurred_at validates_datetime :occurred_at
validates_date :date_of_birth, :before => lambda { 18.years.ago }, validates_date :date_of_birth, before: lambda { 18.years.ago },
:before_message => "must be at least 18 years old" before_message: "must be at least 18 years old"
validates_datetime :finish_time, :after => :start_time # Method symbol validates_datetime :finish_time, after: :start_time # Method symbol
validates_date :booked_at, :on => :create, :on_or_after => :today # See Restriction Shorthand. validates_date :booked_at, on: :create, on_or_after: :today # See Restriction Shorthand.
validates_time :booked_at, :between => ['9.00am', '5:00pm']
validates_time :breakfast_time, :on_or_after => '6:00am', validates_time :booked_at, between: ['9:00am', '5:00pm'] # On or after 9:00AM and on or before 5:00PM
:on_or_after_message => 'must be after opening time', validates_time :booked_at, between: '9:00am'..'5:00pm' # The same as previous example
:before => :lunchtime, validates_time :booked_at, between: '9:00am'...'5:00pm' # On or after 9:00AM and strictly before 5:00PM
:before_message => 'must be before lunch time'
validates_time :breakfast_time, on_or_after: '6:00am',
on_or_after_message: 'must be after opening time',
before: :lunchtime,
before_message: 'must be before lunch time'
== Usage == Usage
@ -74,14 +72,14 @@ To validate a model with a date, time or datetime attribute you just use the
validation method validation method
class Person < ActiveRecord::Base class Person < ActiveRecord::Base
validates_date :date_of_birth, :on_or_before => lambda { Date.current } validates_date :date_of_birth, on_or_before: lambda { Date.current }
# or # or
validates :date_of_birth, :timeliness => {:on_or_before => lambda { Date.current }, :type => :date} validates :date_of_birth, timeliness: {on_or_before: lambda { Date.current }, type: :date}
end end
# or even on a specific record, per ActiveModel API. or even on a specific record, per ActiveModel API.
@person.validates_date :date_of_birth, :on_or_before => lambda { Date.current } @person.validates_date :date_of_birth, on_or_before: lambda { Date.current }
The list of validation methods available are as follows: The list of validation methods available are as follows:
@ -139,8 +137,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
@ -178,8 +176,8 @@ You can also use validation options for custom error messages. The following opt
:after_message :after_message
:on_or_after_message :on_or_after_message
Note: There is no :between_message option. The between error message should be defined using the Note: There is no :between_message option. The between error message should be defined using the :on_or_after and :on_or_before
:on_or_before and :on_or_after messages. (:before in case when :between argument is a Range with excluded high value, see Examples) messages.
It is highly recommended you use the I18n system for error messages. It is highly recommended you use the I18n system for error messages.
@ -208,14 +206,14 @@ plugin allows you to use shorthand symbols for often used relative times or date
Just provide the symbol as the option value like so: Just provide the symbol as the option value like so:
validates_date :birth_date, :on_or_before => :today validates_date :birth_date, on_or_before: :today
The :today symbol is evaluated as <tt>lambda { Date.today }</tt>. The :now and :today The :today symbol is evaluated as <tt>lambda { Date.today }</tt>. The :now and :today
symbols are pre-configured. Configure your own like so: symbols are pre-configured. Configure your own like so:
# in the setup block # in the setup block
config.restriction_shorthand_symbols.update( config.restriction_shorthand_symbols.update(
:yesterday => lambda { 1.day.ago } yesterday: lambda { 1.day.ago }
) )
@ -298,4 +296,4 @@ To see the generous people who have contributed code, take a look at the {contri
== License == License
Copyright (c) 2008-2010 Adam Meehan, released under the MIT license Copyright (c) 2008 Adam Meehan, released under the MIT license

View File

@ -1,7 +1,11 @@
require 'bundler' require 'bundler'
require 'bundler/setup'
require 'appraisal'
Bundler::GemHelper.install_tasks Bundler::GemHelper.install_tasks
require 'rake/rdoctask' require 'rdoc/task'
require 'rspec/core/rake_task' require 'rspec/core/rake_task'
desc "Run specs" desc "Run specs"

View File

@ -1 +0,0 @@
Autotest.add_discovery { "rspec2" }

View File

@ -0,0 +1,14 @@
# This file was generated by Appraisal
source "http://rubygems.org"
gem "rails", "~> 5.0.0"
gem "rspec"
gem "rspec-rails", "~> 3.7"
gem "timecop"
gem "byebug"
gem "appraisal"
gem "sqlite3", "~> 1.3.6"
gem "nokogiri", "~> 1.8"
gemspec path: "../"

View File

@ -0,0 +1,14 @@
# This file was generated by Appraisal
source "http://rubygems.org"
gem "rails", "~> 5.1.0"
gem "rspec"
gem "rspec-rails", "~> 3.7"
gem "timecop"
gem "byebug"
gem "appraisal"
gem "sqlite3", "~> 1.3.6"
gem "nokogiri", "~> 1.8"
gemspec path: "../"

View File

@ -0,0 +1,14 @@
# This file was generated by Appraisal
source "http://rubygems.org"
gem "rails", "~> 5.2.0"
gem "rspec"
gem "rspec-rails", "~> 3.7"
gem "timecop"
gem "byebug"
gem "appraisal"
gem "sqlite3", "~> 1.3.6"
gem "nokogiri", "~> 1.8"
gemspec path: "../"

View File

@ -3,7 +3,6 @@ module ValidatesTimeliness
class InstallGenerator < Rails::Generators::Base class InstallGenerator < Rails::Generators::Base
desc "Copy ValidatesTimeliness default files" desc "Copy ValidatesTimeliness default files"
source_root File.expand_path('../templates', __FILE__) source_root File.expand_path('../templates', __FILE__)
class_option :template_engine
def copy_initializers def copy_initializers
copy_file 'validates_timeliness.rb', 'config/initializers/validates_timeliness.rb' copy_file 'validates_timeliness.rb', 'config/initializers/validates_timeliness.rb'

View File

@ -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
@ -32,7 +32,7 @@ ValidatesTimeliness.setup do |config|
# Remove one or more formats making them invalid. e.g. remove_formats(:date, 'dd/mm/yyy') # Remove one or more formats making them invalid. e.g. remove_formats(:date, 'dd/mm/yyy')
# config.parser.remove_formats() # config.parser.remove_formats()
# #
# Change the amiguous year threshold when parsing a 2 digit year # Change the ambiguous year threshold when parsing a 2 digit year
# config.parser.ambiguous_year_threshold = 30 # config.parser.ambiguous_year_threshold = 30
# #
# Treat ambiguous dates, such as 01/02/1950, as a Non-US date. # Treat ambiguous dates, such as 01/02/1950, as a Non-US date.

View File

@ -24,26 +24,24 @@ module ValidatesTimeliness
class << self class << self
delegate :default_timezone, :default_timezone=, :dummy_date_for_time_type, :dummy_date_for_time_type=, :to => Timeliness delegate :default_timezone, :default_timezone=, :dummy_date_for_time_type, :dummy_date_for_time_type=, :to => Timeliness
attr_accessor :extend_orms, :ignore_restriction_errors, :restriction_shorthand_symbols, :use_plugin_parser
end end
# Extend ORM/ODMs for full support (:active_record, :mongoid). # Extend ORM/ODMs for full support (:active_record).
mattr_accessor :extend_orms self.extend_orms = []
@@extend_orms = []
# Ignore errors when restriction options are evaluated # Ignore errors when restriction options are evaluated
mattr_accessor :ignore_restriction_errors self.ignore_restriction_errors = false
@@ignore_restriction_errors = false
# Shorthand time and date symbols for restrictions # Shorthand time and date symbols for restrictions
mattr_accessor :restriction_shorthand_symbols self.restriction_shorthand_symbols = {
@@restriction_shorthand_symbols = { now: proc { Time.current },
:now => lambda { Time.current }, today: proc { Date.current }
:today => lambda { Date.current }
} }
# Use the plugin date/time parser which is stricter and extensible # Use the plugin date/time parser which is stricter and extensible
mattr_accessor :use_plugin_parser self.use_plugin_parser = false
@@use_plugin_parser = false
# Default timezone # Default timezone
self.default_timezone = :utc self.default_timezone = :utc
@ -51,18 +49,20 @@ module ValidatesTimeliness
# Set the dummy date part for a time type values. # Set the dummy date part for a time type values.
self.dummy_date_for_time_type = [ 2000, 1, 1 ] self.dummy_date_for_time_type = [ 2000, 1, 1 ]
def self.parser
Timeliness
end
# Setup method for plugin configuration # Setup method for plugin configuration
def self.setup def self.setup
yield self yield self
load_orms
end
def self.load_orms
extend_orms.each {|orm| require "validates_timeliness/orm/#{orm}" } extend_orms.each {|orm| require "validates_timeliness/orm/#{orm}" }
end end
def self.parser; Timeliness end
end end
require 'validates_timeliness/conversion' require 'validates_timeliness/converter'
require 'validates_timeliness/validator' require 'validates_timeliness/validator'
require 'validates_timeliness/helper_methods' require 'validates_timeliness/helper_methods'
require 'validates_timeliness/attribute_methods' require 'validates_timeliness/attribute_methods'

View File

@ -6,74 +6,44 @@ module ValidatesTimeliness
class_attribute :timeliness_validated_attributes class_attribute :timeliness_validated_attributes
self.timeliness_validated_attributes = [] self.timeliness_validated_attributes = []
end end
module ClassMethods
public
# Override in ORM shim
def timeliness_attribute_timezone_aware?(attr_name)
false
end
# Override in ORM shim
def timeliness_attribute_type(attr_name)
:datetime
end
protected
def define_timeliness_methods(before_type_cast=false)
return if timeliness_validated_attributes.blank?
timeliness_validated_attributes.each do |attr_name|
define_timeliness_write_method(attr_name)
define_timeliness_before_type_cast_method(attr_name) if before_type_cast
end
end
def define_timeliness_write_method(attr_name)
method_body, line = <<-EOV, __LINE__ + 1
def #{attr_name}=(value)
@timeliness_cache ||= {}
@timeliness_cache["#{attr_name}"] = value
#{ "if value.is_a?(String)\n#{timeliness_type_cast_code(attr_name, 'value')}\nend" if ValidatesTimeliness.use_plugin_parser }
super
end
EOV
generated_timeliness_methods.module_eval(method_body, __FILE__, line)
end
def define_timeliness_before_type_cast_method(attr_name)
method_body, line = <<-EOV, __LINE__ + 1
def #{attr_name}_before_type_cast
_timeliness_raw_value_for('#{attr_name}')
end
EOV
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
end
module InstanceMethods
def _timeliness_raw_value_for(attr_name)
@timeliness_cache && @timeliness_cache[attr_name.to_s]
end
def _clear_timeliness_cache
@timeliness_cache = {}
end
end
end end
end end
ActiveModel::Type::Date.prepend Module.new {
def cast_value(value)
return super unless ValidatesTimeliness.use_plugin_parser
if value.is_a?(::String)
return if value.empty?
value = Timeliness::Parser.parse(value, :date)
value.to_date if value
elsif value.respond_to?(:to_date)
value.to_date
else
value
end
end
}
ActiveModel::Type::Time.prepend Module.new {
def user_input_in_time_zone(value)
return super unless ValidatesTimeliness.use_plugin_parser
if value.is_a?(String)
dummy_time_value = value.sub(/\A(\d\d\d\d-\d\d-\d\d |)/, Date.current.to_s + ' ')
Timeliness::Parser.parse(dummy_time_value, :datetime, zone: :current)
else
value.in_time_zone
end
end
}
ActiveModel::Type::DateTime.prepend Module.new {
def user_input_in_time_zone(value)
if value.is_a?(String) && ValidatesTimeliness.use_plugin_parser
Timeliness::Parser.parse(value, :datetime, zone: :current)
else
value.in_time_zone
end
end
}

View File

@ -1,10 +1,18 @@
module ValidatesTimeliness module ValidatesTimeliness
module Conversion class Converter
attr_reader :type, :format, :ignore_usec
def type_cast_value(value, type) def initialize(type:, format: nil, ignore_usec: false, time_zone_aware: false)
@type = type
@format = format
@ignore_usec = ignore_usec
@time_zone_aware = time_zone_aware
end
def type_cast_value(value)
return nil if value.nil? || !value.respond_to?(:to_time) return nil if value.nil? || !value.respond_to?(:to_time)
value = value.in_time_zone if value.acts_like?(:time) && @timezone_aware value = value.in_time_zone if value.acts_like?(:time) && time_zone_aware?
value = case type value = case type
when :time when :time
dummy_time(value) dummy_time(value)
@ -12,9 +20,11 @@ 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 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 time_zone_aware?))
else else
value value
end end
@ -22,30 +32,30 @@ module ValidatesTimeliness
def dummy_time(value) def dummy_time(value)
time = if value.acts_like?(:time) time = if value.acts_like?(:time)
value = value.in_time_zone if @timezone_aware value = value.in_time_zone if time_zone_aware?
[value.hour, value.min, value.sec] [value.hour, value.min, value.sec]
else else
[0,0,0] [0,0,0]
end end
values = ValidatesTimeliness.dummy_date_for_time_type + time values = ValidatesTimeliness.dummy_date_for_time_type + time
Timeliness::Parser.make_time(values, (:current if @timezone_aware)) Timeliness::Parser.make_time(values, (:current if time_zone_aware?))
end end
def evaluate_option_value(value, record) def evaluate(value, scope=nil)
case value case value
when Time, Date when Time, Date
value value
when String when String
parse(value) parse(value)
when Symbol when Symbol
if !record.respond_to?(value) && restriction_shorthand?(value) if !scope.respond_to?(value) && restriction_shorthand?(value)
ValidatesTimeliness.restriction_shorthand_symbols[value].call ValidatesTimeliness.restriction_shorthand_symbols[value].call
else else
evaluate_option_value(record.send(value), record) evaluate(scope.send(value))
end end
when Proc when Proc
result = value.arity > 0 ? value.call(record) : value.call result = value.arity > 0 ? value.call(scope) : value.call
evaluate_option_value(result, record) evaluate(result, scope)
else else
value value
end end
@ -57,14 +67,18 @@ module ValidatesTimeliness
def parse(value) def parse(value)
return nil if value.nil? return nil if value.nil?
if ValidatesTimeliness.use_plugin_parser if ValidatesTimeliness.use_plugin_parser
Timeliness::Parser.parse(value, @type, :zone => (:current if @timezone_aware), :format => options[:format], :strict => false) Timeliness::Parser.parse(value, type, zone: (:current if time_zone_aware?), format: format, strict: false)
else else
@timezone_aware ? Time.zone.parse(value) : value.to_time(ValidatesTimeliness.default_timezone) time_zone_aware? ? Time.zone.parse(value) : value.to_time(ValidatesTimeliness.default_timezone)
end end
rescue ArgumentError, TypeError rescue ArgumentError, TypeError
nil nil
end end
def time_zone_aware?
@time_zone_aware
end
end end
end end

View File

@ -1,14 +1,13 @@
module ValidatesTimeliness module ValidatesTimeliness
module Extensions module Extensions
autoload :DateTimeSelect, 'validates_timeliness/extensions/date_time_select' autoload :TimelinessDateTimeSelect, '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(:prepend, ValidatesTimeliness::Extensions::TimelinessDateTimeSelect)
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

View File

@ -1,44 +1,50 @@
module ValidatesTimeliness module ValidatesTimeliness
module Extensions module Extensions
module DateTimeSelect module TimelinessDateTimeSelect
extend ActiveSupport::Concern
# Intercepts the date and time select helpers to reuse the values from # Intercepts the date and time select helpers to reuse the values from
# the params rather than the parsed value. This allows invalid date/time # the params rather than the parsed value. This allows invalid date/time
# values to be redisplayed instead of blanks to aid correction by the user. # values to be redisplayed instead of blanks to aid correction by the user.
# 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.
attr_accessor :object_name, :method_name, :template_object, :options, :html_options
included do POSITION = {
alias_method_chain :datetime_selector, :timeliness :year => 1, :month => 2, :day => 3, :hour => 4, :min => 5, :sec => 6
}.freeze
class DateTimeValue
attr_accessor :year, :month, :day, :hour, :min, :sec
def initialize(year:, month:, day: nil, hour: nil, min: nil, sec: nil)
@year, @month, @day, @hour, @min, @sec = year, month, day, hour, min, sec
end end
module InstanceMethods def change(options)
self.class.new(
TimelinessDateTime = Struct.new(:year, :month, :day, :hour, :min, :sec) year: options.fetch(:year, year),
month: options.fetch(:month, month),
def datetime_selector_with_timeliness(*args) day: options.fetch(:day, day),
@timeliness_date_or_time_tag = true hour: options.fetch(:hour, hour),
datetime_selector_without_timeliness(*args) min: options.fetch(:min) { options[:hour] ? 0 : min },
sec: options.fetch(:sec) { options[:hour] || options[:min] ? 0 : sec }
)
end
end end
def value(object) # Splat args to support Rails 5.0 which expects object, and 5.2 which doesn't
unless @timeliness_date_or_time_tag && @template_object.params[@object_name] def value(*object)
return super return super unless @template_object.params[@object_name]
end
pairs = @template_object.params[@object_name].select {|k,v| k =~ /^#{@method_name}\(/ } pairs = @template_object.params[@object_name].select {|k,v| k =~ /^#{@method_name}\(/ }
return super if pairs.empty? return super if pairs.empty?
values = [nil] * 6 values = {}
pairs.map do |(param, value)| pairs.each_pair do |key, value|
position = param.scan(/\(([0-9]*).*\)/).first.first position = key[/\((\d+)\w+\)/, 1]
values[position.to_i-1] = value values[POSITION.key(position.to_i)] = value.to_i
end end
TimelinessDateTime.new(*values) DateTimeValue.new(values)
end end
end end
end
end end
end end

View File

@ -1,69 +1,55 @@
module ValidatesTimeliness module ValidatesTimeliness
module Extensions module Extensions
module MultiparameterHandler class AcceptsMultiparameterTime < Module
extend ActiveSupport::Concern
# Stricter handling of date and time values from multiparameter def initialize(defaults: {})
# assignment from the date/time select view helpers
included do define_method(:cast) do |value|
alias_method_chain :instantiate_time_object, :timeliness if value.is_a?(Hash)
alias_method_chain :execute_callstack_for_multiparameter_attributes, :timeliness value_from_multiparameter_assignment(value)
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)
if Date.valid_civil?(*values[0..2])
instantiate_time_object_without_timeliness(name, values)
else else
invalid_multiparameter_date_or_time_as_string(values) super(value)
end end
end end
def instantiate_date_object(name, values) define_method(:assert_valid_value) do |value|
values = values.map { |v| v.nil? ? 1 : v } if value.is_a?(Hash)
Date.new(*values) value_from_multiparameter_assignment(value)
rescue ArgumentError => ex
invalid_multiparameter_date_or_time_as_string(values)
end
def execute_callstack_for_multiparameter_attributes_with_timeliness(callstack)
errors = []
callstack.each do |name, values_with_empty_parameters|
begin
klass = (self.class.reflect_on_aggregation(name.to_sym) || column_for_attribute(name)).klass
values = values_with_empty_parameters.reject { |v| v.nil? }
if values.empty?
send(name + "=", nil)
else else
super(value)
value = if Time == klass end
instantiate_time_object(name, values)
elsif Date == klass
instantiate_date_object(name, values_with_empty_parameters)
else
klass.new(*values)
end end
send(name + "=", value) define_method(:value_from_multiparameter_assignment) do |values_hash|
defaults.each do |k, v|
values_hash[k] ||= v
end end
rescue => ex return unless values_hash.values_at(1,2,3).all?{ |v| v.present? } &&
errors << ActiveRecord::AttributeAssignmentError.new("error on assignment #{values.inspect} to #{name}", ex, name) Date.valid_civil?(*values_hash.values_at(1,2,3))
end
end values = values_hash.sort.map(&:last)
unless errors.empty? ::Time.send(default_timezone, *values)
raise ActiveRecord::MultiparameterAssignmentErrors.new(errors), "#{errors.size} error(s) on assignment of multiparameter attributes"
end end
private :value_from_multiparameter_assignment
end end
end end
end end
end end
ActiveModel::Type::Date.class_eval do
include ValidatesTimeliness::Extensions::AcceptsMultiparameterTime.new
end
ActiveModel::Type::Time.class_eval do
include ValidatesTimeliness::Extensions::AcceptsMultiparameterTime.new(
defaults: { 1 => 1970, 2 => 1, 3 => 1, 4 => 0, 5 => 0 }
)
end
ActiveModel::Type::DateTime.class_eval do
include ValidatesTimeliness::Extensions::AcceptsMultiparameterTime.new(
defaults: { 4 => 0, 5 => 0 }
)
end

View File

@ -14,12 +14,13 @@ 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)
options = _merge_attributes(attr_names).merge(:type => type) timeliness_validation_for attr_names
if respond_to?(:timeliness_validated_attributes)
self.timeliness_validated_attributes ||= []
self.timeliness_validated_attributes += (attr_names - self.timeliness_validated_attributes)
end end
def timeliness_validation_for(attr_names, type=nil)
options = _merge_attributes(attr_names)
options.update(:type => type) if type
validates_with TimelinessValidator, options validates_with TimelinessValidator, options
end end
end end

View File

@ -0,0 +1,71 @@
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
def define_timeliness_methods(before_type_cast=false)
return if timeliness_validated_attributes.blank?
timeliness_validated_attributes.each do |attr_name|
define_attribute_timeliness_methods(attr_name, before_type_cast)
end
end
def 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
def define_attribute_timeliness_methods(attr_name, before_type_cast=false)
define_timeliness_write_method(attr_name)
define_timeliness_before_type_cast_method(attr_name) if before_type_cast
end
def define_timeliness_write_method(attr_name)
generated_timeliness_methods.module_eval <<-STR, __FILE__, __LINE__ + 1
def #{attr_name}=(value)
@timeliness_cache ||= {}
@timeliness_cache['#{attr_name}'] = value
@attributes['#{attr_name}'] = super
end
STR
end
def define_timeliness_before_type_cast_method(attr_name)
generated_timeliness_methods.module_eval <<-STR, __FILE__, __LINE__ + 1
def #{attr_name}_before_type_cast
read_timeliness_attribute_before_type_cast('#{attr_name}')
end
STR
end
end
def read_timeliness_attribute_before_type_cast(attr_name)
@timeliness_cache && @timeliness_cache[attr_name] || @attributes[attr_name]
end
def _clear_timeliness_cache
@timeliness_cache = {}
end
end
end
end

View File

@ -3,44 +3,15 @@ module ValidatesTimeliness
module ActiveRecord module ActiveRecord
extend ActiveSupport::Concern extend ActiveSupport::Concern
module ClassMethods def read_timeliness_attribute_before_type_cast(attr_name)
def define_attribute_methods read_attribute_before_type_cast(attr_name)
super
# Define write method and before_type_cast method
define_timeliness_methods(true)
end
def timeliness_attribute_timezone_aware?(attr_name)
attr_name = attr_name.to_s
create_time_zone_conversion_attribute?(attr_name, columns_hash[attr_name])
end
def timeliness_attribute_type(attr_name)
columns_hash[attr_name.to_s].type
end
def timeliness_type_cast_code(attr_name, var_name)
type = timeliness_attribute_type(attr_name)
<<-END
#{super}
#{var_name} = #{var_name}.to_date if #{var_name} && :#{type} == :date
END
end
end
module InstanceMethods
def reload(*args)
_clear_timeliness_cache
super
end
end end
end end
end end
end end
class ActiveRecord::Base ActiveSupport.on_load(:active_record) do
include ValidatesTimeliness::AttributeMethods include ValidatesTimeliness::AttributeMethods
include ValidatesTimeliness::ORM::ActiveRecord include ValidatesTimeliness::ORM::ActiveRecord
end end

View File

@ -1,57 +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
# 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_type_cast_code(attr_name, var_name)
type = timeliness_attribute_type(attr_name)
"#{var_name} = Timeliness::Parser.parse(value, :#{type})"
end
def timeliness_attribute_type(attr_name)
{
Date => :date,
Time => :datetime,
DateTime => :datetime
}[fields[attr_name.to_s].type] || :datetime
end
end
end
end
end
module Mongoid::Document
# Due to how Mongoid misuses ActiveSupport::Concern,
# the only way to override a core component method is
# using an append_features hook.
#
module TimelinessConcern
def append_features(base)
super
base.send :include, ValidatesTimeliness::AttributeMethods
base.send :include, ValidatesTimeliness::ORM::Mongoid
end
end
extend TimelinessConcern
def reload_with_timeliness
_clear_timeliness_cache
reload_without_timeliness
end
alias_method_chain :reload, :timeliness
end

View File

@ -3,12 +3,21 @@ module ValidatesTimeliness
initializer "validates_timeliness.initialize_active_record", :after => 'active_record.initialize_timezone' do initializer "validates_timeliness.initialize_active_record", :after => 'active_record.initialize_timezone' do
ActiveSupport.on_load(:active_record) do ActiveSupport.on_load(:active_record) do
ValidatesTimeliness.default_timezone = ActiveRecord::Base.default_timezone ValidatesTimeliness.default_timezone = ActiveRecord::Base.default_timezone
ValidatesTimeliness.extend_orms = [ :active_record ] ValidatesTimeliness.extend_orms << :active_record
ValidatesTimeliness.load_orms
end end
end end
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

View File

@ -1,10 +1,9 @@
require 'active_model'
require 'active_model/validator' require 'active_model/validator'
module ValidatesTimeliness module ValidatesTimeliness
class Validator < ActiveModel::EachValidator class Validator < ActiveModel::EachValidator
include Conversion attr_reader :type, :attributes, :converter
attr_reader :type
RESTRICTIONS = { RESTRICTIONS = {
:is_at => :==, :is_at => :==,
@ -22,26 +21,46 @@ module ValidatesTimeliness
RESTRICTION_ERROR_MESSAGE = "Error occurred validating %s for %s restriction:\n%s" RESTRICTION_ERROR_MESSAGE = "Error occurred validating %s for %s restriction:\n%s"
def self.kind
:timeliness
end
def initialize(options) def initialize(options)
@type = options.delete(:type) || :datetime @type = options.delete(:type) || :datetime
@allow_nil, @allow_blank = options.delete(:allow_nil), options.delete(:allow_blank) @allow_nil, @allow_blank = options.delete(:allow_nil), options.delete(:allow_blank)
if range = options.delete(:between) if range = options.delete(:between)
raise ArgumentError, ":between must be a Range or an Array" unless range.is_a?(Range) || range.is_a?(Array) raise ArgumentError, ":between must be a Range or an Array" unless range.is_a?(Range) || range.is_a?(Array)
options[:on_or_after], options[:on_or_before] = range.first, range.last options[:on_or_after] = range.first
if range.is_a?(Range) && range.exclude_end?
options[:before] = range.last
else
options[:on_or_before] = range.last
end
end 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
def setup_timeliness_validated_attributes(model)
if model.respond_to?(:timeliness_validated_attributes)
model.timeliness_validated_attributes ||= []
model.timeliness_validated_attributes |= attributes
end
end 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?)
@timezone_aware = timezone_aware?(record, attr_name) @converter = initialize_converter(record, attr_name)
value = parse(raw_value) if value.is_a?(String) || options[:format]
value = type_cast_value(value, @type) value = @converter.parse(raw_value) if value.is_a?(String) || options[:format]
value = @converter.type_cast_value(value)
add_error(record, attr_name, :"invalid_#{@type}") and return if value.blank? add_error(record, attr_name, :"invalid_#{@type}") and return if value.blank?
@ -51,7 +70,7 @@ module ValidatesTimeliness
def validate_restrictions(record, attr_name, value) def validate_restrictions(record, attr_name, value)
@restrictions_to_check.each do |restriction| @restrictions_to_check.each do |restriction|
begin begin
restriction_value = type_cast_value(evaluate_option_value(options[restriction], record), @type) restriction_value = @converter.type_cast_value(@converter.evaluate(options[restriction], record))
unless value.send(RESTRICTIONS[restriction], restriction_value) unless value.send(RESTRICTIONS[restriction], restriction_value)
add_error(record, attr_name, restriction, restriction_value) and break add_error(record, attr_name, restriction, restriction_value) and break
end end
@ -66,7 +85,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
@ -76,13 +95,22 @@ 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) record.read_timeliness_attribute_before_type_cast(attr_name.to_s)
end end
def timezone_aware?(record, attr_name) def time_zone_aware?(record, attr_name)
record.class.respond_to?(:timeliness_attribute_timezone_aware?) && record.class.respond_to?(:skip_time_zone_conversion_for_attributes) &&
record.class.timeliness_attribute_timezone_aware?(attr_name) !record.class.skip_time_zone_conversion_for_attributes.include?(attr_name.to_sym)
end
def initialize_converter(record, attr_name)
ValidatesTimeliness::Converter.new(
type: @type,
time_zone_aware: time_zone_aware?(record, attr_name),
format: options[:format],
ignore_usec: options[:ignore_usec]
)
end end
end end

View File

@ -1,3 +1,3 @@
module ValidatesTimeliness module ValidatesTimeliness
VERSION = '3.0.6' VERSION = '5.0.0.beta2'
end end

View File

@ -1,17 +1,21 @@
require 'rspec' require 'rspec'
require 'byebug'
require 'active_model' require 'active_model'
require 'active_model/validations' 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 +28,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
@ -49,9 +49,7 @@ class Person
attribute :birth_date, :date attribute :birth_date, :date
attribute :birth_time, :time attribute :birth_time, :time
attribute :birth_datetime, :datetime attribute :birth_datetime, :datetime
validates_date :birth_date
validates_time :birth_time
validates_datetime :birth_datetime
define_attribute_methods model_attributes.keys define_attribute_methods model_attributes.keys
end end
@ -59,8 +57,10 @@ class PersonWithShim < Person
include TestModelShim include TestModelShim
end end
ActiveRecord::Base.default_timezone = :utc
ActiveRecord::Base.time_zone_aware_attributes = true ActiveRecord::Base.time_zone_aware_attributes = true
ActiveRecord::Base.establish_connection({:adapter => 'sqlite3', :database => ':memory:'}) ActiveRecord::Base.establish_connection({:adapter => 'sqlite3', :database => ':memory:'})
ActiveRecord::Base.time_zone_aware_types = [:datetime, :time]
ActiveRecord::Migration.verbose = false ActiveRecord::Migration.verbose = false
ActiveRecord::Schema.define(:version => 1) do ActiveRecord::Schema.define(:version => 1) do
create_table :employees, :force => true do |t| create_table :employees, :force => true do |t|
@ -73,12 +73,10 @@ ActiveRecord::Schema.define(:version => 1) do
end end
class Employee < ActiveRecord::Base class Employee < ActiveRecord::Base
validates_date :birth_date
validates_time :birth_time
validates_datetime :birth_datetime
define_attribute_methods
attr_accessor :redefined_birth_date_called attr_accessor :redefined_birth_date_called
validates_date :birth_date, :allow_nil => true
validates_time :birth_time, :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
@ -86,17 +84,17 @@ class Employee < ActiveRecord::Base
end end
end end
Rspec.configure do |c| RSpec.configure do |c|
c.mock_with :rspec c.mock_with :rspec
c.include(RspecTagMatchers) c.include(TagMatcher)
c.include(ModelHelpers) c.include(ModelHelpers)
c.include(ConfigHelper) c.include(ConfigHelper)
c.before do c.before do
Person.reset_callbacks(:validate) reset_validation_setup_for(Person)
PersonWithShim.timeliness_validated_attributes = [] reset_validation_setup_for(PersonWithShim)
Person._validators.clear
Employee.reset_callbacks(:validate)
Employee.timeliness_validated_attributes = []
Employee._validators.clear
end end
c.filter_run_excluding :active_record => lambda {|version|
!(::ActiveRecord::VERSION::STRING.to_s =~ /^#{version.to_s}/)
}
end end

View File

@ -10,6 +10,16 @@ module ConfigHelper
ValidatesTimeliness.send(:"#{preference_name}=", old_value) ValidatesTimeliness.send(:"#{preference_name}=", old_value)
end end
def reset_validation_setup_for(model_class)
model_class.reset_callbacks(:validate)
model_class._validators.clear
model_class.timeliness_validated_attributes = [] if model_class.respond_to?(:timeliness_validated_attributes)
model_class.undefine_attribute_methods
# This is a hack to avoid a disabled super method error message after an undef
model_class.instance_variable_set(:@generated_attribute_methods, nil)
model_class.instance_variable_set(:@generated_timeliness_methods, nil)
end
module ClassMethods module ClassMethods
def with_config(preference_name, temporary_value) def with_config(preference_name, temporary_value)
original_config_value = ValidatesTimeliness.send(preference_name) original_config_value = ValidatesTimeliness.send(preference_name)

View File

@ -3,22 +3,21 @@ 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
def with_each_person_value(attr_name, values) def with_each_person_value(attr_name, values)
record = Person.new record = Person.new
values = [values] unless values.is_a?(Array) Array.wrap(values).each do |value|
values.each do |value|
record.send("#{attr_name}=", value) record.send("#{attr_name}=", value)
yield record, value yield record, value
end end

View 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

View File

@ -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
@ -26,12 +25,14 @@ module TestModel
def type_cast(attr_name, value) def type_cast(attr_name, value)
return value unless value.is_a?(String) return value unless value.is_a?(String)
value.send("to_#{model_attributes[attr_name.to_sym]}") rescue nil type_name = model_attributes[attr_name.to_sym]
type = ActiveModel::Type.lookup(type_name)
type.cast(value)
end end
end end
def initialize(attributes = nil) def initialize(attributes = nil)
@attributes = self.class.model_attributes.inject({}) do |hash, column| @attributes = self.class.model_attributes.keys.inject({}) do |hash, column|
hash[column.to_s] = nil hash[column.to_s] = nil
hash hash
end end
@ -39,7 +40,7 @@ module TestModel
end end
def attributes def attributes
@attributes.keys @attributes
end end
def attributes=(new_attributes={}) def attributes=(new_attributes={})
@ -49,14 +50,12 @@ module TestModel
end end
def method_missing(method_id, *args, &block) def method_missing(method_id, *args, &block)
if !self.class.attribute_methods_generated? if !matched_attribute_method(method_id.to_s).nil?
self.class.define_attribute_methods self.class.model_attributes.keys.map(&:to_s) self.class.define_attribute_methods self.class.model_attributes.keys
method_name = method_id.to_s
send(method_id, *args, &block) send(method_id, *args, &block)
else else
super super
end end
end end
end end

View File

@ -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
context "with plugin parser" do context "with plugin parser" do
@ -55,7 +53,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
@ -65,7 +63,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

View File

@ -1,182 +1,197 @@
require 'spec_helper' RSpec.describe ValidatesTimeliness::Converter do
subject(:converter) { described_class.new(type: type, time_zone_aware: time_zone_aware, ignore_usec: ignore_usec) }
describe ValidatesTimeliness::Conversion do
include ValidatesTimeliness::Conversion
let(:options) { Hash.new } let(:options) { Hash.new }
let(:type) { :date }
let(:time_zone_aware) { false }
let(:ignore_usec) { false }
before do before do
Timecop.freeze(Time.mktime(2010, 1, 1, 0, 0, 0)) Timecop.freeze(Time.mktime(2010, 1, 1, 0, 0, 0))
end end
delegate :type_cast_value, :evaluate, :parse, :dummy_time, to: :converter
describe "#type_cast_value" do describe "#type_cast_value" do
describe "for date type" do describe "for date type" do
let(:type) { :date }
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))).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))).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))).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)).to eq(nil)
end end
end end
describe "for time type" do describe "for time type" do
let(:type) { :time }
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))).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))).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))).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))).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)).to eq(nil)
end end
end end
describe "for datetime type" do describe "for datetime type" do
let(:type) { :datetime }
let(:time_zone_aware) { true }
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))).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))).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))).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
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)
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)).to eq(nil)
end end
end end
describe "ignore_usec option" do describe "ignore_usec option" do
let(:options) { {:ignore_usec => true} } let(:type) { :datetime }
let(:ignore_usec) { true }
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)).to eq(Time.utc(2010, 1, 1, 12, 34, 56))
end end
context do
let(:time_zone_aware) { true }
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
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)
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 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 expect(dummy_time(Time.utc(2000, 1, 1, 12, 34, 56))).to eq(Time.zone.local(2000, 1, 1, 23, 34, 56))
dummy_time(Time.utc(2000, 1, 1, 12, 34, 56)).should == 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
end end
describe "#evaluate_option_value" do describe "#evaluate" do
let(:person) { Person.new } let(:person) { Person.new }
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(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(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(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(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(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_datetime = value
evaluate_option_value(:birth_time, person).should == value expect(evaluate(:birth_datetime, 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(value, person)).to eq(Time.utc(2010,1,1,12,0,0))
end end
context do
let(:converter) { described_class.new(type: :date, time_zone_aware: true) }
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
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(value, person)).to eq(Time.zone.local(2010,1,1,12,0,0))
end
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-11-12 13:00:00'
evaluate_option_value(lambda { value }, person).should == Time.utc(2010,1,1,12,0,0) expect(evaluate(lambda { value }, person)).to eq(Time.utc(2010,11,12,13,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 = '13: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(:birth_time, person)).to eq(Time.utc(2000,1,1,13,0,0))
end end
context "restriction shorthand" do context "restriction shorthand" do
@ -185,17 +200,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(: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(: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(:now, person)).to eq(time)
end end
end end
end end
@ -205,7 +220,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
@ -214,21 +229,19 @@ describe ValidatesTimeliness::Conversion do
with_config(:use_plugin_parser, false) with_config(:use_plugin_parser, false)
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 expect(Timeliness::Parser).to_not receive(:parse)
Time.zone.should_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
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

View File

@ -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
@ -14,48 +12,48 @@ describe ValidatesTimeliness::Extensions::DateTimeSelect do
describe "datetime_select" do describe "datetime_select" do
it "should use param values when attribute is nil" do it "should use param values when attribute is nil" do
@params["person"] = { @params["person"] = {
"birth_datetime(1i)" => 2009, "birth_datetime(1i)" => '2009',
"birth_datetime(2i)" => 2, "birth_datetime(2i)" => '2',
"birth_datetime(3i)" => 29, "birth_datetime(3i)" => '29',
"birth_datetime(4i)" => 12, "birth_datetime(4i)" => '12',
"birth_datetime(5i)" => 13, "birth_datetime(5i)" => '13',
"birth_datetime(6i)" => 14, "birth_datetime(6i)" => '14',
} }
person.birth_datetime = nil person.birth_datetime = nil
@output = datetime_select(:person, :birth_datetime, :include_blank => true, :include_seconds => true) @output = datetime_select(:person, :birth_datetime, include_blank: true, include_seconds: true)
should_have_datetime_selected(:birth_datetime, :year => 2009, :month => 'February', :day => 29, :hour => 12, :min => 13, :sec => 14) should_have_datetime_selected(:birth_datetime, year: 2009, month: 'February', day: 29, hour: 12, min: 13, sec: 14)
end end
it "should override object values and use params if present" do it "should override object values and use params if present" do
@params["person"] = { @params["person"] = {
"birth_datetime(1i)" => 2009, "birth_datetime(1i)" => '2009',
"birth_datetime(2i)" => 2, "birth_datetime(2i)" => '2',
"birth_datetime(3i)" => 29, "birth_datetime(3i)" => '29',
"birth_datetime(4i)" => 12, "birth_datetime(4i)" => '12',
"birth_datetime(5i)" => 13, "birth_datetime(5i)" => '13',
"birth_datetime(6i)" => 14, "birth_datetime(6i)" => '14',
} }
person.birth_datetime = "2010-01-01 15:16:17" person.birth_datetime = "2010-01-01 15:16:17"
@output = datetime_select(:person, :birth_datetime, :include_blank => true, :include_seconds => true) @output = datetime_select(:person, :birth_datetime, include_blank: true, include_seconds: true)
should_have_datetime_selected(:birth_datetime, :year => 2009, :month => 'February', :day => 29, :hour => 12, :min => 13, :sec => 14) should_have_datetime_selected(:birth_datetime, year: 2009, month: 'February', day: 29, hour: 12, min: 13, sec: 14)
end end
it "should use attribute values from object if no params" do it "should use attribute values from object if no params" do
person.birth_datetime = "2009-01-02 12:13:14" person.birth_datetime = "2009-01-02 12:13:14"
@output = datetime_select(:person, :birth_datetime, :include_blank => true, :include_seconds => true) @output = datetime_select(:person, :birth_datetime, include_blank: true, include_seconds: true)
should_have_datetime_selected(:birth_datetime, :year => 2009, :month => 'January', :day => 2, :hour => 12, :min => 13, :sec => 14) should_have_datetime_selected(:birth_datetime, year: 2009, month: 'January', day: 2, hour: 12, min: 13, sec: 14)
end end
it "should use attribute values if params does not contain attribute params" do it "should use attribute values if params does not contain attribute params" do
person.birth_datetime = "2009-01-02 12:13:14" person.birth_datetime = "2009-01-02 12:13:14"
@params["person"] = { } @params["person"] = { }
@output = datetime_select(:person, :birth_datetime, :include_blank => true, :include_seconds => true) @output = datetime_select(:person, :birth_datetime, include_blank: true, include_seconds: true)
should_have_datetime_selected(:birth_datetime, :year => 2009, :month => 'January', :day => 2, :hour => 12, :min => 13, :sec => 14) should_have_datetime_selected(:birth_datetime, year: 2009, month: 'January', day: 2, hour: 12, min: 13, sec: 14)
end end
it "should not select values when attribute value is nil and has no param values" do it "should not select values when attribute value is nil and has no param values" do
person.birth_datetime = nil person.birth_datetime = nil
@output = datetime_select(:person, :birth_datetime, :include_blank => true, :include_seconds => true) @output = datetime_select(:person, :birth_datetime, include_blank: true, include_seconds: true)
should_not_have_datetime_selected(:birth_datetime, :year, :month, :day, :hour, :min, :sec) should_not_have_datetime_selected(:birth_datetime, :year, :month, :day, :hour, :min, :sec)
end end
end end
@ -63,44 +61,56 @@ describe ValidatesTimeliness::Extensions::DateTimeSelect do
describe "date_select" do describe "date_select" do
it "should use param values when attribute is nil" do it "should use param values when attribute is nil" do
@params["person"] = { @params["person"] = {
"birth_date(1i)" => 2009, "birth_date(1i)" => '2009',
"birth_date(2i)" => 2, "birth_date(2i)" => '2',
"birth_date(3i)" => 29, "birth_date(3i)" => '29',
} }
person.birth_date = nil person.birth_date = nil
@output = date_select(:person, :birth_date, :include_blank => true, :include_seconds => true) @output = date_select(:person, :birth_date, include_blank: true)
should_have_datetime_selected(:birth_date, :year => 2009, :month => 'February', :day => 29) should_have_datetime_selected(:birth_date, year: 2009, month: 'February', day: 29)
end end
it "should override object values and use params if present" do it "should override object values and use params if present" do
@params["person"] = { @params["person"] = {
"birth_date(1i)" => 2009, "birth_date(1i)" => '2009',
"birth_date(2i)" => 2, "birth_date(2i)" => '2',
"birth_date(3i)" => 29, "birth_date(3i)" => '29',
} }
person.birth_date = "2009-03-01" person.birth_date = "2009-03-01"
@output = date_select(:person, :birth_date, :include_blank => true, :include_seconds => true) @output = date_select(:person, :birth_date, include_blank: true)
should_have_datetime_selected(:birth_date, :year => 2009, :month => 'February', :day => 29) should_have_datetime_selected(:birth_date, year: 2009, month: 'February', day: 29)
end end
it "should select attribute values from object if no params" do it "should select attribute values from object if no params" do
person.birth_date = "2009-01-02" person.birth_date = "2009-01-02"
@output = date_select(:person, :birth_date, :include_blank => true, :include_seconds => true) @output = date_select(:person, :birth_date, include_blank: true)
should_have_datetime_selected(:birth_date, :year => 2009, :month => 'January', :day => 2) should_have_datetime_selected(:birth_date, year: 2009, month: 'January', day: 2)
end end
it "should select attribute values if params does not contain attribute params" do it "should select attribute values if params does not contain attribute params" do
person.birth_date = "2009-01-02" person.birth_date = "2009-01-02"
@params["person"] = { } @params["person"] = { }
@output = date_select(:person, :birth_date, :include_blank => true, :include_seconds => true) @output = date_select(:person, :birth_date, include_blank: true)
should_have_datetime_selected(:birth_date, :year => 2009, :month => 'January', :day => 2) should_have_datetime_selected(:birth_date, year: 2009, month: 'January', day: 2)
end end
it "should not select values when attribute value is nil and has no param values" do it "should not select values when attribute value is nil and has no param values" do
person.birth_date = nil person.birth_date = nil
@output = date_select(:person, :birth_date, :include_blank => true, :include_seconds => true) @output = date_select(:person, :birth_date, include_blank: true)
should_not_have_datetime_selected(:birth_time, :year, :month, :day) should_not_have_datetime_selected(:birth_time, :year, :month, :day)
end end
it "should allow the day part to be discarded" do
@params["person"] = {
"birth_date(1i)" => '2009',
"birth_date(2i)" => '2',
}
@output = date_select(:person, :birth_date, include_blank: true, discard_day: true)
should_have_datetime_selected(:birth_date, year: 2009, month: 'February')
should_not_have_datetime_selected(:birth_time, :day)
expect(@output).to have_tag("input[id=person_birth_date_3i][type=hidden][value='1']")
end
end end
describe "time_select" do describe "time_select" do
@ -110,42 +120,43 @@ describe ValidatesTimeliness::Extensions::DateTimeSelect do
it "should use param values when attribute is nil" do it "should use param values when attribute is nil" do
@params["person"] = { @params["person"] = {
"birth_time(1i)" => 2000, "birth_time(1i)" => '2000',
"birth_time(2i)" => 1, "birth_time(2i)" => '1',
"birth_time(3i)" => 1, "birth_time(3i)" => '1',
"birth_time(4i)" => 12, "birth_time(4i)" => '12',
"birth_time(5i)" => 13, "birth_time(5i)" => '13',
"birth_time(6i)" => 14, "birth_time(6i)" => '14',
} }
person.birth_time = nil person.birth_time = nil
@output = time_select(:person, :birth_time, :include_blank => true, :include_seconds => true) @output = time_select(:person, :birth_time, include_blank: true, include_seconds: true)
should_have_datetime_selected(:birth_time, :hour => 12, :min => 13, :sec => 14) should_have_datetime_selected(:birth_time, hour: 12, min: 13, sec: 14)
end end
it "should select attribute values from object if no params" do it "should select attribute values from object if no params" do
person.birth_time = "2000-01-01 12:13:14" person.birth_time = "2000-01-01 12:13:14"
@output = time_select(:person, :birth_time, :include_blank => true, :include_seconds => true) @output = time_select(:person, :birth_time, include_blank: true, include_seconds: true)
should_have_datetime_selected(:birth_time, :hour => 12, :min => 13, :sec => 14) should_have_datetime_selected(:birth_time, hour: 12, min: 13, sec: 14)
end end
it "should not select values when attribute value is nil and has no param values" do it "should not select values when attribute value is nil and has no param values" do
person.birth_time = nil person.birth_time = nil
@output = time_select(:person, :birth_time, :include_blank => true, :include_seconds => true) @output = time_select(:person, :birth_time, include_blank: true, include_seconds: true)
should_not_have_datetime_selected(:birth_time, :hour, :min, :sec) should_not_have_datetime_selected(:birth_time, :hour, :min, :sec)
end end
end end
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

View File

@ -1,33 +1,43 @@
require 'spec_helper' RSpec.describe 'ValidatesTimeliness::Extensions::MultiparameterHandler' do
describe ValidatesTimeliness::Extensions::MultiparameterHandler do
let(:employee) { Employee.new }
context "time column" do context "time column" do
it 'should return string value for invalid date portion' do it 'should be nil invalid date portion' do
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 == '2000-02-31 12:00:00' expect(employee.birth_datetime).to be_nil
end end
it 'should return Time value for valid datetimes' do it 'should assign a Time value for valid datetimes' do
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 be_kind_of(Time) expect(employee.birth_datetime).to eq Time.zone.local(2000, 2, 28, 12, 0, 0)
end
it 'should be nil for incomplete date portion' do
employee = record_with_multiparameter_attribute(:birth_datetime, [2000, nil, nil])
expect(employee.birth_datetime).to be_nil
end end
end end
context "date column" do context "date column" do
it 'should return string value for invalid date' do it 'should assign nil for invalid date' do
multiparameter_attribute(:birth_date, [2000, 2, 31]) employee = record_with_multiparameter_attribute(:birth_date, [2000, 2, 31])
employee.birth_date_before_type_cast.should == '2000-02-31' expect(employee.birth_date).to be_nil
end end
it 'should return Date value for valid date' do it 'should assign a Date value for valid date' do
multiparameter_attribute(:birth_date, [2000, 2, 28]) employee = record_with_multiparameter_attribute(:birth_date, [2000, 2, 28])
employee.birth_date_before_type_cast.should be_kind_of(Date) expect(employee.birth_date).to eq Date.new(2000, 2, 28)
end
it 'should assign hash values for incomplete date' do
employee = record_with_multiparameter_attribute(:birth_date, [2000, nil, nil])
expect(employee.birth_date).to be_nil
end end
end end
def multiparameter_attribute(name, values) def record_with_multiparameter_attribute(name, values)
employee.send(:execute_callstack_for_multiparameter_attributes, name.to_s => values) hash = {}
values.each_with_index {|value, index| hash["#{name}(#{index+1}i)"] = value.to_s }
Employee.new(hash)
end end
end end

View File

@ -1,21 +1,28 @@
require 'spec_helper' RSpec.describe ValidatesTimeliness, 'HelperMethods' do
let(:record) { Person.new }
describe ValidatesTimeliness, 'HelperMethods' do
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
Person.new.should respond_to(:validates_date) expect(record).to respond_to(:validates_date)
Person.new.should respond_to(:validates_time) expect(record).to respond_to(:validates_time)
Person.new.should respond_to(:validates_datetime) expect(record).to respond_to(:validates_datetime)
end end
it 'should validate instance when validation method called' do it 'should validate instance using class validation defined' do
r = Person.new Person.validates_date :birth_date
r.validates_date :birth_date record.valid?
r.errors[:birth_date].should_not be_empty
expect(record.errors[:birth_date]).not_to be_empty
end
it 'should validate instance using instance valiation method' do
record.validates_date :birth_date
expect(record.errors[:birth_date]).not_to be_empty
end end
end end

View File

@ -1,100 +1,190 @@
require 'spec_helper' RSpec.describe ValidatesTimeliness, 'ActiveRecord' do
describe ValidatesTimeliness, 'ActiveRecord' do
context "validation methods" do context "validation methods" do
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
Employee.new.should respond_to(:validates_date) expect(record).to respond_to(:validates_date)
Employee.new.should respond_to(:validates_time) expect(record).to respond_to(:validates_time)
Employee.new.should respond_to(:validates_datetime) expect(record).to respond_to(:validates_datetime)
end
it "should validate a valid value string" do
record.birth_date = '2012-01-01'
record.valid?
expect(record.errors[:birth_date]).to be_empty
end
it "should validate a invalid value string" do
record.birth_date = 'not a date'
record.valid?
expect(record.birth_date_before_type_cast).to eq 'not a date'
expect(record.errors[:birth_date]).not_to be_empty
end
it "should validate a nil value" do
record.birth_date = nil
record.valid?
expect(record.errors[:birth_date]).to be_empty
end end
end end
it 'should determine type for attribute' do context 'attribute timezone awareness' do
Employee.timeliness_attribute_type(:birth_date).should == :date let(:klass) {
Class.new(ActiveRecord::Base) do
self.table_name = 'employees'
attr_accessor :some_date
attr_accessor :some_time
attr_accessor :some_datetime
validates_date :some_date
validates_time :some_time
validates_datetime :some_datetime
end
}
end end
context "attribute write method" do context "attribute write method" do
class EmployeeWithCache < ActiveRecord::Base class EmployeeWithCache < ActiveRecord::Base
set_table_name 'employees' self.table_name = 'employees'
validates_datetime :birth_datetime validates_date :birth_date, :allow_blank => true
end validates_time :birth_time, :allow_blank => true
validates_datetime :birth_datetime, :allow_blank => true
it 'should cache attribute raw value' do
r = EmployeeWithCache.new
r.birth_datetime = date_string = '2010-01-01'
r._timeliness_raw_value_for(:birth_datetime).should == date_string
end end
context "with plugin parser" do context "with plugin parser" do
with_config(:use_plugin_parser, true) with_config(:use_plugin_parser, true)
let(:record) { EmployeeWithParser.new }
class EmployeeWithParser < ActiveRecord::Base class EmployeeWithParser < ActiveRecord::Base
set_table_name 'employees' self.table_name = 'employees'
validates_date :birth_date validates_date :birth_date, :allow_blank => true
validates_datetime :birth_datetime validates_time :birth_time, :allow_blank => true
validates_datetime :birth_datetime, :allow_blank => true
end end
it 'should parse a string value' do before do
Timeliness::Parser.should_receive(:parse) allow(Timeliness::Parser).to receive(:parse).and_call_original
r = EmployeeWithParser.new
r.birth_date = '2010-01-01'
end end
context "for a date column" do context "for a date column" do
it 'should store a date value after parsing string' do it 'should parse a string value' do
r = EmployeeWithParser.new record.birth_date = '2010-01-01'
r.birth_date = '2010-01-01' expect(record.birth_date).to eq(Date.new(2010, 1, 1))
r.birth_date.should be_kind_of(Date) expect(Timeliness::Parser).to have_received(:parse)
r.birth_date.should == Date.new(2010, 1, 1) end
it 'should parse a invalid string value as nil' do
record.birth_date = 'not valid'
expect(record.birth_date).to be_nil
expect(Timeliness::Parser).to have_received(:parse)
end
it 'should store a Date value after parsing string' do
record.birth_date = '2010-01-01'
expect(record.birth_date).to be_kind_of(Date)
expect(record.birth_date).to eq Date.new(2010, 1, 1)
end
end
context "for a time column" do
around do |example|
time_zone_aware_types = ActiveRecord::Base.time_zone_aware_types.dup
example.call
ActiveRecord::Base.time_zone_aware_types = time_zone_aware_types
end
context 'timezone aware' do
with_config(:default_timezone, 'Australia/Melbourne')
before do
unless ActiveRecord::Base.time_zone_aware_types.include?(:time)
ActiveRecord::Base.time_zone_aware_types.push(:time)
end
end
it 'should parse a string value' do
record.birth_time = '12:30'
expect(record.birth_time).to eq('12:30'.in_time_zone)
expect(Timeliness::Parser).to have_received(:parse)
end
it 'should parse a invalid string value as nil' do
record.birth_time = 'not valid'
expect(record.birth_time).to be_nil
expect(Timeliness::Parser).to have_received(:parse)
end
it 'should store a Time value after parsing string' do
record.birth_time = '12:30'
expect(record.birth_time).to eq('12:30'.in_time_zone)
expect(record.birth_time.utc_offset).to eq '12:30'.in_time_zone.utc_offset
end
end
skip 'not timezone aware' do
before do
ActiveRecord::Base.time_zone_aware_types.delete(:time)
end
it 'should parse a string value' do
record.birth_time = '12:30'
expect(record.birth_time).to eq(Time.utc(2000,1,1,12,30))
expect(Timeliness::Parser).to have_received(:parse)
end
it 'should parse a invalid string value as nil' do
record.birth_time = 'not valid'
expect(record.birth_time).to be_nil
expect(Timeliness::Parser).to have_received(:parse)
end
it 'should store a Time value in utc' do
record.birth_time = '12:30'
expect(record.birth_time.utc_offset).to eq Time.now.utc.utc_offset
end
end end
end end
context "for a datetime column" do context "for a datetime column" do
with_config(:default_timezone, 'Australia/Melbourne') with_config(:default_timezone, 'Australia/Melbourne')
it 'should parse string into Time value' do it 'should parse a string value into Time value' do
r = EmployeeWithParser.new record.birth_datetime = '2010-01-01 12:00'
r.birth_datetime = '2010-01-01 12:00'
r.birth_datetime.should be_kind_of(Time) expect(record.birth_datetime).to eq Time.zone.local(2010,1,1,12,00)
expect(Timeliness::Parser).to have_received(:parse)
end
it 'should parse a invalid string value as nil' do
record.birth_datetime = 'not valid'
expect(record.birth_datetime).to be_nil
expect(Timeliness::Parser).to have_received(:parse)
end end
it 'should parse string as current timezone' do it 'should parse string as current timezone' do
r = EmployeeWithParser.new record.birth_datetime = '2010-06-01 12:00'
r.birth_datetime = '2010-06-01 12:00'
r.birth_datetime.utc_offset.should == Time.zone.utc_offset expect(record.birth_datetime.utc_offset).to eq Time.zone.utc_offset
end end
end end
end end
end end
context "cached value" do
it 'should be cleared on reload' do
r = Employee.create!
r.birth_date = '2010-01-01'
r.reload
r._timeliness_raw_value_for(:birth_date).should be_nil
end
end
context "before_type_cast method" do
it 'should be defined on class if ORM supports it' do
Employee.new.should respond_to(:birth_datetime_before_type_cast)
end
it 'should return original value' do
r = Employee.new
r.birth_datetime = date_string = '2010-01-01'
r.birth_datetime_before_type_cast.should == date_string
end
end
end end

View File

@ -1,105 +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|
name = "validates_timeliness_test"
host = "localhost"
config.master = Mongo::Connection.new.db(name)
config.persist_in_safe_mode = false
end
describe ValidatesTimeliness, 'Mongoid' do
class Article
::ValidatesTimeliness.use_plugin_parser = true
include Mongoid::Document
field :publish_date, :type => Date
field :publish_time, :type => Time
field :publish_datetime, :type => DateTime
validates_date :publish_date, :allow_nil => true
validates_time :publish_time, :allow_nil => true
validates_datetime :publish_datetime, :allow_nil => true
::ValidatesTimeliness.use_plugin_parser = false
end
context "validation methods" do
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
Article.new.should respond_to(:validates_date)
Article.new.should respond_to(:validates_time)
Article.new.should respond_to(:validates_datetime)
end
end
it 'should determine type for attribute' do
Article.timeliness_attribute_type(:publish_date).should == :date
end
context "attribute write method" do
it 'should cache attribute raw value' do
r = Article.new
r.publish_datetime = date_string = '2010-01-01'
r._timeliness_raw_value_for(:publish_datetime).should == date_string
end
context "with plugin parser" do
with_config(:use_plugin_parser, false)
it 'should parse a string value' do
Timeliness::Parser.should_receive(:parse)
r = Article.new
r.publish_date = '2010-01-01'
end
context "for a date column" do
it 'should store a date value after parsing string' do
r = Article.new
r.publish_date = '2010-01-01'
r.publish_date.should be_kind_of(Date)
r.publish_date.should == Date.new(2010, 1, 1)
end
end
context "for a datetime column" do
it 'should parse string into Time value' do
r = Article.new
r.publish_datetime = '2010-01-01 12:00'
r.publish_datetime.should be_kind_of(Time)
r.publish_datetime.should == Time.utc(2010,1,1,12,0)
end
end
end
end
context "cached value" do
it 'should be cleared on reload' do
r = Article.create!
r.publish_date = '2010-01-01'
r.reload
r._timeliness_raw_value_for(:publish_date).should be_nil
end
end
context "before_type_cast method" do
it 'should not be defined if ORM does not support it' do
Article.new.should_not respond_to(:publish_datetime_before_type_cast)
end
end
end
rescue LoadError
puts "Mongoid specs skipped. Mongoid not installed"
rescue StandardError
puts "Mongoid specs skipped. MongoDB connection failed."
end

View 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

View File

@ -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

View File

@ -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

View File

@ -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

View File

@ -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

View File

@ -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

View File

@ -1,23 +1,27 @@
require 'spec_helper' RSpec.describe ValidatesTimeliness::Validator do
describe ValidatesTimeliness::Validator do
NIL = [nil]
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
it 'should add attribute to timeliness attributes set' do
expect(PersonWithShim.timeliness_validated_attributes).not_to include(:birth_time)
PersonWithShim.validates :birth_time, :timeliness => {:is_at => "12:30"}
expect(PersonWithShim.timeliness_validated_attributes).to include(:birth_time)
end end
end end
@ -29,22 +33,33 @@ 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
it 'should not allow nil by default' do it 'should not allow nil by default' do
Person.validates_date :birth_date Person.validates_date :birth_date
invalid!(:birth_date, NIL, 'is not a valid date') invalid!(:birth_date, [nil], 'is not a valid date')
valid!(:birth_date, Date.today) valid!(:birth_date, Date.today)
end end
it 'should allow nil when true' do it 'should allow nil when true' do
Person.validates_date :birth_date, :allow_nil => true Person.validates_date :birth_date, :allow_nil => true
valid!(:birth_date, NIL) valid!(:birth_date, [nil])
end
context "with raw value cache" do
it "should not be valid with an invalid format" do
PersonWithShim.validates_date :birth_date, :allow_nil => true
p = PersonWithShim.new
p.birth_date = 'bogus'
expect(p).not_to be_valid
end
end end
end end
@ -59,6 +74,31 @@ describe ValidatesTimeliness::Validator do
Person.validates_date :birth_date, :allow_blank => true Person.validates_date :birth_date, :allow_blank => true
valid!(:birth_date, '') valid!(:birth_date, '')
end end
context "with raw value cache" do
it "should not be valid with an invalid format" do
PersonWithShim.validates_date :birth_date, :allow_blank => true
p = PersonWithShim.new
p.birth_date = 'bogus'
expect(p).not_to be_valid
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 end
describe ":between option" do describe ":between option" do
@ -66,8 +106,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)
@ -79,14 +119,27 @@ 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)
valid!(:birth_date, on_or_before) valid!(:birth_date, on_or_before)
end end
end end
describe "range with excluded end value" do
it 'should be split option into :on_or_after and :before values' do
on_or_after, before = Date.new(2010,1,1), Date.new(2010,1,3)
Person.validates_date :birth_date, :between => on_or_after...before
expect(Person.validators.first.options[:on_or_after]).to eq(on_or_after)
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, before, "must be before 2010-01-03")
valid!(:birth_date, on_or_after)
valid!(:birth_date, before - 1)
end
end
end end
describe ":ignore_usec option" do describe ":ignore_usec option" do
@ -118,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
@ -138,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 for :is_at restriction") 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
@ -161,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
@ -182,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

View File

@ -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

View File

@ -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 } 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.3"]) s.add_runtime_dependency(%q<timeliness>, [">= 0.3.10", "< 1"])
end end