cloudfoundry/cloud_controller_ng

View on GitHub
spec/request/lifecycle/service_route_bindings_synoptic_spec.rb

Summary

Maintainability
A
1 hr
Test Coverage
require 'spec_helper'
require 'cloud_controller'

# This lifecycle test aims to use different v3 service endpoints together
RSpec.describe 'V3 service route bindings synoptic' do
  before do
    stub_request(:get, 'http://example.org/amazing-service-broker/v2/catalog').
      with(basic_auth: %w[admin password]).
      to_return(status: 200, body: catalog, headers: {})

    stub_request(:put, %r{\Ahttp://example.org/amazing-service-broker/v2/service_instances/.+\z}).
      with(basic_auth: %w[admin password]).
      to_return(status: 201, body: {}.to_json, headers: {})

    stub_request(:get, 'https://main.default.svc.cluster-domain.example/apis/networking.cloudfoundry.org/v1alpha1').
      with(basic_auth: %w[admin password]).
      to_return(status: 200, body: '', headers: {})

    stub_request(:get, %r{\Ahttp://example.org/amazing-service-broker/v2/service_instances/.+/service_bindings/.+\z}).
      with(basic_auth: %w[admin password]).
      to_return(status: 200, body: { parameters: { key1: 'value1', key2: 'value2' } }.to_json, headers: {})

    stub_request(:delete, %r{\Ahttp://example.org/amazing-service-broker/v2/service_instances/.+/service_bindings/.+\z}).
      with(basic_auth: %w[admin password]).
      to_return(status: 410, body: '', headers: {})
    VCAP::CloudController::Config.config.set(:kubernetes, nil)

    stub_request(:post, 'https://uaa.service.cf.internal/oauth/token').
      with(body: 'grant_type=client_credentials').
      to_return(status: 200, body: "{ token_type: 'Bearer', access_token: 'dXNlcm5hbWVfbG9va3VwX2NsaWVudF9uYW1lOnVzZXJuYW1lX2xvb2t1cF9zZWNyZXQ=' }", headers: {})
  end

  it 'works end to end' do
    org_guid = create_org
    space_guid = create_space org_guid

    create_service_broker

    plan_guid = get_route_service_plan using: admin_headers
    make_plan_visible plan_guid

    service_instance_guid = wait_for_service_instance_to_be_created(space_guid, plan_guid)

    domain_guid = create_domain
    route_guid = create_route(domain_guid, space_guid)

    create_request = create_route_binding_request(route_guid, service_instance_guid)
    post '/v3/service_route_bindings', create_request.to_json, admin_headers
    expect(last_response).to have_status_code(202)
    execute_all_jobs(expected_successes: 1, expected_failures: 0)

    get '/v3/service_route_bindings', nil, admin_headers
    route_binding_guid = parsed_response['resources'][0]['guid']
    get "/v3/service_route_bindings/#{route_binding_guid}/parameters", nil, admin_headers
    expect(parsed_response).to match_array(create_request[:parameters].with_indifferent_access)

    updates_metadata(route_binding_guid, create_request)

    delete "/v3/service_route_bindings/#{route_binding_guid}", nil, admin_headers
    expect(last_response).to have_status_code(202)
    execute_all_jobs(expected_successes: 1, expected_failures: 0)
    get "/v3/service_route_bindings/#{route_binding_guid}", nil, admin_headers
    expect(last_response).to have_status_code(404)
  end

  def updates_metadata(binding_guid, create_request)
    update_request = {
      metadata: {
        labels: { key: 'value' },
        annotations: { note: 'detailed information' }
      }
    }
    patch "/v3/service_route_bindings/#{binding_guid}", update_request.to_json, admin_headers
    expect(last_response).to have_status_code(200)
    expect(parsed_response['metadata']['annotations']).to match_array(update_request[:metadata][:annotations].merge(create_request[:metadata][:annotations]).stringify_keys)
    expect(parsed_response['metadata']['labels']).to match_array(update_request[:metadata][:labels].merge(create_request[:metadata][:labels]).stringify_keys)
  end

  def get_route_service_plan(using:)
    get_service_plan_with(name: 'route_plan', using: using)
  end

  def get_service_plan_with(name:, using:)
    headers = using
    get "/v3/service_plans?names=#{name}", nil, headers
    expect(last_response).to have_status_code(200)

    parsed_response['resources'][0]['guid']
  end

  def create_route_binding_request(route_guid, service_instance_guid)
    {
      metadata: {
        annotations: {
          foo: 'bar'
        },
        labels: {
          baz: 'qux'
        }
      },
      relationships: {
        route: {
          data: {
            guid: route_guid
          }
        },
        service_instance: {
          data: {
            guid: service_instance_guid
          }
        }
      },
      parameters: {
        key1: 'value1',
        key2: 'value2'
      }
    }
  end

  def create_domain
    post '/v3/domains', { name: 'example.com', internal: false }.to_json, admin_headers
    expect(last_response).to have_status_code(201)
    parsed_response['guid']
  end

  def create_route(domain_guid, space_guid)
    route_request = {
      host: 'a-hostname',
      path: '/some_path',
      relationships: {
        domain: {
          data: { guid: domain_guid }
        },
        space: {
          data: { guid: space_guid }
        }
      }
    }
    post '/v3/routes', route_request.to_json, admin_headers
    expect(last_response).to have_status_code(201)
    parsed_response['guid']
  end
end