requisite 0.4.2 → 0.4.3
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- checksums.yaml +4 -4
- data/README.md +25 -5
- data/lib/requisite/boundary_object.rb +25 -15
- data/lib/requisite/version.rb +1 -1
- data/test/requisite/api_model_test.rb +2 -2
- data/test/requisite/api_user_test.rb +42 -9
- metadata +6 -7
checksums.yaml
CHANGED
@@ -1,7 +1,7 @@
|
|
1
1
|
---
|
2
2
|
SHA256:
|
3
|
-
metadata.gz:
|
4
|
-
data.tar.gz:
|
3
|
+
metadata.gz: 828a5465ab9ec59c47bda9e6ab72c34e9bdbab05536ab99ad45c3cdaec933f16
|
4
|
+
data.tar.gz: b1c387023439e720d6600310604828d672abd8dd4eabb25d0412bcec423c4529
|
5
5
|
SHA512:
|
6
|
-
metadata.gz:
|
7
|
-
data.tar.gz:
|
6
|
+
metadata.gz: eb0c6e6e4cc10fb351e01334754196f2a65ced999a32980f416a2c226b4ea1a76a77b0a1b0523e408de9b372fc40676ab7e9482290f28d13d983947ae750a9f9
|
7
|
+
data.tar.gz: 3269605161e8264e04dc0640637a107d4db5ef079f635e31c674cc35cb409eb32ccb991a3f17bcebe4f31bedcd6616fbfb09856829a86f5a802f67f998df9ac1
|
data/README.md
CHANGED
@@ -38,7 +38,7 @@ class UserApiModel < Requisite::ApiModel
|
|
38
38
|
attribute! :username
|
39
39
|
attribute :real_name
|
40
40
|
end
|
41
|
-
|
41
|
+
|
42
42
|
# method with the name of of an attribute will be called to calculate the mapped value
|
43
43
|
def real_name
|
44
44
|
"#{attribute_from_model(:first_name)} #{attribute_from_model(:last_name)}"
|
@@ -88,10 +88,10 @@ Example:
|
|
88
88
|
class UserApiModel < Requisite::ApiModel
|
89
89
|
serialized_attributes do
|
90
90
|
attribute :id, stringify: true
|
91
|
-
attribute :custom_attributes, rename: :custom_data
|
91
|
+
attribute :custom_attributes, rename: :custom_data
|
92
92
|
attribute :is_awesome, default: true
|
93
93
|
attribute :awesome_score, rename: :score, stringify: true, default: 9001
|
94
|
-
attribute :age, type:
|
94
|
+
attribute :age, type: Integer,
|
95
95
|
attribute :tired, type: Requisite::Boolean
|
96
96
|
end
|
97
97
|
end
|
@@ -131,7 +131,7 @@ With typed hashes, only values specified with a type are permitted:
|
|
131
131
|
```ruby
|
132
132
|
class UserApiModel < Requisite::ApiModel
|
133
133
|
serialized_attributes do
|
134
|
-
attribute :data, typed_hash: { is_awesome: Requisite::Boolean, score:
|
134
|
+
attribute :data, typed_hash: { is_awesome: Requisite::Boolean, score: Integer, name: String }
|
135
135
|
end
|
136
136
|
end
|
137
137
|
|
@@ -198,7 +198,27 @@ class ApiUser < Requisite::ApiModel
|
|
198
198
|
raise IdentifierNotFoundError unless identifier
|
199
199
|
end
|
200
200
|
end
|
201
|
-
```
|
201
|
+
```
|
202
|
+
|
203
|
+
#### Around each attribute
|
204
|
+
|
205
|
+
An `around_each_attribute` method can be defined to wrap each attribute fetch in a block. This can be useful for instrumenting processing on a per attribute basis.
|
206
|
+
|
207
|
+
```ruby
|
208
|
+
class ApiUser < Requisite::ApiModel
|
209
|
+
serialized_attributes do
|
210
|
+
attribute :id, type: String
|
211
|
+
attribute :email, type: String
|
212
|
+
end
|
213
|
+
|
214
|
+
def around_each_attribute(name, &block)
|
215
|
+
start = Time.now
|
216
|
+
yield
|
217
|
+
end = Time.now
|
218
|
+
puts "Fetching #{name} took #{end - start}"
|
219
|
+
end
|
220
|
+
end
|
221
|
+
```
|
202
222
|
|
203
223
|
#### Thanks
|
204
224
|
|
@@ -4,14 +4,17 @@ module Requisite
|
|
4
4
|
def attribute(name, options={})
|
5
5
|
attribute_keys << name
|
6
6
|
define_method(name) do
|
7
|
-
|
8
|
-
|
9
|
-
|
10
|
-
|
11
|
-
|
12
|
-
|
13
|
-
|
14
|
-
|
7
|
+
result = nil
|
8
|
+
self.send(:around_each_attribute, name) do
|
9
|
+
resolved_name = options[:rename] || name
|
10
|
+
result = self.send(:convert, resolved_name)
|
11
|
+
result = self.send(:parse_typed_hash, resolved_name, options[:typed_hash]) if options[:typed_hash]
|
12
|
+
result = self.send(:parse_scalar_hash, resolved_name) if options[:scalar_hash]
|
13
|
+
result = self.send(:parse_typed_array, resolved_name, options[:typed_array]) if options[:typed_array]
|
14
|
+
result = options[:default] if (options.key?(:default) && empty_result?(result))
|
15
|
+
raise_bad_type_if_type_mismatch(result, options[:type]) if options[:type] && result
|
16
|
+
result = result.to_s if options[:stringify]
|
17
|
+
end
|
15
18
|
result
|
16
19
|
end
|
17
20
|
end
|
@@ -19,13 +22,16 @@ module Requisite
|
|
19
22
|
def attribute!(name, options={})
|
20
23
|
attribute_keys << name
|
21
24
|
define_method(name) do
|
22
|
-
|
23
|
-
|
24
|
-
|
25
|
-
|
26
|
-
|
27
|
-
|
28
|
-
|
25
|
+
result = nil
|
26
|
+
self.send(:around_each_attribute, name) do
|
27
|
+
resolved_name = options[:rename] || name
|
28
|
+
result = self.send(:convert!, resolved_name)
|
29
|
+
result = self.send(:parse_typed_hash, resolved_name, options[:typed_hash]) if options[:typed_hash]
|
30
|
+
result = self.send(:parse_scalar_hash, resolved_name) if options[:scalar_hash]
|
31
|
+
result = self.send(:parse_typed_array, resolved_name, options[:typed_array]) if options[:typed_array]
|
32
|
+
result = result.to_s if options[:stringify]
|
33
|
+
raise_bad_type_if_type_mismatch(result, options[:type]) if options[:type]
|
34
|
+
end
|
29
35
|
result
|
30
36
|
end
|
31
37
|
end
|
@@ -46,6 +52,10 @@ module Requisite
|
|
46
52
|
|
47
53
|
private
|
48
54
|
|
55
|
+
def around_each_attribute(name)
|
56
|
+
yield
|
57
|
+
end
|
58
|
+
|
49
59
|
self.singleton_class.send(:alias_method, :a, :attribute)
|
50
60
|
self.singleton_class.send(:alias_method, :a!, :attribute!)
|
51
61
|
|
data/lib/requisite/version.rb
CHANGED
@@ -107,7 +107,7 @@ module Requisite
|
|
107
107
|
end
|
108
108
|
|
109
109
|
it 'attribute can be stringified after type check' do
|
110
|
-
DummyApiModel.serialized_attributes { attribute :num, stringify: true, type:
|
110
|
+
DummyApiModel.serialized_attributes { attribute :num, stringify: true, type: Integer }
|
111
111
|
response = DummyApiModel.new(params_hash)
|
112
112
|
_(response.to_hash).must_equal(:num => '12')
|
113
113
|
end
|
@@ -202,7 +202,7 @@ module Requisite
|
|
202
202
|
|
203
203
|
describe 'with typed arrays' do
|
204
204
|
it 'allows arrays of one type' do
|
205
|
-
DummyApiModel.serialized_attributes { attribute :ids, typed_array:
|
205
|
+
DummyApiModel.serialized_attributes { attribute :ids, typed_array: Integer }
|
206
206
|
response = DummyApiModel.new({ids: [1, 2, 3]})
|
207
207
|
_(response.to_hash).must_equal(:ids => [1, 2, 3])
|
208
208
|
end
|
@@ -1,15 +1,17 @@
|
|
1
1
|
require 'test_helper'
|
2
|
+
require 'benchmark'
|
2
3
|
|
3
4
|
# Example Object
|
4
5
|
class ApiUser < Requisite::ApiModel
|
6
|
+
|
5
7
|
serialized_attributes do
|
6
8
|
attribute :id, type: String
|
7
9
|
attribute :user_id
|
8
10
|
attribute :email, type: String
|
9
11
|
attribute :name, type: String
|
10
|
-
attribute :created_at, type:
|
12
|
+
attribute :created_at, type: Integer
|
11
13
|
attribute :last_seen_user_agent, type: String
|
12
|
-
attribute :last_request_at, type:
|
14
|
+
attribute :last_request_at, type: Integer
|
13
15
|
attribute :unsubscribed_from_emails, type: Requisite::Boolean
|
14
16
|
attribute :update_last_request_at, type: Requisite::Boolean
|
15
17
|
attribute :new_session, type: Requisite::Boolean
|
@@ -17,7 +19,7 @@ class ApiUser < Requisite::ApiModel
|
|
17
19
|
attribute :company
|
18
20
|
attribute :companies
|
19
21
|
end
|
20
|
-
|
22
|
+
|
21
23
|
# Ensure that at least one identifier is passed
|
22
24
|
def preprocess_model
|
23
25
|
identifier = attribute_from_model(:id)
|
@@ -25,10 +27,30 @@ class ApiUser < Requisite::ApiModel
|
|
25
27
|
identifier ||= attribute_from_model(:email)
|
26
28
|
raise StandardError unless identifier
|
27
29
|
end
|
28
|
-
|
30
|
+
|
31
|
+
def last_attribute_fetch_time
|
32
|
+
@last_attribute_fetch_time
|
33
|
+
end
|
34
|
+
|
35
|
+
def attribute_names
|
36
|
+
@attribute_names
|
37
|
+
end
|
38
|
+
|
39
|
+
def around_each_attribute(name)
|
40
|
+
@last_attribute_fetch_time = nil
|
41
|
+
@attribute_names ||= []
|
42
|
+
|
43
|
+
result = nil
|
44
|
+
|
45
|
+
@last_attribute_fetch_time = Benchmark.measure do
|
46
|
+
result = yield
|
47
|
+
end.total
|
48
|
+
@attribute_names << name
|
49
|
+
end
|
50
|
+
|
29
51
|
# We want to accept someone sending `created_at` or `created` as parameters
|
30
52
|
def created_at
|
31
|
-
with_type!(
|
53
|
+
with_type!(Integer) { attribute_from_model(:created_at) || attribute_from_model(:created) }
|
32
54
|
end
|
33
55
|
end
|
34
56
|
|
@@ -71,13 +93,13 @@ module Requisite
|
|
71
93
|
})
|
72
94
|
user.name.must_equal('Bob')
|
73
95
|
end
|
74
|
-
|
96
|
+
|
75
97
|
it 'raises an error without an identifier' do
|
76
98
|
user_request_params = { :name => 'Bob' }
|
77
99
|
user = ApiUser.new(user_request_params)
|
78
100
|
proc { user.to_hash }.must_raise(StandardError)
|
79
101
|
end
|
80
|
-
|
102
|
+
|
81
103
|
it 'raises an error when created or created_at is not of the right type' do
|
82
104
|
user_request_params = { :user_id => 'abcdef', :created => 'Thursday' }
|
83
105
|
user = ApiUser.new(user_request_params)
|
@@ -116,7 +138,7 @@ module Requisite
|
|
116
138
|
:new_attribute => 'hi'
|
117
139
|
})
|
118
140
|
end
|
119
|
-
|
141
|
+
|
120
142
|
it 'accepts a user model' do
|
121
143
|
user_model = UserModel.new
|
122
144
|
user_model.user_id = 'abcdef'
|
@@ -129,7 +151,7 @@ module Requisite
|
|
129
151
|
})
|
130
152
|
user.name.must_equal('Bob')
|
131
153
|
end
|
132
|
-
|
154
|
+
|
133
155
|
it 'accepts a user model and renders nils if asked' do
|
134
156
|
user_model = UserModel.new
|
135
157
|
user_model.user_id = 'abcdef'
|
@@ -152,5 +174,16 @@ module Requisite
|
|
152
174
|
})
|
153
175
|
user.name.must_equal('Bob')
|
154
176
|
end
|
177
|
+
|
178
|
+
it 'calls around_each_attribute for each attribute' do
|
179
|
+
user_model = UserModel.new
|
180
|
+
user_model.user_id = 'abcdef'
|
181
|
+
user = ApiUser.new(user_model)
|
182
|
+
|
183
|
+
user.to_hash(show_nil: true)
|
184
|
+
|
185
|
+
user.attribute_names.must_equal [:id, :user_id, :email, :name, :last_seen_user_agent, :last_request_at, :unsubscribed_from_emails, :update_last_request_at, :new_session, :custom_data, :company, :companies]
|
186
|
+
user.last_attribute_fetch_time.must_be :>, 0
|
187
|
+
end
|
155
188
|
end
|
156
189
|
end
|
metadata
CHANGED
@@ -1,14 +1,14 @@
|
|
1
1
|
--- !ruby/object:Gem::Specification
|
2
2
|
name: requisite
|
3
3
|
version: !ruby/object:Gem::Version
|
4
|
-
version: 0.4.
|
4
|
+
version: 0.4.3
|
5
5
|
platform: ruby
|
6
6
|
authors:
|
7
7
|
- James Osler
|
8
|
-
autorequire:
|
8
|
+
autorequire:
|
9
9
|
bindir: bin
|
10
10
|
cert_chain: []
|
11
|
-
date:
|
11
|
+
date: 2025-03-25 00:00:00.000000000 Z
|
12
12
|
dependencies:
|
13
13
|
- !ruby/object:Gem::Dependency
|
14
14
|
name: minitest
|
@@ -82,7 +82,7 @@ homepage: https://www.intercom.io
|
|
82
82
|
licenses:
|
83
83
|
- Apache License Version 2.0
|
84
84
|
metadata: {}
|
85
|
-
post_install_message:
|
85
|
+
post_install_message:
|
86
86
|
rdoc_options: []
|
87
87
|
require_paths:
|
88
88
|
- lib
|
@@ -97,9 +97,8 @@ required_rubygems_version: !ruby/object:Gem::Requirement
|
|
97
97
|
- !ruby/object:Gem::Version
|
98
98
|
version: '0'
|
99
99
|
requirements: []
|
100
|
-
|
101
|
-
|
102
|
-
signing_key:
|
100
|
+
rubygems_version: 3.5.22
|
101
|
+
signing_key:
|
103
102
|
specification_version: 4
|
104
103
|
summary: Strongly defined models for HTTP APIs
|
105
104
|
test_files:
|