changes_validator 0.0.1

Sign up to get free protection for your applications and to get access to all the features.
@@ -0,0 +1,7 @@
1
+ ---
2
+ SHA1:
3
+ metadata.gz: a9232b72a8a820e08048cd79cc4f970a1f74e7b4
4
+ data.tar.gz: 1dac8a68b32ae3f430ac4d7122b14175dbac8fcd
5
+ SHA512:
6
+ metadata.gz: 61a6eea8080594c5975109fdc8a6818b11f89b9d672607830880a84ccce4d42ef53d4dc37ab52a6b0d8cc61cbe9d3eda4885f1271e2f772209e35a0fe374d7c5
7
+ data.tar.gz: 36bd3f37116a16085165378ed4c3bf0cf17fa87d813b8880fa6ad6a75871ef4cc33926bd1e4cd1c8ecac152a797d222eeadae26de1debd615f38757694b34997
@@ -0,0 +1,17 @@
1
+ *.gem
2
+ *.rbc
3
+ .bundle
4
+ .config
5
+ .yardoc
6
+ Gemfile.lock
7
+ InstalledFiles
8
+ _yardoc
9
+ coverage
10
+ doc/
11
+ lib/bundler/man
12
+ pkg
13
+ rdoc
14
+ spec/reports
15
+ test/tmp
16
+ test/version_tmp
17
+ tmp
data/Gemfile ADDED
@@ -0,0 +1,4 @@
1
+ source 'https://rubygems.org'
2
+
3
+ # Specify your gem's dependencies in changes_validator.gemspec
4
+ gemspec
@@ -0,0 +1,22 @@
1
+ Copyright (c) 2013 Bogdan Gusiev
2
+
3
+ MIT License
4
+
5
+ Permission is hereby granted, free of charge, to any person obtaining
6
+ a copy of this software and associated documentation files (the
7
+ "Software"), to deal in the Software without restriction, including
8
+ without limitation the rights to use, copy, modify, merge, publish,
9
+ distribute, sublicense, and/or sell copies of the Software, and to
10
+ permit persons to whom the Software is furnished to do so, subject to
11
+ the following conditions:
12
+
13
+ The above copyright notice and this permission notice shall be
14
+ included in all copies or substantial portions of the Software.
15
+
16
+ THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
17
+ EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
18
+ MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
19
+ NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
20
+ LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
21
+ OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
22
+ WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
@@ -0,0 +1,67 @@
1
+ # ChangesValidator
2
+
3
+ ChangesValidator is the most minimalistic state machine implementation focused on validating a state changes rather than define API methods and logic.
4
+
5
+
6
+ ## Why ChangesValidator?
7
+
8
+ The best way to define API and logic in Ruby is using Ruby.
9
+ ChangesValidator does only validation that Object can move from one state to another.
10
+
11
+ ## Dependencies
12
+
13
+ ActiveModel
14
+
15
+ ## Installation
16
+
17
+ Add this line to your application's Gemfile:
18
+
19
+ gem 'changes_validator'
20
+
21
+ And then execute:
22
+
23
+ $ bundle
24
+
25
+ ## Usage
26
+
27
+
28
+ ### Basic setup
29
+
30
+ If the state machine's main goal is to validate changess than let's implement it as a validation:
31
+
32
+ ``` ruby
33
+ class Reward < AR::Base
34
+ validates! :state, :changes => {
35
+ nil => [:pending], # Initial state is always pending
36
+ :pending => [:approved, :rejected], # Pending can be changesed to to approved and rejected
37
+ :approved => :paid # Approved can only be changesed to paid
38
+ }
39
+ end
40
+ ```
41
+
42
+ Recommended to use with strict validation method: `validates!` as wrong state changes use to be programmer mistake but not user input mistake.
43
+ In this case exception will be raise and logged.
44
+
45
+
46
+ ### Advanced Options
47
+
48
+ * `:message` - validation message. Can have %{value} and %{old\_value} interpolation variables.
49
+ * Default: "Can not be changesed to %{value}"
50
+ * Example: "Can not be changesed from %{old\_value} to %{value}"
51
+ * `:allow_nil` - don't apply validator if value is nil
52
+ * `:allow_blank` - don't apply validator if value is blank
53
+ * `:if` - only apply validator if specified method return true
54
+ * `:unless` - only apply validator if specified method return false
55
+ * `:strict` - if true this validator will always raise when record is invalid
56
+ * Default:
57
+
58
+
59
+
60
+
61
+ ## Contributing
62
+
63
+ 1. Fork it
64
+ 2. Create your feature branch (`git checkout -b my-new-feature`)
65
+ 3. Commit your changes (`git commit -am 'Add some feature'`)
66
+ 4. Push to the branch (`git push origin my-new-feature`)
67
+ 5. Create new Pull Request
@@ -0,0 +1 @@
1
+ require "bundler/gem_tasks"
@@ -0,0 +1,26 @@
1
+ # coding: utf-8
2
+ lib = File.expand_path('../lib', __FILE__)
3
+ $LOAD_PATH.unshift(lib) unless $LOAD_PATH.include?(lib)
4
+ require 'changes_validator/version'
5
+
6
+ Gem::Specification.new do |spec|
7
+ spec.name = "changes_validator"
8
+ spec.version = ChangesValidator::VERSION
9
+ spec.authors = ["Bogdan Gusiev"]
10
+ spec.email = ["agresso@gmail.com"]
11
+ spec.description = %q{ActiveModel validator that acts like state machine}
12
+ spec.summary = %q{This validator allows to specify a mapping of allowed changess for given attribute and validate that this attribute can only be transfered according to changess map}
13
+ spec.homepage = "https://github.com/bogdan/changes_validator"
14
+ spec.license = "MIT"
15
+
16
+ spec.files = `git ls-files`.split($/)
17
+ spec.executables = spec.files.grep(%r{^bin/}) { |f| File.basename(f) }
18
+ spec.test_files = spec.files.grep(%r{^(test|spec|features)/})
19
+ spec.require_paths = ["lib"]
20
+
21
+ spec.add_runtime_dependency "activemodel"
22
+ spec.add_development_dependency "bundler", "~> 1.3"
23
+ spec.add_development_dependency "rake"
24
+ spec.add_development_dependency "rspec"
25
+ spec.add_development_dependency "byebug" if RUBY_VERSION >= "2.0"
26
+ end
@@ -0,0 +1,39 @@
1
+ require "active_model"
2
+
3
+ class ChangesValidator < ActiveModel::EachValidator
4
+
5
+ def initialize(*)
6
+ super
7
+ normalize_options
8
+ end
9
+
10
+ def validate_each(record, attribute, value)
11
+ ensure_supports_dirty(record)
12
+ changes = record.changes[attribute.to_s]
13
+ return unless changes
14
+ changess = options[:with]
15
+ start = changes.first
16
+ destination = value
17
+
18
+ allowed_changess = changess[start]
19
+ if !allowed_changess || !Array(allowed_changess).map {|s| s.to_s }.include?(destination.to_s)
20
+ record.errors.add(attribute, :changes, :old_value => start, :message => options[:message])
21
+ end
22
+ end
23
+
24
+ def ensure_supports_dirty(record)
25
+ unless record.respond_to?(:changes)
26
+ raise ConfigurationError, "ChangesValidator can only be applied to model with dirty support (ActiveModel::Dirty)"
27
+ end
28
+ end
29
+
30
+ class ConfigurationError < Exception; end
31
+
32
+ def normalize_options
33
+ unless options[:with]
34
+ @options = {:with => options}
35
+ end
36
+ end
37
+ end
38
+
39
+ I18n.load_path << File.expand_path('../../locales/en.yml', __FILE__)
@@ -0,0 +1,4 @@
1
+ require "active_model"
2
+ class ChangesValidator < ActiveModel::EachValidator
3
+ VERSION = "0.0.1"
4
+ end
@@ -0,0 +1,5 @@
1
+ en:
2
+ errors:
3
+ # The values :model, :attribute, :value and :old_value are always available for interpolation
4
+ messages:
5
+ changes: "can't be changesed to %{value}"
@@ -0,0 +1,112 @@
1
+ require "spec_helper"
2
+
3
+ class User
4
+ include ActiveModel::Dirty
5
+ include ActiveModel::Validations
6
+
7
+ define_attribute_methods [:state]
8
+
9
+ def state
10
+ @state
11
+ end
12
+
13
+ def state=(val)
14
+ state_will_change! unless val == @state
15
+ @state = val
16
+ end
17
+
18
+ def save!
19
+ raise 'invalid' unless valid?
20
+ @previously_changed = changes
21
+ @changed_attributes.clear
22
+ end
23
+
24
+ def truthly?
25
+ true
26
+ end
27
+
28
+ def falsy?
29
+ false
30
+ end
31
+
32
+
33
+ end
34
+
35
+
36
+ describe ChangesValidator do
37
+ before do
38
+ User.clear_validators!
39
+ end
40
+ let!(:user) { User.new }
41
+ it "should work" do
42
+ user.should be_valid
43
+ end
44
+
45
+ it "should support state change according to changess table" do
46
+ User.validates :state, :changes => {nil => [:pending, :approved]}
47
+ user.state = "pending"
48
+ user.should be_valid
49
+ user.state = "approved"
50
+ user.should be_valid
51
+ end
52
+
53
+ it "should return record invalid if changesed to wrong state" do
54
+ User.validates :state, :changes => {nil => [:pending, :approved]}
55
+ user.state = "voided"
56
+ user.should_not be_valid
57
+ user.errors[:state].should_not be_empty
58
+ user.errors[:state].first.should == "can't be changesed to voided"
59
+ end
60
+
61
+
62
+ it "should raise exception when model don't support dirty" do
63
+ klass = Class.new {
64
+ include ActiveModel::Validations
65
+ attr_accessor :state
66
+ validates :state, :changes => {nil => :pending}
67
+ }
68
+ proc {
69
+ klass.new.valid?
70
+ }.should raise_error(ChangesValidator::ConfigurationError)
71
+ end
72
+ it "should be able to use custom message" do
73
+ User.validates :state, :changes => {:with => {nil => [:pending, :approved]}, :message => "can not be changesed like this"}
74
+ user.state = 'declined'
75
+ user.should_not be_valid
76
+ user.errors[:state].should_not be_empty
77
+ user.errors[:state].first.should == "can not be changesed like this"
78
+ end
79
+ it "should be able to use old_value interpolation in custom message" do
80
+ User.validates :state, :changes => {:with => {nil => :pending, :pending => [:approved]}, :message => "can not be changesed from %{old_value} to %{value}"}
81
+ user.state = 'pending'
82
+ user.save!
83
+ user.state = 'declined'
84
+ user.should_not be_valid
85
+ user.errors[:state].should_not be_empty
86
+ user.errors[:state].first.should == "can not be changesed from pending to declined"
87
+ end
88
+ it "should be able to use allow_nil option" do
89
+ User.validates :state, :changes => {:with => {nil => :pending, :pending => [:approved]}}, :allow_nil => true
90
+ user.state = 'pending'
91
+ user.save!
92
+ user.state = nil
93
+ user.should be_valid
94
+ end
95
+ it "should be able to use allow_blank option" do
96
+ User.validates :state, :changes => {:with => {nil => :pending, :pending => [:approved]}}, :allow_blank => true
97
+ user.state = 'pending'
98
+ user.save!
99
+ user.state = ' '
100
+ user.should be_valid
101
+ end
102
+ it "should be able to use if option" do
103
+ User.validates :state, :changes => {:with => {nil => :pending, :pending => [:approved]}}, :if => :falsy?
104
+ user.state = 'approved'
105
+ user.should be_valid
106
+ end
107
+ it "should be able to use unless option" do
108
+ User.validates :state, :changes => {:with => {nil => :pending, :pending => [:approved]}}, :unless => :truthly?
109
+ user.state = 'approved'
110
+ user.should be_valid
111
+ end
112
+ end
@@ -0,0 +1,9 @@
1
+ $:.unshift(File.join(File.dirname(__FILE__), '..', 'lib'))
2
+ $:.unshift(File.dirname(__FILE__))
3
+ require 'rubygems'
4
+ require 'bundler/setup'
5
+ require 'changes_validator' # and any other gems you need
6
+
7
+ RSpec.configure do |config|
8
+ I18n.enforce_available_locales = true
9
+ end
metadata ADDED
@@ -0,0 +1,129 @@
1
+ --- !ruby/object:Gem::Specification
2
+ name: changes_validator
3
+ version: !ruby/object:Gem::Version
4
+ version: 0.0.1
5
+ platform: ruby
6
+ authors:
7
+ - Bogdan Gusiev
8
+ autorequire:
9
+ bindir: bin
10
+ cert_chain: []
11
+ date: 2014-03-25 00:00:00.000000000 Z
12
+ dependencies:
13
+ - !ruby/object:Gem::Dependency
14
+ name: activemodel
15
+ requirement: !ruby/object:Gem::Requirement
16
+ requirements:
17
+ - - ">="
18
+ - !ruby/object:Gem::Version
19
+ version: '0'
20
+ type: :runtime
21
+ prerelease: false
22
+ version_requirements: !ruby/object:Gem::Requirement
23
+ requirements:
24
+ - - ">="
25
+ - !ruby/object:Gem::Version
26
+ version: '0'
27
+ - !ruby/object:Gem::Dependency
28
+ name: bundler
29
+ requirement: !ruby/object:Gem::Requirement
30
+ requirements:
31
+ - - "~>"
32
+ - !ruby/object:Gem::Version
33
+ version: '1.3'
34
+ type: :development
35
+ prerelease: false
36
+ version_requirements: !ruby/object:Gem::Requirement
37
+ requirements:
38
+ - - "~>"
39
+ - !ruby/object:Gem::Version
40
+ version: '1.3'
41
+ - !ruby/object:Gem::Dependency
42
+ name: rake
43
+ requirement: !ruby/object:Gem::Requirement
44
+ requirements:
45
+ - - ">="
46
+ - !ruby/object:Gem::Version
47
+ version: '0'
48
+ type: :development
49
+ prerelease: false
50
+ version_requirements: !ruby/object:Gem::Requirement
51
+ requirements:
52
+ - - ">="
53
+ - !ruby/object:Gem::Version
54
+ version: '0'
55
+ - !ruby/object:Gem::Dependency
56
+ name: rspec
57
+ requirement: !ruby/object:Gem::Requirement
58
+ requirements:
59
+ - - ">="
60
+ - !ruby/object:Gem::Version
61
+ version: '0'
62
+ type: :development
63
+ prerelease: false
64
+ version_requirements: !ruby/object:Gem::Requirement
65
+ requirements:
66
+ - - ">="
67
+ - !ruby/object:Gem::Version
68
+ version: '0'
69
+ - !ruby/object:Gem::Dependency
70
+ name: byebug
71
+ requirement: !ruby/object:Gem::Requirement
72
+ requirements:
73
+ - - ">="
74
+ - !ruby/object:Gem::Version
75
+ version: '0'
76
+ type: :development
77
+ prerelease: false
78
+ version_requirements: !ruby/object:Gem::Requirement
79
+ requirements:
80
+ - - ">="
81
+ - !ruby/object:Gem::Version
82
+ version: '0'
83
+ description: ActiveModel validator that acts like state machine
84
+ email:
85
+ - agresso@gmail.com
86
+ executables: []
87
+ extensions: []
88
+ extra_rdoc_files: []
89
+ files:
90
+ - ".gitignore"
91
+ - Gemfile
92
+ - LICENSE.txt
93
+ - README.md
94
+ - Rakefile
95
+ - changes_validator.gemspec
96
+ - lib/changes_validator.rb
97
+ - lib/changes_validator/version.rb
98
+ - locales/en.yml
99
+ - spec/changes_validator_spec.rb
100
+ - spec/spec_helper.rb
101
+ homepage: https://github.com/bogdan/changes_validator
102
+ licenses:
103
+ - MIT
104
+ metadata: {}
105
+ post_install_message:
106
+ rdoc_options: []
107
+ require_paths:
108
+ - lib
109
+ required_ruby_version: !ruby/object:Gem::Requirement
110
+ requirements:
111
+ - - ">="
112
+ - !ruby/object:Gem::Version
113
+ version: '0'
114
+ required_rubygems_version: !ruby/object:Gem::Requirement
115
+ requirements:
116
+ - - ">="
117
+ - !ruby/object:Gem::Version
118
+ version: '0'
119
+ requirements: []
120
+ rubyforge_project:
121
+ rubygems_version: 2.2.2
122
+ signing_key:
123
+ specification_version: 4
124
+ summary: This validator allows to specify a mapping of allowed changess for given
125
+ attribute and validate that this attribute can only be transfered according to changess
126
+ map
127
+ test_files:
128
+ - spec/changes_validator_spec.rb
129
+ - spec/spec_helper.rb