Resolve merge conflict
[arvados.git] / spec / google / api_client / media_spec.rb
1 # Copyright 2012 Google Inc.
2 #
3 # Licensed under the Apache License, Version 2.0 (the "License");
4 # you may not use this file except in compliance with the License.
5 # You may obtain a copy of the License at
6 #
7 #      http://www.apache.org/licenses/LICENSE-2.0
8 #
9 # Unless required by applicable law or agreed to in writing, software
10 # distributed under the License is distributed on an "AS IS" BASIS,
11 # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12 # See the License for the specific language governing permissions and
13 # limitations under the License.
14
15 require 'spec_helper'
16
17 require 'google/api_client'
18 require 'google/api_client/version'
19
20 fixtures_path = File.expand_path('../../../fixtures', __FILE__)
21
22 describe Google::APIClient::UploadIO do
23   it 'should reject invalid file paths' do
24     (lambda do
25       media = Google::APIClient::UploadIO.new('doesnotexist', 'text/plain')
26     end).should raise_error
27   end
28   
29   describe 'with a file' do
30     before do
31       @file = File.expand_path('files/sample.txt', fixtures_path)
32       @media = Google::APIClient::UploadIO.new(@file, 'text/plain')
33     end
34     
35     it 'should report the correct file length' do
36       @media.length.should == File.size(@file)
37     end
38     
39     it 'should have a mime type' do
40       @media.content_type.should == 'text/plain'
41     end
42   end
43   
44   describe 'with StringIO' do
45     before do
46       @content = "hello world"
47       @media = Google::APIClient::UploadIO.new(StringIO.new(@content), 'text/plain')
48     end
49
50     it 'should report the correct file length' do
51       @media.length.should == @content.length
52     end
53     
54     it 'should have a mime type' do
55       @media.content_type.should == 'text/plain'
56     end
57   end
58 end
59
60 describe Google::APIClient::ResumableUpload do
61   before do
62     @client = Google::APIClient.new
63     @drive = @client.discovered_api('drive', 'v1')
64     @file = File.expand_path('files/sample.txt', fixtures_path)
65     @media = Google::APIClient::UploadIO.new(@file, 'text/plain')
66     @uploader = Google::APIClient::ResumableUpload.new(
67       mock_result(308),
68       @media,
69       'https://www.googleapis.com/upload/drive/v1/files/12345')
70   end
71   
72   it 'should consider 20x status as complete' do
73     api_client = stub('api', :execute => mock_result(200))
74     @uploader.send_chunk(api_client)
75     @uploader.complete?.should == true
76   end
77
78   it 'should consider 30x status as incomplete' do
79     api_client = stub('api', :execute => mock_result(308))
80     @uploader.send_chunk(api_client)
81     @uploader.complete?.should == false
82     @uploader.expired?.should == false
83   end
84
85   it 'should consider 40x status as fatal' do
86     api_client = stub('api', :execute => mock_result(404))
87     @uploader.send_chunk(api_client)
88     @uploader.expired?.should == true
89   end
90
91   it 'should detect changes to location' do
92     api_client = stub('api', :execute => mock_result(308, 'location' => 'https://www.googleapis.com/upload/drive/v1/files/abcdef'))
93     @uploader.send_chunk(api_client)
94     @uploader.location.should == 'https://www.googleapis.com/upload/drive/v1/files/abcdef'
95   end
96   
97   it 'should resume from the saved range reported by the server' do 
98     api_client = mock('api')
99     api_client.should_receive(:execute).and_return(mock_result(308, 'range' => '0-99'))
100     api_client.should_receive(:execute).with(
101       hash_including(:headers => hash_including(
102         "Content-Range" => "bytes 100-299/#{@media.length}",
103         "Content-Length" => "200"
104       ))).and_return(mock_result(308))
105
106     @uploader.chunk_size = 200
107     @uploader.send_chunk(api_client) # Send bytes 0-199, only 0-99 saved
108     @uploader.send_chunk(api_client) # Send bytes 100-299
109   end
110   
111   it 'should resync the offset after 5xx errors' do
112     api_client = mock('api')
113     api_client.should_receive(:execute).and_return(mock_result(500))
114     api_client.should_receive(:execute).with(
115       hash_including(:headers => hash_including(
116         "Content-Range" => "bytes */#{@media.length}",
117         "Content-Length" => "0"
118       ))).and_return(mock_result(308, 'range' => '0-99'))
119     api_client.should_receive(:execute).with(
120       hash_including(:headers => hash_including(
121         "Content-Range" => "bytes 100-299/#{@media.length}",
122         "Content-Length" => "200"
123       ))).and_return(mock_result(308))
124
125     @uploader.chunk_size = 200
126     @uploader.send_chunk(api_client) # 500, invalidate
127     @uploader.send_chunk(api_client) # Just resyncs, doesn't actually upload
128     @uploader.send_chunk(api_client) # Send next chunk at correct range
129   end
130
131   def mock_result(status, headers = {})
132     reference = Google::APIClient::Reference.new(:api_method => @drive.files.insert)
133     stub('result', :status => status, :headers => headers, :reference => reference)
134   end
135   
136 end