-
Notifications
You must be signed in to change notification settings - Fork 186
/
Copy pathactions.rb
334 lines (272 loc) · 10.6 KB
/
actions.rb
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
module ZendeskAPI
module ResponseHandler
def handle_response(response)
if response.body.is_a?(Hash) && response.body[self.class.singular_resource_name]
@attributes.replace(@attributes.deep_merge(response.body[self.class.singular_resource_name]))
end
end
end
module Save
include ResponseHandler
# If this resource hasn't been deleted, then create or save it.
# Executes a POST if it is a {Data#new_record?}, otherwise a PUT.
# Merges returned attributes on success.
# @return [Boolean] Success?
def save!(options = {})
return false if respond_to?(:destroyed?) && destroyed?
if new_record? && !options[:force_update]
method = :post
req_path = path
else
method = :put
req_path = url || path
end
req_path = options[:path] if options[:path]
save_associations
@response = @client.connection.send(method, req_path) do |req|
req.body = attributes_for_save.merge(@global_params)
yield req if block_given?
end
handle_response(@response)
@attributes.clear_changes
clear_associations
true
end
# Saves, returning false if it fails and attaching the errors
def save(options = {}, &block)
save!(options, &block)
rescue ZendeskAPI::Error::RecordInvalid => e
@errors = e.errors
false
rescue ZendeskAPI::Error::ClientError
false
end
# Removes all cached associations
def clear_associations
self.class.associations.each do |association_data|
name = association_data[:name]
instance_variable_set("@#{name}", nil) if instance_variable_defined?("@#{name}")
end
end
# Saves associations
# Takes into account inlining, collections, and id setting on the parent resource.
def save_associations
self.class.associations.each do |association_data|
association_name = association_data[:name]
next unless send("#{association_name}_used?") && association = send(association_name)
inline_creation = association_data[:inline] == :create && new_record?
changed = association.is_a?(Collection) || association.changed?
if association.respond_to?(:save) && changed && !inline_creation && association.save
send("#{association_name}=", association) # set id/ids columns
end
if (association_data[:inline] == true || inline_creation) && changed
attributes[association_name] = association.to_param
end
end
end
end
module Read
include ResponseHandler
include ZendeskAPI::Sideloading
def self.included(base)
base.extend(ClassMethods)
end
# Reloads a resource.
def reload!
response = @client.connection.get(path) do |req|
yield req if block_given?
end
handle_response(response)
attributes.clear_changes
self
end
module ClassMethods
# Finds a resource by an id and any options passed in.
# A custom path to search at can be passed into opts. It defaults to the {Data.resource_name} of the class.
# @param [Client] client The {Client} object to be used
# @param [Hash] options Any additional GET parameters to be added
def find!(client, options = {})
@client = client # so we can use client.logger in rescue
raise ArgumentError, "No :id given" unless options[:id] || options["id"] || ancestors.include?(SingularResource)
association = options.delete(:association) || Association.new(:class => self)
includes = Array(options[:include])
options[:include] = includes.join(",") if includes.any?
response = client.connection.get(association.generate_path(options)) do |req|
req.params = options
yield req if block_given?
end
new_from_response(client, response, includes)
end
# Finds, returning nil if it fails
# @param [Client] client The {Client} object to be used
# @param [Hash] options Any additional GET parameters to be added
def find(client, options = {}, &block)
find!(client, options, &block)
rescue ZendeskAPI::Error::ClientError
nil
end
end
end
module Create
include Save
def self.included(base)
base.extend(ClassMethods)
end
module ClassMethods
# Create a resource given the attributes passed in.
# @param [Client] client The {Client} object to be used
# @param [Hash] attributes The attributes to create.
def create!(client, attributes = {}, &block)
ZendeskAPI::Client.check_deprecated_namespace_usage attributes, singular_resource_name
new(client, attributes).tap do |resource|
resource.save!(&block)
end
end
# Creates, returning nil if it fails
# @param [Client] client The {Client} object to be used
# @param [Hash] options Any additional GET parameters to be added
def create(client, attributes = {}, &block)
create!(client, attributes, &block)
rescue ZendeskAPI::Error::ClientError
nil
end
end
end
module CreateMany
# Creates multiple resources using the create_many endpoint.
# @param [Client] client The {Client} object to be used
# @param [Array] attributes_array An array of resources to be created.
# @return [JobStatus] the {JobStatus} instance for this create job
def create_many!(client, attributes_array, association = Association.new(:class => self))
response = client.connection.post("#{association.generate_path}/create_many") do |req|
req.body = { resource_name => attributes_array }
yield req if block_given?
end
JobStatus.new_from_response(client, response)
end
end
module CreateOrUpdate
# Creates or updates resource using the create_or_update endpoint.
# @param [Client] client The {Client} object to be used
# @param [Hash] attributes The attributes to create.
def create_or_update!(client, attributes, association = Association.new(:class => self))
response = client.connection.post("#{association.generate_path}/create_or_update") do |req|
req.body = { singular_resource_name => attributes }
yield req if block_given?
end
new_from_response(client, response, Array(association.options[:include]))
end
end
module CreateOrUpdateMany
# Creates or updates multiple resources using the create_or_update_many endpoint.
#
# @param [Client] client The {Client} object to be used
# @param [Array<Hash>] attributes The attributes to update resources with
#
# @return [JobStatus] the {JobStatus} instance for this destroy job
def create_or_update_many!(client, attributes)
association = Association.new(:class => self)
response = client.connection.post("#{association.generate_path}/create_or_update_many") do |req|
req.body = { resource_name => attributes }
yield req if block_given?
end
JobStatus.new_from_response(client, response)
end
end
module Destroy
def self.included(klass)
klass.extend(ClassMethod)
end
# Has this object been deleted?
def destroyed?
@destroyed ||= false
end
# If this resource hasn't already been deleted, then do so.
# @return [Boolean] Successful?
def destroy!
return false if destroyed? || new_record?
@client.connection.delete(url || path) do |req|
yield req if block_given?
end
@destroyed = true
end
# Destroys, returning false on error.
def destroy(&block)
destroy!(&block)
rescue ZendeskAPI::Error::ClientError
false
end
module ClassMethod
# Deletes a resource given the id passed in.
# @param [Client] client The {Client} object to be used
# @param [Hash] opts The optional parameters to pass. Defaults to {}
def destroy!(client, opts = {}, &block)
new(client, opts).destroy!(&block)
true
end
# Destroys, returning false on error.
def destroy(client, attributes = {}, &block)
destroy!(client, attributes, &block)
rescue ZendeskAPI::Error::ClientError
false
end
end
end
module DestroyMany
# Destroys multiple resources using the destroy_many endpoint.
# @param [Client] client The {Client} object to be used
# @param [Array] ids An array of ids to destroy
# @return [JobStatus] the {JobStatus} instance for this destroy job
def destroy_many!(client, ids, association = Association.new(:class => self))
response = client.connection.delete("#{association.generate_path}/destroy_many") do |req|
req.params = { :ids => ids.join(',') }
yield req if block_given?
end
JobStatus.new_from_response(client, response)
end
end
module Update
include Save
def self.included(klass)
klass.extend(ClassMethod)
end
module ClassMethod
# Updates, returning false on error.
def update(client, attributes = {}, &block)
update!(client, attributes, &block)
rescue ZendeskAPI::Error::ClientError
false
end
# Updates a resource given the id passed in.
# @param [Client] client The {Client} object to be used
# @param [Hash] attributes The attributes to update. Default to {
def update!(client, attributes = {}, &block)
ZendeskAPI::Client.check_deprecated_namespace_usage attributes, singular_resource_name
resource = new(client, :id => attributes.delete(:id), :global => attributes.delete(:global), :association => attributes.delete(:association))
resource.attributes.merge!(attributes)
resource.save!(:force_update => resource.is_a?(SingularResource), &block)
resource
end
end
end
module UpdateMany
# Updates multiple resources using the update_many endpoint.
# @param [Client] client The {Client} object to be used
# @param [Array] ids_or_attributes An array of ids or arributes including ids to update
# @param [Hash] attributes The attributes to update resources with
# @return [JobStatus] the {JobStatus} instance for this destroy job
def update_many!(client, ids_or_attributes, attributes = {})
association = attributes.delete(:association) || Association.new(:class => self)
response = client.connection.put("#{association.generate_path}/update_many") do |req|
if attributes == {}
req.body = { resource_name => ids_or_attributes }
else
req.params = { :ids => ids_or_attributes.join(',') }
req.body = { singular_resource_name => attributes }
end
yield req if block_given?
end
JobStatus.new_from_response(client, response)
end
end
end