fileUploadSpec.js 9.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292
  1. /**
  2. * ownCloud
  3. *
  4. * @author Vincent Petry
  5. * @copyright 2014 Vincent Petry <pvince81@owncloud.com>
  6. *
  7. * This library is free software; you can redistribute it and/or
  8. * modify it under the terms of the GNU AFFERO GENERAL PUBLIC LICENSE
  9. * License as published by the Free Software Foundation; either
  10. * version 3 of the License, or any later version.
  11. *
  12. * This library is distributed in the hope that it will be useful,
  13. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  14. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  15. * GNU AFFERO GENERAL PUBLIC LICENSE for more details.
  16. *
  17. * You should have received a copy of the GNU Affero General Public
  18. * License along with this library. If not, see <http://www.gnu.org/licenses/>.
  19. *
  20. */
  21. describe('OC.Upload tests', function() {
  22. var $dummyUploader;
  23. var testFile;
  24. var uploader;
  25. var failStub;
  26. var progressBarStub;
  27. beforeEach(function() {
  28. testFile = {
  29. name: 'test.txt',
  30. size: 5000, // 5 KB
  31. type: 'text/plain',
  32. lastModifiedDate: new Date()
  33. };
  34. // need a dummy button because file-upload checks on it
  35. $('#testArea').append(
  36. '<input type="file" id="file_upload_start" name="files[]" multiple="multiple">' +
  37. '<input type="hidden" id="free_space" name="free_space" value="50000000">' + // 50 MB
  38. // TODO: handlebars!
  39. '<div id="new">' +
  40. '<a>New</a>' +
  41. '<ul>' +
  42. '<li data-type="file" data-newname="New text file.txt"><p>Text file</p></li>' +
  43. '</ul>' +
  44. '</div>'
  45. );
  46. $dummyUploader = $('#file_upload_start');
  47. progressBarStub = {on: function(){}};
  48. uploader = new OC.Uploader($dummyUploader, {progressBar: progressBarStub});
  49. failStub = sinon.stub();
  50. uploader.on('fail', failStub);
  51. });
  52. afterEach(function() {
  53. $dummyUploader = undefined;
  54. failStub = undefined;
  55. });
  56. /**
  57. * Add file for upload
  58. * @param {Array.<File>} files array of file data to simulate upload
  59. * @return {Array.<Object>} array of uploadinfo or null if add() returned false
  60. */
  61. function addFiles(uploader, files) {
  62. return _.map(files, function(file) {
  63. var jqXHR = {status: 200};
  64. var uploadInfo = {
  65. originalFiles: files,
  66. files: [file],
  67. jqXHR: jqXHR,
  68. response: sinon.stub().returns(jqXHR),
  69. submit: sinon.stub(),
  70. abort: sinon.stub()
  71. };
  72. if (uploader.fileUploadParam.add.call(
  73. $dummyUploader[0],
  74. {},
  75. uploadInfo
  76. )) {
  77. return uploadInfo;
  78. }
  79. return null;
  80. });
  81. }
  82. describe('Adding files for upload', function() {
  83. it('adds file when size is below limits', function(done) {
  84. var result = addFiles(uploader, [testFile]);
  85. expect(result[0]).not.toEqual(null);
  86. result[0].submit.callsFake(function(){
  87. expect(result[0].submit.calledOnce).toEqual(true);
  88. done();
  89. });
  90. });
  91. it('adds file when free space is unknown', function(done) {
  92. var result;
  93. $('#free_space').val(-2);
  94. result = addFiles(uploader, [testFile]);
  95. expect(result[0]).not.toEqual(null);
  96. result[0].submit.callsFake(function(){
  97. expect(result[0].submit.calledOnce).toEqual(true);
  98. expect(failStub.notCalled).toEqual(true);
  99. done();
  100. });
  101. });
  102. it('does not add file if it exceeds free space', function(done) {
  103. var result;
  104. $('#free_space').val(1000);
  105. failStub.callsFake(function(){
  106. expect(failStub.calledOnce).toEqual(true);
  107. expect(failStub.getCall(0).args[1].textStatus).toEqual('notenoughspace');
  108. expect(failStub.getCall(0).args[1].errorThrown).toEqual(
  109. 'Not enough free space, you are uploading 5 KB but only 1000 B is left'
  110. );
  111. setTimeout(done, 0);
  112. });
  113. result = addFiles(uploader, [testFile]);
  114. expect(result[0]).toEqual(null);
  115. });
  116. });
  117. describe('Upload conflicts', function() {
  118. var conflictDialogStub;
  119. var fileList;
  120. beforeEach(function() {
  121. $('#testArea').append(
  122. '<div id="tableContainer">' +
  123. '<table id="filestable" class="list-container view-grid">' +
  124. '<thead><tr>' +
  125. '<th id="headerName" class="hidden column-name">' +
  126. '<input type="checkbox" id="select_all_files" class="select-all">' +
  127. '<a class="name columntitle" data-sort="name"><span>Name</span><span class="sort-indicator"></span></a>' +
  128. '<span id="selectedActionsList" class="selectedActions hidden">' +
  129. '<a href class="download"><img src="actions/download.svg">Download</a>' +
  130. '<a href class="delete-selected">Delete</a></span>' +
  131. '</th>' +
  132. '<th class="hidden column-size"><a class="columntitle" data-sort="size"><span class="sort-indicator"></span></a></th>' +
  133. '<th class="hidden column-mtime"><a class="columntitle" data-sort="mtime"><span class="sort-indicator"></span></a></th>' +
  134. '</tr></thead>' +
  135. '<tbody id="fileList"></tbody>' +
  136. '<tfoot></tfoot>' +
  137. '</table>' +
  138. '</div>'
  139. );
  140. fileList = new OCA.Files.FileList($('#tableContainer'));
  141. fileList.add({name: 'conflict.txt', mimetype: 'text/plain'});
  142. fileList.add({name: 'conflict2.txt', mimetype: 'text/plain'});
  143. conflictDialogStub = sinon.stub(OC.dialogs, 'fileexists');
  144. uploader = new OC.Uploader($dummyUploader, {
  145. progressBar: progressBarStub,
  146. fileList: fileList
  147. });
  148. var deferred = $.Deferred();
  149. conflictDialogStub.returns(deferred.promise());
  150. deferred.resolve();
  151. });
  152. afterEach(function() {
  153. conflictDialogStub.restore();
  154. fileList.destroy();
  155. });
  156. it('does not show conflict dialog when no client side conflict', function(done) {
  157. $('#free_space').val(200000);
  158. var counter = 0;
  159. var fun = function() {
  160. counter++;
  161. if(counter != 2) {
  162. return;
  163. }
  164. expect(result[0].submit.calledOnce).toEqual(true);
  165. expect(result[1].submit.calledOnce).toEqual(true);
  166. setTimeout(done, 0);
  167. };
  168. var result = addFiles(uploader, [{name: 'noconflict.txt'}, {name: 'noconflict2.txt'}]);
  169. result[0].submit.callsFake(fun);
  170. result[1].submit.callsFake(fun);
  171. expect(conflictDialogStub.notCalled).toEqual(true);
  172. });
  173. it('shows conflict dialog when no client side conflict', function(done) {
  174. var counter = 0;
  175. conflictDialogStub.callsFake(function(){
  176. counter++;
  177. if(counter != 3) {
  178. return $.Deferred().resolve().promise();
  179. }
  180. setTimeout(function() {
  181. expect(conflictDialogStub.callCount).toEqual(3);
  182. expect(conflictDialogStub.getCall(1).args[0].getFileName())
  183. .toEqual('conflict.txt');
  184. expect(conflictDialogStub.getCall(1).args[1])
  185. .toEqual({ name: 'conflict.txt', mimetype: 'text/plain', directory: '/' });
  186. expect(conflictDialogStub.getCall(1).args[2]).toEqual({ name: 'conflict.txt' });
  187. // yes, the dialog must be called several times...
  188. expect(conflictDialogStub.getCall(2).args[0].getFileName()).toEqual('conflict2.txt');
  189. expect(conflictDialogStub.getCall(2).args[1])
  190. .toEqual({ name: 'conflict2.txt', mimetype: 'text/plain', directory: '/' });
  191. expect(conflictDialogStub.getCall(2).args[2]).toEqual({ name: 'conflict2.txt' });
  192. expect(result[0].submit.calledOnce).toEqual(false);
  193. expect(result[1].submit.calledOnce).toEqual(false);
  194. expect(result[2].submit.calledOnce).toEqual(true);
  195. done();
  196. }, 0);
  197. });
  198. var result = addFiles(uploader, [
  199. {name: 'conflict.txt'},
  200. {name: 'conflict2.txt'},
  201. {name: 'noconflict.txt'}
  202. ]);
  203. });
  204. it('cancels upload when skipping file in conflict mode', function(done) {
  205. var fileData = {name: 'conflict.txt'};
  206. var uploadData = addFiles(uploader, [
  207. fileData
  208. ]);
  209. var upload = new OC.FileUpload(uploader, uploadData[0]);
  210. var deleteStub = sinon.stub(upload, 'deleteUpload');
  211. deleteStub.callsFake(function(){
  212. expect(deleteStub.calledOnce).toEqual(true);
  213. done();
  214. });
  215. uploader.onSkip(upload);
  216. });
  217. it('overwrites file when choosing replace in conflict mode', function(done) {
  218. var fileData = {name: 'conflict.txt'};
  219. var uploadData = addFiles(uploader, [
  220. fileData
  221. ]);
  222. expect(uploadData[0].submit.notCalled).toEqual(true);
  223. var upload = new OC.FileUpload(uploader, uploadData[0]);
  224. uploadData[0].submit.callsFake(function(){
  225. expect(upload.getConflictMode()).toEqual(OC.FileUpload.CONFLICT_MODE_OVERWRITE);
  226. expect(uploadData[0].submit.callCount).toEqual(1);
  227. done();
  228. });
  229. uploader.onReplace(upload);
  230. });
  231. it('autorenames file when choosing replace in conflict mode', function(done) {
  232. // needed for _.defer call
  233. var clock = sinon.useFakeTimers();
  234. var fileData = {name: 'conflict.txt'};
  235. var uploadData = addFiles(uploader, [
  236. fileData
  237. ]);
  238. expect(uploadData[0].submit.notCalled).toEqual(true);
  239. var upload = new OC.FileUpload(uploader, uploadData[0]);
  240. var getResponseStatusStub = sinon.stub(upload, 'getResponseStatus');
  241. var counter = 0;
  242. uploadData[0].submit.callsFake(function(){
  243. counter++;
  244. if(counter===1)
  245. {
  246. expect(upload.getConflictMode()).toEqual(OC.FileUpload.CONFLICT_MODE_AUTORENAME);
  247. expect(upload.getFileName()).toEqual('conflict (2).txt');
  248. expect(uploadData[0].submit.calledOnce).toEqual(true);
  249. getResponseStatusStub.returns(412);
  250. uploader.fileUploadParam.fail.call($dummyUploader[0], {}, uploadData[0]);
  251. clock.tick(500);
  252. }
  253. if(counter===2)
  254. {
  255. expect(upload.getFileName()).toEqual('conflict (3).txt');
  256. expect(uploadData[0].submit.calledTwice).toEqual(true);
  257. clock.restore();
  258. done();
  259. }
  260. });
  261. uploader.onAutorename(upload);
  262. // in case of server-side conflict, tries to rename again
  263. });
  264. });
  265. });