omg-activemodel 8.0.0.alpha1
Sign up to get free protection for your applications and to get access to all the features.
- checksums.yaml +7 -0
- data/CHANGELOG.md +67 -0
- data/MIT-LICENSE +21 -0
- data/README.rdoc +266 -0
- data/lib/active_model/access.rb +16 -0
- data/lib/active_model/api.rb +99 -0
- data/lib/active_model/attribute/user_provided_default.rb +55 -0
- data/lib/active_model/attribute.rb +277 -0
- data/lib/active_model/attribute_assignment.rb +78 -0
- data/lib/active_model/attribute_methods.rb +592 -0
- data/lib/active_model/attribute_mutation_tracker.rb +189 -0
- data/lib/active_model/attribute_registration.rb +117 -0
- data/lib/active_model/attribute_set/builder.rb +182 -0
- data/lib/active_model/attribute_set/yaml_encoder.rb +40 -0
- data/lib/active_model/attribute_set.rb +118 -0
- data/lib/active_model/attributes.rb +165 -0
- data/lib/active_model/callbacks.rb +155 -0
- data/lib/active_model/conversion.rb +121 -0
- data/lib/active_model/deprecator.rb +7 -0
- data/lib/active_model/dirty.rb +416 -0
- data/lib/active_model/error.rb +208 -0
- data/lib/active_model/errors.rb +547 -0
- data/lib/active_model/forbidden_attributes_protection.rb +33 -0
- data/lib/active_model/gem_version.rb +17 -0
- data/lib/active_model/lint.rb +118 -0
- data/lib/active_model/locale/en.yml +38 -0
- data/lib/active_model/model.rb +78 -0
- data/lib/active_model/naming.rb +359 -0
- data/lib/active_model/nested_error.rb +22 -0
- data/lib/active_model/railtie.rb +24 -0
- data/lib/active_model/secure_password.rb +231 -0
- data/lib/active_model/serialization.rb +198 -0
- data/lib/active_model/serializers/json.rb +154 -0
- data/lib/active_model/translation.rb +78 -0
- data/lib/active_model/type/big_integer.rb +36 -0
- data/lib/active_model/type/binary.rb +62 -0
- data/lib/active_model/type/boolean.rb +48 -0
- data/lib/active_model/type/date.rb +78 -0
- data/lib/active_model/type/date_time.rb +88 -0
- data/lib/active_model/type/decimal.rb +107 -0
- data/lib/active_model/type/float.rb +64 -0
- data/lib/active_model/type/helpers/accepts_multiparameter_time.rb +53 -0
- data/lib/active_model/type/helpers/mutable.rb +24 -0
- data/lib/active_model/type/helpers/numeric.rb +61 -0
- data/lib/active_model/type/helpers/time_value.rb +127 -0
- data/lib/active_model/type/helpers/timezone.rb +23 -0
- data/lib/active_model/type/helpers.rb +7 -0
- data/lib/active_model/type/immutable_string.rb +71 -0
- data/lib/active_model/type/integer.rb +113 -0
- data/lib/active_model/type/registry.rb +37 -0
- data/lib/active_model/type/serialize_cast_value.rb +47 -0
- data/lib/active_model/type/string.rb +43 -0
- data/lib/active_model/type/time.rb +87 -0
- data/lib/active_model/type/value.rb +157 -0
- data/lib/active_model/type.rb +55 -0
- data/lib/active_model/validations/absence.rb +33 -0
- data/lib/active_model/validations/acceptance.rb +113 -0
- data/lib/active_model/validations/callbacks.rb +119 -0
- data/lib/active_model/validations/clusivity.rb +54 -0
- data/lib/active_model/validations/comparability.rb +18 -0
- data/lib/active_model/validations/comparison.rb +90 -0
- data/lib/active_model/validations/confirmation.rb +80 -0
- data/lib/active_model/validations/exclusion.rb +49 -0
- data/lib/active_model/validations/format.rb +112 -0
- data/lib/active_model/validations/helper_methods.rb +15 -0
- data/lib/active_model/validations/inclusion.rb +47 -0
- data/lib/active_model/validations/length.rb +130 -0
- data/lib/active_model/validations/numericality.rb +222 -0
- data/lib/active_model/validations/presence.rb +39 -0
- data/lib/active_model/validations/resolve_value.rb +26 -0
- data/lib/active_model/validations/validates.rb +175 -0
- data/lib/active_model/validations/with.rb +154 -0
- data/lib/active_model/validations.rb +489 -0
- data/lib/active_model/validator.rb +190 -0
- data/lib/active_model/version.rb +10 -0
- data/lib/active_model.rb +84 -0
- metadata +139 -0
checksums.yaml
ADDED
@@ -0,0 +1,7 @@
|
|
1
|
+
---
|
2
|
+
SHA256:
|
3
|
+
metadata.gz: b9199b4832bbe8dfdddaef904f1830bc5c43aee700c358b48b7ecc205f27d988
|
4
|
+
data.tar.gz: 14f2bbdd71b68ccbaa9613688c58197de86cd6e281260a3f062171ee46084af0
|
5
|
+
SHA512:
|
6
|
+
metadata.gz: 4aaa1d510d524f7133108ec430fdad64f50ee8cb05e113a260c2ff203ed477ea574c7523c6c31cb1955fd4ed704b8e7b3df42c02ca7a918ab9694288f5c8ae12
|
7
|
+
data.tar.gz: 6dba06dff924fcf9bbccf45bf0eb3efd968daee301d619c0ad509cbfc6d3b602180735823d0cc731451c80451f792effd4140cb12a95ef6ae2b4e862a58365c8
|
data/CHANGELOG.md
ADDED
@@ -0,0 +1,67 @@
|
|
1
|
+
* Add a default token generator for password reset tokens when using `has_secure_password`.
|
2
|
+
|
3
|
+
```ruby
|
4
|
+
class User < ApplicationRecord
|
5
|
+
has_secure_password
|
6
|
+
end
|
7
|
+
|
8
|
+
user = User.create!(name: "david", password: "123", password_confirmation: "123")
|
9
|
+
token = user.password_reset_token
|
10
|
+
User.find_by_password_reset_token(token) # returns user
|
11
|
+
|
12
|
+
# 16 minutes later...
|
13
|
+
User.find_by_password_reset_token(token) # returns nil
|
14
|
+
|
15
|
+
# raises ActiveSupport::MessageVerifier::InvalidSignature since the token is expired
|
16
|
+
User.find_by_password_reset_token!(token)
|
17
|
+
```
|
18
|
+
|
19
|
+
*DHH*
|
20
|
+
|
21
|
+
* Add a load hook `active_model_translation` for `ActiveModel::Translation`.
|
22
|
+
|
23
|
+
*Shouichi Kamiya*
|
24
|
+
|
25
|
+
* Add `raise_on_missing_translations` option to `ActiveModel::Translation`.
|
26
|
+
When the option is set, `human_attribute_name` raises an error if a translation of the given attribute is missing.
|
27
|
+
|
28
|
+
```ruby
|
29
|
+
# ActiveModel::Translation.raise_on_missing_translations = false
|
30
|
+
Post.human_attribute_name("title")
|
31
|
+
=> "Title"
|
32
|
+
|
33
|
+
# ActiveModel::Translation.raise_on_missing_translations = true
|
34
|
+
Post.human_attribute_name("title")
|
35
|
+
=> Translation missing. Options considered were: (I18n::MissingTranslationData)
|
36
|
+
- en.activerecord.attributes.post.title
|
37
|
+
- en.attributes.title
|
38
|
+
|
39
|
+
raise exception.respond_to?(:to_exception) ? exception.to_exception : exception
|
40
|
+
^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
|
41
|
+
```
|
42
|
+
|
43
|
+
*Shouichi Kamiya*
|
44
|
+
|
45
|
+
* Introduce `ActiveModel::AttributeAssignment#attribute_writer_missing`
|
46
|
+
|
47
|
+
Provide instances with an opportunity to gracefully handle assigning to an
|
48
|
+
unknown attribute:
|
49
|
+
|
50
|
+
```ruby
|
51
|
+
class Rectangle
|
52
|
+
include ActiveModel::AttributeAssignment
|
53
|
+
|
54
|
+
attr_accessor :length, :width
|
55
|
+
|
56
|
+
def attribute_writer_missing(name, value)
|
57
|
+
Rails.logger.warn "Tried to assign to unknown attribute #{name}"
|
58
|
+
end
|
59
|
+
end
|
60
|
+
|
61
|
+
rectangle = Rectangle.new
|
62
|
+
rectangle.assign_attributes(height: 10) # => Logs "Tried to assign to unknown attribute 'height'"
|
63
|
+
```
|
64
|
+
|
65
|
+
*Sean Doyle*
|
66
|
+
|
67
|
+
Please check [7-2-stable](https://github.com/rails/rails/blob/7-2-stable/activemodel/CHANGELOG.md) for previous changes.
|
data/MIT-LICENSE
ADDED
@@ -0,0 +1,21 @@
|
|
1
|
+
Copyright (c) David Heinemeier Hansson
|
2
|
+
|
3
|
+
Permission is hereby granted, free of charge, to any person obtaining
|
4
|
+
a copy of this software and associated documentation files (the
|
5
|
+
"Software"), to deal in the Software without restriction, including
|
6
|
+
without limitation the rights to use, copy, modify, merge, publish,
|
7
|
+
distribute, sublicense, and/or sell copies of the Software, and to
|
8
|
+
permit persons to whom the Software is furnished to do so, subject to
|
9
|
+
the following conditions:
|
10
|
+
|
11
|
+
The above copyright notice and this permission notice shall be
|
12
|
+
included in all copies or substantial portions of the Software.
|
13
|
+
|
14
|
+
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
15
|
+
EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
16
|
+
MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
|
17
|
+
NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
|
18
|
+
LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
|
19
|
+
OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
|
20
|
+
WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
21
|
+
|
data/README.rdoc
ADDED
@@ -0,0 +1,266 @@
|
|
1
|
+
= Active Model -- model interfaces for \Rails
|
2
|
+
|
3
|
+
Active Model provides a known set of interfaces for usage in model classes.
|
4
|
+
They allow for Action Pack helpers to interact with non-Active Record models,
|
5
|
+
for example. Active Model also helps with building custom ORMs for use outside of
|
6
|
+
the \Rails framework.
|
7
|
+
|
8
|
+
You can read more about Active Model in the {Active Model Basics}[https://guides.rubyonrails.org/active_model_basics.html] guide.
|
9
|
+
|
10
|
+
Prior to \Rails 3.0, if a plugin or gem developer wanted to have an object
|
11
|
+
interact with Action Pack helpers, it was required to either copy chunks of
|
12
|
+
code from \Rails, or monkey patch entire helpers to make them handle objects
|
13
|
+
that did not exactly conform to the Active Record interface. This would result
|
14
|
+
in code duplication and fragile applications that broke on upgrades. Active
|
15
|
+
Model solves this by defining an explicit API. You can read more about the
|
16
|
+
API in +ActiveModel::Lint::Tests+.
|
17
|
+
|
18
|
+
Active Model provides a default module that implements the basic API required
|
19
|
+
to integrate with Action Pack out of the box: ActiveModel::API.
|
20
|
+
|
21
|
+
class Person
|
22
|
+
include ActiveModel::API
|
23
|
+
|
24
|
+
attr_accessor :name, :age
|
25
|
+
validates_presence_of :name
|
26
|
+
end
|
27
|
+
|
28
|
+
person = Person.new(name: 'bob', age: '18')
|
29
|
+
person.name # => 'bob'
|
30
|
+
person.age # => '18'
|
31
|
+
person.valid? # => true
|
32
|
+
|
33
|
+
It includes model name introspections, conversions, translations and
|
34
|
+
validations, resulting in a class suitable to be used with Action Pack.
|
35
|
+
See ActiveModel::API for more examples.
|
36
|
+
|
37
|
+
Active Model also provides the following functionality to have ORM-like
|
38
|
+
behavior out of the box:
|
39
|
+
|
40
|
+
* Add attribute magic to objects
|
41
|
+
|
42
|
+
class Person
|
43
|
+
include ActiveModel::AttributeMethods
|
44
|
+
|
45
|
+
attribute_method_prefix 'clear_'
|
46
|
+
define_attribute_methods :name, :age
|
47
|
+
|
48
|
+
attr_accessor :name, :age
|
49
|
+
|
50
|
+
def clear_attribute(attr)
|
51
|
+
send("#{attr}=", nil)
|
52
|
+
end
|
53
|
+
end
|
54
|
+
|
55
|
+
person = Person.new
|
56
|
+
person.clear_name
|
57
|
+
person.clear_age
|
58
|
+
|
59
|
+
{Learn more}[https://api.rubyonrails.org/classes/ActiveModel/AttributeMethods.html]
|
60
|
+
|
61
|
+
* Callbacks for certain operations
|
62
|
+
|
63
|
+
class Person
|
64
|
+
extend ActiveModel::Callbacks
|
65
|
+
define_model_callbacks :create
|
66
|
+
|
67
|
+
def create
|
68
|
+
run_callbacks :create do
|
69
|
+
# Your create action methods here
|
70
|
+
end
|
71
|
+
end
|
72
|
+
end
|
73
|
+
|
74
|
+
This generates +before_create+, +around_create+ and +after_create+
|
75
|
+
class methods that wrap your create method.
|
76
|
+
|
77
|
+
{Learn more}[https://api.rubyonrails.org/classes/ActiveModel/Callbacks.html]
|
78
|
+
|
79
|
+
* Tracking value changes
|
80
|
+
|
81
|
+
class Person
|
82
|
+
include ActiveModel::Dirty
|
83
|
+
|
84
|
+
define_attribute_methods :name
|
85
|
+
|
86
|
+
def name
|
87
|
+
@name
|
88
|
+
end
|
89
|
+
|
90
|
+
def name=(val)
|
91
|
+
name_will_change! unless val == @name
|
92
|
+
@name = val
|
93
|
+
end
|
94
|
+
|
95
|
+
def save
|
96
|
+
# do persistence work
|
97
|
+
changes_applied
|
98
|
+
end
|
99
|
+
end
|
100
|
+
|
101
|
+
person = Person.new
|
102
|
+
person.name # => nil
|
103
|
+
person.changed? # => false
|
104
|
+
person.name = 'bob'
|
105
|
+
person.changed? # => true
|
106
|
+
person.changed # => ['name']
|
107
|
+
person.changes # => { 'name' => [nil, 'bob'] }
|
108
|
+
person.save
|
109
|
+
person.name = 'robert'
|
110
|
+
person.save
|
111
|
+
person.previous_changes # => {'name' => ['bob, 'robert']}
|
112
|
+
|
113
|
+
{Learn more}[https://api.rubyonrails.org/classes/ActiveModel/Dirty.html]
|
114
|
+
|
115
|
+
* Adding +errors+ interface to objects
|
116
|
+
|
117
|
+
Exposing error messages allows objects to interact with Action Pack
|
118
|
+
helpers seamlessly.
|
119
|
+
|
120
|
+
class Person
|
121
|
+
|
122
|
+
def initialize
|
123
|
+
@errors = ActiveModel::Errors.new(self)
|
124
|
+
end
|
125
|
+
|
126
|
+
attr_accessor :name
|
127
|
+
attr_reader :errors
|
128
|
+
|
129
|
+
def validate!
|
130
|
+
errors.add(:name, "cannot be nil") if name.nil?
|
131
|
+
end
|
132
|
+
|
133
|
+
def self.human_attribute_name(attr, options = {})
|
134
|
+
"Name"
|
135
|
+
end
|
136
|
+
end
|
137
|
+
|
138
|
+
person = Person.new
|
139
|
+
person.name = nil
|
140
|
+
person.validate!
|
141
|
+
person.errors.full_messages
|
142
|
+
# => ["Name cannot be nil"]
|
143
|
+
|
144
|
+
{Learn more}[https://api.rubyonrails.org/classes/ActiveModel/Errors.html]
|
145
|
+
|
146
|
+
* Model name introspection
|
147
|
+
|
148
|
+
class NamedPerson
|
149
|
+
extend ActiveModel::Naming
|
150
|
+
end
|
151
|
+
|
152
|
+
NamedPerson.model_name.name # => "NamedPerson"
|
153
|
+
NamedPerson.model_name.human # => "Named person"
|
154
|
+
|
155
|
+
{Learn more}[https://api.rubyonrails.org/classes/ActiveModel/Naming.html]
|
156
|
+
|
157
|
+
* Making objects serializable
|
158
|
+
|
159
|
+
ActiveModel::Serialization provides a standard interface for your object
|
160
|
+
to provide +to_json+ serialization.
|
161
|
+
|
162
|
+
class SerialPerson
|
163
|
+
include ActiveModel::Serialization
|
164
|
+
|
165
|
+
attr_accessor :name
|
166
|
+
|
167
|
+
def attributes
|
168
|
+
{'name' => name}
|
169
|
+
end
|
170
|
+
end
|
171
|
+
|
172
|
+
s = SerialPerson.new
|
173
|
+
s.serializable_hash # => {"name"=>nil}
|
174
|
+
|
175
|
+
class SerialPerson
|
176
|
+
include ActiveModel::Serializers::JSON
|
177
|
+
end
|
178
|
+
|
179
|
+
s = SerialPerson.new
|
180
|
+
s.to_json # => "{\"name\":null}"
|
181
|
+
|
182
|
+
{Learn more}[https://api.rubyonrails.org/classes/ActiveModel/Serialization.html]
|
183
|
+
|
184
|
+
* Internationalization (i18n) support
|
185
|
+
|
186
|
+
class Person
|
187
|
+
extend ActiveModel::Translation
|
188
|
+
end
|
189
|
+
|
190
|
+
Person.human_attribute_name('my_attribute')
|
191
|
+
# => "My attribute"
|
192
|
+
|
193
|
+
{Learn more}[https://api.rubyonrails.org/classes/ActiveModel/Translation.html]
|
194
|
+
|
195
|
+
* Validation support
|
196
|
+
|
197
|
+
class Person
|
198
|
+
include ActiveModel::Validations
|
199
|
+
|
200
|
+
attr_accessor :first_name, :last_name
|
201
|
+
|
202
|
+
validates_each :first_name, :last_name do |record, attr, value|
|
203
|
+
record.errors.add attr, "starts with z." if value.start_with?("z")
|
204
|
+
end
|
205
|
+
end
|
206
|
+
|
207
|
+
person = Person.new
|
208
|
+
person.first_name = 'zoolander'
|
209
|
+
person.valid? # => false
|
210
|
+
|
211
|
+
{Learn more}[https://api.rubyonrails.org/classes/ActiveModel/Validations.html]
|
212
|
+
|
213
|
+
* Custom validators
|
214
|
+
|
215
|
+
class HasNameValidator < ActiveModel::Validator
|
216
|
+
def validate(record)
|
217
|
+
record.errors.add(:name, "must exist") if record.name.blank?
|
218
|
+
end
|
219
|
+
end
|
220
|
+
|
221
|
+
class ValidatorPerson
|
222
|
+
include ActiveModel::Validations
|
223
|
+
validates_with HasNameValidator
|
224
|
+
attr_accessor :name
|
225
|
+
end
|
226
|
+
|
227
|
+
p = ValidatorPerson.new
|
228
|
+
p.valid? # => false
|
229
|
+
p.errors.full_messages # => ["Name must exist"]
|
230
|
+
p.name = "Bob"
|
231
|
+
p.valid? # => true
|
232
|
+
|
233
|
+
{Learn more}[https://api.rubyonrails.org/classes/ActiveModel/Validator.html]
|
234
|
+
|
235
|
+
|
236
|
+
== Download and installation
|
237
|
+
|
238
|
+
The latest version of Active Model can be installed with RubyGems:
|
239
|
+
|
240
|
+
$ gem install activemodel
|
241
|
+
|
242
|
+
Source code can be downloaded as part of the \Rails project on GitHub
|
243
|
+
|
244
|
+
* https://github.com/rails/rails/tree/main/activemodel
|
245
|
+
|
246
|
+
|
247
|
+
== License
|
248
|
+
|
249
|
+
Active Model is released under the MIT license:
|
250
|
+
|
251
|
+
* https://opensource.org/licenses/MIT
|
252
|
+
|
253
|
+
|
254
|
+
== Support
|
255
|
+
|
256
|
+
API documentation is at:
|
257
|
+
|
258
|
+
* https://api.rubyonrails.org
|
259
|
+
|
260
|
+
Bug reports for the Ruby on \Rails project can be filed here:
|
261
|
+
|
262
|
+
* https://github.com/rails/rails/issues
|
263
|
+
|
264
|
+
Feature requests should be discussed on the rails-core mailing list here:
|
265
|
+
|
266
|
+
* https://discuss.rubyonrails.org/c/rubyonrails-core
|
@@ -0,0 +1,16 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
require "active_support/core_ext/enumerable"
|
4
|
+
require "active_support/core_ext/hash/indifferent_access"
|
5
|
+
|
6
|
+
module ActiveModel
|
7
|
+
module Access # :nodoc:
|
8
|
+
def slice(*methods)
|
9
|
+
methods.flatten.index_with { |method| public_send(method) }.with_indifferent_access
|
10
|
+
end
|
11
|
+
|
12
|
+
def values_at(*methods)
|
13
|
+
methods.flatten.map! { |method| public_send(method) }
|
14
|
+
end
|
15
|
+
end
|
16
|
+
end
|
@@ -0,0 +1,99 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
module ActiveModel
|
4
|
+
# = Active \Model \API
|
5
|
+
#
|
6
|
+
# Includes the required interface for an object to interact with
|
7
|
+
# Action Pack and Action View, using different Active \Model modules.
|
8
|
+
# It includes model name introspections, conversions, translations, and
|
9
|
+
# validations. Besides that, it allows you to initialize the object with a
|
10
|
+
# hash of attributes, pretty much like Active Record does.
|
11
|
+
#
|
12
|
+
# A minimal implementation could be:
|
13
|
+
#
|
14
|
+
# class Person
|
15
|
+
# include ActiveModel::API
|
16
|
+
# attr_accessor :name, :age
|
17
|
+
# end
|
18
|
+
#
|
19
|
+
# person = Person.new(name: 'bob', age: '18')
|
20
|
+
# person.name # => "bob"
|
21
|
+
# person.age # => "18"
|
22
|
+
#
|
23
|
+
# Note that, by default, +ActiveModel::API+ implements #persisted?
|
24
|
+
# to return +false+, which is the most common case. You may want to override
|
25
|
+
# it in your class to simulate a different scenario:
|
26
|
+
#
|
27
|
+
# class Person
|
28
|
+
# include ActiveModel::API
|
29
|
+
# attr_accessor :id, :name
|
30
|
+
#
|
31
|
+
# def persisted?
|
32
|
+
# self.id.present?
|
33
|
+
# end
|
34
|
+
# end
|
35
|
+
#
|
36
|
+
# person = Person.new(id: 1, name: 'bob')
|
37
|
+
# person.persisted? # => true
|
38
|
+
#
|
39
|
+
# Also, if for some reason you need to run code on initialize ( ::new ), make
|
40
|
+
# sure you call +super+ if you want the attributes hash initialization to
|
41
|
+
# happen.
|
42
|
+
#
|
43
|
+
# class Person
|
44
|
+
# include ActiveModel::API
|
45
|
+
# attr_accessor :id, :name, :omg
|
46
|
+
#
|
47
|
+
# def initialize(attributes={})
|
48
|
+
# super
|
49
|
+
# @omg ||= true
|
50
|
+
# end
|
51
|
+
# end
|
52
|
+
#
|
53
|
+
# person = Person.new(id: 1, name: 'bob')
|
54
|
+
# person.omg # => true
|
55
|
+
#
|
56
|
+
# For more detailed information on other functionalities available, please
|
57
|
+
# refer to the specific modules included in +ActiveModel::API+
|
58
|
+
# (see below).
|
59
|
+
module API
|
60
|
+
extend ActiveSupport::Concern
|
61
|
+
include ActiveModel::AttributeAssignment
|
62
|
+
include ActiveModel::Validations
|
63
|
+
include ActiveModel::Conversion
|
64
|
+
|
65
|
+
included do
|
66
|
+
extend ActiveModel::Naming
|
67
|
+
extend ActiveModel::Translation
|
68
|
+
end
|
69
|
+
|
70
|
+
# Initializes a new model with the given +params+.
|
71
|
+
#
|
72
|
+
# class Person
|
73
|
+
# include ActiveModel::API
|
74
|
+
# attr_accessor :name, :age
|
75
|
+
# end
|
76
|
+
#
|
77
|
+
# person = Person.new(name: 'bob', age: '18')
|
78
|
+
# person.name # => "bob"
|
79
|
+
# person.age # => "18"
|
80
|
+
def initialize(attributes = {})
|
81
|
+
assign_attributes(attributes) if attributes
|
82
|
+
|
83
|
+
super()
|
84
|
+
end
|
85
|
+
|
86
|
+
# Indicates if the model is persisted. Default is +false+.
|
87
|
+
#
|
88
|
+
# class Person
|
89
|
+
# include ActiveModel::API
|
90
|
+
# attr_accessor :id, :name
|
91
|
+
# end
|
92
|
+
#
|
93
|
+
# person = Person.new(id: 1, name: 'bob')
|
94
|
+
# person.persisted? # => false
|
95
|
+
def persisted?
|
96
|
+
false
|
97
|
+
end
|
98
|
+
end
|
99
|
+
end
|
@@ -0,0 +1,55 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
require "active_model/attribute"
|
4
|
+
|
5
|
+
module ActiveModel
|
6
|
+
class Attribute # :nodoc:
|
7
|
+
def with_user_default(value)
|
8
|
+
UserProvidedDefault.new(name, value, type, self.is_a?(FromDatabase) ? self : original_attribute)
|
9
|
+
end
|
10
|
+
|
11
|
+
class UserProvidedDefault < FromUser # :nodoc:
|
12
|
+
def initialize(name, value, type, database_default)
|
13
|
+
@user_provided_value = value
|
14
|
+
super(name, value, type, database_default)
|
15
|
+
end
|
16
|
+
|
17
|
+
def value_before_type_cast
|
18
|
+
if user_provided_value.is_a?(Proc)
|
19
|
+
@memoized_value_before_type_cast ||= user_provided_value.call
|
20
|
+
else
|
21
|
+
@user_provided_value
|
22
|
+
end
|
23
|
+
end
|
24
|
+
|
25
|
+
def with_type(type)
|
26
|
+
self.class.new(name, user_provided_value, type, original_attribute)
|
27
|
+
end
|
28
|
+
|
29
|
+
def marshal_dump
|
30
|
+
result = [
|
31
|
+
name,
|
32
|
+
value_before_type_cast,
|
33
|
+
type,
|
34
|
+
original_attribute,
|
35
|
+
]
|
36
|
+
result << value if defined?(@value)
|
37
|
+
result
|
38
|
+
end
|
39
|
+
|
40
|
+
def marshal_load(values)
|
41
|
+
name, user_provided_value, type, original_attribute, value = values
|
42
|
+
@name = name
|
43
|
+
@user_provided_value = user_provided_value
|
44
|
+
@type = type
|
45
|
+
@original_attribute = original_attribute
|
46
|
+
if values.length == 5
|
47
|
+
@value = value
|
48
|
+
end
|
49
|
+
end
|
50
|
+
|
51
|
+
private
|
52
|
+
attr_reader :user_provided_value
|
53
|
+
end
|
54
|
+
end
|
55
|
+
end
|