dynamoid 3.3.0 → 3.7.0
Sign up to get free protection for your applications and to get access to all the features.
- checksums.yaml +4 -4
- data/CHANGELOG.md +104 -1
- data/README.md +146 -52
- data/lib/dynamoid.rb +1 -0
- data/lib/dynamoid/adapter.rb +20 -7
- data/lib/dynamoid/adapter_plugin/aws_sdk_v3.rb +70 -37
- data/lib/dynamoid/adapter_plugin/aws_sdk_v3/batch_get_item.rb +3 -0
- data/lib/dynamoid/adapter_plugin/aws_sdk_v3/create_table.rb +20 -12
- data/lib/dynamoid/adapter_plugin/aws_sdk_v3/item_updater.rb +5 -4
- data/lib/dynamoid/adapter_plugin/aws_sdk_v3/middleware/backoff.rb +2 -2
- data/lib/dynamoid/adapter_plugin/aws_sdk_v3/middleware/limit.rb +2 -3
- data/lib/dynamoid/adapter_plugin/aws_sdk_v3/middleware/start_key.rb +2 -2
- data/lib/dynamoid/adapter_plugin/aws_sdk_v3/query.rb +4 -2
- data/lib/dynamoid/adapter_plugin/aws_sdk_v3/scan.rb +4 -2
- data/lib/dynamoid/adapter_plugin/aws_sdk_v3/table.rb +1 -0
- data/lib/dynamoid/adapter_plugin/aws_sdk_v3/until_past_table_status.rb +2 -1
- data/lib/dynamoid/application_time_zone.rb +1 -0
- data/lib/dynamoid/associations.rb +182 -19
- data/lib/dynamoid/associations/association.rb +10 -2
- data/lib/dynamoid/associations/belongs_to.rb +2 -1
- data/lib/dynamoid/associations/has_and_belongs_to_many.rb +2 -1
- data/lib/dynamoid/associations/has_many.rb +2 -1
- data/lib/dynamoid/associations/has_one.rb +2 -1
- data/lib/dynamoid/associations/many_association.rb +68 -23
- data/lib/dynamoid/associations/single_association.rb +31 -4
- data/lib/dynamoid/components.rb +2 -0
- data/lib/dynamoid/config.rb +15 -3
- data/lib/dynamoid/config/backoff_strategies/constant_backoff.rb +1 -0
- data/lib/dynamoid/config/backoff_strategies/exponential_backoff.rb +1 -0
- data/lib/dynamoid/config/options.rb +1 -0
- data/lib/dynamoid/criteria.rb +9 -1
- data/lib/dynamoid/criteria/chain.rb +421 -46
- data/lib/dynamoid/criteria/ignored_conditions_detector.rb +3 -3
- data/lib/dynamoid/criteria/key_fields_detector.rb +31 -10
- data/lib/dynamoid/criteria/nonexistent_fields_detector.rb +3 -2
- data/lib/dynamoid/criteria/overwritten_conditions_detector.rb +1 -1
- data/lib/dynamoid/dirty.rb +119 -64
- data/lib/dynamoid/document.rb +133 -46
- data/lib/dynamoid/dumping.rb +9 -0
- data/lib/dynamoid/dynamodb_time_zone.rb +1 -0
- data/lib/dynamoid/errors.rb +2 -0
- data/lib/dynamoid/fields.rb +251 -39
- data/lib/dynamoid/fields/declare.rb +86 -0
- data/lib/dynamoid/finders.rb +69 -32
- data/lib/dynamoid/identity_map.rb +6 -0
- data/lib/dynamoid/indexes.rb +86 -17
- data/lib/dynamoid/loadable.rb +2 -2
- data/lib/dynamoid/log/formatter.rb +26 -0
- data/lib/dynamoid/middleware/identity_map.rb +1 -0
- data/lib/dynamoid/persistence.rb +502 -104
- data/lib/dynamoid/persistence/import.rb +2 -1
- data/lib/dynamoid/persistence/save.rb +1 -0
- data/lib/dynamoid/persistence/update_fields.rb +5 -2
- data/lib/dynamoid/persistence/update_validations.rb +18 -0
- data/lib/dynamoid/persistence/upsert.rb +5 -3
- data/lib/dynamoid/primary_key_type_mapping.rb +1 -0
- data/lib/dynamoid/railtie.rb +1 -0
- data/lib/dynamoid/tasks.rb +3 -1
- data/lib/dynamoid/tasks/database.rb +1 -0
- data/lib/dynamoid/type_casting.rb +12 -2
- data/lib/dynamoid/undumping.rb +8 -0
- data/lib/dynamoid/validations.rb +6 -1
- data/lib/dynamoid/version.rb +1 -1
- metadata +48 -75
- data/.coveralls.yml +0 -1
- data/.document +0 -5
- data/.gitignore +0 -74
- data/.rspec +0 -2
- data/.rubocop.yml +0 -71
- data/.rubocop_todo.yml +0 -55
- data/.travis.yml +0 -44
- data/Appraisals +0 -22
- data/Gemfile +0 -8
- data/Rakefile +0 -46
- data/Vagrantfile +0 -29
- data/docker-compose.yml +0 -7
- data/dynamoid.gemspec +0 -57
- data/gemfiles/rails_4_2.gemfile +0 -9
- data/gemfiles/rails_5_0.gemfile +0 -8
- data/gemfiles/rails_5_1.gemfile +0 -8
- data/gemfiles/rails_5_2.gemfile +0 -8
- data/gemfiles/rails_6_0.gemfile +0 -8
data/lib/dynamoid/document.rb
CHANGED
@@ -1,6 +1,6 @@
|
|
1
1
|
# frozen_string_literal: true
|
2
2
|
|
3
|
-
module Dynamoid
|
3
|
+
module Dynamoid
|
4
4
|
# This is the base module for all domain objects that need to be persisted to
|
5
5
|
# the database as documents.
|
6
6
|
module Document
|
@@ -17,28 +17,19 @@ module Dynamoid #:nodoc:
|
|
17
17
|
end
|
18
18
|
|
19
19
|
module ClassMethods
|
20
|
-
#
|
21
|
-
# write capacity.
|
22
|
-
#
|
23
|
-
# @param [Hash] options options to pass for this table
|
24
|
-
# @option options [Symbol] :name the name for the table; this still gets namespaced
|
25
|
-
# @option options [Symbol] :id id column for the table
|
26
|
-
# @option options [Integer] :read_capacity set the read capacity for the table; does not work on existing tables
|
27
|
-
# @option options [Integer] :write_capacity set the write capacity for the table; does not work on existing tables
|
28
|
-
#
|
29
|
-
# @since 0.4.0
|
20
|
+
# @private
|
30
21
|
def table(options = {})
|
31
22
|
self.options = options
|
32
23
|
super if defined? super
|
33
24
|
end
|
34
25
|
|
35
26
|
def attr_readonly(*read_only_attributes)
|
36
|
-
ActiveSupport::Deprecation.warn('[Dynamoid] .attr_readonly is deprecated! Call .find instead of')
|
37
27
|
self.read_only_attributes.concat read_only_attributes.map(&:to_s)
|
38
28
|
end
|
39
29
|
|
40
|
-
# Returns the
|
30
|
+
# Returns the read capacity for this table.
|
41
31
|
#
|
32
|
+
# @return [Integer] read capacity units
|
42
33
|
# @since 0.4.0
|
43
34
|
def read_capacity
|
44
35
|
options[:read_capacity] || Dynamoid::Config.read_capacity
|
@@ -46,25 +37,53 @@ module Dynamoid #:nodoc:
|
|
46
37
|
|
47
38
|
# Returns the write_capacity for this table.
|
48
39
|
#
|
40
|
+
# @return [Integer] write capacity units
|
49
41
|
# @since 0.4.0
|
50
42
|
def write_capacity
|
51
43
|
options[:write_capacity] || Dynamoid::Config.write_capacity
|
52
44
|
end
|
53
45
|
|
46
|
+
# Returns the billing (capacity) mode for this table.
|
47
|
+
#
|
48
|
+
# Could be either +provisioned+ or +on_demand+.
|
49
|
+
#
|
50
|
+
# @return [Symbol]
|
51
|
+
def capacity_mode
|
52
|
+
options[:capacity_mode] || Dynamoid::Config.capacity_mode
|
53
|
+
end
|
54
|
+
|
54
55
|
# Returns the field name used to support STI for this table.
|
56
|
+
#
|
57
|
+
# Default field name is +type+ but it can be overrided in the +table+
|
58
|
+
# method call.
|
59
|
+
#
|
60
|
+
# User.inheritance_field # => :type
|
55
61
|
def inheritance_field
|
56
62
|
options[:inheritance_field] || :type
|
57
63
|
end
|
58
64
|
|
59
|
-
# Returns the
|
65
|
+
# Returns the hash key field name for this class.
|
60
66
|
#
|
67
|
+
# By default +id+ field is used. But it can be overriden in the +table+
|
68
|
+
# method call.
|
69
|
+
#
|
70
|
+
# User.hash_key # => :id
|
71
|
+
#
|
72
|
+
# @return [Symbol] a hash key name
|
61
73
|
# @since 0.4.0
|
62
74
|
def hash_key
|
63
75
|
options[:key] || :id
|
64
76
|
end
|
65
77
|
|
66
|
-
#
|
78
|
+
# Return the count of items for this class.
|
79
|
+
#
|
80
|
+
# It returns aproximate value based on DynamoDB statistic. DynamoDB
|
81
|
+
# updates it periodicaly so the value can be no accurate.
|
82
|
+
#
|
83
|
+
# It's a reletivly cheap operation and doesn't read all the items in a
|
84
|
+
# table. It makes just one HTTP request to DynamoDB.
|
67
85
|
#
|
86
|
+
# @return [Integer] items count in a table
|
68
87
|
# @since 0.6.1
|
69
88
|
def count
|
70
89
|
Dynamoid.adapter.count(table_name)
|
@@ -72,35 +91,67 @@ module Dynamoid #:nodoc:
|
|
72
91
|
|
73
92
|
# Initialize a new object.
|
74
93
|
#
|
75
|
-
#
|
94
|
+
# User.build(name: 'A')
|
76
95
|
#
|
77
|
-
#
|
96
|
+
# Initialize an object and pass it into a block to set other attributes.
|
97
|
+
#
|
98
|
+
# User.build(name: 'A') do |u|
|
99
|
+
# u.age = 21
|
100
|
+
# end
|
101
|
+
#
|
102
|
+
# The only difference between +build+ and +new+ methods is that +build+
|
103
|
+
# supports STI (Single table inheritance) and looks at the inheritance
|
104
|
+
# field. So it can build a model of actual class. For instance:
|
105
|
+
#
|
106
|
+
# class Employee
|
107
|
+
# include Dynamoid::Document
|
78
108
|
#
|
109
|
+
# field :type
|
110
|
+
# field :name
|
111
|
+
# end
|
112
|
+
#
|
113
|
+
# class Manager < Employee
|
114
|
+
# end
|
115
|
+
#
|
116
|
+
# Employee.build(name: 'Alice', type: 'Manager') # => #<Manager:0x00007f945756e3f0 ...>
|
117
|
+
#
|
118
|
+
# @param attrs [Hash] Attributes with which to create the document
|
119
|
+
# @param block [Proc] Block to process a document after initialization
|
120
|
+
# @return [Dynamoid::Document] the new document
|
79
121
|
# @since 0.2.0
|
80
|
-
def build(attrs = {})
|
81
|
-
choose_right_class(attrs).new(attrs)
|
122
|
+
def build(attrs = {}, &block)
|
123
|
+
choose_right_class(attrs).new(attrs, &block)
|
82
124
|
end
|
83
125
|
|
84
|
-
# Does this
|
126
|
+
# Does this model exist in a table?
|
127
|
+
#
|
128
|
+
# User.exists?('713') # => true
|
85
129
|
#
|
86
|
-
#
|
87
|
-
# Multiple keys and single compound primary key should be passed only as Array explicitily.
|
130
|
+
# If a range key is declared it should be specified in the following way:
|
88
131
|
#
|
89
|
-
#
|
132
|
+
# User.exists?([['713', 'range-key-value']]) # => true
|
90
133
|
#
|
91
|
-
#
|
134
|
+
# It's possible to check existence of several models at once:
|
92
135
|
#
|
93
|
-
#
|
136
|
+
# User.exists?(['713', '714', '715'])
|
94
137
|
#
|
95
|
-
#
|
138
|
+
# Or in case when a range key is declared:
|
96
139
|
#
|
97
|
-
#
|
98
|
-
#
|
140
|
+
# User.exists?(
|
141
|
+
# [
|
142
|
+
# ['713', 'range-key-value-1'],
|
143
|
+
# ['714', 'range-key-value-2'],
|
144
|
+
# ['715', 'range-key-value-3']
|
145
|
+
# ]
|
146
|
+
# )
|
99
147
|
#
|
100
|
-
#
|
148
|
+
# It's also possible to specify models not with primary key but with
|
149
|
+
# conditions on the attributes (in the +where+ method style):
|
101
150
|
#
|
102
|
-
#
|
151
|
+
# User.exists?(age: 20, 'created_at.gt': Time.now - 1.day)
|
103
152
|
#
|
153
|
+
# @param id_or_conditions [String|Array[String]|Array[Array]|Hash] the primary id of the model, a list of primary ids or a hash with the options to filter from.
|
154
|
+
# @return [true|false]
|
104
155
|
# @since 0.2.0
|
105
156
|
def exists?(id_or_conditions = {})
|
106
157
|
case id_or_conditions
|
@@ -115,10 +166,12 @@ module Dynamoid #:nodoc:
|
|
115
166
|
end
|
116
167
|
end
|
117
168
|
|
169
|
+
# @private
|
118
170
|
def deep_subclasses
|
119
171
|
subclasses + subclasses.map(&:deep_subclasses).flatten
|
120
172
|
end
|
121
173
|
|
174
|
+
# @private
|
122
175
|
def choose_right_class(attrs)
|
123
176
|
attrs[inheritance_field] ? attrs[inheritance_field].constantize : self
|
124
177
|
end
|
@@ -126,36 +179,50 @@ module Dynamoid #:nodoc:
|
|
126
179
|
|
127
180
|
# Initialize a new object.
|
128
181
|
#
|
129
|
-
#
|
182
|
+
# User.new(name: 'A')
|
130
183
|
#
|
184
|
+
# Initialize an object and pass it into a block to set other attributes.
|
185
|
+
#
|
186
|
+
# User.new(name: 'A') do |u|
|
187
|
+
# u.age = 21
|
188
|
+
# end
|
189
|
+
#
|
190
|
+
# @param attrs [Hash] Attributes with which to create the document
|
191
|
+
# @param block [Proc] Block to process a document after initialization
|
131
192
|
# @return [Dynamoid::Document] the new document
|
132
193
|
#
|
133
194
|
# @since 0.2.0
|
134
|
-
def initialize(attrs = {})
|
195
|
+
def initialize(attrs = {}, &block)
|
135
196
|
run_callbacks :initialize do
|
136
197
|
@new_record = true
|
137
198
|
@attributes ||= {}
|
138
199
|
@associations ||= {}
|
139
200
|
@attributes_before_type_cast ||= {}
|
140
201
|
|
141
|
-
attrs_with_defaults = self.class.attributes.
|
202
|
+
attrs_with_defaults = self.class.attributes.each_with_object({}) do |(attribute, options), res|
|
142
203
|
if attrs.key?(attribute)
|
143
|
-
res
|
204
|
+
res[attribute] = attrs[attribute]
|
144
205
|
elsif options.key?(:default)
|
145
|
-
res
|
146
|
-
else
|
147
|
-
res
|
206
|
+
res[attribute] = evaluate_default_value(options[:default])
|
148
207
|
end
|
149
208
|
end
|
150
209
|
|
151
210
|
attrs_virtual = attrs.slice(*(attrs.keys - self.class.attributes.keys))
|
152
211
|
|
153
212
|
load(attrs_with_defaults.merge(attrs_virtual))
|
213
|
+
|
214
|
+
if block
|
215
|
+
block.call(self)
|
216
|
+
end
|
154
217
|
end
|
155
218
|
end
|
156
219
|
|
157
|
-
#
|
220
|
+
# Check equality of two models.
|
158
221
|
#
|
222
|
+
# A model is equal to another model only if their primary keys (hash key
|
223
|
+
# and optionaly range key) are equal.
|
224
|
+
#
|
225
|
+
# @return [true|false]
|
159
226
|
# @since 0.2.0
|
160
227
|
def ==(other)
|
161
228
|
if self.class.identity_map_on?
|
@@ -167,36 +234,54 @@ module Dynamoid #:nodoc:
|
|
167
234
|
end
|
168
235
|
end
|
169
236
|
|
237
|
+
# Check equality of two models.
|
238
|
+
#
|
239
|
+
# Works exactly like +==+ does.
|
240
|
+
#
|
241
|
+
# @return [true|false]
|
170
242
|
def eql?(other)
|
171
243
|
self == other
|
172
244
|
end
|
173
245
|
|
246
|
+
# Generate an Integer hash value for this model.
|
247
|
+
#
|
248
|
+
# Hash value is based on primary key. So models can be used safely as a
|
249
|
+
# +Hash+ keys.
|
250
|
+
#
|
251
|
+
# @return [Integer]
|
174
252
|
def hash
|
175
253
|
hash_key.hash ^ range_value.hash
|
176
254
|
end
|
177
255
|
|
178
|
-
# Return
|
256
|
+
# Return a model's hash key value.
|
179
257
|
#
|
180
258
|
# @since 0.4.0
|
181
259
|
def hash_key
|
182
|
-
|
260
|
+
self[self.class.hash_key.to_sym]
|
183
261
|
end
|
184
262
|
|
185
|
-
# Assign
|
263
|
+
# Assign a model's hash key value, regardless of what it might be called to
|
264
|
+
# the object.
|
186
265
|
#
|
187
266
|
# @since 0.4.0
|
188
267
|
def hash_key=(value)
|
189
|
-
|
268
|
+
self[self.class.hash_key.to_sym] = value
|
190
269
|
end
|
191
270
|
|
271
|
+
# Return a model's range key value.
|
272
|
+
#
|
273
|
+
# Returns +nil+ if a range key isn't declared for a model.
|
192
274
|
def range_value
|
193
|
-
if
|
194
|
-
|
275
|
+
if self.class.range_key
|
276
|
+
self[self.class.range_key.to_sym]
|
195
277
|
end
|
196
278
|
end
|
197
279
|
|
280
|
+
# Assign a model's range key value.
|
198
281
|
def range_value=(value)
|
199
|
-
|
282
|
+
if self.class.range_key
|
283
|
+
self[self.class.range_key.to_sym] = value
|
284
|
+
end
|
200
285
|
end
|
201
286
|
|
202
287
|
private
|
@@ -208,7 +293,7 @@ module Dynamoid #:nodoc:
|
|
208
293
|
# Evaluates the default value given, this is used by undump
|
209
294
|
# when determining the value of the default given for a field options.
|
210
295
|
#
|
211
|
-
# @param [Object]
|
296
|
+
# @param val [Object] the attribute's default value
|
212
297
|
def evaluate_default_value(val)
|
213
298
|
if val.respond_to?(:call)
|
214
299
|
val.call
|
@@ -220,3 +305,5 @@ module Dynamoid #:nodoc:
|
|
220
305
|
end
|
221
306
|
end
|
222
307
|
end
|
308
|
+
|
309
|
+
ActiveSupport.run_load_hooks(:dynamoid, Dynamoid::Document)
|
data/lib/dynamoid/dumping.rb
CHANGED
@@ -1,6 +1,7 @@
|
|
1
1
|
# frozen_string_literal: true
|
2
2
|
|
3
3
|
module Dynamoid
|
4
|
+
# @private
|
4
5
|
module Dumping
|
5
6
|
def self.dump_attributes(attributes, attributes_options)
|
6
7
|
{}.tap do |h|
|
@@ -35,6 +36,7 @@ module Dynamoid
|
|
35
36
|
when :serialized then SerializedDumper
|
36
37
|
when :raw then RawDumper
|
37
38
|
when :boolean then BooleanDumper
|
39
|
+
when :binary then BinaryDumper
|
38
40
|
when Class then CustomTypeDumper
|
39
41
|
end
|
40
42
|
|
@@ -287,6 +289,13 @@ module Dynamoid
|
|
287
289
|
end
|
288
290
|
end
|
289
291
|
|
292
|
+
# string -> string
|
293
|
+
class BinaryDumper < Base
|
294
|
+
def process(value)
|
295
|
+
Base64.strict_encode64(value)
|
296
|
+
end
|
297
|
+
end
|
298
|
+
|
290
299
|
# any object -> string
|
291
300
|
class CustomTypeDumper < Base
|
292
301
|
def process(value)
|
data/lib/dynamoid/errors.rb
CHANGED
data/lib/dynamoid/fields.rb
CHANGED
@@ -1,11 +1,14 @@
|
|
1
1
|
# frozen_string_literal: true
|
2
2
|
|
3
|
-
|
3
|
+
require 'dynamoid/fields/declare'
|
4
|
+
|
5
|
+
module Dynamoid
|
4
6
|
# All fields on a Dynamoid::Document must be explicitly defined -- if you have fields in the database that are not
|
5
7
|
# specified with field, then they will be ignored.
|
6
8
|
module Fields
|
7
9
|
extend ActiveSupport::Concern
|
8
10
|
|
11
|
+
# @private
|
9
12
|
# Types allowed in indexes:
|
10
13
|
PERMITTED_KEY_TYPES = %i[
|
11
14
|
number
|
@@ -21,8 +24,11 @@ module Dynamoid #:nodoc:
|
|
21
24
|
class_attribute :range_key
|
22
25
|
|
23
26
|
self.attributes = {}
|
24
|
-
|
25
|
-
|
27
|
+
|
28
|
+
# Timestamp fields could be disabled later in `table` method call.
|
29
|
+
# So let's declare them here and remove them later if it will be necessary
|
30
|
+
field :created_at, :datetime if Dynamoid::Config.timestamps
|
31
|
+
field :updated_at, :datetime if Dynamoid::Config.timestamps
|
26
32
|
|
27
33
|
field :id # Default primary key
|
28
34
|
end
|
@@ -30,59 +36,214 @@ module Dynamoid #:nodoc:
|
|
30
36
|
module ClassMethods
|
31
37
|
# Specify a field for a document.
|
32
38
|
#
|
33
|
-
#
|
34
|
-
#
|
39
|
+
# class User
|
40
|
+
# include Dynamoid::Document
|
41
|
+
#
|
42
|
+
# field :last_name
|
43
|
+
# field :age, :integer
|
44
|
+
# field :last_sign_in, :datetime
|
45
|
+
# end
|
46
|
+
#
|
47
|
+
# Its type determines how it is coerced when read in and out of the
|
48
|
+
# data store. You can specify +string+, +integer+, +number+, +set+, +array+,
|
49
|
+
# +map+, +datetime+, +date+, +serialized+, +raw+, +boolean+ and +binary+
|
35
50
|
# or specify a class that defines a serialization strategy.
|
36
51
|
#
|
52
|
+
# By default field type is +string+.
|
53
|
+
#
|
54
|
+
# Set can store elements of the same type only (it's a limitation of
|
55
|
+
# DynamoDB itself). If a set should store elements only of some particular
|
56
|
+
# type then +of+ option should be specified:
|
57
|
+
#
|
58
|
+
# field :hobbies, :set, of: :string
|
59
|
+
#
|
60
|
+
# Only +string+, +integer+, +number+, +date+, +datetime+ and +serialized+
|
61
|
+
# element types are supported.
|
62
|
+
#
|
63
|
+
# Element type can have own options - they should be specified in the
|
64
|
+
# form of +Hash+:
|
65
|
+
#
|
66
|
+
# field :hobbies, :set, of: { serialized: { serializer: JSON } }
|
67
|
+
#
|
68
|
+
# Array can contain element of different types but if supports the same
|
69
|
+
# +of+ option to convert all the provided elements to the declared type.
|
70
|
+
#
|
71
|
+
# field :rates, :array, of: :number
|
72
|
+
#
|
73
|
+
# By default +date+ and +datetime+ fields are stored as integer values.
|
74
|
+
# The format can be changed to string with option +store_as_string+:
|
75
|
+
#
|
76
|
+
# field :published_on, :datetime, store_as_string: true
|
77
|
+
#
|
78
|
+
# Boolean field by default is stored as a string +t+ or +f+. But DynamoDB
|
79
|
+
# supports boolean type natively. In order to switch to the native
|
80
|
+
# boolean type an option +store_as_native_boolean+ should be specified:
|
81
|
+
#
|
82
|
+
# field :active, :boolean, store_as_native_boolean: true
|
83
|
+
#
|
84
|
+
# If you specify the +serialized+ type a value will be serialized to
|
85
|
+
# string in Yaml format by default. Custom way to serialize value to
|
86
|
+
# string can be specified with +serializer+ option. Custom serializer
|
87
|
+
# should have +dump+ and +load+ methods.
|
88
|
+
#
|
37
89
|
# If you specify a class for field type, Dynamoid will serialize using
|
38
|
-
#
|
90
|
+
# +dynamoid_dump+ method and load using +dynamoid_load+ method.
|
91
|
+
#
|
92
|
+
# Default field type is +string+.
|
93
|
+
#
|
94
|
+
# A field can have a default value. It's assigned at initializing a model
|
95
|
+
# if no value is specified:
|
39
96
|
#
|
40
|
-
#
|
97
|
+
# field :age, :integer, default: 1
|
41
98
|
#
|
42
|
-
#
|
43
|
-
#
|
44
|
-
#
|
99
|
+
# If a defautl value should be recalculated every time it can be
|
100
|
+
# specified as a callable object (it should implement a +call+ method
|
101
|
+
# e.g. +Proc+ object):
|
102
|
+
#
|
103
|
+
# field :date_of_birth, :date, default: -> { Date.today }
|
104
|
+
#
|
105
|
+
# For every field Dynamoid creates several methods:
|
106
|
+
#
|
107
|
+
# * getter
|
108
|
+
# * setter
|
109
|
+
# * predicate +<name>?+ to check whether a value set
|
110
|
+
# * +<name>_before_type_cast?+ to get an original field value before it was type casted
|
111
|
+
#
|
112
|
+
# It works in the following way:
|
113
|
+
#
|
114
|
+
# class User
|
115
|
+
# include Dynamoid::Document
|
116
|
+
#
|
117
|
+
# field :age, :integer
|
118
|
+
# end
|
119
|
+
#
|
120
|
+
# user = User.new
|
121
|
+
# user.age # => nil
|
122
|
+
# user.age? # => false
|
123
|
+
#
|
124
|
+
# user.age = 20
|
125
|
+
# user.age? # => true
|
126
|
+
#
|
127
|
+
# user.age = '21'
|
128
|
+
# user.age # => 21 - integer
|
129
|
+
# user.age_before_type_cast # => '21' - string
|
130
|
+
#
|
131
|
+
# There is also an option +alias+ which allows to use another name for a
|
132
|
+
# field:
|
133
|
+
#
|
134
|
+
# class User
|
135
|
+
# include Dynamoid::Document
|
136
|
+
#
|
137
|
+
# field :firstName, :string, alias: :first_name
|
138
|
+
# end
|
139
|
+
#
|
140
|
+
# user = User.new(firstName: 'Michael')
|
141
|
+
# user.firstName # Michael
|
142
|
+
# user.first_name # Michael
|
143
|
+
#
|
144
|
+
# @param name [Symbol] name of the field
|
145
|
+
# @param type [Symbol] type of the field (optional)
|
146
|
+
# @param options [Hash] any additional options for the field type (optional)
|
45
147
|
#
|
46
148
|
# @since 0.2.0
|
47
149
|
def field(name, type = :string, options = {})
|
48
|
-
named = name.to_s
|
49
150
|
if type == :float
|
50
151
|
Dynamoid.logger.warn("Field type :float, which you declared for '#{name}', is deprecated in favor of :number.")
|
51
152
|
type = :number
|
52
153
|
end
|
53
|
-
self.attributes = attributes.merge(name => { type: type }.merge(options))
|
54
|
-
|
55
|
-
define_attribute_method(name) # Dirty API
|
56
154
|
|
57
|
-
|
58
|
-
define_method(named) { read_attribute(named) }
|
59
|
-
define_method("#{named}?") do
|
60
|
-
value = read_attribute(named)
|
61
|
-
case value
|
62
|
-
when true then true
|
63
|
-
when false, nil then false
|
64
|
-
else
|
65
|
-
!value.nil?
|
66
|
-
end
|
67
|
-
end
|
68
|
-
define_method("#{named}=") { |value| write_attribute(named, value) }
|
69
|
-
define_method("#{named}_before_type_cast") { read_attribute_before_type_cast(named) }
|
70
|
-
end
|
155
|
+
Dynamoid::Fields::Declare.new(self, name, type, options).call
|
71
156
|
end
|
72
157
|
|
158
|
+
# Declare a table range key.
|
159
|
+
#
|
160
|
+
# class User
|
161
|
+
# include Dynamoid::Document
|
162
|
+
#
|
163
|
+
# range :last_name
|
164
|
+
# end
|
165
|
+
#
|
166
|
+
# By default a range key is a string. In order to use any other type it
|
167
|
+
# should be specified as a second argument:
|
168
|
+
#
|
169
|
+
# range :age, :integer
|
170
|
+
#
|
171
|
+
# Type options can be specified as well:
|
172
|
+
#
|
173
|
+
# range :date_of_birth, :date, store_as_string: true
|
174
|
+
#
|
175
|
+
# @param name [Symbol] a range key attribute name
|
176
|
+
# @param type [Symbol] a range key type (optional)
|
177
|
+
# @param options [Symbol] type options (optional)
|
73
178
|
def range(name, type = :string, options = {})
|
74
179
|
field(name, type, options)
|
75
180
|
self.range_key = name
|
76
181
|
end
|
77
182
|
|
78
|
-
|
183
|
+
# Set table level properties.
|
184
|
+
#
|
185
|
+
# There are some sensible defaults:
|
186
|
+
#
|
187
|
+
# * table name is based on a model class e.g. +users+ for +User+ class
|
188
|
+
# * hash key name - +id+ by default
|
189
|
+
# * hash key type - +string+ by default
|
190
|
+
# * generating timestamp fields +created_at+ and +updated_at+
|
191
|
+
# * billing mode and read/write capacity units
|
192
|
+
#
|
193
|
+
# The +table+ method can be used to override the defaults:
|
194
|
+
#
|
195
|
+
# class User
|
196
|
+
# include Dynamoid::Document
|
197
|
+
#
|
198
|
+
# table name: :customers, key: :uuid
|
199
|
+
# end
|
200
|
+
#
|
201
|
+
# The hash key field is declared by default and a type is a string. If
|
202
|
+
# another type is needed the field should be declared explicitly:
|
203
|
+
#
|
204
|
+
# class User
|
205
|
+
# include Dynamoid::Document
|
206
|
+
#
|
207
|
+
# field :id, :integer
|
208
|
+
# end
|
209
|
+
#
|
210
|
+
# @param options [Hash] options to override default table settings
|
211
|
+
# @option options [Symbol] :name name of a table
|
212
|
+
# @option options [Symbol] :key name of a hash key attribute
|
213
|
+
# @option options [Symbol] :inheritance_field name of an attribute used for STI
|
214
|
+
# @option options [Symbol] :capacity_mode table billing mode - either +provisioned+ or +on_demand+
|
215
|
+
# @option options [Integer] :write_capacity table write capacity units
|
216
|
+
# @option options [Integer] :read_capacity table read capacity units
|
217
|
+
# @option options [true|false] :timestamps whether generate +created_at+ and +updated_at+ fields or not
|
218
|
+
# @option options [Hash] :expires set up a table TTL and should have following structure +{ field: <attriubute name>, after: <seconds> }+
|
219
|
+
#
|
220
|
+
# @since 0.4.0
|
221
|
+
def table(options)
|
79
222
|
# a default 'id' column is created when Dynamoid::Document is included
|
80
223
|
unless attributes.key? hash_key
|
81
224
|
remove_field :id
|
82
225
|
field(hash_key)
|
83
226
|
end
|
227
|
+
|
228
|
+
if options[:timestamps] && !Dynamoid::Config.timestamps
|
229
|
+
# Timestamp fields weren't declared in `included` hook because they
|
230
|
+
# are disabled globaly
|
231
|
+
field :created_at, :datetime
|
232
|
+
field :updated_at, :datetime
|
233
|
+
elsif options[:timestamps] == false && Dynamoid::Config.timestamps
|
234
|
+
# Timestamp fields were declared in `included` hook but they are
|
235
|
+
# disabled for a table
|
236
|
+
remove_field :created_at
|
237
|
+
remove_field :updated_at
|
238
|
+
end
|
84
239
|
end
|
85
240
|
|
241
|
+
# Remove a field declaration
|
242
|
+
#
|
243
|
+
# Removes a field from the list of fields and removes all te generated
|
244
|
+
# for a field methods.
|
245
|
+
#
|
246
|
+
# @param field [Symbol] a field name
|
86
247
|
def remove_field(field)
|
87
248
|
field = field.to_sym
|
88
249
|
attributes.delete(field) || raise('No such field')
|
@@ -99,8 +260,12 @@ module Dynamoid #:nodoc:
|
|
99
260
|
end
|
100
261
|
end
|
101
262
|
|
102
|
-
private
|
263
|
+
# @private
|
264
|
+
def timestamps_enabled?
|
265
|
+
options[:timestamps] || (options[:timestamps].nil? && Dynamoid::Config.timestamps)
|
266
|
+
end
|
103
267
|
|
268
|
+
# @private
|
104
269
|
def generated_methods
|
105
270
|
@generated_methods ||= begin
|
106
271
|
Module.new.tap do |mod|
|
@@ -114,15 +279,25 @@ module Dynamoid #:nodoc:
|
|
114
279
|
attr_accessor :attributes
|
115
280
|
alias raw_attributes attributes
|
116
281
|
|
117
|
-
# Write an attribute on the object.
|
282
|
+
# Write an attribute on the object.
|
283
|
+
#
|
284
|
+
# user.age = 20
|
285
|
+
# user.write_attribute(:age, 21)
|
286
|
+
# user.age # => 21
|
118
287
|
#
|
119
|
-
#
|
120
|
-
#
|
288
|
+
# Also marks the previous value as dirty.
|
289
|
+
#
|
290
|
+
# @param name [Symbol] the name of the field
|
291
|
+
# @param value [Object] the value to assign to that field
|
121
292
|
#
|
122
293
|
# @since 0.2.0
|
123
294
|
def write_attribute(name, value)
|
124
295
|
name = name.to_sym
|
125
296
|
|
297
|
+
unless attribute_is_present_on_model?(name)
|
298
|
+
raise Dynamoid::Errors::UnknownAttribute.new("Attribute #{name} is not part of the model")
|
299
|
+
end
|
300
|
+
|
126
301
|
if association = @associations[name]
|
127
302
|
association.reset
|
128
303
|
end
|
@@ -138,22 +313,40 @@ module Dynamoid #:nodoc:
|
|
138
313
|
|
139
314
|
# Read an attribute from an object.
|
140
315
|
#
|
141
|
-
#
|
316
|
+
# user.age = 20
|
317
|
+
# user.read_attribute(:age) # => 20
|
142
318
|
#
|
319
|
+
# @param name [Symbol] the name of the field
|
320
|
+
# @return attribute value
|
143
321
|
# @since 0.2.0
|
144
322
|
def read_attribute(name)
|
145
323
|
attributes[name.to_sym]
|
146
324
|
end
|
147
325
|
alias [] read_attribute
|
148
326
|
|
149
|
-
#
|
327
|
+
# Return attributes values before type casting.
|
328
|
+
#
|
329
|
+
# user = User.new
|
330
|
+
# user.age = '21'
|
331
|
+
# user.age # => 21
|
332
|
+
#
|
333
|
+
# user.attributes_before_type_cast # => { age: '21' }
|
334
|
+
#
|
335
|
+
# @return [Hash] original attribute values
|
150
336
|
def attributes_before_type_cast
|
151
337
|
@attributes_before_type_cast
|
152
338
|
end
|
153
339
|
|
154
|
-
#
|
340
|
+
# Return the value of the attribute identified by name before type casting.
|
341
|
+
#
|
342
|
+
# user = User.new
|
343
|
+
# user.age = '21'
|
344
|
+
# user.age # => 21
|
155
345
|
#
|
156
|
-
#
|
346
|
+
# user.read_attribute_before_type_cast(:age) # => '21'
|
347
|
+
#
|
348
|
+
# @param name [Symbol] attribute name
|
349
|
+
# @return original attribute value
|
157
350
|
def read_attribute_before_type_cast(name)
|
158
351
|
return nil unless name.respond_to?(:to_sym)
|
159
352
|
|
@@ -166,18 +359,32 @@ module Dynamoid #:nodoc:
|
|
166
359
|
#
|
167
360
|
# @since 0.2.0
|
168
361
|
def set_created_at
|
169
|
-
self.created_at ||= DateTime.now.in_time_zone(Time.zone) if
|
362
|
+
self.created_at ||= DateTime.now.in_time_zone(Time.zone) if self.class.timestamps_enabled?
|
170
363
|
end
|
171
364
|
|
172
365
|
# Automatically called during the save callback to set the updated_at time.
|
173
366
|
#
|
174
367
|
# @since 0.2.0
|
175
368
|
def set_updated_at
|
176
|
-
|
369
|
+
# @_touch_record=false means explicit disabling
|
370
|
+
if self.class.timestamps_enabled? && !updated_at_changed? && @_touch_record != false
|
177
371
|
self.updated_at = DateTime.now.in_time_zone(Time.zone)
|
178
372
|
end
|
179
373
|
end
|
180
374
|
|
375
|
+
def set_expires_field
|
376
|
+
options = self.class.options[:expires]
|
377
|
+
|
378
|
+
if options.present?
|
379
|
+
name = options[:field]
|
380
|
+
seconds = options[:after]
|
381
|
+
|
382
|
+
if self[name].blank?
|
383
|
+
send("#{name}=", Time.now.to_i + seconds)
|
384
|
+
end
|
385
|
+
end
|
386
|
+
end
|
387
|
+
|
181
388
|
def set_inheritance_field
|
182
389
|
# actually it does only following logic:
|
183
390
|
# self.type ||= self.class.name if self.class.attributes[:type]
|
@@ -187,5 +394,10 @@ module Dynamoid #:nodoc:
|
|
187
394
|
send("#{type}=", self.class.name)
|
188
395
|
end
|
189
396
|
end
|
397
|
+
|
398
|
+
def attribute_is_present_on_model?(attribute_name)
|
399
|
+
setter = "#{attribute_name}=".to_sym
|
400
|
+
respond_to?(setter)
|
401
|
+
end
|
190
402
|
end
|
191
403
|
end
|