require "spec_helper" describe CFoundry::RestClient do let(:token) { nil } let(:target) { "https://api.cloudfoundry.com" } let(:rest_client) { CFoundry::RestClient.new(target, token) } describe '#request' do let(:path) { "some-path" } let(:url) { "#{target}/#{path}" } let(:verb) { "GET" } let(:options) { {} } def check_request(verb = :get, &block) request_stub = stub_request(verb, url).to_return do |req| block.call(req) {} end subject expect(request_stub).to have_been_requested end subject { rest_client.request(verb, path, options) } describe 'headers' do %w[Authorization X-Request-Id Content-Type].each do |header_name| it "should not include the #{header_name} by default" do check_request do |req| expect(req.headers).not_to have_key(header_name) end end end it "should not provide specific accept mimetypes by default" do check_request do |req| expect(req.headers["Accept"]).to eql("*/*") end end it 'should always include a content length' do check_request do |req| expect(req.headers["Content-Length"]).to eql(0) end end context "when a payload is passed" do context "when the payload is a string" do let(:options) { { :payload => "some payload"} } it 'includes a content length matching the payload size' do check_request do |req| expect(req.headers["Content-Length"]).to eql("some payload".length) end end end context "when the payload is a hash and the content-type is JSON" do let(:options) { { :payload => { "key" => "value" }, :content => :json } } it 'includes a content length matching the JSON encoded length' do check_request do |req| expect(req.headers["Content-Length"]).to eql('{"key":"value"}'.length) end end end context "when the payload is a hash (i.e. multipart upload)" do let(:verb) { "PUT" } let(:options) { {:payload => {"key" => "value"}} } it 'includes a nonzero content length' do check_request(:put) do |req| expect(req.headers["Content-Length"].to_i).to be > 0 end end end end context "when params are passed" do context "when params is an empty hash" do let(:options) { {:params => {}} } it "does not add a query string delimiter (the question mark)" do request_stub = stub_request(:get, "https://api.cloudfoundry.com/some-path") subject expect(request_stub).to have_been_requested end end context "when params has values" do let(:options) { {:params => {"key" => "value"}} } it "appends a query string and delimiter" do request_stub = stub_request(:get, "https://api.cloudfoundry.com/some-path?key=value") subject expect(request_stub).to have_been_requested end end end context 'and the token is set' do let(:token_header) { "bearer something" } let(:token) { CFoundry::AuthToken.new(token_header) } it 'should include Authorization in the header' do check_request do |req| expect(req.headers["Authorization"]).to eq(token_header) end end end context 'and the request_id is set' do before { rest_client.request_id = "SomeRequestId" } it 'should include X-Request-Id in the header' do check_request do |req| expect(req.headers["X-Request-Id"]).to eq "SomeRequestId" end end end context 'and the content is passed in' do let(:options) { {:content => "text/xml"} } it 'should include Content-Type in the header' do check_request do |req| expect(req.headers["Content-Type"]).to eq "text/xml" end end end context 'and custom headers are passed in' do let(:options) { {:headers => {"X-Foo" => "Bar"}} } it 'should include the custom header in the header' do check_request do |req| expect(req.headers["X-Foo"]).to eq "Bar" end end context 'and it overrides an existing one' do let(:options) { {:content => "text/xml", :headers => {"Content-Type" => "text/html"}} } it 'uses the custom header' do check_request do |req| expect(req.headers["Content-Type"]).to eq "text/html" end end end end end describe 'errors' do context "when the target refuses the connection" do let(:target) { "http://localhost:2358974958" } it "raises CFoundry::TargetRefused" do stub_request(:get, url).to_raise(Errno::ECONNREFUSED) expect { subject }.to raise_error(CFoundry::TargetRefused) end end context "when the target is not a HTTP(S) URI" do let(:target) { "ftp://foo-bar.com" } it "raises CFoundry::InvalidTarget" do expect { subject }.to raise_error(CFoundry::InvalidTarget) end end context "when the target URI is invalid" do let(:target) { "@#*#^! rubby" } it "raises CFoundry::InvalidTarget" do expect { subject }.to raise_error(CFoundry::InvalidTarget) end end end describe "the return value" do before do stub_request(:get, url).to_return({ :status => 201, :headers => {"Content-Type" => "application/json"}, :body => '{ "foo": 1 }' }) end it "returns a request hash and a response hash" do expect(subject).to be_an(Array) expect(subject.length).to eq(2) end describe "the returned request hash" do it "returns a hash of :headers, :url, :body and :verb" do expect(subject[0]).to eq({ :url => url, :method => "GET", :headers => {"Content-Length" => 0}, :body => nil }) end end describe "the returned response hash" do it "returns a hash of :headers, :status, :body" do expect(subject[1]).to eq({ :status => "201", :headers => {"content-type" => "application/json"}, :body => '{ "foo": 1 }' }) end end end describe "when the path starts with a slash" do let(:path) { "/some-path/some-segment" } it "doesn't add a double slash" do stub = stub_request(:get, "https://api.cloudfoundry.com/some-path/some-segment") subject expect(stub).to have_been_requested end end describe "when the path does not start with a slash" do let(:path) { "some-path/some-segment" } it "doesn't add a double slash" do stub = stub_request(:get, "https://api.cloudfoundry.com/some-path/some-segment") subject expect(stub).to have_been_requested end end describe "when the path is a full url" do let(:path) { "http://example.com" } it "requests the given url" do stub = stub_request(:get, "http://example.com") subject expect(stub).to have_been_requested end end describe "when the path is malformed" do let(:path) { "#%&*$(#%&$%)" } it "blows up" do expect { subject }.to raise_error(URI::InvalidURIError) end end describe 'trace' do before do rest_client.trace = true stub_request(:get, url).to_return(:status => 200, :headers => {"content-type" => "application/json"}, :body => '{"some": "json"}') end it "prints the request and the response" do rest_client.should_receive(:print_request).with({:headers => {"Content-Length" => 0}, :url => "https://api.cloudfoundry.com/some-path", :method => "GET", :body => nil}) rest_client.should_receive(:print_response).with({:status => "200", :headers => {"content-type" => "application/json"}, :body => '{"some": "json"}'}) subject end end describe "following redirects" do before do stub_request(:post, "https://api.cloudfoundry.com/apps").to_return( :status => 301, :headers => {"location" => "https://api.cloudfoundry.com/apps/some-guid"} ) stub_request(:get, "https://api.cloudfoundry.com/apps/some-guid").to_return( :status => 200, :body => '{"some": "json"}' ) end it "follows redirects correctly, returning the response to the 2nd redirect" do request, response = rest_client.request("POST", "apps") expect(response).to eql( :status => "200", :headers => {}, :body => '{"some": "json"}' ) end end end describe CFoundry::RestClient::HTTPFactory do describe ".create" do let(:http_proxy) { '' } let(:https_proxy) { '' } let(:target_uri) { "http://cloudfoundry.com" } subject { CFoundry::RestClient::HTTPFactory.create(URI.parse(target_uri), http_proxy, https_proxy) } context "when no proxy URI is set" do it "should return an instance of the plain Net:HTTP class" do expect(subject).to be_instance_of(Net::HTTP) expect(subject.use_ssl?).to be_false expect(subject.proxy?).to_not be_true end end context "when the target is an https URI" do let(:target_uri) { "https://cloudfoundry.com" } it "should return an instance of the plain Net:HTTP class with use_ssl" do expect(subject).to be_instance_of(Net::HTTP) expect(subject.use_ssl?).to be_true expect(subject.proxy?).to_not be_true end end context "when a http proxy URI without user/password is set " do let(:http_proxy) { "http://exapmle.com:8080" } it "should return an instance of the proxy class" do expect(subject.proxy?).to be_true expect(subject.proxy_address).to eql("exapmle.com") expect(subject.proxy_port).to eql(8080) end end context "when a http proxy URI with user/password is set " do let(:http_proxy) { "http://user:pass@exapmle.com:8080" } it "should return an instance of the proxy class" do expect(subject.proxy?).to be_true expect(subject.proxy_user).to eql("user") expect(subject.proxy_pass).to eql("pass") end end context "when a https proxy URI is set and the target is an https URI" do let(:target_uri) { "https://cloudfoundry.com" } let(:https_proxy) { "http://exapmle.com:8080" } it "should return an instance of the proxy class" do expect(subject.proxy?).to be_true end end context "when a https proxy URI is set and the target is an http URI" do let(:target_uri) { "http://cloudfoundry.com" } let(:https_proxy) { "http://exapmle.com:8080" } it "should return an instance of the plain Net:HTTP class" do expect(subject.proxy?).to be_nil end end context "when a http proxy URI is set and the target is an https URI" do let(:target_uri) { "https://cloudfoundry.com" } let(:http_proxy) { "http://exapmle.com:8080" } it "should return an instance of the plain Net:HTTP class" do expect(subject.proxy?).to be_nil end end context "when an invalid proxy URI is set" do let(:http_proxy) { "invalid URI" } it "should raise an error" do expect { subject }.to raise_error(URI::InvalidURIError) end end end end describe "#target=" do let(:token) { "some-token" } it "invalidates the token if the target has changed" do rest_client.target = "http://some-target.com" expect(rest_client.token).to be_nil end it "does not invalidate the token if the target has not changed" do rest_client.target = target expect(rest_client.token).to eq token end end end