5 class ApiError < StandardError
6 attr_reader :api_response, :api_response_s, :api_status, :request_url
8 def initialize(request_url, errmsg)
9 @request_url = request_url
11 errors = @api_response[:errors]
12 if not errors.is_a?(Array)
13 @api_response[:errors] = [errors || errmsg]
19 class NoApiResponseException < ApiError
20 def initialize(request_url, exception)
21 @api_response_s = exception.to_s
23 "#{exception.class.to_s} error connecting to API server")
27 class InvalidApiResponseException < ApiError
28 def initialize(request_url, api_response)
29 @api_status = api_response.status_code
30 @api_response_s = api_response.content
31 super(request_url, "Unparseable response from API server")
35 class ApiErrorResponseException < ApiError
36 def initialize(request_url, api_response)
37 @api_status = api_response.status_code
38 @api_response_s = api_response.content
39 @api_response = Oj.load(@api_response_s, :symbol_keys => true)
40 errors = @api_response[:errors]
41 if errors.respond_to?(:join)
42 errors = errors.join("\n\n")
46 super(request_url, "#{errors} [API: #{@api_status}]")
50 class AccessForbiddenException < ApiErrorResponseException; end
51 class NotFoundException < ApiErrorResponseException; end
52 class NotLoggedInException < ApiErrorResponseException; end
54 ERROR_CODE_CLASSES = {
55 401 => NotLoggedInException,
56 403 => AccessForbiddenException,
57 404 => NotFoundException,
60 @@profiling_enabled = Rails.configuration.profiling_enabled
63 # An API client object suitable for handling API requests on behalf
64 # of the current thread.
65 def self.new_or_current
66 # If this thread doesn't have an API client yet, *or* this model
67 # has been reloaded since the existing client was created, create
68 # a new client. Otherwise, keep using the latest client created in
70 unless Thread.current[:arvados_api_client].andand.class == self
71 Thread.current[:arvados_api_client] = new
73 Thread.current[:arvados_api_client]
78 @client_mtx = Mutex.new
81 def api(resources_kind, action, data=nil, tokens={})
86 @client_mtx.synchronize do
87 @api_client = HTTPClient.new
88 @api_client.ssl_config.timeout = Rails.configuration.api_client_connect_timeout
89 @api_client.connect_timeout = Rails.configuration.api_client_connect_timeout
90 @api_client.receive_timeout = Rails.configuration.api_client_receive_timeout
91 if Rails.configuration.arvados_insecure_https
92 @api_client.ssl_config.verify_mode = OpenSSL::SSL::VERIFY_NONE
94 # Use system CA certificates
95 ["/etc/ssl/certs/ca-certificates.crt",
96 "/etc/pki/tls/certs/ca-bundle.crt"]
97 .select { |ca_path| File.readable?(ca_path) }
98 .each { |ca_path| @api_client.ssl_config.add_trust_ca(ca_path) }
100 if Rails.configuration.api_response_compression
101 @api_client.transparent_gzip_decompression = true
106 resources_kind = class_kind(resources_kind).pluralize if resources_kind.is_a? Class
107 url = "#{self.arvados_v1_base}/#{resources_kind}#{action}"
109 # Clean up /arvados/v1/../../discovery/v1 to /discovery/v1
110 url.sub! '/arvados/v1/../../', '/'
113 'api_token' => (tokens[:arvados_api_token] ||
114 Thread.current[:arvados_api_token] ||
116 'reader_tokens' => ((tokens[:reader_tokens] ||
117 Thread.current[:reader_tokens] ||
119 [Rails.configuration.anonymous_user_token]).to_json,
120 'current_request_id' => (Thread.current[:current_request_id] || ''),
124 if v.is_a? String or v.nil?
131 query[k] = Oj.dump(v, mode: :compat)
135 query["_method"] = "GET"
138 if @@profiling_enabled
139 query["_profile"] = "true"
142 header = {"Accept" => "application/json"}
144 profile_checkpoint { "Prepare request #{query["_method"] or "POST"} #{url} #{query[:uuid]} #{query.inspect[0,256]}" }
145 msg = @client_mtx.synchronize do
147 @api_client.post(url, query, header: header)
149 raise NoApiResponseException.new(url, exception)
152 profile_checkpoint 'API transaction'
153 if @@profiling_enabled
154 if msg.headers['X-Runtime']
155 Rails.logger.info "API server: #{msg.headers['X-Runtime']} runtime reported"
157 Rails.logger.info "Content-Encoding #{msg.headers['Content-Encoding'].inspect}, Content-Length #{msg.headers['Content-Length'].inspect}, actual content size #{msg.content.size}"
161 resp = Oj.load(msg.content, :symbol_keys => true)
162 rescue Oj::ParseError
166 if not resp.is_a? Hash
167 raise InvalidApiResponseException.new(url, msg)
168 elsif msg.status_code != 200
169 error_class = ERROR_CODE_CLASSES.fetch(msg.status_code,
170 ApiErrorResponseException)
171 raise error_class.new(url, msg)
175 Rails.logger.info "API client: " \
176 "#{resp.delete(:_profile)[:request_time]} request_time"
178 profile_checkpoint 'Parse response'
182 def self.patch_paging_vars(ary, items_available, offset, limit, links=nil)
184 (class << ary; self; end).class_eval { attr_accessor :items_available }
185 ary.items_available = items_available
188 (class << ary; self; end).class_eval { attr_accessor :offset }
192 (class << ary; self; end).class_eval { attr_accessor :limit }
196 (class << ary; self; end).class_eval { attr_accessor :links }
202 def unpack_api_response(j, kind=nil)
203 if j.is_a? Hash and j[:items].is_a? Array and j[:kind].match(/(_list|List)$/)
204 ary = j[:items].collect { |x| unpack_api_response x, x[:kind] }
205 links = ArvadosResourceList.new Link
206 links.results = (j[:links] || []).collect do |x|
207 unpack_api_response x, x[:kind]
209 self.class.patch_paging_vars(ary, j[:items_available], j[:offset], j[:limit], links)
210 elsif j.is_a? Hash and (kind || j[:kind])
211 oclass = self.kind_class(kind || j[:kind])
214 childkind = j["#{k.to_s}_kind".to_sym]
216 j[k] = self.unpack_api_response(j[k], childkind)
219 oclass.new.private_reload(j)
228 def arvados_login_url(params={})
229 if Rails.configuration.respond_to? :arvados_login_base
230 uri = Rails.configuration.arvados_login_base
232 uri = self.arvados_v1_base.sub(%r{/arvados/v\d+.*}, '/login')
235 uri += '?' << params.collect { |k,v|
236 CGI.escape(k.to_s) + '=' + CGI.escape(v.to_s)
242 def arvados_logout_url(params={})
243 arvados_login_url(params).sub('/login','/logout')
247 Rails.configuration.arvados_v1_base
251 @@discovery ||= api '../../discovery/v1/apis/arvados/v1/rest', ''
255 kind.match(/^arvados\#(.+?)(_list|List)?$/)[1].pluralize.classify.constantize rescue nil
258 def class_kind(resource_class)
259 resource_class.to_s.underscore
262 def self.class_kind(resource_class)
263 resource_class.to_s.underscore
267 def profile_checkpoint label=nil
268 return if !@@profiling_enabled
269 label = yield if block_given?
271 if label and @profile_t0
272 Rails.logger.info "API client: #{t - @profile_t0} #{label}"