sbf-dm-core 1.3.0.beta

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 +44 -0
  5. data/.rspec +1 -0
  6. data/.rubocop.yml +468 -0
  7. data/.travis.yml +57 -0
  8. data/.yardopts +1 -0
  9. data/Gemfile +70 -0
  10. data/LICENSE +20 -0
  11. data/README.md +269 -0
  12. data/Rakefile +4 -0
  13. data/dm-core.gemspec +21 -0
  14. data/lib/dm-core/adapters/abstract_adapter.rb +233 -0
  15. data/lib/dm-core/adapters/in_memory_adapter.rb +110 -0
  16. data/lib/dm-core/adapters.rb +249 -0
  17. data/lib/dm-core/associations/many_to_many.rb +477 -0
  18. data/lib/dm-core/associations/many_to_one.rb +282 -0
  19. data/lib/dm-core/associations/one_to_many.rb +332 -0
  20. data/lib/dm-core/associations/one_to_one.rb +84 -0
  21. data/lib/dm-core/associations/relationship.rb +650 -0
  22. data/lib/dm-core/backwards.rb +11 -0
  23. data/lib/dm-core/collection.rb +1486 -0
  24. data/lib/dm-core/core_ext/kernel.rb +21 -0
  25. data/lib/dm-core/core_ext/pathname.rb +4 -0
  26. data/lib/dm-core/core_ext/symbol.rb +10 -0
  27. data/lib/dm-core/identity_map.rb +6 -0
  28. data/lib/dm-core/model/hook.rb +99 -0
  29. data/lib/dm-core/model/is.rb +30 -0
  30. data/lib/dm-core/model/property.rb +244 -0
  31. data/lib/dm-core/model/relationship.rb +366 -0
  32. data/lib/dm-core/model/scope.rb +87 -0
  33. data/lib/dm-core/model.rb +876 -0
  34. data/lib/dm-core/property/binary.rb +19 -0
  35. data/lib/dm-core/property/boolean.rb +35 -0
  36. data/lib/dm-core/property/class.rb +23 -0
  37. data/lib/dm-core/property/date.rb +45 -0
  38. data/lib/dm-core/property/date_time.rb +44 -0
  39. data/lib/dm-core/property/decimal.rb +47 -0
  40. data/lib/dm-core/property/discriminator.rb +40 -0
  41. data/lib/dm-core/property/float.rb +27 -0
  42. data/lib/dm-core/property/integer.rb +32 -0
  43. data/lib/dm-core/property/invalid_value_error.rb +17 -0
  44. data/lib/dm-core/property/lookup.rb +26 -0
  45. data/lib/dm-core/property/numeric.rb +35 -0
  46. data/lib/dm-core/property/object.rb +33 -0
  47. data/lib/dm-core/property/serial.rb +13 -0
  48. data/lib/dm-core/property/string.rb +47 -0
  49. data/lib/dm-core/property/text.rb +12 -0
  50. data/lib/dm-core/property/time.rb +46 -0
  51. data/lib/dm-core/property/typecast/numeric.rb +32 -0
  52. data/lib/dm-core/property/typecast/time.rb +33 -0
  53. data/lib/dm-core/property.rb +856 -0
  54. data/lib/dm-core/property_set.rb +177 -0
  55. data/lib/dm-core/query/conditions/comparison.rb +886 -0
  56. data/lib/dm-core/query/conditions/operation.rb +710 -0
  57. data/lib/dm-core/query/direction.rb +33 -0
  58. data/lib/dm-core/query/operator.rb +34 -0
  59. data/lib/dm-core/query/path.rb +113 -0
  60. data/lib/dm-core/query/sort.rb +38 -0
  61. data/lib/dm-core/query.rb +1352 -0
  62. data/lib/dm-core/relationship_set.rb +69 -0
  63. data/lib/dm-core/repository.rb +226 -0
  64. data/lib/dm-core/resource/persistence_state/clean.rb +36 -0
  65. data/lib/dm-core/resource/persistence_state/deleted.rb +26 -0
  66. data/lib/dm-core/resource/persistence_state/dirty.rb +91 -0
  67. data/lib/dm-core/resource/persistence_state/immutable.rb +32 -0
  68. data/lib/dm-core/resource/persistence_state/persisted.rb +25 -0
  69. data/lib/dm-core/resource/persistence_state/transient.rb +87 -0
  70. data/lib/dm-core/resource/persistence_state.rb +70 -0
  71. data/lib/dm-core/resource.rb +1220 -0
  72. data/lib/dm-core/spec/lib/adapter_helpers.rb +63 -0
  73. data/lib/dm-core/spec/lib/collection_helpers.rb +21 -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 +164 -0
  78. data/lib/dm-core/spec/shared/adapter_spec.rb +366 -0
  79. data/lib/dm-core/spec/shared/public/property_spec.rb +229 -0
  80. data/lib/dm-core/spec/shared/resource_spec.rb +1221 -0
  81. data/lib/dm-core/spec/shared/sel_spec.rb +111 -0
  82. data/lib/dm-core/spec/shared/semipublic/property_spec.rb +184 -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 +388 -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 +13 -0
  102. data/lib/dm-core/support/logger.rb +201 -0
  103. data/lib/dm-core/support/mash.rb +176 -0
  104. data/lib/dm-core/support/naming_conventions.rb +109 -0
  105. data/lib/dm-core/support/ordered_set.rb +381 -0
  106. data/lib/dm-core/support/subject.rb +33 -0
  107. data/lib/dm-core/support/subject_set.rb +251 -0
  108. data/lib/dm-core/version.rb +3 -0
  109. data/lib/dm-core.rb +274 -0
  110. data/script/performance.rb +275 -0
  111. data/script/profile.rb +218 -0
  112. data/spec/lib/rspec_immediate_feedback_formatter.rb +54 -0
  113. data/spec/public/associations/many_to_many/read_multiple_join_spec.rb +69 -0
  114. data/spec/public/associations/many_to_many_spec.rb +197 -0
  115. data/spec/public/associations/many_to_one_spec.rb +83 -0
  116. data/spec/public/associations/many_to_one_with_boolean_cpk_spec.rb +40 -0
  117. data/spec/public/associations/many_to_one_with_custom_fk_spec.rb +49 -0
  118. data/spec/public/associations/one_to_many_spec.rb +81 -0
  119. data/spec/public/associations/one_to_one_spec.rb +176 -0
  120. data/spec/public/associations/one_to_one_with_boolean_cpk_spec.rb +46 -0
  121. data/spec/public/collection_spec.rb +69 -0
  122. data/spec/public/finalize_spec.rb +77 -0
  123. data/spec/public/model/hook_spec.rb +245 -0
  124. data/spec/public/model/property_spec.rb +91 -0
  125. data/spec/public/model/relationship_spec.rb +1040 -0
  126. data/spec/public/model_spec.rb +456 -0
  127. data/spec/public/property/binary_spec.rb +43 -0
  128. data/spec/public/property/boolean_spec.rb +21 -0
  129. data/spec/public/property/class_spec.rb +27 -0
  130. data/spec/public/property/date_spec.rb +21 -0
  131. data/spec/public/property/date_time_spec.rb +21 -0
  132. data/spec/public/property/decimal_spec.rb +23 -0
  133. data/spec/public/property/discriminator_spec.rb +134 -0
  134. data/spec/public/property/float_spec.rb +22 -0
  135. data/spec/public/property/integer_spec.rb +22 -0
  136. data/spec/public/property/object_spec.rb +117 -0
  137. data/spec/public/property/serial_spec.rb +22 -0
  138. data/spec/public/property/string_spec.rb +21 -0
  139. data/spec/public/property/text_spec.rb +62 -0
  140. data/spec/public/property/time_spec.rb +21 -0
  141. data/spec/public/property_spec.rb +333 -0
  142. data/spec/public/resource/state_spec.rb +72 -0
  143. data/spec/public/resource_spec.rb +289 -0
  144. data/spec/public/sel_spec.rb +53 -0
  145. data/spec/public/setup_spec.rb +145 -0
  146. data/spec/public/shared/association_collection_shared_spec.rb +309 -0
  147. data/spec/public/shared/collection_finder_shared_spec.rb +267 -0
  148. data/spec/public/shared/collection_shared_spec.rb +1637 -0
  149. data/spec/public/shared/finder_shared_spec.rb +1647 -0
  150. data/spec/semipublic/adapters/abstract_adapter_spec.rb +30 -0
  151. data/spec/semipublic/adapters/in_memory_adapter_spec.rb +13 -0
  152. data/spec/semipublic/associations/many_to_many_spec.rb +94 -0
  153. data/spec/semipublic/associations/many_to_one_spec.rb +63 -0
  154. data/spec/semipublic/associations/one_to_many_spec.rb +55 -0
  155. data/spec/semipublic/associations/one_to_one_spec.rb +53 -0
  156. data/spec/semipublic/associations/relationship_spec.rb +200 -0
  157. data/spec/semipublic/associations_spec.rb +177 -0
  158. data/spec/semipublic/collection_spec.rb +110 -0
  159. data/spec/semipublic/model_spec.rb +96 -0
  160. data/spec/semipublic/property/binary_spec.rb +13 -0
  161. data/spec/semipublic/property/boolean_spec.rb +47 -0
  162. data/spec/semipublic/property/class_spec.rb +33 -0
  163. data/spec/semipublic/property/date_spec.rb +43 -0
  164. data/spec/semipublic/property/date_time_spec.rb +46 -0
  165. data/spec/semipublic/property/decimal_spec.rb +83 -0
  166. data/spec/semipublic/property/discriminator_spec.rb +19 -0
  167. data/spec/semipublic/property/float_spec.rb +82 -0
  168. data/spec/semipublic/property/integer_spec.rb +82 -0
  169. data/spec/semipublic/property/lookup_spec.rb +29 -0
  170. data/spec/semipublic/property/serial_spec.rb +13 -0
  171. data/spec/semipublic/property/string_spec.rb +13 -0
  172. data/spec/semipublic/property/text_spec.rb +31 -0
  173. data/spec/semipublic/property/time_spec.rb +50 -0
  174. data/spec/semipublic/property_spec.rb +114 -0
  175. data/spec/semipublic/query/conditions/comparison_spec.rb +1502 -0
  176. data/spec/semipublic/query/conditions/operation_spec.rb +1296 -0
  177. data/spec/semipublic/query/path_spec.rb +471 -0
  178. data/spec/semipublic/query_spec.rb +3665 -0
  179. data/spec/semipublic/resource/state/clean_spec.rb +89 -0
  180. data/spec/semipublic/resource/state/deleted_spec.rb +79 -0
  181. data/spec/semipublic/resource/state/dirty_spec.rb +163 -0
  182. data/spec/semipublic/resource/state/immutable_spec.rb +107 -0
  183. data/spec/semipublic/resource/state/transient_spec.rb +163 -0
  184. data/spec/semipublic/resource/state_spec.rb +230 -0
  185. data/spec/semipublic/resource_spec.rb +23 -0
  186. data/spec/semipublic/shared/condition_shared_spec.rb +9 -0
  187. data/spec/semipublic/shared/resource_shared_spec.rb +198 -0
  188. data/spec/semipublic/shared/resource_state_shared_spec.rb +91 -0
  189. data/spec/semipublic/shared/subject_shared_spec.rb +79 -0
  190. data/spec/spec_helper.rb +34 -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 +27 -0
  248. data/spec/unit/hook_spec.rb +1216 -0
  249. data/spec/unit/inflections_spec.rb +14 -0
  250. data/spec/unit/lazy_array_spec.rb +1949 -0
  251. data/spec/unit/mash_spec.rb +289 -0
  252. data/spec/unit/module_spec.rb +70 -0
  253. data/spec/unit/object_spec.rb +38 -0
  254. data/spec/unit/try_dup_spec.rb +46 -0
  255. data/tasks/ci.rake +1 -0
  256. data/tasks/spec.rake +18 -0
  257. data/tasks/yard.rake +9 -0
  258. data/tasks/yardstick.rake +19 -0
  259. metadata +323 -0
@@ -0,0 +1,856 @@
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
+ include Subject, DataMapper::Assertions
304
+ extend Equalizer
305
+
306
+ equalize :model, :name, :options
307
+
308
+ PRIMITIVES = [
309
+ TrueClass,
310
+ ::String,
311
+ ::Float,
312
+ ::Integer,
313
+ ::BigDecimal,
314
+ ::DateTime,
315
+ ::Date,
316
+ ::Time,
317
+ ::Class
318
+ ].to_set.freeze
319
+
320
+ OPTIONS = %i(
321
+ load_as dump_as
322
+ accessor reader writer
323
+ lazy default key field
324
+ index unique_index
325
+ unique allow_nil allow_blank required
326
+ ).freeze
327
+
328
+ # Possible :visibility option values
329
+ VISIBILITY_OPTIONS = %i(public protected private).to_set.freeze
330
+
331
+ # Invalid property names
332
+ INVALID_NAMES = (Resource.instance_methods +
333
+ Resource.private_instance_methods +
334
+ Query::OPTIONS.to_a
335
+ ).map(&:to_s)
336
+
337
+ attr_reader :load_as, :dump_as, :model, :name, :instance_variable_name,
338
+ :reader_visibility, :writer_visibility, :options,
339
+ :default, :repository_name, :allow_nil, :allow_blank, :required
340
+
341
+ alias_method :load_class, :load_as
342
+ alias_method :dump_class, :dump_as
343
+
344
+ class << self
345
+ extend Deprecate
346
+
347
+ deprecate :all_descendants, :descendants
348
+
349
+ # @api semipublic
350
+ def determine_class(type)
351
+ return type if type < DataMapper::Property::Object
352
+
353
+ find_class(DataMapper::Inflector.demodulize(type.name))
354
+ end
355
+
356
+ # @api private
357
+ def demodulized_names
358
+ @demodulized_names ||= {}
359
+ end
360
+
361
+ # @api semipublic
362
+ def find_class(name)
363
+ klass = demodulized_names[name]
364
+ klass ||= const_get(name) if const_defined?(name)
365
+ klass
366
+ end
367
+
368
+ # @api public
369
+ def descendants
370
+ @descendants ||= DescendantSet.new
371
+ end
372
+
373
+ # @api private
374
+ def inherited(descendant)
375
+ # Descendants is a tree rooted in DataMapper::Property that tracks
376
+ # inheritance. We pre-calculate each comparison value (demodulized
377
+ # class name) to achieve a Hash[]-time lookup, rather than walk the
378
+ # entire descendant tree and calculate names on-demand (expensive,
379
+ # redundant).
380
+ #
381
+ # Since the algorithm relegates property class name lookups to a flat
382
+ # namespace, we need to ensure properties defined outside of DM don't
383
+ # override built-ins (Serial, String, etc) by merely defining a property
384
+ # of a same name. We avoid this by only ever adding to the lookup
385
+ # table. Given that DM loads its own property classes first, we can
386
+ # assume that their names are "reserved" when added to the table.
387
+ #
388
+ # External property authors who want to provide "replacements" for
389
+ # builtins (e.g. in a non-DM-supported adapter) should follow the
390
+ # convention of wrapping those properties in a module, and include'ing
391
+ # the module on the model class directly. This bypasses the DM-hooked
392
+ # const_missing lookup that would normally check this table.
393
+ descendants << descendant
394
+
395
+ Property.demodulized_names[DataMapper::Inflector.demodulize(descendant.name)] ||= descendant
396
+
397
+ # inherit accepted options
398
+ descendant.accepted_options.concat(accepted_options)
399
+
400
+ # inherit the option values
401
+ options.each { |key, value| descendant.send(key, value) }
402
+
403
+ super
404
+ end
405
+
406
+ # @api public
407
+ def accepted_options
408
+ @accepted_options ||= []
409
+ end
410
+
411
+ # @api public
412
+ def accept_options(*args)
413
+ accepted_options.concat(args)
414
+
415
+ # create methods for each new option
416
+ args.each do |property_option|
417
+ class_eval <<-RUBY, __FILE__, __LINE__ + 1
418
+ def self.#{property_option}(value = Undefined) # def self.unique(value = Undefined)
419
+ return @#{property_option} if value.equal?(Undefined) # return @unique if value.equal?(Undefined)
420
+ descendants.each do |descendant| # descendants.each do |descendant|
421
+ unless descendant.instance_variable_defined?(:@#{property_option}) # unless descendant.instance_variable_defined?(:@unique)
422
+ descendant.#{property_option}(value) # descendant.unique(value)
423
+ end # end
424
+ end # end
425
+ @#{property_option} = value # @unique = value
426
+ end # end
427
+ RUBY
428
+ end
429
+
430
+ descendants.each { |descendant| descendant.accepted_options.concat(args) }
431
+ end
432
+
433
+ # @api private
434
+ def nullable(*_args)
435
+ # :required is preferable to :allow_nil, but :nullable maps precisely to :allow_nil
436
+ raise "#nullable is deprecated, use #required instead (#{caller.first})"
437
+ end
438
+
439
+ # Gives all the options set on this property
440
+ #
441
+ # @return [Hash] with all options and their values set on this property
442
+ #
443
+ # @api public
444
+ def options
445
+ options = {}
446
+ accepted_options.each do |name|
447
+ options[name] = send(name) if instance_variable_defined?("@#{name}")
448
+ end
449
+ options
450
+ end
451
+
452
+ # @api deprecated
453
+ def primitive(*args)
454
+ warn "DataMapper::Property.primitive is deprecated, use .load_as instead (#{caller.first})"
455
+ load_as(*args)
456
+ end
457
+ end
458
+
459
+ accept_options(*Property::OPTIONS)
460
+
461
+ # A hook to allow properties to extend or modify the model it's bound to.
462
+ # Implementations are not supposed to modify the state of the property
463
+ # class, and should produce no side-effects on the property instance.
464
+ def bind
465
+ # no op
466
+ end
467
+
468
+ # Supplies the field in the data-store which the property corresponds to
469
+ #
470
+ # @return [String] name of field in data-store
471
+ #
472
+ # @api semipublic
473
+ def field(repository_name = nil)
474
+ raise "Passing in +repository_name+ to #{self.class}#field is deprecated (#{caller.first})" if repository_name
475
+
476
+ # defer setting the field with the adapter specific naming
477
+ # conventions until after the adapter has been setup
478
+ @field ||= model.field_naming_convention(self.repository_name).call(self).freeze
479
+ end
480
+
481
+ # Returns true if property is unique. Serial properties and keys
482
+ # are unique by default.
483
+ #
484
+ # @return [Boolean]
485
+ # true if property has uniq index defined, false otherwise
486
+ #
487
+ # @api public
488
+ def unique?
489
+ !!@unique
490
+ end
491
+
492
+ # Returns index name if property has index.
493
+ #
494
+ # @return [Boolean, Symbol, Array]
495
+ # returns true if property is indexed by itself
496
+ # returns a Symbol if the property is indexed with other properties
497
+ # returns an Array if the property belongs to multiple indexes
498
+ # returns false if the property does not belong to any indexes
499
+ #
500
+ # @api public
501
+ attr_reader :index
502
+
503
+ # Returns true if property has unique index. Serial properties and
504
+ # keys are unique by default.
505
+ #
506
+ # @return [Boolean, Symbol, Array]
507
+ # returns true if property is indexed by itself
508
+ # returns a Symbol if the property is indexed with other properties
509
+ # returns an Array if the property belongs to multiple indexes
510
+ # returns false if the property does not belong to any indexes
511
+ #
512
+ # @api public
513
+ attr_reader :unique_index
514
+
515
+ # Returns whether or not the property is to be lazy-loaded
516
+ #
517
+ # @return [Boolean]
518
+ # true if the property is to be lazy-loaded
519
+ #
520
+ # @api public
521
+ def lazy?
522
+ @lazy
523
+ end
524
+
525
+ # Returns whether or not the property is a key or a part of a key
526
+ #
527
+ # @return [Boolean]
528
+ # true if the property is a key or a part of a key
529
+ #
530
+ # @api public
531
+ def key?
532
+ @key
533
+ end
534
+
535
+ # Returns whether or not the property is "serial" (auto-incrementing)
536
+ #
537
+ # @return [Boolean]
538
+ # whether or not the property is "serial"
539
+ #
540
+ # @api public
541
+ def serial?
542
+ @serial
543
+ end
544
+
545
+ # Returns whether or not the property must be non-nil and non-blank
546
+ #
547
+ # @return [Boolean]
548
+ # whether or not the property is required
549
+ #
550
+ # @api public
551
+ def required?
552
+ @required
553
+ end
554
+
555
+ # Returns whether or not the property can accept 'nil' as it's value
556
+ #
557
+ # @return [Boolean]
558
+ # whether or not the property can accept 'nil'
559
+ #
560
+ # @api public
561
+ def allow_nil?
562
+ @allow_nil
563
+ end
564
+
565
+ # Returns whether or not the property can be a blank value
566
+ #
567
+ # @return [Boolean]
568
+ # whether or not the property can be blank
569
+ #
570
+ # @api public
571
+ def allow_blank?
572
+ @allow_blank
573
+ end
574
+
575
+ # Standardized reader method for the property
576
+ #
577
+ # @param [Resource] resource
578
+ # model instance for which this property is to be loaded
579
+ #
580
+ # @return [Object]
581
+ # the value of this property for the provided instance
582
+ #
583
+ # @raise [ArgumentError] "+resource+ should be a Resource, but was ...."
584
+ #
585
+ # @api private
586
+ def get(resource)
587
+ get!(resource)
588
+ end
589
+
590
+ # Fetch the ivar value in the resource
591
+ #
592
+ # @param [Resource] resource
593
+ # model instance for which this property is to be unsafely loaded
594
+ #
595
+ # @return [Object]
596
+ # current @ivar value of this property in +resource+
597
+ #
598
+ # @api private
599
+ def get!(resource)
600
+ resource.instance_variable_get(instance_variable_name)
601
+ end
602
+
603
+ # Provides a standardized setter method for the property
604
+ #
605
+ # @param [Resource] resource
606
+ # the resource to get the value from
607
+ # @param [Object] value
608
+ # the value to set in the resource
609
+ #
610
+ # @return [Object]
611
+ # +value+ after being typecasted according to this property's primitive
612
+ #
613
+ # @raise [ArgumentError] "+resource+ should be a Resource, but was ...."
614
+ #
615
+ # @api private
616
+ def set(resource, value)
617
+ set!(resource, value)
618
+ end
619
+
620
+ # Set the ivar value in the resource
621
+ #
622
+ # @param [Resource] resource
623
+ # the resource to set
624
+ # @param [Object] value
625
+ # the value to set in the resource
626
+ #
627
+ # @return [Object]
628
+ # the value set in the resource
629
+ #
630
+ # @api private
631
+ def set!(resource, value)
632
+ resource.instance_variable_set(instance_variable_name, value)
633
+ end
634
+
635
+ # Check if the attribute corresponding to the property is loaded
636
+ #
637
+ # @param [Resource] resource
638
+ # model instance for which the attribute is to be tested
639
+ #
640
+ # @return [Boolean]
641
+ # true if the attribute is loaded in the resource
642
+ #
643
+ # @api private
644
+ def loaded?(resource)
645
+ resource.instance_variable_defined?(instance_variable_name)
646
+ end
647
+
648
+ # Loads lazy columns when get or set is called.
649
+ #
650
+ # @param [Resource] resource
651
+ # model instance for which lazy loaded attribute are loaded
652
+ #
653
+ # @api private
654
+ def lazy_load(resource)
655
+ return if loaded?(resource)
656
+
657
+ resource.__send__(:lazy_load, lazy_load_properties)
658
+ end
659
+
660
+ # @api private
661
+ def lazy_load_properties
662
+ @lazy_load_properties ||=
663
+ begin
664
+ properties = self.properties
665
+ properties.in_context(lazy? ? [self] : properties.defaults)
666
+ end
667
+ end
668
+
669
+ # @api private
670
+ def properties
671
+ @properties ||= model.properties(repository_name)
672
+ end
673
+
674
+ # @api semipublic
675
+ def typecast(value)
676
+ if (value.nil? || value_loaded?(value)) && !respond_to?(:typecast_to_primitive, true)
677
+ value
678
+ elsif respond_to?(:typecast_to_primitive, true)
679
+ typecast_to_primitive(value)
680
+ end
681
+ end
682
+
683
+ # Test the value to see if it is a valid value for this Property
684
+ #
685
+ # @param [Object] value
686
+ # the value to be tested
687
+ #
688
+ # @return [Boolean]
689
+ # true if the value is valid
690
+ #
691
+ # @api semipublic
692
+ def valid?(value, negated = false)
693
+ dumped_value = dump(value)
694
+
695
+ if required? && dumped_value.nil?
696
+ negated || false
697
+ else
698
+ value_dumped?(dumped_value) || (dumped_value.nil? && (allow_nil? || negated))
699
+ end
700
+ end
701
+
702
+ # Asserts value is valid
703
+ #
704
+ # @param [Object] value
705
+ # the value to be tested
706
+ #
707
+ # @return [Boolean]
708
+ # true if the value is valid
709
+ #
710
+ # @raise [Property::InvalidValueError]
711
+ # if value is not valid
712
+ def assert_valid_value(value)
713
+ raise Property::InvalidValueError.new(self, value) unless valid?(value)
714
+
715
+ true
716
+ end
717
+
718
+ # Returns a concise string representation of the property instance.
719
+ #
720
+ # @return [String]
721
+ # Concise string representation of the property instance.
722
+ #
723
+ # @api public
724
+ def inspect
725
+ "#<#{self.class.name} @model=#{model.inspect} @name=#{name.inspect}>"
726
+ end
727
+
728
+ # Test a value to see if it matches the primitive type
729
+ #
730
+ # @param [Object] value
731
+ # value to test
732
+ #
733
+ # @return [Boolean]
734
+ # true if the value is the correct type
735
+ #
736
+ # @api semipublic
737
+ def primitive?(value)
738
+ warn "#primitive? is deprecated, use #value_dumped? instead (#{caller.first})"
739
+ value_dumped?(value)
740
+ end
741
+
742
+ def primitive
743
+ warn "#primitive is deprecated, use #dump_as instead (#{caller.first})"
744
+ dump_as
745
+ end
746
+
747
+ # @api semipublic
748
+ def value_dumped?(value)
749
+ value.is_a?(dump_as)
750
+ end
751
+
752
+ # @api semipublic
753
+ def value_loaded?(value)
754
+ value.is_a?(load_as)
755
+ end
756
+
757
+ # @api semipublic
758
+ protected def initialize(model, name, options = {})
759
+ options = options.to_hash.dup
760
+
761
+ if INVALID_NAMES.include?(name.to_s) || (is_a?(Boolean) && INVALID_NAMES.include?("#{name}?"))
762
+ raise ArgumentError,
763
+ "+name+ was #{name.inspect}, which cannot be used as a property name since it collides with an existing method or a query option"
764
+ end
765
+
766
+ assert_valid_options(options)
767
+
768
+ predefined_options = self.class.options
769
+
770
+ @repository_name = model.repository_name
771
+ @model = model
772
+ @name = name.to_s.chomp('?').to_sym
773
+ @options = predefined_options.merge(options).freeze
774
+ @instance_variable_name = "@#{@name}".freeze
775
+ @load_as = self.class.load_as
776
+ @dump_as = @options.fetch(:dump_as, @load_as)
777
+ @field = @options[:field].freeze unless @options[:field].nil?
778
+ @default = @options[:default]
779
+
780
+ @serial = @options.fetch(:serial, false)
781
+ @key = @options.fetch(:key, @serial)
782
+ @unique = @options.fetch(:unique, @key ? :key : false)
783
+ @required = @options.fetch(:required, @key)
784
+ @allow_nil = @options.fetch(:allow_nil, !@required)
785
+ @allow_blank = @options.fetch(:allow_blank, !@required)
786
+ @index = @options.fetch(:index, false)
787
+ @unique_index = @options.fetch(:unique_index, @unique)
788
+ @lazy = @options.fetch(:lazy, false) && !@key
789
+
790
+ determine_visibility
791
+
792
+ bind
793
+ end
794
+
795
+ # @api private
796
+ protected def assert_valid_options(options)
797
+ keys = options.keys
798
+
799
+ if (unknown_keys = keys - self.class.accepted_options).any?
800
+ raise ArgumentError, "options #{unknown_keys.map(&:inspect).join(' and ')} are unknown"
801
+ end
802
+
803
+ options.each do |key, value|
804
+ boolean_value = [true, false].include?(value)
805
+
806
+ case key
807
+ when :field
808
+ assert_kind_of "options[:#{key}]", value, ::String
809
+
810
+ when :default
811
+ raise ArgumentError, "options[:#{key}] must not be nil" if value.nil?
812
+
813
+ when :serial, :key, :allow_nil, :allow_blank, :required, :auto_validation
814
+ raise ArgumentError, "options[:#{key}] must be either true or false" unless boolean_value
815
+
816
+ if key == :required && (keys.include?(:allow_nil) || keys.include?(:allow_blank))
817
+ raise ArgumentError, 'options[:required] cannot be mixed with :allow_nil or :allow_blank'
818
+ end
819
+
820
+ when :index, :unique_index, :unique, :lazy
821
+ unless boolean_value || value.is_a?(Symbol) || (value.is_a?(Array) && value.any? && value.all? { |val| val.is_a?(Symbol) })
822
+ raise ArgumentError, "options[:#{key}] must be either true, false, a Symbol or an Array of Symbols"
823
+ end
824
+
825
+ when :length
826
+ assert_kind_of "options[:#{key}]", value, Range, ::Integer
827
+
828
+ when :size, :precision, :scale
829
+ assert_kind_of "options[:#{key}]", value, ::Integer
830
+
831
+ when :reader, :writer, :accessor
832
+ assert_kind_of "options[:#{key}]", value, Symbol
833
+
834
+ raise ArgumentError, "options[:#{key}] must be #{VISIBILITY_OPTIONS.join(' or ')}" unless VISIBILITY_OPTIONS.include?(value)
835
+ end
836
+ end
837
+ end
838
+
839
+ # Assert given visibility value is supported.
840
+ #
841
+ # Will raise ArgumentError if this Property's reader and writer
842
+ # visibilities are not included in VISIBILITY_OPTIONS.
843
+ #
844
+ # @return [undefined]
845
+ #
846
+ # @raise [ArgumentError] "property visibility must be :public, :protected, or :private"
847
+ #
848
+ # @api private
849
+ protected def determine_visibility
850
+ default_accessor = @options.fetch(:accessor, :public)
851
+
852
+ @reader_visibility = @options.fetch(:reader, default_accessor)
853
+ @writer_visibility = @options.fetch(:writer, default_accessor)
854
+ end
855
+ end
856
+ end