simple_unique 0.0.1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- data/.gitignore +19 -0
- data/.rspec +2 -0
- data/Gemfile +4 -0
- data/LICENSE.txt +22 -0
- data/README.md +36 -0
- data/Rakefile +1 -0
- data/lib/simple_unique/validations/uniqueness.rb +45 -0
- data/lib/simple_unique/validations.rb +93 -0
- data/lib/simple_unique/version.rb +3 -0
- data/lib/simple_unique.rb +6 -0
- data/simple_unique.gemspec +23 -0
- data/spec/simple_unique/validations_spec.rb +101 -0
- data/spec/spec_helper.rb +22 -0
- data/spec/support/aws_init.rb.example +4 -0
- metadata +94 -0
data/.gitignore
ADDED
data/.rspec
ADDED
data/Gemfile
ADDED
data/LICENSE.txt
ADDED
@@ -0,0 +1,22 @@
|
|
1
|
+
Copyright (c) 2013 Jeremy Green
|
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.
|
data/README.md
ADDED
@@ -0,0 +1,36 @@
|
|
1
|
+
# SimpleUnique
|
2
|
+
|
3
|
+
A simple gem to add `validates_uniqueness_of` to `AWS::Record::Model`
|
4
|
+
supplied by `aws-sdk`.
|
5
|
+
|
6
|
+
## Installation
|
7
|
+
|
8
|
+
Add this line to your application's Gemfile:
|
9
|
+
|
10
|
+
gem 'simple_unique'
|
11
|
+
|
12
|
+
And then execute:
|
13
|
+
|
14
|
+
$ bundle
|
15
|
+
|
16
|
+
Or install it yourself as:
|
17
|
+
|
18
|
+
$ gem install simple_unique
|
19
|
+
|
20
|
+
## Usage
|
21
|
+
|
22
|
+
```ruby
|
23
|
+
def Person < AWS::Record::Model
|
24
|
+
string_attr :name
|
25
|
+
validates_uniqueness_of :name
|
26
|
+
end
|
27
|
+
```
|
28
|
+
|
29
|
+
|
30
|
+
## Contributing
|
31
|
+
|
32
|
+
1. Fork it
|
33
|
+
2. Create your feature branch (`git checkout -b my-new-feature`)
|
34
|
+
3. Commit your changes (`git commit -am 'Add some feature'`)
|
35
|
+
4. Push to the branch (`git push origin my-new-feature`)
|
36
|
+
5. Create new Pull Request
|
data/Rakefile
ADDED
@@ -0,0 +1 @@
|
|
1
|
+
require "bundler/gem_tasks"
|
@@ -0,0 +1,45 @@
|
|
1
|
+
require 'aws/record/validator'
|
2
|
+
|
3
|
+
module AWS
|
4
|
+
module Record
|
5
|
+
class UniquenessValidator < Validator
|
6
|
+
|
7
|
+
ACCEPTED_OPTIONS = [:message, :scope, :allow_nil, :allow_blank, :if, :unless]
|
8
|
+
|
9
|
+
def validate_attribute record, attribute_name, value
|
10
|
+
|
11
|
+
# @TODO - Model the initial relation setup after the setup
|
12
|
+
# found in ActiveRecord::Validations::UniquenessValidator.
|
13
|
+
# Not sure it we have to go to such lengths or not...
|
14
|
+
relation = record.class
|
15
|
+
|
16
|
+
scope_array(options[:scope]).each do |scope_item|
|
17
|
+
scope_value = record.send(scope_item.to_sym)
|
18
|
+
relation = relation.where(scope_item.to_sym => scope_value)
|
19
|
+
end
|
20
|
+
|
21
|
+
existing_record = relation.where(attribute_name.to_sym => value).first
|
22
|
+
taken = !existing_record.nil?
|
23
|
+
|
24
|
+
record.errors.add(attribute_name, message) if taken #blank
|
25
|
+
|
26
|
+
end
|
27
|
+
|
28
|
+
def message
|
29
|
+
options[:message] || 'has already been taken'
|
30
|
+
end
|
31
|
+
|
32
|
+
protected
|
33
|
+
def scope_array(scope)
|
34
|
+
if scope.nil?
|
35
|
+
[]
|
36
|
+
elsif scope.is_a? Array
|
37
|
+
scope
|
38
|
+
else
|
39
|
+
[scope]
|
40
|
+
end
|
41
|
+
end
|
42
|
+
|
43
|
+
end
|
44
|
+
end
|
45
|
+
end
|
@@ -0,0 +1,93 @@
|
|
1
|
+
require 'simple_unique/validations/uniqueness'
|
2
|
+
|
3
|
+
module AWS
|
4
|
+
module Record
|
5
|
+
|
6
|
+
module Validations
|
7
|
+
|
8
|
+
# Validates whether the value of the specified attributes are unique across
|
9
|
+
# the system. Useful for making sure that only one user can be named “foo”.
|
10
|
+
#
|
11
|
+
# class Person < AWS::Record::Model
|
12
|
+
# validates_uniqueness_of :user_name
|
13
|
+
# end
|
14
|
+
#
|
15
|
+
# It can also validate whether the value of the specified attributes are unique based on a scope parameter:
|
16
|
+
#
|
17
|
+
# class Person < AWS::Record::Model
|
18
|
+
# validates_uniqueness_of :user_name, :scope => :account_id
|
19
|
+
# end
|
20
|
+
#
|
21
|
+
# Or even multiple scope parameters. For example, making sure that a teacher can only be on the schedule once
|
22
|
+
# per semester for a particular class.
|
23
|
+
#
|
24
|
+
# class TeacherSchedule < AWS::Record::Model
|
25
|
+
# validates_uniqueness_of :teacher_id, :scope => [:semester_id, :class_id]
|
26
|
+
# end
|
27
|
+
#
|
28
|
+
# When the record is created, a check is performed to make sure that no record exists in the database
|
29
|
+
# with the given value for the specified attribute (that maps to a column). When the record is updated,
|
30
|
+
# the same check is made but disregarding the record itself.
|
31
|
+
#
|
32
|
+
# Configuration options:
|
33
|
+
# * <tt>:message</tt> - Specifies a custom error message (default is: "has already been taken").
|
34
|
+
# * <tt>:scope</tt> - One or more columns by which to limit the scope of the uniqueness constraint.
|
35
|
+
# * <tt>:case_sensitive</tt> - Always +true+. SimpleDB does not support case insensitive search.
|
36
|
+
# * <tt>:allow_nil</tt> - Always +true+.
|
37
|
+
# * <tt>:allow_blank</tt> - Always +true+.
|
38
|
+
# * <tt>:if</tt> - Specifies a method, proc or string to call to determine if the validation should
|
39
|
+
# occur (e.g. <tt>:if => :allow_validation</tt>, or <tt>:if => Proc.new { |user| user.signup_step > 2 }</tt>).
|
40
|
+
# The method, proc or string should return or evaluate to a true or false value.
|
41
|
+
# * <tt>:unless</tt> - Specifies a method, proc or string to call to determine if the validation should
|
42
|
+
# not occur (e.g. <tt>:unless => :skip_validation</tt>, or
|
43
|
+
# <tt>:unless => Proc.new { |user| user.signup_step <= 2 }</tt>). The method, proc or string should
|
44
|
+
# return or evaluate to a true or false value.
|
45
|
+
#
|
46
|
+
# === Concurrency and integrity
|
47
|
+
#
|
48
|
+
# Using this validation method in conjunction with AWS::Record::Model#save
|
49
|
+
# does not guarantee the absence of duplicate record insertions, because
|
50
|
+
# uniqueness checks on the application level are inherently prone to race
|
51
|
+
# conditions. For example, suppose that two users try to post a Comment at
|
52
|
+
# the same time, and a Comment's title must be unique. At the database-level,
|
53
|
+
# the actions performed by these users could be interleaved in the following manner:
|
54
|
+
#
|
55
|
+
# User 1 | User 2
|
56
|
+
# ------------------------------------+--------------------------------------
|
57
|
+
# # User 1 checks whether there's |
|
58
|
+
# # already a comment with the title |
|
59
|
+
# # 'My Post'. This is not the case. |
|
60
|
+
# SELECT * FROM comments |
|
61
|
+
# WHERE title = 'My Post' |
|
62
|
+
# |
|
63
|
+
# | # User 2 does the same thing and also
|
64
|
+
# | # infers that his title is unique.
|
65
|
+
# | SELECT * FROM comments
|
66
|
+
# | WHERE title = 'My Post'
|
67
|
+
# |
|
68
|
+
# # User 1 inserts his comment. |
|
69
|
+
# INSERT INTO comments |
|
70
|
+
# (title, content) VALUES |
|
71
|
+
# ('My Post', 'hi!') |
|
72
|
+
# |
|
73
|
+
# | # User 2 does the same thing.
|
74
|
+
# | INSERT INTO comments
|
75
|
+
# | (title, content) VALUES
|
76
|
+
# | ('My Post', 'hello!')
|
77
|
+
# |
|
78
|
+
# | # ^^^^^^
|
79
|
+
# | # Boom! We now have a duplicate
|
80
|
+
# | # title!
|
81
|
+
#
|
82
|
+
# To guard against duplicates you should create a process to periodically
|
83
|
+
# scan your data and take appropriate action when duplicates are found. (Ick!)
|
84
|
+
# This means that this validation is useful only in systems where the data creation
|
85
|
+
# events that trigger it are few and far between (relatively speaking).
|
86
|
+
def validates_uniqueness_of *args
|
87
|
+
validators << UniquenessValidator.new(self, *args)
|
88
|
+
end
|
89
|
+
|
90
|
+
end
|
91
|
+
|
92
|
+
end
|
93
|
+
end
|
@@ -0,0 +1,23 @@
|
|
1
|
+
# -*- encoding: utf-8 -*-
|
2
|
+
lib = File.expand_path('../lib', __FILE__)
|
3
|
+
$LOAD_PATH.unshift(lib) unless $LOAD_PATH.include?(lib)
|
4
|
+
require 'simple_unique/version'
|
5
|
+
|
6
|
+
Gem::Specification.new do |gem|
|
7
|
+
gem.name = "simple_unique"
|
8
|
+
gem.version = SimpleUnique::VERSION
|
9
|
+
gem.authors = ["Jeremy Green"]
|
10
|
+
gem.email = ["jeremy@octolabs.com"]
|
11
|
+
gem.description = %q{Validations for AWS::Record::Model}
|
12
|
+
gem.summary = %q{Validations for AWS::Record::Model}
|
13
|
+
gem.homepage = ""
|
14
|
+
|
15
|
+
gem.files = `git ls-files`.split($/)
|
16
|
+
gem.executables = gem.files.grep(%r{^bin/}).map{ |f| File.basename(f) }
|
17
|
+
gem.test_files = gem.files.grep(%r{^(test|spec|features)/})
|
18
|
+
gem.require_paths = ["lib"]
|
19
|
+
|
20
|
+
gem.add_dependency "aws-sdk", "~> 1.8.0"
|
21
|
+
gem.add_development_dependency "rspec", ">= 2.0.0"
|
22
|
+
|
23
|
+
end
|
@@ -0,0 +1,101 @@
|
|
1
|
+
require 'spec_helper'
|
2
|
+
|
3
|
+
describe 'model validations' do
|
4
|
+
|
5
|
+
let(:klass) { Class.new(AWS::Record::Base) }
|
6
|
+
|
7
|
+
before(:each) do
|
8
|
+
klass.create_domain
|
9
|
+
klass.each{|u| u.destroy }
|
10
|
+
sleep(2)
|
11
|
+
end
|
12
|
+
|
13
|
+
describe 'validates_uniqueness_of' do
|
14
|
+
it 'should only allow one record with a given attribute value' do
|
15
|
+
klass.string_attr :name
|
16
|
+
klass.validates_uniqueness_of :name
|
17
|
+
widg = klass.new(:name => "Turd Furguson")
|
18
|
+
widg.valid?.should be_true
|
19
|
+
widg.save
|
20
|
+
sleep(2) # Allow a couple of seconds for SimpleDB to catch up
|
21
|
+
|
22
|
+
widg = klass.new(:name => "Turd Furguson")
|
23
|
+
widg.valid?.should be_false
|
24
|
+
widg.errors[:name].should == ['has already been taken']
|
25
|
+
end
|
26
|
+
|
27
|
+
it 'should only allow one record with a given attribute value within a given scope' do
|
28
|
+
klass.string_attr :name
|
29
|
+
klass.string_attr :category
|
30
|
+
klass.validates_uniqueness_of :name, :scope => :category
|
31
|
+
widg = klass.new(:name => "Turd Furguson", :category => "Funny name")
|
32
|
+
widg.valid?.should be_true
|
33
|
+
widg.save
|
34
|
+
sleep(2) # Allow a couple of seconds for SimpleDB to catch up
|
35
|
+
|
36
|
+
widg = klass.new(:name => "Turd Furguson", :category => "Funny name")
|
37
|
+
widg.valid?.should be_false
|
38
|
+
widg.errors[:name].should == ['has already been taken']
|
39
|
+
|
40
|
+
widg.category = "Silly name"
|
41
|
+
widg.valid?.should be_true
|
42
|
+
end
|
43
|
+
|
44
|
+
|
45
|
+
it 'should skip the validation if the given attribute is nil when :allow_nil => true' do
|
46
|
+
klass.string_attr :name
|
47
|
+
klass.string_attr :category # need to avoid the "can't save empty record" error
|
48
|
+
klass.validates_uniqueness_of :name#, :allow_nil => true
|
49
|
+
widg = klass.new(:name => nil, :category => "A cat")
|
50
|
+
widg.valid?.should be_true
|
51
|
+
widg.save
|
52
|
+
sleep(2) # Allow a couple of seconds for SimpleDB to catch up
|
53
|
+
|
54
|
+
widg = klass.new(:name => nil, :category => "A cat")
|
55
|
+
widg.valid?.should be_true
|
56
|
+
end
|
57
|
+
|
58
|
+
it 'should skip the validation if the given attribute is blank when :allow_blank => true' do
|
59
|
+
klass.string_attr :name
|
60
|
+
klass.string_attr :category # need to avoid the "can't save empty record" error
|
61
|
+
klass.validates_uniqueness_of :name#, :allow_blank => true
|
62
|
+
widg = klass.new(:name => "", :category => "A cat")
|
63
|
+
widg.valid?.should be_true
|
64
|
+
widg.save
|
65
|
+
sleep(2) # Allow a couple of seconds for SimpleDB to catch up
|
66
|
+
|
67
|
+
widg = klass.new(:name => "", :category => "A cat")
|
68
|
+
widg.valid?.should be_true
|
69
|
+
end
|
70
|
+
|
71
|
+
it 'should skip the validation if the :if option evaluates to false' do
|
72
|
+
klass.string_attr :name
|
73
|
+
klass.boolean_attr :should_validate
|
74
|
+
klass.validates_uniqueness_of :name, :if => :should_validate
|
75
|
+
widg = klass.new(:name => "Turd Furguson")
|
76
|
+
widg.valid?.should be_true
|
77
|
+
widg.save
|
78
|
+
sleep(2) # Allow a couple of seconds for SimpleDB to catch up
|
79
|
+
|
80
|
+
widg = klass.new(:name => "Turd Furguson",:should_validate => false)
|
81
|
+
widg.valid?.should be_true
|
82
|
+
end
|
83
|
+
|
84
|
+
it 'should skip the validation if the :unless option evaluates to true' do
|
85
|
+
klass.string_attr :name
|
86
|
+
klass.boolean_attr :should_not_validate
|
87
|
+
klass.validates_uniqueness_of :name, :unless => :should_not_validate
|
88
|
+
widg = klass.new(:name => "Turd Furguson")
|
89
|
+
widg.valid?.should be_true
|
90
|
+
widg.save
|
91
|
+
sleep(2) # Allow a couple of seconds for SimpleDB to catch up
|
92
|
+
|
93
|
+
widg = klass.new(:name => "Turd Furguson",:should_not_validate => true)
|
94
|
+
widg.valid?.should be_true
|
95
|
+
end
|
96
|
+
|
97
|
+
end
|
98
|
+
|
99
|
+
end
|
100
|
+
|
101
|
+
|
data/spec/spec_helper.rb
ADDED
@@ -0,0 +1,22 @@
|
|
1
|
+
# This file was generated by the `rspec --init` command. Conventionally, all
|
2
|
+
# specs live under a `spec` directory, which RSpec adds to the `$LOAD_PATH`.
|
3
|
+
# Require this file using `require "spec_helper"` to ensure that it is only
|
4
|
+
# loaded once.
|
5
|
+
#
|
6
|
+
# See http://rubydoc.info/gems/rspec-core/RSpec/Core/Configuration
|
7
|
+
require 'aws-sdk'
|
8
|
+
require 'support/aws_init'
|
9
|
+
|
10
|
+
require "#{File.dirname(__FILE__)}/../lib/simple_unique"
|
11
|
+
|
12
|
+
RSpec.configure do |config|
|
13
|
+
config.treat_symbols_as_metadata_keys_with_true_values = true
|
14
|
+
config.run_all_when_everything_filtered = true
|
15
|
+
config.filter_run :focus
|
16
|
+
|
17
|
+
# Run specs in random order to surface order dependencies. If you find an
|
18
|
+
# order dependency and want to debug it, you can fix the order by providing
|
19
|
+
# the seed, which is printed after each run.
|
20
|
+
# --seed 1234
|
21
|
+
config.order = 'random'
|
22
|
+
end
|
metadata
ADDED
@@ -0,0 +1,94 @@
|
|
1
|
+
--- !ruby/object:Gem::Specification
|
2
|
+
name: simple_unique
|
3
|
+
version: !ruby/object:Gem::Version
|
4
|
+
version: 0.0.1
|
5
|
+
prerelease:
|
6
|
+
platform: ruby
|
7
|
+
authors:
|
8
|
+
- Jeremy Green
|
9
|
+
autorequire:
|
10
|
+
bindir: bin
|
11
|
+
cert_chain: []
|
12
|
+
date: 2013-01-12 00:00:00.000000000 Z
|
13
|
+
dependencies:
|
14
|
+
- !ruby/object:Gem::Dependency
|
15
|
+
name: aws-sdk
|
16
|
+
requirement: !ruby/object:Gem::Requirement
|
17
|
+
none: false
|
18
|
+
requirements:
|
19
|
+
- - ~>
|
20
|
+
- !ruby/object:Gem::Version
|
21
|
+
version: 1.8.0
|
22
|
+
type: :runtime
|
23
|
+
prerelease: false
|
24
|
+
version_requirements: !ruby/object:Gem::Requirement
|
25
|
+
none: false
|
26
|
+
requirements:
|
27
|
+
- - ~>
|
28
|
+
- !ruby/object:Gem::Version
|
29
|
+
version: 1.8.0
|
30
|
+
- !ruby/object:Gem::Dependency
|
31
|
+
name: rspec
|
32
|
+
requirement: !ruby/object:Gem::Requirement
|
33
|
+
none: false
|
34
|
+
requirements:
|
35
|
+
- - ! '>='
|
36
|
+
- !ruby/object:Gem::Version
|
37
|
+
version: 2.0.0
|
38
|
+
type: :development
|
39
|
+
prerelease: false
|
40
|
+
version_requirements: !ruby/object:Gem::Requirement
|
41
|
+
none: false
|
42
|
+
requirements:
|
43
|
+
- - ! '>='
|
44
|
+
- !ruby/object:Gem::Version
|
45
|
+
version: 2.0.0
|
46
|
+
description: Validations for AWS::Record::Model
|
47
|
+
email:
|
48
|
+
- jeremy@octolabs.com
|
49
|
+
executables: []
|
50
|
+
extensions: []
|
51
|
+
extra_rdoc_files: []
|
52
|
+
files:
|
53
|
+
- .gitignore
|
54
|
+
- .rspec
|
55
|
+
- Gemfile
|
56
|
+
- LICENSE.txt
|
57
|
+
- README.md
|
58
|
+
- Rakefile
|
59
|
+
- lib/simple_unique.rb
|
60
|
+
- lib/simple_unique/validations.rb
|
61
|
+
- lib/simple_unique/validations/uniqueness.rb
|
62
|
+
- lib/simple_unique/version.rb
|
63
|
+
- simple_unique.gemspec
|
64
|
+
- spec/simple_unique/validations_spec.rb
|
65
|
+
- spec/spec_helper.rb
|
66
|
+
- spec/support/aws_init.rb.example
|
67
|
+
homepage: ''
|
68
|
+
licenses: []
|
69
|
+
post_install_message:
|
70
|
+
rdoc_options: []
|
71
|
+
require_paths:
|
72
|
+
- lib
|
73
|
+
required_ruby_version: !ruby/object:Gem::Requirement
|
74
|
+
none: false
|
75
|
+
requirements:
|
76
|
+
- - ! '>='
|
77
|
+
- !ruby/object:Gem::Version
|
78
|
+
version: '0'
|
79
|
+
required_rubygems_version: !ruby/object:Gem::Requirement
|
80
|
+
none: false
|
81
|
+
requirements:
|
82
|
+
- - ! '>='
|
83
|
+
- !ruby/object:Gem::Version
|
84
|
+
version: '0'
|
85
|
+
requirements: []
|
86
|
+
rubyforge_project:
|
87
|
+
rubygems_version: 1.8.24
|
88
|
+
signing_key:
|
89
|
+
specification_version: 3
|
90
|
+
summary: Validations for AWS::Record::Model
|
91
|
+
test_files:
|
92
|
+
- spec/simple_unique/validations_spec.rb
|
93
|
+
- spec/spec_helper.rb
|
94
|
+
- spec/support/aws_init.rb.example
|