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 CHANGED
@@ -1,7 +1,7 @@
1
1
  ---
2
2
  SHA256:
3
- metadata.gz: 397b350ccf27a15694e155a4bb21bae6a01fe94cacceaa14aff89d1db6f542c1
4
- data.tar.gz: da042f3adbf3b00a1b85e01a8dd740b23dec523379adc1c5500f1a367f2a0391
3
+ metadata.gz: 2725c19065bc2a34ba2c1eddc54ab4f7127caf234a9e11be5a08c31388f4d3df
4
+ data.tar.gz: 6797dde1e63a4701eaa5e8c1c743c57feeae914352407e43c1185d092de1f8b8
5
5
  SHA512:
6
- metadata.gz: bf357e666a546058018322985ffe4a342d3004d9105c11bcd9e9eeb9b904ce8ecc915960783987b6b57cab8cca26c63eb7d6a788e74f581dc2e8e566f3a32bf9
7
- data.tar.gz: 8c9d658eff8041b6237c3b554b82dee7ca390a833aafdfd5d0fc4c7fa6f730241f5a9412c0a8b171ae4df1323e53e1fed796d6782df83b166a76cb78a26ab5e7
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(error_message)
8
- error_message[/key '([^']+)'/, 1]&.split('.')&.last
7
+ def unique_index_name(error)
8
+ error.message[/key '([^']+)'/, 1]&.split('.')&.last
9
9
  end
10
10
 
11
- def unique_error_columns(_error_message); end
11
+ def unique_error_columns(_error); end
12
12
 
13
- def foreign_key_error_column(error_message)
14
- error_message[/FOREIGN KEY \(`([^`]+)`\)/, 1]
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(error_message)
8
- error_message[/unique constraint "([^"]+)"/, 1]
7
+ def unique_index_name(error)
8
+ error.message[/unique constraint "([^"]+)"/, 1]
9
9
  end
10
10
 
11
- def unique_error_columns(error_message)
12
- error_message[/Key \((.+)\)=/, 1].split(', ')
11
+ def unique_error_columns(error)
12
+ error.message[/Key \((.+)\)=/, 1].split(', ')
13
13
  end
14
14
 
15
- def foreign_key_error_column(error_message)
16
- error_message[/Key \(([^)]+)\)/, 1]
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(_error_message); end
7
+ def unique_index_name(error)
8
+ error.message[/UNIQUE constraint failed: index '([^']+)'/, 1]
9
+ end
8
10
 
9
- def unique_error_columns(error_message)
10
- error_message.scan(/\w+\.([^,:]+)/).flatten
11
+ def unique_error_columns(error)
12
+ error.message.scan(/\w+\.([^,:]+)/).flatten
11
13
  end
12
14
 
13
- def foreign_key_error_column(error_message)
14
- error_message[/\("([^"]+)"\) VALUES/, 1]
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 Metrics/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 Metrics/LineLength
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
- keys = key_types.map do |key_generator, error_processor|
21
- KeyGenerator.public_send(key_generator, adapter.public_send(error_processor, error.message))
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
- return process_validator(validate, instance, attribute_validator)
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
- false
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 = ActiveRecord::VERSION::MAJOR < 5 ? :blank : :required
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
- if ActiveRecord::VERSION::MAJOR < 5
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 Metrics/LineLength
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? ? case_sensitive_default : @case_sensitive
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 < Cop
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, location: :selector)
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 < Cop
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, location: :selector)
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
@@ -1,3 +1,3 @@
1
1
  module DatabaseValidations
2
- VERSION = '1.1.1'.freeze
2
+ VERSION = '2.0.0'.freeze
3
3
  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: 1.1.1
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: 2022-03-14 00:00:00.000000000 Z
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: 4.2.0
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: 4.2.0
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: '2.7'
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: '2.7'
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: '2.0'
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: '2.0'
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.9'
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.9'
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: '13.0'
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: '13.0'
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: '3.0'
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: '3.0'
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.60'
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.60'
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: '1.30'
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: '1.30'
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: '1.3'
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: '1.3'
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: '0'
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.0.8
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.