220 lines
		
	
	
		
			6.0 KiB
		
	
	
	
		
			Ruby
		
	
	
	
	
	
			
		
		
	
	
			220 lines
		
	
	
		
			6.0 KiB
		
	
	
	
		
			Ruby
		
	
	
	
	
	
| # frozen_string_literal: true
 | |
| 
 | |
| require 'rails_helper'
 | |
| 
 | |
| RSpec.describe Remotable do
 | |
|   let(:foo_class) do
 | |
|     Class.new do
 | |
|       def initialize
 | |
|         @attrs = {}
 | |
|       end
 | |
| 
 | |
|       def [](arg)
 | |
|         @attrs[arg]
 | |
|       end
 | |
| 
 | |
|       def []=(arg1, arg2)
 | |
|         @attrs[arg1] = arg2
 | |
|       end
 | |
| 
 | |
|       def hoge=(arg); end
 | |
| 
 | |
|       def hoge_file_name; end
 | |
| 
 | |
|       def hoge_file_name=(arg); end
 | |
| 
 | |
|       def has_attribute?(arg); end
 | |
| 
 | |
|       def self.attachment_definitions
 | |
|         { hoge: nil }
 | |
|       end
 | |
|     end
 | |
|   end
 | |
| 
 | |
|   let(:attribute_name) { :"#{hoge}_remote_url" }
 | |
|   let(:code)           { 200 }
 | |
|   let(:file)           { 'filename="foo.txt"' }
 | |
|   let(:foo)            { foo_class.new }
 | |
|   let(:headers)        { { 'content-disposition' => file } }
 | |
|   let(:hoge)           { :hoge }
 | |
|   let(:url)            { 'https://google.com' }
 | |
| 
 | |
|   before do
 | |
|     foo_class.include described_class
 | |
|     foo_class.remotable_attachment :hoge, 1.kilobyte
 | |
|   end
 | |
| 
 | |
|   it 'defines a method #hoge_remote_url=' do
 | |
|     expect(foo).to respond_to(:hoge_remote_url=)
 | |
|   end
 | |
| 
 | |
|   it 'defines a method #reset_hoge!' do
 | |
|     expect(foo).to respond_to(:reset_hoge!)
 | |
|   end
 | |
| 
 | |
|   it 'defines a method #download_hoge!' do
 | |
|     expect(foo).to respond_to(:download_hoge!)
 | |
|   end
 | |
| 
 | |
|   describe '#hoge_remote_url=' do
 | |
|     before do
 | |
|       stub_request(:get, url).to_return(status: code, headers: headers)
 | |
|     end
 | |
| 
 | |
|     it 'always returns its argument' do
 | |
|       [nil, '', [], {}].each do |arg|
 | |
|         expect(foo.hoge_remote_url = arg).to be arg
 | |
|       end
 | |
|     end
 | |
| 
 | |
|     context 'with an invalid URL' do
 | |
|       before do
 | |
|         parsed = instance_double(Addressable::URI)
 | |
|         allow(parsed).to receive(:normalize).with(no_args).and_raise(Addressable::URI::InvalidURIError)
 | |
|         allow(Addressable::URI).to receive(:parse).with(url).and_return(parsed)
 | |
|       end
 | |
| 
 | |
|       it 'makes no request' do
 | |
|         foo.hoge_remote_url = url
 | |
|         expect(a_request(:get, url)).to_not have_been_made
 | |
|       end
 | |
|     end
 | |
| 
 | |
|     context 'with scheme that is neither http nor https' do
 | |
|       let(:url) { 'ftp://google.com' }
 | |
| 
 | |
|       it 'makes no request' do
 | |
|         foo.hoge_remote_url = url
 | |
|         expect(a_request(:get, url)).to_not have_been_made
 | |
|       end
 | |
|     end
 | |
| 
 | |
|     context 'with relative URL' do
 | |
|       let(:url) { 'https:///path' }
 | |
| 
 | |
|       it 'makes no request' do
 | |
|         foo.hoge_remote_url = url
 | |
|         expect(a_request(:get, url)).to_not have_been_made
 | |
|       end
 | |
|     end
 | |
| 
 | |
|     context 'when URL has not changed' do
 | |
|       it 'makes no request if file is already saved' do
 | |
|         allow(foo).to receive(:[]).with(attribute_name).and_return(url)
 | |
|         allow(foo).to receive(:hoge_file_name).and_return('foo.jpg')
 | |
| 
 | |
|         foo.hoge_remote_url = url
 | |
|         expect(a_request(:get, url)).to_not have_been_made
 | |
|       end
 | |
| 
 | |
|       it 'makes request if file is not already saved' do
 | |
|         allow(foo).to receive(:[]).with(attribute_name).and_return(url)
 | |
|         allow(foo).to receive(:hoge_file_name).and_return(nil)
 | |
| 
 | |
|         foo.hoge_remote_url = url
 | |
|         expect(a_request(:get, url)).to have_been_made
 | |
|       end
 | |
|     end
 | |
| 
 | |
|     context 'when instance has no attribute for URL' do
 | |
|       before do
 | |
|         allow(foo).to receive(:has_attribute?).with(attribute_name).and_return(false)
 | |
|       end
 | |
| 
 | |
|       it 'does not try to write attribute' do
 | |
|         allow(foo).to receive('[]=').with(attribute_name, url)
 | |
| 
 | |
|         foo.hoge_remote_url = url
 | |
| 
 | |
|         expect(foo).to_not have_received('[]=').with(attribute_name, url)
 | |
|       end
 | |
|     end
 | |
| 
 | |
|     context 'when instance has an attribute for URL' do
 | |
|       before do
 | |
|         allow(foo).to receive(:has_attribute?).with(attribute_name).and_return(true)
 | |
|       end
 | |
| 
 | |
|       it 'does not try to write attribute' do
 | |
|         allow(foo).to receive('[]=').with(attribute_name, url)
 | |
| 
 | |
|         foo.hoge_remote_url = url
 | |
| 
 | |
|         expect(foo).to have_received('[]=').with(attribute_name, url)
 | |
|       end
 | |
|     end
 | |
| 
 | |
|     context 'with a valid URL' do
 | |
|       it 'makes a request' do
 | |
|         foo.hoge_remote_url = url
 | |
|         expect(a_request(:get, url)).to have_been_made
 | |
|       end
 | |
| 
 | |
|       context 'when the response is not successful' do
 | |
|         let(:code) { 500 }
 | |
| 
 | |
|         it 'does not assign file' do
 | |
|           allow(foo).to receive(:public_send)
 | |
|           allow(foo).to receive(:public_send)
 | |
| 
 | |
|           foo.hoge_remote_url = url
 | |
| 
 | |
|           expect(foo).to_not have_received(:public_send).with("#{hoge}=", any_args)
 | |
|           expect(foo).to_not have_received(:public_send).with("#{hoge}_file_name=", any_args)
 | |
|         end
 | |
|       end
 | |
| 
 | |
|       context 'when the response is successful' do
 | |
|         let(:code) { 200 }
 | |
| 
 | |
|         context 'when contains Content-Disposition header' do
 | |
|           let(:file)      { 'filename="foo.txt"' }
 | |
|           let(:headers)   { { 'content-disposition' => file } }
 | |
| 
 | |
|           it 'assigns file' do
 | |
|             response_with_limit = ResponseWithLimit.new(nil, 0)
 | |
| 
 | |
|             allow(ResponseWithLimit).to receive(:new).with(anything, anything).and_return(response_with_limit)
 | |
| 
 | |
|             allow(foo).to receive(:public_send)
 | |
|             foo.hoge_remote_url = url
 | |
|             expect(foo).to have_received(:public_send).with("download_#{hoge}!", url)
 | |
| 
 | |
|             allow(foo).to receive(:public_send)
 | |
|             foo.download_hoge!(url)
 | |
|             expect(foo).to have_received(:public_send).with("#{hoge}=", response_with_limit)
 | |
|           end
 | |
|         end
 | |
|       end
 | |
| 
 | |
|       context 'when an error is raised during the request' do
 | |
|         before do
 | |
|           stub_request(:get, url).to_raise(error_class)
 | |
|         end
 | |
| 
 | |
|         error_classes = [
 | |
|           HTTP::TimeoutError,
 | |
|           HTTP::ConnectionError,
 | |
|           OpenSSL::SSL::SSLError,
 | |
|           Paperclip::Errors::NotIdentifiedByImageMagickError,
 | |
|           Addressable::URI::InvalidURIError,
 | |
|         ]
 | |
| 
 | |
|         error_classes.each do |error_class|
 | |
|           let(:error_class) { error_class }
 | |
| 
 | |
|           it 'calls Rails.logger.debug' do
 | |
|             allow(Rails.logger).to receive(:debug)
 | |
| 
 | |
|             foo.hoge_remote_url = url
 | |
| 
 | |
|             expect(Rails.logger).to have_received(:debug) do |&block|
 | |
|               expect(block.call).to match(/^Error fetching remote #{hoge}: /)
 | |
|             end
 | |
|           end
 | |
|         end
 | |
|       end
 | |
|     end
 | |
|   end
 | |
| end
 |