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 CHANGED
@@ -1,7 +1,7 @@
1
1
  ---
2
2
  SHA256:
3
- metadata.gz: 22ae33af68416be1beb8abf9c3384934e4c93f1a46ab089c05436956a7391e3a
4
- data.tar.gz: ef8a85a1fb0af0aedee42f635283590306016e0101adef510365f065ae6cb908
3
+ metadata.gz: 828a5465ab9ec59c47bda9e6ab72c34e9bdbab05536ab99ad45c3cdaec933f16
4
+ data.tar.gz: b1c387023439e720d6600310604828d672abd8dd4eabb25d0412bcec423c4529
5
5
  SHA512:
6
- metadata.gz: 35f9ebfbf4ff91f05067c422e27b13caed813d926093f0c5bf792375be57cdb03a5451e8690a999417826334d762484e63eb58025c0eafda6bade2db210d5615
7
- data.tar.gz: 362d50d708a1d1f80002dc751fa41cf5baf8ed1dea3cd66725e80472df2fd1dcb9d9224964699cd06a3deecfccd8e7c10466dc852a89731b8b55169f65c0fceb
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: Fixnum,
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: Fixnum, name: String }
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
- resolved_name = options[:rename] || name
8
- result = self.send(:convert, resolved_name)
9
- result = self.send(:parse_typed_hash, resolved_name, options[:typed_hash]) if options[:typed_hash]
10
- result = self.send(:parse_scalar_hash, resolved_name) if options[:scalar_hash]
11
- result = self.send(:parse_typed_array, resolved_name, options[:typed_array]) if options[:typed_array]
12
- result = options[:default] if (options.key?(:default) && empty_result?(result))
13
- raise_bad_type_if_type_mismatch(result, options[:type]) if options[:type] && result
14
- result = result.to_s if options[:stringify]
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
- resolved_name = options[:rename] || name
23
- result = self.send(:convert!, resolved_name)
24
- result = self.send(:parse_typed_hash, resolved_name, options[:typed_hash]) if options[:typed_hash]
25
- result = self.send(:parse_scalar_hash, resolved_name) if options[:scalar_hash]
26
- result = self.send(:parse_typed_array, resolved_name, options[:typed_array]) if options[:typed_array]
27
- result = result.to_s if options[:stringify]
28
- raise_bad_type_if_type_mismatch(result, options[:type]) if options[:type]
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
 
@@ -1,3 +1,3 @@
1
1
  module Requisite
2
- VERSION = '0.4.2'
2
+ VERSION = '0.4.3'
3
3
  end
@@ -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: Fixnum }
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: Fixnum }
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: Fixnum
12
+ attribute :created_at, type: Integer
11
13
  attribute :last_seen_user_agent, type: String
12
- attribute :last_request_at, type: Fixnum
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!(Fixnum) { attribute_from_model(:created_at) || attribute_from_model(:created) }
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.2
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: 2022-09-15 00:00:00.000000000 Z
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
- rubyforge_project:
101
- rubygems_version: 2.7.6.2
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: