ibm_db 3.0.3-x86-mingw32 → 5.0.3-x86-mingw32

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.
Files changed (593) hide show
  1. checksums.yaml +5 -5
  2. data/CHANGES +10 -0
  3. data/LICENSE +1 -1
  4. data/MANIFEST +14 -14
  5. data/ParameterizedQueries README +6 -6
  6. data/README +208 -225
  7. data/ext/Makefile +269 -0
  8. data/ext/Makefile.nt32 +181 -181
  9. data/ext/Makefile.nt32.191 +212 -212
  10. data/ext/extconf.rb +322 -291
  11. data/ext/gil_release_version +3 -0
  12. data/ext/ibm_db-i386-mingw32.def +2 -0
  13. data/ext/ibm_db.c +11879 -11884
  14. data/ext/ibm_db.o +0 -0
  15. data/ext/ibm_db.so +0 -0
  16. data/ext/mkmf.log +110 -0
  17. data/ext/ruby_ibm_db.h +241 -241
  18. data/ext/ruby_ibm_db_cli.c +866 -866
  19. data/ext/ruby_ibm_db_cli.h +500 -500
  20. data/ext/ruby_ibm_db_cli.o +0 -0
  21. data/ext/unicode_support_version +3 -0
  22. data/init.rb +41 -41
  23. data/lib/IBM_DB.rb +27 -27
  24. data/lib/active_record/connection_adapters/ibm_db_adapter.rb +3561 -3186
  25. data/lib/active_record/connection_adapters/ibmdb_adapter.rb +5 -2
  26. data/lib/active_record/vendor/db2-i5-zOS.yaml +328 -328
  27. data/lib/mswin32/ibm_db.rb +91 -123
  28. data/lib/mswin32/rb2x/i386/ibm_db.so +0 -0
  29. data/test/active_record/connection_adapters/fake_adapter.rb +49 -46
  30. data/test/assets/example.log +1 -1
  31. data/test/assets/test.txt +1 -1
  32. data/test/cases/adapter_test.rb +351 -261
  33. data/test/cases/adapters/mysql2/active_schema_test.rb +193 -0
  34. data/test/cases/adapters/mysql2/bind_parameter_test.rb +50 -0
  35. data/test/cases/adapters/mysql2/boolean_test.rb +100 -0
  36. data/test/cases/adapters/mysql2/case_sensitivity_test.rb +63 -0
  37. data/test/cases/adapters/mysql2/charset_collation_test.rb +54 -0
  38. data/test/cases/adapters/mysql2/connection_test.rb +210 -0
  39. data/test/cases/adapters/mysql2/datetime_precision_quoting_test.rb +45 -0
  40. data/test/cases/adapters/mysql2/enum_test.rb +26 -0
  41. data/test/cases/adapters/mysql2/explain_test.rb +21 -0
  42. data/test/cases/adapters/mysql2/json_test.rb +195 -0
  43. data/test/cases/adapters/mysql2/mysql2_adapter_test.rb +83 -0
  44. data/test/cases/adapters/mysql2/reserved_word_test.rb +152 -0
  45. data/test/cases/adapters/mysql2/schema_migrations_test.rb +59 -0
  46. data/test/cases/adapters/mysql2/schema_test.rb +126 -0
  47. data/test/cases/adapters/mysql2/sp_test.rb +36 -0
  48. data/test/cases/adapters/mysql2/sql_types_test.rb +14 -0
  49. data/test/cases/adapters/mysql2/table_options_test.rb +42 -0
  50. data/test/cases/adapters/mysql2/unsigned_type_test.rb +66 -0
  51. data/test/cases/adapters/postgresql/active_schema_test.rb +98 -0
  52. data/test/cases/adapters/postgresql/array_test.rb +339 -0
  53. data/test/cases/adapters/postgresql/bit_string_test.rb +82 -0
  54. data/test/cases/adapters/postgresql/bytea_test.rb +134 -0
  55. data/test/cases/adapters/postgresql/case_insensitive_test.rb +26 -0
  56. data/test/cases/adapters/postgresql/change_schema_test.rb +38 -0
  57. data/test/cases/adapters/postgresql/cidr_test.rb +25 -0
  58. data/test/cases/adapters/postgresql/citext_test.rb +78 -0
  59. data/test/cases/adapters/postgresql/collation_test.rb +53 -0
  60. data/test/cases/adapters/postgresql/composite_test.rb +132 -0
  61. data/test/cases/adapters/postgresql/connection_test.rb +257 -0
  62. data/test/cases/adapters/postgresql/datatype_test.rb +92 -0
  63. data/test/cases/adapters/postgresql/domain_test.rb +47 -0
  64. data/test/cases/adapters/postgresql/enum_test.rb +91 -0
  65. data/test/cases/adapters/postgresql/explain_test.rb +20 -0
  66. data/test/cases/adapters/postgresql/extension_migration_test.rb +63 -0
  67. data/test/cases/adapters/postgresql/full_text_test.rb +44 -0
  68. data/test/cases/adapters/postgresql/geometric_test.rb +378 -0
  69. data/test/cases/adapters/postgresql/hstore_test.rb +382 -0
  70. data/test/cases/adapters/postgresql/infinity_test.rb +69 -0
  71. data/test/cases/adapters/postgresql/integer_test.rb +25 -0
  72. data/test/cases/adapters/postgresql/json_test.rb +237 -0
  73. data/test/cases/adapters/postgresql/ltree_test.rb +53 -0
  74. data/test/cases/adapters/postgresql/money_test.rb +96 -0
  75. data/test/cases/adapters/postgresql/network_test.rb +94 -0
  76. data/test/cases/adapters/postgresql/numbers_test.rb +49 -0
  77. data/test/cases/adapters/postgresql/postgresql_adapter_test.rb +405 -0
  78. data/test/cases/adapters/postgresql/prepared_statements_test.rb +22 -0
  79. data/test/cases/adapters/postgresql/quoting_test.rb +44 -0
  80. data/test/cases/adapters/postgresql/range_test.rb +343 -0
  81. data/test/cases/adapters/postgresql/referential_integrity_test.rb +111 -0
  82. data/test/cases/adapters/postgresql/rename_table_test.rb +34 -0
  83. data/test/cases/adapters/postgresql/schema_authorization_test.rb +119 -0
  84. data/test/cases/adapters/postgresql/schema_test.rb +597 -0
  85. data/test/cases/adapters/postgresql/serial_test.rb +154 -0
  86. data/test/cases/adapters/postgresql/statement_pool_test.rb +41 -0
  87. data/test/cases/adapters/postgresql/timestamp_test.rb +90 -0
  88. data/test/cases/adapters/postgresql/type_lookup_test.rb +33 -0
  89. data/test/cases/adapters/postgresql/utils_test.rb +62 -0
  90. data/test/cases/adapters/postgresql/uuid_test.rb +294 -0
  91. data/test/cases/adapters/postgresql/xml_test.rb +54 -0
  92. data/test/cases/adapters/sqlite3/collation_test.rb +53 -0
  93. data/test/cases/adapters/sqlite3/copy_table_test.rb +98 -0
  94. data/test/cases/adapters/sqlite3/explain_test.rb +21 -0
  95. data/test/cases/adapters/sqlite3/quoting_test.rb +101 -0
  96. data/test/cases/adapters/sqlite3/sqlite3_adapter_test.rb +441 -0
  97. data/test/cases/adapters/sqlite3/sqlite3_create_folder_test.rb +24 -0
  98. data/test/cases/adapters/sqlite3/statement_pool_test.rb +20 -0
  99. data/test/cases/aggregations_test.rb +168 -158
  100. data/test/cases/ar_schema_test.rb +146 -161
  101. data/test/cases/associations/association_scope_test.rb +16 -21
  102. data/test/cases/associations/belongs_to_associations_test.rb +1141 -1029
  103. data/test/cases/associations/bidirectional_destroy_dependencies_test.rb +41 -0
  104. data/test/cases/associations/callbacks_test.rb +190 -192
  105. data/test/cases/associations/cascaded_eager_loading_test.rb +188 -188
  106. data/test/cases/associations/eager_load_includes_full_sti_class_test.rb +36 -36
  107. data/test/cases/associations/eager_load_nested_include_test.rb +126 -128
  108. data/test/cases/associations/eager_singularization_test.rb +148 -148
  109. data/test/cases/associations/eager_test.rb +1514 -1411
  110. data/test/cases/associations/extension_test.rb +87 -82
  111. data/test/cases/associations/has_and_belongs_to_many_associations_test.rb +1004 -932
  112. data/test/cases/associations/has_many_associations_test.rb +2501 -2162
  113. data/test/cases/associations/has_many_through_associations_test.rb +1271 -1204
  114. data/test/cases/associations/has_one_associations_test.rb +707 -610
  115. data/test/cases/associations/has_one_through_associations_test.rb +383 -380
  116. data/test/cases/associations/inner_join_association_test.rb +139 -139
  117. data/test/cases/associations/inverse_associations_test.rb +733 -693
  118. data/test/cases/associations/join_model_test.rb +777 -754
  119. data/test/cases/associations/left_outer_join_association_test.rb +88 -0
  120. data/test/cases/associations/nested_through_associations_test.rb +579 -579
  121. data/test/cases/associations/required_test.rb +102 -82
  122. data/test/cases/associations_test.rb +385 -380
  123. data/test/cases/attribute_decorators_test.rb +126 -125
  124. data/test/cases/attribute_methods/read_test.rb +60 -60
  125. data/test/cases/attribute_methods_test.rb +1009 -952
  126. data/test/cases/attribute_set_test.rb +270 -200
  127. data/test/cases/attribute_test.rb +246 -180
  128. data/test/cases/attributes_test.rb +253 -136
  129. data/test/cases/autosave_association_test.rb +1708 -1595
  130. data/test/cases/base_test.rb +1713 -1638
  131. data/test/cases/batches_test.rb +489 -212
  132. data/test/cases/binary_test.rb +44 -52
  133. data/test/cases/bind_parameter_test.rb +110 -100
  134. data/test/cases/cache_key_test.rb +26 -0
  135. data/test/cases/calculations_test.rb +798 -646
  136. data/test/cases/callbacks_test.rb +636 -543
  137. data/test/cases/clone_test.rb +40 -40
  138. data/test/cases/coders/json_test.rb +15 -0
  139. data/test/cases/coders/yaml_column_test.rb +63 -63
  140. data/test/cases/collection_cache_key_test.rb +115 -0
  141. data/test/cases/column_alias_test.rb +17 -17
  142. data/test/cases/column_definition_test.rb +92 -123
  143. data/test/cases/comment_test.rb +145 -0
  144. data/test/cases/connection_adapters/adapter_leasing_test.rb +56 -54
  145. data/test/cases/connection_adapters/connection_handler_test.rb +160 -53
  146. data/test/cases/connection_adapters/connection_specification_test.rb +12 -12
  147. data/test/cases/connection_adapters/merge_and_resolve_default_url_config_test.rb +255 -293
  148. data/test/cases/connection_adapters/mysql_type_lookup_test.rb +69 -65
  149. data/test/cases/connection_adapters/quoting_test.rb +13 -13
  150. data/test/cases/connection_adapters/schema_cache_test.rb +61 -56
  151. data/test/cases/connection_adapters/type_lookup_test.rb +118 -110
  152. data/test/cases/connection_management_test.rb +112 -122
  153. data/test/cases/connection_pool_test.rb +521 -346
  154. data/test/cases/connection_specification/resolver_test.rb +131 -116
  155. data/test/cases/core_test.rb +112 -112
  156. data/test/cases/counter_cache_test.rb +214 -209
  157. data/test/cases/custom_locking_test.rb +17 -17
  158. data/test/cases/database_statements_test.rb +34 -19
  159. data/test/cases/{invalid_date_test.rb → date_test.rb} +44 -32
  160. data/test/cases/date_time_precision_test.rb +107 -0
  161. data/test/cases/date_time_test.rb +61 -61
  162. data/test/cases/defaults_test.rb +219 -223
  163. data/test/cases/dirty_test.rb +763 -775
  164. data/test/cases/disconnected_test.rb +30 -28
  165. data/test/cases/dup_test.rb +157 -157
  166. data/test/cases/enum_test.rb +444 -290
  167. data/test/cases/errors_test.rb +16 -0
  168. data/test/cases/explain_subscriber_test.rb +64 -64
  169. data/test/cases/explain_test.rb +87 -76
  170. data/test/cases/finder_respond_to_test.rb +60 -60
  171. data/test/cases/finder_test.rb +1294 -1166
  172. data/test/cases/fixture_set/file_test.rb +156 -138
  173. data/test/cases/fixtures_test.rb +988 -897
  174. data/test/cases/forbidden_attributes_protection_test.rb +165 -99
  175. data/test/cases/habtm_destroy_order_test.rb +61 -61
  176. data/test/cases/helper.rb +204 -210
  177. data/test/cases/hot_compatibility_test.rb +142 -54
  178. data/test/cases/i18n_test.rb +45 -45
  179. data/test/cases/inheritance_test.rb +606 -375
  180. data/test/cases/integration_test.rb +155 -139
  181. data/test/cases/invalid_connection_test.rb +24 -22
  182. data/test/cases/invertible_migration_test.rb +387 -295
  183. data/test/cases/json_serialization_test.rb +311 -302
  184. data/test/cases/locking_test.rb +493 -477
  185. data/test/cases/log_subscriber_test.rb +225 -136
  186. data/test/cases/migration/change_schema_test.rb +458 -472
  187. data/test/cases/migration/change_table_test.rb +256 -224
  188. data/test/cases/migration/column_attributes_test.rb +176 -192
  189. data/test/cases/migration/column_positioning_test.rb +56 -56
  190. data/test/cases/migration/columns_test.rb +310 -304
  191. data/test/cases/migration/command_recorder_test.rb +350 -305
  192. data/test/cases/migration/compatibility_test.rb +118 -0
  193. data/test/cases/migration/create_join_table_test.rb +157 -148
  194. data/test/cases/migration/foreign_key_test.rb +362 -360
  195. data/test/cases/migration/helper.rb +39 -39
  196. data/test/cases/migration/index_test.rb +218 -216
  197. data/test/cases/migration/logger_test.rb +36 -36
  198. data/test/cases/migration/pending_migrations_test.rb +52 -53
  199. data/test/cases/migration/references_foreign_key_test.rb +221 -214
  200. data/test/cases/migration/references_index_test.rb +101 -101
  201. data/test/cases/migration/references_statements_test.rb +136 -116
  202. data/test/cases/migration/rename_table_test.rb +93 -93
  203. data/test/cases/migration_test.rb +1157 -959
  204. data/test/cases/migrator_test.rb +471 -388
  205. data/test/cases/mixin_test.rb +68 -70
  206. data/test/cases/modules_test.rb +172 -173
  207. data/test/cases/multiparameter_attributes_test.rb +372 -350
  208. data/test/cases/multiple_db_test.rb +122 -115
  209. data/test/cases/nested_attributes_test.rb +1098 -1057
  210. data/test/cases/nested_attributes_with_callbacks_test.rb +144 -144
  211. data/test/cases/persistence_test.rb +1001 -909
  212. data/test/cases/pooled_connections_test.rb +81 -81
  213. data/test/cases/primary_keys_test.rb +376 -237
  214. data/test/cases/query_cache_test.rb +446 -326
  215. data/test/cases/quoting_test.rb +202 -156
  216. data/test/cases/readonly_test.rb +119 -118
  217. data/test/cases/reaper_test.rb +85 -85
  218. data/test/cases/reflection_test.rb +509 -454
  219. data/test/cases/relation/delegation_test.rb +63 -68
  220. data/test/cases/relation/merging_test.rb +157 -161
  221. data/test/cases/relation/mutation_test.rb +183 -165
  222. data/test/cases/relation/or_test.rb +92 -0
  223. data/test/cases/relation/predicate_builder_test.rb +16 -14
  224. data/test/cases/relation/record_fetch_warning_test.rb +40 -0
  225. data/test/cases/relation/where_chain_test.rb +105 -181
  226. data/test/cases/relation/where_clause_test.rb +182 -0
  227. data/test/cases/relation/where_test.rb +322 -300
  228. data/test/cases/relation_test.rb +328 -297
  229. data/test/cases/relations_test.rb +2026 -1815
  230. data/test/cases/reload_models_test.rb +22 -22
  231. data/test/cases/result_test.rb +90 -80
  232. data/test/cases/sanitize_test.rb +176 -83
  233. data/test/cases/schema_dumper_test.rb +457 -463
  234. data/test/cases/schema_loading_test.rb +52 -0
  235. data/test/cases/scoping/default_scoping_test.rb +528 -454
  236. data/test/cases/scoping/named_scoping_test.rb +561 -524
  237. data/test/cases/scoping/relation_scoping_test.rb +400 -357
  238. data/test/cases/secure_token_test.rb +32 -0
  239. data/test/cases/serialization_test.rb +104 -104
  240. data/test/cases/serialized_attribute_test.rb +364 -277
  241. data/test/cases/statement_cache_test.rb +136 -98
  242. data/test/cases/store_test.rb +195 -194
  243. data/test/cases/suppressor_test.rb +63 -0
  244. data/test/cases/tasks/database_tasks_test.rb +462 -396
  245. data/test/cases/tasks/mysql_rake_test.rb +345 -311
  246. data/test/cases/tasks/postgresql_rake_test.rb +304 -245
  247. data/test/cases/tasks/sqlite_rake_test.rb +220 -193
  248. data/test/cases/test_case.rb +131 -123
  249. data/test/cases/test_fixtures_test.rb +36 -0
  250. data/test/cases/time_precision_test.rb +103 -0
  251. data/test/cases/timestamp_test.rb +501 -468
  252. data/test/cases/touch_later_test.rb +121 -0
  253. data/test/cases/transaction_callbacks_test.rb +518 -452
  254. data/test/cases/transaction_isolation_test.rb +106 -106
  255. data/test/cases/transactions_test.rb +835 -817
  256. data/test/cases/type/adapter_specific_registry_test.rb +133 -0
  257. data/test/cases/type/date_time_test.rb +14 -0
  258. data/test/cases/type/integer_test.rb +27 -121
  259. data/test/cases/type/string_test.rb +22 -36
  260. data/test/cases/type/type_map_test.rb +177 -177
  261. data/test/cases/type_test.rb +39 -0
  262. data/test/cases/types_test.rb +24 -141
  263. data/test/cases/unconnected_test.rb +33 -33
  264. data/test/cases/validations/absence_validation_test.rb +73 -0
  265. data/test/cases/validations/association_validation_test.rb +97 -86
  266. data/test/cases/validations/i18n_generate_message_validation_test.rb +84 -84
  267. data/test/cases/validations/i18n_validation_test.rb +86 -90
  268. data/test/cases/validations/length_validation_test.rb +79 -47
  269. data/test/cases/validations/presence_validation_test.rb +103 -68
  270. data/test/cases/validations/uniqueness_validation_test.rb +548 -434
  271. data/test/cases/validations_repair_helper.rb +19 -23
  272. data/test/cases/validations_test.rb +194 -165
  273. data/test/cases/view_test.rb +216 -113
  274. data/test/cases/yaml_serialization_test.rb +121 -86
  275. data/test/config.example.yml +97 -0
  276. data/test/config.rb +5 -5
  277. data/test/config.yml +154 -154
  278. data/test/connections/native_ibm_db/connection.rb +43 -43
  279. data/test/fixtures/accounts.yml +29 -29
  280. data/test/fixtures/admin/accounts.yml +2 -2
  281. data/test/fixtures/admin/randomly_named_a9.yml +7 -7
  282. data/test/fixtures/admin/randomly_named_b0.yml +7 -7
  283. data/test/fixtures/admin/users.yml +10 -10
  284. data/test/fixtures/author_addresses.yml +17 -17
  285. data/test/fixtures/author_favorites.yml +3 -3
  286. data/test/fixtures/authors.yml +23 -23
  287. data/test/fixtures/bad_posts.yml +9 -0
  288. data/test/fixtures/binaries.yml +133 -133
  289. data/test/fixtures/books.yml +31 -11
  290. data/test/fixtures/bulbs.yml +5 -5
  291. data/test/fixtures/cars.yml +9 -9
  292. data/test/fixtures/categories.yml +19 -19
  293. data/test/fixtures/categories/special_categories.yml +9 -9
  294. data/test/fixtures/categories/subsubdir/arbitrary_filename.yml +4 -4
  295. data/test/fixtures/categories_ordered.yml +7 -7
  296. data/test/fixtures/categories_posts.yml +31 -31
  297. data/test/fixtures/categorizations.yml +23 -23
  298. data/test/fixtures/clubs.yml +8 -8
  299. data/test/fixtures/collections.yml +3 -3
  300. data/test/fixtures/colleges.yml +3 -3
  301. data/test/fixtures/comments.yml +65 -65
  302. data/test/fixtures/companies.yml +67 -67
  303. data/test/fixtures/computers.yml +10 -10
  304. data/test/fixtures/content.yml +3 -0
  305. data/test/fixtures/content_positions.yml +3 -0
  306. data/test/fixtures/courses.yml +8 -8
  307. data/test/fixtures/customers.yml +25 -25
  308. data/test/fixtures/dashboards.yml +6 -6
  309. data/test/fixtures/dead_parrots.yml +5 -0
  310. data/test/fixtures/developers.yml +21 -21
  311. data/test/fixtures/developers_projects.yml +16 -16
  312. data/test/fixtures/dog_lovers.yml +7 -7
  313. data/test/fixtures/dogs.yml +4 -4
  314. data/test/fixtures/doubloons.yml +3 -3
  315. data/test/fixtures/edges.yml +5 -5
  316. data/test/fixtures/entrants.yml +14 -14
  317. data/test/fixtures/essays.yml +6 -6
  318. data/test/fixtures/faces.yml +11 -11
  319. data/test/fixtures/fk_test_has_fk.yml +3 -3
  320. data/test/fixtures/fk_test_has_pk.yml +1 -1
  321. data/test/fixtures/friendships.yml +4 -4
  322. data/test/fixtures/funny_jokes.yml +10 -10
  323. data/test/fixtures/interests.yml +33 -33
  324. data/test/fixtures/items.yml +3 -3
  325. data/test/fixtures/jobs.yml +7 -7
  326. data/test/fixtures/legacy_things.yml +3 -3
  327. data/test/fixtures/live_parrots.yml +4 -0
  328. data/test/fixtures/mateys.yml +4 -4
  329. data/test/fixtures/member_details.yml +8 -8
  330. data/test/fixtures/member_types.yml +6 -6
  331. data/test/fixtures/members.yml +11 -11
  332. data/test/fixtures/memberships.yml +34 -34
  333. data/test/fixtures/men.yml +5 -5
  334. data/test/fixtures/minimalistics.yml +2 -2
  335. data/test/fixtures/minivans.yml +5 -5
  336. data/test/fixtures/mixed_case_monkeys.yml +6 -6
  337. data/test/fixtures/mixins.yml +29 -29
  338. data/test/fixtures/movies.yml +7 -7
  339. data/test/fixtures/naked/yml/accounts.yml +1 -1
  340. data/test/fixtures/naked/yml/companies.yml +1 -1
  341. data/test/fixtures/naked/yml/courses.yml +1 -1
  342. data/test/fixtures/naked/yml/parrots.yml +2 -0
  343. data/test/fixtures/naked/yml/trees.yml +3 -0
  344. data/test/fixtures/nodes.yml +29 -0
  345. data/test/fixtures/organizations.yml +5 -5
  346. data/test/fixtures/other_comments.yml +6 -0
  347. data/test/fixtures/other_dogs.yml +2 -0
  348. data/test/fixtures/other_posts.yml +7 -0
  349. data/test/fixtures/other_topics.yml +42 -42
  350. data/test/fixtures/owners.yml +9 -9
  351. data/test/fixtures/parrots.yml +27 -27
  352. data/test/fixtures/parrots_pirates.yml +7 -7
  353. data/test/fixtures/people.yml +24 -24
  354. data/test/fixtures/peoples_treasures.yml +3 -3
  355. data/test/fixtures/pets.yml +19 -19
  356. data/test/fixtures/pirates.yml +12 -12
  357. data/test/fixtures/posts.yml +80 -80
  358. data/test/fixtures/price_estimates.yml +16 -7
  359. data/test/fixtures/products.yml +4 -4
  360. data/test/fixtures/projects.yml +7 -7
  361. data/test/fixtures/randomly_named_a9.yml +7 -7
  362. data/test/fixtures/ratings.yml +14 -14
  363. data/test/fixtures/readers.yml +11 -11
  364. data/test/fixtures/references.yml +17 -17
  365. data/test/fixtures/reserved_words/distinct.yml +5 -5
  366. data/test/fixtures/reserved_words/distinct_select.yml +11 -11
  367. data/test/fixtures/reserved_words/group.yml +14 -14
  368. data/test/fixtures/reserved_words/select.yml +8 -8
  369. data/test/fixtures/reserved_words/values.yml +7 -7
  370. data/test/fixtures/ships.yml +6 -6
  371. data/test/fixtures/speedometers.yml +8 -8
  372. data/test/fixtures/sponsors.yml +12 -12
  373. data/test/fixtures/string_key_objects.yml +7 -7
  374. data/test/fixtures/subscribers.yml +10 -10
  375. data/test/fixtures/subscriptions.yml +12 -12
  376. data/test/fixtures/taggings.yml +78 -78
  377. data/test/fixtures/tags.yml +11 -11
  378. data/test/fixtures/tasks.yml +7 -7
  379. data/test/fixtures/teapots.yml +3 -3
  380. data/test/fixtures/to_be_linked/accounts.yml +2 -2
  381. data/test/fixtures/to_be_linked/users.yml +10 -10
  382. data/test/fixtures/topics.yml +49 -49
  383. data/test/fixtures/toys.yml +14 -14
  384. data/test/fixtures/traffic_lights.yml +9 -9
  385. data/test/fixtures/treasures.yml +10 -10
  386. data/test/fixtures/trees.yml +3 -0
  387. data/test/fixtures/uuid_children.yml +3 -3
  388. data/test/fixtures/uuid_parents.yml +2 -2
  389. data/test/fixtures/variants.yml +4 -4
  390. data/test/fixtures/vegetables.yml +19 -19
  391. data/test/fixtures/vertices.yml +3 -3
  392. data/test/fixtures/warehouse_things.yml +2 -2
  393. data/test/fixtures/zines.yml +5 -5
  394. data/test/ibm_db_test.rb +24 -24
  395. data/test/migrations/10_urban/9_add_expressions.rb +11 -11
  396. data/test/migrations/decimal/1_give_me_big_numbers.rb +15 -15
  397. data/test/migrations/magic/1_currencies_have_symbols.rb +12 -12
  398. data/test/migrations/missing/1000_people_have_middle_names.rb +9 -9
  399. data/test/migrations/missing/1_people_have_last_names.rb +9 -9
  400. data/test/migrations/missing/3_we_need_reminders.rb +12 -12
  401. data/test/migrations/missing/4_innocent_jointable.rb +12 -12
  402. data/test/migrations/rename/1_we_need_things.rb +11 -11
  403. data/test/migrations/rename/2_rename_things.rb +9 -9
  404. data/test/migrations/to_copy/1_people_have_hobbies.rb +9 -9
  405. data/test/migrations/to_copy/2_people_have_descriptions.rb +9 -9
  406. data/test/migrations/to_copy2/1_create_articles.rb +7 -7
  407. data/test/migrations/to_copy2/2_create_comments.rb +7 -7
  408. data/test/migrations/to_copy_with_name_collision/1_people_have_hobbies.rb +9 -9
  409. data/test/migrations/to_copy_with_timestamps/20090101010101_people_have_hobbies.rb +9 -9
  410. data/test/migrations/to_copy_with_timestamps/20090101010202_people_have_descriptions.rb +9 -9
  411. data/test/migrations/to_copy_with_timestamps2/20090101010101_create_articles.rb +7 -7
  412. data/test/migrations/to_copy_with_timestamps2/20090101010202_create_comments.rb +7 -7
  413. data/test/migrations/valid/1_valid_people_have_last_names.rb +9 -9
  414. data/test/migrations/valid/2_we_need_reminders.rb +12 -12
  415. data/test/migrations/valid/3_innocent_jointable.rb +12 -12
  416. data/test/migrations/valid_with_subdirectories/1_valid_people_have_last_names.rb +9 -9
  417. data/test/migrations/valid_with_subdirectories/sub/2_we_need_reminders.rb +12 -12
  418. data/test/migrations/valid_with_subdirectories/sub1/3_innocent_jointable.rb +12 -12
  419. data/test/migrations/valid_with_timestamps/20100101010101_valid_with_timestamps_people_have_last_names.rb +9 -9
  420. data/test/migrations/valid_with_timestamps/20100201010101_valid_with_timestamps_we_need_reminders.rb +12 -12
  421. data/test/migrations/valid_with_timestamps/20100301010101_valid_with_timestamps_innocent_jointable.rb +12 -12
  422. data/test/migrations/version_check/20131219224947_migration_version_check.rb +8 -8
  423. data/test/models/admin.rb +4 -4
  424. data/test/models/admin/account.rb +2 -2
  425. data/test/models/admin/randomly_named_c1.rb +7 -3
  426. data/test/models/admin/user.rb +40 -40
  427. data/test/models/aircraft.rb +5 -4
  428. data/test/models/arunit2_model.rb +3 -3
  429. data/test/models/author.rb +209 -212
  430. data/test/models/auto_id.rb +4 -4
  431. data/test/models/autoloadable/extra_firm.rb +2 -2
  432. data/test/models/binary.rb +1 -1
  433. data/test/models/bird.rb +12 -12
  434. data/test/models/book.rb +23 -18
  435. data/test/models/boolean.rb +2 -2
  436. data/test/models/bulb.rb +52 -51
  437. data/test/models/cake_designer.rb +3 -3
  438. data/test/models/car.rb +29 -26
  439. data/test/models/carrier.rb +2 -2
  440. data/test/models/cat.rb +10 -0
  441. data/test/models/categorization.rb +19 -19
  442. data/test/models/category.rb +35 -35
  443. data/test/models/chef.rb +8 -3
  444. data/test/models/citation.rb +3 -3
  445. data/test/models/club.rb +25 -23
  446. data/test/models/college.rb +10 -10
  447. data/test/models/column.rb +3 -3
  448. data/test/models/column_name.rb +3 -3
  449. data/test/models/comment.rb +76 -64
  450. data/test/models/company.rb +230 -225
  451. data/test/models/company_in_module.rb +98 -98
  452. data/test/models/computer.rb +3 -3
  453. data/test/models/contact.rb +41 -41
  454. data/test/models/content.rb +40 -0
  455. data/test/models/contract.rb +20 -20
  456. data/test/models/country.rb +7 -7
  457. data/test/models/course.rb +6 -6
  458. data/test/models/customer.rb +83 -77
  459. data/test/models/customer_carrier.rb +14 -14
  460. data/test/models/dashboard.rb +3 -3
  461. data/test/models/default.rb +2 -2
  462. data/test/models/department.rb +4 -4
  463. data/test/models/developer.rb +274 -252
  464. data/test/models/dog.rb +5 -5
  465. data/test/models/dog_lover.rb +5 -5
  466. data/test/models/doubloon.rb +12 -12
  467. data/test/models/drink_designer.rb +3 -3
  468. data/test/models/edge.rb +5 -5
  469. data/test/models/electron.rb +5 -5
  470. data/test/models/engine.rb +4 -4
  471. data/test/models/entrant.rb +3 -3
  472. data/test/models/essay.rb +5 -5
  473. data/test/models/event.rb +2 -2
  474. data/test/models/eye.rb +37 -37
  475. data/test/models/face.rb +9 -9
  476. data/test/models/friendship.rb +6 -6
  477. data/test/models/guid.rb +1 -1
  478. data/test/models/guitar.rb +4 -0
  479. data/test/models/hotel.rb +11 -6
  480. data/test/models/image.rb +3 -3
  481. data/test/models/interest.rb +5 -5
  482. data/test/models/invoice.rb +4 -4
  483. data/test/models/item.rb +7 -7
  484. data/test/models/job.rb +7 -7
  485. data/test/models/joke.rb +7 -7
  486. data/test/models/keyboard.rb +3 -3
  487. data/test/models/legacy_thing.rb +3 -3
  488. data/test/models/lesson.rb +11 -11
  489. data/test/models/line_item.rb +3 -3
  490. data/test/models/liquid.rb +4 -4
  491. data/test/models/man.rb +11 -11
  492. data/test/models/matey.rb +4 -4
  493. data/test/models/member.rb +42 -41
  494. data/test/models/member_detail.rb +8 -7
  495. data/test/models/member_type.rb +3 -3
  496. data/test/models/membership.rb +35 -35
  497. data/test/models/mentor.rb +3 -0
  498. data/test/models/minimalistic.rb +2 -2
  499. data/test/models/minivan.rb +9 -9
  500. data/test/models/mixed_case_monkey.rb +3 -3
  501. data/test/models/mocktail_designer.rb +2 -0
  502. data/test/models/molecule.rb +6 -6
  503. data/test/models/movie.rb +5 -5
  504. data/test/models/node.rb +5 -0
  505. data/test/models/non_primary_key.rb +2 -0
  506. data/test/models/notification.rb +3 -0
  507. data/test/models/order.rb +4 -4
  508. data/test/models/organization.rb +14 -14
  509. data/test/models/other_dog.rb +5 -0
  510. data/test/models/owner.rb +37 -34
  511. data/test/models/parrot.rb +28 -29
  512. data/test/models/person.rb +142 -143
  513. data/test/models/personal_legacy_thing.rb +4 -4
  514. data/test/models/pet.rb +18 -15
  515. data/test/models/pet_treasure.rb +6 -0
  516. data/test/models/pirate.rb +92 -92
  517. data/test/models/possession.rb +3 -3
  518. data/test/models/post.rb +273 -264
  519. data/test/models/price_estimate.rb +4 -4
  520. data/test/models/professor.rb +5 -5
  521. data/test/models/project.rb +40 -29
  522. data/test/models/publisher.rb +2 -2
  523. data/test/models/publisher/article.rb +4 -4
  524. data/test/models/publisher/magazine.rb +3 -3
  525. data/test/models/randomly_named_c1.rb +3 -3
  526. data/test/models/rating.rb +4 -4
  527. data/test/models/reader.rb +23 -23
  528. data/test/models/recipe.rb +3 -0
  529. data/test/models/record.rb +2 -2
  530. data/test/models/reference.rb +22 -22
  531. data/test/models/reply.rb +61 -61
  532. data/test/models/ship.rb +39 -33
  533. data/test/models/ship_part.rb +7 -7
  534. data/test/models/shop.rb +17 -17
  535. data/test/models/shop_account.rb +6 -6
  536. data/test/models/speedometer.rb +6 -6
  537. data/test/models/sponsor.rb +7 -7
  538. data/test/models/string_key_object.rb +3 -3
  539. data/test/models/student.rb +4 -4
  540. data/test/models/subject.rb +16 -16
  541. data/test/models/subscriber.rb +8 -8
  542. data/test/models/subscription.rb +4 -4
  543. data/test/models/tag.rb +13 -7
  544. data/test/models/tagging.rb +13 -13
  545. data/test/models/task.rb +5 -5
  546. data/test/models/topic.rb +118 -124
  547. data/test/models/toy.rb +6 -6
  548. data/test/models/traffic_light.rb +4 -4
  549. data/test/models/treasure.rb +14 -14
  550. data/test/models/treaty.rb +7 -7
  551. data/test/models/tree.rb +3 -0
  552. data/test/models/tuning_peg.rb +4 -0
  553. data/test/models/tyre.rb +11 -11
  554. data/test/models/user.rb +14 -0
  555. data/test/models/uuid_child.rb +3 -3
  556. data/test/models/uuid_item.rb +6 -0
  557. data/test/models/uuid_parent.rb +3 -3
  558. data/test/models/vegetables.rb +24 -24
  559. data/test/models/vehicle.rb +6 -6
  560. data/test/models/vertex.rb +9 -9
  561. data/test/models/warehouse_thing.rb +5 -5
  562. data/test/models/wheel.rb +3 -3
  563. data/test/models/without_table.rb +3 -3
  564. data/test/models/zine.rb +3 -3
  565. data/test/schema/mysql2_specific_schema.rb +68 -58
  566. data/test/schema/oracle_specific_schema.rb +40 -43
  567. data/test/schema/postgresql_specific_schema.rb +114 -202
  568. data/test/schema/schema.rb +1057 -938
  569. data/test/schema/schema.rb.original +1057 -0
  570. data/test/schema/sqlite_specific_schema.rb +18 -22
  571. data/test/support/config.rb +43 -43
  572. data/test/support/connection.rb +23 -22
  573. data/test/support/connection_helper.rb +14 -14
  574. data/test/support/ddl_helper.rb +8 -8
  575. data/test/support/schema_dumping_helper.rb +20 -20
  576. data/test/support/yaml_compatibility_fixtures/rails_4_1.yml +22 -0
  577. data/test/support/yaml_compatibility_fixtures/rails_4_2_0.yml +182 -0
  578. metadata +146 -30
  579. data/lib/mswin32/rb19x/ibm_db.so +0 -0
  580. data/lib/mswin32/rb21x/i386/ibm_db.so +0 -0
  581. data/lib/mswin32/rb22x/i386/ibm_db.so +0 -0
  582. data/lib/mswin32/rb23x/i386/ibm_db.so +0 -0
  583. data/test/cases/associations/deprecated_counter_cache_on_has_many_through_test.rb +0 -26
  584. data/test/cases/attribute_methods/serialization_test.rb +0 -29
  585. data/test/cases/migration/change_schema_test - Copy.rb +0 -448
  586. data/test/cases/migration/foreign_key_test - Changed.rb +0 -325
  587. data/test/cases/migration/table_and_index_test.rb +0 -24
  588. data/test/cases/relation/where_test2.rb +0 -36
  589. data/test/cases/type/decimal_test.rb +0 -51
  590. data/test/cases/type/unsigned_integer_test.rb +0 -18
  591. data/test/cases/xml_serialization_test.rb +0 -457
  592. data/test/fixtures/naked/csv/accounts.csv +0 -1
  593. data/test/schema/mysql_specific_schema.rb +0 -70
@@ -1,68 +1,103 @@
1
- # encoding: utf-8
2
- require "cases/helper"
3
- require 'models/man'
4
- require 'models/face'
5
- require 'models/interest'
6
- require 'models/speedometer'
7
- require 'models/dashboard'
8
-
9
- class PresenceValidationTest < ActiveRecord::TestCase
10
- class Boy < Man; end
11
-
12
- repair_validations(Boy)
13
-
14
- def test_validates_presence_of_non_association
15
- Boy.validates_presence_of(:name)
16
- b = Boy.new
17
- assert b.invalid?
18
-
19
- b.name = "Alex"
20
- assert b.valid?
21
- end
22
-
23
- def test_validates_presence_of_has_one
24
- Boy.validates_presence_of(:face)
25
- b = Boy.new
26
- assert b.invalid?, "should not be valid if has_one association missing"
27
- assert_equal 1, b.errors[:face].size, "validates_presence_of should only add one error"
28
- end
29
-
30
- def test_validates_presence_of_has_one_marked_for_destruction
31
- Boy.validates_presence_of(:face)
32
- b = Boy.new
33
- f = Face.new
34
- b.face = f
35
- assert b.valid?
36
-
37
- f.mark_for_destruction
38
- assert b.invalid?
39
- end
40
-
41
- def test_validates_presence_of_has_many_marked_for_destruction
42
- Boy.validates_presence_of(:interests)
43
- b = Boy.new
44
- b.interests << [i1 = Interest.new, i2 = Interest.new]
45
- assert b.valid?
46
-
47
- i1.mark_for_destruction
48
- assert b.valid?
49
-
50
- i2.mark_for_destruction
51
- assert b.invalid?
52
- end
53
-
54
- def test_validates_presence_doesnt_convert_to_array
55
- speedometer = Class.new(Speedometer)
56
- speedometer.validates_presence_of :dashboard
57
-
58
- dash = Dashboard.new
59
-
60
- # dashboard has to_a method
61
- def dash.to_a; ['(/)', '(\)']; end
62
-
63
- s = speedometer.new
64
- s.dashboard = dash
65
-
66
- assert_nothing_raised { s.valid? }
67
- end
68
- end
1
+ require "cases/helper"
2
+ require 'models/man'
3
+ require 'models/face'
4
+ require 'models/interest'
5
+ require 'models/speedometer'
6
+ require 'models/dashboard'
7
+
8
+ class PresenceValidationTest < ActiveRecord::TestCase
9
+ class Boy < Man; end
10
+
11
+ repair_validations(Boy)
12
+
13
+ def test_validates_presence_of_non_association
14
+ Boy.validates_presence_of(:name)
15
+ b = Boy.new
16
+ assert b.invalid?
17
+
18
+ b.name = "Alex"
19
+ assert b.valid?
20
+ end
21
+
22
+ def test_validates_presence_of_has_one
23
+ Boy.validates_presence_of(:face)
24
+ b = Boy.new
25
+ assert b.invalid?, "should not be valid if has_one association missing"
26
+ assert_equal 1, b.errors[:face].size, "validates_presence_of should only add one error"
27
+ end
28
+
29
+ def test_validates_presence_of_has_one_marked_for_destruction
30
+ Boy.validates_presence_of(:face)
31
+ b = Boy.new
32
+ f = Face.new
33
+ b.face = f
34
+ assert b.valid?
35
+
36
+ f.mark_for_destruction
37
+ assert b.invalid?
38
+ end
39
+
40
+ def test_validates_presence_of_has_many_marked_for_destruction
41
+ Boy.validates_presence_of(:interests)
42
+ b = Boy.new
43
+ b.interests << [i1 = Interest.new, i2 = Interest.new]
44
+ assert b.valid?
45
+
46
+ i1.mark_for_destruction
47
+ assert b.valid?
48
+
49
+ i2.mark_for_destruction
50
+ assert b.invalid?
51
+ end
52
+
53
+ def test_validates_presence_doesnt_convert_to_array
54
+ speedometer = Class.new(Speedometer)
55
+ speedometer.validates_presence_of :dashboard
56
+
57
+ dash = Dashboard.new
58
+
59
+ # dashboard has to_a method
60
+ def dash.to_a; ['(/)', '(\)']; end
61
+
62
+ s = speedometer.new
63
+ s.dashboard = dash
64
+
65
+ assert_nothing_raised { s.valid? }
66
+ end
67
+
68
+ def test_validates_presence_of_virtual_attribute_on_model
69
+ repair_validations(Interest) do
70
+ Interest.send(:attr_accessor, :abbreviation)
71
+ Interest.validates_presence_of(:topic)
72
+ Interest.validates_presence_of(:abbreviation)
73
+
74
+ interest = Interest.create!(topic: 'Thought Leadering', abbreviation: 'tl')
75
+ assert interest.valid?
76
+
77
+ interest.abbreviation = ''
78
+
79
+ assert interest.invalid?
80
+ end
81
+ end
82
+
83
+ def test_validations_run_on_persisted_record
84
+ repair_validations(Interest) do
85
+ interest = Interest.new
86
+ interest.save!
87
+ assert_predicate interest, :valid?
88
+
89
+ Interest.validates_presence_of(:topic)
90
+
91
+ assert_not_predicate interest, :valid?
92
+ end
93
+ end
94
+
95
+ def test_validates_presence_with_on_context
96
+ repair_validations(Interest) do
97
+ Interest.validates_presence_of(:topic, on: :required_name)
98
+ interest = Interest.new
99
+ interest.save!
100
+ assert_not interest.valid?(:required_name)
101
+ end
102
+ end
103
+ end
@@ -1,434 +1,548 @@
1
- # encoding: utf-8
2
- require "cases/helper"
3
- require 'models/topic'
4
- require 'models/reply'
5
- require 'models/warehouse_thing'
6
- require 'models/guid'
7
- require 'models/event'
8
-
9
- class Wizard < ActiveRecord::Base
10
- self.abstract_class = true
11
-
12
- validates_uniqueness_of :name
13
- end
14
-
15
- class IneptWizard < Wizard
16
- validates_uniqueness_of :city
17
- end
18
-
19
- class Conjurer < IneptWizard
20
- end
21
-
22
- class Thaumaturgist < IneptWizard
23
- end
24
-
25
- class ReplyTitle; end
26
-
27
- class ReplyWithTitleObject < Reply
28
- validates_uniqueness_of :content, :scope => :title
29
-
30
- def title; ReplyTitle.new; end
31
- end
32
-
33
- class Employee < ActiveRecord::Base
34
- self.table_name = 'postgresql_arrays'
35
- validates_uniqueness_of :nicknames
36
- end
37
-
38
- class TopicWithUniqEvent < Topic
39
- belongs_to :event, foreign_key: :parent_id
40
- validates :event, uniqueness: true
41
- end
42
-
43
- class BigIntTest < ActiveRecord::Base
44
- INT_MAX_VALUE = 2147483647
45
- self.table_name = 'cars'
46
- validates :engines_count, uniqueness: true, inclusion: { in: 0..INT_MAX_VALUE }
47
- end
48
-
49
- class BigIntReverseTest < ActiveRecord::Base
50
- INT_MAX_VALUE = 2147483647
51
- self.table_name = 'cars'
52
- validates :engines_count, inclusion: { in: 0..INT_MAX_VALUE }
53
- validates :engines_count, uniqueness: true
54
- end
55
-
56
- class UniquenessValidationTest < ActiveRecord::TestCase
57
- INT_MAX_VALUE = 2147483647
58
-
59
- fixtures :topics, 'warehouse_things'
60
-
61
- repair_validations(Topic, Reply)
62
-
63
- def test_validate_uniqueness
64
- Topic.validates_uniqueness_of(:title)
65
-
66
- t = Topic.new("title" => "I'm uniqué!")
67
- assert t.save, "Should save t as unique"
68
-
69
- t.content = "Remaining unique"
70
- assert t.save, "Should still save t as unique"
71
-
72
- t2 = Topic.new("title" => "I'm uniqué!")
73
- assert !t2.valid?, "Shouldn't be valid"
74
- assert !t2.save, "Shouldn't save t2 as unique"
75
- assert_equal ["has already been taken"], t2.errors[:title]
76
-
77
- t2.title = "Now I am really also unique"
78
- assert t2.save, "Should now save t2 as unique"
79
- end
80
-
81
- def test_validate_uniqueness_with_alias_attribute
82
- Topic.alias_attribute :new_title, :title
83
- Topic.validates_uniqueness_of(:new_title)
84
-
85
- topic = Topic.new(new_title: 'abc')
86
- assert topic.valid?
87
- end
88
-
89
- def test_validates_uniqueness_with_nil_value
90
- Topic.validates_uniqueness_of(:title)
91
-
92
- t = Topic.new("title" => nil)
93
- assert t.save, "Should save t as unique"
94
-
95
- t2 = Topic.new("title" => nil)
96
- assert !t2.valid?, "Shouldn't be valid"
97
- assert !t2.save, "Shouldn't save t2 as unique"
98
- assert_equal ["has already been taken"], t2.errors[:title]
99
- end
100
-
101
- def test_validates_uniqueness_with_validates
102
- Topic.validates :title, :uniqueness => true
103
- Topic.create!('title' => 'abc')
104
-
105
- t2 = Topic.new('title' => 'abc')
106
- assert !t2.valid?
107
- assert t2.errors[:title]
108
- end
109
-
110
- def test_validate_uniqueness_when_integer_out_of_range
111
- entry = BigIntTest.create(engines_count: INT_MAX_VALUE + 1)
112
- assert_equal entry.errors[:engines_count], ['is not included in the list']
113
- end
114
-
115
- def test_validate_uniqueness_when_integer_out_of_range_show_order_does_not_matter
116
- entry = BigIntReverseTest.create(engines_count: INT_MAX_VALUE + 1)
117
- assert_equal entry.errors[:engines_count], ['is not included in the list']
118
- end
119
-
120
- def test_validates_uniqueness_with_newline_chars
121
- Topic.validates_uniqueness_of(:title, :case_sensitive => false)
122
-
123
- t = Topic.new("title" => "new\nline")
124
- assert t.save, "Should save t as unique"
125
- end
126
-
127
- def test_validate_uniqueness_with_scope
128
- Reply.validates_uniqueness_of(:content, :scope => "parent_id")
129
-
130
- t = Topic.create("title" => "I'm unique!")
131
-
132
- r1 = t.replies.create "title" => "r1", "content" => "hello world"
133
- assert r1.valid?, "Saving r1"
134
-
135
- r2 = t.replies.create "title" => "r2", "content" => "hello world"
136
- assert !r2.valid?, "Saving r2 first time"
137
-
138
- r2.content = "something else"
139
- assert r2.save, "Saving r2 second time"
140
-
141
- t2 = Topic.create("title" => "I'm unique too!")
142
- r3 = t2.replies.create "title" => "r3", "content" => "hello world"
143
- assert r3.valid?, "Saving r3"
144
- end
145
-
146
- def test_validate_uniqueness_with_object_scope
147
- Reply.validates_uniqueness_of(:content, :scope => :topic)
148
-
149
- t = Topic.create("title" => "I'm unique!")
150
-
151
- r1 = t.replies.create "title" => "r1", "content" => "hello world"
152
- assert r1.valid?, "Saving r1"
153
-
154
- r2 = t.replies.create "title" => "r2", "content" => "hello world"
155
- assert !r2.valid?, "Saving r2 first time"
156
- end
157
-
158
- def test_validate_uniqueness_with_composed_attribute_scope
159
- r1 = ReplyWithTitleObject.create "title" => "r1", "content" => "hello world"
160
- assert r1.valid?, "Saving r1"
161
-
162
- r2 = ReplyWithTitleObject.create "title" => "r1", "content" => "hello world"
163
- assert !r2.valid?, "Saving r2 first time"
164
- end
165
-
166
- def test_validate_uniqueness_with_object_arg
167
- Reply.validates_uniqueness_of(:topic)
168
-
169
- t = Topic.create("title" => "I'm unique!")
170
-
171
- r1 = t.replies.create "title" => "r1", "content" => "hello world"
172
- assert r1.valid?, "Saving r1"
173
-
174
- r2 = t.replies.create "title" => "r2", "content" => "hello world"
175
- assert !r2.valid?, "Saving r2 first time"
176
- end
177
-
178
- def test_validate_uniqueness_scoped_to_defining_class
179
- t = Topic.create("title" => "What, me worry?")
180
-
181
- r1 = t.unique_replies.create "title" => "r1", "content" => "a barrel of fun"
182
- assert r1.valid?, "Saving r1"
183
-
184
- r2 = t.silly_unique_replies.create "title" => "r2", "content" => "a barrel of fun"
185
- assert !r2.valid?, "Saving r2"
186
-
187
- # Should succeed as validates_uniqueness_of only applies to
188
- # UniqueReply and its subclasses
189
- r3 = t.replies.create "title" => "r2", "content" => "a barrel of fun"
190
- assert r3.valid?, "Saving r3"
191
- end
192
-
193
- def test_validate_uniqueness_with_scope_array
194
- Reply.validates_uniqueness_of(:author_name, :scope => [:author_email_address, :parent_id])
195
-
196
- t = Topic.create("title" => "The earth is actually flat!")
197
-
198
- r1 = t.replies.create "author_name" => "jeremy", "author_email_address" => "jeremy@rubyonrails.com", "title" => "You're crazy!", "content" => "Crazy reply"
199
- assert r1.valid?, "Saving r1"
200
-
201
- r2 = t.replies.create "author_name" => "jeremy", "author_email_address" => "jeremy@rubyonrails.com", "title" => "You're crazy!", "content" => "Crazy reply again..."
202
- assert !r2.valid?, "Saving r2. Double reply by same author."
203
-
204
- r2.author_email_address = "jeremy_alt_email@rubyonrails.com"
205
- assert r2.save, "Saving r2 the second time."
206
-
207
- r3 = t.replies.create "author_name" => "jeremy", "author_email_address" => "jeremy_alt_email@rubyonrails.com", "title" => "You're wrong", "content" => "It's cubic"
208
- assert !r3.valid?, "Saving r3"
209
-
210
- r3.author_name = "jj"
211
- assert r3.save, "Saving r3 the second time."
212
-
213
- r3.author_name = "jeremy"
214
- assert !r3.save, "Saving r3 the third time."
215
- end
216
-
217
- def test_validate_case_insensitive_uniqueness
218
- Topic.validates_uniqueness_of(:title, :parent_id, :case_sensitive => false, :allow_nil => true)
219
-
220
- t = Topic.new("title" => "I'm unique!", :parent_id => 2)
221
- assert t.save, "Should save t as unique"
222
-
223
- t.content = "Remaining unique"
224
- assert t.save, "Should still save t as unique"
225
-
226
- t2 = Topic.new("title" => "I'm UNIQUE!", :parent_id => 1)
227
- assert !t2.valid?, "Shouldn't be valid"
228
- assert !t2.save, "Shouldn't save t2 as unique"
229
- assert t2.errors[:title].any?
230
- assert t2.errors[:parent_id].any?
231
- assert_equal ["has already been taken"], t2.errors[:title]
232
-
233
- t2.title = "I'm truly UNIQUE!"
234
- assert !t2.valid?, "Shouldn't be valid"
235
- assert !t2.save, "Shouldn't save t2 as unique"
236
- assert t2.errors[:title].empty?
237
- assert t2.errors[:parent_id].any?
238
-
239
- t2.parent_id = 4
240
- assert t2.save, "Should now save t2 as unique"
241
-
242
- t2.parent_id = nil
243
- t2.title = nil
244
- assert t2.valid?, "should validate with nil"
245
- assert t2.save, "should save with nil"
246
-
247
- t_utf8 = Topic.new("title" => "Я тоже уникальный!")
248
- assert t_utf8.save, "Should save t_utf8 as unique"
249
-
250
- # If database hasn't UTF-8 character set, this test fails
251
- if Topic.all.merge!(:select => 'LOWER(title) AS title').find(t_utf8.id).title == "я тоже уникальный!"
252
- t2_utf8 = Topic.new("title" => тоже УНИКАЛЬНЫЙ!")
253
- assert !t2_utf8.valid?, "Shouldn't be valid"
254
- assert !t2_utf8.save, "Shouldn't save t2_utf8 as unique"
255
- end
256
- end
257
-
258
- def test_validate_case_sensitive_uniqueness_with_special_sql_like_chars
259
- Topic.validates_uniqueness_of(:title, :case_sensitive => true)
260
-
261
- t = Topic.new("title" => "I'm unique!")
262
- assert t.save, "Should save t as unique"
263
-
264
- t2 = Topic.new("title" => "I'm %")
265
- assert t2.save, "Should save t2 as unique"
266
-
267
- t3 = Topic.new("title" => "I'm uniqu_!")
268
- assert t3.save, "Should save t3 as unique"
269
- end
270
-
271
- def test_validate_case_insensitive_uniqueness_with_special_sql_like_chars
272
- Topic.validates_uniqueness_of(:title, :case_sensitive => false)
273
-
274
- t = Topic.new("title" => "I'm unique!")
275
- assert t.save, "Should save t as unique"
276
-
277
- t2 = Topic.new("title" => "I'm %")
278
- assert t2.save, "Should save t2 as unique"
279
-
280
- t3 = Topic.new("title" => "I'm uniqu_!")
281
- assert t3.save, "Should save t3 as unique"
282
- end
283
-
284
- def test_validate_case_sensitive_uniqueness
285
- Topic.validates_uniqueness_of(:title, :case_sensitive => true, :allow_nil => true)
286
-
287
- t = Topic.new("title" => "I'm unique!")
288
- assert t.save, "Should save t as unique"
289
-
290
- t.content = "Remaining unique"
291
- assert t.save, "Should still save t as unique"
292
-
293
- t2 = Topic.new("title" => "I'M UNIQUE!")
294
- assert t2.valid?, "Should be valid"
295
- assert t2.save, "Should save t2 as unique"
296
- assert t2.errors[:title].empty?
297
- assert t2.errors[:parent_id].empty?
298
- assert_not_equal ["has already been taken"], t2.errors[:title]
299
-
300
- t3 = Topic.new("title" => "I'M uNiQUe!")
301
- assert t3.valid?, "Should be valid"
302
- assert t3.save, "Should save t2 as unique"
303
- assert t3.errors[:title].empty?
304
- assert t3.errors[:parent_id].empty?
305
- assert_not_equal ["has already been taken"], t3.errors[:title]
306
- end
307
-
308
- def test_validate_case_sensitive_uniqueness_with_attribute_passed_as_integer
309
- Topic.validates_uniqueness_of(:title, :case_sensitive => true)
310
- Topic.create!('title' => 101)
311
-
312
- t2 = Topic.new('title' => 101)
313
- assert !t2.valid?
314
- assert t2.errors[:title]
315
- end
316
-
317
- def test_validate_uniqueness_with_non_standard_table_names
318
- i1 = WarehouseThing.create(:value => 1000)
319
- assert !i1.valid?, "i1 should not be valid"
320
- assert i1.errors[:value].any?, "Should not be empty"
321
- end
322
-
323
- def test_validates_uniqueness_inside_scoping
324
- Topic.validates_uniqueness_of(:title)
325
-
326
- Topic.where(:author_name => "David").scoping do
327
- t1 = Topic.new("title" => "I'm unique!", "author_name" => "Mary")
328
- assert t1.save
329
- t2 = Topic.new("title" => "I'm unique!", "author_name" => "David")
330
- assert !t2.valid?
331
- end
332
- end
333
-
334
- def test_validate_uniqueness_with_columns_which_are_sql_keywords
335
- repair_validations(Guid) do
336
- Guid.validates_uniqueness_of :key
337
- g = Guid.new
338
- g.key = "foo"
339
- assert_nothing_raised { !g.valid? }
340
- end
341
- end
342
-
343
- def test_validate_uniqueness_with_limit
344
- # Event.title is limited to 5 characters
345
- e1 = Event.create(:title => "abcde")
346
- assert e1.valid?, "Could not create an event with a unique, 5 character title"
347
- e2 = Event.create(:title => "abcdefgh")
348
- assert !e2.valid?, "Created an event whose title, with limit taken into account, is not unique"
349
- end
350
-
351
- def test_validate_uniqueness_with_limit_and_utf8
352
- unless current_adapter?(:IBM_DBAdapter)
353
- # Limit for the varchar field is number of bytes and not characters for DB2. Hence the below test cases is expected to fail.
354
- # Event.title is limited to 5 characters
355
- e1 = Event.create(:title => "一二三四五")
356
- assert e1.valid?, "Could not create an event with a unique, 5 character title"
357
- e2 = Event.create(:title => "一二三四五六七八")
358
- assert !e2.valid?, "Created an event whose title, with limit taken into account, is not unique"
359
- end
360
- end
361
-
362
- def test_validate_straight_inheritance_uniqueness
363
- w1 = IneptWizard.create(:name => "Rincewind", :city => "Ankh-Morpork")
364
- assert w1.valid?, "Saving w1"
365
-
366
- # Should use validation from base class (which is abstract)
367
- w2 = IneptWizard.new(:name => "Rincewind", :city => "Quirm")
368
- assert !w2.valid?, "w2 shouldn't be valid"
369
- assert w2.errors[:name].any?, "Should have errors for name"
370
- assert_equal ["has already been taken"], w2.errors[:name], "Should have uniqueness message for name"
371
-
372
- w3 = Conjurer.new(:name => "Rincewind", :city => "Quirm")
373
- assert !w3.valid?, "w3 shouldn't be valid"
374
- assert w3.errors[:name].any?, "Should have errors for name"
375
- assert_equal ["has already been taken"], w3.errors[:name], "Should have uniqueness message for name"
376
-
377
- w4 = Conjurer.create(:name => "The Amazing Bonko", :city => "Quirm")
378
- assert w4.valid?, "Saving w4"
379
-
380
- w5 = Thaumaturgist.new(:name => "The Amazing Bonko", :city => "Lancre")
381
- assert !w5.valid?, "w5 shouldn't be valid"
382
- assert w5.errors[:name].any?, "Should have errors for name"
383
- assert_equal ["has already been taken"], w5.errors[:name], "Should have uniqueness message for name"
384
-
385
- w6 = Thaumaturgist.new(:name => "Mustrum Ridcully", :city => "Quirm")
386
- assert !w6.valid?, "w6 shouldn't be valid"
387
- assert w6.errors[:city].any?, "Should have errors for city"
388
- assert_equal ["has already been taken"], w6.errors[:city], "Should have uniqueness message for city"
389
- end
390
-
391
- def test_validate_uniqueness_with_conditions
392
- Topic.validates_uniqueness_of :title, conditions: -> { where(approved: true) }
393
- Topic.create("title" => "I'm a topic", "approved" => true)
394
- Topic.create("title" => "I'm an unapproved topic", "approved" => false)
395
-
396
- t3 = Topic.new("title" => "I'm a topic", "approved" => true)
397
- assert !t3.valid?, "t3 shouldn't be valid"
398
-
399
- t4 = Topic.new("title" => "I'm an unapproved topic", "approved" => false)
400
- assert t4.valid?, "t4 should be valid"
401
- end
402
-
403
- def test_validate_uniqueness_with_non_callable_conditions_is_not_supported
404
- assert_raises(ArgumentError) {
405
- Topic.validates_uniqueness_of :title, conditions: Topic.where(approved: true)
406
- }
407
- end
408
-
409
- if current_adapter? :PostgreSQLAdapter
410
- def test_validate_uniqueness_with_array_column
411
- e1 = Employee.create("nicknames" => ["john", "johnny"], "commission_by_quarter" => [1000, 1200])
412
- assert e1.persisted?, "Saving e1"
413
-
414
- e2 = Employee.create("nicknames" => ["john", "johnny"], "commission_by_quarter" => [2200])
415
- assert !e2.persisted?, "e2 shouldn't be valid"
416
- assert e2.errors[:nicknames].any?, "Should have errors for nicknames"
417
- assert_equal ["has already been taken"], e2.errors[:nicknames], "Should have uniqueness message for nicknames"
418
- end
419
- end
420
-
421
- def test_validate_uniqueness_on_existing_relation
422
- event = Event.create
423
- assert TopicWithUniqEvent.create(event: event).valid?
424
-
425
- topic = TopicWithUniqEvent.new(event: event)
426
- assert_not topic.valid?
427
- assert_equal ['has already been taken'], topic.errors[:event]
428
- end
429
-
430
- def test_validate_uniqueness_on_empty_relation
431
- topic = TopicWithUniqEvent.new
432
- assert topic.valid?
433
- end
434
- end
1
+ require "cases/helper"
2
+ require "models/topic"
3
+ require "models/reply"
4
+ require "models/warehouse_thing"
5
+ require "models/guid"
6
+ require "models/event"
7
+ require "models/dashboard"
8
+ require "models/uuid_item"
9
+ require "models/author"
10
+ require "models/person"
11
+ require "models/essay"
12
+
13
+ class Wizard < ActiveRecord::Base
14
+ self.abstract_class = true
15
+
16
+ validates_uniqueness_of :name
17
+ end
18
+
19
+ class IneptWizard < Wizard
20
+ validates_uniqueness_of :city
21
+ end
22
+
23
+ class Conjurer < IneptWizard
24
+ end
25
+
26
+ class Thaumaturgist < IneptWizard
27
+ end
28
+
29
+ class ReplyTitle; end
30
+
31
+ class ReplyWithTitleObject < Reply
32
+ validates_uniqueness_of :content, :scope => :title
33
+
34
+ def title; ReplyTitle.new; end
35
+ end
36
+
37
+ class TopicWithUniqEvent < Topic
38
+ belongs_to :event, foreign_key: :parent_id
39
+ validates :event, uniqueness: true
40
+ end
41
+
42
+ class BigIntTest < ActiveRecord::Base
43
+ INT_MAX_VALUE = 2147483647
44
+ self.table_name = 'cars'
45
+ validates :engines_count, uniqueness: true, inclusion: { in: 0..INT_MAX_VALUE }
46
+ end
47
+
48
+ class BigIntReverseTest < ActiveRecord::Base
49
+ INT_MAX_VALUE = 2147483647
50
+ self.table_name = 'cars'
51
+ validates :engines_count, inclusion: { in: 0..INT_MAX_VALUE }
52
+ validates :engines_count, uniqueness: true
53
+ end
54
+
55
+ class CoolTopic < Topic
56
+ validates_uniqueness_of :id
57
+ end
58
+
59
+ class TopicWithAfterCreate < Topic
60
+ after_create :set_author
61
+
62
+ def set_author
63
+ update_attributes!(:author_name => "#{title} #{id}")
64
+ end
65
+ end
66
+
67
+ class UniquenessValidationTest < ActiveRecord::TestCase
68
+ INT_MAX_VALUE = 2147483647
69
+
70
+ fixtures :topics, 'warehouse_things'
71
+
72
+ repair_validations(Topic, Reply)
73
+
74
+ def test_validate_uniqueness
75
+ Topic.validates_uniqueness_of(:title)
76
+
77
+ t = Topic.new("title" => "I'm uniqué!")
78
+ assert t.save, "Should save t as unique"
79
+
80
+ t.content = "Remaining unique"
81
+ assert t.save, "Should still save t as unique"
82
+
83
+ t2 = Topic.new("title" => "I'm uniqué!")
84
+ assert !t2.valid?, "Shouldn't be valid"
85
+ assert !t2.save, "Shouldn't save t2 as unique"
86
+ assert_equal ["has already been taken"], t2.errors[:title]
87
+
88
+ t2.title = "Now I am really also unique"
89
+ assert t2.save, "Should now save t2 as unique"
90
+ end
91
+
92
+ def test_validate_uniqueness_with_alias_attribute
93
+ Topic.alias_attribute :new_title, :title
94
+ Topic.validates_uniqueness_of(:new_title)
95
+
96
+ topic = Topic.new(new_title: 'abc')
97
+ assert topic.valid?
98
+ end
99
+
100
+ def test_validates_uniqueness_with_nil_value
101
+ Topic.validates_uniqueness_of(:title)
102
+
103
+ t = Topic.new("title" => nil)
104
+ assert t.save, "Should save t as unique"
105
+
106
+ t2 = Topic.new("title" => nil)
107
+ assert !t2.valid?, "Shouldn't be valid"
108
+ assert !t2.save, "Shouldn't save t2 as unique"
109
+ assert_equal ["has already been taken"], t2.errors[:title]
110
+ end
111
+
112
+ def test_validates_uniqueness_with_validates
113
+ Topic.validates :title, :uniqueness => true
114
+ Topic.create!('title' => 'abc')
115
+
116
+ t2 = Topic.new('title' => 'abc')
117
+ assert !t2.valid?
118
+ assert t2.errors[:title]
119
+ end
120
+
121
+ def test_validate_uniqueness_when_integer_out_of_range
122
+ entry = BigIntTest.create(engines_count: INT_MAX_VALUE + 1)
123
+ assert_equal entry.errors[:engines_count], ['is not included in the list']
124
+ end
125
+
126
+ def test_validate_uniqueness_when_integer_out_of_range_show_order_does_not_matter
127
+ entry = BigIntReverseTest.create(engines_count: INT_MAX_VALUE + 1)
128
+ assert_equal entry.errors[:engines_count], ['is not included in the list']
129
+ end
130
+
131
+ def test_validates_uniqueness_with_newline_chars
132
+ Topic.validates_uniqueness_of(:title, :case_sensitive => false)
133
+
134
+ t = Topic.new("title" => "new\nline")
135
+ assert t.save, "Should save t as unique"
136
+ end
137
+
138
+ def test_validate_uniqueness_with_scope
139
+ Reply.validates_uniqueness_of(:content, :scope => "parent_id")
140
+
141
+ t = Topic.create("title" => "I'm unique!")
142
+
143
+ r1 = t.replies.create "title" => "r1", "content" => "hello world"
144
+ assert r1.valid?, "Saving r1"
145
+
146
+ r2 = t.replies.create "title" => "r2", "content" => "hello world"
147
+ assert !r2.valid?, "Saving r2 first time"
148
+
149
+ r2.content = "something else"
150
+ assert r2.save, "Saving r2 second time"
151
+
152
+ t2 = Topic.create("title" => "I'm unique too!")
153
+ r3 = t2.replies.create "title" => "r3", "content" => "hello world"
154
+ assert r3.valid?, "Saving r3"
155
+ end
156
+
157
+ def test_validate_uniqueness_with_object_scope
158
+ Reply.validates_uniqueness_of(:content, :scope => :topic)
159
+
160
+ t = Topic.create("title" => "I'm unique!")
161
+
162
+ r1 = t.replies.create "title" => "r1", "content" => "hello world"
163
+ assert r1.valid?, "Saving r1"
164
+
165
+ r2 = t.replies.create "title" => "r2", "content" => "hello world"
166
+ assert !r2.valid?, "Saving r2 first time"
167
+ end
168
+
169
+ def test_validate_uniqueness_with_polymorphic_object_scope
170
+ Essay.validates_uniqueness_of(:name, scope: :writer)
171
+
172
+ a = Author.create(name: "Sergey")
173
+ p = Person.create(first_name: "Sergey")
174
+
175
+ e1 = a.essays.create(name: "Essay")
176
+ assert e1.valid?, "Saving e1"
177
+
178
+ e2 = p.essays.create(name: "Essay")
179
+ assert e2.valid?, "Saving e2"
180
+ end
181
+
182
+ def test_validate_uniqueness_with_composed_attribute_scope
183
+ r1 = ReplyWithTitleObject.create "title" => "r1", "content" => "hello world"
184
+ assert r1.valid?, "Saving r1"
185
+
186
+ r2 = ReplyWithTitleObject.create "title" => "r1", "content" => "hello world"
187
+ assert !r2.valid?, "Saving r2 first time"
188
+ end
189
+
190
+ def test_validate_uniqueness_with_object_arg
191
+ Reply.validates_uniqueness_of(:topic)
192
+
193
+ t = Topic.create("title" => "I'm unique!")
194
+
195
+ r1 = t.replies.create "title" => "r1", "content" => "hello world"
196
+ assert r1.valid?, "Saving r1"
197
+
198
+ r2 = t.replies.create "title" => "r2", "content" => "hello world"
199
+ assert !r2.valid?, "Saving r2 first time"
200
+ end
201
+
202
+ def test_validate_uniqueness_scoped_to_defining_class
203
+ t = Topic.create("title" => "What, me worry?")
204
+
205
+ r1 = t.unique_replies.create "title" => "r1", "content" => "a barrel of fun"
206
+ assert r1.valid?, "Saving r1"
207
+
208
+ r2 = t.silly_unique_replies.create "title" => "r2", "content" => "a barrel of fun"
209
+ assert !r2.valid?, "Saving r2"
210
+
211
+ # Should succeed as validates_uniqueness_of only applies to
212
+ # UniqueReply and its subclasses
213
+ r3 = t.replies.create "title" => "r2", "content" => "a barrel of fun"
214
+ assert r3.valid?, "Saving r3"
215
+ end
216
+
217
+ def test_validate_uniqueness_with_scope_array
218
+ Reply.validates_uniqueness_of(:author_name, :scope => [:author_email_address, :parent_id])
219
+
220
+ t = Topic.create("title" => "The earth is actually flat!")
221
+
222
+ r1 = t.replies.create "author_name" => "jeremy", "author_email_address" => "jeremy@rubyonrails.com", "title" => "You're crazy!", "content" => "Crazy reply"
223
+ assert r1.valid?, "Saving r1"
224
+
225
+ r2 = t.replies.create "author_name" => "jeremy", "author_email_address" => "jeremy@rubyonrails.com", "title" => "You're crazy!", "content" => "Crazy reply again..."
226
+ assert !r2.valid?, "Saving r2. Double reply by same author."
227
+
228
+ r2.author_email_address = "jeremy_alt_email@rubyonrails.com"
229
+ assert r2.save, "Saving r2 the second time."
230
+
231
+ r3 = t.replies.create "author_name" => "jeremy", "author_email_address" => "jeremy_alt_email@rubyonrails.com", "title" => "You're wrong", "content" => "It's cubic"
232
+ assert !r3.valid?, "Saving r3"
233
+
234
+ r3.author_name = "jj"
235
+ assert r3.save, "Saving r3 the second time."
236
+
237
+ r3.author_name = "jeremy"
238
+ assert !r3.save, "Saving r3 the third time."
239
+ end
240
+
241
+ def test_validate_case_insensitive_uniqueness
242
+ Topic.validates_uniqueness_of(:title, :parent_id, :case_sensitive => false, :allow_nil => true)
243
+
244
+ t = Topic.new("title" => "I'm unique!", :parent_id => 2)
245
+ assert t.save, "Should save t as unique"
246
+
247
+ t.content = "Remaining unique"
248
+ assert t.save, "Should still save t as unique"
249
+
250
+ t2 = Topic.new("title" => "I'm UNIQUE!", :parent_id => 1)
251
+ assert !t2.valid?, "Shouldn't be valid"
252
+ assert !t2.save, "Shouldn't save t2 as unique"
253
+ assert t2.errors[:title].any?
254
+ assert t2.errors[:parent_id].any?
255
+ assert_equal ["has already been taken"], t2.errors[:title]
256
+
257
+ t2.title = "I'm truly UNIQUE!"
258
+ assert !t2.valid?, "Shouldn't be valid"
259
+ assert !t2.save, "Shouldn't save t2 as unique"
260
+ assert t2.errors[:title].empty?
261
+ assert t2.errors[:parent_id].any?
262
+
263
+ t2.parent_id = 4
264
+ assert t2.save, "Should now save t2 as unique"
265
+
266
+ t2.parent_id = nil
267
+ t2.title = nil
268
+ assert t2.valid?, "should validate with nil"
269
+ assert t2.save, "should save with nil"
270
+
271
+ t_utf8 = Topic.new("title" => "Я тоже уникальный!")
272
+ assert t_utf8.save, "Should save t_utf8 as unique"
273
+
274
+ # If database hasn't UTF-8 character set, this test fails
275
+ if Topic.all.merge!(:select => 'LOWER(title) AS title').find(t_utf8.id).title == "я тоже уникальный!"
276
+ t2_utf8 = Topic.new("title" => "я тоже УНИКАЛЬНЫЙ!")
277
+ assert !t2_utf8.valid?, "Shouldn't be valid"
278
+ assert !t2_utf8.save, "Shouldn't save t2_utf8 as unique"
279
+ end
280
+ end
281
+
282
+ def test_validate_case_sensitive_uniqueness_with_special_sql_like_chars
283
+ Topic.validates_uniqueness_of(:title, :case_sensitive => true)
284
+
285
+ t = Topic.new("title" => "I'm unique!")
286
+ assert t.save, "Should save t as unique"
287
+
288
+ t2 = Topic.new("title" => "I'm %")
289
+ assert t2.save, "Should save t2 as unique"
290
+
291
+ t3 = Topic.new("title" => "I'm uniqu_!")
292
+ assert t3.save, "Should save t3 as unique"
293
+ end
294
+
295
+ def test_validate_case_insensitive_uniqueness_with_special_sql_like_chars
296
+ Topic.validates_uniqueness_of(:title, :case_sensitive => false)
297
+
298
+ t = Topic.new("title" => "I'm unique!")
299
+ assert t.save, "Should save t as unique"
300
+
301
+ t2 = Topic.new("title" => "I'm %")
302
+ assert t2.save, "Should save t2 as unique"
303
+
304
+ t3 = Topic.new("title" => "I'm uniqu_!")
305
+ assert t3.save, "Should save t3 as unique"
306
+ end
307
+
308
+ def test_validate_case_sensitive_uniqueness
309
+ Topic.validates_uniqueness_of(:title, :case_sensitive => true, :allow_nil => true)
310
+
311
+ t = Topic.new("title" => "I'm unique!")
312
+ assert t.save, "Should save t as unique"
313
+
314
+ t.content = "Remaining unique"
315
+ assert t.save, "Should still save t as unique"
316
+
317
+ t2 = Topic.new("title" => "I'M UNIQUE!")
318
+ assert t2.valid?, "Should be valid"
319
+ assert t2.save, "Should save t2 as unique"
320
+ assert t2.errors[:title].empty?
321
+ assert t2.errors[:parent_id].empty?
322
+ assert_not_equal ["has already been taken"], t2.errors[:title]
323
+
324
+ t3 = Topic.new("title" => "I'M uNiQUe!")
325
+ assert t3.valid?, "Should be valid"
326
+ assert t3.save, "Should save t2 as unique"
327
+ assert t3.errors[:title].empty?
328
+ assert t3.errors[:parent_id].empty?
329
+ assert_not_equal ["has already been taken"], t3.errors[:title]
330
+ end
331
+
332
+ def test_validate_case_sensitive_uniqueness_with_attribute_passed_as_integer
333
+ Topic.validates_uniqueness_of(:title, :case_sensitive => true)
334
+ Topic.create!('title' => 101)
335
+
336
+ t2 = Topic.new('title' => 101)
337
+ assert !t2.valid?
338
+ assert t2.errors[:title]
339
+ end
340
+
341
+ def test_validate_uniqueness_with_non_standard_table_names
342
+ i1 = WarehouseThing.create(:value => 1000)
343
+ assert !i1.valid?, "i1 should not be valid"
344
+ assert i1.errors[:value].any?, "Should not be empty"
345
+ end
346
+
347
+ def test_validates_uniqueness_inside_scoping
348
+ Topic.validates_uniqueness_of(:title)
349
+
350
+ Topic.where(:author_name => "David").scoping do
351
+ t1 = Topic.new("title" => "I'm unique!", "author_name" => "Mary")
352
+ assert t1.save
353
+ t2 = Topic.new("title" => "I'm unique!", "author_name" => "David")
354
+ assert !t2.valid?
355
+ end
356
+ end
357
+
358
+ def test_validate_uniqueness_with_columns_which_are_sql_keywords
359
+ repair_validations(Guid) do
360
+ Guid.validates_uniqueness_of :key
361
+ g = Guid.new
362
+ g.key = "foo"
363
+ assert_nothing_raised { !g.valid? }
364
+ end
365
+ end
366
+
367
+ def test_validate_uniqueness_with_limit
368
+ if current_adapter?(:SQLite3Adapter)
369
+ # Event.title has limit 5, but SQLite doesn't truncate.
370
+ e1 = Event.create(title: "abcdefgh")
371
+ assert e1.valid?, "Could not create an event with a unique 8 characters title"
372
+
373
+ e2 = Event.create(title: "abcdefgh")
374
+ assert_not e2.valid?, "Created an event whose title is not unique"
375
+ elsif current_adapter?(:Mysql2Adapter, :PostgreSQLAdapter, :OracleAdapter, :SQLServerAdapter)
376
+ assert_raise(ActiveRecord::ValueTooLong) do
377
+ Event.create(title: "abcdefgh")
378
+ end
379
+ else
380
+ assert_raise(ActiveRecord::StatementInvalid) do
381
+ Event.create(title: "abcdefgh")
382
+ end
383
+ end
384
+ end
385
+
386
+ def test_validate_uniqueness_with_limit_and_utf8
387
+ if current_adapter?(:SQLite3Adapter)
388
+ # Event.title has limit 5, but does SQLite doesn't truncate.
389
+ e1 = Event.create(title: "一二三四五六七八")
390
+ assert e1.valid?, "Could not create an event with a unique 8 characters title"
391
+
392
+ e2 = Event.create(title: "一二三四五六七八")
393
+ assert_not e2.valid?, "Created an event whose title is not unique"
394
+ elsif current_adapter?(:Mysql2Adapter, :PostgreSQLAdapter, :OracleAdapter, :SQLServerAdapter)
395
+ assert_raise(ActiveRecord::ValueTooLong) do
396
+ Event.create(title: "一二三四五六七八")
397
+ end
398
+ else
399
+ assert_raise(ActiveRecord::StatementInvalid) do
400
+ Event.create(title: "一二三四五六七八")
401
+ end
402
+ end
403
+ end
404
+
405
+ def test_validate_straight_inheritance_uniqueness
406
+ w1 = IneptWizard.create(:name => "Rincewind", :city => "Ankh-Morpork")
407
+ assert w1.valid?, "Saving w1"
408
+
409
+ # Should use validation from base class (which is abstract)
410
+ w2 = IneptWizard.new(:name => "Rincewind", :city => "Quirm")
411
+ assert !w2.valid?, "w2 shouldn't be valid"
412
+ assert w2.errors[:name].any?, "Should have errors for name"
413
+ assert_equal ["has already been taken"], w2.errors[:name], "Should have uniqueness message for name"
414
+
415
+ w3 = Conjurer.new(:name => "Rincewind", :city => "Quirm")
416
+ assert !w3.valid?, "w3 shouldn't be valid"
417
+ assert w3.errors[:name].any?, "Should have errors for name"
418
+ assert_equal ["has already been taken"], w3.errors[:name], "Should have uniqueness message for name"
419
+
420
+ w4 = Conjurer.create(:name => "The Amazing Bonko", :city => "Quirm")
421
+ assert w4.valid?, "Saving w4"
422
+
423
+ w5 = Thaumaturgist.new(:name => "The Amazing Bonko", :city => "Lancre")
424
+ assert !w5.valid?, "w5 shouldn't be valid"
425
+ assert w5.errors[:name].any?, "Should have errors for name"
426
+ assert_equal ["has already been taken"], w5.errors[:name], "Should have uniqueness message for name"
427
+
428
+ w6 = Thaumaturgist.new(:name => "Mustrum Ridcully", :city => "Quirm")
429
+ assert !w6.valid?, "w6 shouldn't be valid"
430
+ assert w6.errors[:city].any?, "Should have errors for city"
431
+ assert_equal ["has already been taken"], w6.errors[:city], "Should have uniqueness message for city"
432
+ end
433
+
434
+ def test_validate_uniqueness_with_conditions
435
+ Topic.validates_uniqueness_of :title, conditions: -> { where(approved: true) }
436
+ Topic.create("title" => "I'm a topic", "approved" => true)
437
+ Topic.create("title" => "I'm an unapproved topic", "approved" => false)
438
+
439
+ t3 = Topic.new("title" => "I'm a topic", "approved" => true)
440
+ assert !t3.valid?, "t3 shouldn't be valid"
441
+
442
+ t4 = Topic.new("title" => "I'm an unapproved topic", "approved" => false)
443
+ assert t4.valid?, "t4 should be valid"
444
+ end
445
+
446
+ def test_validate_uniqueness_with_non_callable_conditions_is_not_supported
447
+ assert_raises(ArgumentError) {
448
+ Topic.validates_uniqueness_of :title, conditions: Topic.where(approved: true)
449
+ }
450
+ end
451
+
452
+ def test_validate_uniqueness_on_existing_relation
453
+ event = Event.create
454
+ assert TopicWithUniqEvent.create(event: event).valid?
455
+
456
+ topic = TopicWithUniqEvent.new(event: event)
457
+ assert_not topic.valid?
458
+ assert_equal ['has already been taken'], topic.errors[:event]
459
+ end
460
+
461
+ def test_validate_uniqueness_on_empty_relation
462
+ topic = TopicWithUniqEvent.new
463
+ assert topic.valid?
464
+ end
465
+
466
+ def test_validate_uniqueness_of_custom_primary_key
467
+ klass = Class.new(ActiveRecord::Base) do
468
+ self.table_name = "keyboards"
469
+ self.primary_key = :key_number
470
+
471
+ validates_uniqueness_of :key_number
472
+
473
+ def self.name
474
+ "Keyboard"
475
+ end
476
+ end
477
+
478
+ klass.create!(key_number: 10)
479
+ key2 = klass.create!(key_number: 11)
480
+
481
+ key2.key_number = 10
482
+ assert_not key2.valid?
483
+ end
484
+
485
+ def test_validate_uniqueness_without_primary_key
486
+ klass = Class.new(ActiveRecord::Base) do
487
+ self.table_name = "dashboards"
488
+
489
+ validates_uniqueness_of :dashboard_id
490
+
491
+ def self.name; "Dashboard" end
492
+ end
493
+
494
+ abc = klass.create!(dashboard_id: "abc")
495
+ assert klass.new(dashboard_id: "xyz").valid?
496
+ assert_not klass.new(dashboard_id: "abc").valid?
497
+
498
+ abc.dashboard_id = "def"
499
+
500
+ e = assert_raises ActiveRecord::UnknownPrimaryKey do
501
+ abc.save!
502
+ end
503
+ assert_match(/\AUnknown primary key for table dashboards in model/, e.message)
504
+ assert_match(/Can not validate uniqueness for persisted record without primary key.\z/, e.message)
505
+ end
506
+
507
+ def test_validate_uniqueness_ignores_itself_when_primary_key_changed
508
+ Topic.validates_uniqueness_of(:title)
509
+
510
+ t = Topic.new("title" => "This is a unique title")
511
+ assert t.save, "Should save t as unique"
512
+
513
+ t.id += 1
514
+ assert t.valid?, "Should be valid"
515
+ assert t.save, "Should still save t as unique"
516
+ end
517
+
518
+ def test_validate_uniqueness_with_after_create_performing_save
519
+ TopicWithAfterCreate.validates_uniqueness_of(:title)
520
+ topic = TopicWithAfterCreate.create!(:title => "Title1")
521
+ assert topic.author_name.start_with?("Title1")
522
+
523
+ topic2 = TopicWithAfterCreate.new(:title => "Title1")
524
+ refute topic2.valid?
525
+ assert_equal(["has already been taken"], topic2.errors[:title])
526
+ end
527
+
528
+ def test_validate_uniqueness_uuid
529
+ skip unless current_adapter?(:PostgreSQLAdapter)
530
+ item = UuidItem.create!(uuid: SecureRandom.uuid, title: 'item1')
531
+ item.update(title: 'item1-title2')
532
+ assert_empty item.errors
533
+
534
+ item2 = UuidValidatingItem.create!(uuid: SecureRandom.uuid, title: 'item2')
535
+ item2.update(title: 'item2-title2')
536
+ assert_empty item2.errors
537
+ end
538
+
539
+ def test_validate_uniqueness_regular_id
540
+ item = CoolTopic.create!(title: 'MyItem')
541
+ assert_empty item.errors
542
+
543
+ item2 = CoolTopic.new(id: item.id, title: 'MyItem2')
544
+ refute item2.valid?
545
+
546
+ assert_equal(["has already been taken"], item2.errors[:id])
547
+ end
548
+ end