store_attribute 1.0.2 → 1.1.1
Sign up to get free protection for your applications and to get access to all the features.
- checksums.yaml +4 -4
- data/CHANGELOG.md +24 -0
- data/LICENSE.txt +1 -1
- data/README.md +19 -6
- data/lib/store_attribute/active_record/store.rb +22 -6
- data/lib/store_attribute/active_record/type/typed_store.rb +13 -1
- data/lib/store_attribute/version.rb +1 -1
- data/lib/store_attribute.rb +3 -0
- metadata +3 -3
checksums.yaml
CHANGED
@@ -1,7 +1,7 @@
|
|
1
1
|
---
|
2
2
|
SHA256:
|
3
|
-
metadata.gz:
|
4
|
-
data.tar.gz:
|
3
|
+
metadata.gz: 6078ba7214bee86726edd35c8cfbac054907c2aada0e5644d63b906b1884cb37
|
4
|
+
data.tar.gz: b3ac4c23dfdc1f8412fccf67fe8c7e9a441943ae7d3adba95eb735e7cd5e0302
|
5
5
|
SHA512:
|
6
|
-
metadata.gz:
|
7
|
-
data.tar.gz:
|
6
|
+
metadata.gz: 2c8b70decca97c29ec6b112265d318f7fcb598a87dccbcc6531fc7f9e1de8701830f7d963f6890cfb453704ad19b14e2b6550d9d513c33fbfadf85952a9491a4
|
7
|
+
data.tar.gz: 41a9e77501f4c9e4df33a06fa343c0377e8cb982b0dfda250003345e16d4afc635b39b99c9570ded1f7af27a6781b76c8dba013b3586bf699c048260159bf507
|
data/CHANGELOG.md
CHANGED
@@ -2,6 +2,30 @@
|
|
2
2
|
|
3
3
|
## master
|
4
4
|
|
5
|
+
## 1.1.1 (2023-06-27)
|
6
|
+
|
7
|
+
- Lookup store attribute types only after schema load.
|
8
|
+
|
9
|
+
## 1.1.0 (2023-03-08) 🌷
|
10
|
+
|
11
|
+
- Add configuration option to return default values when attribute key is not present in the serialized value ([@markedmondson][], [@palkan][]).
|
12
|
+
|
13
|
+
Add to the class (preferrable `ApplicationRecord` or some other base class):
|
14
|
+
|
15
|
+
```ruby
|
16
|
+
class ApplicationRecord < ActiveRecord::Base
|
17
|
+
self.store_attribute_unset_values_fallback_to_default = true
|
18
|
+
|
19
|
+
store_attribute :extra, :color, :string, default: "grey"
|
20
|
+
end
|
21
|
+
|
22
|
+
user = User.create!(extra: {})
|
23
|
+
# without the fallback
|
24
|
+
user.color #=> nil
|
25
|
+
# with fallback
|
26
|
+
user.color #=> "grey"
|
27
|
+
```
|
28
|
+
|
5
29
|
## 1.0.2 (2022-07-29)
|
6
30
|
|
7
31
|
- Fix possible conflicts with Active Model objects. ([@palkan][])
|
data/LICENSE.txt
CHANGED
data/README.md
CHANGED
@@ -36,7 +36,7 @@ Where:
|
|
36
36
|
- `store_name` The name of the store.
|
37
37
|
- `name` The name of the accessor to the store.
|
38
38
|
- `type` A symbol such as `:string` or `:integer`, or a type object to be used for the accessor.
|
39
|
-
- `options` (optional) A hash of cast type options such as `precision`, `limit`, `scale`, `default`.
|
39
|
+
- `options` (optional) A hash of cast type options such as `precision`, `limit`, `scale`, `default`. Regular `store_accessor` options, such as `prefix`, `suffix` are also supported.
|
40
40
|
|
41
41
|
Type casting occurs every time you write data through accessor or update store itself
|
42
42
|
and when object is loaded from database.
|
@@ -131,17 +131,30 @@ end
|
|
131
131
|
Date.current #=> 2022-03-17
|
132
132
|
|
133
133
|
user = User.new
|
134
|
-
user.name #=> "
|
134
|
+
user.name #=> "Joe"
|
135
135
|
user.expired_at #=> 2022-03-19
|
136
136
|
user.save!
|
137
137
|
|
138
138
|
raw_user = RawUser.find(user.id)
|
139
|
-
|
140
|
-
|
139
|
+
raw_user.name #=> "Joe"
|
140
|
+
raw_user.expired_at #=> 2022-03-19
|
141
141
|
|
142
142
|
another_raw_user = RawUser.create!
|
143
143
|
another_user = User.find(another_raw_user.id)
|
144
144
|
|
145
|
-
|
146
|
-
|
145
|
+
another_user.name #=> nil
|
146
|
+
another_user.expired_at #=> nil
|
147
147
|
```
|
148
|
+
|
149
|
+
It is possible to configure `store_attribute` to return the default value even when the record is persisted and the attribute name is not present. By using the `store_attribute_unset_values_fallback_to_default` class option, default values will be returned for missing keys. For example:
|
150
|
+
|
151
|
+
```ruby
|
152
|
+
class User < ApplicationRecord
|
153
|
+
self.store_attribute_unset_values_fallback_to_default = true
|
154
|
+
end
|
155
|
+
|
156
|
+
user = User.create!(extra: {})
|
157
|
+
user.expired_at #=> 2022-03-19
|
158
|
+
```
|
159
|
+
|
160
|
+
**IMPORTANT:** Due to implementation limitations, it's not recommended to toggle the value of `store_attribute_unset_values_fallback_to_default` in sub-classes. We recommend to set this value in base classes (e.g., `ApplicationRecord`).
|
@@ -10,6 +10,8 @@ module ActiveRecord
|
|
10
10
|
alias_method :_orig_store_without_types, :store
|
11
11
|
alias_method :_orig_store_accessor_without_types, :store_accessor
|
12
12
|
|
13
|
+
attr_writer :store_attribute_unset_values_fallback_to_default
|
14
|
+
|
13
15
|
# Defines store on this model.
|
14
16
|
#
|
15
17
|
# +store_name+ The name of the store.
|
@@ -122,12 +124,18 @@ module ActiveRecord
|
|
122
124
|
_define_predicate_method(name, prefix: prefix, suffix: suffix) if type == :boolean
|
123
125
|
|
124
126
|
_define_store_attribute(store_name) if !_local_typed_stored_attributes? || _local_typed_stored_attributes[store_name].empty?
|
125
|
-
|
127
|
+
_local_typed_stored_attributes[store_name][name] = [type, options]
|
126
128
|
end
|
127
129
|
|
128
|
-
def
|
129
|
-
|
130
|
-
|
130
|
+
def store_attribute_unset_values_fallback_to_default
|
131
|
+
return @store_attribute_unset_values_fallback_to_default if instance_variable_defined?(:@store_attribute_unset_values_fallback_to_default)
|
132
|
+
|
133
|
+
@store_attribute_unset_values_fallback_to_default =
|
134
|
+
if superclass.respond_to?(:store_attribute_unset_values_fallback_to_default)
|
135
|
+
superclass.store_attribute_unset_values_fallback_to_default
|
136
|
+
else
|
137
|
+
false
|
138
|
+
end
|
131
139
|
end
|
132
140
|
|
133
141
|
def _local_typed_stored_attributes?
|
@@ -156,12 +164,17 @@ module ActiveRecord
|
|
156
164
|
|
157
165
|
defaultik = Type::TypedStore::Defaultik.new
|
158
166
|
|
167
|
+
owner = self
|
168
|
+
|
159
169
|
if use_decorator
|
160
170
|
decorate_attribute_type(attr_name, "typed_accessor_for_#{attr_name}") do |subtype|
|
161
171
|
subtypes = _local_typed_stored_attributes[attr_name]
|
162
172
|
type = Type::TypedStore.create_from_type(subtype)
|
173
|
+
type.owner = owner
|
163
174
|
defaultik.type = type
|
164
|
-
subtypes.each
|
175
|
+
subtypes.each do |name, (cast_type, options)|
|
176
|
+
type.add_typed_key(name, cast_type, **options.symbolize_keys)
|
177
|
+
end
|
165
178
|
|
166
179
|
define_default_attribute(attr_name, defaultik.proc, type, from_user: true)
|
167
180
|
|
@@ -173,8 +186,11 @@ module ActiveRecord
|
|
173
186
|
subtype = _lookup_cast_type(attr_name, was_type, {}) if defined?(_lookup_cast_type)
|
174
187
|
|
175
188
|
type = Type::TypedStore.create_from_type(subtype)
|
189
|
+
type.owner = owner
|
176
190
|
defaultik.type = type
|
177
|
-
subtypes.each
|
191
|
+
subtypes.each do |name, (cast_type, options)|
|
192
|
+
type.add_typed_key(name, cast_type, **options.symbolize_keys)
|
193
|
+
end
|
178
194
|
|
179
195
|
type
|
180
196
|
end
|
@@ -25,6 +25,8 @@ module ActiveRecord
|
|
25
25
|
new(basetype)
|
26
26
|
end
|
27
27
|
|
28
|
+
attr_writer :owner
|
29
|
+
|
28
30
|
def initialize(subtype)
|
29
31
|
@accessor_types = {}
|
30
32
|
@defaults = {}
|
@@ -47,6 +49,8 @@ module ActiveRecord
|
|
47
49
|
accessor_types.each do |key, type|
|
48
50
|
if hash.key?(key)
|
49
51
|
hash[key] = type.deserialize(hash[key])
|
52
|
+
elsif fallback_to_default?(key)
|
53
|
+
hash[key] = built_defaults[key]
|
50
54
|
end
|
51
55
|
end
|
52
56
|
hash
|
@@ -106,6 +110,10 @@ module ActiveRecord
|
|
106
110
|
|
107
111
|
protected
|
108
112
|
|
113
|
+
def built_defaults
|
114
|
+
@built_defaults ||= build_defaults
|
115
|
+
end
|
116
|
+
|
109
117
|
# We cannot rely on string keys 'cause user input can contain symbol keys
|
110
118
|
def key_to_cast(val, key)
|
111
119
|
return key if val.key?(key)
|
@@ -121,7 +129,11 @@ module ActiveRecord
|
|
121
129
|
accessor_types.fetch(key.to_s)
|
122
130
|
end
|
123
131
|
|
124
|
-
|
132
|
+
def fallback_to_default?(key)
|
133
|
+
owner&.store_attribute_unset_values_fallback_to_default && defaults.key?(key)
|
134
|
+
end
|
135
|
+
|
136
|
+
attr_reader :accessor_types, :defaults, :store_accessor, :owner
|
125
137
|
end
|
126
138
|
end
|
127
139
|
end
|
data/lib/store_attribute.rb
CHANGED
metadata
CHANGED
@@ -1,14 +1,14 @@
|
|
1
1
|
--- !ruby/object:Gem::Specification
|
2
2
|
name: store_attribute
|
3
3
|
version: !ruby/object:Gem::Version
|
4
|
-
version: 1.
|
4
|
+
version: 1.1.1
|
5
5
|
platform: ruby
|
6
6
|
authors:
|
7
7
|
- palkan
|
8
8
|
autorequire:
|
9
9
|
bindir: bin
|
10
10
|
cert_chain: []
|
11
|
-
date:
|
11
|
+
date: 2023-06-27 00:00:00.000000000 Z
|
12
12
|
dependencies:
|
13
13
|
- !ruby/object:Gem::Dependency
|
14
14
|
name: activerecord
|
@@ -106,7 +106,7 @@ required_rubygems_version: !ruby/object:Gem::Requirement
|
|
106
106
|
- !ruby/object:Gem::Version
|
107
107
|
version: '0'
|
108
108
|
requirements: []
|
109
|
-
rubygems_version: 3.
|
109
|
+
rubygems_version: 3.4.8
|
110
110
|
signing_key:
|
111
111
|
specification_version: 4
|
112
112
|
summary: ActiveRecord extension which adds typecasting to store accessors
|