module Restfulie module Client module HTTP #:nodoc: # Request Adapter provides a minimal interface to exchange information between server over HTTP protocol through simple adapters. # # All the concrete adapters follow the interface laid down in this module. # Default connection provider is net/http # #==Example # # @re = ::Restfulie::Client::HTTP::RequestExecutor.new('http://restfulie.com') #this class includes RequestAdapter module. # puts @re.as('application/atom+xml').get!('/posts').title #=> 'Hello World!' # module RequestAdapter attr_reader :host attr_accessor :cookies attr_writer :default_headers def host=(host) if host.is_a?(::URI) @host = host else @host = ::URI.parse(host) end end def default_headers @default_headers ||= {} end # GET HTTP verb without {Error} # * path: '/posts' # * headers: {'Accept' => '*/*', 'Content-Type' => 'application/atom+xml'} def get(path, *args) request(:get, path, *args) end # HEAD HTTP verb without {Error} # * path: '/posts' # * headers: {'Accept' => '*/*', 'Content-Type' => 'application/atom+xml'} def head(path, *args) request(:head, path, *args) end # POST HTTP verb without {Error} # * path: '/posts' # * payload: 'some text' # * headers: {'Accept' => '*/*', 'Content-Type' => 'application/atom+xml'} def post(path, payload, *args) request(:post, path, payload, *args) end # PATCH HTTP verb without {Error} # * path: '/posts' # * payload: 'some text' # * headers: {'Accept' => '*/*', 'Content-Type' => 'application/atom+xml'} def patch(path, payload, *args) request(:patch, path, payload, *args) end # PUT HTTP verb without {Error} # * path: '/posts' # * payload: 'some text' # * headers: {'Accept' => '*/*', 'Content-Type' => 'application/atom+xml'} def put(path, payload, *args) request(:put, path, payload, *args) end # DELETE HTTP verb without {Error} # * path: '/posts' # * headers: {'Accept' => '*/*', 'Content-Type' => 'application/atom+xml'} def delete(path, *args) request(:delete, path, *args) end # GET HTTP verb {Error} # * path: '/posts' # * headers: {'Accept' => '*/*', 'Content-Type' => 'application/atom+xml'} def get!(path, *args) request!(:get, path, *args) end # HEAD HTTP verb {Error} # * path: '/posts' # * headers: {'Accept' => '*/*', 'Content-Type' => 'application/atom+xml'} def head!(path, *args) request!(:head, path, *args) end # POST HTTP verb {Error} # * path: '/posts' # * payload: 'some text' # * headers: {'Accept' => '*/*', 'Content-Type' => 'application/atom+xml'} def post!(path, payload, *args) request!(:post, path, payload, *args) end # PATCH HTTP verb {Error} # * path: '/posts' # * payload: 'some text' # * headers: {'Accept' => '*/*', 'Content-Type' => 'application/atom+xml'} def patch!(path, payload, *args) request!(:patch, path, payload, *args) end # PUT HTTP verb {Error} # * path: '/posts' # * payload: 'some text' # * headers: {'Accept' => '*/*', 'Content-Type' => 'application/atom+xml'} def put!(path, payload, *args) request!(:put, path, payload, *args) end # DELETE HTTP verb {Error} # * path: '/posts' # * headers: {'Accept' => '*/*', 'Content-Type' => 'application/atom+xml'} def delete!(path, *args) request!(:delete, path, *args) end # Executes a request against your server and return a response instance without {Error} # * method: :get,:post,:delete,:head,:put # * path: '/posts' # * args: payload: 'some text' and/or headers: {'Accept' => '*/*', 'Content-Type' => 'application/atom+xml'} def request(method, path, *args) request!(method, path, *args) rescue Error::RESTError => se [[@host, path], nil, se.response] end # Executes a request against your server and return a response instance. # * method: :get,:post,:delete,:head,:put # * path: '/posts' # * args: payload: 'some text' and/or headers: {'Accept' => '*/*', 'Content-Type' => 'application/atom+xml'} def request!(method, path, *args) headers = default_headers.merge(args.extract_options!) unless @host.user.blank? && @host.password.blank? headers["Authorization"] = "Basic " + ["#{@host.user}:#{@host.password}"].pack("m").delete("\r\n") end headers['cookie'] = @cookies if @cookies args << headers ::Restfulie::Common::Logger.logger.info(request_to_s(method, path, *args)) if ::Restfulie::Common::Logger.logger begin http_request = get_connection_provider response = Restfulie::Client.cache_provider.get([@host, path], http_request, method) return [[@host, path], http_request, response] if response response = ResponseHandler.handle(method, path, http_request.send(method, path, *args)) rescue Exception => e Restfulie::Common::Logger.logger.error(e) raise Error::ServerNotAvailableError.new(self, Response.new(method, path, 503, nil, {}), e ) end case response.code when 100..299 [[@host, path], http_request, response] when 300..399 raise Error::Redirection.new(self, response) when 400 raise Error::BadRequest.new(self, response) when 401 raise Error::Unauthorized.new(self, response) when 403 raise Error::Forbidden.new(self, response) when 404 raise Error::NotFound.new(self, response) when 405 raise Error::MethodNotAllowed.new(self, response) when 407 raise Error::ProxyAuthenticationRequired.new(self, response) when 409 raise Error::Conflict.new(self, response) when 410 raise Error::Gone.new(self, response) when 412 raise Error::PreconditionFailed.new(self, response) when 402, 406, 408, 411, 413..499 raise Error::ClientError.new(self, response) when 501 raise Error::NotImplemented.new(self, response) when 500, 502..599 raise Error::ServerError.new(self, response) else raise Error::UnknownError.new(self, response) end end private def get_connection_provider @connection ||= ::Net::HTTP.new(@host.host, @host.port) end protected def request_to_s(method, path, *args) result = ["#{method.to_s.upcase} #{path}"] arguments = args.dup headers = arguments.extract_options! if [:post, :put].include?(method) body = arguments.shift end result << headers.collect { |key, value| "#{key}: #{value}" }.join("\n") (result + [body ? (body.inspect + "\n") : nil]).compact.join("\n") << "\n" end end end end end