require 'spec_helper' describe CopycopterClient do def build_client(config = {}) config[:logger] ||= FakeLogger.new default_config = CopycopterClient::Configuration.new.to_hash CopycopterClient::Client.new(default_config.update(config)) end def add_project api_key = 'xyz123' FakeCopycopterApp.add_project(api_key) end def build_client_with_project(config = {}) project = add_project config[:api_key] = project.api_key build_client(config) end describe 'opening a connection' do let(:config) { CopycopterClient::Configuration.new } let(:http) { Net::HTTP.new(config.host, config.port) } before do Net::HTTP.stubs(:new => http) end it 'should timeout when connecting' do project = add_project client = build_client(:api_key => project.api_key, :http_open_timeout => 4) client.download { |ignore| } http.open_timeout.should == 4 end it 'should timeout when reading' do project = add_project client = build_client(:api_key => project.api_key, :http_read_timeout => 4) client.download { |ignore| } http.read_timeout.should == 4 end it 'uses verified ssl when secure' do project = add_project client = build_client(:api_key => project.api_key, :secure => true) client.download { |ignore| } http.use_ssl?.should == true http.verify_mode.should == OpenSSL::SSL::VERIFY_PEER end it 'does not use ssl when insecure' do project = add_project client = build_client(:api_key => project.api_key, :secure => false) client.download { |ignore| } http.use_ssl?.should == false end it 'wraps HTTP errors with ConnectionError' do errors = [ Timeout::Error.new, Errno::EINVAL.new, Errno::ECONNRESET.new, EOFError.new, Net::HTTPBadResponse.new, Net::HTTPHeaderSyntaxError.new, Net::ProtocolError.new, SocketError.new, OpenSSL::SSL::SSLError.new, Errno::ECONNREFUSED.new ] errors.each do |original_error| http.stubs(:request).raises(original_error) client = build_client_with_project expect { client.download { |ignore| } }. to raise_error(CopycopterClient::ConnectionError) { |error| error.message. should == "#{original_error.class.name}: #{original_error.message}" } end end it 'handles 500 errors from downloads with ConnectionError' do client = build_client(:api_key => 'raise_error') expect { client.download { |ignore| } }. to raise_error(CopycopterClient::ConnectionError) end it 'handles 500 errors from uploads with ConnectionError' do client = build_client(:api_key => 'raise_error') expect { client.upload({}) }.to raise_error(CopycopterClient::ConnectionError) end it 'handles 404 errors from downloads with ConnectionError' do client = build_client(:api_key => 'bogus') expect { client.download { |ignore| } }. to raise_error(CopycopterClient::InvalidApiKey) end it 'handles 404 errors from uploads with ConnectionError' do client = build_client(:api_key => 'bogus') expect { client.upload({}) }.to raise_error(CopycopterClient::InvalidApiKey) end end it 'downloads published blurbs for an existing project' do project = add_project project.update({ 'draft' => { 'key.one' => 'unexpected one', 'key.three' => 'unexpected three' }, 'published' => { 'key.one' => 'expected one', 'key.two' => 'expected two' } }) client = build_client(:api_key => project.api_key, :public => true) blurbs = nil client.download { |yielded| blurbs = yielded } blurbs.should == { 'key.one' => 'expected one', 'key.two' => 'expected two' } end it 'logs that it performed a download' do logger = FakeLogger.new client = build_client_with_project(:logger => logger) client.download { |ignore| } logger.should have_entry(:info, 'Downloaded translations') end it 'downloads draft blurbs for an existing project' do project = add_project project.update({ 'draft' => { 'key.one' => 'expected one', 'key.two' => 'expected two' }, 'published' => { 'key.one' => 'unexpected one', 'key.three' => 'unexpected three' } }) client = build_client(:api_key => project.api_key, :public => false) blurbs = nil client.download { |yielded| blurbs = yielded } blurbs.should == { 'key.one' => 'expected one', 'key.two' => 'expected two' } end it "handles a 304 response when downloading" do project = add_project project.update('draft' => { 'key.one' => "expected one" }) logger = FakeLogger.new client = build_client(:api_key => project.api_key, :public => false, :logger => logger) yields = 0 2.times do client.download { |ignore| yields += 1 } end yields.should == 1 logger.should have_entry(:info, "No new translations") end it "uploads defaults for missing blurbs in an existing project" do project = add_project blurbs = { 'key.one' => 'expected one', 'key.two' => 'expected two' } client = build_client(:api_key => project.api_key, :public => true) client.upload(blurbs) project.reload.draft.should == blurbs end it "logs that it performed an upload" do logger = FakeLogger.new client = build_client_with_project(:logger => logger) client.upload({}) logger.should have_entry(:info, "Uploaded missing translations") end it "deploys from the top-level constant" do client = build_client CopycopterClient.configure do |config| config.client = client end client.stubs(:deploy) CopycopterClient.deploy client.should have_received(:deploy) end it "deploys" do project = add_project project.update({ 'draft' => { 'key.one' => "expected one", 'key.two' => "expected two" }, 'published' => { 'key.one' => "unexpected one", 'key.two' => "unexpected one", } }) logger = FakeLogger.new client = build_client(:api_key => project.api_key, :logger => logger) client.deploy project.reload.published.should == { 'key.one' => "expected one", 'key.two' => "expected two" } logger.should have_entry(:info, "Deployed") end it "handles deploy errors" do expect { build_client.deploy }.to raise_error(CopycopterClient::InvalidApiKey) end end