# rubocop:disable Metrics/ClassLength require 'sitehub/route_builder' class SiteHub describe RouteBuilder do include_context :middleware_test describe '::from_hash' do include_context :sitehub_json subject do described_class.from_hash(proxy_1, :expected).routes[route_1[:label]] end context 'splits' do context 'sitehub_cookie_name' do it 'sets it' do expect(subject.sitehub_cookie_name).to eq(:expected) end end context 'sitehub_cookie_path' do it 'sets it' do expect(subject.sitehub_cookie_path).to eq(proxy_1[:sitehub_cookie_path]) end end pending 'returns core with splits' end context 'routes' do context 'sitehub_cookie_name' do pending 'sets it' end context 'sitehub_cookie_path' do pending 'sets it' end pending 'returns core with routes' end context 'default' do it 'sets the default' end end subject do described_class.new(sitehub_cookie_name: :cookie_name, mapped_path: '/path') end describe '#routes' do it 'returns RouteCollection by default' do expect(subject.routes).to be_a(Collection::RouteCollection) end it 'returns the same intance everytime' do collection = Collection::SplitRouteCollection.new subject.routes(collection) expect(subject.routes).to be(collection) end context 'endpoints already set' do context 'different object supplied' do it 'raises an error' do subject.routes(Collection::SplitRouteCollection.new) expect { subject.routes(Collection::RouteCollection.new) } .to raise_error(RouteBuilder::InvalidDefinitionException) end end end end it 'supports middleware' do expect(described_class).to include(Middleware) end describe '#initialize' do let(:named_parameters) do { sitehub_cookie_name: :name, mapped_path: '/path' } end context 'with a block' do it 'evaluates the block in the context of the instance' do self_inside_block = nil instance = described_class.new(named_parameters) do self_inside_block = self default(url: :url) end expect(self_inside_block).to eq(instance) end end context 'id' do context 'id supplied' do it 'sets the id using it' do subject = described_class.new(named_parameters.merge(id: :custom_id)) expect(subject.id).to eq(:custom_id) end end end end describe 'valid?' do context 'default defined' do context 'splits add up to 100' do it 'returns true' do subject.split(percentage: 100, label: :old, url: :url) expect(subject.valid?).to eq(true) end end context 'splits do not add up to 100' do it 'returns true' do subject.split(percentage: 10, label: :old, url: :url) subject.default(url: :url) expect(subject.valid?).to eq(true) end end end context 'default not defined' do context 'splits add up to 100' do it 'returns true' do subject.default(url: :url) expect(subject.valid?).to eq(true) end end context 'splits do not add up to 100' do it 'returns false' do subject.split(percentage: 10, label: :old, url: :url) subject.split(percentage: 10, label: :new, url: :url) expect(subject).to_not be_valid end end end end describe '#split' do it 'setups up a splits collection' do subject.split percentage: 10, url: :url, label: :label expect(subject.routes).to be_a(Collection::SplitRouteCollection) end end describe '#route' do it 'sets up the routes collection' do subject.route url: :url, label: :current expect(subject.routes).to be_a(Collection::RouteCollection) end end describe '#add_endpoint' do it 'stores the route against the given label' do subject.add_route url: :url, label: :current proxy = ForwardProxy.new(mapped_url: :url, mapped_path: subject.mapped_path) expected_route = Route.new(proxy, id: :current, sitehub_cookie_name: :cookie_name, sitehub_cookie_path: nil) expect(subject.routes[:current]).to eq(expected_route) end it 'accepts a rule' do endpoint = subject.add_route url: :url, label: :current, rule: :rule expect(endpoint.rule).to eq(:rule) end it 'accepts a percentage' do subject.routes(Collection::SplitRouteCollection.new) endpoint = subject.add_route url: :url, label: :current, percentage: 50 expect(endpoint.upper).to eq(50) end context 'block supplied' do let(:block) do proc do route url: :url, label: :label2, rule: :rule do route url: :url, label: :label3 end end end it 'stores the nested route_builder against the label' do rule = proc { true } subject.add_route(rule: rule, label: :label1, &block) subject.use middleware expected_endpoints = RouteBuilder.new(rule: rule, id: :label1, sitehub_cookie_name: :cookie_name, mapped_path: '/path', &block).build expect(subject.routes[:label1]).to eq(expected_endpoints) subject.build end describe '#errors and warnings' do context 'precentage and rule not supplied' do it 'raise an error' do expected_message = described_class::RULE_NOT_SPECIFIED_MSG expect { subject.add_route(label: :label) {} } .to raise_exception described_class::InvalidDefinitionException, expected_message end end context 'url' do it 'gives a warning to say that the url will not be used' do expect(subject).to receive(:warn).with(described_class::IGNORING_URL_MSG) subject.add_route(rule: :rule, url: :url, label: :label, &block) end end end it 'stores a proxy builder' do rule = proc { true } subject.add_route(rule: rule, label: :label, &block) expected_endpoints = described_class.new(id: :label, sitehub_cookie_name: :cookie_name, rule: rule, mapped_path: subject.mapped_path, &block).tap do |builder| builder.sitehub_cookie_name subject.sitehub_cookie_name end.build expect(subject.routes.values).to eq([expected_endpoints]) end context 'invalid definitions inside block' do it 'raises an error' do rule = proc { true } expect do subject.add_route rule: rule, label: :label do split percentage: 20, url: :url, label: :label1 end end.to raise_exception described_class::InvalidDefinitionException end end end end describe '#build' do let(:rule) { proc {} } context 'middleware not specified' do it 'leaves it the proxies alone' do subject.route url: :url, label: :current expect(subject.routes[:current]).to be_using_rack_stack(ForwardProxy) subject.build expect(subject.routes[:current]).to be_using_rack_stack(ForwardProxy) end end context 'middleware specified' do before do subject.use middleware end it 'wraps the forward proxies in the middleware' do subject.route url: :url, label: :current subject.build expect(subject.routes[:current]).to be_using_rack_stack(middleware, ForwardProxy) end it 'wraps the default in the middleware' do subject.default url: :url subject.build expect(subject.default_route).to be_using_rack_stack(middleware, ForwardProxy) end context 'nested routes' do pending 'what should it do?' end end end describe '#resolve' do context 'id not supplied' do context 'routes defined' do it 'returns that route' do subject.route url: :url, label: :current expect(subject.resolve(env: {})).to eq(subject.routes.values.first) end it 'passes the env to the when resolving the correct route' do expect_any_instance_of(subject.routes.class).to receive(:resolve).with(env: :env).and_call_original subject.resolve(env: :env) end end context 'splits not defined' do it 'returns the default' do subject.default url: :url expect(subject.resolve(env: {})).to eq(subject.default_route) end end end context 'id supplied' do context 'nested routes' do let!(:expected) do result = nil subject.split percentage: 0, label: :experiment1 do result = split percentage: 100, url: :url1, label: :new end subject.split percentage: 100, label: :experiment2, url: :url result.value end it 'returns that route' do expect(subject.resolve(id: expected.id, env: {})).to eq(expected) end end context 'id does not exist' do let!(:expected) do subject.default url: :url subject.default_route end it 'returns the default route' do expect(subject.resolve(id: :missing, env: {})).to eq(expected) end end context 'non nested route' do let!(:expected) do subject.split(percentage: 100, label: :experiment1, url: :url).value end it 'returns that route' do expect(subject.resolve(id: expected.id, env: {})).to eq(expected) end end end end context '#endpoints' do context 'called with a collection' do it 'sets endpoints to be that collection' do subject.routes(:collection) expect(subject.routes).to eq(:collection) end end context 'already set with a different collection' do it 'raise an error' do subject.routes(:collection1) expect { subject.routes(:collection2) }.to raise_exception described_class::InvalidDefinitionException end end end describe '#forward_proxy' do subject do described_class.new(sitehub_cookie_name: :cookie_name, mapped_path: '/path') end it 'sets the sitehub_cookie_path' do subject.sitehub_cookie_path :cookie_path proxy = subject.forward_proxy(label: :label, url: :url) expect(proxy.sitehub_cookie_path).to eq(:cookie_path) end it 'sets the sitehub_cookie_name' do subject.sitehub_cookie_name :expected_cookie_name proxy = subject.forward_proxy(label: :label, url: :url) expect(proxy.sitehub_cookie_name).to eq(:expected_cookie_name) end end end end