activerecord 3.2.22.5 → 5.2.8

Sign up to get free protection for your applications and to get access to all the features.

Potentially problematic release.


This version of activerecord might be problematic. Click here for more details.

Files changed (275) hide show
  1. checksums.yaml +5 -5
  2. data/CHANGELOG.md +657 -621
  3. data/MIT-LICENSE +2 -2
  4. data/README.rdoc +41 -46
  5. data/examples/performance.rb +55 -42
  6. data/examples/simple.rb +6 -5
  7. data/lib/active_record/aggregations.rb +264 -236
  8. data/lib/active_record/association_relation.rb +40 -0
  9. data/lib/active_record/associations/alias_tracker.rb +47 -42
  10. data/lib/active_record/associations/association.rb +127 -75
  11. data/lib/active_record/associations/association_scope.rb +126 -92
  12. data/lib/active_record/associations/belongs_to_association.rb +78 -27
  13. data/lib/active_record/associations/belongs_to_polymorphic_association.rb +9 -4
  14. data/lib/active_record/associations/builder/association.rb +117 -32
  15. data/lib/active_record/associations/builder/belongs_to.rb +135 -60
  16. data/lib/active_record/associations/builder/collection_association.rb +61 -54
  17. data/lib/active_record/associations/builder/has_and_belongs_to_many.rb +120 -42
  18. data/lib/active_record/associations/builder/has_many.rb +10 -64
  19. data/lib/active_record/associations/builder/has_one.rb +19 -51
  20. data/lib/active_record/associations/builder/singular_association.rb +28 -18
  21. data/lib/active_record/associations/collection_association.rb +226 -293
  22. data/lib/active_record/associations/collection_proxy.rb +1067 -69
  23. data/lib/active_record/associations/foreign_association.rb +13 -0
  24. data/lib/active_record/associations/has_many_association.rb +83 -47
  25. data/lib/active_record/associations/has_many_through_association.rb +98 -65
  26. data/lib/active_record/associations/has_one_association.rb +57 -20
  27. data/lib/active_record/associations/has_one_through_association.rb +18 -9
  28. data/lib/active_record/associations/join_dependency/join_association.rb +48 -126
  29. data/lib/active_record/associations/join_dependency/join_base.rb +11 -12
  30. data/lib/active_record/associations/join_dependency/join_part.rb +35 -42
  31. data/lib/active_record/associations/join_dependency.rb +212 -164
  32. data/lib/active_record/associations/preloader/association.rb +95 -89
  33. data/lib/active_record/associations/preloader/through_association.rb +84 -44
  34. data/lib/active_record/associations/preloader.rb +123 -111
  35. data/lib/active_record/associations/singular_association.rb +33 -24
  36. data/lib/active_record/associations/through_association.rb +60 -26
  37. data/lib/active_record/associations.rb +1759 -1506
  38. data/lib/active_record/attribute_assignment.rb +60 -193
  39. data/lib/active_record/attribute_decorators.rb +90 -0
  40. data/lib/active_record/attribute_methods/before_type_cast.rb +55 -8
  41. data/lib/active_record/attribute_methods/dirty.rb +113 -74
  42. data/lib/active_record/attribute_methods/primary_key.rb +106 -77
  43. data/lib/active_record/attribute_methods/query.rb +8 -5
  44. data/lib/active_record/attribute_methods/read.rb +63 -114
  45. data/lib/active_record/attribute_methods/serialization.rb +60 -90
  46. data/lib/active_record/attribute_methods/time_zone_conversion.rb +69 -43
  47. data/lib/active_record/attribute_methods/write.rb +43 -45
  48. data/lib/active_record/attribute_methods.rb +366 -149
  49. data/lib/active_record/attributes.rb +266 -0
  50. data/lib/active_record/autosave_association.rb +312 -225
  51. data/lib/active_record/base.rb +114 -505
  52. data/lib/active_record/callbacks.rb +145 -67
  53. data/lib/active_record/coders/json.rb +15 -0
  54. data/lib/active_record/coders/yaml_column.rb +32 -23
  55. data/lib/active_record/collection_cache_key.rb +53 -0
  56. data/lib/active_record/connection_adapters/abstract/connection_pool.rb +883 -284
  57. data/lib/active_record/connection_adapters/abstract/database_limits.rb +16 -2
  58. data/lib/active_record/connection_adapters/abstract/database_statements.rb +350 -200
  59. data/lib/active_record/connection_adapters/abstract/query_cache.rb +82 -27
  60. data/lib/active_record/connection_adapters/abstract/quoting.rb +150 -65
  61. data/lib/active_record/connection_adapters/abstract/savepoints.rb +23 -0
  62. data/lib/active_record/connection_adapters/abstract/schema_creation.rb +146 -0
  63. data/lib/active_record/connection_adapters/abstract/schema_definitions.rb +477 -284
  64. data/lib/active_record/connection_adapters/abstract/schema_dumper.rb +95 -0
  65. data/lib/active_record/connection_adapters/abstract/schema_statements.rb +1100 -310
  66. data/lib/active_record/connection_adapters/abstract/transaction.rb +283 -0
  67. data/lib/active_record/connection_adapters/abstract_adapter.rb +450 -118
  68. data/lib/active_record/connection_adapters/abstract_mysql_adapter.rb +657 -446
  69. data/lib/active_record/connection_adapters/column.rb +50 -255
  70. data/lib/active_record/connection_adapters/connection_specification.rb +287 -0
  71. data/lib/active_record/connection_adapters/determine_if_preparable_visitor.rb +33 -0
  72. data/lib/active_record/connection_adapters/mysql/column.rb +27 -0
  73. data/lib/active_record/connection_adapters/mysql/database_statements.rb +140 -0
  74. data/lib/active_record/connection_adapters/mysql/explain_pretty_printer.rb +72 -0
  75. data/lib/active_record/connection_adapters/mysql/quoting.rb +44 -0
  76. data/lib/active_record/connection_adapters/mysql/schema_creation.rb +73 -0
  77. data/lib/active_record/connection_adapters/mysql/schema_definitions.rb +87 -0
  78. data/lib/active_record/connection_adapters/mysql/schema_dumper.rb +80 -0
  79. data/lib/active_record/connection_adapters/mysql/schema_statements.rb +148 -0
  80. data/lib/active_record/connection_adapters/mysql/type_metadata.rb +35 -0
  81. data/lib/active_record/connection_adapters/mysql2_adapter.rb +59 -210
  82. data/lib/active_record/connection_adapters/postgresql/column.rb +44 -0
  83. data/lib/active_record/connection_adapters/postgresql/database_statements.rb +163 -0
  84. data/lib/active_record/connection_adapters/postgresql/explain_pretty_printer.rb +44 -0
  85. data/lib/active_record/connection_adapters/postgresql/oid/array.rb +92 -0
  86. data/lib/active_record/connection_adapters/postgresql/oid/bit.rb +56 -0
  87. data/lib/active_record/connection_adapters/postgresql/oid/bit_varying.rb +15 -0
  88. data/lib/active_record/connection_adapters/postgresql/oid/bytea.rb +17 -0
  89. data/lib/active_record/connection_adapters/postgresql/oid/cidr.rb +50 -0
  90. data/lib/active_record/connection_adapters/postgresql/oid/date.rb +23 -0
  91. data/lib/active_record/connection_adapters/postgresql/oid/date_time.rb +23 -0
  92. data/lib/active_record/connection_adapters/postgresql/oid/decimal.rb +15 -0
  93. data/lib/active_record/connection_adapters/postgresql/oid/enum.rb +21 -0
  94. data/lib/active_record/connection_adapters/postgresql/oid/hstore.rb +71 -0
  95. data/lib/active_record/connection_adapters/postgresql/oid/inet.rb +15 -0
  96. data/lib/active_record/connection_adapters/postgresql/oid/jsonb.rb +15 -0
  97. data/lib/active_record/connection_adapters/postgresql/oid/legacy_point.rb +45 -0
  98. data/lib/active_record/connection_adapters/postgresql/oid/money.rb +41 -0
  99. data/lib/active_record/connection_adapters/postgresql/oid/oid.rb +15 -0
  100. data/lib/active_record/connection_adapters/postgresql/oid/point.rb +65 -0
  101. data/lib/active_record/connection_adapters/postgresql/oid/range.rb +97 -0
  102. data/lib/active_record/connection_adapters/postgresql/oid/specialized_string.rb +18 -0
  103. data/lib/active_record/connection_adapters/postgresql/oid/type_map_initializer.rb +111 -0
  104. data/lib/active_record/connection_adapters/postgresql/oid/uuid.rb +23 -0
  105. data/lib/active_record/connection_adapters/postgresql/oid/vector.rb +28 -0
  106. data/lib/active_record/connection_adapters/postgresql/oid/xml.rb +30 -0
  107. data/lib/active_record/connection_adapters/postgresql/oid.rb +34 -0
  108. data/lib/active_record/connection_adapters/postgresql/quoting.rb +168 -0
  109. data/lib/active_record/connection_adapters/postgresql/referential_integrity.rb +43 -0
  110. data/lib/active_record/connection_adapters/postgresql/schema_creation.rb +65 -0
  111. data/lib/active_record/connection_adapters/postgresql/schema_definitions.rb +206 -0
  112. data/lib/active_record/connection_adapters/postgresql/schema_dumper.rb +50 -0
  113. data/lib/active_record/connection_adapters/postgresql/schema_statements.rb +774 -0
  114. data/lib/active_record/connection_adapters/postgresql/type_metadata.rb +39 -0
  115. data/lib/active_record/connection_adapters/postgresql/utils.rb +81 -0
  116. data/lib/active_record/connection_adapters/postgresql_adapter.rb +620 -1080
  117. data/lib/active_record/connection_adapters/schema_cache.rb +85 -36
  118. data/lib/active_record/connection_adapters/sql_type_metadata.rb +34 -0
  119. data/lib/active_record/connection_adapters/sqlite3/explain_pretty_printer.rb +21 -0
  120. data/lib/active_record/connection_adapters/sqlite3/quoting.rb +67 -0
  121. data/lib/active_record/connection_adapters/sqlite3/schema_creation.rb +17 -0
  122. data/lib/active_record/connection_adapters/sqlite3/schema_definitions.rb +19 -0
  123. data/lib/active_record/connection_adapters/sqlite3/schema_dumper.rb +18 -0
  124. data/lib/active_record/connection_adapters/sqlite3/schema_statements.rb +106 -0
  125. data/lib/active_record/connection_adapters/sqlite3_adapter.rb +545 -27
  126. data/lib/active_record/connection_adapters/statement_pool.rb +34 -13
  127. data/lib/active_record/connection_handling.rb +145 -0
  128. data/lib/active_record/core.rb +559 -0
  129. data/lib/active_record/counter_cache.rb +200 -105
  130. data/lib/active_record/define_callbacks.rb +22 -0
  131. data/lib/active_record/dynamic_matchers.rb +107 -69
  132. data/lib/active_record/enum.rb +244 -0
  133. data/lib/active_record/errors.rb +245 -60
  134. data/lib/active_record/explain.rb +35 -71
  135. data/lib/active_record/explain_registry.rb +32 -0
  136. data/lib/active_record/explain_subscriber.rb +18 -9
  137. data/lib/active_record/fixture_set/file.rb +82 -0
  138. data/lib/active_record/fixtures.rb +418 -275
  139. data/lib/active_record/gem_version.rb +17 -0
  140. data/lib/active_record/inheritance.rb +209 -100
  141. data/lib/active_record/integration.rb +116 -21
  142. data/lib/active_record/internal_metadata.rb +45 -0
  143. data/lib/active_record/legacy_yaml_adapter.rb +48 -0
  144. data/lib/active_record/locale/en.yml +9 -1
  145. data/lib/active_record/locking/optimistic.rb +107 -94
  146. data/lib/active_record/locking/pessimistic.rb +20 -8
  147. data/lib/active_record/log_subscriber.rb +99 -34
  148. data/lib/active_record/migration/command_recorder.rb +199 -64
  149. data/lib/active_record/migration/compatibility.rb +217 -0
  150. data/lib/active_record/migration/join_table.rb +17 -0
  151. data/lib/active_record/migration.rb +893 -296
  152. data/lib/active_record/model_schema.rb +328 -175
  153. data/lib/active_record/nested_attributes.rb +338 -242
  154. data/lib/active_record/no_touching.rb +58 -0
  155. data/lib/active_record/null_relation.rb +68 -0
  156. data/lib/active_record/persistence.rb +557 -170
  157. data/lib/active_record/query_cache.rb +14 -43
  158. data/lib/active_record/querying.rb +36 -24
  159. data/lib/active_record/railtie.rb +147 -52
  160. data/lib/active_record/railties/console_sandbox.rb +5 -4
  161. data/lib/active_record/railties/controller_runtime.rb +13 -6
  162. data/lib/active_record/railties/databases.rake +206 -488
  163. data/lib/active_record/readonly_attributes.rb +4 -6
  164. data/lib/active_record/reflection.rb +734 -228
  165. data/lib/active_record/relation/batches/batch_enumerator.rb +69 -0
  166. data/lib/active_record/relation/batches.rb +249 -52
  167. data/lib/active_record/relation/calculations.rb +330 -284
  168. data/lib/active_record/relation/delegation.rb +135 -37
  169. data/lib/active_record/relation/finder_methods.rb +450 -287
  170. data/lib/active_record/relation/from_clause.rb +26 -0
  171. data/lib/active_record/relation/merger.rb +193 -0
  172. data/lib/active_record/relation/predicate_builder/array_handler.rb +48 -0
  173. data/lib/active_record/relation/predicate_builder/association_query_value.rb +46 -0
  174. data/lib/active_record/relation/predicate_builder/base_handler.rb +19 -0
  175. data/lib/active_record/relation/predicate_builder/basic_object_handler.rb +20 -0
  176. data/lib/active_record/relation/predicate_builder/polymorphic_array_value.rb +56 -0
  177. data/lib/active_record/relation/predicate_builder/range_handler.rb +42 -0
  178. data/lib/active_record/relation/predicate_builder/relation_handler.rb +19 -0
  179. data/lib/active_record/relation/predicate_builder.rb +132 -43
  180. data/lib/active_record/relation/query_attribute.rb +45 -0
  181. data/lib/active_record/relation/query_methods.rb +1037 -221
  182. data/lib/active_record/relation/record_fetch_warning.rb +51 -0
  183. data/lib/active_record/relation/spawn_methods.rb +48 -151
  184. data/lib/active_record/relation/where_clause.rb +186 -0
  185. data/lib/active_record/relation/where_clause_factory.rb +34 -0
  186. data/lib/active_record/relation.rb +451 -359
  187. data/lib/active_record/result.rb +129 -20
  188. data/lib/active_record/runtime_registry.rb +24 -0
  189. data/lib/active_record/sanitization.rb +164 -136
  190. data/lib/active_record/schema.rb +31 -19
  191. data/lib/active_record/schema_dumper.rb +154 -107
  192. data/lib/active_record/schema_migration.rb +56 -0
  193. data/lib/active_record/scoping/default.rb +108 -98
  194. data/lib/active_record/scoping/named.rb +125 -112
  195. data/lib/active_record/scoping.rb +77 -123
  196. data/lib/active_record/secure_token.rb +40 -0
  197. data/lib/active_record/serialization.rb +10 -6
  198. data/lib/active_record/statement_cache.rb +121 -0
  199. data/lib/active_record/store.rb +175 -16
  200. data/lib/active_record/suppressor.rb +61 -0
  201. data/lib/active_record/table_metadata.rb +82 -0
  202. data/lib/active_record/tasks/database_tasks.rb +337 -0
  203. data/lib/active_record/tasks/mysql_database_tasks.rb +115 -0
  204. data/lib/active_record/tasks/postgresql_database_tasks.rb +143 -0
  205. data/lib/active_record/tasks/sqlite_database_tasks.rb +83 -0
  206. data/lib/active_record/timestamp.rb +80 -41
  207. data/lib/active_record/touch_later.rb +64 -0
  208. data/lib/active_record/transactions.rb +240 -119
  209. data/lib/active_record/translation.rb +2 -0
  210. data/lib/active_record/type/adapter_specific_registry.rb +136 -0
  211. data/lib/active_record/type/date.rb +9 -0
  212. data/lib/active_record/type/date_time.rb +9 -0
  213. data/lib/active_record/type/decimal_without_scale.rb +15 -0
  214. data/lib/active_record/type/hash_lookup_type_map.rb +25 -0
  215. data/lib/active_record/type/internal/timezone.rb +17 -0
  216. data/lib/active_record/type/json.rb +30 -0
  217. data/lib/active_record/type/serialized.rb +71 -0
  218. data/lib/active_record/type/text.rb +11 -0
  219. data/lib/active_record/type/time.rb +21 -0
  220. data/lib/active_record/type/type_map.rb +62 -0
  221. data/lib/active_record/type/unsigned_integer.rb +17 -0
  222. data/lib/active_record/type.rb +79 -0
  223. data/lib/active_record/type_caster/connection.rb +33 -0
  224. data/lib/active_record/type_caster/map.rb +23 -0
  225. data/lib/active_record/type_caster.rb +9 -0
  226. data/lib/active_record/validations/absence.rb +25 -0
  227. data/lib/active_record/validations/associated.rb +35 -18
  228. data/lib/active_record/validations/length.rb +26 -0
  229. data/lib/active_record/validations/presence.rb +68 -0
  230. data/lib/active_record/validations/uniqueness.rb +133 -75
  231. data/lib/active_record/validations.rb +53 -43
  232. data/lib/active_record/version.rb +7 -7
  233. data/lib/active_record.rb +89 -57
  234. data/lib/rails/generators/active_record/application_record/application_record_generator.rb +27 -0
  235. data/lib/rails/generators/active_record/application_record/templates/application_record.rb.tt +5 -0
  236. data/lib/rails/generators/active_record/migration/migration_generator.rb +61 -8
  237. data/lib/rails/generators/active_record/migration/templates/create_table_migration.rb.tt +24 -0
  238. data/lib/rails/generators/active_record/migration/templates/migration.rb.tt +46 -0
  239. data/lib/rails/generators/active_record/migration.rb +28 -8
  240. data/lib/rails/generators/active_record/model/model_generator.rb +23 -22
  241. data/lib/rails/generators/active_record/model/templates/model.rb.tt +13 -0
  242. data/lib/rails/generators/active_record/model/templates/{module.rb → module.rb.tt} +1 -1
  243. data/lib/rails/generators/active_record.rb +10 -16
  244. metadata +141 -62
  245. data/examples/associations.png +0 -0
  246. data/lib/active_record/associations/has_and_belongs_to_many_association.rb +0 -63
  247. data/lib/active_record/associations/join_helper.rb +0 -55
  248. data/lib/active_record/associations/preloader/belongs_to.rb +0 -17
  249. data/lib/active_record/associations/preloader/collection_association.rb +0 -24
  250. data/lib/active_record/associations/preloader/has_and_belongs_to_many.rb +0 -60
  251. data/lib/active_record/associations/preloader/has_many.rb +0 -17
  252. data/lib/active_record/associations/preloader/has_many_through.rb +0 -15
  253. data/lib/active_record/associations/preloader/has_one.rb +0 -23
  254. data/lib/active_record/associations/preloader/has_one_through.rb +0 -9
  255. data/lib/active_record/associations/preloader/singular_association.rb +0 -21
  256. data/lib/active_record/attribute_methods/deprecated_underscore_read.rb +0 -32
  257. data/lib/active_record/connection_adapters/abstract/connection_specification.rb +0 -191
  258. data/lib/active_record/connection_adapters/mysql_adapter.rb +0 -441
  259. data/lib/active_record/connection_adapters/sqlite_adapter.rb +0 -583
  260. data/lib/active_record/dynamic_finder_match.rb +0 -68
  261. data/lib/active_record/dynamic_scope_match.rb +0 -23
  262. data/lib/active_record/fixtures/file.rb +0 -65
  263. data/lib/active_record/identity_map.rb +0 -162
  264. data/lib/active_record/observer.rb +0 -121
  265. data/lib/active_record/railties/jdbcmysql_error.rb +0 -16
  266. data/lib/active_record/serializers/xml_serializer.rb +0 -203
  267. data/lib/active_record/session_store.rb +0 -360
  268. data/lib/active_record/test_case.rb +0 -73
  269. data/lib/rails/generators/active_record/migration/templates/migration.rb +0 -34
  270. data/lib/rails/generators/active_record/model/templates/migration.rb +0 -15
  271. data/lib/rails/generators/active_record/model/templates/model.rb +0 -12
  272. data/lib/rails/generators/active_record/observer/observer_generator.rb +0 -15
  273. data/lib/rails/generators/active_record/observer/templates/observer.rb +0 -4
  274. data/lib/rails/generators/active_record/session_migration/session_migration_generator.rb +0 -25
  275. data/lib/rails/generators/active_record/session_migration/templates/migration.rb +0 -12
@@ -0,0 +1,26 @@
1
+ # frozen_string_literal: true
2
+
3
+ module ActiveRecord
4
+ module Validations
5
+ class LengthValidator < ActiveModel::Validations::LengthValidator # :nodoc:
6
+ def validate_each(record, attribute, association_or_value)
7
+ if association_or_value.respond_to?(:loaded?) && association_or_value.loaded?
8
+ association_or_value = association_or_value.target.reject(&:marked_for_destruction?)
9
+ end
10
+ super
11
+ end
12
+ end
13
+
14
+ module ClassMethods
15
+ # Validates that the specified attributes match the length restrictions supplied.
16
+ # If the attribute is an association, records that are marked for destruction are not counted.
17
+ #
18
+ # See ActiveModel::Validations::HelperMethods.validates_length_of for more information.
19
+ def validates_length_of(*attr_names)
20
+ validates_with LengthValidator, _merge_attributes(attr_names)
21
+ end
22
+
23
+ alias_method :validates_size_of, :validates_length_of
24
+ end
25
+ end
26
+ end
@@ -0,0 +1,68 @@
1
+ # frozen_string_literal: true
2
+
3
+ module ActiveRecord
4
+ module Validations
5
+ class PresenceValidator < ActiveModel::Validations::PresenceValidator # :nodoc:
6
+ def validate_each(record, attribute, association_or_value)
7
+ if record.class._reflect_on_association(attribute)
8
+ association_or_value = Array.wrap(association_or_value).reject(&:marked_for_destruction?)
9
+ end
10
+ super
11
+ end
12
+ end
13
+
14
+ module ClassMethods
15
+ # Validates that the specified attributes are not blank (as defined by
16
+ # Object#blank?), and, if the attribute is an association, that the
17
+ # associated object is not marked for destruction. Happens by default
18
+ # on save.
19
+ #
20
+ # class Person < ActiveRecord::Base
21
+ # has_one :face
22
+ # validates_presence_of :face
23
+ # end
24
+ #
25
+ # The face attribute must be in the object and it cannot be blank or marked
26
+ # for destruction.
27
+ #
28
+ # If you want to validate the presence of a boolean field (where the real values
29
+ # are true and false), you will want to use
30
+ # <tt>validates_inclusion_of :field_name, in: [true, false]</tt>.
31
+ #
32
+ # This is due to the way Object#blank? handles boolean values:
33
+ # <tt>false.blank? # => true</tt>.
34
+ #
35
+ # This validator defers to the Active Model validation for presence, adding the
36
+ # check to see that an associated object is not marked for destruction. This
37
+ # prevents the parent object from validating successfully and saving, which then
38
+ # deletes the associated object, thus putting the parent object into an invalid
39
+ # state.
40
+ #
41
+ # NOTE: This validation will not fail while using it with an association
42
+ # if the latter was assigned but not valid. If you want to ensure that
43
+ # it is both present and valid, you also need to use
44
+ # {validates_associated}[rdoc-ref:Validations::ClassMethods#validates_associated].
45
+ #
46
+ # Configuration options:
47
+ # * <tt>:message</tt> - A custom error message (default is: "can't be blank").
48
+ # * <tt>:on</tt> - Specifies the contexts where this validation is active.
49
+ # Runs in all validation contexts by default +nil+. You can pass a symbol
50
+ # or an array of symbols. (e.g. <tt>on: :create</tt> or
51
+ # <tt>on: :custom_validation_context</tt> or
52
+ # <tt>on: [:create, :custom_validation_context]</tt>)
53
+ # * <tt>:if</tt> - Specifies a method, proc or string to call to determine if
54
+ # the validation should occur (e.g. <tt>if: :allow_validation</tt>, or
55
+ # <tt>if: Proc.new { |user| user.signup_step > 2 }</tt>). The method, proc
56
+ # or string should return or evaluate to a +true+ or +false+ value.
57
+ # * <tt>:unless</tt> - Specifies a method, proc or string to call to determine
58
+ # if the validation should not occur (e.g. <tt>unless: :skip_validation</tt>,
59
+ # or <tt>unless: Proc.new { |user| user.signup_step <= 2 }</tt>). The method,
60
+ # proc or string should return or evaluate to a +true+ or +false+ value.
61
+ # * <tt>:strict</tt> - Specifies whether validation should be strict.
62
+ # See ActiveModel::Validations#validates! for more information.
63
+ def validates_presence_of(*attr_names)
64
+ validates_with PresenceValidator, _merge_attributes(attr_names)
65
+ end
66
+ end
67
+ end
68
+ end
@@ -1,116 +1,177 @@
1
- require 'active_support/core_ext/array/wrap'
1
+ # frozen_string_literal: true
2
2
 
3
3
  module ActiveRecord
4
4
  module Validations
5
- class UniquenessValidator < ActiveModel::EachValidator
5
+ class UniquenessValidator < ActiveModel::EachValidator # :nodoc:
6
6
  def initialize(options)
7
- super(options.reverse_merge(:case_sensitive => true))
8
- end
9
-
10
- # Unfortunately, we have to tie Uniqueness validators to a class.
11
- def setup(klass)
12
- @klass = klass
7
+ if options[:conditions] && !options[:conditions].respond_to?(:call)
8
+ raise ArgumentError, "#{options[:conditions]} was passed as :conditions but is not callable. " \
9
+ "Pass a callable instead: `conditions: -> { where(approved: true) }`"
10
+ end
11
+ unless Array(options[:scope]).all? { |scope| scope.respond_to?(:to_sym) }
12
+ raise ArgumentError, "#{options[:scope]} is not supported format for :scope option. " \
13
+ "Pass a symbol or an array of symbols instead: `scope: :user_id`"
14
+ end
15
+ super({ case_sensitive: true }.merge!(options))
16
+ @klass = options[:class]
13
17
  end
14
18
 
15
19
  def validate_each(record, attribute, value)
16
20
  finder_class = find_finder_class_for(record)
17
- table = finder_class.arel_table
18
-
19
- coder = record.class.serialized_attributes[attribute.to_s]
20
-
21
- if value && coder
22
- value = coder.dump value
21
+ value = map_enum_attribute(finder_class, attribute, value)
22
+
23
+ relation = build_relation(finder_class, attribute, value)
24
+ if record.persisted?
25
+ if finder_class.primary_key
26
+ relation = relation.where.not(finder_class.primary_key => record.id_in_database)
27
+ else
28
+ raise UnknownPrimaryKey.new(finder_class, "Can not validate uniqueness for persisted record without primary key.")
29
+ end
23
30
  end
31
+ relation = scope_relation(record, relation)
32
+ relation = relation.merge(options[:conditions]) if options[:conditions]
24
33
 
25
- relation = build_relation(finder_class, table, attribute, value)
26
- relation = relation.and(table[finder_class.primary_key.to_sym].not_eq(record.send(:id))) if record.persisted?
34
+ if relation.exists?
35
+ error_options = options.except(:case_sensitive, :scope, :conditions)
36
+ error_options[:value] = value
27
37
 
28
- Array.wrap(options[:scope]).each do |scope_item|
29
- scope_value = record.read_attribute(scope_item)
30
- relation = relation.and(table[scope_item].eq(scope_value))
31
- end
32
-
33
- if finder_class.unscoped.where(relation).exists?
34
- record.errors.add(attribute, :taken, options.except(:case_sensitive, :scope).merge(:value => value))
38
+ record.errors.add(attribute, :taken, error_options)
35
39
  end
36
40
  end
37
41
 
38
- protected
39
-
42
+ private
40
43
  # The check for an existing value should be run from a class that
41
44
  # isn't abstract. This means working down from the current class
42
45
  # (self), to the first non-abstract class. Since classes don't know
43
46
  # their subclasses, we have to build the hierarchy between self and
44
47
  # the record's class.
45
- def find_finder_class_for(record) #:nodoc:
48
+ def find_finder_class_for(record)
46
49
  class_hierarchy = [record.class]
47
50
 
48
51
  while class_hierarchy.first != @klass
49
- class_hierarchy.insert(0, class_hierarchy.first.superclass)
52
+ class_hierarchy.unshift(class_hierarchy.first.superclass)
50
53
  end
51
54
 
52
55
  class_hierarchy.detect { |klass| !klass.abstract_class? }
53
56
  end
54
57
 
55
- def build_relation(klass, table, attribute, value) #:nodoc:
56
- column = klass.columns_hash[attribute.to_s]
57
- value = column.limit ? value.to_s.mb_chars[0, column.limit] : value.to_s if value && column.text?
58
+ def build_relation(klass, attribute, value)
59
+ if reflection = klass._reflect_on_association(attribute)
60
+ attribute = reflection.foreign_key
61
+ value = value.attributes[reflection.klass.primary_key] unless value.nil?
62
+ end
63
+
64
+ if value.nil?
65
+ return klass.unscoped.where!(attribute => value)
66
+ end
67
+
68
+ # the attribute may be an aliased attribute
69
+ if klass.attribute_alias?(attribute)
70
+ attribute = klass.attribute_alias(attribute)
71
+ end
72
+
73
+ attribute_name = attribute.to_s
74
+ value = klass.predicate_builder.build_bind_attribute(attribute_name, value)
58
75
 
59
- if !options[:case_sensitive] && value && column.text?
76
+ table = klass.arel_table
77
+ column = klass.columns_hash[attribute_name]
78
+
79
+ comparison = if !options[:case_sensitive]
60
80
  # will use SQL LOWER function before comparison, unless it detects a case insensitive collation
61
- relation = klass.connection.case_insensitive_comparison(table, attribute, column, value)
81
+ klass.connection.case_insensitive_comparison(table, attribute, column, value)
62
82
  else
63
- value = klass.connection.case_sensitive_modifier(value) if value
64
- relation = table[attribute].eq(value)
83
+ klass.connection.case_sensitive_comparison(table, attribute, column, value)
84
+ end
85
+ klass.unscoped.where!(comparison)
86
+ end
87
+
88
+ def scope_relation(record, relation)
89
+ Array(options[:scope]).each do |scope_item|
90
+ scope_value = if record.class._reflect_on_association(scope_item)
91
+ record.association(scope_item).reader
92
+ else
93
+ record._read_attribute(scope_item)
94
+ end
95
+ relation = relation.where(scope_item => scope_value)
65
96
  end
66
97
 
67
98
  relation
68
99
  end
100
+
101
+ def map_enum_attribute(klass, attribute, value)
102
+ mapping = klass.defined_enums[attribute.to_s]
103
+ value = mapping[value] if value && mapping
104
+ value
105
+ end
69
106
  end
70
107
 
71
108
  module ClassMethods
72
- # Validates whether the value of the specified attributes are unique across the system.
73
- # Useful for making sure that only one user
109
+ # Validates whether the value of the specified attributes are unique
110
+ # across the system. Useful for making sure that only one user
74
111
  # can be named "davidhh".
75
112
  #
76
113
  # class Person < ActiveRecord::Base
77
114
  # validates_uniqueness_of :user_name
78
115
  # end
79
116
  #
80
- # It can also validate whether the value of the specified attributes are unique based on a scope parameter:
117
+ # It can also validate whether the value of the specified attributes are
118
+ # unique based on a <tt>:scope</tt> parameter:
81
119
  #
82
120
  # class Person < ActiveRecord::Base
83
- # validates_uniqueness_of :user_name, :scope => :account_id
121
+ # validates_uniqueness_of :user_name, scope: :account_id
84
122
  # end
85
123
  #
86
- # Or even multiple scope parameters. For example, making sure that a teacher can only be on the schedule once
87
- # per semester for a particular class.
124
+ # Or even multiple scope parameters. For example, making sure that a
125
+ # teacher can only be on the schedule once per semester for a particular
126
+ # class.
88
127
  #
89
128
  # class TeacherSchedule < ActiveRecord::Base
90
- # validates_uniqueness_of :teacher_id, :scope => [:semester_id, :class_id]
129
+ # validates_uniqueness_of :teacher_id, scope: [:semester_id, :class_id]
130
+ # end
131
+ #
132
+ # It is also possible to limit the uniqueness constraint to a set of
133
+ # records matching certain conditions. In this example archived articles
134
+ # are not being taken into consideration when validating uniqueness
135
+ # of the title attribute:
136
+ #
137
+ # class Article < ActiveRecord::Base
138
+ # validates_uniqueness_of :title, conditions: -> { where.not(status: 'archived') }
91
139
  # end
92
140
  #
93
- # When the record is created, a check is performed to make sure that no record exists in the database
94
- # with the given value for the specified attribute (that maps to a column). When the record is updated,
141
+ # When the record is created, a check is performed to make sure that no
142
+ # record exists in the database with the given value for the specified
143
+ # attribute (that maps to a column). When the record is updated,
95
144
  # the same check is made but disregarding the record itself.
96
145
  #
97
146
  # Configuration options:
98
- # * <tt>:message</tt> - Specifies a custom error message (default is: "has already been taken").
99
- # * <tt>:scope</tt> - One or more columns by which to limit the scope of the uniqueness constraint.
100
- # * <tt>:case_sensitive</tt> - Looks for an exact match. Ignored by non-text columns (+true+ by default).
101
- # * <tt>:allow_nil</tt> - If set to true, skips this validation if the attribute is +nil+ (default is +false+).
102
- # * <tt>:allow_blank</tt> - If set to true, skips this validation if the attribute is blank (default is +false+).
103
- # * <tt>:if</tt> - Specifies a method, proc or string to call to determine if the validation should
104
- # occur (e.g. <tt>:if => :allow_validation</tt>, or <tt>:if => Proc.new { |user| user.signup_step > 2 }</tt>).
105
- # The method, proc or string should return or evaluate to a true or false value.
106
- # * <tt>:unless</tt> - Specifies a method, proc or string to call to determine if the validation should
107
- # not occur (e.g. <tt>:unless => :skip_validation</tt>, or
108
- # <tt>:unless => Proc.new { |user| user.signup_step <= 2 }</tt>). The method, proc or string should
109
- # return or evaluate to a true or false value.
147
+ #
148
+ # * <tt>:message</tt> - Specifies a custom error message (default is:
149
+ # "has already been taken").
150
+ # * <tt>:scope</tt> - One or more columns by which to limit the scope of
151
+ # the uniqueness constraint.
152
+ # * <tt>:conditions</tt> - Specify the conditions to be included as a
153
+ # <tt>WHERE</tt> SQL fragment to limit the uniqueness constraint lookup
154
+ # (e.g. <tt>conditions: -> { where(status: 'active') }</tt>).
155
+ # * <tt>:case_sensitive</tt> - Looks for an exact match. Ignored by
156
+ # non-text columns (+true+ by default).
157
+ # * <tt>:allow_nil</tt> - If set to +true+, skips this validation if the
158
+ # attribute is +nil+ (default is +false+).
159
+ # * <tt>:allow_blank</tt> - If set to +true+, skips this validation if the
160
+ # attribute is blank (default is +false+).
161
+ # * <tt>:if</tt> - Specifies a method, proc or string to call to determine
162
+ # if the validation should occur (e.g. <tt>if: :allow_validation</tt>,
163
+ # or <tt>if: Proc.new { |user| user.signup_step > 2 }</tt>). The method,
164
+ # proc or string should return or evaluate to a +true+ or +false+ value.
165
+ # * <tt>:unless</tt> - Specifies a method, proc or string to call to
166
+ # determine if the validation should not occur (e.g. <tt>unless: :skip_validation</tt>,
167
+ # or <tt>unless: Proc.new { |user| user.signup_step <= 2 }</tt>). The
168
+ # method, proc or string should return or evaluate to a +true+ or +false+
169
+ # value.
110
170
  #
111
171
  # === Concurrency and integrity
112
172
  #
113
- # Using this validation method in conjunction with ActiveRecord::Base#save
173
+ # Using this validation method in conjunction with
174
+ # {ActiveRecord::Base#save}[rdoc-ref:Persistence#save]
114
175
  # does not guarantee the absence of duplicate record insertions, because
115
176
  # uniqueness checks on the application level are inherently prone to race
116
177
  # conditions. For example, suppose that two users try to post a Comment at
@@ -126,11 +187,11 @@ module ActiveRecord
126
187
  # WHERE title = 'My Post' |
127
188
  # |
128
189
  # | # User 2 does the same thing and also
129
- # | # infers that his title is unique.
190
+ # | # infers that their title is unique.
130
191
  # | SELECT * FROM comments
131
192
  # | WHERE title = 'My Post'
132
193
  # |
133
- # # User 1 inserts his comment. |
194
+ # # User 1 inserts their comment. |
134
195
  # INSERT INTO comments |
135
196
  # (title, content) VALUES |
136
197
  # ('My Post', 'hi!') |
@@ -144,34 +205,31 @@ module ActiveRecord
144
205
  # | # Boom! We now have a duplicate
145
206
  # | # title!
146
207
  #
147
- # This could even happen if you use transactions with the 'serializable'
148
- # isolation level. The best way to work around this problem is to add a unique
149
- # index to the database table using
150
- # ActiveRecord::ConnectionAdapters::SchemaStatements#add_index. In the
151
- # rare case that a race condition occurs, the database will guarantee
208
+ # The best way to work around this problem is to add a unique index to the database table using
209
+ # {connection.add_index}[rdoc-ref:ConnectionAdapters::SchemaStatements#add_index].
210
+ # In the rare case that a race condition occurs, the database will guarantee
152
211
  # the field's uniqueness.
153
212
  #
154
213
  # When the database catches such a duplicate insertion,
155
- # ActiveRecord::Base#save will raise an ActiveRecord::StatementInvalid
214
+ # {ActiveRecord::Base#save}[rdoc-ref:Persistence#save] will raise an ActiveRecord::StatementInvalid
156
215
  # exception. You can either choose to let this error propagate (which
157
216
  # will result in the default Rails exception page being shown), or you
158
217
  # can catch it and restart the transaction (e.g. by telling the user
159
- # that the title already exists, and asking him to re-enter the title).
160
- # This technique is also known as optimistic concurrency control:
161
- # http://en.wikipedia.org/wiki/Optimistic_concurrency_control
218
+ # that the title already exists, and asking them to re-enter the title).
219
+ # This technique is also known as
220
+ # {optimistic concurrency control}[https://en.wikipedia.org/wiki/Optimistic_concurrency_control].
162
221
  #
163
222
  # The bundled ActiveRecord::ConnectionAdapters distinguish unique index
164
223
  # constraint errors from other types of database errors by throwing an
165
- # ActiveRecord::RecordNotUnique exception.
166
- # For other adapters you will have to parse the (database-specific) exception
167
- # message to detect such a case.
224
+ # ActiveRecord::RecordNotUnique exception. For other adapters you will
225
+ # have to parse the (database-specific) exception message to detect such
226
+ # a case.
227
+ #
168
228
  # The following bundled adapters throw the ActiveRecord::RecordNotUnique exception:
169
- # * ActiveRecord::ConnectionAdapters::MysqlAdapter
170
- # * ActiveRecord::ConnectionAdapters::Mysql2Adapter
171
- # * ActiveRecord::ConnectionAdapters::SQLiteAdapter
172
- # * ActiveRecord::ConnectionAdapters::SQLite3Adapter
173
- # * ActiveRecord::ConnectionAdapters::PostgreSQLAdapter
174
229
  #
230
+ # * ActiveRecord::ConnectionAdapters::Mysql2Adapter.
231
+ # * ActiveRecord::ConnectionAdapters::SQLite3Adapter.
232
+ # * ActiveRecord::ConnectionAdapters::PostgreSQLAdapter.
175
233
  def validates_uniqueness_of(*attr_names)
176
234
  validates_with UniquenessValidator, _merge_attributes(attr_names)
177
235
  end
@@ -1,83 +1,93 @@
1
+ # frozen_string_literal: true
2
+
1
3
  module ActiveRecord
2
- # = Active Record RecordInvalid
4
+ # = Active Record \RecordInvalid
3
5
  #
4
- # Raised by <tt>save!</tt> and <tt>create!</tt> when the record is invalid. Use the
5
- # +record+ method to retrieve the record which did not validate.
6
+ # Raised by {ActiveRecord::Base#save!}[rdoc-ref:Persistence#save!] and
7
+ # {ActiveRecord::Base#create!}[rdoc-ref:Persistence::ClassMethods#create!] when the record is invalid.
8
+ # Use the #record method to retrieve the record which did not validate.
6
9
  #
7
10
  # begin
8
- # complex_operation_that_calls_save!_internally
11
+ # complex_operation_that_internally_calls_save!
9
12
  # rescue ActiveRecord::RecordInvalid => invalid
10
13
  # puts invalid.record.errors
11
14
  # end
12
15
  class RecordInvalid < ActiveRecordError
13
16
  attr_reader :record
14
- def initialize(record)
15
- @record = record
16
- errors = @record.errors.full_messages.join(", ")
17
- super(I18n.t("activerecord.errors.messages.record_invalid", :errors => errors))
17
+
18
+ def initialize(record = nil)
19
+ if record
20
+ @record = record
21
+ errors = @record.errors.full_messages.join(", ")
22
+ message = I18n.t(:"#{@record.class.i18n_scope}.errors.messages.record_invalid", errors: errors, default: :"errors.messages.record_invalid")
23
+ else
24
+ message = "Record invalid"
25
+ end
26
+
27
+ super(message)
18
28
  end
19
29
  end
20
30
 
21
- # = Active Record Validations
31
+ # = Active Record \Validations
22
32
  #
23
- # Active Record includes the majority of its validations from <tt>ActiveModel::Validations</tt>
33
+ # Active Record includes the majority of its validations from ActiveModel::Validations
24
34
  # all of which accept the <tt>:on</tt> argument to define the context where the
25
35
  # validations are active. Active Record will always supply either the context of
26
36
  # <tt>:create</tt> or <tt>:update</tt> dependent on whether the model is a
27
- # <tt>new_record?</tt>.
37
+ # {new_record?}[rdoc-ref:Persistence#new_record?].
28
38
  module Validations
29
39
  extend ActiveSupport::Concern
30
40
  include ActiveModel::Validations
31
41
 
32
- module ClassMethods
33
- # Creates an object just like Base.create but calls <tt>save!</tt> instead of +save+
34
- # so an exception is raised if the record is invalid.
35
- def create!(attributes = nil, options = {}, &block)
36
- if attributes.is_a?(Array)
37
- attributes.collect { |attr| create!(attr, options, &block) }
38
- else
39
- object = new(attributes, options)
40
- yield(object) if block_given?
41
- object.save!
42
- object
43
- end
44
- end
45
- end
46
-
47
- # The validation process on save can be skipped by passing <tt>:validate => false</tt>. The regular Base#save method is
48
- # replaced with this when the validations module is mixed in, which it is by default.
49
- def save(options={})
42
+ # The validation process on save can be skipped by passing <tt>validate: false</tt>.
43
+ # The regular {ActiveRecord::Base#save}[rdoc-ref:Persistence#save] method is replaced
44
+ # with this when the validations module is mixed in, which it is by default.
45
+ def save(options = {})
50
46
  perform_validations(options) ? super : false
51
47
  end
52
48
 
53
- # Attempts to save the record just like Base#save but will raise a +RecordInvalid+ exception instead of returning false
54
- # if the record is not valid.
55
- def save!(options={})
56
- perform_validations(options) ? super : raise(RecordInvalid.new(self))
49
+ # Attempts to save the record just like {ActiveRecord::Base#save}[rdoc-ref:Base#save] but
50
+ # will raise an ActiveRecord::RecordInvalid exception instead of returning +false+ if the record is not valid.
51
+ def save!(options = {})
52
+ perform_validations(options) ? super : raise_validation_error
57
53
  end
58
54
 
59
- # Runs all the validations within the specified context. Returns true if no errors are found,
60
- # false otherwise.
55
+ # Runs all the validations within the specified context. Returns +true+ if
56
+ # no errors are found, +false+ otherwise.
61
57
  #
62
- # If the argument is false (default is +nil+), the context is set to <tt>:create</tt> if
63
- # <tt>new_record?</tt> is true, and to <tt>:update</tt> if it is not.
58
+ # Aliased as #validate.
64
59
  #
65
- # Validations with no <tt>:on</tt> option will run no matter the context. Validations with
60
+ # If the argument is +false+ (default is +nil+), the context is set to <tt>:create</tt> if
61
+ # {new_record?}[rdoc-ref:Persistence#new_record?] is +true+, and to <tt>:update</tt> if it is not.
62
+ #
63
+ # \Validations with no <tt>:on</tt> option will run no matter the context. \Validations with
66
64
  # some <tt>:on</tt> option will only run in the specified context.
67
65
  def valid?(context = nil)
68
- context ||= (new_record? ? :create : :update)
66
+ context ||= default_validation_context
69
67
  output = super(context)
70
68
  errors.empty? && output
71
69
  end
72
70
 
73
- protected
71
+ alias_method :validate, :valid?
72
+
73
+ private
74
+
75
+ def default_validation_context
76
+ new_record? ? :create : :update
77
+ end
78
+
79
+ def raise_validation_error
80
+ raise(RecordInvalid.new(self))
81
+ end
74
82
 
75
- def perform_validations(options={})
76
- perform_validation = options[:validate] != false
77
- perform_validation ? valid?(options[:context]) : true
83
+ def perform_validations(options = {})
84
+ options[:validate] == false || valid?(options[:context])
78
85
  end
79
86
  end
80
87
  end
81
88
 
82
89
  require "active_record/validations/associated"
83
90
  require "active_record/validations/uniqueness"
91
+ require "active_record/validations/presence"
92
+ require "active_record/validations/absence"
93
+ require "active_record/validations/length"
@@ -1,10 +1,10 @@
1
- module ActiveRecord
2
- module VERSION #:nodoc:
3
- MAJOR = 3
4
- MINOR = 2
5
- TINY = 22
6
- PRE = "5"
1
+ # frozen_string_literal: true
2
+
3
+ require_relative "gem_version"
7
4
 
8
- STRING = [MAJOR, MINOR, TINY, PRE].compact.join('.')
5
+ module ActiveRecord
6
+ # Returns the version of the currently loaded ActiveRecord as a <tt>Gem::Version</tt>
7
+ def self.version
8
+ gem_version
9
9
  end
10
10
  end