Core.test.js 59 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187
  1. import Core from './Core'
  2. import utils from './Utils'
  3. import Plugin from './Plugin'
  4. import AcquirerPlugin1 from '../../test/mocks/acquirerPlugin1'
  5. import AcquirerPlugin2 from '../../test/mocks/acquirerPlugin2'
  6. import InvalidPlugin from '../../test/mocks/invalidPlugin'
  7. import InvalidPluginWithoutId from '../../test/mocks/invalidPluginWithoutId'
  8. import InvalidPluginWithoutType from '../../test/mocks/invalidPluginWithoutType'
  9. const sampleImageDataURI =
  10. ''
  11. describe('src/Core', () => {
  12. const RealCreateObjectUrl = global.URL.createObjectURL
  13. beforeEach(() => {
  14. jest.spyOn(utils, 'findDOMElement').mockImplementation(path => {
  15. return 'some config...'
  16. })
  17. jest.spyOn(utils, 'createThumbnail').mockImplementation(path => {
  18. return Promise.resolve(sampleImageDataURI)
  19. })
  20. utils.createThumbnail.mockClear()
  21. global.URL.createObjectURL = jest.fn().mockReturnValue('newUrl')
  22. })
  23. afterEach(() => {
  24. global.URL.createObjectURL = RealCreateObjectUrl
  25. })
  26. it('should expose a class', () => {
  27. const core = Core()
  28. expect(core.constructor.name).toEqual('Uppy')
  29. })
  30. it('should have a string `id` option that defaults to "uppy"', () => {
  31. const core = Core()
  32. expect(core.getID()).toEqual('uppy')
  33. const core2 = Core({ id: 'profile' })
  34. expect(core2.getID()).toEqual('profile')
  35. })
  36. describe('plugins', () => {
  37. it('should add a plugin to the plugin stack', () => {
  38. const core = Core()
  39. core.use(AcquirerPlugin1)
  40. expect(Object.keys(core.plugins.acquirer).length).toEqual(1)
  41. })
  42. it('should prevent the same plugin from being added more than once', () => {
  43. const core = Core()
  44. core.use(AcquirerPlugin1)
  45. expect(() => {
  46. core.use(AcquirerPlugin1)
  47. }).toThrowErrorMatchingSnapshot()
  48. })
  49. it('should not be able to add an invalid plugin', () => {
  50. const core = Core()
  51. expect(() => {
  52. core.use(InvalidPlugin)
  53. }).toThrowErrorMatchingSnapshot()
  54. })
  55. it('should not be able to add a plugin that has no id', () => {
  56. const core = Core()
  57. expect(() =>
  58. core.use(InvalidPluginWithoutId)
  59. ).toThrowErrorMatchingSnapshot()
  60. })
  61. it('should not be able to add a plugin that has no type', () => {
  62. const core = Core()
  63. expect(() =>
  64. core.use(InvalidPluginWithoutType)
  65. ).toThrowErrorMatchingSnapshot()
  66. })
  67. it('should return the plugin that matches the specified name', () => {
  68. const core = new Core()
  69. expect(core.getPlugin('foo')).toEqual(false)
  70. core.use(AcquirerPlugin1)
  71. const plugin = core.getPlugin('TestSelector1')
  72. expect(plugin.id).toEqual('TestSelector1')
  73. expect(plugin instanceof Plugin)
  74. })
  75. it('should call the specified method on all the plugins', () => {
  76. const core = new Core()
  77. core.use(AcquirerPlugin1)
  78. core.use(AcquirerPlugin2)
  79. core.iteratePlugins(plugin => {
  80. plugin.run('hello')
  81. })
  82. expect(core.plugins.acquirer[0].mocks.run.mock.calls.length).toEqual(1)
  83. expect(core.plugins.acquirer[0].mocks.run.mock.calls[0]).toEqual([
  84. 'hello'
  85. ])
  86. expect(core.plugins.acquirer[1].mocks.run.mock.calls.length).toEqual(1)
  87. expect(core.plugins.acquirer[1].mocks.run.mock.calls[0]).toEqual([
  88. 'hello'
  89. ])
  90. })
  91. it('should uninstall and the remove the specified plugin', () => {
  92. const core = new Core()
  93. core.use(AcquirerPlugin1)
  94. core.use(AcquirerPlugin2)
  95. expect(Object.keys(core.plugins.acquirer).length).toEqual(2)
  96. const plugin = core.getPlugin('TestSelector1')
  97. core.removePlugin(plugin)
  98. expect(Object.keys(core.plugins.acquirer).length).toEqual(1)
  99. expect(plugin.mocks.uninstall.mock.calls.length).toEqual(1)
  100. expect(core.plugins.acquirer[0].mocks.run.mock.calls.length).toEqual(0)
  101. })
  102. })
  103. describe('state', () => {
  104. it('should update all the plugins with the new state when the updateAll method is called', () => {
  105. const core = new Core()
  106. core.use(AcquirerPlugin1)
  107. core.use(AcquirerPlugin2)
  108. core.updateAll({ foo: 'bar' })
  109. expect(core.plugins.acquirer[0].mocks.update.mock.calls.length).toEqual(1)
  110. expect(core.plugins.acquirer[0].mocks.update.mock.calls[0]).toEqual([
  111. { foo: 'bar' }
  112. ])
  113. expect(core.plugins.acquirer[1].mocks.update.mock.calls.length).toEqual(1)
  114. expect(core.plugins.acquirer[1].mocks.update.mock.calls[0]).toEqual([
  115. { foo: 'bar' }
  116. ])
  117. })
  118. it('should update the state', () => {
  119. const core = new Core()
  120. const stateUpdateEventMock = jest.fn()
  121. core.on('state-update', stateUpdateEventMock)
  122. core.use(AcquirerPlugin1)
  123. core.use(AcquirerPlugin2)
  124. core.setState({ foo: 'bar', bee: 'boo' })
  125. core.setState({ foo: 'baar' })
  126. const newState = {
  127. bee: 'boo',
  128. capabilities: { resumableUploads: false },
  129. files: {},
  130. foo: 'baar',
  131. info: { isHidden: true, message: '', type: 'info' },
  132. meta: {},
  133. plugins: {},
  134. totalProgress: 0
  135. }
  136. expect(core.state).toEqual(newState)
  137. expect(core.plugins.acquirer[0].mocks.update.mock.calls[1]).toEqual([
  138. newState
  139. ])
  140. expect(core.plugins.acquirer[1].mocks.update.mock.calls[1]).toEqual([
  141. newState
  142. ])
  143. expect(stateUpdateEventMock.mock.calls.length).toEqual(2)
  144. // current state
  145. expect(stateUpdateEventMock.mock.calls[1][0]).toEqual({
  146. bee: 'boo',
  147. capabilities: { resumableUploads: false },
  148. files: {},
  149. foo: 'bar',
  150. info: { isHidden: true, message: '', type: 'info' },
  151. meta: {},
  152. plugins: {},
  153. totalProgress: 0
  154. })
  155. // new state
  156. expect(stateUpdateEventMock.mock.calls[1][1]).toEqual({
  157. bee: 'boo',
  158. capabilities: { resumableUploads: false },
  159. files: {},
  160. foo: 'baar',
  161. info: { isHidden: true, message: '', type: 'info' },
  162. meta: {},
  163. plugins: {},
  164. totalProgress: 0
  165. })
  166. })
  167. it('should get the state', () => {
  168. const core = new Core()
  169. core.setState({ foo: 'bar' })
  170. expect(core.getState()).toEqual({
  171. capabilities: { resumableUploads: false },
  172. files: {},
  173. foo: 'bar',
  174. info: { isHidden: true, message: '', type: 'info' },
  175. meta: {},
  176. plugins: {},
  177. totalProgress: 0
  178. })
  179. })
  180. })
  181. it('should reset when the reset method is called', () => {
  182. const core = new Core()
  183. // const corePauseEventMock = jest.fn()
  184. const coreCancelEventMock = jest.fn()
  185. const coreStateUpdateEventMock = jest.fn()
  186. core.on('cancel-all', coreCancelEventMock)
  187. core.on('state-update', coreStateUpdateEventMock)
  188. core.setState({ foo: 'bar', totalProgress: 30 })
  189. core.reset()
  190. // expect(corePauseEventMock.mock.calls.length).toEqual(1)
  191. expect(coreCancelEventMock.mock.calls.length).toEqual(1)
  192. expect(coreStateUpdateEventMock.mock.calls.length).toEqual(2)
  193. expect(coreStateUpdateEventMock.mock.calls[1][1]).toEqual({
  194. capabilities: { resumableUploads: false },
  195. files: {},
  196. foo: 'bar',
  197. info: { isHidden: true, message: '', type: 'info' },
  198. meta: {},
  199. plugins: {},
  200. totalProgress: 0
  201. })
  202. })
  203. it('should close, reset and uninstall when the close method is called', () => {
  204. const core = new Core()
  205. core.use(AcquirerPlugin1)
  206. // const corePauseEventMock = jest.fn()
  207. const coreCancelEventMock = jest.fn()
  208. const coreStateUpdateEventMock = jest.fn()
  209. // core.on('pause-all', corePauseEventMock)
  210. core.on('cancel-all', coreCancelEventMock)
  211. core.on('state-update', coreStateUpdateEventMock)
  212. core.close()
  213. // expect(corePauseEventMock.mock.calls.length).toEqual(1)
  214. expect(coreCancelEventMock.mock.calls.length).toEqual(1)
  215. expect(coreStateUpdateEventMock.mock.calls.length).toEqual(1)
  216. expect(coreStateUpdateEventMock.mock.calls[0][1]).toEqual({
  217. capabilities: { resumableUploads: false },
  218. files: {},
  219. info: { isHidden: true, message: '', type: 'info' },
  220. meta: {},
  221. plugins: {},
  222. totalProgress: 0
  223. })
  224. expect(core.plugins.acquirer[0].mocks.uninstall.mock.calls.length).toEqual(
  225. 1
  226. )
  227. })
  228. describe('preprocessors', () => {
  229. it('should add a preprocessor', () => {
  230. const core = new Core()
  231. const preprocessor = function () {}
  232. core.addPreProcessor(preprocessor)
  233. expect(core.preProcessors[0]).toEqual(preprocessor)
  234. })
  235. it('should remove a preprocessor', () => {
  236. const core = new Core()
  237. const preprocessor1 = function () {}
  238. const preprocessor2 = function () {}
  239. const preprocessor3 = function () {}
  240. core.addPreProcessor(preprocessor1)
  241. core.addPreProcessor(preprocessor2)
  242. core.addPreProcessor(preprocessor3)
  243. expect(core.preProcessors.length).toEqual(3)
  244. core.removePreProcessor(preprocessor2)
  245. expect(core.preProcessors.length).toEqual(2)
  246. })
  247. it('should execute all the preprocessors when uploading a file', () => {
  248. const core = new Core()
  249. const preprocessor1 = jest.fn()
  250. const preprocessor2 = jest.fn()
  251. core.addPreProcessor(preprocessor1)
  252. core.addPreProcessor(preprocessor2)
  253. return core
  254. .addFile({
  255. source: 'jest',
  256. name: 'foo.jpg',
  257. type: 'image/jpeg',
  258. data: utils.dataURItoFile(sampleImageDataURI, {})
  259. })
  260. .then(() => core.upload())
  261. .then(() => {
  262. const fileId = Object.keys(core.state.files)[0]
  263. expect(preprocessor1.mock.calls.length).toEqual(1)
  264. expect(preprocessor1.mock.calls[0][0].length).toEqual(1)
  265. expect(preprocessor1.mock.calls[0][0][0]).toEqual(fileId)
  266. expect(preprocessor2.mock.calls[0][0].length).toEqual(1)
  267. expect(preprocessor2.mock.calls[0][0][0]).toEqual(fileId)
  268. })
  269. })
  270. it('should update the file progress state when preprocess-progress event is fired', () => {
  271. const core = new Core()
  272. core.run()
  273. return core
  274. .addFile({
  275. source: 'jest',
  276. name: 'foo.jpg',
  277. type: 'image/jpeg',
  278. data: utils.dataURItoFile(sampleImageDataURI, {})
  279. })
  280. .then(() => {
  281. const fileId = Object.keys(core.state.files)[0]
  282. core.emit('preprocess-progress', fileId, {
  283. mode: 'determinate',
  284. message: 'something',
  285. value: 0
  286. })
  287. expect(core.state.files[fileId].progress).toEqual({
  288. percentage: 0,
  289. bytesUploaded: 0,
  290. bytesTotal: 17175,
  291. uploadComplete: false,
  292. uploadStarted: false,
  293. preprocess: { mode: 'determinate', message: 'something', value: 0 }
  294. })
  295. })
  296. })
  297. it('should update the file progress state when preprocess-complete event is fired', () => {
  298. const core = new Core()
  299. core.run()
  300. return core
  301. .addFile({
  302. source: 'jest',
  303. name: 'foo.jpg',
  304. type: 'image/jpeg',
  305. data: utils.dataURItoFile(sampleImageDataURI, {})
  306. })
  307. .then(() => {
  308. const fileId = Object.keys(core.state.files)[0]
  309. core.emit('preprocess-complete', fileId, {
  310. mode: 'determinate',
  311. message: 'something',
  312. value: 0
  313. })
  314. expect(core.state.files[fileId].progress).toEqual({
  315. percentage: 0,
  316. bytesUploaded: 0,
  317. bytesTotal: 17175,
  318. uploadComplete: false,
  319. uploadStarted: false
  320. })
  321. })
  322. })
  323. })
  324. describe('postprocessors', () => {
  325. it('should add a postprocessor', () => {
  326. const core = new Core()
  327. const postprocessor = function () {}
  328. core.addPostProcessor(postprocessor)
  329. expect(core.postProcessors[0]).toEqual(postprocessor)
  330. })
  331. it('should remove a postprocessor', () => {
  332. const core = new Core()
  333. const postprocessor1 = function () {}
  334. const postprocessor2 = function () {}
  335. const postprocessor3 = function () {}
  336. core.addPostProcessor(postprocessor1)
  337. core.addPostProcessor(postprocessor2)
  338. core.addPostProcessor(postprocessor3)
  339. expect(core.postProcessors.length).toEqual(3)
  340. core.removePostProcessor(postprocessor2)
  341. expect(core.postProcessors.length).toEqual(2)
  342. })
  343. it('should execute all the postprocessors when uploading a file', () => {
  344. const core = new Core()
  345. const postprocessor1 = jest.fn()
  346. const postprocessor2 = jest.fn()
  347. core.addPostProcessor(postprocessor1)
  348. core.addPostProcessor(postprocessor2)
  349. return core
  350. .addFile({
  351. source: 'jest',
  352. name: 'foo.jpg',
  353. type: 'image/jpeg',
  354. data: utils.dataURItoFile(sampleImageDataURI, {})
  355. })
  356. .then(() => core.upload())
  357. .then(() => {
  358. expect(postprocessor1.mock.calls.length).toEqual(1)
  359. // const lastModifiedTime = new Date()
  360. // const fileId = 'foojpg' + lastModifiedTime.getTime()
  361. const fileId = 'uppy-foojpg-image'
  362. expect(postprocessor1.mock.calls[0][0].length).toEqual(1)
  363. expect(postprocessor1.mock.calls[0][0][0].substring(0, 17)).toEqual(
  364. fileId.substring(0, 17)
  365. )
  366. expect(postprocessor2.mock.calls[0][0].length).toEqual(1)
  367. expect(postprocessor2.mock.calls[0][0][0].substring(0, 17)).toEqual(
  368. fileId.substring(0, 17)
  369. )
  370. })
  371. })
  372. it('should update the file progress state when postprocess-progress event is fired', () => {
  373. const core = new Core()
  374. core.run()
  375. return core
  376. .addFile({
  377. source: 'jest',
  378. name: 'foo.jpg',
  379. type: 'image/jpeg',
  380. data: utils.dataURItoFile(sampleImageDataURI, {})
  381. })
  382. .then(() => {
  383. const fileId = Object.keys(core.state.files)[0]
  384. core.emit('postprocess-progress', fileId, {
  385. mode: 'determinate',
  386. message: 'something',
  387. value: 0
  388. })
  389. expect(core.state.files[fileId].progress).toEqual({
  390. percentage: 0,
  391. bytesUploaded: 0,
  392. bytesTotal: 17175,
  393. uploadComplete: false,
  394. uploadStarted: false,
  395. postprocess: { mode: 'determinate', message: 'something', value: 0 }
  396. })
  397. })
  398. })
  399. it('should update the file progress state when postprocess-complete event is fired', () => {
  400. const core = new Core()
  401. core.run()
  402. return core
  403. .addFile({
  404. source: 'jest',
  405. name: 'foo.jpg',
  406. type: 'image/jpeg',
  407. data: utils.dataURItoFile(sampleImageDataURI, {})
  408. })
  409. .then(() => {
  410. const fileId = Object.keys(core.state.files)[0]
  411. core.emit('postprocess-complete', fileId, {
  412. mode: 'determinate',
  413. message: 'something',
  414. value: 0
  415. })
  416. expect(core.state.files[fileId].progress).toEqual({
  417. percentage: 0,
  418. bytesUploaded: 0,
  419. bytesTotal: 17175,
  420. uploadComplete: false,
  421. uploadStarted: false
  422. })
  423. })
  424. })
  425. })
  426. describe('uploaders', () => {
  427. it('should add an uploader', () => {
  428. const core = new Core()
  429. const uploader = function () {}
  430. core.addUploader(uploader)
  431. expect(core.uploaders[0]).toEqual(uploader)
  432. })
  433. it('should remove an uploader', () => {
  434. const core = new Core()
  435. const uploader1 = function () {}
  436. const uploader2 = function () {}
  437. const uploader3 = function () {}
  438. core.addUploader(uploader1)
  439. core.addUploader(uploader2)
  440. core.addUploader(uploader3)
  441. expect(core.uploaders.length).toEqual(3)
  442. core._removeUploader(uploader2)
  443. expect(core.uploaders.length).toEqual(2)
  444. })
  445. })
  446. describe('adding a file', () => {
  447. it('should call onBeforeFileAdded if it was specified in the options when initailising the class', () => {
  448. const onBeforeFileAdded = jest.fn(value => {
  449. return Promise.resolve()
  450. })
  451. const core = new Core({
  452. onBeforeFileAdded
  453. })
  454. return core
  455. .addFile({
  456. source: 'jest',
  457. name: 'foo.jpg',
  458. type: 'image/jpeg',
  459. data: utils.dataURItoFile(sampleImageDataURI, {})
  460. })
  461. .then(() => {
  462. expect(onBeforeFileAdded.mock.calls.length).toEqual(1)
  463. expect(onBeforeFileAdded.mock.calls[0][0].name).toEqual('foo.jpg')
  464. expect(onBeforeFileAdded.mock.calls[0][1]).toEqual({})
  465. })
  466. })
  467. it('should add a file', () => {
  468. const fileData = utils.dataURItoFile(sampleImageDataURI, {})
  469. const fileAddedEventMock = jest.fn()
  470. const core = new Core()
  471. core.run()
  472. core.on('file-added', fileAddedEventMock)
  473. return core
  474. .addFile({
  475. source: 'jest',
  476. name: 'foo.jpg',
  477. type: 'image/jpeg',
  478. data: fileData
  479. })
  480. .then(() => {
  481. const fileId = Object.keys(core.state.files)[0]
  482. const newFile = {
  483. extension: 'jpg',
  484. id: fileId,
  485. isRemote: false,
  486. meta: { name: 'foo.jpg', type: 'image/jpeg' },
  487. name: 'foo.jpg',
  488. preview: sampleImageDataURI,
  489. data: fileData,
  490. progress: {
  491. bytesTotal: 17175,
  492. bytesUploaded: 0,
  493. percentage: 0,
  494. uploadComplete: false,
  495. uploadStarted: false
  496. },
  497. remote: '',
  498. size: 17175,
  499. source: 'jest',
  500. type: 'image/jpeg'
  501. }
  502. expect(core.state.files[fileId]).toEqual(newFile)
  503. newFile.preview = undefined // not sure why this happens.. needs further investigation
  504. expect(fileAddedEventMock.mock.calls[0][0]).toEqual(newFile)
  505. })
  506. })
  507. it('should not allow a file that does not meet the restrictions', () => {
  508. const core = new Core({
  509. restrictions: {
  510. allowedFileTypes: ['image/gif']
  511. }
  512. })
  513. return core
  514. .addFile({
  515. source: 'jest',
  516. name: 'foo.jpg',
  517. type: 'image/jpeg',
  518. data: utils.dataURItoFile(sampleImageDataURI, {})
  519. })
  520. .then(() => {
  521. throw new Error('File was allowed through')
  522. })
  523. .catch(e => {
  524. expect(e.message).toEqual('File not allowed')
  525. })
  526. })
  527. it('should work with restriction errors that are not Error class instances', () => {
  528. const core = new Core({
  529. onBeforeFileAdded () {
  530. return Promise.reject('a plain string') // eslint-disable-line prefer-promise-reject-errors
  531. }
  532. })
  533. return expect(core.addFile({
  534. source: 'jest',
  535. name: 'foo.jpg',
  536. type: 'image/jpeg',
  537. data: null
  538. })).rejects.toMatchObject({ message: 'onBeforeFileAdded: a plain string' })
  539. })
  540. it('should call utils.generatePreview when file-added is triggered and thumbnail generation is allowed', () => {
  541. const core = new Core({
  542. }).run()
  543. const file = {
  544. type: 'image/jpeg',
  545. isRemote: false
  546. }
  547. core.emit('file-added', file)
  548. expect(utils.createThumbnail).toHaveBeenCalledTimes(1)
  549. expect(utils.createThumbnail.mock.calls[0][1]).toEqual(200)
  550. })
  551. it('should return an object url of the image when file-added is triggered and thumbnail generation is disabled', () => {
  552. const core = new Core({
  553. thumbnailGeneration: false
  554. }).run()
  555. const file = {
  556. type: 'image/jpeg',
  557. isRemote: false,
  558. data: 'foo'
  559. }
  560. core.emit('file-added', file)
  561. expect(URL.createObjectURL).toHaveBeenCalledTimes(1)
  562. expect(URL.createObjectURL).toHaveBeenCalledWith('foo')
  563. })
  564. })
  565. describe('uploading a file', () => {
  566. it('should return a { successful, failed } pair containing file objects', () => {
  567. const core = new Core().run()
  568. core.addUploader((fileIDs) => Promise.resolve())
  569. return Promise.all([
  570. core.addFile({ source: 'jest', name: 'foo.jpg', type: 'image/jpeg', data: new Uint8Array() }),
  571. core.addFile({ source: 'jest', name: 'bar.jpg', type: 'image/jpeg', data: new Uint8Array() })
  572. ]).then(() => {
  573. return expect(core.upload()).resolves.toMatchObject({
  574. successful: [
  575. { name: 'foo.jpg' },
  576. { name: 'bar.jpg' }
  577. ],
  578. failed: []
  579. })
  580. })
  581. })
  582. it('should return files with errors in the { failed } key', () => {
  583. const core = new Core().run()
  584. core.addUploader((fileIDs) => {
  585. fileIDs.forEach((fileID) => {
  586. if (/bar/.test(core.getFile(fileID).name)) {
  587. core.emit('upload-error', fileID, new Error('This is bar and I do not like bar'))
  588. }
  589. })
  590. return Promise.resolve()
  591. })
  592. return Promise.all([
  593. core.addFile({ source: 'jest', name: 'foo.jpg', type: 'image/jpeg', data: new Uint8Array() }),
  594. core.addFile({ source: 'jest', name: 'bar.jpg', type: 'image/jpeg', data: new Uint8Array() })
  595. ]).then(() => {
  596. return expect(core.upload()).resolves.toMatchObject({
  597. successful: [
  598. { name: 'foo.jpg' }
  599. ],
  600. failed: [
  601. { name: 'bar.jpg', error: 'This is bar and I do not like bar' }
  602. ]
  603. })
  604. })
  605. })
  606. })
  607. describe('removing a file', () => {
  608. it('should remove the file', () => {
  609. const fileRemovedEventMock = jest.fn()
  610. const core = new Core()
  611. core.on('file-removed', fileRemovedEventMock)
  612. core.run()
  613. return core
  614. .addFile({
  615. source: 'jest',
  616. name: 'foo.jpg',
  617. type: 'image/jpeg',
  618. data: utils.dataURItoFile(sampleImageDataURI, {})
  619. })
  620. .then(() => {
  621. const fileId = Object.keys(core.state.files)[0]
  622. expect(Object.keys(core.state.files).length).toEqual(1)
  623. core.setState({
  624. totalProgress: 50
  625. })
  626. core.removeFile(fileId)
  627. expect(Object.keys(core.state.files).length).toEqual(0)
  628. expect(fileRemovedEventMock.mock.calls[0][0]).toEqual(fileId)
  629. expect(core.state.totalProgress).toEqual(0)
  630. })
  631. })
  632. })
  633. describe('restoring a file', () => {
  634. xit('should restore a file', () => {})
  635. xit("should fail to restore a file if it doesn't exist", () => {})
  636. })
  637. describe('get a file', () => {
  638. it('should get the specified file', () => {
  639. const core = new Core()
  640. return core
  641. .addFile({
  642. source: 'jest',
  643. name: 'foo.jpg',
  644. type: 'image/jpeg',
  645. data: utils.dataURItoFile(sampleImageDataURI, {})
  646. })
  647. .then(() => {
  648. const fileId = Object.keys(core.state.files)[0]
  649. expect(core.getFile(fileId).name).toEqual('foo.jpg')
  650. expect(core.getFile('non existant file')).toEqual(undefined)
  651. })
  652. })
  653. })
  654. describe('meta data', () => {
  655. it('should set meta data by calling setMeta', () => {
  656. const core = new Core({
  657. meta: { foo2: 'bar2' }
  658. })
  659. core.setMeta({ foo: 'bar', bur: 'mur' })
  660. core.setMeta({ boo: 'moo', bur: 'fur' })
  661. expect(core.state.meta).toEqual({
  662. foo: 'bar',
  663. foo2: 'bar2',
  664. boo: 'moo',
  665. bur: 'fur'
  666. })
  667. })
  668. it('should update meta data for a file by calling updateMeta', () => {
  669. const core = new Core()
  670. return core
  671. .addFile({
  672. source: 'jest',
  673. name: 'foo.jpg',
  674. type: 'image/jpeg',
  675. data: utils.dataURItoFile(sampleImageDataURI, {})
  676. })
  677. .then(() => {
  678. const fileId = Object.keys(core.state.files)[0]
  679. core.setFileMeta(fileId, { foo: 'bar', bur: 'mur' })
  680. core.setFileMeta(fileId, { boo: 'moo', bur: 'fur' })
  681. expect(core.state.files[fileId].meta).toEqual({
  682. name: 'foo.jpg',
  683. type: 'image/jpeg',
  684. foo: 'bar',
  685. bur: 'fur',
  686. boo: 'moo'
  687. })
  688. })
  689. })
  690. })
  691. describe('progress', () => {
  692. it('should calculate the progress of a file upload', () => {
  693. const core = new Core()
  694. return core
  695. .addFile({
  696. source: 'jest',
  697. name: 'foo.jpg',
  698. type: 'image/jpeg',
  699. data: utils.dataURItoFile(sampleImageDataURI, {})
  700. })
  701. .then(() => {
  702. const fileId = Object.keys(core.state.files)[0]
  703. core._calculateProgress({
  704. id: fileId,
  705. bytesUploaded: 12345,
  706. bytesTotal: 17175
  707. })
  708. expect(core.state.files[fileId].progress).toEqual({
  709. percentage: 71,
  710. bytesUploaded: 12345,
  711. bytesTotal: 17175,
  712. uploadComplete: false,
  713. uploadStarted: false
  714. })
  715. core._calculateProgress({
  716. id: fileId,
  717. bytesUploaded: 17175,
  718. bytesTotal: 17175
  719. })
  720. expect(core.state.files[fileId].progress).toEqual({
  721. percentage: 100,
  722. bytesUploaded: 17175,
  723. bytesTotal: 17175,
  724. uploadComplete: false,
  725. uploadStarted: false
  726. })
  727. })
  728. })
  729. it('should calculate the total progress of all file uploads', () => {
  730. const core = new Core()
  731. return core
  732. .addFile({
  733. source: 'jest',
  734. name: 'foo.jpg',
  735. type: 'image/jpeg',
  736. data: utils.dataURItoFile(sampleImageDataURI, {})
  737. })
  738. .then(() => {
  739. return core
  740. .addFile({
  741. source: 'jest',
  742. name: 'foo2.jpg',
  743. type: 'image/jpeg',
  744. data: utils.dataURItoFile(sampleImageDataURI, {})
  745. })
  746. }).then(() => {
  747. const fileId1 = Object.keys(core.state.files)[0]
  748. const fileId2 = Object.keys(core.state.files)[1]
  749. core.state.files[fileId1].progress.uploadStarted = new Date()
  750. core.state.files[fileId2].progress.uploadStarted = new Date()
  751. core._calculateProgress({
  752. id: fileId1,
  753. bytesUploaded: 12345,
  754. bytesTotal: 17175
  755. })
  756. core._calculateProgress({
  757. id: fileId2,
  758. bytesUploaded: 10201,
  759. bytesTotal: 17175
  760. })
  761. core._calculateTotalProgress()
  762. expect(core.state.totalProgress).toEqual(65)
  763. })
  764. })
  765. it('should reset the progress', () => {
  766. const resetProgressEvent = jest.fn()
  767. const core = new Core()
  768. core.run()
  769. core.on('reset-progress', resetProgressEvent)
  770. return core
  771. .addFile({
  772. source: 'jest',
  773. name: 'foo.jpg',
  774. type: 'image/jpeg',
  775. data: utils.dataURItoFile(sampleImageDataURI, {})
  776. })
  777. .then(() => {
  778. return core
  779. .addFile({
  780. source: 'jest',
  781. name: 'foo2.jpg',
  782. type: 'image/jpeg',
  783. data: utils.dataURItoFile(sampleImageDataURI, {})
  784. })
  785. }).then(() => {
  786. const fileId1 = Object.keys(core.state.files)[0]
  787. const fileId2 = Object.keys(core.state.files)[1]
  788. core.state.files[fileId1].progress.uploadStarted = new Date()
  789. core.state.files[fileId2].progress.uploadStarted = new Date()
  790. core._calculateProgress({
  791. id: fileId1,
  792. bytesUploaded: 12345,
  793. bytesTotal: 17175
  794. })
  795. core._calculateProgress({
  796. id: fileId2,
  797. bytesUploaded: 10201,
  798. bytesTotal: 17175
  799. })
  800. core._calculateTotalProgress()
  801. expect(core.state.totalProgress).toEqual(65)
  802. core.resetProgress()
  803. expect(core.state.files[fileId1].progress).toEqual({
  804. percentage: 0,
  805. bytesUploaded: 0,
  806. bytesTotal: 17175,
  807. uploadComplete: false,
  808. uploadStarted: false
  809. })
  810. expect(core.state.files[fileId2].progress).toEqual({
  811. percentage: 0,
  812. bytesUploaded: 0,
  813. bytesTotal: 17175,
  814. uploadComplete: false,
  815. uploadStarted: false
  816. })
  817. expect(core.state.totalProgress).toEqual(0)
  818. expect(resetProgressEvent.mock.calls.length).toEqual(1)
  819. })
  820. })
  821. })
  822. describe('checkRestrictions', () => {
  823. it('should enforce the maxNumberOfFiles rule', () => {
  824. const core = new Core({
  825. autoProceed: false,
  826. restrictions: {
  827. maxNumberOfFiles: 1
  828. }
  829. })
  830. // add 2 files
  831. core.addFile({
  832. source: 'jest',
  833. name: 'foo1.jpg',
  834. type: 'image/jpeg',
  835. data: utils.dataURItoFile(sampleImageDataURI, {})
  836. })
  837. return expect(core.addFile({
  838. source: 'jest',
  839. name: 'foo2.jpg',
  840. type: 'image/jpeg',
  841. data: utils.dataURItoFile(sampleImageDataURI, {})
  842. })).rejects.toMatchObject({ message: 'File not allowed' }).then(() => {
  843. expect(core.state.info.message).toEqual('You can only upload 1 file')
  844. })
  845. })
  846. xit('should enforce the minNumberOfFiles rule', () => {})
  847. it('should enfore the allowedFileTypes rule', () => {
  848. const core = new Core({
  849. autoProceed: false,
  850. restrictions: {
  851. allowedFileTypes: ['image/gif', 'image/png']
  852. }
  853. })
  854. return expect(core.addFile({
  855. source: 'jest',
  856. name: 'foo2.jpg',
  857. type: 'image/jpeg',
  858. data: utils.dataURItoFile(sampleImageDataURI, {})
  859. })).rejects.toMatchObject({ message: 'File not allowed' }).then(() => {
  860. expect(core.state.info.message).toEqual('You can only upload: image/gif, image/png')
  861. })
  862. })
  863. it('should enforce the maxFileSize rule', () => {
  864. const core = new Core({
  865. autoProceed: false,
  866. restrictions: {
  867. maxFileSize: 1234
  868. }
  869. })
  870. return expect(core.addFile({
  871. source: 'jest',
  872. name: 'foo.jpg',
  873. type: 'image/jpeg',
  874. data: utils.dataURItoFile(sampleImageDataURI, {})
  875. })).rejects.toMatchObject({ message: 'File not allowed' }).then(() => {
  876. expect(core.state.info.message).toEqual('This file exceeds maximum allowed size of 1.2 KB')
  877. })
  878. })
  879. })
  880. describe('actions', () => {
  881. it('should update the state when receiving the error event', () => {
  882. const core = new Core()
  883. core.run()
  884. core.emit('error', new Error('foooooo'))
  885. expect(core.state.error).toEqual('foooooo')
  886. })
  887. it('should update the state when receiving the upload-error event', () => {
  888. const core = new Core()
  889. core.run()
  890. core.state.files['fileId'] = {
  891. name: 'filename'
  892. }
  893. core.emit('upload-error', 'fileId', { message: 'this is the error' })
  894. expect(core.state.info).toEqual({'message': 'Failed to upload filename', 'details': 'this is the error', 'isHidden': false, 'type': 'error'})
  895. })
  896. it('should reset the error state when receiving the upload event', () => {
  897. const core = new Core()
  898. core.run()
  899. core.emit('error', { foo: 'bar' })
  900. core.emit('upload')
  901. expect(core.state.error).toEqual(null)
  902. })
  903. })
  904. describe('updateOnlineStatus', () => {
  905. const RealNavigatorOnline = global.window.navigator.onLine
  906. function mockNavigatorOnline (status) {
  907. Object.defineProperty(
  908. global.window.navigator,
  909. 'onLine',
  910. {
  911. value: status,
  912. writable: true
  913. }
  914. )
  915. }
  916. afterEach(() => {
  917. global.window.navigator.onLine = RealNavigatorOnline
  918. })
  919. it('should emit the correct event based on whether there is a network connection', () => {
  920. const onlineEventMock = jest.fn()
  921. const offlineEventMock = jest.fn()
  922. const backOnlineEventMock = jest.fn()
  923. const core = new Core()
  924. core.on('is-offline', offlineEventMock)
  925. core.on('is-online', onlineEventMock)
  926. core.on('back-online', backOnlineEventMock)
  927. mockNavigatorOnline(true)
  928. core.updateOnlineStatus()
  929. expect(onlineEventMock.mock.calls.length).toEqual(1)
  930. expect(offlineEventMock.mock.calls.length).toEqual(0)
  931. expect(backOnlineEventMock.mock.calls.length).toEqual(0)
  932. mockNavigatorOnline(false)
  933. core.updateOnlineStatus()
  934. expect(onlineEventMock.mock.calls.length).toEqual(1)
  935. expect(offlineEventMock.mock.calls.length).toEqual(1)
  936. expect(backOnlineEventMock.mock.calls.length).toEqual(0)
  937. mockNavigatorOnline(true)
  938. core.updateOnlineStatus()
  939. expect(onlineEventMock.mock.calls.length).toEqual(2)
  940. expect(offlineEventMock.mock.calls.length).toEqual(1)
  941. expect(backOnlineEventMock.mock.calls.length).toEqual(1)
  942. })
  943. })
  944. describe('info', () => {
  945. it('should set a string based message to be displayed infinitely', () => {
  946. const infoVisibleEvent = jest.fn()
  947. const core = new Core()
  948. core.run()
  949. core.on('info-visible', infoVisibleEvent)
  950. core.info('This is the message', 'info', 0)
  951. expect(core.state.info).toEqual({
  952. isHidden: false,
  953. type: 'info',
  954. message: 'This is the message',
  955. details: null
  956. })
  957. expect(infoVisibleEvent.mock.calls.length).toEqual(1)
  958. expect(typeof core.infoTimeoutID).toEqual('undefined')
  959. })
  960. it('should set a object based message to be displayed infinitely', () => {
  961. const infoVisibleEvent = jest.fn()
  962. const core = new Core()
  963. core.run()
  964. core.on('info-visible', infoVisibleEvent)
  965. core.info({
  966. message: 'This is the message',
  967. details: {
  968. foo: 'bar'
  969. }
  970. }, 'warning', 0)
  971. expect(core.state.info).toEqual({
  972. isHidden: false,
  973. type: 'warning',
  974. message: 'This is the message',
  975. details: {
  976. foo: 'bar'
  977. }
  978. })
  979. expect(infoVisibleEvent.mock.calls.length).toEqual(1)
  980. expect(typeof core.infoTimeoutID).toEqual('undefined')
  981. })
  982. it('should set an info message to be displayed for a period of time before hiding', (done) => {
  983. const infoVisibleEvent = jest.fn()
  984. const infoHiddenEvent = jest.fn()
  985. const core = new Core()
  986. core.run()
  987. core.on('info-visible', infoVisibleEvent)
  988. core.on('info-hidden', infoHiddenEvent)
  989. core.info('This is the message', 'info', 100)
  990. expect(typeof core.infoTimeoutID).toEqual('number')
  991. expect(infoHiddenEvent.mock.calls.length).toEqual(0)
  992. setTimeout(() => {
  993. expect(infoHiddenEvent.mock.calls.length).toEqual(1)
  994. expect(core.state.info).toEqual({
  995. isHidden: true,
  996. type: 'info',
  997. message: 'This is the message',
  998. details: null
  999. })
  1000. done()
  1001. }, 110)
  1002. })
  1003. it('should hide an info message', () => {
  1004. const infoVisibleEvent = jest.fn()
  1005. const infoHiddenEvent = jest.fn()
  1006. const core = new Core()
  1007. core.run()
  1008. core.on('info-visible', infoVisibleEvent)
  1009. core.on('info-hidden', infoHiddenEvent)
  1010. core.info('This is the message', 'info', 0)
  1011. expect(typeof core.infoTimeoutID).toEqual('undefined')
  1012. expect(infoHiddenEvent.mock.calls.length).toEqual(0)
  1013. core.hideInfo()
  1014. expect(infoHiddenEvent.mock.calls.length).toEqual(1)
  1015. expect(core.state.info).toEqual({
  1016. isHidden: true,
  1017. type: 'info',
  1018. message: 'This is the message',
  1019. details: null
  1020. })
  1021. })
  1022. })
  1023. describe('createUpload', () => {
  1024. it('should assign the specified files to a new upload', () => {
  1025. const core = new Core()
  1026. core.run()
  1027. return core.addFile({
  1028. source: 'jest',
  1029. name: 'foo.jpg',
  1030. type: 'image/jpeg',
  1031. data: utils.dataURItoFile(sampleImageDataURI, {})
  1032. }).then(() => {
  1033. core._createUpload(Object.keys(core.state.files))
  1034. const uploadId = Object.keys(core.state.currentUploads)[0]
  1035. const currentUploadsState = {}
  1036. currentUploadsState[uploadId] = {
  1037. fileIDs: Object.keys(core.state.files),
  1038. step: 0
  1039. }
  1040. expect(core.state.currentUploads).toEqual(currentUploadsState)
  1041. })
  1042. })
  1043. })
  1044. describe('removeUpload', () => {
  1045. it('should remove all files from the specified upload', () => {
  1046. // this uploader will run once the upload has started
  1047. const uploader = () => {
  1048. return Promise.resolve().then(() => {
  1049. const uploadId = Object.keys(core.state.currentUploads)[0]
  1050. expect(typeof core.state.currentUploads[uploadId]).toEqual('object')
  1051. expect(core.state.currentUploads[uploadId].fileIDs.length).toEqual(1)
  1052. core._removeUpload(uploadId)
  1053. expect(typeof core.state.currentUploads[uploadId]).toEqual('undefined')
  1054. })
  1055. }
  1056. const core = new Core()
  1057. core.run()
  1058. core.addUploader(uploader)
  1059. return core
  1060. .addFile({
  1061. source: 'jest',
  1062. name: 'foo.jpg',
  1063. type: 'image/jpeg',
  1064. data: utils.dataURItoFile(sampleImageDataURI, {})
  1065. })
  1066. .then(() => {
  1067. return core.upload(true)
  1068. })
  1069. })
  1070. })
  1071. })