database_validations 1.1.1 → 2.0.0
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.
- checksums.yaml +4 -4
- data/lib/database_validations/lib/adapters/mysql_adapter.rb +6 -5
- data/lib/database_validations/lib/adapters/postgresql_adapter.rb +7 -6
- data/lib/database_validations/lib/adapters/sqlite_adapter.rb +12 -5
- data/lib/database_validations/lib/checkers/db_uniqueness_validator.rb +2 -2
- data/lib/database_validations/lib/rescuer.rb +42 -5
- data/lib/database_validations/lib/validators/db_presence_validator.rb +3 -7
- data/lib/database_validations/rspec/uniqueness_validator_matcher.rb +1 -3
- data/lib/database_validations/rubocop/cop/belongs_to.rb +3 -4
- data/lib/database_validations/rubocop/cop/uniqueness_of.rb +4 -3
- data/lib/database_validations/version.rb +1 -1
- metadata +36 -39
checksums.yaml
CHANGED
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
---
|
|
2
2
|
SHA256:
|
|
3
|
-
metadata.gz:
|
|
4
|
-
data.tar.gz:
|
|
3
|
+
metadata.gz: 2725c19065bc2a34ba2c1eddc54ab4f7127caf234a9e11be5a08c31388f4d3df
|
|
4
|
+
data.tar.gz: 6797dde1e63a4701eaa5e8c1c743c57feeae914352407e43c1185d092de1f8b8
|
|
5
5
|
SHA512:
|
|
6
|
-
metadata.gz:
|
|
7
|
-
data.tar.gz:
|
|
6
|
+
metadata.gz: a734d277d173977b8c30c3f6952f9458af89bf6eec3e79da3b218c222c8c575f94d163054bfcaf4d9c23b9cf6af2666a3b6794a22c7c013b61518605f57e2663
|
|
7
|
+
data.tar.gz: 3fba9fd273ca3640fd19854401e42c2c091794dfb763cee8cdaa0c9e80260723cdad9fe66cf9ef0824a476e49766f6d6648309d942d8ad99744963bf0b0d58d9
|
|
@@ -4,14 +4,15 @@ module DatabaseValidations
|
|
|
4
4
|
ADAPTER = :mysql2
|
|
5
5
|
|
|
6
6
|
class << self
|
|
7
|
-
def unique_index_name(
|
|
8
|
-
|
|
7
|
+
def unique_index_name(error)
|
|
8
|
+
error.message[/key '([^']+)'/, 1]&.split('.')&.last
|
|
9
9
|
end
|
|
10
10
|
|
|
11
|
-
def unique_error_columns(
|
|
11
|
+
def unique_error_columns(_error); end
|
|
12
12
|
|
|
13
|
-
def foreign_key_error_column(
|
|
14
|
-
|
|
13
|
+
def foreign_key_error_column(error)
|
|
14
|
+
column = error.message[/FOREIGN KEY \(`([^`]+)`\)/, 1]
|
|
15
|
+
column ? [column] : []
|
|
15
16
|
end
|
|
16
17
|
end
|
|
17
18
|
end
|
|
@@ -4,16 +4,17 @@ module DatabaseValidations
|
|
|
4
4
|
ADAPTER = :postgresql
|
|
5
5
|
|
|
6
6
|
class << self
|
|
7
|
-
def unique_index_name(
|
|
8
|
-
|
|
7
|
+
def unique_index_name(error)
|
|
8
|
+
error.message[/unique constraint "([^"]+)"/, 1]
|
|
9
9
|
end
|
|
10
10
|
|
|
11
|
-
def unique_error_columns(
|
|
12
|
-
|
|
11
|
+
def unique_error_columns(error)
|
|
12
|
+
error.message[/Key \((.+)\)=/, 1].split(', ')
|
|
13
13
|
end
|
|
14
14
|
|
|
15
|
-
def foreign_key_error_column(
|
|
16
|
-
|
|
15
|
+
def foreign_key_error_column(error)
|
|
16
|
+
column = error.message[/Key \(([^)]+)\)/, 1]
|
|
17
|
+
column ? [column] : []
|
|
17
18
|
end
|
|
18
19
|
end
|
|
19
20
|
end
|
|
@@ -4,14 +4,21 @@ module DatabaseValidations
|
|
|
4
4
|
ADAPTER = :sqlite3
|
|
5
5
|
|
|
6
6
|
class << self
|
|
7
|
-
def unique_index_name(
|
|
7
|
+
def unique_index_name(error)
|
|
8
|
+
error.message[/UNIQUE constraint failed: index '([^']+)'/, 1]
|
|
9
|
+
end
|
|
8
10
|
|
|
9
|
-
def unique_error_columns(
|
|
10
|
-
|
|
11
|
+
def unique_error_columns(error)
|
|
12
|
+
error.message.scan(/\w+\.([^,:]+)/).flatten
|
|
11
13
|
end
|
|
12
14
|
|
|
13
|
-
def foreign_key_error_column(
|
|
14
|
-
|
|
15
|
+
def foreign_key_error_column(error)
|
|
16
|
+
return [] unless error.respond_to?(:sql) && error.sql
|
|
17
|
+
|
|
18
|
+
columns_clause = error.sql[/\(([^)]+)\)\s*VALUES/i, 1]
|
|
19
|
+
return [] unless columns_clause
|
|
20
|
+
|
|
21
|
+
columns_clause.scan(/"([^"]+)"/).flatten
|
|
15
22
|
end
|
|
16
23
|
end
|
|
17
24
|
end
|
|
@@ -40,8 +40,8 @@ module DatabaseValidations
|
|
|
40
40
|
|
|
41
41
|
validator.attributes.map do |attribute|
|
|
42
42
|
columns = KeyGenerator.unify_columns(attribute, validator.options[:scope])
|
|
43
|
-
index = validator.index_name ? adapter.find_unique_index_by_name(validator.index_name.to_s) : adapter.find_unique_index(columns, validator.where) # rubocop:disable
|
|
44
|
-
raise Errors::IndexNotFound.new(columns, validator.where, validator.index_name, adapter.unique_indexes, adapter.table_name) unless index && valid_index?(columns, index) # rubocop:disable
|
|
43
|
+
index = validator.index_name ? adapter.find_unique_index_by_name(validator.index_name.to_s) : adapter.find_unique_index(columns, validator.where) # rubocop:disable Layout/LineLength
|
|
44
|
+
raise Errors::IndexNotFound.new(columns, validator.where, validator.index_name, adapter.unique_indexes, adapter.table_name) unless index && valid_index?(columns, index) # rubocop:disable Layout/LineLength
|
|
45
45
|
end
|
|
46
46
|
end
|
|
47
47
|
end
|
|
@@ -15,21 +15,58 @@ module DatabaseValidations
|
|
|
15
15
|
end
|
|
16
16
|
|
|
17
17
|
def process(validate, instance, error, key_types)
|
|
18
|
+
keys = resolve_keys(instance, error, key_types)
|
|
19
|
+
attribute_validator = find_matching_validator(instance, keys)
|
|
20
|
+
return false unless attribute_validator
|
|
21
|
+
|
|
22
|
+
process_validator(validate, instance, attribute_validator)
|
|
23
|
+
end
|
|
24
|
+
|
|
25
|
+
def resolve_keys(instance, error, key_types)
|
|
18
26
|
adapter = Adapters.factory(instance.class)
|
|
19
27
|
|
|
20
|
-
|
|
21
|
-
|
|
28
|
+
key_types.flat_map do |key_generator, error_processor|
|
|
29
|
+
result = adapter.public_send(error_processor, error)
|
|
30
|
+
|
|
31
|
+
# FK adapters return an array of candidate columns, each generating a
|
|
32
|
+
# separate key. Uniqueness adapters return columns that form a single
|
|
33
|
+
# composite key, passed together to the key generator.
|
|
34
|
+
if key_generator == :for_db_presence
|
|
35
|
+
Array(result).map { |column| KeyGenerator.public_send(key_generator, column) }
|
|
36
|
+
else
|
|
37
|
+
[KeyGenerator.public_send(key_generator, result)]
|
|
38
|
+
end
|
|
22
39
|
end
|
|
40
|
+
end
|
|
41
|
+
|
|
42
|
+
def find_matching_validator(instance, keys)
|
|
43
|
+
first_match = nil
|
|
23
44
|
|
|
24
45
|
keys.each do |key|
|
|
25
46
|
attribute_validator = instance._db_validators[key]
|
|
26
|
-
|
|
27
47
|
next unless attribute_validator
|
|
28
48
|
|
|
29
|
-
|
|
49
|
+
first_match ||= attribute_validator
|
|
50
|
+
|
|
51
|
+
next if (keys.size > 1) && !foreign_key_invalid?(instance, attribute_validator)
|
|
52
|
+
|
|
53
|
+
return attribute_validator
|
|
30
54
|
end
|
|
31
55
|
|
|
32
|
-
|
|
56
|
+
# TOCTOU fallback: if disambiguate queries all passed (concurrent insert),
|
|
57
|
+
# use the first matching validator rather than leaving the error unhandled.
|
|
58
|
+
first_match
|
|
59
|
+
end
|
|
60
|
+
|
|
61
|
+
def foreign_key_invalid?(instance, attribute_validator)
|
|
62
|
+
attribute = attribute_validator.attribute
|
|
63
|
+
reflection = instance.class._reflect_on_association(attribute)
|
|
64
|
+
return true unless reflection
|
|
65
|
+
|
|
66
|
+
fk_value = instance.read_attribute(reflection.foreign_key)
|
|
67
|
+
return true if fk_value.blank?
|
|
68
|
+
|
|
69
|
+
!reflection.klass.exists?(reflection.association_primary_key => fk_value)
|
|
33
70
|
end
|
|
34
71
|
|
|
35
72
|
def process_validator(validate, instance, attribute_validator)
|
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
module DatabaseValidations
|
|
2
2
|
class DbPresenceValidator < ActiveRecord::Validations::PresenceValidator
|
|
3
|
-
REFLECTION_MESSAGE =
|
|
3
|
+
REFLECTION_MESSAGE = :required
|
|
4
4
|
|
|
5
5
|
attr_reader :klass
|
|
6
6
|
|
|
@@ -53,15 +53,11 @@ module DatabaseValidations
|
|
|
53
53
|
end
|
|
54
54
|
|
|
55
55
|
def db_belongs_to(name, scope = nil, **options)
|
|
56
|
-
|
|
57
|
-
options[:required] = false
|
|
58
|
-
else
|
|
59
|
-
options[:optional] = true
|
|
60
|
-
end
|
|
56
|
+
options[:optional] = true
|
|
61
57
|
|
|
62
58
|
belongs_to(name, scope, **options)
|
|
63
59
|
|
|
64
|
-
validates_with DatabaseValidations::DbPresenceValidator, _merge_attributes([name, message: DatabaseValidations::DbPresenceValidator::REFLECTION_MESSAGE]) # rubocop:disable
|
|
60
|
+
validates_with DatabaseValidations::DbPresenceValidator, _merge_attributes([name, { message: DatabaseValidations::DbPresenceValidator::REFLECTION_MESSAGE }]) # rubocop:disable Layout/LineLength
|
|
65
61
|
end
|
|
66
62
|
end
|
|
67
63
|
end
|
|
@@ -63,15 +63,13 @@ RSpec::Matchers.define :validate_db_uniqueness_of do |field| # rubocop:disable M
|
|
|
63
63
|
end
|
|
64
64
|
end
|
|
65
65
|
|
|
66
|
-
case_sensitive_default = ActiveRecord::VERSION::MAJOR >= 6 ? nil : true
|
|
67
|
-
|
|
68
66
|
@validators.include?(
|
|
69
67
|
field: field,
|
|
70
68
|
scope: Array.wrap(@scope),
|
|
71
69
|
where: @where,
|
|
72
70
|
message: @message,
|
|
73
71
|
index_name: @index_name,
|
|
74
|
-
case_sensitive: @case_sensitive.nil? ?
|
|
72
|
+
case_sensitive: @case_sensitive.nil? ? nil : @case_sensitive
|
|
75
73
|
)
|
|
76
74
|
end
|
|
77
75
|
|
|
@@ -10,8 +10,9 @@ module RuboCop
|
|
|
10
10
|
# # good
|
|
11
11
|
# db_belongs_to :company
|
|
12
12
|
#
|
|
13
|
-
class BelongsTo <
|
|
13
|
+
class BelongsTo < Base
|
|
14
14
|
MSG = 'Use `db_belongs_to`.'.freeze
|
|
15
|
+
RESTRICT_ON_SEND = %i[belongs_to].freeze
|
|
15
16
|
|
|
16
17
|
NON_SUPPORTED_OPTIONS = %i[
|
|
17
18
|
optional
|
|
@@ -20,14 +21,12 @@ module RuboCop
|
|
|
20
21
|
foreign_type
|
|
21
22
|
].freeze
|
|
22
23
|
|
|
23
|
-
def_node_matcher :belongs_to?, '(send nil? :belongs_to ...)'
|
|
24
24
|
def_node_matcher :option_name, '(pair (sym $_) ...)'
|
|
25
25
|
|
|
26
26
|
def on_send(node)
|
|
27
|
-
return unless belongs_to?(node)
|
|
28
27
|
return unless supported?(node)
|
|
29
28
|
|
|
30
|
-
add_offense(node
|
|
29
|
+
add_offense(node.loc.selector)
|
|
31
30
|
end
|
|
32
31
|
|
|
33
32
|
private
|
|
@@ -14,17 +14,18 @@ module RuboCop
|
|
|
14
14
|
# validates_db_uniqueness_of :address, scope: :user_id
|
|
15
15
|
# validates_db_uniqueness_of :title
|
|
16
16
|
#
|
|
17
|
-
class UniquenessOf <
|
|
17
|
+
class UniquenessOf < Base
|
|
18
18
|
MSG = 'Use `validates_db_uniqueness_of`.'.freeze
|
|
19
|
+
RESTRICT_ON_SEND = %i[validates validates_uniqueness_of].freeze
|
|
19
20
|
|
|
20
21
|
def_node_matcher :uniquness_validation?, '(pair (sym :uniqueness) _)'
|
|
21
22
|
|
|
22
23
|
def on_send(node)
|
|
23
24
|
if node.method_name == :validates_uniqueness_of
|
|
24
|
-
add_offense(node
|
|
25
|
+
add_offense(node.loc.selector)
|
|
25
26
|
elsif node.method_name == :validates
|
|
26
27
|
uniqueness(node) do |option|
|
|
27
|
-
add_offense(option)
|
|
28
|
+
add_offense(option.loc.expression)
|
|
28
29
|
end
|
|
29
30
|
end
|
|
30
31
|
end
|
metadata
CHANGED
|
@@ -1,14 +1,13 @@
|
|
|
1
1
|
--- !ruby/object:Gem::Specification
|
|
2
2
|
name: database_validations
|
|
3
3
|
version: !ruby/object:Gem::Version
|
|
4
|
-
version:
|
|
4
|
+
version: 2.0.0
|
|
5
5
|
platform: ruby
|
|
6
6
|
authors:
|
|
7
7
|
- Evgeniy Demin
|
|
8
|
-
autorequire:
|
|
9
8
|
bindir: bin
|
|
10
9
|
cert_chain: []
|
|
11
|
-
date:
|
|
10
|
+
date: 1980-01-02 00:00:00.000000000 Z
|
|
12
11
|
dependencies:
|
|
13
12
|
- !ruby/object:Gem::Dependency
|
|
14
13
|
name: activerecord
|
|
@@ -16,56 +15,56 @@ dependencies:
|
|
|
16
15
|
requirements:
|
|
17
16
|
- - ">="
|
|
18
17
|
- !ruby/object:Gem::Version
|
|
19
|
-
version:
|
|
18
|
+
version: 7.2.0
|
|
20
19
|
type: :runtime
|
|
21
20
|
prerelease: false
|
|
22
21
|
version_requirements: !ruby/object:Gem::Requirement
|
|
23
22
|
requirements:
|
|
24
23
|
- - ">="
|
|
25
24
|
- !ruby/object:Gem::Version
|
|
26
|
-
version:
|
|
25
|
+
version: 7.2.0
|
|
27
26
|
- !ruby/object:Gem::Dependency
|
|
28
27
|
name: benchmark-ips
|
|
29
28
|
requirement: !ruby/object:Gem::Requirement
|
|
30
29
|
requirements:
|
|
31
|
-
- - "
|
|
30
|
+
- - ">="
|
|
32
31
|
- !ruby/object:Gem::Version
|
|
33
|
-
version: '
|
|
32
|
+
version: '0'
|
|
34
33
|
type: :development
|
|
35
34
|
prerelease: false
|
|
36
35
|
version_requirements: !ruby/object:Gem::Requirement
|
|
37
36
|
requirements:
|
|
38
|
-
- - "
|
|
37
|
+
- - ">="
|
|
39
38
|
- !ruby/object:Gem::Version
|
|
40
|
-
version: '
|
|
39
|
+
version: '0'
|
|
41
40
|
- !ruby/object:Gem::Dependency
|
|
42
41
|
name: bundler
|
|
43
42
|
requirement: !ruby/object:Gem::Requirement
|
|
44
43
|
requirements:
|
|
45
|
-
- - "
|
|
44
|
+
- - ">="
|
|
46
45
|
- !ruby/object:Gem::Version
|
|
47
|
-
version: '
|
|
46
|
+
version: '0'
|
|
48
47
|
type: :development
|
|
49
48
|
prerelease: false
|
|
50
49
|
version_requirements: !ruby/object:Gem::Requirement
|
|
51
50
|
requirements:
|
|
52
|
-
- - "
|
|
51
|
+
- - ">="
|
|
53
52
|
- !ruby/object:Gem::Version
|
|
54
|
-
version: '
|
|
53
|
+
version: '0'
|
|
55
54
|
- !ruby/object:Gem::Dependency
|
|
56
55
|
name: db-query-matchers
|
|
57
56
|
requirement: !ruby/object:Gem::Requirement
|
|
58
57
|
requirements:
|
|
59
58
|
- - ">="
|
|
60
59
|
- !ruby/object:Gem::Version
|
|
61
|
-
version: '0
|
|
60
|
+
version: '0'
|
|
62
61
|
type: :development
|
|
63
62
|
prerelease: false
|
|
64
63
|
version_requirements: !ruby/object:Gem::Requirement
|
|
65
64
|
requirements:
|
|
66
65
|
- - ">="
|
|
67
66
|
- !ruby/object:Gem::Version
|
|
68
|
-
version: '0
|
|
67
|
+
version: '0'
|
|
69
68
|
- !ruby/object:Gem::Dependency
|
|
70
69
|
name: mysql2
|
|
71
70
|
requirement: !ruby/object:Gem::Requirement
|
|
@@ -98,72 +97,72 @@ dependencies:
|
|
|
98
97
|
name: rake
|
|
99
98
|
requirement: !ruby/object:Gem::Requirement
|
|
100
99
|
requirements:
|
|
101
|
-
- - "
|
|
100
|
+
- - ">="
|
|
102
101
|
- !ruby/object:Gem::Version
|
|
103
|
-
version: '
|
|
102
|
+
version: '0'
|
|
104
103
|
type: :development
|
|
105
104
|
prerelease: false
|
|
106
105
|
version_requirements: !ruby/object:Gem::Requirement
|
|
107
106
|
requirements:
|
|
108
|
-
- - "
|
|
107
|
+
- - ">="
|
|
109
108
|
- !ruby/object:Gem::Version
|
|
110
|
-
version: '
|
|
109
|
+
version: '0'
|
|
111
110
|
- !ruby/object:Gem::Dependency
|
|
112
111
|
name: rspec
|
|
113
112
|
requirement: !ruby/object:Gem::Requirement
|
|
114
113
|
requirements:
|
|
115
|
-
- - "
|
|
114
|
+
- - ">="
|
|
116
115
|
- !ruby/object:Gem::Version
|
|
117
|
-
version: '
|
|
116
|
+
version: '0'
|
|
118
117
|
type: :development
|
|
119
118
|
prerelease: false
|
|
120
119
|
version_requirements: !ruby/object:Gem::Requirement
|
|
121
120
|
requirements:
|
|
122
|
-
- - "
|
|
121
|
+
- - ">="
|
|
123
122
|
- !ruby/object:Gem::Version
|
|
124
|
-
version: '
|
|
123
|
+
version: '0'
|
|
125
124
|
- !ruby/object:Gem::Dependency
|
|
126
125
|
name: rubocop
|
|
127
126
|
requirement: !ruby/object:Gem::Requirement
|
|
128
127
|
requirements:
|
|
129
|
-
- - "
|
|
128
|
+
- - ">="
|
|
130
129
|
- !ruby/object:Gem::Version
|
|
131
|
-
version: '0
|
|
130
|
+
version: '0'
|
|
132
131
|
type: :development
|
|
133
132
|
prerelease: false
|
|
134
133
|
version_requirements: !ruby/object:Gem::Requirement
|
|
135
134
|
requirements:
|
|
136
|
-
- - "
|
|
135
|
+
- - ">="
|
|
137
136
|
- !ruby/object:Gem::Version
|
|
138
|
-
version: '0
|
|
137
|
+
version: '0'
|
|
139
138
|
- !ruby/object:Gem::Dependency
|
|
140
139
|
name: rubocop-rspec
|
|
141
140
|
requirement: !ruby/object:Gem::Requirement
|
|
142
141
|
requirements:
|
|
143
|
-
- - "
|
|
142
|
+
- - ">="
|
|
144
143
|
- !ruby/object:Gem::Version
|
|
145
|
-
version: '
|
|
144
|
+
version: '0'
|
|
146
145
|
type: :development
|
|
147
146
|
prerelease: false
|
|
148
147
|
version_requirements: !ruby/object:Gem::Requirement
|
|
149
148
|
requirements:
|
|
150
|
-
- - "
|
|
149
|
+
- - ">="
|
|
151
150
|
- !ruby/object:Gem::Version
|
|
152
|
-
version: '
|
|
151
|
+
version: '0'
|
|
153
152
|
- !ruby/object:Gem::Dependency
|
|
154
153
|
name: sqlite3
|
|
155
154
|
requirement: !ruby/object:Gem::Requirement
|
|
156
155
|
requirements:
|
|
157
|
-
- - "
|
|
156
|
+
- - ">="
|
|
158
157
|
- !ruby/object:Gem::Version
|
|
159
|
-
version: '
|
|
158
|
+
version: '0'
|
|
160
159
|
type: :development
|
|
161
160
|
prerelease: false
|
|
162
161
|
version_requirements: !ruby/object:Gem::Requirement
|
|
163
162
|
requirements:
|
|
164
|
-
- - "
|
|
163
|
+
- - ">="
|
|
165
164
|
- !ruby/object:Gem::Version
|
|
166
|
-
version: '
|
|
165
|
+
version: '0'
|
|
167
166
|
description: |-
|
|
168
167
|
ActiveRecord provides validations on app level but it won't guarantee the
|
|
169
168
|
consistent. In some cases, like `validates_uniqueness_of` it executes
|
|
@@ -208,7 +207,6 @@ homepage: https://github.com/toptal/database_validations
|
|
|
208
207
|
licenses:
|
|
209
208
|
- MIT
|
|
210
209
|
metadata: {}
|
|
211
|
-
post_install_message:
|
|
212
210
|
rdoc_options: []
|
|
213
211
|
require_paths:
|
|
214
212
|
- lib
|
|
@@ -216,15 +214,14 @@ required_ruby_version: !ruby/object:Gem::Requirement
|
|
|
216
214
|
requirements:
|
|
217
215
|
- - ">="
|
|
218
216
|
- !ruby/object:Gem::Version
|
|
219
|
-
version:
|
|
217
|
+
version: 3.2.0
|
|
220
218
|
required_rubygems_version: !ruby/object:Gem::Requirement
|
|
221
219
|
requirements:
|
|
222
220
|
- - ">="
|
|
223
221
|
- !ruby/object:Gem::Version
|
|
224
222
|
version: '0'
|
|
225
223
|
requirements: []
|
|
226
|
-
rubygems_version: 3.
|
|
227
|
-
signing_key:
|
|
224
|
+
rubygems_version: 3.6.9
|
|
228
225
|
specification_version: 4
|
|
229
226
|
summary: Provide compatibility between database constraints and ActiveRecord validations
|
|
230
227
|
with better performance and consistency.
|