Core.test.js 58 KB

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