require 'singleton' require 'json' require 'excon' require_relative './error' module Kong class Client include Singleton attr_accessor :default_headers attr_reader :http_client # Initialize api client # def initialize Excon.defaults[:ssl_verify_peer] = false if ignore_ssl_errors? @api_url = api_url @http_client = Excon.new(@api_url, omit_default_port: true) @default_headers = { 'Accept' => 'application/json' } end def self.api_url self.instance.api_url end def self.api_url=(url) self.instance.api_url = url end # Kong Admin API URL # # @return [String] def api_url @api_url ||= ENV['KONG_URI'] || 'http://localhost:8001' end def api_url=(url) @api_url = url end # Get request # # @param [String] path # @param [Hash,NilClass] params # @param [Hash] headers # @return [Hash] def get(path, params = nil, headers = {}) response = http_client.get( path: path, query: encode_params(params), headers: request_headers(headers) ) if response.status == 200 parse_response(response) else handle_error_response(response) end end # Post request # # @param [String] path # @param [Object] obj # @param [Hash] params # @param [Hash] headers # @return [Hash] def post(path, obj, params = {}, headers = {}) request_headers = request_headers(headers) request_options = { path: path, headers: request_headers, body: encode_body(obj, request_headers['Content-Type']), query: encode_params(params) } response = http_client.post(request_options) if [200, 201].include?(response.status) parse_response(response) else handle_error_response(response) end end # Put request # # @param [String] path # @param [Object] obj # @param [Hash] params # @param [Hash] headers # @return [Hash] def patch(path, obj, params = {}, headers = {}) request_headers = request_headers(headers) request_options = { path: path, headers: request_headers, body: encode_body(obj, request_headers['Content-Type']), query: encode_params(params) } response = http_client.patch(request_options) if [200, 201].include?(response.status) parse_response(response) else handle_error_response(response) end end # Put request # # @param [String] path # @param [Object] obj # @param [Hash] params # @param [Hash] headers # @return [Hash] def put(path, obj, params = {}, headers = {}) request_headers = request_headers(headers) request_options = { path: path, headers: request_headers, body: encode_body(obj, request_headers['Content-Type']), query: encode_params(params) } response = http_client.put(request_options) if [200, 201].include?(response.status) parse_response(response) else handle_error_response(response) end end # Delete request # # @param [String] path # @param [Hash,String] body # @param [Hash] params # @param [Hash] headers # @return [Hash] def delete(path, body = nil, params = {}, headers = {}) request_headers = request_headers(headers) request_options = { path: path, headers: request_headers, body: encode_body(body, request_headers['Content-Type']), query: encode_params(params) } response = http_client.delete(request_options) unless response.status == 204 handle_error_response(response) end end private ## # Get request headers # # @param [Hash] headers # @return [Hash] def request_headers(headers = {}) @default_headers.merge(headers) end ## # Encode body based on content type # # @param [Object] body # @param [String] content_type def encode_body(body, content_type) if content_type == 'application/json' dump_json(body) else body end end def encode_params(params) return nil if params.nil? URI.encode_www_form(params) end ## # Parse response # # @param [HTTP::Message] # @return [Object] def parse_response(response) if response.headers['Content-Type'].include?('application/json') parse_json(response.body) else response.body end end ## # Parse json # # @param [String] json # @return [Hash,Object,NilClass] def parse_json(json) JSON.parse(json) rescue nil end ## # Dump json # # @param [Object] obj # @return [String] def dump_json(obj) JSON.dump(obj) end def ignore_ssl_errors? ENV['SSL_IGNORE_ERRORS'] == 'true' end def handle_error_response(response) message = response.body if response.status == 404 && message == '' message = 'Not found' end raise Error.new(response.status, message) end end end