/** * @copyright 2014 Vincent Petry * * @author Christoph Wurst * @author Jan-Christoph Borchardt * @author Julius Härtl * @author Morris Jobke * @author Tomasz Grobelny * @author Vincent Petry * * @license AGPL-3.0-or-later * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU Affero General Public License as * published by the Free Software Foundation, either version 3 of the * License, or (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU Affero General Public License for more details. * * You should have received a copy of the GNU Affero General Public License * along with this program. If not, see . * */ describe('OC.Upload tests', function() { var $dummyUploader; var testFile; var uploader; var failStub; var progressBarStub; beforeEach(function() { testFile = { name: 'test.txt', size: 5000, // 5 KB type: 'text/plain', lastModifiedDate: new Date() }; // need a dummy button because file-upload checks on it $('#testArea').append( '' + '' + // 50 MB // TODO: handlebars! '
' + 'New' + '
    ' + '
  • Text file

  • ' + '
' + '
' ); $dummyUploader = $('#file_upload_start'); progressBarStub = {on: function(){}}; uploader = new OC.Uploader($dummyUploader, {progressBar: progressBarStub}); failStub = sinon.stub(); uploader.on('fail', failStub); }); afterEach(function() { $dummyUploader = undefined; failStub = undefined; }); /** * Add file for upload * @param {Array.} files array of file data to simulate upload * @return {Array.} array of uploadinfo or null if add() returned false */ function addFiles(uploader, files) { return _.map(files, function(file) { var jqXHR = {status: 200}; var uploadInfo = { originalFiles: files, files: [file], jqXHR: jqXHR, response: sinon.stub().returns(jqXHR), targetDir: "/", submit: sinon.stub(), abort: sinon.stub() }; if (uploader.fileUploadParam.add.call( $dummyUploader[0], {}, uploadInfo )) { return uploadInfo; } return null; }); } describe('Adding files for upload', function() { it('adds file when size is below limits', function(done) { var result = addFiles(uploader, [testFile]); expect(result[0]).not.toEqual(null); result[0].submit.callsFake(function(){ expect(result[0].submit.calledOnce).toEqual(true); done(); }); }); it('adds file when free space is unknown', function(done) { var result; $('#free_space').val(-2); result = addFiles(uploader, [testFile]); expect(result[0]).not.toEqual(null); result[0].submit.callsFake(function(){ expect(result[0].submit.calledOnce).toEqual(true); expect(failStub.notCalled).toEqual(true); done(); }); }); it('does not add file if it exceeds free space', function(done) { var result; $('#free_space').val(1000); failStub.callsFake(function(){ expect(failStub.calledOnce).toEqual(true); expect(failStub.getCall(0).args[1].textStatus).toEqual('notenoughspace'); expect(failStub.getCall(0).args[1].errorThrown).toEqual( 'Not enough free space, you are uploading 5 KB but only 1000 B is left' ); setTimeout(done, 0); }); result = addFiles(uploader, [testFile]); expect(result[0]).toEqual(null); }); }); describe('Upload conflicts', function() { var conflictDialogStub; var clock; var fileList; beforeEach(function() { $('#testArea').append( '
' + '' + '' + '' + '' + '' + '' + '' + '' + '
' + '
' ); fileList = new OCA.Files.FileList($('#tableContainer')); fileList.add({name: 'conflict.txt', mimetype: 'text/plain'}); fileList.add({name: 'conflict2.txt', mimetype: 'text/plain'}); conflictDialogStub = sinon.stub(OC.dialogs, 'fileexists'); uploader = new OC.Uploader($dummyUploader, { progressBar: progressBarStub, fileList: fileList }); var deferred = $.Deferred(); conflictDialogStub.returns(deferred.promise()); deferred.resolve(); }); afterEach(function() { if (clock) { clock.restore(); clock = undefined } conflictDialogStub.restore(); fileList.destroy(); }); it('does not show conflict dialog when no client side conflict', function(done) { $('#free_space').val(200000); var counter = 0; var fun = function() { counter++; if(counter != 2) { return; } expect(result[0].submit.calledOnce).toEqual(true); expect(result[1].submit.calledOnce).toEqual(true); setTimeout(done, 0); }; var result = addFiles(uploader, [{name: 'noconflict.txt'}, {name: 'noconflict2.txt'}]); result[0].submit.callsFake(fun); result[1].submit.callsFake(fun); expect(conflictDialogStub.notCalled).toEqual(true); }); it('shows conflict dialog when no client side conflict', function(done) { var counter = 0; conflictDialogStub.callsFake(function(){ counter++; if(counter != 3) { return $.Deferred().resolve().promise(); } setTimeout(function() { expect(conflictDialogStub.callCount).toEqual(3); expect(conflictDialogStub.getCall(1).args[0].getFileName()) .toEqual('conflict.txt'); expect(conflictDialogStub.getCall(1).args[1]) .toEqual({ name: 'conflict.txt', mimetype: 'text/plain', directory: '/' }); expect(conflictDialogStub.getCall(1).args[2]).toEqual({ name: 'conflict.txt' }); // yes, the dialog must be called several times... expect(conflictDialogStub.getCall(2).args[0].getFileName()).toEqual('conflict2.txt'); expect(conflictDialogStub.getCall(2).args[1]) .toEqual({ name: 'conflict2.txt', mimetype: 'text/plain', directory: '/' }); expect(conflictDialogStub.getCall(2).args[2]).toEqual({ name: 'conflict2.txt' }); expect(result[0].submit.calledOnce).toEqual(false); expect(result[1].submit.calledOnce).toEqual(false); expect(result[2].submit.calledOnce).toEqual(true); done(); }, 10); }); var result = addFiles(uploader, [ {name: 'conflict.txt'}, {name: 'conflict2.txt'}, {name: 'noconflict.txt'} ]); }); it('cancels upload when skipping file in conflict mode', function(done) { var fileData = {name: 'conflict.txt'}; var uploadData = addFiles(uploader, [ fileData ]); var upload = new OC.FileUpload(uploader, uploadData[0]); var deleteStub = sinon.stub(upload, 'deleteUpload'); deleteStub.callsFake(function(){ expect(deleteStub.calledOnce).toEqual(true); done(); }); uploader.onSkip(upload); }); it('overwrites file when choosing replace in conflict mode', function(done) { var fileData = {name: 'conflict.txt'}; var uploadData = addFiles(uploader, [ fileData ]); expect(uploadData[0].submit.notCalled).toEqual(true); var upload = new OC.FileUpload(uploader, uploadData[0]); uploadData[0].submit.callsFake(function(){ expect(upload.getConflictMode()).toEqual(OC.FileUpload.CONFLICT_MODE_OVERWRITE); expect(uploadData[0].submit.callCount).toEqual(1); done(); }); uploader.onReplace(upload); }); it('autorenames file when choosing replace in conflict mode', function(done) { var fileData = {name: 'conflict.txt'}; var uploadData = addFiles(uploader, [ fileData ]); expect(uploadData[0].submit.notCalled).toEqual(true); var upload = new OC.FileUpload(uploader, uploadData[0]); var getResponseStatusStub = sinon.stub(upload, 'getResponseStatus'); var counter = 0; uploadData[0].submit.callsFake(function(){ counter++; if(counter===1) { expect(upload.getConflictMode()).toEqual(OC.FileUpload.CONFLICT_MODE_AUTORENAME); expect(upload.getFileName()).toEqual('conflict (2).txt'); expect(uploadData[0].submit.calledOnce).toEqual(true); getResponseStatusStub.returns(412); uploader.fileUploadParam.fail.call($dummyUploader[0], {}, uploadData[0]); } if(counter===2) { _.defer(function() { expect(upload.getFileName()).toEqual('conflict (3).txt'); expect(uploadData[0].submit.calledTwice).toEqual(true); done(); }) } }); uploader.onAutorename(upload); // in case of server-side conflict, tries to rename again }); }); });