ibm_db 3.0.5-x86-mingw32 → 4.0.0-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 (586) hide show
  1. checksums.yaml +4 -4
  2. data/CHANGES +4 -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.nt32 +181 -181
  8. data/ext/Makefile.nt32.191 +212 -212
  9. data/ext/extconf.rb +291 -291
  10. data/ext/ibm_db.c +11887 -11887
  11. data/ext/ruby_ibm_db.h +241 -241
  12. data/ext/ruby_ibm_db_cli.c +866 -866
  13. data/ext/ruby_ibm_db_cli.h +500 -500
  14. data/init.rb +41 -41
  15. data/lib/IBM_DB.rb +27 -27
  16. data/lib/active_record/connection_adapters/ibm_db_adapter.rb +3452 -3177
  17. data/lib/active_record/connection_adapters/ibmdb_adapter.rb +5 -2
  18. data/lib/active_record/vendor/db2-i5-zOS.yaml +328 -328
  19. data/lib/mswin32/ibm_db.rb +91 -123
  20. data/lib/mswin32/rb2x/i386/ibm_db.so +0 -0
  21. data/test/active_record/connection_adapters/fake_adapter.rb +49 -46
  22. data/test/assets/example.log +1 -1
  23. data/test/assets/test.txt +1 -1
  24. data/test/cases/adapter_test.rb +351 -276
  25. data/test/cases/adapters/mysql2/active_schema_test.rb +193 -0
  26. data/test/cases/adapters/mysql2/bind_parameter_test.rb +50 -0
  27. data/test/cases/adapters/mysql2/boolean_test.rb +100 -0
  28. data/test/cases/adapters/mysql2/case_sensitivity_test.rb +63 -0
  29. data/test/cases/adapters/mysql2/charset_collation_test.rb +54 -0
  30. data/test/cases/adapters/mysql2/connection_test.rb +210 -0
  31. data/test/cases/adapters/mysql2/datetime_precision_quoting_test.rb +45 -0
  32. data/test/cases/adapters/mysql2/enum_test.rb +26 -0
  33. data/test/cases/adapters/mysql2/explain_test.rb +21 -0
  34. data/test/cases/adapters/mysql2/json_test.rb +195 -0
  35. data/test/cases/adapters/mysql2/mysql2_adapter_test.rb +83 -0
  36. data/test/cases/adapters/mysql2/reserved_word_test.rb +152 -0
  37. data/test/cases/adapters/mysql2/schema_migrations_test.rb +59 -0
  38. data/test/cases/adapters/mysql2/schema_test.rb +126 -0
  39. data/test/cases/adapters/mysql2/sp_test.rb +36 -0
  40. data/test/cases/adapters/mysql2/sql_types_test.rb +14 -0
  41. data/test/cases/adapters/mysql2/table_options_test.rb +42 -0
  42. data/test/cases/adapters/mysql2/unsigned_type_test.rb +66 -0
  43. data/test/cases/adapters/postgresql/active_schema_test.rb +98 -0
  44. data/test/cases/adapters/postgresql/array_test.rb +339 -0
  45. data/test/cases/adapters/postgresql/bit_string_test.rb +82 -0
  46. data/test/cases/adapters/postgresql/bytea_test.rb +134 -0
  47. data/test/cases/adapters/postgresql/case_insensitive_test.rb +26 -0
  48. data/test/cases/adapters/postgresql/change_schema_test.rb +38 -0
  49. data/test/cases/adapters/postgresql/cidr_test.rb +25 -0
  50. data/test/cases/adapters/postgresql/citext_test.rb +78 -0
  51. data/test/cases/adapters/postgresql/collation_test.rb +53 -0
  52. data/test/cases/adapters/postgresql/composite_test.rb +132 -0
  53. data/test/cases/adapters/postgresql/connection_test.rb +257 -0
  54. data/test/cases/adapters/postgresql/datatype_test.rb +92 -0
  55. data/test/cases/adapters/postgresql/domain_test.rb +47 -0
  56. data/test/cases/adapters/postgresql/enum_test.rb +91 -0
  57. data/test/cases/adapters/postgresql/explain_test.rb +20 -0
  58. data/test/cases/adapters/postgresql/extension_migration_test.rb +63 -0
  59. data/test/cases/adapters/postgresql/full_text_test.rb +44 -0
  60. data/test/cases/adapters/postgresql/geometric_test.rb +378 -0
  61. data/test/cases/adapters/postgresql/hstore_test.rb +382 -0
  62. data/test/cases/adapters/postgresql/infinity_test.rb +69 -0
  63. data/test/cases/adapters/postgresql/integer_test.rb +25 -0
  64. data/test/cases/adapters/postgresql/json_test.rb +237 -0
  65. data/test/cases/adapters/postgresql/ltree_test.rb +53 -0
  66. data/test/cases/adapters/postgresql/money_test.rb +96 -0
  67. data/test/cases/adapters/postgresql/network_test.rb +94 -0
  68. data/test/cases/adapters/postgresql/numbers_test.rb +49 -0
  69. data/test/cases/adapters/postgresql/postgresql_adapter_test.rb +405 -0
  70. data/test/cases/adapters/postgresql/prepared_statements_test.rb +22 -0
  71. data/test/cases/adapters/postgresql/quoting_test.rb +44 -0
  72. data/test/cases/adapters/postgresql/range_test.rb +343 -0
  73. data/test/cases/adapters/postgresql/referential_integrity_test.rb +111 -0
  74. data/test/cases/adapters/postgresql/rename_table_test.rb +34 -0
  75. data/test/cases/adapters/postgresql/schema_authorization_test.rb +119 -0
  76. data/test/cases/adapters/postgresql/schema_test.rb +597 -0
  77. data/test/cases/adapters/postgresql/serial_test.rb +154 -0
  78. data/test/cases/adapters/postgresql/statement_pool_test.rb +41 -0
  79. data/test/cases/adapters/postgresql/timestamp_test.rb +90 -0
  80. data/test/cases/adapters/postgresql/type_lookup_test.rb +33 -0
  81. data/test/cases/adapters/postgresql/utils_test.rb +62 -0
  82. data/test/cases/adapters/postgresql/uuid_test.rb +294 -0
  83. data/test/cases/adapters/postgresql/xml_test.rb +54 -0
  84. data/test/cases/adapters/sqlite3/collation_test.rb +53 -0
  85. data/test/cases/adapters/sqlite3/copy_table_test.rb +98 -0
  86. data/test/cases/adapters/sqlite3/explain_test.rb +21 -0
  87. data/test/cases/adapters/sqlite3/quoting_test.rb +101 -0
  88. data/test/cases/adapters/sqlite3/sqlite3_adapter_test.rb +441 -0
  89. data/test/cases/adapters/sqlite3/sqlite3_create_folder_test.rb +24 -0
  90. data/test/cases/adapters/sqlite3/statement_pool_test.rb +20 -0
  91. data/test/cases/aggregations_test.rb +168 -158
  92. data/test/cases/ar_schema_test.rb +146 -161
  93. data/test/cases/associations/association_scope_test.rb +16 -21
  94. data/test/cases/associations/belongs_to_associations_test.rb +1141 -1029
  95. data/test/cases/associations/bidirectional_destroy_dependencies_test.rb +41 -0
  96. data/test/cases/associations/callbacks_test.rb +190 -192
  97. data/test/cases/associations/cascaded_eager_loading_test.rb +188 -188
  98. data/test/cases/associations/eager_load_includes_full_sti_class_test.rb +36 -36
  99. data/test/cases/associations/eager_load_nested_include_test.rb +126 -128
  100. data/test/cases/associations/eager_singularization_test.rb +148 -148
  101. data/test/cases/associations/eager_test.rb +1514 -1429
  102. data/test/cases/associations/extension_test.rb +87 -82
  103. data/test/cases/associations/has_and_belongs_to_many_associations_test.rb +1004 -972
  104. data/test/cases/associations/has_many_associations_test.rb +2501 -2182
  105. data/test/cases/associations/has_many_through_associations_test.rb +1271 -1204
  106. data/test/cases/associations/has_one_associations_test.rb +707 -610
  107. data/test/cases/associations/has_one_through_associations_test.rb +383 -380
  108. data/test/cases/associations/inner_join_association_test.rb +139 -139
  109. data/test/cases/associations/inverse_associations_test.rb +733 -706
  110. data/test/cases/associations/join_model_test.rb +777 -754
  111. data/test/cases/associations/left_outer_join_association_test.rb +88 -0
  112. data/test/cases/associations/nested_through_associations_test.rb +579 -579
  113. data/test/cases/associations/required_test.rb +102 -82
  114. data/test/cases/associations_test.rb +385 -380
  115. data/test/cases/attribute_decorators_test.rb +125 -125
  116. data/test/cases/attribute_methods/read_test.rb +60 -60
  117. data/test/cases/attribute_methods_test.rb +1009 -952
  118. data/test/cases/attribute_set_test.rb +270 -210
  119. data/test/cases/attribute_test.rb +246 -180
  120. data/test/cases/attributes_test.rb +253 -136
  121. data/test/cases/autosave_association_test.rb +1708 -1595
  122. data/test/cases/base_test.rb +1713 -1664
  123. data/test/cases/batches_test.rb +489 -212
  124. data/test/cases/binary_test.rb +44 -52
  125. data/test/cases/bind_parameter_test.rb +110 -100
  126. data/test/cases/cache_key_test.rb +25 -0
  127. data/test/cases/calculations_test.rb +798 -646
  128. data/test/cases/callbacks_test.rb +636 -543
  129. data/test/cases/clone_test.rb +40 -40
  130. data/test/cases/coders/json_test.rb +15 -0
  131. data/test/cases/coders/yaml_column_test.rb +63 -63
  132. data/test/cases/collection_cache_key_test.rb +115 -0
  133. data/test/cases/column_alias_test.rb +17 -17
  134. data/test/cases/column_definition_test.rb +92 -123
  135. data/test/cases/comment_test.rb +143 -0
  136. data/test/cases/connection_adapters/adapter_leasing_test.rb +56 -54
  137. data/test/cases/connection_adapters/connection_handler_test.rb +160 -53
  138. data/test/cases/connection_adapters/connection_specification_test.rb +12 -12
  139. data/test/cases/connection_adapters/merge_and_resolve_default_url_config_test.rb +255 -293
  140. data/test/cases/connection_adapters/mysql_type_lookup_test.rb +69 -65
  141. data/test/cases/connection_adapters/quoting_test.rb +13 -13
  142. data/test/cases/connection_adapters/schema_cache_test.rb +61 -56
  143. data/test/cases/connection_adapters/type_lookup_test.rb +118 -110
  144. data/test/cases/connection_management_test.rb +112 -122
  145. data/test/cases/connection_pool_test.rb +521 -346
  146. data/test/cases/connection_specification/resolver_test.rb +131 -116
  147. data/test/cases/core_test.rb +112 -112
  148. data/test/cases/counter_cache_test.rb +214 -209
  149. data/test/cases/custom_locking_test.rb +17 -17
  150. data/test/cases/database_statements_test.rb +34 -19
  151. data/test/cases/{invalid_date_test.rb → date_test.rb} +44 -32
  152. data/test/cases/date_time_precision_test.rb +106 -0
  153. data/test/cases/date_time_test.rb +61 -61
  154. data/test/cases/defaults_test.rb +218 -223
  155. data/test/cases/dirty_test.rb +763 -785
  156. data/test/cases/disconnected_test.rb +30 -28
  157. data/test/cases/dup_test.rb +157 -157
  158. data/test/cases/enum_test.rb +444 -290
  159. data/test/cases/errors_test.rb +16 -0
  160. data/test/cases/explain_subscriber_test.rb +64 -64
  161. data/test/cases/explain_test.rb +87 -76
  162. data/test/cases/finder_respond_to_test.rb +60 -60
  163. data/test/cases/finder_test.rb +1294 -1169
  164. data/test/cases/fixture_set/file_test.rb +156 -138
  165. data/test/cases/fixtures_test.rb +988 -908
  166. data/test/cases/forbidden_attributes_protection_test.rb +165 -99
  167. data/test/cases/habtm_destroy_order_test.rb +61 -61
  168. data/test/cases/helper.rb +204 -210
  169. data/test/cases/hot_compatibility_test.rb +142 -54
  170. data/test/cases/i18n_test.rb +45 -45
  171. data/test/cases/inheritance_test.rb +606 -375
  172. data/test/cases/integration_test.rb +155 -139
  173. data/test/cases/invalid_connection_test.rb +24 -22
  174. data/test/cases/invertible_migration_test.rb +387 -295
  175. data/test/cases/json_serialization_test.rb +311 -302
  176. data/test/cases/locking_test.rb +493 -477
  177. data/test/cases/log_subscriber_test.rb +225 -136
  178. data/test/cases/migration/change_schema_test.rb +458 -512
  179. data/test/cases/migration/change_table_test.rb +256 -224
  180. data/test/cases/migration/column_attributes_test.rb +176 -192
  181. data/test/cases/migration/column_positioning_test.rb +56 -56
  182. data/test/cases/migration/columns_test.rb +310 -304
  183. data/test/cases/migration/command_recorder_test.rb +350 -305
  184. data/test/cases/migration/compatibility_test.rb +118 -0
  185. data/test/cases/migration/create_join_table_test.rb +157 -148
  186. data/test/cases/migration/foreign_key_test.rb +360 -328
  187. data/test/cases/migration/helper.rb +39 -39
  188. data/test/cases/migration/index_test.rb +218 -216
  189. data/test/cases/migration/logger_test.rb +36 -36
  190. data/test/cases/migration/pending_migrations_test.rb +52 -53
  191. data/test/cases/migration/references_foreign_key_test.rb +216 -169
  192. data/test/cases/migration/references_index_test.rb +101 -101
  193. data/test/cases/migration/references_statements_test.rb +136 -116
  194. data/test/cases/migration/rename_table_test.rb +93 -93
  195. data/test/cases/migration_test.rb +1157 -959
  196. data/test/cases/migrator_test.rb +470 -388
  197. data/test/cases/mixin_test.rb +68 -70
  198. data/test/cases/modules_test.rb +172 -173
  199. data/test/cases/multiparameter_attributes_test.rb +372 -350
  200. data/test/cases/multiple_db_test.rb +122 -115
  201. data/test/cases/nested_attributes_test.rb +1098 -1070
  202. data/test/cases/nested_attributes_with_callbacks_test.rb +144 -144
  203. data/test/cases/persistence_test.rb +1001 -909
  204. data/test/cases/pooled_connections_test.rb +81 -81
  205. data/test/cases/primary_keys_test.rb +376 -237
  206. data/test/cases/query_cache_test.rb +446 -326
  207. data/test/cases/quoting_test.rb +202 -156
  208. data/test/cases/readonly_test.rb +119 -118
  209. data/test/cases/reaper_test.rb +85 -85
  210. data/test/cases/reflection_test.rb +509 -463
  211. data/test/cases/relation/delegation_test.rb +63 -68
  212. data/test/cases/relation/merging_test.rb +157 -161
  213. data/test/cases/relation/mutation_test.rb +183 -165
  214. data/test/cases/relation/or_test.rb +92 -0
  215. data/test/cases/relation/predicate_builder_test.rb +16 -14
  216. data/test/cases/relation/record_fetch_warning_test.rb +40 -0
  217. data/test/cases/relation/where_chain_test.rb +105 -181
  218. data/test/cases/relation/where_clause_test.rb +182 -0
  219. data/test/cases/relation/where_test.rb +322 -300
  220. data/test/cases/relation_test.rb +328 -319
  221. data/test/cases/relations_test.rb +2026 -1815
  222. data/test/cases/reload_models_test.rb +22 -22
  223. data/test/cases/result_test.rb +90 -80
  224. data/test/cases/sanitize_test.rb +176 -83
  225. data/test/cases/schema_dumper_test.rb +457 -463
  226. data/test/cases/schema_loading_test.rb +52 -0
  227. data/test/cases/scoping/default_scoping_test.rb +528 -454
  228. data/test/cases/scoping/named_scoping_test.rb +561 -524
  229. data/test/cases/scoping/relation_scoping_test.rb +400 -357
  230. data/test/cases/secure_token_test.rb +32 -0
  231. data/test/cases/serialization_test.rb +104 -104
  232. data/test/cases/serialized_attribute_test.rb +364 -277
  233. data/test/cases/statement_cache_test.rb +136 -98
  234. data/test/cases/store_test.rb +195 -194
  235. data/test/cases/suppressor_test.rb +63 -0
  236. data/test/cases/tasks/database_tasks_test.rb +462 -398
  237. data/test/cases/tasks/mysql_rake_test.rb +345 -324
  238. data/test/cases/tasks/postgresql_rake_test.rb +304 -250
  239. data/test/cases/tasks/sqlite_rake_test.rb +220 -193
  240. data/test/cases/test_case.rb +131 -123
  241. data/test/cases/test_fixtures_test.rb +36 -0
  242. data/test/cases/time_precision_test.rb +102 -0
  243. data/test/cases/timestamp_test.rb +501 -467
  244. data/test/cases/touch_later_test.rb +121 -0
  245. data/test/cases/transaction_callbacks_test.rb +518 -452
  246. data/test/cases/transaction_isolation_test.rb +106 -106
  247. data/test/cases/transactions_test.rb +834 -817
  248. data/test/cases/type/adapter_specific_registry_test.rb +133 -0
  249. data/test/cases/type/date_time_test.rb +14 -0
  250. data/test/cases/type/integer_test.rb +27 -121
  251. data/test/cases/type/string_test.rb +22 -36
  252. data/test/cases/type/type_map_test.rb +177 -177
  253. data/test/cases/type_test.rb +39 -0
  254. data/test/cases/types_test.rb +24 -141
  255. data/test/cases/unconnected_test.rb +33 -33
  256. data/test/cases/validations/absence_validation_test.rb +73 -0
  257. data/test/cases/validations/association_validation_test.rb +97 -86
  258. data/test/cases/validations/i18n_generate_message_validation_test.rb +84 -84
  259. data/test/cases/validations/i18n_validation_test.rb +86 -90
  260. data/test/cases/validations/length_validation_test.rb +79 -47
  261. data/test/cases/validations/presence_validation_test.rb +103 -68
  262. data/test/cases/validations/uniqueness_validation_test.rb +548 -457
  263. data/test/cases/validations_repair_helper.rb +19 -23
  264. data/test/cases/validations_test.rb +194 -165
  265. data/test/cases/view_test.rb +216 -119
  266. data/test/cases/yaml_serialization_test.rb +121 -126
  267. data/test/config.example.yml +97 -0
  268. data/test/config.rb +5 -5
  269. data/test/fixtures/accounts.yml +29 -29
  270. data/test/fixtures/admin/accounts.yml +2 -2
  271. data/test/fixtures/admin/users.yml +10 -10
  272. data/test/fixtures/author_addresses.original +11 -0
  273. data/test/fixtures/author_addresses.yml +17 -17
  274. data/test/fixtures/author_favorites.yml +3 -3
  275. data/test/fixtures/authors.original +17 -0
  276. data/test/fixtures/authors.yml +23 -23
  277. data/test/fixtures/bad_posts.yml +9 -0
  278. data/test/fixtures/binaries.yml +133 -133
  279. data/test/fixtures/books.yml +31 -11
  280. data/test/fixtures/bulbs.yml +5 -5
  281. data/test/fixtures/cars.yml +9 -9
  282. data/test/fixtures/categories.yml +19 -19
  283. data/test/fixtures/categories/special_categories.yml +9 -9
  284. data/test/fixtures/categories/subsubdir/arbitrary_filename.yml +4 -4
  285. data/test/fixtures/categories_ordered.yml +7 -7
  286. data/test/fixtures/categories_posts.yml +31 -31
  287. data/test/fixtures/categorizations.yml +23 -23
  288. data/test/fixtures/clubs.yml +8 -8
  289. data/test/fixtures/collections.yml +3 -3
  290. data/test/fixtures/colleges.yml +3 -3
  291. data/test/fixtures/comments.yml +65 -65
  292. data/test/fixtures/companies.yml +67 -67
  293. data/test/fixtures/computers.yml +10 -10
  294. data/test/fixtures/content.yml +3 -0
  295. data/test/fixtures/content_positions.yml +3 -0
  296. data/test/fixtures/courses.yml +8 -8
  297. data/test/fixtures/customers.yml +25 -25
  298. data/test/fixtures/dashboards.yml +6 -6
  299. data/test/fixtures/dead_parrots.yml +5 -0
  300. data/test/fixtures/developers.yml +22 -22
  301. data/test/fixtures/developers_projects.yml +16 -16
  302. data/test/fixtures/dog_lovers.yml +7 -7
  303. data/test/fixtures/dogs.yml +4 -4
  304. data/test/fixtures/doubloons.yml +3 -3
  305. data/test/fixtures/edges.yml +5 -5
  306. data/test/fixtures/entrants.yml +14 -14
  307. data/test/fixtures/essays.yml +6 -6
  308. data/test/fixtures/faces.yml +11 -11
  309. data/test/fixtures/fk_test_has_fk.yml +3 -3
  310. data/test/fixtures/fk_test_has_pk.yml +1 -1
  311. data/test/fixtures/friendships.yml +4 -4
  312. data/test/fixtures/funny_jokes.yml +10 -10
  313. data/test/fixtures/interests.yml +33 -33
  314. data/test/fixtures/items.yml +3 -3
  315. data/test/fixtures/jobs.yml +7 -7
  316. data/test/fixtures/legacy_things.yml +3 -3
  317. data/test/fixtures/live_parrots.yml +4 -0
  318. data/test/fixtures/mateys.yml +4 -4
  319. data/test/fixtures/member_details.yml +8 -8
  320. data/test/fixtures/member_types.yml +6 -6
  321. data/test/fixtures/members.yml +11 -11
  322. data/test/fixtures/memberships.yml +34 -34
  323. data/test/fixtures/men.yml +5 -5
  324. data/test/fixtures/minimalistics.yml +2 -2
  325. data/test/fixtures/minivans.yml +5 -5
  326. data/test/fixtures/mixed_case_monkeys.yml +6 -6
  327. data/test/fixtures/mixins.yml +29 -29
  328. data/test/fixtures/movies.yml +7 -7
  329. data/test/fixtures/naked/yml/accounts.yml +1 -1
  330. data/test/fixtures/naked/yml/companies.yml +1 -1
  331. data/test/fixtures/naked/yml/courses.yml +1 -1
  332. data/test/fixtures/naked/yml/parrots.yml +2 -0
  333. data/test/fixtures/naked/yml/trees.yml +3 -0
  334. data/test/fixtures/nodes.yml +29 -0
  335. data/test/fixtures/organizations.yml +5 -5
  336. data/test/fixtures/other_comments.yml +6 -0
  337. data/test/fixtures/other_dogs.yml +2 -0
  338. data/test/fixtures/other_posts.yml +7 -0
  339. data/test/fixtures/other_topics.yml +42 -42
  340. data/test/fixtures/owners.yml +9 -9
  341. data/test/fixtures/parrots.yml +27 -27
  342. data/test/fixtures/parrots_pirates.yml +7 -7
  343. data/test/fixtures/people.yml +24 -24
  344. data/test/fixtures/peoples_treasures.yml +3 -3
  345. data/test/fixtures/pets.yml +19 -19
  346. data/test/fixtures/pirates.yml +15 -12
  347. data/test/fixtures/posts.yml +80 -80
  348. data/test/fixtures/price_estimates.yml +16 -7
  349. data/test/fixtures/products.yml +4 -4
  350. data/test/fixtures/projects.yml +7 -7
  351. data/test/fixtures/ratings.yml +14 -14
  352. data/test/fixtures/readers.yml +11 -11
  353. data/test/fixtures/references.yml +17 -17
  354. data/test/fixtures/reserved_words/distinct.yml +5 -5
  355. data/test/fixtures/reserved_words/distinct_select.yml +11 -11
  356. data/test/fixtures/reserved_words/group.yml +14 -14
  357. data/test/fixtures/reserved_words/select.yml +8 -8
  358. data/test/fixtures/reserved_words/values.yml +7 -7
  359. data/test/fixtures/ships.yml +6 -6
  360. data/test/fixtures/speedometers.yml +8 -8
  361. data/test/fixtures/sponsors.yml +12 -12
  362. data/test/fixtures/string_key_objects.yml +7 -7
  363. data/test/fixtures/subscribers.yml +10 -10
  364. data/test/fixtures/subscriptions.yml +12 -12
  365. data/test/fixtures/taggings.yml +78 -78
  366. data/test/fixtures/tags.yml +11 -11
  367. data/test/fixtures/tasks.yml +7 -7
  368. data/test/fixtures/teapots.yml +3 -3
  369. data/test/fixtures/to_be_linked/accounts.yml +2 -2
  370. data/test/fixtures/to_be_linked/users.yml +10 -10
  371. data/test/fixtures/topics.yml +49 -49
  372. data/test/fixtures/toys.yml +14 -14
  373. data/test/fixtures/traffic_lights.yml +9 -9
  374. data/test/fixtures/treasures.yml +10 -10
  375. data/test/fixtures/trees.yml +3 -0
  376. data/test/fixtures/uuid_children.yml +3 -3
  377. data/test/fixtures/uuid_parents.yml +2 -2
  378. data/test/fixtures/variants.yml +4 -4
  379. data/test/fixtures/vegetables.yml +19 -19
  380. data/test/fixtures/vertices.yml +3 -3
  381. data/test/fixtures/warehouse_things.yml +2 -2
  382. data/test/fixtures/zines.yml +5 -5
  383. data/test/migrations/10_urban/9_add_expressions.rb +11 -11
  384. data/test/migrations/decimal/1_give_me_big_numbers.rb +15 -15
  385. data/test/migrations/magic/1_currencies_have_symbols.rb +12 -12
  386. data/test/migrations/missing/1000_people_have_middle_names.rb +9 -9
  387. data/test/migrations/missing/1_people_have_last_names.rb +9 -9
  388. data/test/migrations/missing/3_we_need_reminders.rb +12 -12
  389. data/test/migrations/missing/4_innocent_jointable.rb +12 -12
  390. data/test/migrations/rename/1_we_need_things.rb +11 -11
  391. data/test/migrations/rename/2_rename_things.rb +9 -9
  392. data/test/migrations/to_copy/1_people_have_hobbies.rb +9 -9
  393. data/test/migrations/to_copy/2_people_have_descriptions.rb +9 -9
  394. data/test/migrations/to_copy2/1_create_articles.rb +7 -7
  395. data/test/migrations/to_copy2/2_create_comments.rb +7 -7
  396. data/test/migrations/to_copy_with_name_collision/1_people_have_hobbies.rb +9 -9
  397. data/test/migrations/to_copy_with_timestamps/20090101010101_people_have_hobbies.rb +9 -9
  398. data/test/migrations/to_copy_with_timestamps/20090101010202_people_have_descriptions.rb +9 -9
  399. data/test/migrations/to_copy_with_timestamps2/20090101010101_create_articles.rb +7 -7
  400. data/test/migrations/to_copy_with_timestamps2/20090101010202_create_comments.rb +7 -7
  401. data/test/migrations/valid/1_valid_people_have_last_names.rb +9 -9
  402. data/test/migrations/valid/2_we_need_reminders.rb +12 -12
  403. data/test/migrations/valid/3_innocent_jointable.rb +12 -12
  404. data/test/migrations/valid_with_subdirectories/1_valid_people_have_last_names.rb +9 -9
  405. data/test/migrations/valid_with_subdirectories/sub/2_we_need_reminders.rb +12 -12
  406. data/test/migrations/valid_with_subdirectories/sub1/3_innocent_jointable.rb +12 -12
  407. data/test/migrations/valid_with_timestamps/20100101010101_valid_with_timestamps_people_have_last_names.rb +9 -9
  408. data/test/migrations/valid_with_timestamps/20100201010101_valid_with_timestamps_we_need_reminders.rb +12 -12
  409. data/test/migrations/valid_with_timestamps/20100301010101_valid_with_timestamps_innocent_jointable.rb +12 -12
  410. data/test/migrations/version_check/20131219224947_migration_version_check.rb +8 -8
  411. data/test/models/admin.rb +5 -5
  412. data/test/models/admin/account.rb +3 -3
  413. data/test/models/admin/randomly_named_c1.rb +6 -2
  414. data/test/models/admin/user.rb +40 -40
  415. data/test/models/aircraft.rb +5 -4
  416. data/test/models/arunit2_model.rb +3 -3
  417. data/test/models/author.rb +209 -212
  418. data/test/models/auto_id.rb +4 -4
  419. data/test/models/autoloadable/extra_firm.rb +2 -2
  420. data/test/models/binary.rb +2 -2
  421. data/test/models/bird.rb +12 -12
  422. data/test/models/book.rb +23 -18
  423. data/test/models/boolean.rb +2 -2
  424. data/test/models/bulb.rb +52 -51
  425. data/test/models/cake_designer.rb +3 -3
  426. data/test/models/car.rb +29 -26
  427. data/test/models/carrier.rb +2 -2
  428. data/test/models/cat.rb +10 -0
  429. data/test/models/categorization.rb +19 -19
  430. data/test/models/category.rb +35 -35
  431. data/test/models/chef.rb +8 -7
  432. data/test/models/citation.rb +3 -3
  433. data/test/models/club.rb +25 -23
  434. data/test/models/college.rb +10 -10
  435. data/test/models/column.rb +3 -3
  436. data/test/models/column_name.rb +3 -3
  437. data/test/models/comment.rb +76 -64
  438. data/test/models/company.rb +230 -228
  439. data/test/models/company_in_module.rb +98 -98
  440. data/test/models/computer.rb +3 -3
  441. data/test/models/contact.rb +41 -41
  442. data/test/models/content.rb +40 -0
  443. data/test/models/contract.rb +20 -20
  444. data/test/models/country.rb +7 -7
  445. data/test/models/course.rb +6 -6
  446. data/test/models/customer.rb +83 -77
  447. data/test/models/customer_carrier.rb +14 -14
  448. data/test/models/dashboard.rb +3 -3
  449. data/test/models/default.rb +2 -2
  450. data/test/models/department.rb +4 -4
  451. data/test/models/developer.rb +274 -255
  452. data/test/models/dog.rb +5 -5
  453. data/test/models/dog_lover.rb +5 -5
  454. data/test/models/doubloon.rb +12 -12
  455. data/test/models/drink_designer.rb +3 -3
  456. data/test/models/edge.rb +5 -5
  457. data/test/models/electron.rb +5 -5
  458. data/test/models/engine.rb +4 -4
  459. data/test/models/entrant.rb +3 -3
  460. data/test/models/essay.rb +5 -5
  461. data/test/models/event.rb +3 -3
  462. data/test/models/eye.rb +37 -37
  463. data/test/models/face.rb +9 -9
  464. data/test/models/friendship.rb +6 -6
  465. data/test/models/guid.rb +2 -2
  466. data/test/models/guitar.rb +4 -0
  467. data/test/models/hotel.rb +11 -9
  468. data/test/models/image.rb +3 -3
  469. data/test/models/interest.rb +5 -5
  470. data/test/models/invoice.rb +4 -4
  471. data/test/models/item.rb +7 -7
  472. data/test/models/job.rb +7 -7
  473. data/test/models/joke.rb +7 -7
  474. data/test/models/keyboard.rb +3 -3
  475. data/test/models/legacy_thing.rb +3 -3
  476. data/test/models/lesson.rb +11 -11
  477. data/test/models/line_item.rb +3 -3
  478. data/test/models/liquid.rb +4 -4
  479. data/test/models/man.rb +11 -11
  480. data/test/models/matey.rb +4 -4
  481. data/test/models/member.rb +42 -41
  482. data/test/models/member_detail.rb +8 -7
  483. data/test/models/member_type.rb +3 -3
  484. data/test/models/membership.rb +35 -35
  485. data/test/models/mentor.rb +3 -0
  486. data/test/models/minimalistic.rb +2 -2
  487. data/test/models/minivan.rb +9 -9
  488. data/test/models/mixed_case_monkey.rb +3 -3
  489. data/test/models/mocktail_designer.rb +2 -0
  490. data/test/models/molecule.rb +6 -6
  491. data/test/models/movie.rb +5 -5
  492. data/test/models/node.rb +5 -0
  493. data/test/models/non_primary_key.rb +2 -0
  494. data/test/models/notification.rb +3 -0
  495. data/test/models/order.rb +4 -4
  496. data/test/models/organization.rb +14 -14
  497. data/test/models/other_dog.rb +5 -0
  498. data/test/models/owner.rb +37 -34
  499. data/test/models/parrot.rb +28 -29
  500. data/test/models/person.rb +142 -143
  501. data/test/models/personal_legacy_thing.rb +4 -4
  502. data/test/models/pet.rb +18 -15
  503. data/test/models/pet_treasure.rb +6 -0
  504. data/test/models/pirate.rb +92 -92
  505. data/test/models/possession.rb +3 -3
  506. data/test/models/post.rb +273 -264
  507. data/test/models/price_estimate.rb +4 -4
  508. data/test/models/professor.rb +5 -5
  509. data/test/models/project.rb +40 -31
  510. data/test/models/publisher.rb +2 -2
  511. data/test/models/publisher/article.rb +4 -4
  512. data/test/models/publisher/magazine.rb +3 -3
  513. data/test/models/randomly_named_c1.rb +1 -1
  514. data/test/models/rating.rb +4 -4
  515. data/test/models/reader.rb +23 -23
  516. data/test/models/recipe.rb +3 -0
  517. data/test/models/record.rb +2 -2
  518. data/test/models/reference.rb +22 -22
  519. data/test/models/reply.rb +61 -61
  520. data/test/models/ship.rb +39 -33
  521. data/test/models/ship_part.rb +8 -8
  522. data/test/models/shop.rb +17 -17
  523. data/test/models/shop_account.rb +6 -6
  524. data/test/models/speedometer.rb +6 -6
  525. data/test/models/sponsor.rb +7 -7
  526. data/test/models/string_key_object.rb +3 -3
  527. data/test/models/student.rb +4 -4
  528. data/test/models/subject.rb +16 -16
  529. data/test/models/subscriber.rb +8 -8
  530. data/test/models/subscription.rb +4 -4
  531. data/test/models/tag.rb +13 -7
  532. data/test/models/tagging.rb +13 -13
  533. data/test/models/task.rb +5 -5
  534. data/test/models/topic.rb +118 -124
  535. data/test/models/toy.rb +6 -6
  536. data/test/models/traffic_light.rb +4 -4
  537. data/test/models/treasure.rb +14 -14
  538. data/test/models/treaty.rb +7 -7
  539. data/test/models/tree.rb +3 -0
  540. data/test/models/tuning_peg.rb +4 -0
  541. data/test/models/tyre.rb +11 -11
  542. data/test/models/user.rb +14 -0
  543. data/test/models/uuid_child.rb +3 -3
  544. data/test/models/uuid_item.rb +6 -0
  545. data/test/models/uuid_parent.rb +3 -3
  546. data/test/models/vegetables.rb +24 -24
  547. data/test/models/vehicle.rb +6 -6
  548. data/test/models/vertex.rb +9 -9
  549. data/test/models/warehouse_thing.rb +5 -5
  550. data/test/models/wheel.rb +3 -3
  551. data/test/models/without_table.rb +3 -3
  552. data/test/models/zine.rb +3 -3
  553. data/test/schema/mysql2_specific_schema.rb +68 -58
  554. data/test/schema/oracle_specific_schema.rb +40 -43
  555. data/test/schema/postgresql_specific_schema.rb +114 -202
  556. data/test/schema/schema.rb +1057 -952
  557. data/test/schema/schema.rb.original +1057 -0
  558. data/test/schema/sqlite_specific_schema.rb +18 -22
  559. data/test/support/config.rb +43 -43
  560. data/test/support/connection.rb +23 -22
  561. data/test/support/connection_helper.rb +14 -14
  562. data/test/support/ddl_helper.rb +8 -8
  563. data/test/support/schema_dumping_helper.rb +20 -20
  564. data/test/support/yaml_compatibility_fixtures/rails_4_1.yml +22 -0
  565. data/test/support/yaml_compatibility_fixtures/rails_4_2_0.yml +182 -0
  566. metadata +129 -28
  567. data/lib/mswin32/rb19x/ibm_db.so +0 -0
  568. data/lib/mswin32/rb21x/i386/ibm_db.so +0 -0
  569. data/lib/mswin32/rb22x/i386/ibm_db.so +0 -0
  570. data/lib/mswin32/rb23x/i386/ibm_db.so +0 -0
  571. data/test/cases/associations/deprecated_counter_cache_on_has_many_through_test.rb +0 -26
  572. data/test/cases/attribute_methods/serialization_test.rb +0 -29
  573. data/test/cases/migration/change_schema_test - Copy.rb +0 -448
  574. data/test/cases/migration/foreign_key_test - Changed.rb +0 -325
  575. data/test/cases/migration/table_and_index_test.rb +0 -24
  576. data/test/cases/relation/where_test2.rb +0 -36
  577. data/test/cases/type/decimal_test.rb +0 -56
  578. data/test/cases/type/unsigned_integer_test.rb +0 -18
  579. data/test/cases/xml_serialization_test.rb +0 -457
  580. data/test/connections/native_ibm_db/connection.rb +0 -44
  581. data/test/fixtures/naked/csv/accounts.csv +0 -1
  582. data/test/schema/i5/ibm_db_specific_schema.rb +0 -137
  583. data/test/schema/ids/ibm_db_specific_schema.rb +0 -140
  584. data/test/schema/luw/ibm_db_specific_schema.rb +0 -137
  585. data/test/schema/mysql_specific_schema.rb +0 -70
  586. data/test/schema/zOS/ibm_db_specific_schema.rb +0 -208
@@ -1,54 +1,142 @@
1
- require 'cases/helper'
2
-
3
- class HotCompatibilityTest < ActiveRecord::TestCase
4
- self.use_transactional_fixtures = false
5
-
6
- setup do
7
- @klass = Class.new(ActiveRecord::Base) do
8
- connection.create_table :hot_compatibilities, force: true do |t|
9
- t.string :foo
10
- t.string :bar
11
- end
12
-
13
- def self.name; 'HotCompatibility'; end
14
- end
15
- end
16
-
17
- teardown do
18
- ActiveRecord::Base.connection.drop_table :hot_compatibilities
19
- end
20
-
21
- test "insert after remove_column" do
22
- # warm cache
23
- @klass.create!
24
-
25
- # we have 3 columns
26
- assert_equal 3, @klass.columns.length
27
-
28
- # remove one of them
29
- @klass.connection.remove_column :hot_compatibilities, :bar
30
-
31
- # we still have 3 columns in the cache
32
- assert_equal 3, @klass.columns.length
33
-
34
- # but we can successfully create a record so long as we don't
35
- # reference the removed column
36
- record = @klass.create! foo: 'foo'
37
- record.reload
38
- assert_equal 'foo', record.foo
39
- end
40
-
41
- test "update after remove_column" do
42
- record = @klass.create! foo: 'foo'
43
- assert_equal 3, @klass.columns.length
44
- @klass.connection.remove_column :hot_compatibilities, :bar
45
- assert_equal 3, @klass.columns.length
46
-
47
- record.reload
48
- assert_equal 'foo', record.foo
49
- record.foo = 'bar'
50
- record.save!
51
- record.reload
52
- assert_equal 'bar', record.foo
53
- end
54
- end
1
+ require 'cases/helper'
2
+ require 'support/connection_helper'
3
+
4
+ class HotCompatibilityTest < ActiveRecord::TestCase
5
+ self.use_transactional_tests = false
6
+ include ConnectionHelper
7
+
8
+ setup do
9
+ @klass = Class.new(ActiveRecord::Base) do
10
+ connection.create_table :hot_compatibilities, force: true do |t|
11
+ t.string :foo
12
+ t.string :bar
13
+ end
14
+
15
+ def self.name; 'HotCompatibility'; end
16
+ end
17
+ end
18
+
19
+ teardown do
20
+ ActiveRecord::Base.connection.drop_table :hot_compatibilities
21
+ end
22
+
23
+ test "insert after remove_column" do
24
+ # warm cache
25
+ @klass.create!
26
+
27
+ # we have 3 columns
28
+ assert_equal 3, @klass.columns.length
29
+
30
+ # remove one of them
31
+ @klass.connection.remove_column :hot_compatibilities, :bar
32
+
33
+ # we still have 3 columns in the cache
34
+ assert_equal 3, @klass.columns.length
35
+
36
+ # but we can successfully create a record so long as we don't
37
+ # reference the removed column
38
+ record = @klass.create! foo: 'foo'
39
+ record.reload
40
+ assert_equal 'foo', record.foo
41
+ end
42
+
43
+ test "update after remove_column" do
44
+ record = @klass.create! foo: 'foo'
45
+ assert_equal 3, @klass.columns.length
46
+ @klass.connection.remove_column :hot_compatibilities, :bar
47
+ assert_equal 3, @klass.columns.length
48
+
49
+ record.reload
50
+ assert_equal 'foo', record.foo
51
+ record.foo = 'bar'
52
+ record.save!
53
+ record.reload
54
+ assert_equal 'bar', record.foo
55
+ end
56
+
57
+ if current_adapter?(:PostgreSQLAdapter)
58
+ test "cleans up after prepared statement failure in a transaction" do
59
+ with_two_connections do |original_connection, ddl_connection|
60
+ record = @klass.create! bar: 'bar'
61
+
62
+ # prepare the reload statement in a transaction
63
+ @klass.transaction do
64
+ record.reload
65
+ end
66
+
67
+ assert get_prepared_statement_cache(@klass.connection).any?,
68
+ "expected prepared statement cache to have something in it"
69
+
70
+ # add a new column
71
+ ddl_connection.add_column :hot_compatibilities, :baz, :string
72
+
73
+ assert_raise(ActiveRecord::PreparedStatementCacheExpired) do
74
+ @klass.transaction do
75
+ record.reload
76
+ end
77
+ end
78
+
79
+ assert_empty get_prepared_statement_cache(@klass.connection),
80
+ "expected prepared statement cache to be empty but it wasn't"
81
+ end
82
+ end
83
+
84
+ test "cleans up after prepared statement failure in nested transactions" do
85
+ with_two_connections do |original_connection, ddl_connection|
86
+ record = @klass.create! bar: 'bar'
87
+
88
+ # prepare the reload statement in a transaction
89
+ @klass.transaction do
90
+ record.reload
91
+ end
92
+
93
+ assert get_prepared_statement_cache(@klass.connection).any?,
94
+ "expected prepared statement cache to have something in it"
95
+
96
+ # add a new column
97
+ ddl_connection.add_column :hot_compatibilities, :baz, :string
98
+
99
+ assert_raise(ActiveRecord::PreparedStatementCacheExpired) do
100
+ @klass.transaction do
101
+ @klass.transaction do
102
+ @klass.transaction do
103
+ record.reload
104
+ end
105
+ end
106
+ end
107
+ end
108
+
109
+ assert_empty get_prepared_statement_cache(@klass.connection),
110
+ "expected prepared statement cache to be empty but it wasn't"
111
+ end
112
+ end
113
+ end
114
+
115
+ private
116
+
117
+ def get_prepared_statement_cache(connection)
118
+ connection.instance_variable_get(:@statements)
119
+ .instance_variable_get(:@cache)[Process.pid]
120
+ end
121
+
122
+ # Rails will automatically clear the prepared statements on the connection
123
+ # that runs the migration, so we use two connections to simulate what would
124
+ # actually happen on a production system; we'd have one connection running the
125
+ # migration from the rake task ("ddl_connection" here), and we'd have another
126
+ # connection in a web worker.
127
+ def with_two_connections
128
+ run_without_connection do |original_connection|
129
+ ActiveRecord::Base.establish_connection(original_connection.merge(pool_size: 2))
130
+ begin
131
+ ddl_connection = ActiveRecord::Base.connection_pool.checkout
132
+ begin
133
+ yield original_connection, ddl_connection
134
+ ensure
135
+ ActiveRecord::Base.connection_pool.checkin ddl_connection
136
+ end
137
+ ensure
138
+ ActiveRecord::Base.clear_all_connections!
139
+ end
140
+ end
141
+ end
142
+ end
@@ -1,45 +1,45 @@
1
- require "cases/helper"
2
- require 'models/topic'
3
- require 'models/reply'
4
-
5
- class ActiveRecordI18nTests < ActiveRecord::TestCase
6
-
7
- def setup
8
- I18n.backend = I18n::Backend::Simple.new
9
- end
10
-
11
- def test_translated_model_attributes
12
- I18n.backend.store_translations 'en', :activerecord => {:attributes => {:topic => {:title => 'topic title attribute'} } }
13
- assert_equal 'topic title attribute', Topic.human_attribute_name('title')
14
- end
15
-
16
- def test_translated_model_attributes_with_symbols
17
- I18n.backend.store_translations 'en', :activerecord => {:attributes => {:topic => {:title => 'topic title attribute'} } }
18
- assert_equal 'topic title attribute', Topic.human_attribute_name(:title)
19
- end
20
-
21
- def test_translated_model_attributes_with_sti
22
- I18n.backend.store_translations 'en', :activerecord => {:attributes => {:reply => {:title => 'reply title attribute'} } }
23
- assert_equal 'reply title attribute', Reply.human_attribute_name('title')
24
- end
25
-
26
- def test_translated_model_attributes_with_sti_fallback
27
- I18n.backend.store_translations 'en', :activerecord => {:attributes => {:topic => {:title => 'topic title attribute'} } }
28
- assert_equal 'topic title attribute', Reply.human_attribute_name('title')
29
- end
30
-
31
- def test_translated_model_names
32
- I18n.backend.store_translations 'en', :activerecord => {:models => {:topic => 'topic model'} }
33
- assert_equal 'topic model', Topic.model_name.human
34
- end
35
-
36
- def test_translated_model_names_with_sti
37
- I18n.backend.store_translations 'en', :activerecord => {:models => {:reply => 'reply model'} }
38
- assert_equal 'reply model', Reply.model_name.human
39
- end
40
-
41
- def test_translated_model_names_with_sti_fallback
42
- I18n.backend.store_translations 'en', :activerecord => {:models => {:topic => 'topic model'} }
43
- assert_equal 'topic model', Reply.model_name.human
44
- end
45
- end
1
+ require "cases/helper"
2
+ require 'models/topic'
3
+ require 'models/reply'
4
+
5
+ class ActiveRecordI18nTests < ActiveRecord::TestCase
6
+
7
+ def setup
8
+ I18n.backend = I18n::Backend::Simple.new
9
+ end
10
+
11
+ def test_translated_model_attributes
12
+ I18n.backend.store_translations 'en', :activerecord => {:attributes => {:topic => {:title => 'topic title attribute'} } }
13
+ assert_equal 'topic title attribute', Topic.human_attribute_name('title')
14
+ end
15
+
16
+ def test_translated_model_attributes_with_symbols
17
+ I18n.backend.store_translations 'en', :activerecord => {:attributes => {:topic => {:title => 'topic title attribute'} } }
18
+ assert_equal 'topic title attribute', Topic.human_attribute_name(:title)
19
+ end
20
+
21
+ def test_translated_model_attributes_with_sti
22
+ I18n.backend.store_translations 'en', :activerecord => {:attributes => {:reply => {:title => 'reply title attribute'} } }
23
+ assert_equal 'reply title attribute', Reply.human_attribute_name('title')
24
+ end
25
+
26
+ def test_translated_model_attributes_with_sti_fallback
27
+ I18n.backend.store_translations 'en', :activerecord => {:attributes => {:topic => {:title => 'topic title attribute'} } }
28
+ assert_equal 'topic title attribute', Reply.human_attribute_name('title')
29
+ end
30
+
31
+ def test_translated_model_names
32
+ I18n.backend.store_translations 'en', :activerecord => {:models => {:topic => 'topic model'} }
33
+ assert_equal 'topic model', Topic.model_name.human
34
+ end
35
+
36
+ def test_translated_model_names_with_sti
37
+ I18n.backend.store_translations 'en', :activerecord => {:models => {:reply => 'reply model'} }
38
+ assert_equal 'reply model', Reply.model_name.human
39
+ end
40
+
41
+ def test_translated_model_names_with_sti_fallback
42
+ I18n.backend.store_translations 'en', :activerecord => {:models => {:topic => 'topic model'} }
43
+ assert_equal 'topic model', Reply.model_name.human
44
+ end
45
+ end
@@ -1,375 +1,606 @@
1
- require 'cases/helper'
2
- require 'models/company'
3
- require 'models/person'
4
- require 'models/post'
5
- require 'models/project'
6
- require 'models/subscriber'
7
- require 'models/vegetables'
8
- require 'models/shop'
9
-
10
- class InheritanceTest < ActiveRecord::TestCase
11
- fixtures :companies, :projects, :subscribers, :accounts, :vegetables
12
-
13
- def test_class_with_store_full_sti_class_returns_full_name
14
- old = ActiveRecord::Base.store_full_sti_class
15
- ActiveRecord::Base.store_full_sti_class = true
16
- assert_equal 'Namespaced::Company', Namespaced::Company.sti_name
17
- ensure
18
- ActiveRecord::Base.store_full_sti_class = old
19
- end
20
-
21
- def test_class_with_blank_sti_name
22
- company = Company.first
23
- company = company.dup
24
- company.extend(Module.new {
25
- def _read_attribute(name)
26
- return ' ' if name == 'type'
27
- super
28
- end
29
- })
30
- company.save!
31
- company = Company.all.to_a.find { |x| x.id == company.id }
32
- assert_equal ' ', company.type
33
- end
34
-
35
- def test_class_without_store_full_sti_class_returns_demodulized_name
36
- old = ActiveRecord::Base.store_full_sti_class
37
- ActiveRecord::Base.store_full_sti_class = false
38
- assert_equal 'Company', Namespaced::Company.sti_name
39
- ensure
40
- ActiveRecord::Base.store_full_sti_class = old
41
- end
42
-
43
- def test_should_store_demodulized_class_name_with_store_full_sti_class_option_disabled
44
- old = ActiveRecord::Base.store_full_sti_class
45
- ActiveRecord::Base.store_full_sti_class = false
46
- item = Namespaced::Company.new
47
- assert_equal 'Company', item[:type]
48
- ensure
49
- ActiveRecord::Base.store_full_sti_class = old
50
- end
51
-
52
- def test_should_store_full_class_name_with_store_full_sti_class_option_enabled
53
- old = ActiveRecord::Base.store_full_sti_class
54
- ActiveRecord::Base.store_full_sti_class = true
55
- item = Namespaced::Company.new
56
- assert_equal 'Namespaced::Company', item[:type]
57
- ensure
58
- ActiveRecord::Base.store_full_sti_class = old
59
- end
60
-
61
- def test_different_namespace_subclass_should_load_correctly_with_store_full_sti_class_option
62
- old = ActiveRecord::Base.store_full_sti_class
63
- ActiveRecord::Base.store_full_sti_class = true
64
- item = Namespaced::Company.create :name => "Wolverine 2"
65
- assert_not_nil Company.find(item.id)
66
- assert_not_nil Namespaced::Company.find(item.id)
67
- ensure
68
- ActiveRecord::Base.store_full_sti_class = old
69
- end
70
-
71
- def test_company_descends_from_active_record
72
- assert !ActiveRecord::Base.descends_from_active_record?
73
- assert AbstractCompany.descends_from_active_record?, 'AbstractCompany should descend from ActiveRecord::Base'
74
- assert Company.descends_from_active_record?, 'Company should descend from ActiveRecord::Base'
75
- assert !Class.new(Company).descends_from_active_record?, 'Company subclass should not descend from ActiveRecord::Base'
76
- end
77
-
78
- def test_inheritance_base_class
79
- assert_equal Post, Post.base_class
80
- assert_equal Post, SpecialPost.base_class
81
- assert_equal Post, StiPost.base_class
82
- assert_equal SubStiPost, SubStiPost.base_class
83
- end
84
-
85
- def test_abstract_inheritance_base_class
86
- assert_equal LoosePerson, LoosePerson.base_class
87
- assert_equal LooseDescendant, LooseDescendant.base_class
88
- assert_equal TightPerson, TightPerson.base_class
89
- assert_equal TightPerson, TightDescendant.base_class
90
- end
91
-
92
- def test_base_class_activerecord_error
93
- klass = Class.new { include ActiveRecord::Inheritance }
94
- assert_raise(ActiveRecord::ActiveRecordError) { klass.base_class }
95
- end
96
-
97
- def test_a_bad_type_column
98
- Company.connection.insert "INSERT INTO companies (id, #{QUOTED_TYPE}, name) VALUES(100, 'bad_class!', 'Not happening')"
99
-
100
- assert_raise(ActiveRecord::SubclassNotFound) { Company.find(100) }
101
- end
102
-
103
- def test_inheritance_find
104
- assert_kind_of Firm, Company.find(1), "37signals should be a firm"
105
- assert_kind_of Firm, Firm.find(1), "37signals should be a firm"
106
- assert_kind_of Client, Company.find(2), "Summit should be a client"
107
- assert_kind_of Client, Client.find(2), "Summit should be a client"
108
- end
109
-
110
- def test_alt_inheritance_find
111
- assert_kind_of Cucumber, Vegetable.find(1)
112
- assert_kind_of Cucumber, Cucumber.find(1)
113
- assert_kind_of Cabbage, Vegetable.find(2)
114
- assert_kind_of Cabbage, Cabbage.find(2)
115
- end
116
-
117
- def test_alt_becomes_works_with_sti
118
- vegetable = Vegetable.find(1)
119
- assert_kind_of Vegetable, vegetable
120
- cabbage = vegetable.becomes(Cabbage)
121
- assert_kind_of Cabbage, cabbage
122
- end
123
-
124
- def test_becomes_and_change_tracking_for_inheritance_columns
125
- cucumber = Vegetable.find(1)
126
- cabbage = cucumber.becomes!(Cabbage)
127
- assert_equal ['Cucumber', 'Cabbage'], cabbage.custom_type_change
128
- end
129
-
130
- def test_alt_becomes_bang_resets_inheritance_type_column
131
- vegetable = Vegetable.create!(name: "Red Pepper")
132
- assert_nil vegetable.custom_type
133
-
134
- cabbage = vegetable.becomes!(Cabbage)
135
- assert_equal "Cabbage", cabbage.custom_type
136
-
137
- vegetable = cabbage.becomes!(Vegetable)
138
- assert_nil cabbage.custom_type
139
- end
140
-
141
- def test_inheritance_find_all
142
- companies = Company.all.merge!(:order => 'id').to_a
143
- assert_kind_of Firm, companies[0], "37signals should be a firm"
144
- assert_kind_of Client, companies[1], "Summit should be a client"
145
- end
146
-
147
- def test_alt_inheritance_find_all
148
- companies = Vegetable.all.merge!(:order => 'id').to_a
149
- assert_kind_of Cucumber, companies[0]
150
- assert_kind_of Cabbage, companies[1]
151
- end
152
-
153
- def test_inheritance_save
154
- firm = Firm.new
155
- firm.name = "Next Angle"
156
- firm.save
157
-
158
- next_angle = Company.find(firm.id)
159
- assert_kind_of Firm, next_angle, "Next Angle should be a firm"
160
- end
161
-
162
- def test_alt_inheritance_save
163
- cabbage = Cabbage.new(:name => 'Savoy')
164
- cabbage.save!
165
-
166
- savoy = Vegetable.find(cabbage.id)
167
- assert_kind_of Cabbage, savoy
168
- end
169
-
170
- def test_inheritance_new_with_default_class
171
- company = Company.new
172
- assert_equal Company, company.class
173
- end
174
-
175
- def test_inheritance_new_with_base_class
176
- company = Company.new(:type => 'Company')
177
- assert_equal Company, company.class
178
- end
179
-
180
- def test_inheritance_new_with_subclass
181
- firm = Company.new(:type => 'Firm')
182
- assert_equal Firm, firm.class
183
- end
184
-
185
- def test_new_with_abstract_class
186
- e = assert_raises(NotImplementedError) do
187
- AbstractCompany.new
188
- end
189
- assert_equal("AbstractCompany is an abstract class and cannot be instantiated.", e.message)
190
- end
191
-
192
- def test_new_with_ar_base
193
- e = assert_raises(NotImplementedError) do
194
- ActiveRecord::Base.new
195
- end
196
- assert_equal("ActiveRecord::Base is an abstract class and cannot be instantiated.", e.message)
197
- end
198
-
199
- def test_new_with_invalid_type
200
- assert_raise(ActiveRecord::SubclassNotFound) { Company.new(:type => 'InvalidType') }
201
- end
202
-
203
- def test_new_with_unrelated_type
204
- assert_raise(ActiveRecord::SubclassNotFound) { Company.new(:type => 'Account') }
205
- end
206
-
207
- def test_new_with_complex_inheritance
208
- assert_nothing_raised { Client.new(type: 'VerySpecialClient') }
209
- end
210
-
211
- def test_new_with_autoload_paths
212
- path = File.expand_path('../../models/autoloadable', __FILE__)
213
- ActiveSupport::Dependencies.autoload_paths << path
214
-
215
- firm = Company.new(:type => 'ExtraFirm')
216
- assert_equal ExtraFirm, firm.class
217
- ensure
218
- ActiveSupport::Dependencies.autoload_paths.reject! { |p| p == path }
219
- ActiveSupport::Dependencies.clear
220
- end
221
-
222
- def test_inheritance_condition
223
- assert_equal 11, Company.count
224
- assert_equal 2, Firm.count
225
- assert_equal 5, Client.count
226
- end
227
-
228
- def test_alt_inheritance_condition
229
- assert_equal 4, Vegetable.count
230
- assert_equal 1, Cucumber.count
231
- assert_equal 3, Cabbage.count
232
- end
233
-
234
- def test_finding_incorrect_type_data
235
- assert_raise(ActiveRecord::RecordNotFound) { Firm.find(2) }
236
- assert_nothing_raised { Firm.find(1) }
237
- end
238
-
239
- def test_alt_finding_incorrect_type_data
240
- assert_raise(ActiveRecord::RecordNotFound) { Cucumber.find(2) }
241
- assert_nothing_raised { Cucumber.find(1) }
242
- end
243
-
244
- def test_update_all_within_inheritance
245
- Client.update_all "name = 'I am a client'"
246
- assert_equal "I am a client", Client.first.name
247
- # Order by added as otherwise Oracle tests were failing because of different order of results
248
- assert_equal "37signals", Firm.all.merge!(:order => "id").to_a.first.name
249
- end
250
-
251
- def test_alt_update_all_within_inheritance
252
- Cabbage.update_all "name = 'the cabbage'"
253
- assert_equal "the cabbage", Cabbage.first.name
254
- assert_equal ["my cucumber"], Cucumber.all.map(&:name).uniq
255
- end
256
-
257
- def test_destroy_all_within_inheritance
258
- Client.destroy_all
259
- assert_equal 0, Client.count
260
- assert_equal 2, Firm.count
261
- end
262
-
263
- def test_alt_destroy_all_within_inheritance
264
- Cabbage.destroy_all
265
- assert_equal 0, Cabbage.count
266
- assert_equal 1, Cucumber.count
267
- end
268
-
269
- def test_find_first_within_inheritance
270
- assert_kind_of Firm, Company.all.merge!(:where => "name = '37signals'").first
271
- assert_kind_of Firm, Firm.all.merge!(:where => "name = '37signals'").first
272
- assert_nil Client.all.merge!(:where => "name = '37signals'").first
273
- end
274
-
275
- def test_alt_find_first_within_inheritance
276
- assert_kind_of Cabbage, Vegetable.all.merge!(:where => "name = 'his cabbage'").first
277
- assert_kind_of Cabbage, Cabbage.all.merge!(:where => "name = 'his cabbage'").first
278
- assert_nil Cucumber.all.merge!(:where => "name = 'his cabbage'").first
279
- end
280
-
281
- def test_complex_inheritance
282
- very_special_client = VerySpecialClient.create("name" => "veryspecial")
283
- assert_equal very_special_client, VerySpecialClient.where("name = 'veryspecial'").first
284
- assert_equal very_special_client, SpecialClient.all.merge!(:where => "name = 'veryspecial'").first
285
- assert_equal very_special_client, Company.all.merge!(:where => "name = 'veryspecial'").first
286
- assert_equal very_special_client, Client.all.merge!(:where => "name = 'veryspecial'").first
287
- assert_equal 1, Client.all.merge!(:where => "name = 'Summit'").to_a.size
288
- assert_equal very_special_client, Client.find(very_special_client.id)
289
- end
290
-
291
- def test_alt_complex_inheritance
292
- king_cole = KingCole.create("name" => "uniform heads")
293
- assert_equal king_cole, KingCole.where("name = 'uniform heads'").first
294
- assert_equal king_cole, GreenCabbage.all.merge!(:where => "name = 'uniform heads'").first
295
- assert_equal king_cole, Cabbage.all.merge!(:where => "name = 'uniform heads'").first
296
- assert_equal king_cole, Vegetable.all.merge!(:where => "name = 'uniform heads'").first
297
- assert_equal 1, Cabbage.all.merge!(:where => "name = 'his cabbage'").to_a.size
298
- assert_equal king_cole, Cabbage.find(king_cole.id)
299
- end
300
-
301
- def test_eager_load_belongs_to_something_inherited
302
- account = Account.all.merge!(:includes => :firm).find(1)
303
- assert account.association(:firm).loaded?, "association was not eager loaded"
304
- end
305
-
306
- def test_alt_eager_loading
307
- cabbage = RedCabbage.all.merge!(:includes => :seller).find(4)
308
- assert cabbage.association(:seller).loaded?, "association was not eager loaded"
309
- end
310
-
311
- def test_eager_load_belongs_to_primary_key_quoting
312
- con = Account.connection
313
- assert_sql(/#{con.quote_table_name('companies')}.#{con.quote_column_name('id')} IN \(1\)/) do
314
- Account.all.merge!(:includes => :firm).find(1)
315
- end
316
- end
317
-
318
- def test_inherits_custom_primary_key
319
- assert_equal Subscriber.primary_key, SpecialSubscriber.primary_key
320
- end
321
-
322
- def test_inheritance_without_mapping
323
- assert_kind_of SpecialSubscriber, SpecialSubscriber.find("webster132")
324
- assert_nothing_raised { s = SpecialSubscriber.new("name" => "And breaaaaathe!"); s.id = 'roger'; s.save }
325
- end
326
-
327
- def test_scope_inherited_properly
328
- assert_nothing_raised { Company.of_first_firm }
329
- assert_nothing_raised { Client.of_first_firm }
330
- end
331
- end
332
-
333
- class InheritanceComputeTypeTest < ActiveRecord::TestCase
334
- fixtures :companies
335
-
336
- def setup
337
- ActiveSupport::Dependencies.log_activity = true
338
- end
339
-
340
- teardown do
341
- ActiveSupport::Dependencies.log_activity = false
342
- self.class.const_remove :FirmOnTheFly rescue nil
343
- Firm.const_remove :FirmOnTheFly rescue nil
344
- end
345
-
346
- def test_instantiation_doesnt_try_to_require_corresponding_file
347
- ActiveRecord::Base.store_full_sti_class = false
348
- foo = Firm.first.clone
349
- foo.type = 'FirmOnTheFly'
350
- foo.save!
351
-
352
- # Should fail without FirmOnTheFly in the type condition.
353
- assert_raise(ActiveRecord::RecordNotFound) { Firm.find(foo.id) }
354
-
355
- # Nest FirmOnTheFly in the test case where Dependencies won't see it.
356
- self.class.const_set :FirmOnTheFly, Class.new(Firm)
357
- assert_raise(ActiveRecord::SubclassNotFound) { Firm.find(foo.id) }
358
-
359
- # Nest FirmOnTheFly in Firm where Dependencies will see it.
360
- # This is analogous to nesting models in a migration.
361
- Firm.const_set :FirmOnTheFly, Class.new(Firm)
362
-
363
- # And instantiate will find the existing constant rather than trying
364
- # to require firm_on_the_fly.
365
- assert_nothing_raised { assert_kind_of Firm::FirmOnTheFly, Firm.find(foo.id) }
366
- ensure
367
- ActiveRecord::Base.store_full_sti_class = true
368
- end
369
-
370
- def test_sti_type_from_attributes_disabled_in_non_sti_class
371
- phone = Shop::Product::Type.new(name: 'Phone')
372
- product = Shop::Product.new(:type => phone)
373
- assert product.save
374
- end
375
- end
1
+ require 'cases/helper'
2
+ require 'models/author'
3
+ require 'models/company'
4
+ require 'models/membership'
5
+ require 'models/person'
6
+ require 'models/post'
7
+ require 'models/project'
8
+ require 'models/subscriber'
9
+ require 'models/vegetables'
10
+ require 'models/shop'
11
+ require 'models/sponsor'
12
+
13
+ module InheritanceTestHelper
14
+ def with_store_full_sti_class(&block)
15
+ assign_store_full_sti_class true, &block
16
+ end
17
+
18
+ def without_store_full_sti_class(&block)
19
+ assign_store_full_sti_class false, &block
20
+ end
21
+
22
+ def assign_store_full_sti_class(flag)
23
+ old_store_full_sti_class = ActiveRecord::Base.store_full_sti_class
24
+ ActiveRecord::Base.store_full_sti_class = flag
25
+ yield
26
+ ensure
27
+ ActiveRecord::Base.store_full_sti_class = old_store_full_sti_class
28
+ end
29
+ end
30
+
31
+ class InheritanceTest < ActiveRecord::TestCase
32
+ include InheritanceTestHelper
33
+ fixtures :companies, :projects, :subscribers, :accounts, :vegetables, :memberships
34
+
35
+ def test_class_with_store_full_sti_class_returns_full_name
36
+ with_store_full_sti_class do
37
+ assert_equal 'Namespaced::Company', Namespaced::Company.sti_name
38
+ end
39
+ end
40
+
41
+ def test_class_with_blank_sti_name
42
+ company = Company.first
43
+ company = company.dup
44
+ company.extend(Module.new {
45
+ def _read_attribute(name)
46
+ return ' ' if name == 'type'
47
+ super
48
+ end
49
+ })
50
+ company.save!
51
+ company = Company.all.to_a.find { |x| x.id == company.id }
52
+ assert_equal ' ', company.type
53
+ end
54
+
55
+ def test_class_without_store_full_sti_class_returns_demodulized_name
56
+ without_store_full_sti_class do
57
+ assert_equal 'Company', Namespaced::Company.sti_name
58
+ end
59
+ end
60
+
61
+ def test_compute_type_success
62
+ assert_equal Author, ActiveRecord::Base.send(:compute_type, 'Author')
63
+ end
64
+
65
+ def test_compute_type_nonexistent_constant
66
+ e = assert_raises NameError do
67
+ ActiveRecord::Base.send :compute_type, 'NonexistentModel'
68
+ end
69
+ assert_equal 'uninitialized constant ActiveRecord::Base::NonexistentModel', e.message
70
+ assert_equal 'ActiveRecord::Base::NonexistentModel', e.name
71
+ end
72
+
73
+ def test_compute_type_no_method_error
74
+ ActiveSupport::Dependencies.stub(:safe_constantize, proc{ raise NoMethodError }) do
75
+ assert_raises NoMethodError do
76
+ ActiveRecord::Base.send :compute_type, 'InvalidModel'
77
+ end
78
+ end
79
+ end
80
+
81
+ def test_compute_type_on_undefined_method
82
+ error = nil
83
+ begin
84
+ Class.new(Author) do
85
+ alias_method :foo, :bar
86
+ end
87
+ rescue => e
88
+ error = e
89
+ end
90
+
91
+ ActiveSupport::Dependencies.stub(:safe_constantize, proc{ raise e }) do
92
+
93
+ exception = assert_raises NameError do
94
+ ActiveRecord::Base.send :compute_type, 'InvalidModel'
95
+ end
96
+ assert_equal error.message, exception.message
97
+ end
98
+ end
99
+
100
+ def test_compute_type_argument_error
101
+ ActiveSupport::Dependencies.stub(:safe_constantize, proc{ raise ArgumentError }) do
102
+ assert_raises ArgumentError do
103
+ ActiveRecord::Base.send :compute_type, 'InvalidModel'
104
+ end
105
+ end
106
+ end
107
+
108
+ def test_should_store_demodulized_class_name_with_store_full_sti_class_option_disabled
109
+ without_store_full_sti_class do
110
+ item = Namespaced::Company.new
111
+ assert_equal 'Company', item[:type]
112
+ end
113
+ end
114
+
115
+ def test_should_store_full_class_name_with_store_full_sti_class_option_enabled
116
+ with_store_full_sti_class do
117
+ item = Namespaced::Company.new
118
+ assert_equal 'Namespaced::Company', item[:type]
119
+ end
120
+ end
121
+
122
+ def test_different_namespace_subclass_should_load_correctly_with_store_full_sti_class_option
123
+ with_store_full_sti_class do
124
+ item = Namespaced::Company.create name: "Wolverine 2"
125
+ assert_not_nil Company.find(item.id)
126
+ assert_not_nil Namespaced::Company.find(item.id)
127
+ end
128
+ end
129
+
130
+ def test_descends_from_active_record
131
+ assert !ActiveRecord::Base.descends_from_active_record?
132
+
133
+ # Abstract subclass of AR::Base.
134
+ assert LoosePerson.descends_from_active_record?
135
+
136
+ # Concrete subclass of an abstract class.
137
+ assert LooseDescendant.descends_from_active_record?
138
+
139
+ # Concrete subclass of AR::Base.
140
+ assert TightPerson.descends_from_active_record?
141
+
142
+ # Concrete subclass of a concrete class but has no type column.
143
+ assert TightDescendant.descends_from_active_record?
144
+
145
+ # Concrete subclass of AR::Base.
146
+ assert Post.descends_from_active_record?
147
+
148
+ # Abstract subclass of a concrete class which has a type column.
149
+ # This is pathological, as you'll never have Sub < Abstract < Concrete.
150
+ assert !StiPost.descends_from_active_record?
151
+
152
+ # Concrete subclasses an abstract class which has a type column.
153
+ assert !SubStiPost.descends_from_active_record?
154
+ end
155
+
156
+ def test_company_descends_from_active_record
157
+ assert !ActiveRecord::Base.descends_from_active_record?
158
+ assert AbstractCompany.descends_from_active_record?, 'AbstractCompany should descend from ActiveRecord::Base'
159
+ assert Company.descends_from_active_record?, 'Company should descend from ActiveRecord::Base'
160
+ assert !Class.new(Company).descends_from_active_record?, 'Company subclass should not descend from ActiveRecord::Base'
161
+ end
162
+
163
+ def test_abstract_class
164
+ assert !ActiveRecord::Base.abstract_class?
165
+ assert LoosePerson.abstract_class?
166
+ assert !LooseDescendant.abstract_class?
167
+ end
168
+
169
+ def test_inheritance_base_class
170
+ assert_equal Post, Post.base_class
171
+ assert_equal Post, SpecialPost.base_class
172
+ assert_equal Post, StiPost.base_class
173
+ assert_equal SubStiPost, SubStiPost.base_class
174
+ end
175
+
176
+ def test_abstract_inheritance_base_class
177
+ assert_equal LoosePerson, LoosePerson.base_class
178
+ assert_equal LooseDescendant, LooseDescendant.base_class
179
+ assert_equal TightPerson, TightPerson.base_class
180
+ assert_equal TightPerson, TightDescendant.base_class
181
+ end
182
+
183
+ def test_base_class_activerecord_error
184
+ klass = Class.new { include ActiveRecord::Inheritance }
185
+ assert_raise(ActiveRecord::ActiveRecordError) { klass.base_class }
186
+ end
187
+
188
+ def test_a_bad_type_column
189
+ Company.connection.insert "INSERT INTO companies (id, #{QUOTED_TYPE}, name) VALUES(100, 'bad_class!', 'Not happening')"
190
+
191
+ assert_raise(ActiveRecord::SubclassNotFound) { Company.find(100) }
192
+ end
193
+
194
+ def test_inheritance_find
195
+ assert_kind_of Firm, Company.find(1), "37signals should be a firm"
196
+ assert_kind_of Firm, Firm.find(1), "37signals should be a firm"
197
+ assert_kind_of Client, Company.find(2), "Summit should be a client"
198
+ assert_kind_of Client, Client.find(2), "Summit should be a client"
199
+ end
200
+
201
+ def test_alt_inheritance_find
202
+ assert_kind_of Cucumber, Vegetable.find(1)
203
+ assert_kind_of Cucumber, Cucumber.find(1)
204
+ assert_kind_of Cabbage, Vegetable.find(2)
205
+ assert_kind_of Cabbage, Cabbage.find(2)
206
+ end
207
+
208
+ def test_alt_becomes_works_with_sti
209
+ vegetable = Vegetable.find(1)
210
+ assert_kind_of Vegetable, vegetable
211
+ cabbage = vegetable.becomes(Cabbage)
212
+ assert_kind_of Cabbage, cabbage
213
+ end
214
+
215
+ def test_becomes_and_change_tracking_for_inheritance_columns
216
+ cucumber = Vegetable.find(1)
217
+ cabbage = cucumber.becomes!(Cabbage)
218
+ assert_equal ['Cucumber', 'Cabbage'], cabbage.custom_type_change
219
+ end
220
+
221
+ def test_alt_becomes_bang_resets_inheritance_type_column
222
+ vegetable = Vegetable.create!(name: "Red Pepper")
223
+ assert_nil vegetable.custom_type
224
+
225
+ cabbage = vegetable.becomes!(Cabbage)
226
+ assert_equal "Cabbage", cabbage.custom_type
227
+
228
+ vegetable = cabbage.becomes!(Vegetable)
229
+ assert_nil cabbage.custom_type
230
+ end
231
+
232
+ def test_inheritance_find_all
233
+ companies = Company.all.merge!(:order => 'id').to_a
234
+ assert_kind_of Firm, companies[0], "37signals should be a firm"
235
+ assert_kind_of Client, companies[1], "Summit should be a client"
236
+ end
237
+
238
+ def test_alt_inheritance_find_all
239
+ companies = Vegetable.all.merge!(:order => 'id').to_a
240
+ assert_kind_of Cucumber, companies[0]
241
+ assert_kind_of Cabbage, companies[1]
242
+ end
243
+
244
+ def test_inheritance_save
245
+ firm = Firm.new
246
+ firm.name = "Next Angle"
247
+ firm.save
248
+
249
+ next_angle = Company.find(firm.id)
250
+ assert_kind_of Firm, next_angle, "Next Angle should be a firm"
251
+ end
252
+
253
+ def test_alt_inheritance_save
254
+ cabbage = Cabbage.new(:name => 'Savoy')
255
+ cabbage.save!
256
+
257
+ savoy = Vegetable.find(cabbage.id)
258
+ assert_kind_of Cabbage, savoy
259
+ end
260
+
261
+ def test_inheritance_new_with_default_class
262
+ company = Company.new
263
+ assert_equal Company, company.class
264
+ end
265
+
266
+ def test_inheritance_new_with_base_class
267
+ company = Company.new(:type => 'Company')
268
+ assert_equal Company, company.class
269
+ end
270
+
271
+ def test_inheritance_new_with_subclass
272
+ firm = Company.new(:type => 'Firm')
273
+ assert_equal Firm, firm.class
274
+ end
275
+
276
+ def test_new_with_abstract_class
277
+ e = assert_raises(NotImplementedError) do
278
+ AbstractCompany.new
279
+ end
280
+ assert_equal("AbstractCompany is an abstract class and cannot be instantiated.", e.message)
281
+ end
282
+
283
+ def test_new_with_ar_base
284
+ e = assert_raises(NotImplementedError) do
285
+ ActiveRecord::Base.new
286
+ end
287
+ assert_equal("ActiveRecord::Base is an abstract class and cannot be instantiated.", e.message)
288
+ end
289
+
290
+ def test_new_with_invalid_type
291
+ assert_raise(ActiveRecord::SubclassNotFound) { Company.new(:type => 'InvalidType') }
292
+ end
293
+
294
+ def test_new_with_unrelated_type
295
+ assert_raise(ActiveRecord::SubclassNotFound) { Company.new(:type => 'Account') }
296
+ end
297
+
298
+ def test_new_with_unrelated_namespaced_type
299
+ without_store_full_sti_class do
300
+ e = assert_raises ActiveRecord::SubclassNotFound do
301
+ Namespaced::Company.new(type: 'Firm')
302
+ end
303
+
304
+ assert_equal "Invalid single-table inheritance type: Namespaced::Firm is not a subclass of Namespaced::Company", e.message
305
+ end
306
+ end
307
+
308
+ def test_new_with_complex_inheritance
309
+ assert_nothing_raised { Client.new(type: 'VerySpecialClient') }
310
+ end
311
+
312
+ def test_new_without_storing_full_sti_class
313
+ without_store_full_sti_class do
314
+ item = Company.new(type: 'SpecialCo')
315
+ assert_instance_of Company::SpecialCo, item
316
+ end
317
+ end
318
+
319
+ def test_new_with_autoload_paths
320
+ path = File.expand_path('../../models/autoloadable', __FILE__)
321
+ ActiveSupport::Dependencies.autoload_paths << path
322
+
323
+ firm = Company.new(:type => 'ExtraFirm')
324
+ assert_equal ExtraFirm, firm.class
325
+ ensure
326
+ ActiveSupport::Dependencies.autoload_paths.reject! { |p| p == path }
327
+ ActiveSupport::Dependencies.clear
328
+ end
329
+
330
+ def test_inheritance_condition
331
+ assert_equal 11, Company.count
332
+ assert_equal 2, Firm.count
333
+ assert_equal 5, Client.count
334
+ end
335
+
336
+ def test_alt_inheritance_condition
337
+ assert_equal 4, Vegetable.count
338
+ assert_equal 1, Cucumber.count
339
+ assert_equal 3, Cabbage.count
340
+ end
341
+
342
+ def test_finding_incorrect_type_data
343
+ assert_raise(ActiveRecord::RecordNotFound) { Firm.find(2) }
344
+ assert_nothing_raised { Firm.find(1) }
345
+ end
346
+
347
+ def test_alt_finding_incorrect_type_data
348
+ assert_raise(ActiveRecord::RecordNotFound) { Cucumber.find(2) }
349
+ assert_nothing_raised { Cucumber.find(1) }
350
+ end
351
+
352
+ def test_update_all_within_inheritance
353
+ Client.update_all "name = 'I am a client'"
354
+ assert_equal "I am a client", Client.first.name
355
+ # Order by added as otherwise Oracle tests were failing because of different order of results
356
+ assert_equal "37signals", Firm.all.merge!(:order => "id").to_a.first.name
357
+ end
358
+
359
+ def test_alt_update_all_within_inheritance
360
+ Cabbage.update_all "name = 'the cabbage'"
361
+ assert_equal "the cabbage", Cabbage.first.name
362
+ assert_equal ["my cucumber"], Cucumber.all.map(&:name).uniq
363
+ end
364
+
365
+ def test_destroy_all_within_inheritance
366
+ Client.destroy_all
367
+ assert_equal 0, Client.count
368
+ assert_equal 2, Firm.count
369
+ end
370
+
371
+ def test_alt_destroy_all_within_inheritance
372
+ Cabbage.destroy_all
373
+ assert_equal 0, Cabbage.count
374
+ assert_equal 1, Cucumber.count
375
+ end
376
+
377
+ def test_find_first_within_inheritance
378
+ assert_kind_of Firm, Company.all.merge!(:where => "name = '37signals'").first
379
+ assert_kind_of Firm, Firm.all.merge!(:where => "name = '37signals'").first
380
+ assert_nil Client.all.merge!(:where => "name = '37signals'").first
381
+ end
382
+
383
+ def test_alt_find_first_within_inheritance
384
+ assert_kind_of Cabbage, Vegetable.all.merge!(:where => "name = 'his cabbage'").first
385
+ assert_kind_of Cabbage, Cabbage.all.merge!(:where => "name = 'his cabbage'").first
386
+ assert_nil Cucumber.all.merge!(:where => "name = 'his cabbage'").first
387
+ end
388
+
389
+ def test_complex_inheritance
390
+ very_special_client = VerySpecialClient.create("name" => "veryspecial")
391
+ assert_equal very_special_client, VerySpecialClient.where("name = 'veryspecial'").first
392
+ assert_equal very_special_client, SpecialClient.all.merge!(:where => "name = 'veryspecial'").first
393
+ assert_equal very_special_client, Company.all.merge!(:where => "name = 'veryspecial'").first
394
+ assert_equal very_special_client, Client.all.merge!(:where => "name = 'veryspecial'").first
395
+ assert_equal 1, Client.all.merge!(:where => "name = 'Summit'").to_a.size
396
+ assert_equal very_special_client, Client.find(very_special_client.id)
397
+ end
398
+
399
+ def test_alt_complex_inheritance
400
+ king_cole = KingCole.create("name" => "uniform heads")
401
+ assert_equal king_cole, KingCole.where("name = 'uniform heads'").first
402
+ assert_equal king_cole, GreenCabbage.all.merge!(:where => "name = 'uniform heads'").first
403
+ assert_equal king_cole, Cabbage.all.merge!(:where => "name = 'uniform heads'").first
404
+ assert_equal king_cole, Vegetable.all.merge!(:where => "name = 'uniform heads'").first
405
+ assert_equal 1, Cabbage.all.merge!(:where => "name = 'his cabbage'").to_a.size
406
+ assert_equal king_cole, Cabbage.find(king_cole.id)
407
+ end
408
+
409
+ def test_eager_load_belongs_to_something_inherited
410
+ account = Account.all.merge!(:includes => :firm).find(1)
411
+ assert account.association(:firm).loaded?, "association was not eager loaded"
412
+ end
413
+
414
+ def test_alt_eager_loading
415
+ cabbage = RedCabbage.all.merge!(:includes => :seller).find(4)
416
+ assert cabbage.association(:seller).loaded?, "association was not eager loaded"
417
+ end
418
+
419
+ def test_eager_load_belongs_to_primary_key_quoting
420
+ con = Account.connection
421
+ assert_sql(/#{con.quote_table_name('companies')}.#{con.quote_column_name('id')} = 1/) do
422
+ Account.all.merge!(:includes => :firm).find(1)
423
+ end
424
+ end
425
+
426
+ def test_inherits_custom_primary_key
427
+ assert_equal Subscriber.primary_key, SpecialSubscriber.primary_key
428
+ end
429
+
430
+ def test_inheritance_without_mapping
431
+ assert_kind_of SpecialSubscriber, SpecialSubscriber.find("webster132")
432
+ assert_nothing_raised { s = SpecialSubscriber.new("name" => "And breaaaaathe!"); s.id = 'roger'; s.save }
433
+ end
434
+
435
+ def test_scope_inherited_properly
436
+ assert_nothing_raised { Company.of_first_firm }
437
+ assert_nothing_raised { Client.of_first_firm }
438
+ end
439
+
440
+ def test_inheritance_with_default_scope
441
+ assert_equal 1, SelectedMembership.count(:all)
442
+ end
443
+ end
444
+
445
+ class InheritanceComputeTypeTest < ActiveRecord::TestCase
446
+ include InheritanceTestHelper
447
+ fixtures :companies
448
+
449
+ teardown do
450
+ self.class.const_remove :FirmOnTheFly rescue nil
451
+ Firm.const_remove :FirmOnTheFly rescue nil
452
+ end
453
+
454
+ def test_instantiation_doesnt_try_to_require_corresponding_file
455
+ without_store_full_sti_class do
456
+ foo = Firm.first.clone
457
+ foo.type = 'FirmOnTheFly'
458
+ foo.save!
459
+
460
+ # Should fail without FirmOnTheFly in the type condition.
461
+ assert_raise(ActiveRecord::RecordNotFound) { Firm.find(foo.id) }
462
+
463
+ # Nest FirmOnTheFly in the test case where Dependencies won't see it.
464
+ self.class.const_set :FirmOnTheFly, Class.new(Firm)
465
+ assert_raise(ActiveRecord::SubclassNotFound) { Firm.find(foo.id) }
466
+
467
+ # Nest FirmOnTheFly in Firm where Dependencies will see it.
468
+ # This is analogous to nesting models in a migration.
469
+ Firm.const_set :FirmOnTheFly, Class.new(Firm)
470
+
471
+ # And instantiate will find the existing constant rather than trying
472
+ # to require firm_on_the_fly.
473
+ assert_nothing_raised { assert_kind_of Firm::FirmOnTheFly, Firm.find(foo.id) }
474
+ end
475
+ end
476
+
477
+ def test_sti_type_from_attributes_disabled_in_non_sti_class
478
+ phone = Shop::Product::Type.new(name: 'Phone')
479
+ product = Shop::Product.new(:type => phone)
480
+ assert product.save
481
+ end
482
+
483
+ def test_inheritance_new_with_subclass_as_default
484
+ original_type = Company.columns_hash["type"].default
485
+ ActiveRecord::Base.connection.change_column_default :companies, :type, 'Firm'
486
+ Company.reset_column_information
487
+
488
+ firm = Company.new # without arguments
489
+ assert_equal 'Firm', firm.type
490
+ assert_instance_of Firm, firm
491
+
492
+ firm = Company.new(firm_name: 'Shri Hans Plastic') # with arguments
493
+ assert_equal 'Firm', firm.type
494
+ assert_instance_of Firm, firm
495
+
496
+ client = Client.new
497
+ assert_equal 'Client', client.type
498
+ assert_instance_of Client, client
499
+
500
+ firm = Company.new(type: 'Client') # overwrite the default type
501
+ assert_equal 'Client', firm.type
502
+ assert_instance_of Client, firm
503
+ ensure
504
+ ActiveRecord::Base.connection.change_column_default :companies, :type, original_type
505
+ Company.reset_column_information
506
+ end
507
+ end
508
+
509
+ class InheritanceAttributeTest < ActiveRecord::TestCase
510
+
511
+ class Company < ActiveRecord::Base
512
+ self.table_name = 'companies'
513
+ attribute :type, :string, default: "InheritanceAttributeTest::Startup"
514
+ end
515
+
516
+ class Startup < Company
517
+ end
518
+
519
+ class Empire < Company
520
+ end
521
+
522
+ def test_inheritance_new_with_subclass_as_default
523
+ startup = Company.new # without arguments
524
+ assert_equal 'InheritanceAttributeTest::Startup', startup.type
525
+ assert_instance_of Startup, startup
526
+
527
+ empire = Company.new(type: 'InheritanceAttributeTest::Empire') # without arguments
528
+ assert_equal 'InheritanceAttributeTest::Empire', empire.type
529
+ assert_instance_of Empire, empire
530
+ end
531
+ end
532
+
533
+ class InheritanceAttributeMappingTest < ActiveRecord::TestCase
534
+ setup do
535
+ @old_registry = ActiveRecord::Type.registry
536
+ ActiveRecord::Type.registry = ActiveRecord::Type::AdapterSpecificRegistry.new
537
+ ActiveRecord::Type.register :omg_sti, InheritanceAttributeMappingTest::OmgStiType
538
+ Company.delete_all
539
+ Sponsor.delete_all
540
+ end
541
+
542
+ teardown do
543
+ ActiveRecord::Type.registry = @old_registry
544
+ end
545
+
546
+ class OmgStiType < ActiveRecord::Type::String
547
+ def cast_value(value)
548
+ if value =~ /\Aomg_(.+)\z/
549
+ $1.classify
550
+ else
551
+ value
552
+ end
553
+ end
554
+
555
+ def serialize(value)
556
+ if value
557
+ "omg_%s" % value.underscore
558
+ end
559
+ end
560
+ end
561
+
562
+ class Company < ActiveRecord::Base
563
+ self.table_name = 'companies'
564
+ attribute :type, :omg_sti
565
+ end
566
+
567
+ class Startup < Company; end
568
+ class Empire < Company; end
569
+
570
+ class Sponsor < ActiveRecord::Base
571
+ self.table_name = 'sponsors'
572
+ attribute :sponsorable_type, :omg_sti
573
+
574
+ belongs_to :sponsorable, polymorphic: true
575
+ end
576
+
577
+ def test_sti_with_custom_type
578
+ Startup.create! name: 'a Startup'
579
+ Empire.create! name: 'an Empire'
580
+
581
+ assert_equal [["a Startup", "omg_inheritance_attribute_mapping_test/startup"],
582
+ ["an Empire", "omg_inheritance_attribute_mapping_test/empire"]], ActiveRecord::Base.connection.select_rows('SELECT name, type FROM companies').sort
583
+ assert_equal [["a Startup", "InheritanceAttributeMappingTest::Startup"],
584
+ ["an Empire", "InheritanceAttributeMappingTest::Empire"]], Company.all.map { |a| [a.name, a.type] }.sort
585
+
586
+ startup = Startup.first
587
+ startup.becomes! Empire
588
+ startup.save!
589
+
590
+ assert_equal [["a Startup", "omg_inheritance_attribute_mapping_test/empire"],
591
+ ["an Empire", "omg_inheritance_attribute_mapping_test/empire"]], ActiveRecord::Base.connection.select_rows('SELECT name, type FROM companies').sort
592
+
593
+ assert_equal [["a Startup", "InheritanceAttributeMappingTest::Empire"],
594
+ ["an Empire", "InheritanceAttributeMappingTest::Empire"]], Company.all.map { |a| [a.name, a.type] }.sort
595
+ end
596
+
597
+ def test_polymorphic_associations_custom_type
598
+ startup = Startup.create! name: 'a Startup'
599
+ sponsor = Sponsor.create! sponsorable: startup
600
+
601
+ assert_equal ["omg_inheritance_attribute_mapping_test/company"], ActiveRecord::Base.connection.select_values('SELECT sponsorable_type FROM sponsors')
602
+
603
+ sponsor = Sponsor.first
604
+ assert_equal startup, sponsor.sponsorable
605
+ end
606
+ end