ardm-core 1.2.1

Sign up to get free protection for your applications and to get access to all the features.
Files changed (259) hide show
  1. checksums.yaml +7 -0
  2. data/.autotest +29 -0
  3. data/.document +5 -0
  4. data/.gitignore +35 -0
  5. data/.travis.yml +23 -0
  6. data/.yardopts +1 -0
  7. data/Gemfile +63 -0
  8. data/LICENSE +20 -0
  9. data/README.rdoc +237 -0
  10. data/Rakefile +4 -0
  11. data/VERSION +1 -0
  12. data/ardm-core.gemspec +25 -0
  13. data/lib/ardm-core.rb +1 -0
  14. data/lib/dm-core.rb +285 -0
  15. data/lib/dm-core/adapters.rb +222 -0
  16. data/lib/dm-core/adapters/abstract_adapter.rb +236 -0
  17. data/lib/dm-core/adapters/in_memory_adapter.rb +113 -0
  18. data/lib/dm-core/associations/many_to_many.rb +496 -0
  19. data/lib/dm-core/associations/many_to_one.rb +296 -0
  20. data/lib/dm-core/associations/one_to_many.rb +345 -0
  21. data/lib/dm-core/associations/one_to_one.rb +86 -0
  22. data/lib/dm-core/associations/relationship.rb +663 -0
  23. data/lib/dm-core/backwards.rb +13 -0
  24. data/lib/dm-core/collection.rb +1514 -0
  25. data/lib/dm-core/core_ext/kernel.rb +23 -0
  26. data/lib/dm-core/core_ext/pathname.rb +6 -0
  27. data/lib/dm-core/core_ext/symbol.rb +10 -0
  28. data/lib/dm-core/identity_map.rb +7 -0
  29. data/lib/dm-core/model.rb +869 -0
  30. data/lib/dm-core/model/hook.rb +102 -0
  31. data/lib/dm-core/model/is.rb +32 -0
  32. data/lib/dm-core/model/property.rb +253 -0
  33. data/lib/dm-core/model/relationship.rb +377 -0
  34. data/lib/dm-core/model/scope.rb +89 -0
  35. data/lib/dm-core/property.rb +839 -0
  36. data/lib/dm-core/property/binary.rb +22 -0
  37. data/lib/dm-core/property/boolean.rb +31 -0
  38. data/lib/dm-core/property/class.rb +24 -0
  39. data/lib/dm-core/property/date.rb +45 -0
  40. data/lib/dm-core/property/date_time.rb +44 -0
  41. data/lib/dm-core/property/decimal.rb +50 -0
  42. data/lib/dm-core/property/discriminator.rb +46 -0
  43. data/lib/dm-core/property/float.rb +28 -0
  44. data/lib/dm-core/property/integer.rb +32 -0
  45. data/lib/dm-core/property/lookup.rb +29 -0
  46. data/lib/dm-core/property/numeric.rb +40 -0
  47. data/lib/dm-core/property/object.rb +28 -0
  48. data/lib/dm-core/property/serial.rb +13 -0
  49. data/lib/dm-core/property/string.rb +50 -0
  50. data/lib/dm-core/property/text.rb +12 -0
  51. data/lib/dm-core/property/time.rb +46 -0
  52. data/lib/dm-core/property/typecast/numeric.rb +32 -0
  53. data/lib/dm-core/property/typecast/time.rb +33 -0
  54. data/lib/dm-core/property_set.rb +177 -0
  55. data/lib/dm-core/query.rb +1444 -0
  56. data/lib/dm-core/query/conditions/comparison.rb +910 -0
  57. data/lib/dm-core/query/conditions/operation.rb +720 -0
  58. data/lib/dm-core/query/direction.rb +36 -0
  59. data/lib/dm-core/query/operator.rb +35 -0
  60. data/lib/dm-core/query/path.rb +114 -0
  61. data/lib/dm-core/query/sort.rb +39 -0
  62. data/lib/dm-core/relationship_set.rb +72 -0
  63. data/lib/dm-core/repository.rb +226 -0
  64. data/lib/dm-core/resource.rb +1228 -0
  65. data/lib/dm-core/resource/persistence_state.rb +75 -0
  66. data/lib/dm-core/resource/persistence_state/clean.rb +40 -0
  67. data/lib/dm-core/resource/persistence_state/deleted.rb +30 -0
  68. data/lib/dm-core/resource/persistence_state/dirty.rb +96 -0
  69. data/lib/dm-core/resource/persistence_state/immutable.rb +34 -0
  70. data/lib/dm-core/resource/persistence_state/persisted.rb +29 -0
  71. data/lib/dm-core/resource/persistence_state/transient.rb +78 -0
  72. data/lib/dm-core/spec/lib/adapter_helpers.rb +54 -0
  73. data/lib/dm-core/spec/lib/collection_helpers.rb +20 -0
  74. data/lib/dm-core/spec/lib/counter_adapter.rb +38 -0
  75. data/lib/dm-core/spec/lib/pending_helpers.rb +50 -0
  76. data/lib/dm-core/spec/lib/spec_helper.rb +74 -0
  77. data/lib/dm-core/spec/setup.rb +173 -0
  78. data/lib/dm-core/spec/shared/adapter_spec.rb +326 -0
  79. data/lib/dm-core/spec/shared/public/property_spec.rb +229 -0
  80. data/lib/dm-core/spec/shared/resource_spec.rb +1236 -0
  81. data/lib/dm-core/spec/shared/sel_spec.rb +111 -0
  82. data/lib/dm-core/spec/shared/semipublic/property_spec.rb +134 -0
  83. data/lib/dm-core/spec/shared/semipublic/query/conditions/abstract_comparison_spec.rb +261 -0
  84. data/lib/dm-core/support/assertions.rb +8 -0
  85. data/lib/dm-core/support/chainable.rb +18 -0
  86. data/lib/dm-core/support/deprecate.rb +12 -0
  87. data/lib/dm-core/support/descendant_set.rb +89 -0
  88. data/lib/dm-core/support/equalizer.rb +48 -0
  89. data/lib/dm-core/support/ext/array.rb +22 -0
  90. data/lib/dm-core/support/ext/blank.rb +25 -0
  91. data/lib/dm-core/support/ext/hash.rb +67 -0
  92. data/lib/dm-core/support/ext/module.rb +47 -0
  93. data/lib/dm-core/support/ext/object.rb +57 -0
  94. data/lib/dm-core/support/ext/string.rb +24 -0
  95. data/lib/dm-core/support/ext/try_dup.rb +12 -0
  96. data/lib/dm-core/support/hook.rb +402 -0
  97. data/lib/dm-core/support/inflections.rb +60 -0
  98. data/lib/dm-core/support/inflector/inflections.rb +211 -0
  99. data/lib/dm-core/support/inflector/methods.rb +151 -0
  100. data/lib/dm-core/support/lazy_array.rb +451 -0
  101. data/lib/dm-core/support/local_object_space.rb +12 -0
  102. data/lib/dm-core/support/logger.rb +199 -0
  103. data/lib/dm-core/support/mash.rb +176 -0
  104. data/lib/dm-core/support/naming_conventions.rb +90 -0
  105. data/lib/dm-core/support/ordered_set.rb +380 -0
  106. data/lib/dm-core/support/subject.rb +33 -0
  107. data/lib/dm-core/support/subject_set.rb +250 -0
  108. data/lib/dm-core/version.rb +3 -0
  109. data/script/performance.rb +275 -0
  110. data/script/profile.rb +218 -0
  111. data/spec/lib/rspec_immediate_feedback_formatter.rb +54 -0
  112. data/spec/public/associations/many_to_many/read_multiple_join_spec.rb +68 -0
  113. data/spec/public/associations/many_to_many_spec.rb +197 -0
  114. data/spec/public/associations/many_to_one_spec.rb +83 -0
  115. data/spec/public/associations/many_to_one_with_boolean_cpk_spec.rb +40 -0
  116. data/spec/public/associations/many_to_one_with_custom_fk_spec.rb +49 -0
  117. data/spec/public/associations/one_to_many_spec.rb +81 -0
  118. data/spec/public/associations/one_to_one_spec.rb +176 -0
  119. data/spec/public/associations/one_to_one_with_boolean_cpk_spec.rb +46 -0
  120. data/spec/public/collection_spec.rb +69 -0
  121. data/spec/public/finalize_spec.rb +76 -0
  122. data/spec/public/model/hook_spec.rb +246 -0
  123. data/spec/public/model/property_spec.rb +88 -0
  124. data/spec/public/model/relationship_spec.rb +1040 -0
  125. data/spec/public/model_spec.rb +458 -0
  126. data/spec/public/property/binary_spec.rb +41 -0
  127. data/spec/public/property/boolean_spec.rb +22 -0
  128. data/spec/public/property/class_spec.rb +28 -0
  129. data/spec/public/property/date_spec.rb +22 -0
  130. data/spec/public/property/date_time_spec.rb +22 -0
  131. data/spec/public/property/decimal_spec.rb +23 -0
  132. data/spec/public/property/discriminator_spec.rb +135 -0
  133. data/spec/public/property/float_spec.rb +22 -0
  134. data/spec/public/property/integer_spec.rb +22 -0
  135. data/spec/public/property/object_spec.rb +107 -0
  136. data/spec/public/property/serial_spec.rb +22 -0
  137. data/spec/public/property/string_spec.rb +22 -0
  138. data/spec/public/property/text_spec.rb +63 -0
  139. data/spec/public/property/time_spec.rb +22 -0
  140. data/spec/public/property_spec.rb +341 -0
  141. data/spec/public/resource_spec.rb +284 -0
  142. data/spec/public/sel_spec.rb +53 -0
  143. data/spec/public/setup_spec.rb +145 -0
  144. data/spec/public/shared/association_collection_shared_spec.rb +309 -0
  145. data/spec/public/shared/collection_finder_shared_spec.rb +267 -0
  146. data/spec/public/shared/collection_shared_spec.rb +1669 -0
  147. data/spec/public/shared/finder_shared_spec.rb +1629 -0
  148. data/spec/rcov.opts +6 -0
  149. data/spec/semipublic/adapters/abstract_adapter_spec.rb +30 -0
  150. data/spec/semipublic/adapters/in_memory_adapter_spec.rb +12 -0
  151. data/spec/semipublic/associations/many_to_many_spec.rb +94 -0
  152. data/spec/semipublic/associations/many_to_one_spec.rb +63 -0
  153. data/spec/semipublic/associations/one_to_many_spec.rb +55 -0
  154. data/spec/semipublic/associations/one_to_one_spec.rb +53 -0
  155. data/spec/semipublic/associations/relationship_spec.rb +200 -0
  156. data/spec/semipublic/associations_spec.rb +177 -0
  157. data/spec/semipublic/collection_spec.rb +110 -0
  158. data/spec/semipublic/model_spec.rb +96 -0
  159. data/spec/semipublic/property/binary_spec.rb +13 -0
  160. data/spec/semipublic/property/boolean_spec.rb +47 -0
  161. data/spec/semipublic/property/class_spec.rb +33 -0
  162. data/spec/semipublic/property/date_spec.rb +43 -0
  163. data/spec/semipublic/property/date_time_spec.rb +46 -0
  164. data/spec/semipublic/property/decimal_spec.rb +83 -0
  165. data/spec/semipublic/property/discriminator_spec.rb +19 -0
  166. data/spec/semipublic/property/float_spec.rb +82 -0
  167. data/spec/semipublic/property/integer_spec.rb +82 -0
  168. data/spec/semipublic/property/lookup_spec.rb +29 -0
  169. data/spec/semipublic/property/serial_spec.rb +13 -0
  170. data/spec/semipublic/property/string_spec.rb +13 -0
  171. data/spec/semipublic/property/text_spec.rb +31 -0
  172. data/spec/semipublic/property/time_spec.rb +50 -0
  173. data/spec/semipublic/property_spec.rb +114 -0
  174. data/spec/semipublic/query/conditions/comparison_spec.rb +1501 -0
  175. data/spec/semipublic/query/conditions/operation_spec.rb +1294 -0
  176. data/spec/semipublic/query/path_spec.rb +471 -0
  177. data/spec/semipublic/query_spec.rb +3777 -0
  178. data/spec/semipublic/resource/state/clean_spec.rb +88 -0
  179. data/spec/semipublic/resource/state/deleted_spec.rb +78 -0
  180. data/spec/semipublic/resource/state/dirty_spec.rb +156 -0
  181. data/spec/semipublic/resource/state/immutable_spec.rb +105 -0
  182. data/spec/semipublic/resource/state/transient_spec.rb +162 -0
  183. data/spec/semipublic/resource/state_spec.rb +230 -0
  184. data/spec/semipublic/resource_spec.rb +23 -0
  185. data/spec/semipublic/shared/condition_shared_spec.rb +9 -0
  186. data/spec/semipublic/shared/resource_shared_spec.rb +199 -0
  187. data/spec/semipublic/shared/resource_state_shared_spec.rb +79 -0
  188. data/spec/semipublic/shared/subject_shared_spec.rb +79 -0
  189. data/spec/spec.opts +5 -0
  190. data/spec/spec_helper.rb +37 -0
  191. data/spec/support/core_ext/hash.rb +10 -0
  192. data/spec/support/core_ext/inheritable_attributes.rb +46 -0
  193. data/spec/support/properties/huge_integer.rb +17 -0
  194. data/spec/unit/array_spec.rb +23 -0
  195. data/spec/unit/blank_spec.rb +73 -0
  196. data/spec/unit/data_mapper/ordered_set/append_spec.rb +26 -0
  197. data/spec/unit/data_mapper/ordered_set/clear_spec.rb +24 -0
  198. data/spec/unit/data_mapper/ordered_set/delete_spec.rb +28 -0
  199. data/spec/unit/data_mapper/ordered_set/each_spec.rb +19 -0
  200. data/spec/unit/data_mapper/ordered_set/empty_spec.rb +20 -0
  201. data/spec/unit/data_mapper/ordered_set/entries_spec.rb +22 -0
  202. data/spec/unit/data_mapper/ordered_set/eql_spec.rb +51 -0
  203. data/spec/unit/data_mapper/ordered_set/equal_value_spec.rb +84 -0
  204. data/spec/unit/data_mapper/ordered_set/hash_spec.rb +12 -0
  205. data/spec/unit/data_mapper/ordered_set/include_spec.rb +23 -0
  206. data/spec/unit/data_mapper/ordered_set/index_spec.rb +28 -0
  207. data/spec/unit/data_mapper/ordered_set/initialize_spec.rb +32 -0
  208. data/spec/unit/data_mapper/ordered_set/merge_spec.rb +36 -0
  209. data/spec/unit/data_mapper/ordered_set/shared/append_spec.rb +24 -0
  210. data/spec/unit/data_mapper/ordered_set/shared/clear_spec.rb +9 -0
  211. data/spec/unit/data_mapper/ordered_set/shared/delete_spec.rb +25 -0
  212. data/spec/unit/data_mapper/ordered_set/shared/each_spec.rb +17 -0
  213. data/spec/unit/data_mapper/ordered_set/shared/empty_spec.rb +9 -0
  214. data/spec/unit/data_mapper/ordered_set/shared/entries_spec.rb +9 -0
  215. data/spec/unit/data_mapper/ordered_set/shared/include_spec.rb +9 -0
  216. data/spec/unit/data_mapper/ordered_set/shared/index_spec.rb +13 -0
  217. data/spec/unit/data_mapper/ordered_set/shared/initialize_spec.rb +28 -0
  218. data/spec/unit/data_mapper/ordered_set/shared/merge_spec.rb +28 -0
  219. data/spec/unit/data_mapper/ordered_set/shared/size_spec.rb +13 -0
  220. data/spec/unit/data_mapper/ordered_set/shared/to_ary_spec.rb +11 -0
  221. data/spec/unit/data_mapper/ordered_set/size_spec.rb +27 -0
  222. data/spec/unit/data_mapper/ordered_set/to_ary_spec.rb +23 -0
  223. data/spec/unit/data_mapper/subject_set/append_spec.rb +47 -0
  224. data/spec/unit/data_mapper/subject_set/clear_spec.rb +34 -0
  225. data/spec/unit/data_mapper/subject_set/delete_spec.rb +40 -0
  226. data/spec/unit/data_mapper/subject_set/each_spec.rb +30 -0
  227. data/spec/unit/data_mapper/subject_set/empty_spec.rb +31 -0
  228. data/spec/unit/data_mapper/subject_set/entries_spec.rb +31 -0
  229. data/spec/unit/data_mapper/subject_set/get_spec.rb +34 -0
  230. data/spec/unit/data_mapper/subject_set/include_spec.rb +32 -0
  231. data/spec/unit/data_mapper/subject_set/named_spec.rb +33 -0
  232. data/spec/unit/data_mapper/subject_set/shared/append_spec.rb +18 -0
  233. data/spec/unit/data_mapper/subject_set/shared/clear_spec.rb +9 -0
  234. data/spec/unit/data_mapper/subject_set/shared/delete_spec.rb +9 -0
  235. data/spec/unit/data_mapper/subject_set/shared/each_spec.rb +9 -0
  236. data/spec/unit/data_mapper/subject_set/shared/empty_spec.rb +9 -0
  237. data/spec/unit/data_mapper/subject_set/shared/entries_spec.rb +9 -0
  238. data/spec/unit/data_mapper/subject_set/shared/get_spec.rb +9 -0
  239. data/spec/unit/data_mapper/subject_set/shared/include_spec.rb +9 -0
  240. data/spec/unit/data_mapper/subject_set/shared/named_spec.rb +9 -0
  241. data/spec/unit/data_mapper/subject_set/shared/size_spec.rb +13 -0
  242. data/spec/unit/data_mapper/subject_set/shared/to_ary_spec.rb +9 -0
  243. data/spec/unit/data_mapper/subject_set/shared/values_at_spec.rb +44 -0
  244. data/spec/unit/data_mapper/subject_set/size_spec.rb +42 -0
  245. data/spec/unit/data_mapper/subject_set/to_ary_spec.rb +34 -0
  246. data/spec/unit/data_mapper/subject_set/values_at_spec.rb +57 -0
  247. data/spec/unit/hash_spec.rb +28 -0
  248. data/spec/unit/hook_spec.rb +1235 -0
  249. data/spec/unit/lazy_array_spec.rb +1949 -0
  250. data/spec/unit/mash_spec.rb +312 -0
  251. data/spec/unit/module_spec.rb +71 -0
  252. data/spec/unit/object_spec.rb +38 -0
  253. data/spec/unit/try_dup_spec.rb +46 -0
  254. data/tasks/ci.rake +1 -0
  255. data/tasks/db.rake +11 -0
  256. data/tasks/spec.rake +38 -0
  257. data/tasks/yard.rake +9 -0
  258. data/tasks/yardstick.rake +19 -0
  259. metadata +491 -0
@@ -0,0 +1,89 @@
1
+ module DataMapper
2
+ module Model
3
+ # Module with query scoping functionality.
4
+ #
5
+ # Scopes are implemented using simple array based
6
+ # stack that is thread local. Default scope can be set
7
+ # on a per repository basis.
8
+ #
9
+ # Scopes are merged as new queries are nested.
10
+ # It is also possible to get exclusive scope access
11
+ # using +with_exclusive_scope+
12
+ module Scope
13
+ # @api private
14
+ def default_scope(repository_name = default_repository_name)
15
+ @default_scope ||= {}
16
+
17
+ default_repository_name = self.default_repository_name
18
+
19
+ @default_scope[repository_name] ||= if repository_name == default_repository_name
20
+ {}
21
+ else
22
+ default_scope(default_repository_name).dup
23
+ end
24
+ end
25
+
26
+ # Returns query on top of scope stack
27
+ #
28
+ # @api private
29
+ def query
30
+ repository.new_query(self, current_scope).freeze
31
+ end
32
+
33
+ # @api private
34
+ def current_scope
35
+ scope_stack.last || default_scope(repository.name)
36
+ end
37
+
38
+ protected
39
+
40
+ # Pushes given query on top of the stack
41
+ #
42
+ # @param [Hash, Query] Query to add to current scope nesting
43
+ #
44
+ # @api private
45
+ def with_scope(query)
46
+ options = if query.kind_of?(Hash)
47
+ query
48
+ else
49
+ query.options
50
+ end
51
+
52
+ # merge the current scope with the passed in query
53
+ with_exclusive_scope(self.query.merge(options)) { |*block_args| yield(*block_args) }
54
+ end
55
+
56
+ # Pushes given query on top of scope stack and yields
57
+ # given block, then pops the stack. During block execution
58
+ # queries previously pushed onto the stack
59
+ # have no effect.
60
+ #
61
+ # @api private
62
+ def with_exclusive_scope(query)
63
+ query = if query.kind_of?(Hash)
64
+ repository.new_query(self, query)
65
+ else
66
+ query.dup
67
+ end
68
+
69
+ scope_stack = self.scope_stack
70
+ scope_stack << query.options
71
+
72
+ begin
73
+ yield query.freeze
74
+ ensure
75
+ scope_stack.pop
76
+ end
77
+ end
78
+
79
+ # Initializes (if necessary) and returns current scope stack
80
+ # @api private
81
+ def scope_stack
82
+ scope_stack_for = Thread.current[:dm_scope_stack] ||= {}
83
+ scope_stack_for[object_id] ||= []
84
+ end
85
+ end # module Scope
86
+
87
+ include Scope
88
+ end # module Model
89
+ end # module DataMapper
@@ -0,0 +1,839 @@
1
+ module DataMapper
2
+ # = Properties
3
+ # Properties for a model are not derived from a database structure, but
4
+ # instead explicitly declared inside your model class definitions. These
5
+ # properties then map (or, if using automigrate, generate) fields in your
6
+ # repository/database.
7
+ #
8
+ # If you are coming to DataMapper from another ORM framework, such as
9
+ # ActiveRecord, this may be a fundamental difference in thinking to you.
10
+ # However, there are several advantages to defining your properties in your
11
+ # models:
12
+ #
13
+ # * information about your model is centralized in one place: rather than
14
+ # having to dig out migrations, xml or other configuration files.
15
+ # * use of mixins can be applied to model properties: better code reuse
16
+ # * having information centralized in your models, encourages you and the
17
+ # developers on your team to take a model-centric view of development.
18
+ # * it provides the ability to use Ruby's access control functions.
19
+ # * and, because DataMapper only cares about properties explicitly defined
20
+ # in your models, DataMapper plays well with legacy databases, and shares
21
+ # databases easily with other applications.
22
+ #
23
+ # == Declaring Properties
24
+ # Inside your class, you call the property method for each property you want
25
+ # to add. The only two required arguments are the name and type, everything
26
+ # else is optional.
27
+ #
28
+ # class Post
29
+ # include DataMapper::Resource
30
+ #
31
+ # property :title, String, :required => true # Cannot be null
32
+ # property :publish, Boolean, :default => false # Default value for new records is false
33
+ # end
34
+ #
35
+ # By default, DataMapper supports the following primitive (Ruby) types
36
+ # also called core properties:
37
+ #
38
+ # * Boolean
39
+ # * Class (datastore primitive is the same as String. Used for Inheritance)
40
+ # * Date
41
+ # * DateTime
42
+ # * Decimal
43
+ # * Float
44
+ # * Integer
45
+ # * Object (marshalled out during serialization)
46
+ # * String (default length is 50)
47
+ # * Text (limit of 65k characters by default)
48
+ # * Time
49
+ #
50
+ # == Limiting Access
51
+ # Property access control is uses the same terminology Ruby does. Properties
52
+ # are public by default, but can also be declared private or protected as
53
+ # needed (via the :accessor option).
54
+ #
55
+ # class Post
56
+ # include DataMapper::Resource
57
+ #
58
+ # property :title, String, :accessor => :private # Both reader and writer are private
59
+ # property :body, Text, :accessor => :protected # Both reader and writer are protected
60
+ # end
61
+ #
62
+ # Access control is also analogous to Ruby attribute readers and writers, and can
63
+ # be declared using :reader and :writer, in addition to :accessor.
64
+ #
65
+ # class Post
66
+ # include DataMapper::Resource
67
+ #
68
+ # property :title, String, :writer => :private # Only writer is private
69
+ # property :tags, String, :reader => :protected # Only reader is protected
70
+ # end
71
+ #
72
+ # == Overriding Accessors
73
+ # The reader/writer for any property can be overridden in the same manner that Ruby
74
+ # attr readers/writers can be. After the property is defined, just add your custom
75
+ # reader or writer:
76
+ #
77
+ # class Post
78
+ # include DataMapper::Resource
79
+ #
80
+ # property :title, String
81
+ #
82
+ # def title=(new_title)
83
+ # raise ArgumentError if new_title != 'Lee is l337'
84
+ # super(new_title)
85
+ # end
86
+ # end
87
+ #
88
+ # Calling super ensures that any validators defined for the property are kept active.
89
+ #
90
+ # == Lazy Loading
91
+ # By default, some properties are not loaded when an object is fetched in
92
+ # DataMapper. These lazily loaded properties are fetched on demand when their
93
+ # accessor is called for the first time (as it is often unnecessary to
94
+ # instantiate -every- property -every- time an object is loaded). For
95
+ # instance, DataMapper::Property::Text fields are lazy loading by default,
96
+ # although you can over-ride this behavior if you wish:
97
+ #
98
+ # Example:
99
+ #
100
+ # class Post
101
+ # include DataMapper::Resource
102
+ #
103
+ # property :title, String # Loads normally
104
+ # property :body, Text # Is lazily loaded by default
105
+ # end
106
+ #
107
+ # If you want to over-ride the lazy loading on any field you can set it to a
108
+ # context or false to disable it with the :lazy option. Contexts allow
109
+ # multiple lazy properties to be loaded at one time. If you set :lazy to
110
+ # true, it is placed in the :default context
111
+ #
112
+ # class Post
113
+ # include DataMapper::Resource
114
+ #
115
+ # property :title, String # Loads normally
116
+ # property :body, Text, :lazy => false # The default is now over-ridden
117
+ # property :comment, String, :lazy => [ :detailed ] # Loads in the :detailed context
118
+ # property :author, String, :lazy => [ :summary, :detailed ] # Loads in :summary & :detailed context
119
+ # end
120
+ #
121
+ # Delaying the request for lazy-loaded attributes even applies to objects
122
+ # accessed through associations. In a sense, DataMapper anticipates that
123
+ # you will likely be iterating over objects in associations and rolls all
124
+ # of the load commands for lazy-loaded properties into one request from
125
+ # the database.
126
+ #
127
+ # Example:
128
+ #
129
+ # Widget.get(1).components
130
+ # # loads when the post object is pulled from database, by default
131
+ #
132
+ # Widget.get(1).components.first.body
133
+ # # loads the values for the body property on all objects in the
134
+ # # association, rather than just this one.
135
+ #
136
+ # Widget.get(1).components.first.comment
137
+ # # loads both comment and author for all objects in the association
138
+ # # since they are both in the :detailed context
139
+ #
140
+ # == Keys
141
+ # Properties can be declared as primary or natural keys on a table.
142
+ # You should a property as the primary key of the table:
143
+ #
144
+ # Examples:
145
+ #
146
+ # property :id, Serial # auto-incrementing key
147
+ # property :legacy_pk, String, :key => true # 'natural' key
148
+ #
149
+ # This is roughly equivalent to ActiveRecord's <tt>set_primary_key</tt>,
150
+ # though non-integer data types may be used, thus DataMapper supports natural
151
+ # keys. When a property is declared as a natural key, accessing the object
152
+ # using the indexer syntax <tt>Class[key]</tt> remains valid.
153
+ #
154
+ # User.get(1)
155
+ # # when :id is the primary key on the users table
156
+ # User.get('bill')
157
+ # # when :name is the primary (natural) key on the users table
158
+ #
159
+ # == Indices
160
+ # You can add indices for your properties by using the <tt>:index</tt>
161
+ # option. If you use <tt>true</tt> as the option value, the index will be
162
+ # automatically named. If you want to name the index yourself, use a symbol
163
+ # as the value.
164
+ #
165
+ # property :last_name, String, :index => true
166
+ # property :first_name, String, :index => :name
167
+ #
168
+ # You can create multi-column composite indices by using the same symbol in
169
+ # all the columns belonging to the index. The columns will appear in the
170
+ # index in the order they are declared.
171
+ #
172
+ # property :last_name, String, :index => :name
173
+ # property :first_name, String, :index => :name
174
+ # # => index on (last_name, first_name)
175
+ #
176
+ # If you want to make the indices unique, use <tt>:unique_index</tt> instead
177
+ # of <tt>:index</tt>
178
+ #
179
+ # == Inferred Validations
180
+ # If you require the dm-validations plugin, auto-validations will
181
+ # automatically be mixed-in in to your model classes: validation rules that
182
+ # are inferred when properties are declared with specific column restrictions.
183
+ #
184
+ # class Post
185
+ # include DataMapper::Resource
186
+ #
187
+ # property :title, String, :length => 250, :min => 0, :max => 250
188
+ # # => infers 'validates_length :title'
189
+ #
190
+ # property :title, String, :required => true
191
+ # # => infers 'validates_present :title'
192
+ #
193
+ # property :email, String, :format => :email_address
194
+ # # => infers 'validates_format :email, :with => :email_address'
195
+ #
196
+ # property :title, String, :length => 255, :required => true
197
+ # # => infers both 'validates_length' as well as 'validates_present'
198
+ # # better: property :title, String, :length => 1..255
199
+ # end
200
+ #
201
+ # This functionality is available with the dm-validations gem. For more information
202
+ # about validations, check the documentation for dm-validations.
203
+ #
204
+ # == Default Values
205
+ # To set a default for a property, use the <tt>:default</tt> key. The
206
+ # property will be set to the value associated with that key the first time
207
+ # it is accessed, or when the resource is saved if it hasn't been set with
208
+ # another value already. This value can be a static value, such as 'hello'
209
+ # but it can also be a proc that will be evaluated when the property is read
210
+ # before its value has been set. The property is set to the return of the
211
+ # proc. The proc is passed two values, the resource the property is being set
212
+ # for and the property itself.
213
+ #
214
+ # property :display_name, String, :default => lambda { |resource, property| resource.login }
215
+ #
216
+ # Word of warning. Don't try to read the value of the property you're setting
217
+ # the default for in the proc. An infinite loop will ensue.
218
+ #
219
+ # == Embedded Values (not implemented yet)
220
+ # As an alternative to extraneous has_one relationships, consider using an
221
+ # EmbeddedValue.
222
+ #
223
+ # == Property options reference
224
+ #
225
+ # :accessor if false, neither reader nor writer methods are
226
+ # created for this property
227
+ #
228
+ # :reader if false, reader method is not created for this property
229
+ #
230
+ # :writer if false, writer method is not created for this property
231
+ #
232
+ # :lazy if true, property value is only loaded when on first read
233
+ # if false, property value is always loaded
234
+ # if a symbol, property value is loaded with other properties
235
+ # in the same group
236
+ #
237
+ # :default default value of this property
238
+ #
239
+ # :allow_nil if true, property may have a nil value on save
240
+ #
241
+ # :key name of the key associated with this property.
242
+ #
243
+ # :field field in the data-store which the property corresponds to
244
+ #
245
+ # :length string field length
246
+ #
247
+ # :format format for autovalidation. Use with dm-validations plugin.
248
+ #
249
+ # :index if true, index is created for the property. If a Symbol, index
250
+ # is named after Symbol value instead of being based on property name.
251
+ #
252
+ # :unique_index true specifies that index on this property should be unique
253
+ #
254
+ # :auto_validation if true, automatic validation is performed on the property
255
+ #
256
+ # :validates validation context. Use together with dm-validations.
257
+ #
258
+ # :unique if true, property column is unique. Properties of type Serial
259
+ # are unique by default.
260
+ #
261
+ # :precision Indicates the number of significant digits. Usually only makes sense
262
+ # for float type properties. Must be >= scale option value. Default is 10.
263
+ #
264
+ # :scale The number of significant digits to the right of the decimal point.
265
+ # Only makes sense for float type properties. Must be > 0.
266
+ # Default is nil for Float type and 10 for BigDecimal
267
+ #
268
+ # == Overriding default Property options
269
+ #
270
+ # There is the ability to reconfigure a Property and it's subclasses by explicitly
271
+ # setting a value in the Property, eg:
272
+ #
273
+ # # set all String properties to have a default length of 255
274
+ # DataMapper::Property::String.length(255)
275
+ #
276
+ # # set all Boolean properties to not allow nil (force true or false)
277
+ # DataMapper::Property::Boolean.allow_nil(false)
278
+ #
279
+ # # set all properties to be required by default
280
+ # DataMapper::Property.required(true)
281
+ #
282
+ # # turn off auto-validation for all properties by default
283
+ # DataMapper::Property.auto_validation(false)
284
+ #
285
+ # # set all mutator methods to be private by default
286
+ # DataMapper::Property.writer(:private)
287
+ #
288
+ # Please note that this has no effect when a subclass has explicitly
289
+ # defined it's own option. For example, setting the String length to
290
+ # 255 will not affect the Text property even though it inherits from
291
+ # String, because it sets it's own default length to 65535.
292
+ #
293
+ # == Misc. Notes
294
+ # * Properties declared as strings will default to a length of 50, rather than
295
+ # 255 (typical max varchar column size). To overload the default, pass
296
+ # <tt>:length => 255</tt> or <tt>:length => 0..255</tt>. Since DataMapper
297
+ # does not introspect for properties, this means that legacy database tables
298
+ # may need their <tt>String</tt> columns defined with a <tt>:length</tt> so
299
+ # that DM does not apply an un-needed length validation, or allow overflow.
300
+ # * You may declare a Property with the data-type of <tt>Class</tt>.
301
+ # see SingleTableInheritance for more on how to use <tt>Class</tt> columns.
302
+ class Property
303
+ module PassThroughLoadDump
304
+ # @api semipublic
305
+ def load(value)
306
+ typecast(value) unless value.nil?
307
+ end
308
+
309
+ # Stub instance method for dumping
310
+ #
311
+ # @param value [Object, nil] value to dump
312
+ #
313
+ # @return [Object] Dumped object
314
+ #
315
+ # @api semipublic
316
+ def dump(value)
317
+ value
318
+ end
319
+ end
320
+
321
+ include DataMapper::Assertions
322
+ include Subject
323
+ extend Equalizer
324
+
325
+ equalize :model, :name, :options
326
+
327
+ PRIMITIVES = [
328
+ TrueClass,
329
+ ::String,
330
+ ::Float,
331
+ ::Integer,
332
+ ::BigDecimal,
333
+ ::DateTime,
334
+ ::Date,
335
+ ::Time,
336
+ ::Class
337
+ ].to_set.freeze
338
+
339
+ OPTIONS = [
340
+ :accessor, :reader, :writer,
341
+ :lazy, :default, :key, :field,
342
+ :index, :unique_index,
343
+ :unique, :allow_nil, :allow_blank, :required
344
+ ]
345
+
346
+ # Possible :visibility option values
347
+ VISIBILITY_OPTIONS = [ :public, :protected, :private ].to_set.freeze
348
+
349
+ # Invalid property names
350
+ INVALID_NAMES = (Resource.instance_methods +
351
+ Resource.private_instance_methods +
352
+ Query::OPTIONS.to_a
353
+ ).map { |name| name.to_s }
354
+
355
+ attr_reader :primitive, :model, :name, :instance_variable_name,
356
+ :reader_visibility, :writer_visibility, :options,
357
+ :default, :repository_name, :allow_nil, :allow_blank, :required
358
+
359
+ class << self
360
+ extend Deprecate
361
+
362
+ deprecate :all_descendants, :descendants
363
+
364
+ # @api semipublic
365
+ def determine_class(type)
366
+ return type if type < DataMapper::Property::Object
367
+ find_class(DataMapper::Inflector.demodulize(type.name))
368
+ end
369
+
370
+ # @api private
371
+ def demodulized_names
372
+ @demodulized_names ||= {}
373
+ end
374
+
375
+ # @api semipublic
376
+ def find_class(name)
377
+ klass = demodulized_names[name]
378
+ klass ||= const_get(name) if const_defined?(name)
379
+ klass
380
+ end
381
+
382
+ # @api public
383
+ def descendants
384
+ @descendants ||= DescendantSet.new
385
+ end
386
+
387
+ # @api private
388
+ def inherited(descendant)
389
+ # Descendants is a tree rooted in DataMapper::Property that tracks
390
+ # inheritance. We pre-calculate each comparison value (demodulized
391
+ # class name) to achieve a Hash[]-time lookup, rather than walk the
392
+ # entire descendant tree and calculate names on-demand (expensive,
393
+ # redundant).
394
+ #
395
+ # Since the algorithm relegates property class name lookups to a flat
396
+ # namespace, we need to ensure properties defined outside of DM don't
397
+ # override built-ins (Serial, String, etc) by merely defining a property
398
+ # of a same name. We avoid this by only ever adding to the lookup
399
+ # table. Given that DM loads its own property classes first, we can
400
+ # assume that their names are "reserved" when added to the table.
401
+ #
402
+ # External property authors who want to provide "replacements" for
403
+ # builtins (e.g. in a non-DM-supported adapter) should follow the
404
+ # convention of wrapping those properties in a module, and include'ing
405
+ # the module on the model class directly. This bypasses the DM-hooked
406
+ # const_missing lookup that would normally check this table.
407
+ descendants << descendant
408
+
409
+ Property.demodulized_names[DataMapper::Inflector.demodulize(descendant.name)] ||= descendant
410
+
411
+ # inherit accepted options
412
+ descendant.accepted_options.concat(accepted_options)
413
+
414
+ # inherit the option values
415
+ options.each { |key, value| descendant.send(key, value) }
416
+ end
417
+
418
+ # @api public
419
+ def accepted_options
420
+ @accepted_options ||= []
421
+ end
422
+
423
+ # @api public
424
+ def accept_options(*args)
425
+ accepted_options.concat(args)
426
+
427
+ # create methods for each new option
428
+ args.each do |property_option|
429
+ class_eval <<-RUBY, __FILE__, __LINE__ + 1
430
+ def self.#{property_option}(value = Undefined) # def self.unique(value = Undefined)
431
+ return @#{property_option} if value.equal?(Undefined) # return @unique if value.equal?(Undefined)
432
+ descendants.each do |descendant| # descendants.each do |descendant|
433
+ unless descendant.instance_variable_defined?(:@#{property_option}) # unless descendant.instance_variable_defined?(:@unique)
434
+ descendant.#{property_option}(value) # descendant.unique(value)
435
+ end # end
436
+ end # end
437
+ @#{property_option} = value # @unique = value
438
+ end # end
439
+ RUBY
440
+ end
441
+
442
+ descendants.each { |descendant| descendant.accepted_options.concat(args) }
443
+ end
444
+
445
+ # @api private
446
+ def nullable(*args)
447
+ # :required is preferable to :allow_nil, but :nullable maps precisely to :allow_nil
448
+ raise "#nullable is deprecated, use #required instead (#{caller.first})"
449
+ end
450
+
451
+ # Gives all the options set on this property
452
+ #
453
+ # @return [Hash] with all options and their values set on this property
454
+ #
455
+ # @api public
456
+ def options
457
+ options = {}
458
+ accepted_options.each do |name|
459
+ options[name] = send(name) if instance_variable_defined?("@#{name}")
460
+ end
461
+ options
462
+ end
463
+ end
464
+
465
+ accept_options :primitive, *Property::OPTIONS
466
+
467
+ # A hook to allow properties to extend or modify the model it's bound to.
468
+ # Implementations are not supposed to modify the state of the property
469
+ # class, and should produce no side-effects on the property instance.
470
+ def bind
471
+ # no op
472
+ end
473
+
474
+ # Supplies the field in the data-store which the property corresponds to
475
+ #
476
+ # @return [String] name of field in data-store
477
+ #
478
+ # @api semipublic
479
+ def field(repository_name = nil)
480
+ if repository_name
481
+ raise "Passing in +repository_name+ to #{self.class}#field is deprecated (#{caller.first})"
482
+ end
483
+
484
+ # defer setting the field with the adapter specific naming
485
+ # conventions until after the adapter has been setup
486
+ @field ||= model.field_naming_convention(self.repository_name).call(self).freeze
487
+ end
488
+
489
+ # Returns true if property is unique. Serial properties and keys
490
+ # are unique by default.
491
+ #
492
+ # @return [Boolean]
493
+ # true if property has uniq index defined, false otherwise
494
+ #
495
+ # @api public
496
+ def unique?
497
+ !!@unique
498
+ end
499
+
500
+ # Returns index name if property has index.
501
+ #
502
+ # @return [Boolean, Symbol, Array]
503
+ # returns true if property is indexed by itself
504
+ # returns a Symbol if the property is indexed with other properties
505
+ # returns an Array if the property belongs to multiple indexes
506
+ # returns false if the property does not belong to any indexes
507
+ #
508
+ # @api public
509
+ attr_reader :index
510
+
511
+ # Returns true if property has unique index. Serial properties and
512
+ # keys are unique by default.
513
+ #
514
+ # @return [Boolean, Symbol, Array]
515
+ # returns true if property is indexed by itself
516
+ # returns a Symbol if the property is indexed with other properties
517
+ # returns an Array if the property belongs to multiple indexes
518
+ # returns false if the property does not belong to any indexes
519
+ #
520
+ # @api public
521
+ attr_reader :unique_index
522
+
523
+ # Returns whether or not the property is to be lazy-loaded
524
+ #
525
+ # @return [Boolean]
526
+ # true if the property is to be lazy-loaded
527
+ #
528
+ # @api public
529
+ def lazy?
530
+ @lazy
531
+ end
532
+
533
+ # Returns whether or not the property is a key or a part of a key
534
+ #
535
+ # @return [Boolean]
536
+ # true if the property is a key or a part of a key
537
+ #
538
+ # @api public
539
+ def key?
540
+ @key
541
+ end
542
+
543
+ # Returns whether or not the property is "serial" (auto-incrementing)
544
+ #
545
+ # @return [Boolean]
546
+ # whether or not the property is "serial"
547
+ #
548
+ # @api public
549
+ def serial?
550
+ @serial
551
+ end
552
+
553
+ # Returns whether or not the property must be non-nil and non-blank
554
+ #
555
+ # @return [Boolean]
556
+ # whether or not the property is required
557
+ #
558
+ # @api public
559
+ def required?
560
+ @required
561
+ end
562
+
563
+ # Returns whether or not the property can accept 'nil' as it's value
564
+ #
565
+ # @return [Boolean]
566
+ # whether or not the property can accept 'nil'
567
+ #
568
+ # @api public
569
+ def allow_nil?
570
+ @allow_nil
571
+ end
572
+
573
+ # Returns whether or not the property can be a blank value
574
+ #
575
+ # @return [Boolean]
576
+ # whether or not the property can be blank
577
+ #
578
+ # @api public
579
+ def allow_blank?
580
+ @allow_blank
581
+ end
582
+
583
+ # Standardized reader method for the property
584
+ #
585
+ # @param [Resource] resource
586
+ # model instance for which this property is to be loaded
587
+ #
588
+ # @return [Object]
589
+ # the value of this property for the provided instance
590
+ #
591
+ # @raise [ArgumentError] "+resource+ should be a Resource, but was ...."
592
+ #
593
+ # @api private
594
+ def get(resource)
595
+ get!(resource)
596
+ end
597
+
598
+ # Fetch the ivar value in the resource
599
+ #
600
+ # @param [Resource] resource
601
+ # model instance for which this property is to be unsafely loaded
602
+ #
603
+ # @return [Object]
604
+ # current @ivar value of this property in +resource+
605
+ #
606
+ # @api private
607
+ def get!(resource)
608
+ resource.instance_variable_get(instance_variable_name)
609
+ end
610
+
611
+ # Provides a standardized setter method for the property
612
+ #
613
+ # @param [Resource] resource
614
+ # the resource to get the value from
615
+ # @param [Object] value
616
+ # the value to set in the resource
617
+ #
618
+ # @return [Object]
619
+ # +value+ after being typecasted according to this property's primitive
620
+ #
621
+ # @raise [ArgumentError] "+resource+ should be a Resource, but was ...."
622
+ #
623
+ # @api private
624
+ def set(resource, value)
625
+ set!(resource, typecast(value))
626
+ end
627
+
628
+ # Set the ivar value in the resource
629
+ #
630
+ # @param [Resource] resource
631
+ # the resource to set
632
+ # @param [Object] value
633
+ # the value to set in the resource
634
+ #
635
+ # @return [Object]
636
+ # the value set in the resource
637
+ #
638
+ # @api private
639
+ def set!(resource, value)
640
+ resource.instance_variable_set(instance_variable_name, value)
641
+ end
642
+
643
+ # Check if the attribute corresponding to the property is loaded
644
+ #
645
+ # @param [Resource] resource
646
+ # model instance for which the attribute is to be tested
647
+ #
648
+ # @return [Boolean]
649
+ # true if the attribute is loaded in the resource
650
+ #
651
+ # @api private
652
+ def loaded?(resource)
653
+ resource.instance_variable_defined?(instance_variable_name)
654
+ end
655
+
656
+ # Loads lazy columns when get or set is called.
657
+ #
658
+ # @param [Resource] resource
659
+ # model instance for which lazy loaded attribute are loaded
660
+ #
661
+ # @api private
662
+ def lazy_load(resource)
663
+ return if loaded?(resource)
664
+ resource.__send__(:lazy_load, lazy_load_properties)
665
+ end
666
+
667
+ # @api private
668
+ def lazy_load_properties
669
+ @lazy_load_properties ||=
670
+ begin
671
+ properties = self.properties
672
+ properties.in_context(lazy? ? [ self ] : properties.defaults)
673
+ end
674
+ end
675
+
676
+ # @api private
677
+ def properties
678
+ @properties ||= model.properties(repository_name)
679
+ end
680
+
681
+ # @api semipublic
682
+ def typecast(value)
683
+ if value.nil? || primitive?(value)
684
+ value
685
+ elsif respond_to?(:typecast_to_primitive)
686
+ typecast_to_primitive(value)
687
+ end
688
+ end
689
+
690
+ # Test the value to see if it is a valid value for this Property
691
+ #
692
+ # @param [Object] loaded_value
693
+ # the value to be tested
694
+ #
695
+ # @return [Boolean]
696
+ # true if the value is valid
697
+ #
698
+ # @api semipulic
699
+ def valid?(value, negated = false)
700
+ dumped_value = dump(value)
701
+
702
+ if required? && dumped_value.nil?
703
+ negated || false
704
+ else
705
+ primitive?(dumped_value) || (dumped_value.nil? && (allow_nil? || negated))
706
+ end
707
+ end
708
+
709
+ # Returns a concise string representation of the property instance.
710
+ #
711
+ # @return [String]
712
+ # Concise string representation of the property instance.
713
+ #
714
+ # @api public
715
+ def inspect
716
+ "#<#{self.class.name} @model=#{model.inspect} @name=#{name.inspect}>"
717
+ end
718
+
719
+ # Test a value to see if it matches the primitive type
720
+ #
721
+ # @param [Object] value
722
+ # value to test
723
+ #
724
+ # @return [Boolean]
725
+ # true if the value is the correct type
726
+ #
727
+ # @api semipublic
728
+ def primitive?(value)
729
+ value.kind_of?(primitive)
730
+ end
731
+
732
+ protected
733
+
734
+ # @api semipublic
735
+ def initialize(model, name, options = {})
736
+ options = options.to_hash.dup
737
+
738
+ if INVALID_NAMES.include?(name.to_s) || (kind_of?(Boolean) && INVALID_NAMES.include?("#{name}?"))
739
+ raise ArgumentError,
740
+ "+name+ was #{name.inspect}, which cannot be used as a property name since it collides with an existing method or a query option"
741
+ end
742
+
743
+ assert_valid_options(options)
744
+
745
+ predefined_options = self.class.options
746
+
747
+ @repository_name = model.repository_name
748
+ @model = model
749
+ @name = name.to_s.chomp('?').to_sym
750
+ @options = predefined_options.merge(options).freeze
751
+ @instance_variable_name = "@#{@name}".freeze
752
+
753
+ @primitive = self.class.primitive
754
+ @field = @options[:field].freeze unless @options[:field].nil?
755
+ @default = @options[:default]
756
+
757
+ @serial = @options.fetch(:serial, false)
758
+ @key = @options.fetch(:key, @serial)
759
+ @unique = @options.fetch(:unique, @key ? :key : false)
760
+ @required = @options.fetch(:required, @key)
761
+ @allow_nil = @options.fetch(:allow_nil, !@required)
762
+ @allow_blank = @options.fetch(:allow_blank, !@required)
763
+ @index = @options.fetch(:index, false)
764
+ @unique_index = @options.fetch(:unique_index, @unique)
765
+ @lazy = @options.fetch(:lazy, false) && !@key
766
+
767
+ determine_visibility
768
+
769
+ bind
770
+ end
771
+
772
+ # @api private
773
+ def assert_valid_options(options)
774
+ keys = options.keys
775
+
776
+ if (unknown_keys = keys - self.class.accepted_options).any?
777
+ raise ArgumentError, "options #{unknown_keys.map { |key| key.inspect }.join(' and ')} are unknown"
778
+ end
779
+
780
+ options.each do |key, value|
781
+ boolean_value = value == true || value == false
782
+
783
+ case key
784
+ when :field
785
+ assert_kind_of "options[:#{key}]", value, ::String
786
+
787
+ when :default
788
+ if value.nil?
789
+ raise ArgumentError, "options[:#{key}] must not be nil"
790
+ end
791
+
792
+ when :serial, :key, :allow_nil, :allow_blank, :required, :auto_validation
793
+ unless boolean_value
794
+ raise ArgumentError, "options[:#{key}] must be either true or false"
795
+ end
796
+
797
+ if key == :required && (keys.include?(:allow_nil) || keys.include?(:allow_blank))
798
+ raise ArgumentError, 'options[:required] cannot be mixed with :allow_nil or :allow_blank'
799
+ end
800
+
801
+ when :index, :unique_index, :unique, :lazy
802
+ unless boolean_value || value.kind_of?(Symbol) || (value.kind_of?(Array) && value.any? && value.all? { |val| val.kind_of?(Symbol) })
803
+ raise ArgumentError, "options[:#{key}] must be either true, false, a Symbol or an Array of Symbols"
804
+ end
805
+
806
+ when :length
807
+ assert_kind_of "options[:#{key}]", value, Range, ::Integer
808
+
809
+ when :size, :precision, :scale
810
+ assert_kind_of "options[:#{key}]", value, ::Integer
811
+
812
+ when :reader, :writer, :accessor
813
+ assert_kind_of "options[:#{key}]", value, Symbol
814
+
815
+ unless VISIBILITY_OPTIONS.include?(value)
816
+ raise ArgumentError, "options[:#{key}] must be #{VISIBILITY_OPTIONS.join(' or ')}"
817
+ end
818
+ end
819
+ end
820
+ end
821
+
822
+ # Assert given visibility value is supported.
823
+ #
824
+ # Will raise ArgumentError if this Property's reader and writer
825
+ # visibilities are not included in VISIBILITY_OPTIONS.
826
+ #
827
+ # @return [undefined]
828
+ #
829
+ # @raise [ArgumentError] "property visibility must be :public, :protected, or :private"
830
+ #
831
+ # @api private
832
+ def determine_visibility
833
+ default_accessor = @options.fetch(:accessor, :public)
834
+
835
+ @reader_visibility = @options.fetch(:reader, default_accessor)
836
+ @writer_visibility = @options.fetch(:writer, default_accessor)
837
+ end
838
+ end # class Property
839
+ end