activerecord_authorails 1.0.0

Sign up to get free protection for your applications and to get access to all the features.
Files changed (270) hide show
  1. data/CHANGELOG +3043 -0
  2. data/README +360 -0
  3. data/RUNNING_UNIT_TESTS +64 -0
  4. data/Rakefile +226 -0
  5. data/examples/associations.png +0 -0
  6. data/examples/associations.rb +87 -0
  7. data/examples/shared_setup.rb +15 -0
  8. data/examples/validation.rb +85 -0
  9. data/install.rb +30 -0
  10. data/lib/active_record.rb +85 -0
  11. data/lib/active_record/acts/list.rb +244 -0
  12. data/lib/active_record/acts/nested_set.rb +211 -0
  13. data/lib/active_record/acts/tree.rb +89 -0
  14. data/lib/active_record/aggregations.rb +191 -0
  15. data/lib/active_record/associations.rb +1637 -0
  16. data/lib/active_record/associations/association_collection.rb +190 -0
  17. data/lib/active_record/associations/association_proxy.rb +158 -0
  18. data/lib/active_record/associations/belongs_to_association.rb +56 -0
  19. data/lib/active_record/associations/belongs_to_polymorphic_association.rb +50 -0
  20. data/lib/active_record/associations/has_and_belongs_to_many_association.rb +169 -0
  21. data/lib/active_record/associations/has_many_association.rb +210 -0
  22. data/lib/active_record/associations/has_many_through_association.rb +247 -0
  23. data/lib/active_record/associations/has_one_association.rb +80 -0
  24. data/lib/active_record/attribute_methods.rb +75 -0
  25. data/lib/active_record/base.rb +2164 -0
  26. data/lib/active_record/calculations.rb +270 -0
  27. data/lib/active_record/callbacks.rb +367 -0
  28. data/lib/active_record/connection_adapters/abstract/connection_specification.rb +279 -0
  29. data/lib/active_record/connection_adapters/abstract/database_statements.rb +130 -0
  30. data/lib/active_record/connection_adapters/abstract/quoting.rb +58 -0
  31. data/lib/active_record/connection_adapters/abstract/schema_definitions.rb +343 -0
  32. data/lib/active_record/connection_adapters/abstract/schema_statements.rb +310 -0
  33. data/lib/active_record/connection_adapters/abstract_adapter.rb +161 -0
  34. data/lib/active_record/connection_adapters/db2_adapter.rb +228 -0
  35. data/lib/active_record/connection_adapters/firebird_adapter.rb +728 -0
  36. data/lib/active_record/connection_adapters/frontbase_adapter.rb +861 -0
  37. data/lib/active_record/connection_adapters/mysql_adapter.rb +414 -0
  38. data/lib/active_record/connection_adapters/openbase_adapter.rb +350 -0
  39. data/lib/active_record/connection_adapters/oracle_adapter.rb +689 -0
  40. data/lib/active_record/connection_adapters/postgresql_adapter.rb +584 -0
  41. data/lib/active_record/connection_adapters/sqlite_adapter.rb +407 -0
  42. data/lib/active_record/connection_adapters/sqlserver_adapter.rb +591 -0
  43. data/lib/active_record/connection_adapters/sybase_adapter.rb +662 -0
  44. data/lib/active_record/deprecated_associations.rb +104 -0
  45. data/lib/active_record/deprecated_finders.rb +44 -0
  46. data/lib/active_record/fixtures.rb +628 -0
  47. data/lib/active_record/locking/optimistic.rb +106 -0
  48. data/lib/active_record/locking/pessimistic.rb +77 -0
  49. data/lib/active_record/migration.rb +394 -0
  50. data/lib/active_record/observer.rb +178 -0
  51. data/lib/active_record/query_cache.rb +64 -0
  52. data/lib/active_record/reflection.rb +222 -0
  53. data/lib/active_record/schema.rb +58 -0
  54. data/lib/active_record/schema_dumper.rb +149 -0
  55. data/lib/active_record/timestamp.rb +51 -0
  56. data/lib/active_record/transactions.rb +136 -0
  57. data/lib/active_record/validations.rb +843 -0
  58. data/lib/active_record/vendor/db2.rb +362 -0
  59. data/lib/active_record/vendor/mysql.rb +1214 -0
  60. data/lib/active_record/vendor/simple.rb +693 -0
  61. data/lib/active_record/version.rb +9 -0
  62. data/lib/active_record/wrappers/yaml_wrapper.rb +15 -0
  63. data/lib/active_record/wrappings.rb +58 -0
  64. data/lib/active_record/xml_serialization.rb +308 -0
  65. data/test/aaa_create_tables_test.rb +59 -0
  66. data/test/abstract_unit.rb +77 -0
  67. data/test/active_schema_test_mysql.rb +31 -0
  68. data/test/adapter_test.rb +87 -0
  69. data/test/adapter_test_sqlserver.rb +81 -0
  70. data/test/aggregations_test.rb +95 -0
  71. data/test/all.sh +8 -0
  72. data/test/ar_schema_test.rb +33 -0
  73. data/test/association_inheritance_reload.rb +14 -0
  74. data/test/associations/callbacks_test.rb +126 -0
  75. data/test/associations/cascaded_eager_loading_test.rb +138 -0
  76. data/test/associations/eager_test.rb +393 -0
  77. data/test/associations/extension_test.rb +42 -0
  78. data/test/associations/join_model_test.rb +497 -0
  79. data/test/associations_test.rb +1809 -0
  80. data/test/attribute_methods_test.rb +49 -0
  81. data/test/base_test.rb +1586 -0
  82. data/test/binary_test.rb +37 -0
  83. data/test/calculations_test.rb +219 -0
  84. data/test/callbacks_test.rb +377 -0
  85. data/test/class_inheritable_attributes_test.rb +32 -0
  86. data/test/column_alias_test.rb +17 -0
  87. data/test/connection_test_firebird.rb +8 -0
  88. data/test/connections/native_db2/connection.rb +25 -0
  89. data/test/connections/native_firebird/connection.rb +26 -0
  90. data/test/connections/native_frontbase/connection.rb +27 -0
  91. data/test/connections/native_mysql/connection.rb +24 -0
  92. data/test/connections/native_openbase/connection.rb +21 -0
  93. data/test/connections/native_oracle/connection.rb +27 -0
  94. data/test/connections/native_postgresql/connection.rb +23 -0
  95. data/test/connections/native_sqlite/connection.rb +34 -0
  96. data/test/connections/native_sqlite3/connection.rb +34 -0
  97. data/test/connections/native_sqlite3/in_memory_connection.rb +18 -0
  98. data/test/connections/native_sqlserver/connection.rb +23 -0
  99. data/test/connections/native_sqlserver_odbc/connection.rb +25 -0
  100. data/test/connections/native_sybase/connection.rb +23 -0
  101. data/test/copy_table_sqlite.rb +64 -0
  102. data/test/datatype_test_postgresql.rb +52 -0
  103. data/test/default_test_firebird.rb +16 -0
  104. data/test/defaults_test.rb +60 -0
  105. data/test/deprecated_associations_test.rb +396 -0
  106. data/test/deprecated_finder_test.rb +151 -0
  107. data/test/empty_date_time_test.rb +25 -0
  108. data/test/finder_test.rb +504 -0
  109. data/test/fixtures/accounts.yml +28 -0
  110. data/test/fixtures/author.rb +99 -0
  111. data/test/fixtures/author_favorites.yml +4 -0
  112. data/test/fixtures/authors.yml +7 -0
  113. data/test/fixtures/auto_id.rb +4 -0
  114. data/test/fixtures/bad_fixtures/attr_with_numeric_first_char +1 -0
  115. data/test/fixtures/bad_fixtures/attr_with_spaces +1 -0
  116. data/test/fixtures/bad_fixtures/blank_line +3 -0
  117. data/test/fixtures/bad_fixtures/duplicate_attributes +3 -0
  118. data/test/fixtures/bad_fixtures/missing_value +1 -0
  119. data/test/fixtures/binary.rb +2 -0
  120. data/test/fixtures/categories.yml +14 -0
  121. data/test/fixtures/categories/special_categories.yml +9 -0
  122. data/test/fixtures/categories/subsubdir/arbitrary_filename.yml +4 -0
  123. data/test/fixtures/categories_ordered.yml +7 -0
  124. data/test/fixtures/categories_posts.yml +23 -0
  125. data/test/fixtures/categorization.rb +5 -0
  126. data/test/fixtures/categorizations.yml +17 -0
  127. data/test/fixtures/category.rb +20 -0
  128. data/test/fixtures/column_name.rb +3 -0
  129. data/test/fixtures/comment.rb +23 -0
  130. data/test/fixtures/comments.yml +59 -0
  131. data/test/fixtures/companies.yml +55 -0
  132. data/test/fixtures/company.rb +107 -0
  133. data/test/fixtures/company_in_module.rb +59 -0
  134. data/test/fixtures/computer.rb +3 -0
  135. data/test/fixtures/computers.yml +4 -0
  136. data/test/fixtures/course.rb +3 -0
  137. data/test/fixtures/courses.yml +7 -0
  138. data/test/fixtures/customer.rb +55 -0
  139. data/test/fixtures/customers.yml +17 -0
  140. data/test/fixtures/db_definitions/db2.drop.sql +32 -0
  141. data/test/fixtures/db_definitions/db2.sql +231 -0
  142. data/test/fixtures/db_definitions/db22.drop.sql +2 -0
  143. data/test/fixtures/db_definitions/db22.sql +5 -0
  144. data/test/fixtures/db_definitions/firebird.drop.sql +63 -0
  145. data/test/fixtures/db_definitions/firebird.sql +304 -0
  146. data/test/fixtures/db_definitions/firebird2.drop.sql +2 -0
  147. data/test/fixtures/db_definitions/firebird2.sql +6 -0
  148. data/test/fixtures/db_definitions/frontbase.drop.sql +32 -0
  149. data/test/fixtures/db_definitions/frontbase.sql +268 -0
  150. data/test/fixtures/db_definitions/frontbase2.drop.sql +1 -0
  151. data/test/fixtures/db_definitions/frontbase2.sql +4 -0
  152. data/test/fixtures/db_definitions/mysql.drop.sql +32 -0
  153. data/test/fixtures/db_definitions/mysql.sql +234 -0
  154. data/test/fixtures/db_definitions/mysql2.drop.sql +2 -0
  155. data/test/fixtures/db_definitions/mysql2.sql +5 -0
  156. data/test/fixtures/db_definitions/openbase.drop.sql +2 -0
  157. data/test/fixtures/db_definitions/openbase.sql +302 -0
  158. data/test/fixtures/db_definitions/openbase2.drop.sql +2 -0
  159. data/test/fixtures/db_definitions/openbase2.sql +7 -0
  160. data/test/fixtures/db_definitions/oracle.drop.sql +65 -0
  161. data/test/fixtures/db_definitions/oracle.sql +325 -0
  162. data/test/fixtures/db_definitions/oracle2.drop.sql +2 -0
  163. data/test/fixtures/db_definitions/oracle2.sql +6 -0
  164. data/test/fixtures/db_definitions/postgresql.drop.sql +37 -0
  165. data/test/fixtures/db_definitions/postgresql.sql +263 -0
  166. data/test/fixtures/db_definitions/postgresql2.drop.sql +2 -0
  167. data/test/fixtures/db_definitions/postgresql2.sql +5 -0
  168. data/test/fixtures/db_definitions/schema.rb +60 -0
  169. data/test/fixtures/db_definitions/sqlite.drop.sql +32 -0
  170. data/test/fixtures/db_definitions/sqlite.sql +215 -0
  171. data/test/fixtures/db_definitions/sqlite2.drop.sql +2 -0
  172. data/test/fixtures/db_definitions/sqlite2.sql +5 -0
  173. data/test/fixtures/db_definitions/sqlserver.drop.sql +34 -0
  174. data/test/fixtures/db_definitions/sqlserver.sql +243 -0
  175. data/test/fixtures/db_definitions/sqlserver2.drop.sql +2 -0
  176. data/test/fixtures/db_definitions/sqlserver2.sql +5 -0
  177. data/test/fixtures/db_definitions/sybase.drop.sql +34 -0
  178. data/test/fixtures/db_definitions/sybase.sql +218 -0
  179. data/test/fixtures/db_definitions/sybase2.drop.sql +4 -0
  180. data/test/fixtures/db_definitions/sybase2.sql +5 -0
  181. data/test/fixtures/default.rb +2 -0
  182. data/test/fixtures/developer.rb +52 -0
  183. data/test/fixtures/developers.yml +21 -0
  184. data/test/fixtures/developers_projects.yml +17 -0
  185. data/test/fixtures/developers_projects/david_action_controller +3 -0
  186. data/test/fixtures/developers_projects/david_active_record +3 -0
  187. data/test/fixtures/developers_projects/jamis_active_record +2 -0
  188. data/test/fixtures/edge.rb +5 -0
  189. data/test/fixtures/edges.yml +6 -0
  190. data/test/fixtures/entrant.rb +3 -0
  191. data/test/fixtures/entrants.yml +14 -0
  192. data/test/fixtures/fk_test_has_fk.yml +3 -0
  193. data/test/fixtures/fk_test_has_pk.yml +2 -0
  194. data/test/fixtures/flowers.jpg +0 -0
  195. data/test/fixtures/funny_jokes.yml +10 -0
  196. data/test/fixtures/joke.rb +6 -0
  197. data/test/fixtures/keyboard.rb +3 -0
  198. data/test/fixtures/legacy_thing.rb +3 -0
  199. data/test/fixtures/legacy_things.yml +3 -0
  200. data/test/fixtures/migrations/1_people_have_last_names.rb +9 -0
  201. data/test/fixtures/migrations/2_we_need_reminders.rb +12 -0
  202. data/test/fixtures/migrations/3_innocent_jointable.rb +12 -0
  203. data/test/fixtures/migrations_with_decimal/1_give_me_big_numbers.rb +15 -0
  204. data/test/fixtures/migrations_with_duplicate/1_people_have_last_names.rb +9 -0
  205. data/test/fixtures/migrations_with_duplicate/2_we_need_reminders.rb +12 -0
  206. data/test/fixtures/migrations_with_duplicate/3_foo.rb +7 -0
  207. data/test/fixtures/migrations_with_duplicate/3_innocent_jointable.rb +12 -0
  208. data/test/fixtures/migrations_with_missing_versions/1000_people_have_middle_names.rb +9 -0
  209. data/test/fixtures/migrations_with_missing_versions/1_people_have_last_names.rb +9 -0
  210. data/test/fixtures/migrations_with_missing_versions/3_we_need_reminders.rb +12 -0
  211. data/test/fixtures/migrations_with_missing_versions/4_innocent_jointable.rb +12 -0
  212. data/test/fixtures/mixed_case_monkey.rb +3 -0
  213. data/test/fixtures/mixed_case_monkeys.yml +6 -0
  214. data/test/fixtures/mixin.rb +63 -0
  215. data/test/fixtures/mixins.yml +127 -0
  216. data/test/fixtures/movie.rb +5 -0
  217. data/test/fixtures/movies.yml +7 -0
  218. data/test/fixtures/naked/csv/accounts.csv +1 -0
  219. data/test/fixtures/naked/yml/accounts.yml +1 -0
  220. data/test/fixtures/naked/yml/companies.yml +1 -0
  221. data/test/fixtures/naked/yml/courses.yml +1 -0
  222. data/test/fixtures/order.rb +4 -0
  223. data/test/fixtures/people.yml +3 -0
  224. data/test/fixtures/person.rb +4 -0
  225. data/test/fixtures/post.rb +58 -0
  226. data/test/fixtures/posts.yml +48 -0
  227. data/test/fixtures/project.rb +27 -0
  228. data/test/fixtures/projects.yml +7 -0
  229. data/test/fixtures/reader.rb +4 -0
  230. data/test/fixtures/readers.yml +4 -0
  231. data/test/fixtures/reply.rb +37 -0
  232. data/test/fixtures/subject.rb +4 -0
  233. data/test/fixtures/subscriber.rb +6 -0
  234. data/test/fixtures/subscribers/first +2 -0
  235. data/test/fixtures/subscribers/second +2 -0
  236. data/test/fixtures/tag.rb +7 -0
  237. data/test/fixtures/tagging.rb +6 -0
  238. data/test/fixtures/taggings.yml +18 -0
  239. data/test/fixtures/tags.yml +7 -0
  240. data/test/fixtures/task.rb +3 -0
  241. data/test/fixtures/tasks.yml +7 -0
  242. data/test/fixtures/topic.rb +25 -0
  243. data/test/fixtures/topics.yml +22 -0
  244. data/test/fixtures/vertex.rb +9 -0
  245. data/test/fixtures/vertices.yml +4 -0
  246. data/test/fixtures_test.rb +401 -0
  247. data/test/inheritance_test.rb +205 -0
  248. data/test/lifecycle_test.rb +137 -0
  249. data/test/locking_test.rb +190 -0
  250. data/test/method_scoping_test.rb +416 -0
  251. data/test/migration_test.rb +768 -0
  252. data/test/migration_test_firebird.rb +124 -0
  253. data/test/mixin_nested_set_test.rb +196 -0
  254. data/test/mixin_test.rb +550 -0
  255. data/test/modules_test.rb +34 -0
  256. data/test/multiple_db_test.rb +60 -0
  257. data/test/pk_test.rb +104 -0
  258. data/test/readonly_test.rb +107 -0
  259. data/test/reflection_test.rb +159 -0
  260. data/test/schema_authorization_test_postgresql.rb +75 -0
  261. data/test/schema_dumper_test.rb +96 -0
  262. data/test/schema_test_postgresql.rb +64 -0
  263. data/test/synonym_test_oracle.rb +17 -0
  264. data/test/table_name_test_sqlserver.rb +23 -0
  265. data/test/threaded_connections_test.rb +48 -0
  266. data/test/transactions_test.rb +230 -0
  267. data/test/unconnected_test.rb +32 -0
  268. data/test/validations_test.rb +1097 -0
  269. data/test/xml_serialization_test.rb +125 -0
  270. metadata +365 -0
@@ -0,0 +1,49 @@
1
+ require 'abstract_unit'
2
+
3
+ class AttributeMethodsTest < Test::Unit::TestCase
4
+ def setup
5
+ @old_suffixes = ActiveRecord::Base.send(:attribute_method_suffixes).dup
6
+ @target = Class.new(ActiveRecord::Base)
7
+ @target.table_name = 'topics'
8
+ end
9
+
10
+ def teardown
11
+ ActiveRecord::Base.send(:attribute_method_suffixes).clear
12
+ ActiveRecord::Base.attribute_method_suffix *@old_suffixes
13
+ end
14
+
15
+
16
+ def test_match_attribute_method_query_returns_match_data
17
+ assert_not_nil md = @target.match_attribute_method?('title=')
18
+ assert_equal 'title', md.pre_match
19
+ assert_equal ['='], md.captures
20
+
21
+ %w(_hello_world ist! _maybe?).each do |suffix|
22
+ @target.class_eval "def attribute#{suffix}(*args) args end"
23
+ @target.attribute_method_suffix suffix
24
+
25
+ assert_not_nil md = @target.match_attribute_method?("title#{suffix}")
26
+ assert_equal 'title', md.pre_match
27
+ assert_equal [suffix], md.captures
28
+ end
29
+ end
30
+
31
+ def test_declared_attribute_method_affects_respond_to_and_method_missing
32
+ topic = @target.new(:title => 'Budget')
33
+ assert topic.respond_to?('title')
34
+ assert_equal 'Budget', topic.title
35
+ assert !topic.respond_to?('title_hello_world')
36
+ assert_raise(NoMethodError) { topic.title_hello_world }
37
+
38
+ %w(_hello_world _it! _candidate= able?).each do |suffix|
39
+ @target.class_eval "def attribute#{suffix}(*args) args end"
40
+ @target.attribute_method_suffix suffix
41
+
42
+ meth = "title#{suffix}"
43
+ assert topic.respond_to?(meth)
44
+ assert_equal ['title'], topic.send(meth)
45
+ assert_equal ['title', 'a'], topic.send(meth, 'a')
46
+ assert_equal ['title', 1, 2, 3], topic.send(meth, 1, 2, 3)
47
+ end
48
+ end
49
+ end
@@ -0,0 +1,1586 @@
1
+ require 'abstract_unit'
2
+ require 'fixtures/topic'
3
+ require 'fixtures/reply'
4
+ require 'fixtures/company'
5
+ require 'fixtures/customer'
6
+ require 'fixtures/developer'
7
+ require 'fixtures/project'
8
+ require 'fixtures/default'
9
+ require 'fixtures/auto_id'
10
+ require 'fixtures/column_name'
11
+ require 'fixtures/subscriber'
12
+ require 'fixtures/keyboard'
13
+ require 'fixtures/post'
14
+
15
+ class Category < ActiveRecord::Base; end
16
+ class Smarts < ActiveRecord::Base; end
17
+ class CreditCard < ActiveRecord::Base
18
+ class PinNumber < ActiveRecord::Base
19
+ class CvvCode < ActiveRecord::Base; end
20
+ class SubCvvCode < CvvCode; end
21
+ end
22
+ class SubPinNumber < PinNumber; end
23
+ class Brand < Category; end
24
+ end
25
+ class MasterCreditCard < ActiveRecord::Base; end
26
+ class Post < ActiveRecord::Base; end
27
+ class Computer < ActiveRecord::Base; end
28
+ class NonExistentTable < ActiveRecord::Base; end
29
+ class TestOracleDefault < ActiveRecord::Base; end
30
+
31
+ class LoosePerson < ActiveRecord::Base
32
+ self.table_name = 'people'
33
+ self.abstract_class = true
34
+ attr_protected :credit_rating, :administrator
35
+ end
36
+
37
+ class LooseDescendant < LoosePerson
38
+ attr_protected :phone_number
39
+ end
40
+
41
+ class TightPerson < ActiveRecord::Base
42
+ self.table_name = 'people'
43
+ attr_accessible :name, :address
44
+ end
45
+
46
+ class TightDescendant < TightPerson
47
+ attr_accessible :phone_number
48
+ end
49
+
50
+ class Booleantest < ActiveRecord::Base; end
51
+
52
+ class Task < ActiveRecord::Base
53
+ attr_protected :starting
54
+ end
55
+
56
+ class BasicsTest < Test::Unit::TestCase
57
+ fixtures :topics, :companies, :developers, :projects, :computers, :accounts
58
+
59
+ def test_table_exists
60
+ assert !NonExistentTable.table_exists?
61
+ assert Topic.table_exists?
62
+ end
63
+
64
+ def test_set_attributes
65
+ topic = Topic.find(1)
66
+ topic.attributes = { "title" => "Budget", "author_name" => "Jason" }
67
+ topic.save
68
+ assert_equal("Budget", topic.title)
69
+ assert_equal("Jason", topic.author_name)
70
+ assert_equal(topics(:first).author_email_address, Topic.find(1).author_email_address)
71
+ end
72
+
73
+ def test_integers_as_nil
74
+ test = AutoId.create('value' => '')
75
+ assert_nil AutoId.find(test.id).value
76
+ end
77
+
78
+ def test_set_attributes_with_block
79
+ topic = Topic.new do |t|
80
+ t.title = "Budget"
81
+ t.author_name = "Jason"
82
+ end
83
+
84
+ assert_equal("Budget", topic.title)
85
+ assert_equal("Jason", topic.author_name)
86
+ end
87
+
88
+ def test_respond_to?
89
+ topic = Topic.find(1)
90
+ assert topic.respond_to?("title")
91
+ assert topic.respond_to?("title?")
92
+ assert topic.respond_to?("title=")
93
+ assert topic.respond_to?(:title)
94
+ assert topic.respond_to?(:title?)
95
+ assert topic.respond_to?(:title=)
96
+ assert topic.respond_to?("author_name")
97
+ assert topic.respond_to?("attribute_names")
98
+ assert !topic.respond_to?("nothingness")
99
+ assert !topic.respond_to?(:nothingness)
100
+ end
101
+
102
+ def test_array_content
103
+ topic = Topic.new
104
+ topic.content = %w( one two three )
105
+ topic.save
106
+
107
+ assert_equal(%w( one two three ), Topic.find(topic.id).content)
108
+ end
109
+
110
+ def test_hash_content
111
+ topic = Topic.new
112
+ topic.content = { "one" => 1, "two" => 2 }
113
+ topic.save
114
+
115
+ assert_equal 2, Topic.find(topic.id).content["two"]
116
+
117
+ topic.content["three"] = 3
118
+ topic.save
119
+
120
+ assert_equal 3, Topic.find(topic.id).content["three"]
121
+ end
122
+
123
+ def test_update_array_content
124
+ topic = Topic.new
125
+ topic.content = %w( one two three )
126
+
127
+ topic.content.push "four"
128
+ assert_equal(%w( one two three four ), topic.content)
129
+
130
+ topic.save
131
+
132
+ topic = Topic.find(topic.id)
133
+ topic.content << "five"
134
+ assert_equal(%w( one two three four five ), topic.content)
135
+ end
136
+
137
+ def test_case_sensitive_attributes_hash
138
+ # DB2 is not case-sensitive
139
+ return true if current_adapter?(:DB2Adapter)
140
+
141
+ assert_equal @loaded_fixtures['computers']['workstation'].to_hash, Computer.find(:first).attributes
142
+ end
143
+
144
+ def test_create
145
+ topic = Topic.new
146
+ topic.title = "New Topic"
147
+ topic.save
148
+ topic_reloaded = Topic.find(topic.id)
149
+ assert_equal("New Topic", topic_reloaded.title)
150
+ end
151
+
152
+ def test_save!
153
+ topic = Topic.new(:title => "New Topic")
154
+ assert topic.save!
155
+
156
+ reply = Reply.new
157
+ assert_raise(ActiveRecord::RecordInvalid) { reply.save! }
158
+ end
159
+
160
+ def test_save_null_string_attributes
161
+ topic = Topic.find(1)
162
+ topic.attributes = { "title" => "null", "author_name" => "null" }
163
+ topic.save!
164
+ topic.reload
165
+ assert_equal("null", topic.title)
166
+ assert_equal("null", topic.author_name)
167
+ end
168
+
169
+ def test_save_nil_string_attributes
170
+ topic = Topic.find(1)
171
+ topic.title = nil
172
+ topic.save!
173
+ topic.reload
174
+ assert_nil topic.title
175
+ end
176
+
177
+ def test_hashes_not_mangled
178
+ new_topic = { :title => "New Topic" }
179
+ new_topic_values = { :title => "AnotherTopic" }
180
+
181
+ topic = Topic.new(new_topic)
182
+ assert_equal new_topic[:title], topic.title
183
+
184
+ topic.attributes= new_topic_values
185
+ assert_equal new_topic_values[:title], topic.title
186
+ end
187
+
188
+ def test_create_many
189
+ topics = Topic.create([ { "title" => "first" }, { "title" => "second" }])
190
+ assert_equal 2, topics.size
191
+ assert_equal "first", topics.first.title
192
+ end
193
+
194
+ def test_create_columns_not_equal_attributes
195
+ topic = Topic.new
196
+ topic.title = 'Another New Topic'
197
+ topic.send :write_attribute, 'does_not_exist', 'test'
198
+ assert_nothing_raised { topic.save }
199
+ end
200
+
201
+ def test_create_through_factory
202
+ topic = Topic.create("title" => "New Topic")
203
+ topicReloaded = Topic.find(topic.id)
204
+ assert_equal(topic, topicReloaded)
205
+ end
206
+
207
+ def test_update
208
+ topic = Topic.new
209
+ topic.title = "Another New Topic"
210
+ topic.written_on = "2003-12-12 23:23:00"
211
+ topic.save
212
+ topicReloaded = Topic.find(topic.id)
213
+ assert_equal("Another New Topic", topicReloaded.title)
214
+
215
+ topicReloaded.title = "Updated topic"
216
+ topicReloaded.save
217
+
218
+ topicReloadedAgain = Topic.find(topic.id)
219
+
220
+ assert_equal("Updated topic", topicReloadedAgain.title)
221
+ end
222
+
223
+ def test_update_columns_not_equal_attributes
224
+ topic = Topic.new
225
+ topic.title = "Still another topic"
226
+ topic.save
227
+
228
+ topicReloaded = Topic.find(topic.id)
229
+ topicReloaded.title = "A New Topic"
230
+ topicReloaded.send :write_attribute, 'does_not_exist', 'test'
231
+ assert_nothing_raised { topicReloaded.save }
232
+ end
233
+
234
+ def test_write_attribute
235
+ topic = Topic.new
236
+ topic.send(:write_attribute, :title, "Still another topic")
237
+ assert_equal "Still another topic", topic.title
238
+
239
+ topic.send(:write_attribute, "title", "Still another topic: part 2")
240
+ assert_equal "Still another topic: part 2", topic.title
241
+ end
242
+
243
+ def test_read_attribute
244
+ topic = Topic.new
245
+ topic.title = "Don't change the topic"
246
+ assert_equal "Don't change the topic", topic.send(:read_attribute, "title")
247
+ assert_equal "Don't change the topic", topic["title"]
248
+
249
+ assert_equal "Don't change the topic", topic.send(:read_attribute, :title)
250
+ assert_equal "Don't change the topic", topic[:title]
251
+ end
252
+
253
+ def test_read_attribute_when_false
254
+ topic = topics(:first)
255
+ topic.approved = false
256
+ assert !topic.approved?, "approved should be false"
257
+ topic.approved = "false"
258
+ assert !topic.approved?, "approved should be false"
259
+ end
260
+
261
+ def test_read_attribute_when_true
262
+ topic = topics(:first)
263
+ topic.approved = true
264
+ assert topic.approved?, "approved should be true"
265
+ topic.approved = "true"
266
+ assert topic.approved?, "approved should be true"
267
+ end
268
+
269
+ def test_read_write_boolean_attribute
270
+ topic = Topic.new
271
+ # puts ""
272
+ # puts "New Topic"
273
+ # puts topic.inspect
274
+ topic.approved = "false"
275
+ # puts "Expecting false"
276
+ # puts topic.inspect
277
+ assert !topic.approved?, "approved should be false"
278
+ topic.approved = "false"
279
+ # puts "Expecting false"
280
+ # puts topic.inspect
281
+ assert !topic.approved?, "approved should be false"
282
+ topic.approved = "true"
283
+ # puts "Expecting true"
284
+ # puts topic.inspect
285
+ assert topic.approved?, "approved should be true"
286
+ topic.approved = "true"
287
+ # puts "Expecting true"
288
+ # puts topic.inspect
289
+ assert topic.approved?, "approved should be true"
290
+ # puts ""
291
+ end
292
+
293
+ def test_reader_generation
294
+ Topic.find(:first).title
295
+ Firm.find(:first).name
296
+ Client.find(:first).name
297
+ if ActiveRecord::Base.generate_read_methods
298
+ assert_readers(Topic, %w(type replies_count))
299
+ assert_readers(Firm, %w(type))
300
+ assert_readers(Client, %w(type ruby_type rating?))
301
+ else
302
+ [Topic, Firm, Client].each {|klass| assert_equal klass.read_methods, {}}
303
+ end
304
+ end
305
+
306
+ def test_reader_for_invalid_column_names
307
+ # column names which aren't legal ruby ids
308
+ topic = Topic.find(:first)
309
+ topic.send(:define_read_method, "mumub-jumbo".to_sym, "mumub-jumbo", nil)
310
+ assert !Topic.read_methods.include?("mumub-jumbo")
311
+ end
312
+
313
+ def test_non_attribute_access_and_assignment
314
+ topic = Topic.new
315
+ assert !topic.respond_to?("mumbo")
316
+ assert_raises(NoMethodError) { topic.mumbo }
317
+ assert_raises(NoMethodError) { topic.mumbo = 5 }
318
+ end
319
+
320
+ def test_preserving_date_objects
321
+ # SQL Server doesn't have a separate column type just for dates, so all are returned as time
322
+ return true if current_adapter?(:SQLServerAdapter)
323
+
324
+ if current_adapter?(:SybaseAdapter)
325
+ # Sybase ctlib does not (yet?) support the date type; use datetime instead.
326
+ assert_kind_of(
327
+ Time, Topic.find(1).last_read,
328
+ "The last_read attribute should be of the Time class"
329
+ )
330
+ else
331
+ assert_kind_of(
332
+ Date, Topic.find(1).last_read,
333
+ "The last_read attribute should be of the Date class"
334
+ )
335
+ end
336
+ end
337
+
338
+ def test_preserving_time_objects
339
+ assert_kind_of(
340
+ Time, Topic.find(1).bonus_time,
341
+ "The bonus_time attribute should be of the Time class"
342
+ )
343
+
344
+ assert_kind_of(
345
+ Time, Topic.find(1).written_on,
346
+ "The written_on attribute should be of the Time class"
347
+ )
348
+
349
+ # For adapters which support microsecond resolution.
350
+ if current_adapter?(:PostgreSQLAdapter)
351
+ assert_equal 11, Topic.find(1).written_on.sec
352
+ assert_equal 223300, Topic.find(1).written_on.usec
353
+ assert_equal 9900, Topic.find(2).written_on.usec
354
+ end
355
+ end
356
+
357
+ def test_destroy
358
+ topic = Topic.find(1)
359
+ assert_equal topic, topic.destroy, 'topic.destroy did not return self'
360
+ assert topic.frozen?, 'topic not frozen after destroy'
361
+ assert_raise(ActiveRecord::RecordNotFound) { Topic.find(topic.id) }
362
+ end
363
+
364
+ def test_record_not_found_exception
365
+ assert_raises(ActiveRecord::RecordNotFound) { topicReloaded = Topic.find(99999) }
366
+ end
367
+
368
+ def test_initialize_with_attributes
369
+ topic = Topic.new({
370
+ "title" => "initialized from attributes", "written_on" => "2003-12-12 23:23"
371
+ })
372
+
373
+ assert_equal("initialized from attributes", topic.title)
374
+ end
375
+
376
+ def test_initialize_with_invalid_attribute
377
+ begin
378
+ topic = Topic.new({ "title" => "test",
379
+ "last_read(1i)" => "2005", "last_read(2i)" => "2", "last_read(3i)" => "31"})
380
+ rescue ActiveRecord::MultiparameterAssignmentErrors => ex
381
+ assert_equal(1, ex.errors.size)
382
+ assert_equal("last_read", ex.errors[0].attribute)
383
+ end
384
+ end
385
+
386
+ def test_load
387
+ topics = Topic.find(:all, :order => 'id')
388
+ assert_equal(2, topics.size)
389
+ assert_equal(topics(:first).title, topics.first.title)
390
+ end
391
+
392
+ def test_load_with_condition
393
+ topics = Topic.find(:all, :conditions => "author_name = 'Mary'")
394
+
395
+ assert_equal(1, topics.size)
396
+ assert_equal(topics(:second).title, topics.first.title)
397
+ end
398
+
399
+ def test_table_name_guesses
400
+ classes = [Category, Smarts, CreditCard, CreditCard::PinNumber, CreditCard::PinNumber::CvvCode, CreditCard::SubPinNumber, CreditCard::Brand, MasterCreditCard]
401
+
402
+ assert_equal "topics", Topic.table_name
403
+
404
+ assert_equal "categories", Category.table_name
405
+ assert_equal "smarts", Smarts.table_name
406
+ assert_equal "credit_cards", CreditCard.table_name
407
+ assert_equal "credit_card_pin_numbers", CreditCard::PinNumber.table_name
408
+ assert_equal "credit_card_pin_number_cvv_codes", CreditCard::PinNumber::CvvCode.table_name
409
+ assert_equal "credit_card_pin_numbers", CreditCard::SubPinNumber.table_name
410
+ assert_equal "categories", CreditCard::Brand.table_name
411
+ assert_equal "master_credit_cards", MasterCreditCard.table_name
412
+
413
+ ActiveRecord::Base.pluralize_table_names = false
414
+ classes.each(&:reset_table_name)
415
+
416
+ assert_equal "category", Category.table_name
417
+ assert_equal "smarts", Smarts.table_name
418
+ assert_equal "credit_card", CreditCard.table_name
419
+ assert_equal "credit_card_pin_number", CreditCard::PinNumber.table_name
420
+ assert_equal "credit_card_pin_number_cvv_code", CreditCard::PinNumber::CvvCode.table_name
421
+ assert_equal "credit_card_pin_number", CreditCard::SubPinNumber.table_name
422
+ assert_equal "category", CreditCard::Brand.table_name
423
+ assert_equal "master_credit_card", MasterCreditCard.table_name
424
+
425
+ ActiveRecord::Base.pluralize_table_names = true
426
+ classes.each(&:reset_table_name)
427
+
428
+ ActiveRecord::Base.table_name_prefix = "test_"
429
+ Category.reset_table_name
430
+ assert_equal "test_categories", Category.table_name
431
+ ActiveRecord::Base.table_name_suffix = "_test"
432
+ Category.reset_table_name
433
+ assert_equal "test_categories_test", Category.table_name
434
+ ActiveRecord::Base.table_name_prefix = ""
435
+ Category.reset_table_name
436
+ assert_equal "categories_test", Category.table_name
437
+ ActiveRecord::Base.table_name_suffix = ""
438
+ Category.reset_table_name
439
+ assert_equal "categories", Category.table_name
440
+
441
+ ActiveRecord::Base.pluralize_table_names = false
442
+ ActiveRecord::Base.table_name_prefix = "test_"
443
+ Category.reset_table_name
444
+ assert_equal "test_category", Category.table_name
445
+ ActiveRecord::Base.table_name_suffix = "_test"
446
+ Category.reset_table_name
447
+ assert_equal "test_category_test", Category.table_name
448
+ ActiveRecord::Base.table_name_prefix = ""
449
+ Category.reset_table_name
450
+ assert_equal "category_test", Category.table_name
451
+ ActiveRecord::Base.table_name_suffix = ""
452
+ Category.reset_table_name
453
+ assert_equal "category", Category.table_name
454
+
455
+ ActiveRecord::Base.pluralize_table_names = true
456
+ classes.each(&:reset_table_name)
457
+ end
458
+
459
+ def test_destroy_all
460
+ assert_equal 2, Topic.count
461
+
462
+ Topic.destroy_all "author_name = 'Mary'"
463
+ assert_equal 1, Topic.count
464
+ end
465
+
466
+ def test_destroy_many
467
+ assert_equal 3, Client.count
468
+ Client.destroy([2, 3])
469
+ assert_equal 1, Client.count
470
+ end
471
+
472
+ def test_delete_many
473
+ Topic.delete([1, 2])
474
+ assert_equal 0, Topic.count
475
+ end
476
+
477
+ def test_boolean_attributes
478
+ assert ! Topic.find(1).approved?
479
+ assert Topic.find(2).approved?
480
+ end
481
+
482
+ def test_increment_counter
483
+ Topic.increment_counter("replies_count", 1)
484
+ assert_equal 2, Topic.find(1).replies_count
485
+
486
+ Topic.increment_counter("replies_count", 1)
487
+ assert_equal 3, Topic.find(1).replies_count
488
+ end
489
+
490
+ def test_decrement_counter
491
+ Topic.decrement_counter("replies_count", 2)
492
+ assert_equal -1, Topic.find(2).replies_count
493
+
494
+ Topic.decrement_counter("replies_count", 2)
495
+ assert_equal -2, Topic.find(2).replies_count
496
+ end
497
+
498
+ def test_update_all
499
+ # The ADO library doesn't support the number of affected rows
500
+ return true if current_adapter?(:SQLServerAdapter)
501
+
502
+ assert_equal 2, Topic.update_all("content = 'bulk updated!'")
503
+ assert_equal "bulk updated!", Topic.find(1).content
504
+ assert_equal "bulk updated!", Topic.find(2).content
505
+ assert_equal 2, Topic.update_all(['content = ?', 'bulk updated again!'])
506
+ assert_equal "bulk updated again!", Topic.find(1).content
507
+ assert_equal "bulk updated again!", Topic.find(2).content
508
+ end
509
+
510
+ def test_update_many
511
+ topic_data = { 1 => { "content" => "1 updated" }, 2 => { "content" => "2 updated" } }
512
+ updated = Topic.update(topic_data.keys, topic_data.values)
513
+
514
+ assert_equal 2, updated.size
515
+ assert_equal "1 updated", Topic.find(1).content
516
+ assert_equal "2 updated", Topic.find(2).content
517
+ end
518
+
519
+ def test_delete_all
520
+ # The ADO library doesn't support the number of affected rows
521
+ return true if current_adapter?(:SQLServerAdapter)
522
+
523
+ assert_equal 2, Topic.delete_all
524
+ end
525
+
526
+ def test_update_by_condition
527
+ Topic.update_all "content = 'bulk updated!'", ["approved = ?", true]
528
+ assert_equal "Have a nice day", Topic.find(1).content
529
+ assert_equal "bulk updated!", Topic.find(2).content
530
+ end
531
+
532
+ def test_attribute_present
533
+ t = Topic.new
534
+ t.title = "hello there!"
535
+ t.written_on = Time.now
536
+ assert t.attribute_present?("title")
537
+ assert t.attribute_present?("written_on")
538
+ assert !t.attribute_present?("content")
539
+ end
540
+
541
+ def test_attribute_keys_on_new_instance
542
+ t = Topic.new
543
+ assert_equal nil, t.title, "The topics table has a title column, so it should be nil"
544
+ assert_raise(NoMethodError) { t.title2 }
545
+ end
546
+
547
+ def test_class_name
548
+ assert_equal "Firm", ActiveRecord::Base.class_name("firms")
549
+ assert_equal "Category", ActiveRecord::Base.class_name("categories")
550
+ assert_equal "AccountHolder", ActiveRecord::Base.class_name("account_holder")
551
+
552
+ ActiveRecord::Base.pluralize_table_names = false
553
+ assert_equal "Firms", ActiveRecord::Base.class_name( "firms" )
554
+ ActiveRecord::Base.pluralize_table_names = true
555
+
556
+ ActiveRecord::Base.table_name_prefix = "test_"
557
+ assert_equal "Firm", ActiveRecord::Base.class_name( "test_firms" )
558
+ ActiveRecord::Base.table_name_suffix = "_tests"
559
+ assert_equal "Firm", ActiveRecord::Base.class_name( "test_firms_tests" )
560
+ ActiveRecord::Base.table_name_prefix = ""
561
+ assert_equal "Firm", ActiveRecord::Base.class_name( "firms_tests" )
562
+ ActiveRecord::Base.table_name_suffix = ""
563
+ assert_equal "Firm", ActiveRecord::Base.class_name( "firms" )
564
+ end
565
+
566
+ def test_null_fields
567
+ assert_nil Topic.find(1).parent_id
568
+ assert_nil Topic.create("title" => "Hey you").parent_id
569
+ end
570
+
571
+ def test_default_values
572
+ topic = Topic.new
573
+ assert topic.approved?
574
+ assert_nil topic.written_on
575
+ assert_nil topic.bonus_time
576
+ assert_nil topic.last_read
577
+
578
+ topic.save
579
+
580
+ topic = Topic.find(topic.id)
581
+ assert topic.approved?
582
+ assert_nil topic.last_read
583
+
584
+ # Oracle has some funky default handling, so it requires a bit of
585
+ # extra testing. See ticket #2788.
586
+ if current_adapter?(:OracleAdapter)
587
+ test = TestOracleDefault.new
588
+ assert_equal "X", test.test_char
589
+ assert_equal "hello", test.test_string
590
+ assert_equal 3, test.test_int
591
+ end
592
+ end
593
+
594
+ # Oracle, SQLServer, and Sybase do not have a TIME datatype.
595
+ unless current_adapter?(:SQLServerAdapter, :OracleAdapter, :SybaseAdapter)
596
+ def test_utc_as_time_zone
597
+ Topic.default_timezone = :utc
598
+ attributes = { "bonus_time" => "5:42:00AM" }
599
+ topic = Topic.find(1)
600
+ topic.attributes = attributes
601
+ assert_equal Time.utc(2000, 1, 1, 5, 42, 0), topic.bonus_time
602
+ Topic.default_timezone = :local
603
+ end
604
+
605
+ def test_utc_as_time_zone_and_new
606
+ Topic.default_timezone = :utc
607
+ attributes = { "bonus_time(1i)"=>"2000",
608
+ "bonus_time(2i)"=>"1",
609
+ "bonus_time(3i)"=>"1",
610
+ "bonus_time(4i)"=>"10",
611
+ "bonus_time(5i)"=>"35",
612
+ "bonus_time(6i)"=>"50" }
613
+ topic = Topic.new(attributes)
614
+ assert_equal Time.utc(2000, 1, 1, 10, 35, 50), topic.bonus_time
615
+ Topic.default_timezone = :local
616
+ end
617
+ end
618
+
619
+ def test_default_values_on_empty_strings
620
+ topic = Topic.new
621
+ topic.approved = nil
622
+ topic.last_read = nil
623
+
624
+ topic.save
625
+
626
+ topic = Topic.find(topic.id)
627
+ assert_nil topic.last_read
628
+
629
+ # Sybase adapter does not allow nulls in boolean columns
630
+ if current_adapter?(:SybaseAdapter)
631
+ assert topic.approved == false
632
+ else
633
+ assert_nil topic.approved
634
+ end
635
+ end
636
+
637
+ def test_equality
638
+ assert_equal Topic.find(1), Topic.find(2).topic
639
+ end
640
+
641
+ def test_equality_of_new_records
642
+ assert_not_equal Topic.new, Topic.new
643
+ end
644
+
645
+ def test_hashing
646
+ assert_equal [ Topic.find(1) ], [ Topic.find(2).topic ] & [ Topic.find(1) ]
647
+ end
648
+
649
+ def test_destroy_new_record
650
+ client = Client.new
651
+ client.destroy
652
+ assert client.frozen?
653
+ end
654
+
655
+ def test_destroy_record_with_associations
656
+ client = Client.find(3)
657
+ client.destroy
658
+ assert client.frozen?
659
+ assert_kind_of Firm, client.firm
660
+ assert_raises(TypeError) { client.name = "something else" }
661
+ end
662
+
663
+ def test_update_attribute
664
+ assert !Topic.find(1).approved?
665
+ Topic.find(1).update_attribute("approved", true)
666
+ assert Topic.find(1).approved?
667
+
668
+ Topic.find(1).update_attribute(:approved, false)
669
+ assert !Topic.find(1).approved?
670
+ end
671
+
672
+ def test_update_attributes
673
+ topic = Topic.find(1)
674
+ assert !topic.approved?
675
+ assert_equal "The First Topic", topic.title
676
+
677
+ topic.update_attributes("approved" => true, "title" => "The First Topic Updated")
678
+ topic.reload
679
+ assert topic.approved?
680
+ assert_equal "The First Topic Updated", topic.title
681
+
682
+ topic.update_attributes(:approved => false, :title => "The First Topic")
683
+ topic.reload
684
+ assert !topic.approved?
685
+ assert_equal "The First Topic", topic.title
686
+ end
687
+
688
+ def test_update_attributes!
689
+ reply = Reply.find(2)
690
+ assert_equal "The Second Topic's of the day", reply.title
691
+ assert_equal "Have a nice day", reply.content
692
+
693
+ reply.update_attributes!("title" => "The Second Topic's of the day updated", "content" => "Have a nice evening")
694
+ reply.reload
695
+ assert_equal "The Second Topic's of the day updated", reply.title
696
+ assert_equal "Have a nice evening", reply.content
697
+
698
+ reply.update_attributes!(:title => "The Second Topic's of the day", :content => "Have a nice day")
699
+ reply.reload
700
+ assert_equal "The Second Topic's of the day", reply.title
701
+ assert_equal "Have a nice day", reply.content
702
+
703
+ assert_raise(ActiveRecord::RecordInvalid) { reply.update_attributes!(:title => nil, :content => "Have a nice evening") }
704
+ end
705
+
706
+ def test_mass_assignment_protection
707
+ firm = Firm.new
708
+ firm.attributes = { "name" => "Next Angle", "rating" => 5 }
709
+ assert_equal 1, firm.rating
710
+ end
711
+
712
+ def test_mass_assignment_protection_against_class_attribute_writers
713
+ [:logger, :configurations, :primary_key_prefix_type, :table_name_prefix, :table_name_suffix, :pluralize_table_names, :colorize_logging,
714
+ :default_timezone, :allow_concurrency, :generate_read_methods, :schema_format, :verification_timeout, :lock_optimistically, :record_timestamps].each do |method|
715
+ assert Task.respond_to?(method)
716
+ assert Task.respond_to?("#{method}=")
717
+ assert Task.new.respond_to?(method)
718
+ assert !Task.new.respond_to?("#{method}=")
719
+ end
720
+ end
721
+
722
+ def test_customized_primary_key_remains_protected
723
+ subscriber = Subscriber.new(:nick => 'webster123', :name => 'nice try')
724
+ assert_nil subscriber.id
725
+
726
+ keyboard = Keyboard.new(:key_number => 9, :name => 'nice try')
727
+ assert_nil keyboard.id
728
+ end
729
+
730
+ def test_customized_primary_key_remains_protected_when_refered_to_as_id
731
+ subscriber = Subscriber.new(:id => 'webster123', :name => 'nice try')
732
+ assert_nil subscriber.id
733
+
734
+ keyboard = Keyboard.new(:id => 9, :name => 'nice try')
735
+ assert_nil keyboard.id
736
+ end
737
+
738
+ def test_mass_assignment_protection_on_defaults
739
+ firm = Firm.new
740
+ firm.attributes = { "id" => 5, "type" => "Client" }
741
+ assert_nil firm.id
742
+ assert_equal "Firm", firm[:type]
743
+ end
744
+
745
+ def test_mass_assignment_accessible
746
+ reply = Reply.new("title" => "hello", "content" => "world", "approved" => true)
747
+ reply.save
748
+
749
+ assert reply.approved?
750
+
751
+ reply.approved = false
752
+ reply.save
753
+
754
+ assert !reply.approved?
755
+ end
756
+
757
+ def test_mass_assignment_protection_inheritance
758
+ assert_nil LoosePerson.accessible_attributes
759
+ assert_equal [ :credit_rating, :administrator ], LoosePerson.protected_attributes
760
+
761
+ assert_nil LooseDescendant.accessible_attributes
762
+ assert_equal [ :credit_rating, :administrator, :phone_number ], LooseDescendant.protected_attributes
763
+
764
+ assert_nil TightPerson.protected_attributes
765
+ assert_equal [ :name, :address ], TightPerson.accessible_attributes
766
+
767
+ assert_nil TightDescendant.protected_attributes
768
+ assert_equal [ :name, :address, :phone_number ], TightDescendant.accessible_attributes
769
+ end
770
+
771
+ def test_multiparameter_attributes_on_date
772
+ attributes = { "last_read(1i)" => "2004", "last_read(2i)" => "6", "last_read(3i)" => "24" }
773
+ topic = Topic.find(1)
774
+ topic.attributes = attributes
775
+ # note that extra #to_date call allows test to pass for Oracle, which
776
+ # treats dates/times the same
777
+ assert_date_from_db Date.new(2004, 6, 24), topic.last_read.to_date
778
+ end
779
+
780
+ def test_multiparameter_attributes_on_date_with_empty_date
781
+ attributes = { "last_read(1i)" => "2004", "last_read(2i)" => "6", "last_read(3i)" => "" }
782
+ topic = Topic.find(1)
783
+ topic.attributes = attributes
784
+ # note that extra #to_date call allows test to pass for Oracle, which
785
+ # treats dates/times the same
786
+ assert_date_from_db Date.new(2004, 6, 1), topic.last_read.to_date
787
+ end
788
+
789
+ def test_multiparameter_attributes_on_date_with_all_empty
790
+ attributes = { "last_read(1i)" => "", "last_read(2i)" => "", "last_read(3i)" => "" }
791
+ topic = Topic.find(1)
792
+ topic.attributes = attributes
793
+ assert_nil topic.last_read
794
+ end
795
+
796
+ def test_multiparameter_attributes_on_time
797
+ attributes = {
798
+ "written_on(1i)" => "2004", "written_on(2i)" => "6", "written_on(3i)" => "24",
799
+ "written_on(4i)" => "16", "written_on(5i)" => "24", "written_on(6i)" => "00"
800
+ }
801
+ topic = Topic.find(1)
802
+ topic.attributes = attributes
803
+ assert_equal Time.local(2004, 6, 24, 16, 24, 0), topic.written_on
804
+ end
805
+
806
+ def test_multiparameter_attributes_on_time_with_empty_seconds
807
+ attributes = {
808
+ "written_on(1i)" => "2004", "written_on(2i)" => "6", "written_on(3i)" => "24",
809
+ "written_on(4i)" => "16", "written_on(5i)" => "24", "written_on(6i)" => ""
810
+ }
811
+ topic = Topic.find(1)
812
+ topic.attributes = attributes
813
+ assert_equal Time.local(2004, 6, 24, 16, 24, 0), topic.written_on
814
+ end
815
+
816
+ def test_multiparameter_mass_assignment_protector
817
+ task = Task.new
818
+ time = Time.mktime(2000, 1, 1, 1)
819
+ task.starting = time
820
+ attributes = { "starting(1i)" => "2004", "starting(2i)" => "6", "starting(3i)" => "24" }
821
+ task.attributes = attributes
822
+ assert_equal time, task.starting
823
+ end
824
+
825
+ def test_multiparameter_assignment_of_aggregation
826
+ customer = Customer.new
827
+ address = Address.new("The Street", "The City", "The Country")
828
+ attributes = { "address(1)" => address.street, "address(2)" => address.city, "address(3)" => address.country }
829
+ customer.attributes = attributes
830
+ assert_equal address, customer.address
831
+ end
832
+
833
+ def test_attributes_on_dummy_time
834
+ # Oracle, SQL Server, and Sybase do not have a TIME datatype.
835
+ return true if current_adapter?(:SQLServerAdapter, :OracleAdapter, :SybaseAdapter)
836
+
837
+ attributes = {
838
+ "bonus_time" => "5:42:00AM"
839
+ }
840
+ topic = Topic.find(1)
841
+ topic.attributes = attributes
842
+ assert_equal Time.local(2000, 1, 1, 5, 42, 0), topic.bonus_time
843
+ end
844
+
845
+ def test_boolean
846
+ b_false = Booleantest.create({ "value" => false })
847
+ false_id = b_false.id
848
+ b_true = Booleantest.create({ "value" => true })
849
+ true_id = b_true.id
850
+
851
+ b_false = Booleantest.find(false_id)
852
+ assert !b_false.value?
853
+ b_true = Booleantest.find(true_id)
854
+ assert b_true.value?
855
+ end
856
+
857
+ def test_boolean_cast_from_string
858
+ b_false = Booleantest.create({ "value" => "0" })
859
+ false_id = b_false.id
860
+ b_true = Booleantest.create({ "value" => "1" })
861
+ true_id = b_true.id
862
+
863
+ b_false = Booleantest.find(false_id)
864
+ assert !b_false.value?
865
+ b_true = Booleantest.find(true_id)
866
+ assert b_true.value?
867
+ end
868
+
869
+ def test_clone
870
+ topic = Topic.find(1)
871
+ cloned_topic = nil
872
+ assert_nothing_raised { cloned_topic = topic.clone }
873
+ assert_equal topic.title, cloned_topic.title
874
+ assert cloned_topic.new_record?
875
+
876
+ # test if the attributes have been cloned
877
+ topic.title = "a"
878
+ cloned_topic.title = "b"
879
+ assert_equal "a", topic.title
880
+ assert_equal "b", cloned_topic.title
881
+
882
+ # test if the attribute values have been cloned
883
+ topic.title = {"a" => "b"}
884
+ cloned_topic = topic.clone
885
+ cloned_topic.title["a"] = "c"
886
+ assert_equal "b", topic.title["a"]
887
+
888
+ cloned_topic.save
889
+ assert !cloned_topic.new_record?
890
+ assert cloned_topic.id != topic.id
891
+ end
892
+
893
+ def test_clone_with_aggregate_of_same_name_as_attribute
894
+ dev = DeveloperWithAggregate.find(1)
895
+ assert_kind_of DeveloperSalary, dev.salary
896
+
897
+ clone = nil
898
+ assert_nothing_raised { clone = dev.clone }
899
+ assert_kind_of DeveloperSalary, clone.salary
900
+ assert_equal dev.salary.amount, clone.salary.amount
901
+ assert clone.new_record?
902
+
903
+ # test if the attributes have been cloned
904
+ original_amount = clone.salary.amount
905
+ dev.salary.amount = 1
906
+ assert_equal original_amount, clone.salary.amount
907
+
908
+ assert clone.save
909
+ assert !clone.new_record?
910
+ assert clone.id != dev.id
911
+ end
912
+
913
+ def test_clone_preserves_subtype
914
+ clone = nil
915
+ assert_nothing_raised { clone = Company.find(3).clone }
916
+ assert_kind_of Client, clone
917
+ end
918
+
919
+ def test_bignum
920
+ company = Company.find(1)
921
+ company.rating = 2147483647
922
+ company.save
923
+ company = Company.find(1)
924
+ assert_equal 2147483647, company.rating
925
+ end
926
+
927
+ # TODO: extend defaults tests to other databases!
928
+ if current_adapter?(:PostgreSQLAdapter)
929
+ def test_default
930
+ default = Default.new
931
+
932
+ # fixed dates / times
933
+ assert_equal Date.new(2004, 1, 1), default.fixed_date
934
+ assert_equal Time.local(2004, 1,1,0,0,0,0), default.fixed_time
935
+
936
+ # char types
937
+ assert_equal 'Y', default.char1
938
+ assert_equal 'a varchar field', default.char2
939
+ assert_equal 'a text field', default.char3
940
+ end
941
+
942
+ class Geometric < ActiveRecord::Base; end
943
+ def test_geometric_content
944
+
945
+ # accepted format notes:
946
+ # ()'s aren't required
947
+ # values can be a mix of float or integer
948
+
949
+ g = Geometric.new(
950
+ :a_point => '(5.0, 6.1)',
951
+ #:a_line => '((2.0, 3), (5.5, 7.0))' # line type is currently unsupported in postgresql
952
+ :a_line_segment => '(2.0, 3), (5.5, 7.0)',
953
+ :a_box => '2.0, 3, 5.5, 7.0',
954
+ :a_path => '[(2.0, 3), (5.5, 7.0), (8.5, 11.0)]', # [ ] is an open path
955
+ :a_polygon => '((2.0, 3), (5.5, 7.0), (8.5, 11.0))',
956
+ :a_circle => '<(5.3, 10.4), 2>'
957
+ )
958
+
959
+ assert g.save
960
+
961
+ # Reload and check that we have all the geometric attributes.
962
+ h = Geometric.find(g.id)
963
+
964
+ assert_equal '(5,6.1)', h.a_point
965
+ assert_equal '[(2,3),(5.5,7)]', h.a_line_segment
966
+ assert_equal '(5.5,7),(2,3)', h.a_box # reordered to store upper right corner then bottom left corner
967
+ assert_equal '[(2,3),(5.5,7),(8.5,11)]', h.a_path
968
+ assert_equal '((2,3),(5.5,7),(8.5,11))', h.a_polygon
969
+ assert_equal '<(5.3,10.4),2>', h.a_circle
970
+
971
+ # use a geometric function to test for an open path
972
+ objs = Geometric.find_by_sql ["select isopen(a_path) from geometrics where id = ?", g.id]
973
+ assert_equal objs[0].isopen, 't'
974
+
975
+ # test alternate formats when defining the geometric types
976
+
977
+ g = Geometric.new(
978
+ :a_point => '5.0, 6.1',
979
+ #:a_line => '((2.0, 3), (5.5, 7.0))' # line type is currently unsupported in postgresql
980
+ :a_line_segment => '((2.0, 3), (5.5, 7.0))',
981
+ :a_box => '(2.0, 3), (5.5, 7.0)',
982
+ :a_path => '((2.0, 3), (5.5, 7.0), (8.5, 11.0))', # ( ) is a closed path
983
+ :a_polygon => '2.0, 3, 5.5, 7.0, 8.5, 11.0',
984
+ :a_circle => '((5.3, 10.4), 2)'
985
+ )
986
+
987
+ assert g.save
988
+
989
+ # Reload and check that we have all the geometric attributes.
990
+ h = Geometric.find(g.id)
991
+
992
+ assert_equal '(5,6.1)', h.a_point
993
+ assert_equal '[(2,3),(5.5,7)]', h.a_line_segment
994
+ assert_equal '(5.5,7),(2,3)', h.a_box # reordered to store upper right corner then bottom left corner
995
+ assert_equal '((2,3),(5.5,7),(8.5,11))', h.a_path
996
+ assert_equal '((2,3),(5.5,7),(8.5,11))', h.a_polygon
997
+ assert_equal '<(5.3,10.4),2>', h.a_circle
998
+
999
+ # use a geometric function to test for an closed path
1000
+ objs = Geometric.find_by_sql ["select isclosed(a_path) from geometrics where id = ?", g.id]
1001
+ assert_equal objs[0].isclosed, 't'
1002
+ end
1003
+ end
1004
+
1005
+ class NumericData < ActiveRecord::Base
1006
+ self.table_name = 'numeric_data'
1007
+ end
1008
+
1009
+ def test_numeric_fields
1010
+ m = NumericData.new(
1011
+ :bank_balance => 1586.43,
1012
+ :big_bank_balance => BigDecimal("1000234000567.95"),
1013
+ :world_population => 6000000000,
1014
+ :my_house_population => 3
1015
+ )
1016
+ assert m.save
1017
+
1018
+ m1 = NumericData.find(m.id)
1019
+ assert_not_nil m1
1020
+
1021
+ # As with migration_test.rb, we should make world_population >= 2**62
1022
+ # to cover 64-bit platforms and test it is a Bignum, but the main thing
1023
+ # is that it's an Integer.
1024
+ assert_kind_of Integer, m1.world_population
1025
+ assert_equal 6000000000, m1.world_population
1026
+
1027
+ assert_kind_of Fixnum, m1.my_house_population
1028
+ assert_equal 3, m1.my_house_population
1029
+
1030
+ assert_kind_of BigDecimal, m1.bank_balance
1031
+ assert_equal BigDecimal("1586.43"), m1.bank_balance
1032
+
1033
+ assert_kind_of BigDecimal, m1.big_bank_balance
1034
+ assert_equal BigDecimal("1000234000567.95"), m1.big_bank_balance
1035
+ end
1036
+
1037
+ def test_auto_id
1038
+ auto = AutoId.new
1039
+ auto.save
1040
+ assert (auto.id > 0)
1041
+ end
1042
+
1043
+ def quote_column_name(name)
1044
+ "<#{name}>"
1045
+ end
1046
+
1047
+ def test_quote_keys
1048
+ ar = AutoId.new
1049
+ source = {"foo" => "bar", "baz" => "quux"}
1050
+ actual = ar.send(:quote_columns, self, source)
1051
+ inverted = actual.invert
1052
+ assert_equal("<foo>", inverted["bar"])
1053
+ assert_equal("<baz>", inverted["quux"])
1054
+ end
1055
+
1056
+ def test_sql_injection_via_find
1057
+ assert_raises(ActiveRecord::RecordNotFound, ActiveRecord::StatementInvalid) do
1058
+ Topic.find("123456 OR id > 0")
1059
+ end
1060
+ end
1061
+
1062
+ def test_column_name_properly_quoted
1063
+ col_record = ColumnName.new
1064
+ col_record.references = 40
1065
+ assert col_record.save
1066
+ col_record.references = 41
1067
+ assert col_record.save
1068
+ assert_not_nil c2 = ColumnName.find(col_record.id)
1069
+ assert_equal(41, c2.references)
1070
+ end
1071
+
1072
+ def test_quoting_arrays
1073
+ replies = Reply.find(:all, :conditions => [ "id IN (?)", topics(:first).replies.collect(&:id) ])
1074
+ assert_equal topics(:first).replies.size, replies.size
1075
+
1076
+ replies = Reply.find(:all, :conditions => [ "id IN (?)", [] ])
1077
+ assert_equal 0, replies.size
1078
+ end
1079
+
1080
+ MyObject = Struct.new :attribute1, :attribute2
1081
+
1082
+ def test_serialized_attribute
1083
+ myobj = MyObject.new('value1', 'value2')
1084
+ topic = Topic.create("content" => myobj)
1085
+ Topic.serialize("content", MyObject)
1086
+ assert_equal(myobj, topic.content)
1087
+ end
1088
+
1089
+ def test_serialized_attribute_with_class_constraint
1090
+ myobj = MyObject.new('value1', 'value2')
1091
+ topic = Topic.create("content" => myobj)
1092
+ Topic.serialize(:content, Hash)
1093
+
1094
+ assert_raise(ActiveRecord::SerializationTypeMismatch) { Topic.find(topic.id).content }
1095
+
1096
+ settings = { "color" => "blue" }
1097
+ Topic.find(topic.id).update_attribute("content", settings)
1098
+ assert_equal(settings, Topic.find(topic.id).content)
1099
+ Topic.serialize(:content)
1100
+ end
1101
+
1102
+ def test_quote
1103
+ author_name = "\\ \001 ' \n \\n \""
1104
+ topic = Topic.create('author_name' => author_name)
1105
+ assert_equal author_name, Topic.find(topic.id).author_name
1106
+ end
1107
+
1108
+ def test_quote_chars
1109
+ str = 'The Narrator'
1110
+ topic = Topic.create(:author_name => str)
1111
+ assert_equal str, topic.author_name
1112
+
1113
+ assert_kind_of ActiveSupport::Multibyte::Chars, str.chars
1114
+ topic = Topic.find_by_author_name(str.chars)
1115
+
1116
+ assert_kind_of Topic, topic
1117
+ assert_equal str, topic.author_name, "The right topic should have been found by name even with name passed as Chars"
1118
+ end
1119
+
1120
+ def test_class_level_destroy
1121
+ should_be_destroyed_reply = Reply.create("title" => "hello", "content" => "world")
1122
+ Topic.find(1).replies << should_be_destroyed_reply
1123
+
1124
+ Topic.destroy(1)
1125
+ assert_raise(ActiveRecord::RecordNotFound) { Topic.find(1) }
1126
+ assert_raise(ActiveRecord::RecordNotFound) { Reply.find(should_be_destroyed_reply.id) }
1127
+ end
1128
+
1129
+ def test_class_level_delete
1130
+ should_be_destroyed_reply = Reply.create("title" => "hello", "content" => "world")
1131
+ Topic.find(1).replies << should_be_destroyed_reply
1132
+
1133
+ Topic.delete(1)
1134
+ assert_raise(ActiveRecord::RecordNotFound) { Topic.find(1) }
1135
+ assert_nothing_raised { Reply.find(should_be_destroyed_reply.id) }
1136
+ end
1137
+
1138
+ def test_increment_attribute
1139
+ assert_equal 1, topics(:first).replies_count
1140
+ topics(:first).increment! :replies_count
1141
+ assert_equal 2, topics(:first, :reload).replies_count
1142
+
1143
+ topics(:first).increment(:replies_count).increment!(:replies_count)
1144
+ assert_equal 4, topics(:first, :reload).replies_count
1145
+ end
1146
+
1147
+ def test_increment_nil_attribute
1148
+ assert_nil topics(:first).parent_id
1149
+ topics(:first).increment! :parent_id
1150
+ assert_equal 1, topics(:first).parent_id
1151
+ end
1152
+
1153
+ def test_decrement_attribute
1154
+ topics(:first).increment(:replies_count).increment!(:replies_count)
1155
+ assert_equal 3, topics(:first).replies_count
1156
+
1157
+ topics(:first).decrement!(:replies_count)
1158
+ assert_equal 2, topics(:first, :reload).replies_count
1159
+
1160
+ topics(:first).decrement(:replies_count).decrement!(:replies_count)
1161
+ assert_equal 0, topics(:first, :reload).replies_count
1162
+ end
1163
+
1164
+ def test_toggle_attribute
1165
+ assert !topics(:first).approved?
1166
+ topics(:first).toggle!(:approved)
1167
+ assert topics(:first).approved?
1168
+ topic = topics(:first)
1169
+ topic.toggle(:approved)
1170
+ assert !topic.approved?
1171
+ topic.reload
1172
+ assert topic.approved?
1173
+ end
1174
+
1175
+ def test_reload
1176
+ t1 = Topic.find(1)
1177
+ t2 = Topic.find(1)
1178
+ t1.title = "something else"
1179
+ t1.save
1180
+ t2.reload
1181
+ assert_equal t1.title, t2.title
1182
+ end
1183
+
1184
+ def test_define_attr_method_with_value
1185
+ k = Class.new( ActiveRecord::Base )
1186
+ k.send(:define_attr_method, :table_name, "foo")
1187
+ assert_equal "foo", k.table_name
1188
+ end
1189
+
1190
+ def test_define_attr_method_with_block
1191
+ k = Class.new( ActiveRecord::Base )
1192
+ k.send(:define_attr_method, :primary_key) { "sys_" + original_primary_key }
1193
+ assert_equal "sys_id", k.primary_key
1194
+ end
1195
+
1196
+ def test_set_table_name_with_value
1197
+ k = Class.new( ActiveRecord::Base )
1198
+ k.table_name = "foo"
1199
+ assert_equal "foo", k.table_name
1200
+ k.set_table_name "bar"
1201
+ assert_equal "bar", k.table_name
1202
+ end
1203
+
1204
+ def test_set_table_name_with_block
1205
+ k = Class.new( ActiveRecord::Base )
1206
+ k.set_table_name { "ks" }
1207
+ assert_equal "ks", k.table_name
1208
+ end
1209
+
1210
+ def test_set_primary_key_with_value
1211
+ k = Class.new( ActiveRecord::Base )
1212
+ k.primary_key = "foo"
1213
+ assert_equal "foo", k.primary_key
1214
+ k.set_primary_key "bar"
1215
+ assert_equal "bar", k.primary_key
1216
+ end
1217
+
1218
+ def test_set_primary_key_with_block
1219
+ k = Class.new( ActiveRecord::Base )
1220
+ k.set_primary_key { "sys_" + original_primary_key }
1221
+ assert_equal "sys_id", k.primary_key
1222
+ end
1223
+
1224
+ def test_set_inheritance_column_with_value
1225
+ k = Class.new( ActiveRecord::Base )
1226
+ k.inheritance_column = "foo"
1227
+ assert_equal "foo", k.inheritance_column
1228
+ k.set_inheritance_column "bar"
1229
+ assert_equal "bar", k.inheritance_column
1230
+ end
1231
+
1232
+ def test_set_inheritance_column_with_block
1233
+ k = Class.new( ActiveRecord::Base )
1234
+ k.set_inheritance_column { original_inheritance_column + "_id" }
1235
+ assert_equal "type_id", k.inheritance_column
1236
+ end
1237
+
1238
+ def test_count_with_join
1239
+ res = Post.count_by_sql "SELECT COUNT(*) FROM posts LEFT JOIN comments ON posts.id=comments.post_id WHERE posts.#{QUOTED_TYPE} = 'Post'"
1240
+ res2 = nil
1241
+ assert_deprecated 'count' do
1242
+ res2 = Post.count("posts.#{QUOTED_TYPE} = 'Post'",
1243
+ "LEFT JOIN comments ON posts.id=comments.post_id")
1244
+ end
1245
+ assert_equal res, res2
1246
+
1247
+ res3 = nil
1248
+ assert_nothing_raised do
1249
+ res3 = Post.count(:conditions => "posts.#{QUOTED_TYPE} = 'Post'",
1250
+ :joins => "LEFT JOIN comments ON posts.id=comments.post_id")
1251
+ end
1252
+ assert_equal res, res3
1253
+
1254
+ res4 = Post.count_by_sql "SELECT COUNT(p.id) FROM posts p, comments co WHERE p.#{QUOTED_TYPE} = 'Post' AND p.id=co.post_id"
1255
+ res5 = nil
1256
+ assert_nothing_raised do
1257
+ res5 = Post.count(:conditions => "p.#{QUOTED_TYPE} = 'Post' AND p.id=co.post_id",
1258
+ :joins => "p, comments co",
1259
+ :select => "p.id")
1260
+ end
1261
+
1262
+ assert_equal res4, res5
1263
+
1264
+ res6 = Post.count_by_sql "SELECT COUNT(DISTINCT p.id) FROM posts p, comments co WHERE p.#{QUOTED_TYPE} = 'Post' AND p.id=co.post_id"
1265
+ res7 = nil
1266
+ assert_nothing_raised do
1267
+ res7 = Post.count(:conditions => "p.#{QUOTED_TYPE} = 'Post' AND p.id=co.post_id",
1268
+ :joins => "p, comments co",
1269
+ :select => "p.id",
1270
+ :distinct => true)
1271
+ end
1272
+ assert_equal res6, res7
1273
+ end
1274
+
1275
+ def test_clear_association_cache_stored
1276
+ firm = Firm.find(1)
1277
+ assert_kind_of Firm, firm
1278
+
1279
+ firm.clear_association_cache
1280
+ assert_equal Firm.find(1).clients.collect{ |x| x.name }.sort, firm.clients.collect{ |x| x.name }.sort
1281
+ end
1282
+
1283
+ def test_clear_association_cache_new_record
1284
+ firm = Firm.new
1285
+ client_stored = Client.find(3)
1286
+ client_new = Client.new
1287
+ client_new.name = "The Joneses"
1288
+ clients = [ client_stored, client_new ]
1289
+
1290
+ firm.clients << clients
1291
+
1292
+ firm.clear_association_cache
1293
+
1294
+ assert_equal firm.clients.collect{ |x| x.name }.sort, clients.collect{ |x| x.name }.sort
1295
+ end
1296
+
1297
+ def test_interpolate_sql
1298
+ assert_nothing_raised { Category.new.send(:interpolate_sql, 'foo@bar') }
1299
+ assert_nothing_raised { Category.new.send(:interpolate_sql, 'foo bar) baz') }
1300
+ assert_nothing_raised { Category.new.send(:interpolate_sql, 'foo bar} baz') }
1301
+ end
1302
+
1303
+ def test_scoped_find_conditions
1304
+ scoped_developers = Developer.with_scope(:find => { :conditions => 'salary > 90000' }) do
1305
+ Developer.find(:all, :conditions => 'id < 5')
1306
+ end
1307
+ assert !scoped_developers.include?(developers(:david)) # David's salary is less than 90,000
1308
+ assert_equal 3, scoped_developers.size
1309
+ end
1310
+
1311
+ def test_scoped_find_limit_offset
1312
+ scoped_developers = Developer.with_scope(:find => { :limit => 3, :offset => 2 }) do
1313
+ Developer.find(:all, :order => 'id')
1314
+ end
1315
+ assert !scoped_developers.include?(developers(:david))
1316
+ assert !scoped_developers.include?(developers(:jamis))
1317
+ assert_equal 3, scoped_developers.size
1318
+
1319
+ # Test without scoped find conditions to ensure we get the whole thing
1320
+ developers = Developer.find(:all, :order => 'id')
1321
+ assert_equal Developer.count, developers.size
1322
+ end
1323
+
1324
+ def test_scoped_find_order
1325
+ # Test order in scope
1326
+ scoped_developers = Developer.with_scope(:find => { :limit => 1, :order => 'salary DESC' }) do
1327
+ Developer.find(:all)
1328
+ end
1329
+ assert_equal 'Jamis', scoped_developers.first.name
1330
+ assert scoped_developers.include?(developers(:jamis))
1331
+ # Test scope without order and order in find
1332
+ scoped_developers = Developer.with_scope(:find => { :limit => 1 }) do
1333
+ Developer.find(:all, :order => 'salary DESC')
1334
+ end
1335
+ # Test scope order + find order, find has priority
1336
+ scoped_developers = Developer.with_scope(:find => { :limit => 3, :order => 'id DESC' }) do
1337
+ Developer.find(:all, :order => 'salary ASC')
1338
+ end
1339
+ assert scoped_developers.include?(developers(:poor_jamis))
1340
+ assert scoped_developers.include?(developers(:david))
1341
+ assert scoped_developers.include?(developers(:dev_10))
1342
+ # Test without scoped find conditions to ensure we get the right thing
1343
+ developers = Developer.find(:all, :order => 'id', :limit => 1)
1344
+ assert scoped_developers.include?(developers(:david))
1345
+ end
1346
+
1347
+ def test_scoped_find_limit_offset_including_has_many_association
1348
+ topics = Topic.with_scope(:find => {:limit => 1, :offset => 1, :include => :replies}) do
1349
+ Topic.find(:all, :order => "topics.id")
1350
+ end
1351
+ assert_equal 1, topics.size
1352
+ assert_equal 2, topics.first.id
1353
+ end
1354
+
1355
+ def test_scoped_find_order_including_has_many_association
1356
+ developers = Developer.with_scope(:find => { :order => 'developers.salary DESC', :include => :projects }) do
1357
+ Developer.find(:all)
1358
+ end
1359
+ assert developers.size >= 2
1360
+ for i in 1...developers.size
1361
+ assert developers[i-1].salary >= developers[i].salary
1362
+ end
1363
+ end
1364
+
1365
+ def test_abstract_class
1366
+ assert !ActiveRecord::Base.abstract_class?
1367
+ assert LoosePerson.abstract_class?
1368
+ assert !LooseDescendant.abstract_class?
1369
+ end
1370
+
1371
+ def test_base_class
1372
+ assert_equal LoosePerson, LoosePerson.base_class
1373
+ assert_equal LooseDescendant, LooseDescendant.base_class
1374
+ assert_equal TightPerson, TightPerson.base_class
1375
+ assert_equal TightPerson, TightDescendant.base_class
1376
+
1377
+ assert_equal Post, Post.base_class
1378
+ assert_equal Post, SpecialPost.base_class
1379
+ assert_equal Post, StiPost.base_class
1380
+ assert_equal SubStiPost, SubStiPost.base_class
1381
+ end
1382
+
1383
+ def test_descends_from_active_record
1384
+ # Tries to call Object.abstract_class?
1385
+ assert_raise(NoMethodError) do
1386
+ ActiveRecord::Base.descends_from_active_record?
1387
+ end
1388
+
1389
+ # Abstract subclass of AR::Base.
1390
+ assert LoosePerson.descends_from_active_record?
1391
+
1392
+ # Concrete subclass of an abstract class.
1393
+ assert LooseDescendant.descends_from_active_record?
1394
+
1395
+ # Concrete subclass of AR::Base.
1396
+ assert TightPerson.descends_from_active_record?
1397
+
1398
+ # Concrete subclass of a concrete class but has no type column.
1399
+ assert TightDescendant.descends_from_active_record?
1400
+
1401
+ # Concrete subclass of AR::Base.
1402
+ assert Post.descends_from_active_record?
1403
+
1404
+ # Abstract subclass of a concrete class which has a type column.
1405
+ # This is pathological, as you'll never have Sub < Abstract < Concrete.
1406
+ assert !StiPost.descends_from_active_record?
1407
+
1408
+ # Concrete subclasses an abstract class which has a type column.
1409
+ assert !SubStiPost.descends_from_active_record?
1410
+ end
1411
+
1412
+ def test_find_on_abstract_base_class_doesnt_use_type_condition
1413
+ old_class = LooseDescendant
1414
+ Object.send :remove_const, :LooseDescendant
1415
+
1416
+ descendant = old_class.create!
1417
+ assert_not_nil LoosePerson.find(descendant.id), "Should have found instance of LooseDescendant when finding abstract LoosePerson: #{descendant.inspect}"
1418
+ ensure
1419
+ unless Object.const_defined?(:LooseDescendant)
1420
+ Object.const_set :LooseDescendant, old_class
1421
+ end
1422
+ end
1423
+
1424
+ def test_assert_queries
1425
+ query = lambda { ActiveRecord::Base.connection.execute 'select count(*) from developers' }
1426
+ assert_queries(2) { 2.times { query.call } }
1427
+ assert_queries 1, &query
1428
+ assert_no_queries { assert true }
1429
+ end
1430
+
1431
+ def test_to_xml
1432
+ xml = topics(:first).to_xml(:indent => 0, :skip_instruct => true)
1433
+ bonus_time_in_current_timezone = topics(:first).bonus_time.xmlschema
1434
+ written_on_in_current_timezone = topics(:first).written_on.xmlschema
1435
+ last_read_in_current_timezone = topics(:first).last_read.xmlschema
1436
+ assert_equal "<topic>", xml.first(7)
1437
+ assert xml.include?(%(<title>The First Topic</title>))
1438
+ assert xml.include?(%(<author-name>David</author-name>))
1439
+ assert xml.include?(%(<id type="integer">1</id>))
1440
+ assert xml.include?(%(<replies-count type="integer">1</replies-count>))
1441
+ assert xml.include?(%(<written-on type="datetime">#{written_on_in_current_timezone}</written-on>))
1442
+ assert xml.include?(%(<content>Have a nice day</content>))
1443
+ assert xml.include?(%(<author-email-address>david@loudthinking.com</author-email-address>))
1444
+ assert xml.match(%(<parent-id type="integer"></parent-id>))
1445
+ if current_adapter?(:SybaseAdapter, :SQLServerAdapter, :OracleAdapter)
1446
+ assert xml.include?(%(<last-read type="datetime">#{last_read_in_current_timezone}</last-read>))
1447
+ else
1448
+ assert xml.include?(%(<last-read type="date">2004-04-15</last-read>))
1449
+ end
1450
+ # Oracle and DB2 don't have true boolean or time-only fields
1451
+ unless current_adapter?(:OracleAdapter, :DB2Adapter)
1452
+ assert xml.include?(%(<approved type="boolean">false</approved>)), "Approved should be a boolean"
1453
+ assert xml.include?(%(<bonus-time type="datetime">#{bonus_time_in_current_timezone}</bonus-time>))
1454
+ end
1455
+ end
1456
+
1457
+ def test_to_xml_skipping_attributes
1458
+ xml = topics(:first).to_xml(:indent => 0, :skip_instruct => true, :except => [:title, :replies_count])
1459
+ assert_equal "<topic>", xml.first(7)
1460
+ assert !xml.include?(%(<title>The First Topic</title>))
1461
+ assert xml.include?(%(<author-name>David</author-name>))
1462
+
1463
+ xml = topics(:first).to_xml(:indent => 0, :skip_instruct => true, :except => [:title, :author_name, :replies_count])
1464
+ assert !xml.include?(%(<title>The First Topic</title>))
1465
+ assert !xml.include?(%(<author-name>David</author-name>))
1466
+ end
1467
+
1468
+ def test_to_xml_including_has_many_association
1469
+ xml = topics(:first).to_xml(:indent => 0, :skip_instruct => true, :include => :replies, :except => :replies_count)
1470
+ assert_equal "<topic>", xml.first(7)
1471
+ assert xml.include?(%(<replies><reply>))
1472
+ assert xml.include?(%(<title>The Second Topic's of the day</title>))
1473
+ end
1474
+
1475
+ def test_array_to_xml_including_has_many_association
1476
+ xml = [ topics(:first), topics(:second) ].to_xml(:indent => 0, :skip_instruct => true, :include => :replies)
1477
+ assert xml.include?(%(<replies><reply>))
1478
+ end
1479
+
1480
+ def test_array_to_xml_including_methods
1481
+ xml = [ topics(:first), topics(:second) ].to_xml(:indent => 0, :skip_instruct => true, :methods => [ :topic_id ])
1482
+ assert xml.include?(%(<topic-id type="integer">#{topics(:first).topic_id}</topic-id>)), xml
1483
+ assert xml.include?(%(<topic-id type="integer">#{topics(:second).topic_id}</topic-id>)), xml
1484
+ end
1485
+
1486
+ def test_array_to_xml_including_has_one_association
1487
+ xml = [ companies(:first_firm), companies(:rails_core) ].to_xml(:indent => 0, :skip_instruct => true, :include => :account)
1488
+ assert xml.include?(companies(:first_firm).account.to_xml(:indent => 0, :skip_instruct => true))
1489
+ assert xml.include?(companies(:rails_core).account.to_xml(:indent => 0, :skip_instruct => true))
1490
+ end
1491
+
1492
+ def test_array_to_xml_including_belongs_to_association
1493
+ xml = [ companies(:first_client), companies(:second_client), companies(:another_client) ].to_xml(:indent => 0, :skip_instruct => true, :include => :firm)
1494
+ assert xml.include?(companies(:first_client).to_xml(:indent => 0, :skip_instruct => true))
1495
+ assert xml.include?(companies(:second_client).firm.to_xml(:indent => 0, :skip_instruct => true))
1496
+ assert xml.include?(companies(:another_client).firm.to_xml(:indent => 0, :skip_instruct => true))
1497
+ end
1498
+
1499
+ def test_to_xml_including_belongs_to_association
1500
+ xml = companies(:first_client).to_xml(:indent => 0, :skip_instruct => true, :include => :firm)
1501
+ assert !xml.include?("<firm>")
1502
+
1503
+ xml = companies(:second_client).to_xml(:indent => 0, :skip_instruct => true, :include => :firm)
1504
+ assert xml.include?("<firm>")
1505
+ end
1506
+
1507
+ def test_to_xml_including_multiple_associations
1508
+ xml = companies(:first_firm).to_xml(:indent => 0, :skip_instruct => true, :include => [ :clients, :account ])
1509
+ assert_equal "<firm>", xml.first(6)
1510
+ assert xml.include?(%(<account>))
1511
+ assert xml.include?(%(<clients><client>))
1512
+ end
1513
+
1514
+ def test_to_xml_including_multiple_associations_with_options
1515
+ xml = companies(:first_firm).to_xml(
1516
+ :indent => 0, :skip_instruct => true,
1517
+ :include => { :clients => { :only => :name } }
1518
+ )
1519
+
1520
+ assert_equal "<firm>", xml.first(6)
1521
+ assert xml.include?(%(<client><name>Summit</name></client>))
1522
+ assert xml.include?(%(<clients><client>))
1523
+ end
1524
+
1525
+ def test_to_xml_including_methods
1526
+ xml = Company.new.to_xml(:methods => :arbitrary_method, :skip_instruct => true)
1527
+ assert_equal "<company>", xml.first(9)
1528
+ assert xml.include?(%(<arbitrary-method>I am Jack's profound disappointment</arbitrary-method>))
1529
+ end
1530
+
1531
+ def test_except_attributes
1532
+ assert_equal(
1533
+ %w( author_name type id approved replies_count bonus_time written_on content author_email_address parent_id last_read),
1534
+ topics(:first).attributes(:except => :title).keys
1535
+ )
1536
+
1537
+ assert_equal(
1538
+ %w( replies_count bonus_time written_on content author_email_address parent_id last_read),
1539
+ topics(:first).attributes(:except => [ :title, :id, :type, :approved, :author_name ]).keys
1540
+ )
1541
+ end
1542
+
1543
+ def test_include_attributes
1544
+ assert_equal(%w( title ), topics(:first).attributes(:only => :title).keys)
1545
+ assert_equal(%w( title author_name type id approved ), topics(:first).attributes(:only => [ :title, :id, :type, :approved, :author_name ]).keys)
1546
+ end
1547
+
1548
+ def test_type_name_with_module_should_handle_beginning
1549
+ assert_equal 'ActiveRecord::Person', ActiveRecord::Base.send(:type_name_with_module, 'Person')
1550
+ assert_equal '::Person', ActiveRecord::Base.send(:type_name_with_module, '::Person')
1551
+ end
1552
+
1553
+ def test_to_param_should_return_string
1554
+ assert_kind_of String, Client.find(:first).to_param
1555
+ end
1556
+
1557
+ # FIXME: this test ought to run, but it needs to run sandboxed so that it
1558
+ # doesn't b0rk the current test environment by undefing everything.
1559
+ #
1560
+ #def test_dev_mode_memory_leak
1561
+ # counts = []
1562
+ # 2.times do
1563
+ # require_dependency 'fixtures/company'
1564
+ # Firm.find(:first)
1565
+ # Dependencies.clear
1566
+ # ActiveRecord::Base.reset_subclasses
1567
+ # Dependencies.remove_subclasses_for(ActiveRecord::Base)
1568
+ #
1569
+ # GC.start
1570
+ #
1571
+ # count = 0
1572
+ # ObjectSpace.each_object(Proc) { count += 1 }
1573
+ # counts << count
1574
+ # end
1575
+ # assert counts.last <= counts.first,
1576
+ # "expected last count (#{counts.last}) to be <= first count (#{counts.first})"
1577
+ #end
1578
+
1579
+ private
1580
+ def assert_readers(model, exceptions)
1581
+ expected_readers = Set.new(model.column_names - ['id'])
1582
+ expected_readers += expected_readers.map { |col| "#{col}?" }
1583
+ expected_readers -= exceptions
1584
+ assert_equal expected_readers, model.read_methods
1585
+ end
1586
+ end