module Gibbon class APIRequest include Helpers def initialize(builder: nil) @request_builder = builder end def post(params: nil, headers: nil, body: nil) validate_api_key begin response = self.rest_client.post do |request| configure_request(request: request, params: params, headers: headers, body: MultiJson.dump(body)) end parse_response(response) rescue => e handle_error(e) end end def patch(params: nil, headers: nil, body: nil) validate_api_key begin response = self.rest_client.patch do |request| configure_request(request: request, params: params, headers: headers, body: MultiJson.dump(body)) end parse_response(response) rescue => e handle_error(e) end end def put(params: nil, headers: nil, body: nil) validate_api_key begin response = self.rest_client.put do |request| configure_request(request: request, params: params, headers: headers, body: MultiJson.dump(body)) end parse_response(response) rescue => e handle_error(e) end end def get(params: nil, headers: nil) validate_api_key begin response = self.rest_client.get do |request| configure_request(request: request, params: params, headers: headers) end parse_response(response) rescue => e handle_error(e) end end def delete(params: nil, headers: nil) validate_api_key begin response = self.rest_client.delete do |request| configure_request(request: request, params: params, headers: headers) end parse_response(response) rescue => e handle_error(e) end end protected # Convenience accessors def api_key @request_builder.api_key end def api_endpoint @request_builder.api_endpoint end def timeout @request_builder.timeout end def open_timeout @request_builder.open_timeout end def proxy @request_builder.proxy end def adapter @request_builder.faraday_adapter end def symbolize_keys @request_builder.symbolize_keys end # Helpers def handle_error(error) error_params = {} begin if error.is_a?(Faraday::Error::ClientError) && error.response error_params[:status_code] = error.response[:status] error_params[:raw_body] = error.response[:body] parsed_response = MultiJson.load(error.response[:body], symbolize_keys: symbolize_keys) if parsed_response error_params[:body] = parsed_response error_params[:title] = parsed_response["title"] if parsed_response["title"] error_params[:detail] = parsed_response["detail"] if parsed_response["detail"] end end rescue MultiJson::ParseError end error_to_raise = MailChimpError.new(error.message, error_params) raise error_to_raise end def configure_request(request: nil, params: nil, headers: nil, body: nil) if request request.params.merge!(params) if params request.headers['Content-Type'] = 'application/json' request.headers.merge!(headers) if headers request.body = body if body request.options.timeout = self.timeout request.options.open_timeout = self.open_timeout end end def rest_client client = Faraday.new(self.api_url, proxy: self.proxy) do |faraday| faraday.response :raise_error faraday.adapter adapter if @request_builder.debug faraday.response :logger, @request_builder.logger, bodies: true end end client.basic_auth('apikey', self.api_key) client end def parse_response(response) parsed_response = nil if response.body && !response.body.empty? begin headers = response.headers body = MultiJson.load(response.body, symbolize_keys: symbolize_keys) parsed_response = Response.new(headers: headers, body: body) rescue MultiJson::ParseError error = MailChimpError.new("Unparseable response: #{response.body}") error.title = "UNPARSEABLE_RESPONSE" error.status_code = 500 raise error end end parsed_response end def validate_api_key api_key = self.api_key unless api_key && (api_key["-"] || self.api_endpoint) raise Gibbon::GibbonError, "You must set an api_key prior to making a call" end end def api_url base_api_url + @request_builder.path end def base_api_url computed_api_endpoint = "https://#{get_data_center_from_api_key(self.api_key)}api.mailchimp.com" "#{self.api_endpoint || computed_api_endpoint}/3.0/" end end end