ibm_db 3.0.5 → 4.0.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (580) 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/test/active_record/connection_adapters/fake_adapter.rb +49 -46
  20. data/test/assets/example.log +1 -1
  21. data/test/assets/test.txt +1 -1
  22. data/test/cases/adapter_test.rb +351 -276
  23. data/test/cases/adapters/mysql2/active_schema_test.rb +193 -0
  24. data/test/cases/adapters/mysql2/bind_parameter_test.rb +50 -0
  25. data/test/cases/adapters/mysql2/boolean_test.rb +100 -0
  26. data/test/cases/adapters/mysql2/case_sensitivity_test.rb +63 -0
  27. data/test/cases/adapters/mysql2/charset_collation_test.rb +54 -0
  28. data/test/cases/adapters/mysql2/connection_test.rb +210 -0
  29. data/test/cases/adapters/mysql2/datetime_precision_quoting_test.rb +45 -0
  30. data/test/cases/adapters/mysql2/enum_test.rb +26 -0
  31. data/test/cases/adapters/mysql2/explain_test.rb +21 -0
  32. data/test/cases/adapters/mysql2/json_test.rb +195 -0
  33. data/test/cases/adapters/mysql2/mysql2_adapter_test.rb +83 -0
  34. data/test/cases/adapters/mysql2/reserved_word_test.rb +152 -0
  35. data/test/cases/adapters/mysql2/schema_migrations_test.rb +59 -0
  36. data/test/cases/adapters/mysql2/schema_test.rb +126 -0
  37. data/test/cases/adapters/mysql2/sp_test.rb +36 -0
  38. data/test/cases/adapters/mysql2/sql_types_test.rb +14 -0
  39. data/test/cases/adapters/mysql2/table_options_test.rb +42 -0
  40. data/test/cases/adapters/mysql2/unsigned_type_test.rb +66 -0
  41. data/test/cases/adapters/postgresql/active_schema_test.rb +98 -0
  42. data/test/cases/adapters/postgresql/array_test.rb +339 -0
  43. data/test/cases/adapters/postgresql/bit_string_test.rb +82 -0
  44. data/test/cases/adapters/postgresql/bytea_test.rb +134 -0
  45. data/test/cases/adapters/postgresql/case_insensitive_test.rb +26 -0
  46. data/test/cases/adapters/postgresql/change_schema_test.rb +38 -0
  47. data/test/cases/adapters/postgresql/cidr_test.rb +25 -0
  48. data/test/cases/adapters/postgresql/citext_test.rb +78 -0
  49. data/test/cases/adapters/postgresql/collation_test.rb +53 -0
  50. data/test/cases/adapters/postgresql/composite_test.rb +132 -0
  51. data/test/cases/adapters/postgresql/connection_test.rb +257 -0
  52. data/test/cases/adapters/postgresql/datatype_test.rb +92 -0
  53. data/test/cases/adapters/postgresql/domain_test.rb +47 -0
  54. data/test/cases/adapters/postgresql/enum_test.rb +91 -0
  55. data/test/cases/adapters/postgresql/explain_test.rb +20 -0
  56. data/test/cases/adapters/postgresql/extension_migration_test.rb +63 -0
  57. data/test/cases/adapters/postgresql/full_text_test.rb +44 -0
  58. data/test/cases/adapters/postgresql/geometric_test.rb +378 -0
  59. data/test/cases/adapters/postgresql/hstore_test.rb +382 -0
  60. data/test/cases/adapters/postgresql/infinity_test.rb +69 -0
  61. data/test/cases/adapters/postgresql/integer_test.rb +25 -0
  62. data/test/cases/adapters/postgresql/json_test.rb +237 -0
  63. data/test/cases/adapters/postgresql/ltree_test.rb +53 -0
  64. data/test/cases/adapters/postgresql/money_test.rb +96 -0
  65. data/test/cases/adapters/postgresql/network_test.rb +94 -0
  66. data/test/cases/adapters/postgresql/numbers_test.rb +49 -0
  67. data/test/cases/adapters/postgresql/postgresql_adapter_test.rb +405 -0
  68. data/test/cases/adapters/postgresql/prepared_statements_test.rb +22 -0
  69. data/test/cases/adapters/postgresql/quoting_test.rb +44 -0
  70. data/test/cases/adapters/postgresql/range_test.rb +343 -0
  71. data/test/cases/adapters/postgresql/referential_integrity_test.rb +111 -0
  72. data/test/cases/adapters/postgresql/rename_table_test.rb +34 -0
  73. data/test/cases/adapters/postgresql/schema_authorization_test.rb +119 -0
  74. data/test/cases/adapters/postgresql/schema_test.rb +597 -0
  75. data/test/cases/adapters/postgresql/serial_test.rb +154 -0
  76. data/test/cases/adapters/postgresql/statement_pool_test.rb +41 -0
  77. data/test/cases/adapters/postgresql/timestamp_test.rb +90 -0
  78. data/test/cases/adapters/postgresql/type_lookup_test.rb +33 -0
  79. data/test/cases/adapters/postgresql/utils_test.rb +62 -0
  80. data/test/cases/adapters/postgresql/uuid_test.rb +294 -0
  81. data/test/cases/adapters/postgresql/xml_test.rb +54 -0
  82. data/test/cases/adapters/sqlite3/collation_test.rb +53 -0
  83. data/test/cases/adapters/sqlite3/copy_table_test.rb +98 -0
  84. data/test/cases/adapters/sqlite3/explain_test.rb +21 -0
  85. data/test/cases/adapters/sqlite3/quoting_test.rb +101 -0
  86. data/test/cases/adapters/sqlite3/sqlite3_adapter_test.rb +441 -0
  87. data/test/cases/adapters/sqlite3/sqlite3_create_folder_test.rb +24 -0
  88. data/test/cases/adapters/sqlite3/statement_pool_test.rb +20 -0
  89. data/test/cases/aggregations_test.rb +168 -158
  90. data/test/cases/ar_schema_test.rb +146 -161
  91. data/test/cases/associations/association_scope_test.rb +16 -21
  92. data/test/cases/associations/belongs_to_associations_test.rb +1141 -1029
  93. data/test/cases/associations/bidirectional_destroy_dependencies_test.rb +41 -0
  94. data/test/cases/associations/callbacks_test.rb +190 -192
  95. data/test/cases/associations/cascaded_eager_loading_test.rb +188 -188
  96. data/test/cases/associations/eager_load_includes_full_sti_class_test.rb +36 -36
  97. data/test/cases/associations/eager_load_nested_include_test.rb +126 -128
  98. data/test/cases/associations/eager_singularization_test.rb +148 -148
  99. data/test/cases/associations/eager_test.rb +1514 -1429
  100. data/test/cases/associations/extension_test.rb +87 -82
  101. data/test/cases/associations/has_and_belongs_to_many_associations_test.rb +1004 -972
  102. data/test/cases/associations/has_many_associations_test.rb +2501 -2182
  103. data/test/cases/associations/has_many_through_associations_test.rb +1271 -1204
  104. data/test/cases/associations/has_one_associations_test.rb +707 -610
  105. data/test/cases/associations/has_one_through_associations_test.rb +383 -380
  106. data/test/cases/associations/inner_join_association_test.rb +139 -139
  107. data/test/cases/associations/inverse_associations_test.rb +733 -706
  108. data/test/cases/associations/join_model_test.rb +777 -754
  109. data/test/cases/associations/left_outer_join_association_test.rb +88 -0
  110. data/test/cases/associations/nested_through_associations_test.rb +579 -579
  111. data/test/cases/associations/required_test.rb +102 -82
  112. data/test/cases/associations_test.rb +385 -380
  113. data/test/cases/attribute_decorators_test.rb +125 -125
  114. data/test/cases/attribute_methods/read_test.rb +60 -60
  115. data/test/cases/attribute_methods_test.rb +1009 -952
  116. data/test/cases/attribute_set_test.rb +270 -210
  117. data/test/cases/attribute_test.rb +246 -180
  118. data/test/cases/attributes_test.rb +253 -136
  119. data/test/cases/autosave_association_test.rb +1708 -1595
  120. data/test/cases/base_test.rb +1713 -1664
  121. data/test/cases/batches_test.rb +489 -212
  122. data/test/cases/binary_test.rb +44 -52
  123. data/test/cases/bind_parameter_test.rb +110 -100
  124. data/test/cases/cache_key_test.rb +25 -0
  125. data/test/cases/calculations_test.rb +798 -646
  126. data/test/cases/callbacks_test.rb +636 -543
  127. data/test/cases/clone_test.rb +40 -40
  128. data/test/cases/coders/json_test.rb +15 -0
  129. data/test/cases/coders/yaml_column_test.rb +63 -63
  130. data/test/cases/collection_cache_key_test.rb +115 -0
  131. data/test/cases/column_alias_test.rb +17 -17
  132. data/test/cases/column_definition_test.rb +92 -123
  133. data/test/cases/comment_test.rb +143 -0
  134. data/test/cases/connection_adapters/adapter_leasing_test.rb +56 -54
  135. data/test/cases/connection_adapters/connection_handler_test.rb +160 -53
  136. data/test/cases/connection_adapters/connection_specification_test.rb +12 -12
  137. data/test/cases/connection_adapters/merge_and_resolve_default_url_config_test.rb +255 -293
  138. data/test/cases/connection_adapters/mysql_type_lookup_test.rb +69 -65
  139. data/test/cases/connection_adapters/quoting_test.rb +13 -13
  140. data/test/cases/connection_adapters/schema_cache_test.rb +61 -56
  141. data/test/cases/connection_adapters/type_lookup_test.rb +118 -110
  142. data/test/cases/connection_management_test.rb +112 -122
  143. data/test/cases/connection_pool_test.rb +521 -346
  144. data/test/cases/connection_specification/resolver_test.rb +131 -116
  145. data/test/cases/core_test.rb +112 -112
  146. data/test/cases/counter_cache_test.rb +214 -209
  147. data/test/cases/custom_locking_test.rb +17 -17
  148. data/test/cases/database_statements_test.rb +34 -19
  149. data/test/cases/{invalid_date_test.rb → date_test.rb} +44 -32
  150. data/test/cases/date_time_precision_test.rb +106 -0
  151. data/test/cases/date_time_test.rb +61 -61
  152. data/test/cases/defaults_test.rb +218 -223
  153. data/test/cases/dirty_test.rb +763 -785
  154. data/test/cases/disconnected_test.rb +30 -28
  155. data/test/cases/dup_test.rb +157 -157
  156. data/test/cases/enum_test.rb +444 -290
  157. data/test/cases/errors_test.rb +16 -0
  158. data/test/cases/explain_subscriber_test.rb +64 -64
  159. data/test/cases/explain_test.rb +87 -76
  160. data/test/cases/finder_respond_to_test.rb +60 -60
  161. data/test/cases/finder_test.rb +1294 -1169
  162. data/test/cases/fixture_set/file_test.rb +156 -138
  163. data/test/cases/fixtures_test.rb +988 -908
  164. data/test/cases/forbidden_attributes_protection_test.rb +165 -99
  165. data/test/cases/habtm_destroy_order_test.rb +61 -61
  166. data/test/cases/helper.rb +204 -210
  167. data/test/cases/hot_compatibility_test.rb +142 -54
  168. data/test/cases/i18n_test.rb +45 -45
  169. data/test/cases/inheritance_test.rb +606 -375
  170. data/test/cases/integration_test.rb +155 -139
  171. data/test/cases/invalid_connection_test.rb +24 -22
  172. data/test/cases/invertible_migration_test.rb +387 -295
  173. data/test/cases/json_serialization_test.rb +311 -302
  174. data/test/cases/locking_test.rb +493 -477
  175. data/test/cases/log_subscriber_test.rb +225 -136
  176. data/test/cases/migration/change_schema_test.rb +458 -512
  177. data/test/cases/migration/change_table_test.rb +256 -224
  178. data/test/cases/migration/column_attributes_test.rb +176 -192
  179. data/test/cases/migration/column_positioning_test.rb +56 -56
  180. data/test/cases/migration/columns_test.rb +310 -304
  181. data/test/cases/migration/command_recorder_test.rb +350 -305
  182. data/test/cases/migration/compatibility_test.rb +118 -0
  183. data/test/cases/migration/create_join_table_test.rb +157 -148
  184. data/test/cases/migration/foreign_key_test.rb +360 -328
  185. data/test/cases/migration/helper.rb +39 -39
  186. data/test/cases/migration/index_test.rb +218 -216
  187. data/test/cases/migration/logger_test.rb +36 -36
  188. data/test/cases/migration/pending_migrations_test.rb +52 -53
  189. data/test/cases/migration/references_foreign_key_test.rb +216 -169
  190. data/test/cases/migration/references_index_test.rb +101 -101
  191. data/test/cases/migration/references_statements_test.rb +136 -116
  192. data/test/cases/migration/rename_table_test.rb +93 -93
  193. data/test/cases/migration_test.rb +1157 -959
  194. data/test/cases/migrator_test.rb +470 -388
  195. data/test/cases/mixin_test.rb +68 -70
  196. data/test/cases/modules_test.rb +172 -173
  197. data/test/cases/multiparameter_attributes_test.rb +372 -350
  198. data/test/cases/multiple_db_test.rb +122 -115
  199. data/test/cases/nested_attributes_test.rb +1098 -1070
  200. data/test/cases/nested_attributes_with_callbacks_test.rb +144 -144
  201. data/test/cases/persistence_test.rb +1001 -909
  202. data/test/cases/pooled_connections_test.rb +81 -81
  203. data/test/cases/primary_keys_test.rb +376 -237
  204. data/test/cases/query_cache_test.rb +446 -326
  205. data/test/cases/quoting_test.rb +202 -156
  206. data/test/cases/readonly_test.rb +119 -118
  207. data/test/cases/reaper_test.rb +85 -85
  208. data/test/cases/reflection_test.rb +509 -463
  209. data/test/cases/relation/delegation_test.rb +63 -68
  210. data/test/cases/relation/merging_test.rb +157 -161
  211. data/test/cases/relation/mutation_test.rb +183 -165
  212. data/test/cases/relation/or_test.rb +92 -0
  213. data/test/cases/relation/predicate_builder_test.rb +16 -14
  214. data/test/cases/relation/record_fetch_warning_test.rb +40 -0
  215. data/test/cases/relation/where_chain_test.rb +105 -181
  216. data/test/cases/relation/where_clause_test.rb +182 -0
  217. data/test/cases/relation/where_test.rb +322 -300
  218. data/test/cases/relation_test.rb +328 -319
  219. data/test/cases/relations_test.rb +2026 -1815
  220. data/test/cases/reload_models_test.rb +22 -22
  221. data/test/cases/result_test.rb +90 -80
  222. data/test/cases/sanitize_test.rb +176 -83
  223. data/test/cases/schema_dumper_test.rb +457 -463
  224. data/test/cases/schema_loading_test.rb +52 -0
  225. data/test/cases/scoping/default_scoping_test.rb +528 -454
  226. data/test/cases/scoping/named_scoping_test.rb +561 -524
  227. data/test/cases/scoping/relation_scoping_test.rb +400 -357
  228. data/test/cases/secure_token_test.rb +32 -0
  229. data/test/cases/serialization_test.rb +104 -104
  230. data/test/cases/serialized_attribute_test.rb +364 -277
  231. data/test/cases/statement_cache_test.rb +136 -98
  232. data/test/cases/store_test.rb +195 -194
  233. data/test/cases/suppressor_test.rb +63 -0
  234. data/test/cases/tasks/database_tasks_test.rb +462 -398
  235. data/test/cases/tasks/mysql_rake_test.rb +345 -324
  236. data/test/cases/tasks/postgresql_rake_test.rb +304 -250
  237. data/test/cases/tasks/sqlite_rake_test.rb +220 -193
  238. data/test/cases/test_case.rb +131 -123
  239. data/test/cases/test_fixtures_test.rb +36 -0
  240. data/test/cases/time_precision_test.rb +102 -0
  241. data/test/cases/timestamp_test.rb +501 -467
  242. data/test/cases/touch_later_test.rb +121 -0
  243. data/test/cases/transaction_callbacks_test.rb +518 -452
  244. data/test/cases/transaction_isolation_test.rb +106 -106
  245. data/test/cases/transactions_test.rb +834 -817
  246. data/test/cases/type/adapter_specific_registry_test.rb +133 -0
  247. data/test/cases/type/date_time_test.rb +14 -0
  248. data/test/cases/type/integer_test.rb +27 -121
  249. data/test/cases/type/string_test.rb +22 -36
  250. data/test/cases/type/type_map_test.rb +177 -177
  251. data/test/cases/type_test.rb +39 -0
  252. data/test/cases/types_test.rb +24 -141
  253. data/test/cases/unconnected_test.rb +33 -33
  254. data/test/cases/validations/absence_validation_test.rb +73 -0
  255. data/test/cases/validations/association_validation_test.rb +97 -86
  256. data/test/cases/validations/i18n_generate_message_validation_test.rb +84 -84
  257. data/test/cases/validations/i18n_validation_test.rb +86 -90
  258. data/test/cases/validations/length_validation_test.rb +79 -47
  259. data/test/cases/validations/presence_validation_test.rb +103 -68
  260. data/test/cases/validations/uniqueness_validation_test.rb +548 -457
  261. data/test/cases/validations_repair_helper.rb +19 -23
  262. data/test/cases/validations_test.rb +194 -165
  263. data/test/cases/view_test.rb +216 -119
  264. data/test/cases/yaml_serialization_test.rb +121 -126
  265. data/test/config.example.yml +97 -0
  266. data/test/config.rb +5 -5
  267. data/test/fixtures/accounts.yml +29 -29
  268. data/test/fixtures/admin/accounts.yml +2 -2
  269. data/test/fixtures/admin/users.yml +10 -10
  270. data/test/fixtures/author_addresses.original +11 -0
  271. data/test/fixtures/author_addresses.yml +17 -17
  272. data/test/fixtures/author_favorites.yml +3 -3
  273. data/test/fixtures/authors.original +17 -0
  274. data/test/fixtures/authors.yml +23 -23
  275. data/test/fixtures/bad_posts.yml +9 -0
  276. data/test/fixtures/binaries.yml +133 -133
  277. data/test/fixtures/books.yml +31 -11
  278. data/test/fixtures/bulbs.yml +5 -5
  279. data/test/fixtures/cars.yml +9 -9
  280. data/test/fixtures/categories.yml +19 -19
  281. data/test/fixtures/categories/special_categories.yml +9 -9
  282. data/test/fixtures/categories/subsubdir/arbitrary_filename.yml +4 -4
  283. data/test/fixtures/categories_ordered.yml +7 -7
  284. data/test/fixtures/categories_posts.yml +31 -31
  285. data/test/fixtures/categorizations.yml +23 -23
  286. data/test/fixtures/clubs.yml +8 -8
  287. data/test/fixtures/collections.yml +3 -3
  288. data/test/fixtures/colleges.yml +3 -3
  289. data/test/fixtures/comments.yml +65 -65
  290. data/test/fixtures/companies.yml +67 -67
  291. data/test/fixtures/computers.yml +10 -10
  292. data/test/fixtures/content.yml +3 -0
  293. data/test/fixtures/content_positions.yml +3 -0
  294. data/test/fixtures/courses.yml +8 -8
  295. data/test/fixtures/customers.yml +25 -25
  296. data/test/fixtures/dashboards.yml +6 -6
  297. data/test/fixtures/dead_parrots.yml +5 -0
  298. data/test/fixtures/developers.yml +22 -22
  299. data/test/fixtures/developers_projects.yml +16 -16
  300. data/test/fixtures/dog_lovers.yml +7 -7
  301. data/test/fixtures/dogs.yml +4 -4
  302. data/test/fixtures/doubloons.yml +3 -3
  303. data/test/fixtures/edges.yml +5 -5
  304. data/test/fixtures/entrants.yml +14 -14
  305. data/test/fixtures/essays.yml +6 -6
  306. data/test/fixtures/faces.yml +11 -11
  307. data/test/fixtures/fk_test_has_fk.yml +3 -3
  308. data/test/fixtures/fk_test_has_pk.yml +1 -1
  309. data/test/fixtures/friendships.yml +4 -4
  310. data/test/fixtures/funny_jokes.yml +10 -10
  311. data/test/fixtures/interests.yml +33 -33
  312. data/test/fixtures/items.yml +3 -3
  313. data/test/fixtures/jobs.yml +7 -7
  314. data/test/fixtures/legacy_things.yml +3 -3
  315. data/test/fixtures/live_parrots.yml +4 -0
  316. data/test/fixtures/mateys.yml +4 -4
  317. data/test/fixtures/member_details.yml +8 -8
  318. data/test/fixtures/member_types.yml +6 -6
  319. data/test/fixtures/members.yml +11 -11
  320. data/test/fixtures/memberships.yml +34 -34
  321. data/test/fixtures/men.yml +5 -5
  322. data/test/fixtures/minimalistics.yml +2 -2
  323. data/test/fixtures/minivans.yml +5 -5
  324. data/test/fixtures/mixed_case_monkeys.yml +6 -6
  325. data/test/fixtures/mixins.yml +29 -29
  326. data/test/fixtures/movies.yml +7 -7
  327. data/test/fixtures/naked/yml/accounts.yml +1 -1
  328. data/test/fixtures/naked/yml/companies.yml +1 -1
  329. data/test/fixtures/naked/yml/courses.yml +1 -1
  330. data/test/fixtures/naked/yml/parrots.yml +2 -0
  331. data/test/fixtures/naked/yml/trees.yml +3 -0
  332. data/test/fixtures/nodes.yml +29 -0
  333. data/test/fixtures/organizations.yml +5 -5
  334. data/test/fixtures/other_comments.yml +6 -0
  335. data/test/fixtures/other_dogs.yml +2 -0
  336. data/test/fixtures/other_posts.yml +7 -0
  337. data/test/fixtures/other_topics.yml +42 -42
  338. data/test/fixtures/owners.yml +9 -9
  339. data/test/fixtures/parrots.yml +27 -27
  340. data/test/fixtures/parrots_pirates.yml +7 -7
  341. data/test/fixtures/people.yml +24 -24
  342. data/test/fixtures/peoples_treasures.yml +3 -3
  343. data/test/fixtures/pets.yml +19 -19
  344. data/test/fixtures/pirates.yml +15 -12
  345. data/test/fixtures/posts.yml +80 -80
  346. data/test/fixtures/price_estimates.yml +16 -7
  347. data/test/fixtures/products.yml +4 -4
  348. data/test/fixtures/projects.yml +7 -7
  349. data/test/fixtures/ratings.yml +14 -14
  350. data/test/fixtures/readers.yml +11 -11
  351. data/test/fixtures/references.yml +17 -17
  352. data/test/fixtures/reserved_words/distinct.yml +5 -5
  353. data/test/fixtures/reserved_words/distinct_select.yml +11 -11
  354. data/test/fixtures/reserved_words/group.yml +14 -14
  355. data/test/fixtures/reserved_words/select.yml +8 -8
  356. data/test/fixtures/reserved_words/values.yml +7 -7
  357. data/test/fixtures/ships.yml +6 -6
  358. data/test/fixtures/speedometers.yml +8 -8
  359. data/test/fixtures/sponsors.yml +12 -12
  360. data/test/fixtures/string_key_objects.yml +7 -7
  361. data/test/fixtures/subscribers.yml +10 -10
  362. data/test/fixtures/subscriptions.yml +12 -12
  363. data/test/fixtures/taggings.yml +78 -78
  364. data/test/fixtures/tags.yml +11 -11
  365. data/test/fixtures/tasks.yml +7 -7
  366. data/test/fixtures/teapots.yml +3 -3
  367. data/test/fixtures/to_be_linked/accounts.yml +2 -2
  368. data/test/fixtures/to_be_linked/users.yml +10 -10
  369. data/test/fixtures/topics.yml +49 -49
  370. data/test/fixtures/toys.yml +14 -14
  371. data/test/fixtures/traffic_lights.yml +9 -9
  372. data/test/fixtures/treasures.yml +10 -10
  373. data/test/fixtures/trees.yml +3 -0
  374. data/test/fixtures/uuid_children.yml +3 -3
  375. data/test/fixtures/uuid_parents.yml +2 -2
  376. data/test/fixtures/variants.yml +4 -4
  377. data/test/fixtures/vegetables.yml +19 -19
  378. data/test/fixtures/vertices.yml +3 -3
  379. data/test/fixtures/warehouse_things.yml +2 -2
  380. data/test/fixtures/zines.yml +5 -5
  381. data/test/migrations/10_urban/9_add_expressions.rb +11 -11
  382. data/test/migrations/decimal/1_give_me_big_numbers.rb +15 -15
  383. data/test/migrations/magic/1_currencies_have_symbols.rb +12 -12
  384. data/test/migrations/missing/1000_people_have_middle_names.rb +9 -9
  385. data/test/migrations/missing/1_people_have_last_names.rb +9 -9
  386. data/test/migrations/missing/3_we_need_reminders.rb +12 -12
  387. data/test/migrations/missing/4_innocent_jointable.rb +12 -12
  388. data/test/migrations/rename/1_we_need_things.rb +11 -11
  389. data/test/migrations/rename/2_rename_things.rb +9 -9
  390. data/test/migrations/to_copy/1_people_have_hobbies.rb +9 -9
  391. data/test/migrations/to_copy/2_people_have_descriptions.rb +9 -9
  392. data/test/migrations/to_copy2/1_create_articles.rb +7 -7
  393. data/test/migrations/to_copy2/2_create_comments.rb +7 -7
  394. data/test/migrations/to_copy_with_name_collision/1_people_have_hobbies.rb +9 -9
  395. data/test/migrations/to_copy_with_timestamps/20090101010101_people_have_hobbies.rb +9 -9
  396. data/test/migrations/to_copy_with_timestamps/20090101010202_people_have_descriptions.rb +9 -9
  397. data/test/migrations/to_copy_with_timestamps2/20090101010101_create_articles.rb +7 -7
  398. data/test/migrations/to_copy_with_timestamps2/20090101010202_create_comments.rb +7 -7
  399. data/test/migrations/valid/1_valid_people_have_last_names.rb +9 -9
  400. data/test/migrations/valid/2_we_need_reminders.rb +12 -12
  401. data/test/migrations/valid/3_innocent_jointable.rb +12 -12
  402. data/test/migrations/valid_with_subdirectories/1_valid_people_have_last_names.rb +9 -9
  403. data/test/migrations/valid_with_subdirectories/sub/2_we_need_reminders.rb +12 -12
  404. data/test/migrations/valid_with_subdirectories/sub1/3_innocent_jointable.rb +12 -12
  405. data/test/migrations/valid_with_timestamps/20100101010101_valid_with_timestamps_people_have_last_names.rb +9 -9
  406. data/test/migrations/valid_with_timestamps/20100201010101_valid_with_timestamps_we_need_reminders.rb +12 -12
  407. data/test/migrations/valid_with_timestamps/20100301010101_valid_with_timestamps_innocent_jointable.rb +12 -12
  408. data/test/migrations/version_check/20131219224947_migration_version_check.rb +8 -8
  409. data/test/models/admin.rb +5 -5
  410. data/test/models/admin/account.rb +3 -3
  411. data/test/models/admin/randomly_named_c1.rb +6 -2
  412. data/test/models/admin/user.rb +40 -40
  413. data/test/models/aircraft.rb +5 -4
  414. data/test/models/arunit2_model.rb +3 -3
  415. data/test/models/author.rb +209 -212
  416. data/test/models/auto_id.rb +4 -4
  417. data/test/models/autoloadable/extra_firm.rb +2 -2
  418. data/test/models/binary.rb +2 -2
  419. data/test/models/bird.rb +12 -12
  420. data/test/models/book.rb +23 -18
  421. data/test/models/boolean.rb +2 -2
  422. data/test/models/bulb.rb +52 -51
  423. data/test/models/cake_designer.rb +3 -3
  424. data/test/models/car.rb +29 -26
  425. data/test/models/carrier.rb +2 -2
  426. data/test/models/cat.rb +10 -0
  427. data/test/models/categorization.rb +19 -19
  428. data/test/models/category.rb +35 -35
  429. data/test/models/chef.rb +8 -7
  430. data/test/models/citation.rb +3 -3
  431. data/test/models/club.rb +25 -23
  432. data/test/models/college.rb +10 -10
  433. data/test/models/column.rb +3 -3
  434. data/test/models/column_name.rb +3 -3
  435. data/test/models/comment.rb +76 -64
  436. data/test/models/company.rb +230 -228
  437. data/test/models/company_in_module.rb +98 -98
  438. data/test/models/computer.rb +3 -3
  439. data/test/models/contact.rb +41 -41
  440. data/test/models/content.rb +40 -0
  441. data/test/models/contract.rb +20 -20
  442. data/test/models/country.rb +7 -7
  443. data/test/models/course.rb +6 -6
  444. data/test/models/customer.rb +83 -77
  445. data/test/models/customer_carrier.rb +14 -14
  446. data/test/models/dashboard.rb +3 -3
  447. data/test/models/default.rb +2 -2
  448. data/test/models/department.rb +4 -4
  449. data/test/models/developer.rb +274 -255
  450. data/test/models/dog.rb +5 -5
  451. data/test/models/dog_lover.rb +5 -5
  452. data/test/models/doubloon.rb +12 -12
  453. data/test/models/drink_designer.rb +3 -3
  454. data/test/models/edge.rb +5 -5
  455. data/test/models/electron.rb +5 -5
  456. data/test/models/engine.rb +4 -4
  457. data/test/models/entrant.rb +3 -3
  458. data/test/models/essay.rb +5 -5
  459. data/test/models/event.rb +3 -3
  460. data/test/models/eye.rb +37 -37
  461. data/test/models/face.rb +9 -9
  462. data/test/models/friendship.rb +6 -6
  463. data/test/models/guid.rb +2 -2
  464. data/test/models/guitar.rb +4 -0
  465. data/test/models/hotel.rb +11 -9
  466. data/test/models/image.rb +3 -3
  467. data/test/models/interest.rb +5 -5
  468. data/test/models/invoice.rb +4 -4
  469. data/test/models/item.rb +7 -7
  470. data/test/models/job.rb +7 -7
  471. data/test/models/joke.rb +7 -7
  472. data/test/models/keyboard.rb +3 -3
  473. data/test/models/legacy_thing.rb +3 -3
  474. data/test/models/lesson.rb +11 -11
  475. data/test/models/line_item.rb +3 -3
  476. data/test/models/liquid.rb +4 -4
  477. data/test/models/man.rb +11 -11
  478. data/test/models/matey.rb +4 -4
  479. data/test/models/member.rb +42 -41
  480. data/test/models/member_detail.rb +8 -7
  481. data/test/models/member_type.rb +3 -3
  482. data/test/models/membership.rb +35 -35
  483. data/test/models/mentor.rb +3 -0
  484. data/test/models/minimalistic.rb +2 -2
  485. data/test/models/minivan.rb +9 -9
  486. data/test/models/mixed_case_monkey.rb +3 -3
  487. data/test/models/mocktail_designer.rb +2 -0
  488. data/test/models/molecule.rb +6 -6
  489. data/test/models/movie.rb +5 -5
  490. data/test/models/node.rb +5 -0
  491. data/test/models/non_primary_key.rb +2 -0
  492. data/test/models/notification.rb +3 -0
  493. data/test/models/order.rb +4 -4
  494. data/test/models/organization.rb +14 -14
  495. data/test/models/other_dog.rb +5 -0
  496. data/test/models/owner.rb +37 -34
  497. data/test/models/parrot.rb +28 -29
  498. data/test/models/person.rb +142 -143
  499. data/test/models/personal_legacy_thing.rb +4 -4
  500. data/test/models/pet.rb +18 -15
  501. data/test/models/pet_treasure.rb +6 -0
  502. data/test/models/pirate.rb +92 -92
  503. data/test/models/possession.rb +3 -3
  504. data/test/models/post.rb +273 -264
  505. data/test/models/price_estimate.rb +4 -4
  506. data/test/models/professor.rb +5 -5
  507. data/test/models/project.rb +40 -31
  508. data/test/models/publisher.rb +2 -2
  509. data/test/models/publisher/article.rb +4 -4
  510. data/test/models/publisher/magazine.rb +3 -3
  511. data/test/models/randomly_named_c1.rb +1 -1
  512. data/test/models/rating.rb +4 -4
  513. data/test/models/reader.rb +23 -23
  514. data/test/models/recipe.rb +3 -0
  515. data/test/models/record.rb +2 -2
  516. data/test/models/reference.rb +22 -22
  517. data/test/models/reply.rb +61 -61
  518. data/test/models/ship.rb +39 -33
  519. data/test/models/ship_part.rb +8 -8
  520. data/test/models/shop.rb +17 -17
  521. data/test/models/shop_account.rb +6 -6
  522. data/test/models/speedometer.rb +6 -6
  523. data/test/models/sponsor.rb +7 -7
  524. data/test/models/string_key_object.rb +3 -3
  525. data/test/models/student.rb +4 -4
  526. data/test/models/subject.rb +16 -16
  527. data/test/models/subscriber.rb +8 -8
  528. data/test/models/subscription.rb +4 -4
  529. data/test/models/tag.rb +13 -7
  530. data/test/models/tagging.rb +13 -13
  531. data/test/models/task.rb +5 -5
  532. data/test/models/topic.rb +118 -124
  533. data/test/models/toy.rb +6 -6
  534. data/test/models/traffic_light.rb +4 -4
  535. data/test/models/treasure.rb +14 -14
  536. data/test/models/treaty.rb +7 -7
  537. data/test/models/tree.rb +3 -0
  538. data/test/models/tuning_peg.rb +4 -0
  539. data/test/models/tyre.rb +11 -11
  540. data/test/models/user.rb +14 -0
  541. data/test/models/uuid_child.rb +3 -3
  542. data/test/models/uuid_item.rb +6 -0
  543. data/test/models/uuid_parent.rb +3 -3
  544. data/test/models/vegetables.rb +24 -24
  545. data/test/models/vehicle.rb +6 -6
  546. data/test/models/vertex.rb +9 -9
  547. data/test/models/warehouse_thing.rb +5 -5
  548. data/test/models/wheel.rb +3 -3
  549. data/test/models/without_table.rb +3 -3
  550. data/test/models/zine.rb +3 -3
  551. data/test/schema/mysql2_specific_schema.rb +68 -58
  552. data/test/schema/oracle_specific_schema.rb +40 -43
  553. data/test/schema/postgresql_specific_schema.rb +114 -202
  554. data/test/schema/schema.rb +1057 -952
  555. data/test/schema/schema.rb.original +1057 -0
  556. data/test/schema/sqlite_specific_schema.rb +18 -22
  557. data/test/support/config.rb +43 -43
  558. data/test/support/connection.rb +23 -22
  559. data/test/support/connection_helper.rb +14 -14
  560. data/test/support/ddl_helper.rb +8 -8
  561. data/test/support/schema_dumping_helper.rb +20 -20
  562. data/test/support/yaml_compatibility_fixtures/rails_4_1.yml +22 -0
  563. data/test/support/yaml_compatibility_fixtures/rails_4_2_0.yml +182 -0
  564. metadata +129 -24
  565. data/test/cases/associations/deprecated_counter_cache_on_has_many_through_test.rb +0 -26
  566. data/test/cases/attribute_methods/serialization_test.rb +0 -29
  567. data/test/cases/migration/change_schema_test - Copy.rb +0 -448
  568. data/test/cases/migration/foreign_key_test - Changed.rb +0 -325
  569. data/test/cases/migration/table_and_index_test.rb +0 -24
  570. data/test/cases/relation/where_test2.rb +0 -36
  571. data/test/cases/type/decimal_test.rb +0 -56
  572. data/test/cases/type/unsigned_integer_test.rb +0 -18
  573. data/test/cases/xml_serialization_test.rb +0 -457
  574. data/test/connections/native_ibm_db/connection.rb +0 -44
  575. data/test/fixtures/naked/csv/accounts.csv +0 -1
  576. data/test/schema/i5/ibm_db_specific_schema.rb +0 -137
  577. data/test/schema/ids/ibm_db_specific_schema.rb +0 -140
  578. data/test/schema/luw/ibm_db_specific_schema.rb +0 -137
  579. data/test/schema/mysql_specific_schema.rb +0 -70
  580. data/test/schema/zOS/ibm_db_specific_schema.rb +0 -208
@@ -0,0 +1,32 @@
1
+ require 'cases/helper'
2
+ require 'models/user'
3
+
4
+ class SecureTokenTest < ActiveRecord::TestCase
5
+ setup do
6
+ @user = User.new
7
+ end
8
+
9
+ def test_token_values_are_generated_for_specified_attributes_and_persisted_on_save
10
+ @user.save
11
+ assert_not_nil @user.token
12
+ assert_not_nil @user.auth_token
13
+ end
14
+
15
+ def test_regenerating_the_secure_token
16
+ @user.save
17
+ old_token = @user.token
18
+ old_auth_token = @user.auth_token
19
+ @user.regenerate_token
20
+ @user.regenerate_auth_token
21
+
22
+ assert_not_equal @user.token, old_token
23
+ assert_not_equal @user.auth_token, old_auth_token
24
+ end
25
+
26
+ def test_token_value_not_overwritten_when_present
27
+ @user.token = "custom-secure-token"
28
+ @user.save
29
+
30
+ assert_equal @user.token, "custom-secure-token"
31
+ end
32
+ end
@@ -1,104 +1,104 @@
1
- require "cases/helper"
2
- require 'models/contact'
3
- require 'models/topic'
4
- require 'models/book'
5
- require 'models/author'
6
- require 'models/post'
7
-
8
- class SerializationTest < ActiveRecord::TestCase
9
- fixtures :books
10
-
11
- FORMATS = [ :xml, :json ]
12
-
13
- def setup
14
- @contact_attributes = {
15
- :name => 'aaron stack',
16
- :age => 25,
17
- :avatar => 'binarydata',
18
- :created_at => Time.utc(2006, 8, 1),
19
- :awesome => false,
20
- :preferences => { :gem => '<strong>ruby</strong>' },
21
- :alternative_id => nil,
22
- :id => nil
23
- }
24
- end
25
-
26
- def test_include_root_in_json_is_false_by_default
27
- assert_equal false, ActiveRecord::Base.include_root_in_json, "include_root_in_json should be false by default but was not"
28
- end
29
-
30
- def test_serialize_should_be_reversible
31
- FORMATS.each do |format|
32
- @serialized = Contact.new.send("to_#{format}")
33
- contact = Contact.new.send("from_#{format}", @serialized)
34
-
35
- assert_equal @contact_attributes.keys.collect(&:to_s).sort, contact.attributes.keys.collect(&:to_s).sort, "For #{format}"
36
- end
37
- end
38
-
39
- def test_serialize_should_allow_attribute_only_filtering
40
- FORMATS.each do |format|
41
- @serialized = Contact.new(@contact_attributes).send("to_#{format}", :only => [ :age, :name ])
42
- contact = Contact.new.send("from_#{format}", @serialized)
43
- assert_equal @contact_attributes[:name], contact.name, "For #{format}"
44
- assert_nil contact.avatar, "For #{format}"
45
- end
46
- end
47
-
48
- def test_serialize_should_allow_attribute_except_filtering
49
- FORMATS.each do |format|
50
- @serialized = Contact.new(@contact_attributes).send("to_#{format}", :except => [ :age, :name ])
51
- contact = Contact.new.send("from_#{format}", @serialized)
52
- assert_nil contact.name, "For #{format}"
53
- assert_nil contact.age, "For #{format}"
54
- assert_equal @contact_attributes[:awesome], contact.awesome, "For #{format}"
55
- end
56
- end
57
-
58
- def test_include_root_in_json_allows_inheritance
59
- original_root_in_json = ActiveRecord::Base.include_root_in_json
60
- ActiveRecord::Base.include_root_in_json = true
61
-
62
- klazz = Class.new(ActiveRecord::Base)
63
- klazz.table_name = 'topics'
64
- assert klazz.include_root_in_json
65
-
66
- klazz.include_root_in_json = false
67
- assert ActiveRecord::Base.include_root_in_json
68
- assert !klazz.include_root_in_json
69
- assert !klazz.new.include_root_in_json
70
- ensure
71
- ActiveRecord::Base.include_root_in_json = original_root_in_json
72
- end
73
-
74
- def test_read_attribute_for_serialization_with_format_without_method_missing
75
- klazz = Class.new(ActiveRecord::Base)
76
- klazz.table_name = 'books'
77
-
78
- book = klazz.new
79
- assert_nil book.read_attribute_for_serialization(:format)
80
- end
81
-
82
- def test_read_attribute_for_serialization_with_format_after_init
83
- klazz = Class.new(ActiveRecord::Base)
84
- klazz.table_name = 'books'
85
-
86
- book = klazz.new(format: 'paperback')
87
- assert_equal 'paperback', book.read_attribute_for_serialization(:format)
88
- end
89
-
90
- def test_read_attribute_for_serialization_with_format_after_find
91
- klazz = Class.new(ActiveRecord::Base)
92
- klazz.table_name = 'books'
93
-
94
- book = klazz.find(books(:awdr).id)
95
- assert_equal 'paperback', book.read_attribute_for_serialization(:format)
96
- end
97
-
98
- def test_find_records_by_serialized_attributes_through_join
99
- author = Author.create!(name: "David")
100
- author.serialized_posts.create!(title: "Hello")
101
-
102
- assert_equal 1, Author.joins(:serialized_posts).where(name: "David", serialized_posts: { title: "Hello" }).length
103
- end
104
- end
1
+ require "cases/helper"
2
+ require 'models/contact'
3
+ require 'models/topic'
4
+ require 'models/book'
5
+ require 'models/author'
6
+ require 'models/post'
7
+
8
+ class SerializationTest < ActiveRecord::TestCase
9
+ fixtures :books
10
+
11
+ FORMATS = [ :json ]
12
+
13
+ def setup
14
+ @contact_attributes = {
15
+ :name => 'aaron stack',
16
+ :age => 25,
17
+ :avatar => 'binarydata',
18
+ :created_at => Time.utc(2006, 8, 1),
19
+ :awesome => false,
20
+ :preferences => { :gem => '<strong>ruby</strong>' },
21
+ :alternative_id => nil,
22
+ :id => nil
23
+ }
24
+ end
25
+
26
+ def test_include_root_in_json_is_false_by_default
27
+ assert_equal false, ActiveRecord::Base.include_root_in_json, "include_root_in_json should be false by default but was not"
28
+ end
29
+
30
+ def test_serialize_should_be_reversible
31
+ FORMATS.each do |format|
32
+ @serialized = Contact.new.send("to_#{format}")
33
+ contact = Contact.new.send("from_#{format}", @serialized)
34
+
35
+ assert_equal @contact_attributes.keys.collect(&:to_s).sort, contact.attributes.keys.collect(&:to_s).sort, "For #{format}"
36
+ end
37
+ end
38
+
39
+ def test_serialize_should_allow_attribute_only_filtering
40
+ FORMATS.each do |format|
41
+ @serialized = Contact.new(@contact_attributes).send("to_#{format}", :only => [ :age, :name ])
42
+ contact = Contact.new.send("from_#{format}", @serialized)
43
+ assert_equal @contact_attributes[:name], contact.name, "For #{format}"
44
+ assert_nil contact.avatar, "For #{format}"
45
+ end
46
+ end
47
+
48
+ def test_serialize_should_allow_attribute_except_filtering
49
+ FORMATS.each do |format|
50
+ @serialized = Contact.new(@contact_attributes).send("to_#{format}", :except => [ :age, :name ])
51
+ contact = Contact.new.send("from_#{format}", @serialized)
52
+ assert_nil contact.name, "For #{format}"
53
+ assert_nil contact.age, "For #{format}"
54
+ assert_equal @contact_attributes[:awesome], contact.awesome, "For #{format}"
55
+ end
56
+ end
57
+
58
+ def test_include_root_in_json_allows_inheritance
59
+ original_root_in_json = ActiveRecord::Base.include_root_in_json
60
+ ActiveRecord::Base.include_root_in_json = true
61
+
62
+ klazz = Class.new(ActiveRecord::Base)
63
+ klazz.table_name = 'topics'
64
+ assert klazz.include_root_in_json
65
+
66
+ klazz.include_root_in_json = false
67
+ assert ActiveRecord::Base.include_root_in_json
68
+ assert !klazz.include_root_in_json
69
+ assert !klazz.new.include_root_in_json
70
+ ensure
71
+ ActiveRecord::Base.include_root_in_json = original_root_in_json
72
+ end
73
+
74
+ def test_read_attribute_for_serialization_with_format_without_method_missing
75
+ klazz = Class.new(ActiveRecord::Base)
76
+ klazz.table_name = 'books'
77
+
78
+ book = klazz.new
79
+ assert_nil book.read_attribute_for_serialization(:format)
80
+ end
81
+
82
+ def test_read_attribute_for_serialization_with_format_after_init
83
+ klazz = Class.new(ActiveRecord::Base)
84
+ klazz.table_name = 'books'
85
+
86
+ book = klazz.new(format: 'paperback')
87
+ assert_equal 'paperback', book.read_attribute_for_serialization(:format)
88
+ end
89
+
90
+ def test_read_attribute_for_serialization_with_format_after_find
91
+ klazz = Class.new(ActiveRecord::Base)
92
+ klazz.table_name = 'books'
93
+
94
+ book = klazz.find(books(:awdr).id)
95
+ assert_equal 'paperback', book.read_attribute_for_serialization(:format)
96
+ end
97
+
98
+ def test_find_records_by_serialized_attributes_through_join
99
+ author = Author.create!(name: "David")
100
+ author.serialized_posts.create!(title: "Hello")
101
+
102
+ assert_equal 1, Author.joins(:serialized_posts).where(name: "David", serialized_posts: { title: "Hello" }).length
103
+ end
104
+ end
@@ -1,277 +1,364 @@
1
- require 'cases/helper'
2
- require 'models/topic'
3
- require 'models/reply'
4
- require 'models/person'
5
- require 'models/traffic_light'
6
- require 'models/post'
7
- require 'bcrypt'
8
-
9
- class SerializedAttributeTest < ActiveRecord::TestCase
10
- fixtures :topics, :posts
11
-
12
- MyObject = Struct.new :attribute1, :attribute2
13
-
14
- teardown do
15
- Topic.serialize("content")
16
- end
17
-
18
- def test_serialize_does_not_eagerly_load_columns
19
- Topic.reset_column_information
20
- assert_no_queries do
21
- Topic.serialize(:content)
22
- end
23
- end
24
-
25
- def test_list_of_serialized_attributes
26
- assert_deprecated do
27
- assert_equal %w(content), Topic.serialized_attributes.keys
28
- end
29
- end
30
-
31
- def test_serialized_attribute
32
- Topic.serialize("content", MyObject)
33
-
34
- myobj = MyObject.new('value1', 'value2')
35
- topic = Topic.create("content" => myobj)
36
- assert_equal(myobj, topic.content)
37
-
38
- topic.reload
39
- assert_equal(myobj, topic.content)
40
- end
41
-
42
- def test_serialized_attribute_in_base_class
43
- Topic.serialize("content", Hash)
44
-
45
- hash = { 'content1' => 'value1', 'content2' => 'value2' }
46
- important_topic = ImportantTopic.create("content" => hash)
47
- assert_equal(hash, important_topic.content)
48
-
49
- important_topic.reload
50
- assert_equal(hash, important_topic.content)
51
- end
52
-
53
- def test_serialized_attributes_from_database_on_subclass
54
- Topic.serialize :content, Hash
55
-
56
- t = Reply.new(content: { foo: :bar })
57
- assert_equal({ foo: :bar }, t.content)
58
- t.save!
59
- t = Reply.last
60
- assert_equal({ foo: :bar }, t.content)
61
- end
62
-
63
- def test_serialized_attribute_calling_dup_method
64
- Topic.serialize :content, JSON
65
-
66
- orig = Topic.new(content: { foo: :bar })
67
- clone = orig.dup
68
- assert_equal(orig.content, clone.content)
69
- end
70
-
71
- def test_serialized_json_attribute_returns_unserialized_value
72
- Topic.serialize :content, JSON
73
- my_post = posts(:welcome)
74
-
75
- t = Topic.new(content: my_post)
76
- t.save!
77
- t.reload
78
-
79
- assert_instance_of(Hash, t.content)
80
- assert_equal(my_post.id, t.content["id"])
81
- assert_equal(my_post.title, t.content["title"])
82
- end
83
-
84
- def test_json_read_legacy_null
85
- Topic.serialize :content, JSON
86
-
87
- # Force a row to have a JSON "null" instead of a database NULL (this is how
88
- # null values are saved on 4.1 and before)
89
- id = Topic.connection.insert "INSERT INTO topics (content) VALUES('null')"
90
- t = Topic.find(id)
91
-
92
- assert_nil t.content
93
- end
94
-
95
- def test_json_read_db_null
96
- Topic.serialize :content, JSON
97
-
98
- # Force a row to have a database NULL instead of a JSON "null"
99
- id = Topic.connection.insert "INSERT INTO topics (content) VALUES(NULL)"
100
- t = Topic.find(id)
101
-
102
- assert_nil t.content
103
- end
104
-
105
- def test_serialized_attribute_declared_in_subclass
106
- hash = { 'important1' => 'value1', 'important2' => 'value2' }
107
- important_topic = ImportantTopic.create("important" => hash)
108
- assert_equal(hash, important_topic.important)
109
-
110
- important_topic.reload
111
- assert_equal(hash, important_topic.important)
112
- assert_equal(hash, important_topic.read_attribute(:important))
113
- end
114
-
115
- def test_serialized_time_attribute
116
- myobj = Time.local(2008,1,1,1,0)
117
- topic = Topic.create("content" => myobj).reload
118
- assert_equal(myobj, topic.content)
119
- end
120
-
121
- def test_serialized_string_attribute
122
- myobj = "Yes"
123
- topic = Topic.create("content" => myobj).reload
124
- assert_equal(myobj, topic.content)
125
- end
126
-
127
- def test_nil_serialized_attribute_without_class_constraint
128
- topic = Topic.new
129
- assert_nil topic.content
130
- end
131
-
132
- def test_nil_not_serialized_without_class_constraint
133
- assert Topic.new(:content => nil).save
134
- assert_equal 1, Topic.where(:content => nil).count
135
- end
136
-
137
- def test_nil_not_serialized_with_class_constraint
138
- Topic.serialize :content, Hash
139
- assert Topic.new(:content => nil).save
140
- assert_equal 1, Topic.where(:content => nil).count
141
- end
142
-
143
- def test_serialized_attribute_should_raise_exception_on_assignment_with_wrong_type
144
- Topic.serialize(:content, Hash)
145
- assert_raise(ActiveRecord::SerializationTypeMismatch) do
146
- Topic.new(content: 'string')
147
- end
148
- end
149
-
150
- def test_should_raise_exception_on_serialized_attribute_with_type_mismatch
151
- myobj = MyObject.new('value1', 'value2')
152
- topic = Topic.new(:content => myobj)
153
- assert topic.save
154
- Topic.serialize(:content, Hash)
155
- assert_raise(ActiveRecord::SerializationTypeMismatch) { Topic.find(topic.id).content }
156
- end
157
-
158
- def test_serialized_attribute_with_class_constraint
159
- settings = { "color" => "blue" }
160
- Topic.serialize(:content, Hash)
161
- topic = Topic.new(:content => settings)
162
- assert topic.save
163
- assert_equal(settings, Topic.find(topic.id).content)
164
- end
165
-
166
- def test_serialized_default_class
167
- Topic.serialize(:content, Hash)
168
- topic = Topic.new
169
- assert_equal Hash, topic.content.class
170
- assert_equal Hash, topic.read_attribute(:content).class
171
- topic.content["beer"] = "MadridRb"
172
- assert topic.save
173
- topic.reload
174
- assert_equal Hash, topic.content.class
175
- assert_equal "MadridRb", topic.content["beer"]
176
- end
177
-
178
- def test_serialized_no_default_class_for_object
179
- topic = Topic.new
180
- assert_nil topic.content
181
- end
182
-
183
- def test_serialized_boolean_value_true
184
- topic = Topic.new(:content => true)
185
- assert topic.save
186
- topic = topic.reload
187
- assert_equal topic.content, true
188
- end
189
-
190
- def test_serialized_boolean_value_false
191
- topic = Topic.new(:content => false)
192
- assert topic.save
193
- topic = topic.reload
194
- assert_equal topic.content, false
195
- end
196
-
197
- def test_serialize_with_coder
198
- some_class = Struct.new(:foo) do
199
- def self.dump(value)
200
- value.foo
201
- end
202
-
203
- def self.load(value)
204
- new(value)
205
- end
206
- end
207
-
208
- Topic.serialize(:content, some_class)
209
- topic = Topic.new(:content => some_class.new('my value'))
210
- topic.save!
211
- topic.reload
212
- assert_kind_of some_class, topic.content
213
- assert_equal topic.content, some_class.new('my value')
214
- end
215
-
216
- def test_serialize_attribute_via_select_method_when_time_zone_available
217
- with_timezone_config aware_attributes: true do
218
- Topic.serialize(:content, MyObject)
219
-
220
- myobj = MyObject.new('value1', 'value2')
221
- topic = Topic.create(content: myobj)
222
-
223
- assert_equal(myobj, Topic.select(:content).find(topic.id).content)
224
- assert_raise(ActiveModel::MissingAttributeError) { Topic.select(:id).find(topic.id).content }
225
- end
226
- end
227
-
228
- def test_serialize_attribute_can_be_serialized_in_an_integer_column
229
- insures = ['life']
230
- person = SerializedPerson.new(first_name: 'David', insures: insures)
231
- assert person.save
232
- person = person.reload
233
- assert_equal(insures, person.insures)
234
- end
235
-
236
- def test_regression_serialized_default_on_text_column_with_null_false
237
- light = TrafficLight.new
238
- assert_equal [], light.state
239
- assert_equal [], light.long_state
240
- end
241
-
242
- def test_serialized_column_should_unserialize_after_update_column
243
- t = Topic.create(content: "first")
244
- assert_equal("first", t.content)
245
-
246
- t.update_column(:content, ["second"])
247
- assert_equal(["second"], t.content)
248
- assert_equal(["second"], t.reload.content)
249
- end
250
-
251
- def test_serialized_column_should_unserialize_after_update_attribute
252
- t = Topic.create(content: "first")
253
- assert_equal("first", t.content)
254
-
255
- t.update_attribute(:content, "second")
256
- assert_equal("second", t.content)
257
- assert_equal("second", t.reload.content)
258
- end
259
-
260
- def test_nil_is_not_changed_when_serialized_with_a_class
261
- Topic.serialize(:content, Array)
262
-
263
- topic = Topic.new(content: nil)
264
-
265
- assert_not topic.content_changed?
266
- end
267
-
268
- def test_newly_emptied_serialized_hash_is_changed
269
- Topic.serialize(:content, Hash)
270
- topic = Topic.create(content: { "things" => "stuff" })
271
- topic.content.delete("things")
272
- topic.save!
273
- topic.reload
274
-
275
- assert_equal({}, topic.content)
276
- end
277
- end
1
+ require 'cases/helper'
2
+ require 'models/topic'
3
+ require 'models/reply'
4
+ require 'models/person'
5
+ require 'models/traffic_light'
6
+ require 'models/post'
7
+ require 'bcrypt'
8
+
9
+ class SerializedAttributeTest < ActiveRecord::TestCase
10
+ fixtures :topics, :posts
11
+
12
+ MyObject = Struct.new :attribute1, :attribute2
13
+
14
+ teardown do
15
+ Topic.serialize("content")
16
+ end
17
+
18
+ def test_serialize_does_not_eagerly_load_columns
19
+ Topic.reset_column_information
20
+ assert_no_queries do
21
+ Topic.serialize(:content)
22
+ end
23
+ end
24
+
25
+ def test_serialized_attribute
26
+ Topic.serialize("content", MyObject)
27
+
28
+ myobj = MyObject.new('value1', 'value2')
29
+ topic = Topic.create("content" => myobj)
30
+ assert_equal(myobj, topic.content)
31
+
32
+ topic.reload
33
+ assert_equal(myobj, topic.content)
34
+ end
35
+
36
+ def test_serialized_attribute_in_base_class
37
+ Topic.serialize("content", Hash)
38
+
39
+ hash = { 'content1' => 'value1', 'content2' => 'value2' }
40
+ important_topic = ImportantTopic.create("content" => hash)
41
+ assert_equal(hash, important_topic.content)
42
+
43
+ important_topic.reload
44
+ assert_equal(hash, important_topic.content)
45
+ end
46
+
47
+ def test_serialized_attributes_from_database_on_subclass
48
+ Topic.serialize :content, Hash
49
+
50
+ t = Reply.new(content: { foo: :bar })
51
+ assert_equal({ foo: :bar }, t.content)
52
+ t.save!
53
+ t = Reply.last
54
+ assert_equal({ foo: :bar }, t.content)
55
+ end
56
+
57
+ def test_serialized_attribute_calling_dup_method
58
+ Topic.serialize :content, JSON
59
+
60
+ orig = Topic.new(content: { foo: :bar })
61
+ clone = orig.dup
62
+ assert_equal(orig.content, clone.content)
63
+ end
64
+
65
+ def test_serialized_json_attribute_returns_unserialized_value
66
+ Topic.serialize :content, JSON
67
+ my_post = posts(:welcome)
68
+
69
+ t = Topic.new(content: my_post)
70
+ t.save!
71
+ t.reload
72
+
73
+ assert_instance_of(Hash, t.content)
74
+ assert_equal(my_post.id, t.content["id"])
75
+ assert_equal(my_post.title, t.content["title"])
76
+ end
77
+
78
+ def test_json_read_legacy_null
79
+ Topic.serialize :content, JSON
80
+
81
+ # Force a row to have a JSON "null" instead of a database NULL (this is how
82
+ # null values are saved on 4.1 and before)
83
+ id = Topic.connection.insert "INSERT INTO topics (content) VALUES('null')"
84
+ t = Topic.find(id)
85
+
86
+ assert_nil t.content
87
+ end
88
+
89
+ def test_json_read_db_null
90
+ Topic.serialize :content, JSON
91
+
92
+ # Force a row to have a database NULL instead of a JSON "null"
93
+ id = Topic.connection.insert "INSERT INTO topics (content) VALUES(NULL)"
94
+ t = Topic.find(id)
95
+
96
+ assert_nil t.content
97
+ end
98
+
99
+ def test_serialized_attribute_declared_in_subclass
100
+ hash = { 'important1' => 'value1', 'important2' => 'value2' }
101
+ important_topic = ImportantTopic.create("important" => hash)
102
+ assert_equal(hash, important_topic.important)
103
+
104
+ important_topic.reload
105
+ assert_equal(hash, important_topic.important)
106
+ assert_equal(hash, important_topic.read_attribute(:important))
107
+ end
108
+
109
+ def test_serialized_time_attribute
110
+ myobj = Time.local(2008,1,1,1,0)
111
+ topic = Topic.create("content" => myobj).reload
112
+ assert_equal(myobj, topic.content)
113
+ end
114
+
115
+ def test_serialized_string_attribute
116
+ myobj = "Yes"
117
+ topic = Topic.create("content" => myobj).reload
118
+ assert_equal(myobj, topic.content)
119
+ end
120
+
121
+ def test_nil_serialized_attribute_without_class_constraint
122
+ topic = Topic.new
123
+ assert_nil topic.content
124
+ end
125
+
126
+ def test_nil_not_serialized_without_class_constraint
127
+ assert Topic.new(:content => nil).save
128
+ assert_equal 1, Topic.where(:content => nil).count
129
+ end
130
+
131
+ def test_nil_not_serialized_with_class_constraint
132
+ Topic.serialize :content, Hash
133
+ assert Topic.new(:content => nil).save
134
+ assert_equal 1, Topic.where(:content => nil).count
135
+ end
136
+
137
+ def test_serialized_attribute_should_raise_exception_on_assignment_with_wrong_type
138
+ Topic.serialize(:content, Hash)
139
+ assert_raise(ActiveRecord::SerializationTypeMismatch) do
140
+ Topic.new(content: 'string')
141
+ end
142
+ end
143
+
144
+ def test_should_raise_exception_on_serialized_attribute_with_type_mismatch
145
+ myobj = MyObject.new('value1', 'value2')
146
+ topic = Topic.new(:content => myobj)
147
+ assert topic.save
148
+ Topic.serialize(:content, Hash)
149
+ assert_raise(ActiveRecord::SerializationTypeMismatch) { Topic.find(topic.id).content }
150
+ end
151
+
152
+ def test_serialized_attribute_with_class_constraint
153
+ settings = { "color" => "blue" }
154
+ Topic.serialize(:content, Hash)
155
+ topic = Topic.new(:content => settings)
156
+ assert topic.save
157
+ assert_equal(settings, Topic.find(topic.id).content)
158
+ end
159
+
160
+ def test_serialized_default_class
161
+ Topic.serialize(:content, Hash)
162
+ topic = Topic.new
163
+ assert_equal Hash, topic.content.class
164
+ assert_equal Hash, topic.read_attribute(:content).class
165
+ topic.content["beer"] = "MadridRb"
166
+ assert topic.save
167
+ topic.reload
168
+ assert_equal Hash, topic.content.class
169
+ assert_equal "MadridRb", topic.content["beer"]
170
+ end
171
+
172
+ def test_serialized_no_default_class_for_object
173
+ topic = Topic.new
174
+ assert_nil topic.content
175
+ end
176
+
177
+ def test_serialized_boolean_value_true
178
+ topic = Topic.new(:content => true)
179
+ assert topic.save
180
+ topic = topic.reload
181
+ assert_equal topic.content, true
182
+ end
183
+
184
+ def test_serialized_boolean_value_false
185
+ topic = Topic.new(:content => false)
186
+ assert topic.save
187
+ topic = topic.reload
188
+ assert_equal topic.content, false
189
+ end
190
+
191
+ def test_serialize_with_coder
192
+ some_class = Struct.new(:foo) do
193
+ def self.dump(value)
194
+ value.foo
195
+ end
196
+
197
+ def self.load(value)
198
+ new(value)
199
+ end
200
+ end
201
+
202
+ Topic.serialize(:content, some_class)
203
+ topic = Topic.new(:content => some_class.new('my value'))
204
+ topic.save!
205
+ topic.reload
206
+ assert_kind_of some_class, topic.content
207
+ assert_equal topic.content, some_class.new('my value')
208
+ end
209
+
210
+ def test_serialize_attribute_via_select_method_when_time_zone_available
211
+ with_timezone_config aware_attributes: true do
212
+ Topic.serialize(:content, MyObject)
213
+
214
+ myobj = MyObject.new('value1', 'value2')
215
+ topic = Topic.create(content: myobj)
216
+
217
+ assert_equal(myobj, Topic.select(:content).find(topic.id).content)
218
+ assert_raise(ActiveModel::MissingAttributeError) { Topic.select(:id).find(topic.id).content }
219
+ end
220
+ end
221
+
222
+ def test_serialize_attribute_can_be_serialized_in_an_integer_column
223
+ insures = ['life']
224
+ person = SerializedPerson.new(first_name: 'David', insures: insures)
225
+ assert person.save
226
+ person = person.reload
227
+ assert_equal(insures, person.insures)
228
+ end
229
+
230
+ def test_regression_serialized_default_on_text_column_with_null_false
231
+ light = TrafficLight.new
232
+ assert_equal [], light.state
233
+ assert_equal [], light.long_state
234
+ end
235
+
236
+ def test_serialized_column_should_unserialize_after_update_column
237
+ t = Topic.create(content: "first")
238
+ assert_equal("first", t.content)
239
+
240
+ t.update_column(:content, ["second"])
241
+ assert_equal(["second"], t.content)
242
+ assert_equal(["second"], t.reload.content)
243
+ end
244
+
245
+ def test_serialized_column_should_unserialize_after_update_attribute
246
+ t = Topic.create(content: "first")
247
+ assert_equal("first", t.content)
248
+
249
+ t.update_attribute(:content, "second")
250
+ assert_equal("second", t.content)
251
+ assert_equal("second", t.reload.content)
252
+ end
253
+
254
+ def test_nil_is_not_changed_when_serialized_with_a_class
255
+ Topic.serialize(:content, Array)
256
+
257
+ topic = Topic.new(content: nil)
258
+
259
+ assert_not topic.content_changed?
260
+ end
261
+
262
+ def test_classes_without_no_arg_constructors_are_not_supported
263
+ assert_raises(ArgumentError) do
264
+ Topic.serialize(:content, Regexp)
265
+ end
266
+ end
267
+
268
+ def test_newly_emptied_serialized_hash_is_changed
269
+ Topic.serialize(:content, Hash)
270
+ topic = Topic.create(content: { "things" => "stuff" })
271
+ topic.content.delete("things")
272
+ topic.save!
273
+ topic.reload
274
+
275
+ assert_equal({}, topic.content)
276
+ end
277
+
278
+ def test_values_cast_from_nil_are_persisted_as_nil
279
+ # This is required to fulfil the following contract, which must be universally
280
+ # true in Active Record:
281
+ #
282
+ # model.attribute = value
283
+ # assert_equal model.attribute, model.tap(&:save).reload.attribute
284
+ Topic.serialize(:content, Hash)
285
+ topic = Topic.create!(content: {})
286
+ topic2 = Topic.create!(content: nil)
287
+
288
+ assert_equal [topic, topic2], Topic.where(content: nil)
289
+ end
290
+
291
+ def test_nil_is_always_persisted_as_null
292
+ Topic.serialize(:content, Hash)
293
+
294
+ topic = Topic.create!(content: { foo: "bar" })
295
+ topic.update_attribute :content, nil
296
+ assert_equal [topic], Topic.where(content: nil)
297
+ end
298
+
299
+ def test_mutation_detection_does_not_double_serialize
300
+ coder = Object.new
301
+ def coder.dump(value)
302
+ return if value.nil?
303
+ value + " encoded"
304
+ end
305
+ def coder.load(value)
306
+ return if value.nil?
307
+ value.gsub(" encoded", "")
308
+ end
309
+ type = Class.new(ActiveModel::Type::Value) do
310
+ include ActiveModel::Type::Helpers::Mutable
311
+
312
+ def serialize(value)
313
+ return if value.nil?
314
+ value + " serialized"
315
+ end
316
+
317
+ def deserialize(value)
318
+ return if value.nil?
319
+ value.gsub(" serialized", "")
320
+ end
321
+ end.new
322
+ model = Class.new(Topic) do
323
+ attribute :foo, type
324
+ serialize :foo, coder
325
+ end
326
+
327
+ topic = model.create!(foo: "bar")
328
+ topic.foo
329
+ refute topic.changed?
330
+ end
331
+ end
332
+
333
+ class ThreadedSerializedAttributeTest < ActiveRecord::TestCase
334
+ self.use_transactional_tests = false
335
+ fixtures :topics
336
+
337
+ def test_serialized_attribute_works_under_concurrent_initial_access
338
+ model = Topic.dup
339
+
340
+ topic = model.last
341
+ topic.update group: "1"
342
+
343
+ model.serialize :group, JSON
344
+ model.reset_column_information
345
+
346
+ # This isn't strictly necessary for the test, but a little bit of
347
+ # knowledge of internals allows us to make failures far more likely.
348
+ model.define_singleton_method(:define_attribute) do |*args|
349
+ Thread.pass
350
+ super(*args)
351
+ end
352
+
353
+ threads = 4.times.map do
354
+ Thread.new do
355
+ topic.reload.group
356
+ end
357
+ end
358
+
359
+ # All the threads should retrieve the value knowing it is JSON, and
360
+ # thus decode it. If this fails, some threads will instead see the
361
+ # raw string ("1"), or raise an exception.
362
+ assert_equal [1] * threads.size, threads.map(&:value)
363
+ end
364
+ end