import Core from './Core' import utils from './Utils' import Plugin from './Plugin' import AcquirerPlugin1 from '../../test/mocks/acquirerPlugin1' import AcquirerPlugin2 from '../../test/mocks/acquirerPlugin2' import InvalidPlugin from '../../test/mocks/invalidPlugin' import InvalidPluginWithoutId from '../../test/mocks/invalidPluginWithoutId' import InvalidPluginWithoutType from '../../test/mocks/invalidPluginWithoutType' const sampleImageDataURI = '' describe('src/Core', () => { const RealCreateObjectUrl = global.URL.createObjectURL beforeEach(() => { jest.spyOn(utils, 'findDOMElement').mockImplementation(path => { return 'some config...' }) jest.spyOn(utils, 'createThumbnail').mockImplementation(path => { return Promise.resolve(sampleImageDataURI) }) utils.createThumbnail.mockClear() global.URL.createObjectURL = jest.fn().mockReturnValue('newUrl') }) afterEach(() => { global.URL.createObjectURL = RealCreateObjectUrl }) it('should expose a class', () => { const core = Core() expect(core.constructor.name).toEqual('Uppy') }) it('should have a string `id` option that defaults to "uppy"', () => { const core = Core() expect(core.getID()).toEqual('uppy') const core2 = Core({ id: 'profile' }) expect(core2.getID()).toEqual('profile') }) describe('plugins', () => { it('should add a plugin to the plugin stack', () => { const core = Core() core.use(AcquirerPlugin1) expect(Object.keys(core.plugins.acquirer).length).toEqual(1) }) it('should prevent the same plugin from being added more than once', () => { const core = Core() core.use(AcquirerPlugin1) expect(() => { core.use(AcquirerPlugin1) }).toThrowErrorMatchingSnapshot() }) it('should not be able to add an invalid plugin', () => { const core = Core() expect(() => { core.use(InvalidPlugin) }).toThrowErrorMatchingSnapshot() }) it('should not be able to add a plugin that has no id', () => { const core = Core() expect(() => core.use(InvalidPluginWithoutId) ).toThrowErrorMatchingSnapshot() }) it('should not be able to add a plugin that has no type', () => { const core = Core() expect(() => core.use(InvalidPluginWithoutType) ).toThrowErrorMatchingSnapshot() }) it('should return the plugin that matches the specified name', () => { const core = new Core() expect(core.getPlugin('foo')).toEqual(false) core.use(AcquirerPlugin1) const plugin = core.getPlugin('TestSelector1') expect(plugin.id).toEqual('TestSelector1') expect(plugin instanceof Plugin) }) it('should call the specified method on all the plugins', () => { const core = new Core() core.use(AcquirerPlugin1) core.use(AcquirerPlugin2) core.iteratePlugins(plugin => { plugin.run('hello') }) expect(core.plugins.acquirer[0].mocks.run.mock.calls.length).toEqual(1) expect(core.plugins.acquirer[0].mocks.run.mock.calls[0]).toEqual([ 'hello' ]) expect(core.plugins.acquirer[1].mocks.run.mock.calls.length).toEqual(1) expect(core.plugins.acquirer[1].mocks.run.mock.calls[0]).toEqual([ 'hello' ]) }) it('should uninstall and the remove the specified plugin', () => { const core = new Core() core.use(AcquirerPlugin1) core.use(AcquirerPlugin2) expect(Object.keys(core.plugins.acquirer).length).toEqual(2) const plugin = core.getPlugin('TestSelector1') core.removePlugin(plugin) expect(Object.keys(core.plugins.acquirer).length).toEqual(1) expect(plugin.mocks.uninstall.mock.calls.length).toEqual(1) expect(core.plugins.acquirer[0].mocks.run.mock.calls.length).toEqual(0) }) }) describe('state', () => { it('should update all the plugins with the new state when the updateAll method is called', () => { const core = new Core() core.use(AcquirerPlugin1) core.use(AcquirerPlugin2) core.updateAll({ foo: 'bar' }) expect(core.plugins.acquirer[0].mocks.update.mock.calls.length).toEqual(1) expect(core.plugins.acquirer[0].mocks.update.mock.calls[0]).toEqual([ { foo: 'bar' } ]) expect(core.plugins.acquirer[1].mocks.update.mock.calls.length).toEqual(1) expect(core.plugins.acquirer[1].mocks.update.mock.calls[0]).toEqual([ { foo: 'bar' } ]) }) it('should update the state', () => { const core = new Core() const stateUpdateEventMock = jest.fn() core.on('state-update', stateUpdateEventMock) core.use(AcquirerPlugin1) core.use(AcquirerPlugin2) core.setState({ foo: 'bar', bee: 'boo' }) core.setState({ foo: 'baar' }) const newState = { bee: 'boo', capabilities: { resumableUploads: false }, files: {}, currentUploads: {}, foo: 'baar', info: { isHidden: true, message: '', type: 'info' }, meta: {}, plugins: {}, totalProgress: 0 } expect(core.state).toEqual(newState) expect(core.plugins.acquirer[0].mocks.update.mock.calls[1]).toEqual([ newState ]) expect(core.plugins.acquirer[1].mocks.update.mock.calls[1]).toEqual([ newState ]) expect(stateUpdateEventMock.mock.calls.length).toEqual(2) // current state expect(stateUpdateEventMock.mock.calls[1][0]).toEqual({ bee: 'boo', capabilities: { resumableUploads: false }, files: {}, currentUploads: {}, foo: 'bar', info: { isHidden: true, message: '', type: 'info' }, meta: {}, plugins: {}, totalProgress: 0 }) // new state expect(stateUpdateEventMock.mock.calls[1][1]).toEqual({ bee: 'boo', capabilities: { resumableUploads: false }, files: {}, currentUploads: {}, foo: 'baar', info: { isHidden: true, message: '', type: 'info' }, meta: {}, plugins: {}, totalProgress: 0 }) }) it('should get the state', () => { const core = new Core() core.setState({ foo: 'bar' }) expect(core.getState()).toEqual({ capabilities: { resumableUploads: false }, files: {}, currentUploads: {}, foo: 'bar', info: { isHidden: true, message: '', type: 'info' }, meta: {}, plugins: {}, totalProgress: 0 }) }) }) it('should reset when the reset method is called', () => { const core = new Core() // const corePauseEventMock = jest.fn() const coreCancelEventMock = jest.fn() const coreStateUpdateEventMock = jest.fn() core.on('cancel-all', coreCancelEventMock) core.on('state-update', coreStateUpdateEventMock) core.setState({ foo: 'bar', totalProgress: 30 }) core.reset() // expect(corePauseEventMock.mock.calls.length).toEqual(1) expect(coreCancelEventMock.mock.calls.length).toEqual(1) expect(coreStateUpdateEventMock.mock.calls.length).toEqual(2) expect(coreStateUpdateEventMock.mock.calls[1][1]).toEqual({ capabilities: { resumableUploads: false }, files: {}, currentUploads: {}, foo: 'bar', info: { isHidden: true, message: '', type: 'info' }, meta: {}, plugins: {}, totalProgress: 0 }) }) it('should close, reset and uninstall when the close method is called', () => { const core = new Core() core.use(AcquirerPlugin1) // const corePauseEventMock = jest.fn() const coreCancelEventMock = jest.fn() const coreStateUpdateEventMock = jest.fn() // core.on('pause-all', corePauseEventMock) core.on('cancel-all', coreCancelEventMock) core.on('state-update', coreStateUpdateEventMock) core.close() // expect(corePauseEventMock.mock.calls.length).toEqual(1) expect(coreCancelEventMock.mock.calls.length).toEqual(1) expect(coreStateUpdateEventMock.mock.calls.length).toEqual(1) expect(coreStateUpdateEventMock.mock.calls[0][1]).toEqual({ capabilities: { resumableUploads: false }, files: {}, currentUploads: {}, info: { isHidden: true, message: '', type: 'info' }, meta: {}, plugins: {}, totalProgress: 0 }) expect(core.plugins.acquirer[0].mocks.uninstall.mock.calls.length).toEqual( 1 ) }) describe('upload hooks', () => { it('should add data returned from upload hooks to the .upload() result', () => { const core = new Core() core.addPreProcessor((fileIDs, uploadID) => { core.addResultData(uploadID, { pre: 'ok' }) }) core.addPostProcessor((fileIDs, uploadID) => { core.addResultData(uploadID, { post: 'ok' }) }) core.addUploader((fileIDs, uploadID) => { core.addResultData(uploadID, { upload: 'ok' }) }) core.run() return core.upload().then((result) => { expect(result.pre).toBe('ok') expect(result.upload).toBe('ok') expect(result.post).toBe('ok') }) }) }) describe('preprocessors', () => { it('should add a preprocessor', () => { const core = new Core() const preprocessor = function () {} core.addPreProcessor(preprocessor) expect(core.preProcessors[0]).toEqual(preprocessor) }) it('should remove a preprocessor', () => { const core = new Core() const preprocessor1 = function () {} const preprocessor2 = function () {} const preprocessor3 = function () {} core.addPreProcessor(preprocessor1) core.addPreProcessor(preprocessor2) core.addPreProcessor(preprocessor3) expect(core.preProcessors.length).toEqual(3) core.removePreProcessor(preprocessor2) expect(core.preProcessors.length).toEqual(2) }) it('should execute all the preprocessors when uploading a file', () => { const core = new Core() const preprocessor1 = jest.fn() const preprocessor2 = jest.fn() core.addPreProcessor(preprocessor1) core.addPreProcessor(preprocessor2) return core .addFile({ source: 'jest', name: 'foo.jpg', type: 'image/jpeg', data: utils.dataURItoFile(sampleImageDataURI, {}) }) .then(() => core.upload()) .then(() => { const fileId = Object.keys(core.state.files)[0] expect(preprocessor1.mock.calls.length).toEqual(1) expect(preprocessor1.mock.calls[0][0].length).toEqual(1) expect(preprocessor1.mock.calls[0][0][0]).toEqual(fileId) expect(preprocessor2.mock.calls[0][0].length).toEqual(1) expect(preprocessor2.mock.calls[0][0][0]).toEqual(fileId) }) }) it('should update the file progress state when preprocess-progress event is fired', () => { const core = new Core() core.run() return core .addFile({ source: 'jest', name: 'foo.jpg', type: 'image/jpeg', data: utils.dataURItoFile(sampleImageDataURI, {}) }) .then(() => { const fileId = Object.keys(core.state.files)[0] core.emit('preprocess-progress', fileId, { mode: 'determinate', message: 'something', value: 0 }) expect(core.state.files[fileId].progress).toEqual({ percentage: 0, bytesUploaded: 0, bytesTotal: 17175, uploadComplete: false, uploadStarted: false, preprocess: { mode: 'determinate', message: 'something', value: 0 } }) }) }) it('should update the file progress state when preprocess-complete event is fired', () => { const core = new Core() core.run() return core .addFile({ source: 'jest', name: 'foo.jpg', type: 'image/jpeg', data: utils.dataURItoFile(sampleImageDataURI, {}) }) .then(() => { const fileId = Object.keys(core.state.files)[0] core.emit('preprocess-complete', fileId, { mode: 'determinate', message: 'something', value: 0 }) expect(core.state.files[fileId].progress).toEqual({ percentage: 0, bytesUploaded: 0, bytesTotal: 17175, uploadComplete: false, uploadStarted: false }) }) }) }) describe('postprocessors', () => { it('should add a postprocessor', () => { const core = new Core() const postprocessor = function () {} core.addPostProcessor(postprocessor) expect(core.postProcessors[0]).toEqual(postprocessor) }) it('should remove a postprocessor', () => { const core = new Core() const postprocessor1 = function () {} const postprocessor2 = function () {} const postprocessor3 = function () {} core.addPostProcessor(postprocessor1) core.addPostProcessor(postprocessor2) core.addPostProcessor(postprocessor3) expect(core.postProcessors.length).toEqual(3) core.removePostProcessor(postprocessor2) expect(core.postProcessors.length).toEqual(2) }) it('should execute all the postprocessors when uploading a file', () => { const core = new Core() const postprocessor1 = jest.fn() const postprocessor2 = jest.fn() core.addPostProcessor(postprocessor1) core.addPostProcessor(postprocessor2) return core .addFile({ source: 'jest', name: 'foo.jpg', type: 'image/jpeg', data: utils.dataURItoFile(sampleImageDataURI, {}) }) .then(() => core.upload()) .then(() => { expect(postprocessor1.mock.calls.length).toEqual(1) // const lastModifiedTime = new Date() // const fileId = 'foojpg' + lastModifiedTime.getTime() const fileId = 'uppy-foojpg-image' expect(postprocessor1.mock.calls[0][0].length).toEqual(1) expect(postprocessor1.mock.calls[0][0][0].substring(0, 17)).toEqual( fileId.substring(0, 17) ) expect(postprocessor2.mock.calls[0][0].length).toEqual(1) expect(postprocessor2.mock.calls[0][0][0].substring(0, 17)).toEqual( fileId.substring(0, 17) ) }) }) it('should update the file progress state when postprocess-progress event is fired', () => { const core = new Core() core.run() return core .addFile({ source: 'jest', name: 'foo.jpg', type: 'image/jpeg', data: utils.dataURItoFile(sampleImageDataURI, {}) }) .then(() => { const fileId = Object.keys(core.state.files)[0] core.emit('postprocess-progress', fileId, { mode: 'determinate', message: 'something', value: 0 }) expect(core.state.files[fileId].progress).toEqual({ percentage: 0, bytesUploaded: 0, bytesTotal: 17175, uploadComplete: false, uploadStarted: false, postprocess: { mode: 'determinate', message: 'something', value: 0 } }) }) }) it('should update the file progress state when postprocess-complete event is fired', () => { const core = new Core() core.run() return core .addFile({ source: 'jest', name: 'foo.jpg', type: 'image/jpeg', data: utils.dataURItoFile(sampleImageDataURI, {}) }) .then(() => { const fileId = Object.keys(core.state.files)[0] core.emit('postprocess-complete', fileId, { mode: 'determinate', message: 'something', value: 0 }) expect(core.state.files[fileId].progress).toEqual({ percentage: 0, bytesUploaded: 0, bytesTotal: 17175, uploadComplete: false, uploadStarted: false }) }) }) }) describe('uploaders', () => { it('should add an uploader', () => { const core = new Core() const uploader = function () {} core.addUploader(uploader) expect(core.uploaders[0]).toEqual(uploader) }) it('should remove an uploader', () => { const core = new Core() const uploader1 = function () {} const uploader2 = function () {} const uploader3 = function () {} core.addUploader(uploader1) core.addUploader(uploader2) core.addUploader(uploader3) expect(core.uploaders.length).toEqual(3) core.removeUploader(uploader2) expect(core.uploaders.length).toEqual(2) }) }) describe('adding a file', () => { it('should call onBeforeFileAdded if it was specified in the options when initailising the class', () => { const onBeforeFileAdded = jest.fn(value => { return Promise.resolve() }) const core = new Core({ onBeforeFileAdded }) return core .addFile({ source: 'jest', name: 'foo.jpg', type: 'image/jpeg', data: utils.dataURItoFile(sampleImageDataURI, {}) }) .then(() => { expect(onBeforeFileAdded.mock.calls.length).toEqual(1) expect(onBeforeFileAdded.mock.calls[0][0].name).toEqual('foo.jpg') expect(onBeforeFileAdded.mock.calls[0][1]).toEqual({}) }) }) it('should add a file', () => { const fileData = utils.dataURItoFile(sampleImageDataURI, {}) const fileAddedEventMock = jest.fn() const core = new Core() core.run() core.on('file-added', fileAddedEventMock) return core .addFile({ source: 'jest', name: 'foo.jpg', type: 'image/jpeg', data: fileData }) .then(() => { const fileId = Object.keys(core.state.files)[0] const newFile = { extension: 'jpg', id: fileId, isRemote: false, meta: { name: 'foo.jpg', type: 'image/jpeg' }, name: 'foo.jpg', preview: undefined, data: fileData, progress: { bytesTotal: 17175, bytesUploaded: 0, percentage: 0, uploadComplete: false, uploadStarted: false }, remote: '', size: 17175, source: 'jest', type: 'image/jpeg' } expect(core.state.files[fileId]).toEqual(newFile) expect(fileAddedEventMock.mock.calls[0][0]).toEqual(newFile) }) }) it('should not allow a file that does not meet the restrictions', () => { const core = new Core({ restrictions: { allowedFileTypes: ['image/gif'] } }) return core .addFile({ source: 'jest', name: 'foo.jpg', type: 'image/jpeg', data: utils.dataURItoFile(sampleImageDataURI, {}) }) .then(() => { throw new Error('File was allowed through') }) .catch(e => { expect(e.message).toEqual('File not allowed') }) }) it('should work with restriction errors that are not Error class instances', () => { const core = new Core({ onBeforeFileAdded () { return Promise.reject('a plain string') // eslint-disable-line prefer-promise-reject-errors } }) return expect(core.addFile({ source: 'jest', name: 'foo.jpg', type: 'image/jpeg', data: null })).rejects.toMatchObject(new Error('onBeforeFileAdded: a plain string')) }) }) describe('uploading a file', () => { it('should return a { successful, failed } pair containing file objects', () => { const core = new Core().run() core.addUploader((fileIDs) => Promise.resolve()) return Promise.all([ core.addFile({ source: 'jest', name: 'foo.jpg', type: 'image/jpeg', data: new Uint8Array() }), core.addFile({ source: 'jest', name: 'bar.jpg', type: 'image/jpeg', data: new Uint8Array() }) ]).then(() => { return expect(core.upload()).resolves.toMatchObject({ successful: [ { name: 'foo.jpg' }, { name: 'bar.jpg' } ], failed: [] }) }) }) it('should return files with errors in the { failed } key', () => { const core = new Core().run() core.addUploader((fileIDs) => { fileIDs.forEach((fileID) => { if (/bar/.test(core.getFile(fileID).name)) { core.emit('upload-error', fileID, new Error('This is bar and I do not like bar')) } }) return Promise.resolve() }) return Promise.all([ core.addFile({ source: 'jest', name: 'foo.jpg', type: 'image/jpeg', data: new Uint8Array() }), core.addFile({ source: 'jest', name: 'bar.jpg', type: 'image/jpeg', data: new Uint8Array() }) ]).then(() => { return expect(core.upload()).resolves.toMatchObject({ successful: [ { name: 'foo.jpg' } ], failed: [ { name: 'bar.jpg', error: 'This is bar and I do not like bar' } ] }) }) }) }) describe('removing a file', () => { it('should remove the file', () => { const fileRemovedEventMock = jest.fn() const core = new Core() core.on('file-removed', fileRemovedEventMock) core.run() return core .addFile({ source: 'jest', name: 'foo.jpg', type: 'image/jpeg', data: utils.dataURItoFile(sampleImageDataURI, {}) }) .then(() => { const fileId = Object.keys(core.state.files)[0] expect(Object.keys(core.state.files).length).toEqual(1) core.setState({ totalProgress: 50 }) core.removeFile(fileId) expect(Object.keys(core.state.files).length).toEqual(0) expect(fileRemovedEventMock.mock.calls[0][0]).toEqual(fileId) expect(core.state.totalProgress).toEqual(0) }) }) }) describe('restoring a file', () => { xit('should restore a file', () => {}) xit("should fail to restore a file if it doesn't exist", () => {}) }) describe('get a file', () => { it('should get the specified file', () => { const core = new Core() return core .addFile({ source: 'jest', name: 'foo.jpg', type: 'image/jpeg', data: utils.dataURItoFile(sampleImageDataURI, {}) }) .then(() => { const fileId = Object.keys(core.state.files)[0] expect(core.getFile(fileId).name).toEqual('foo.jpg') expect(core.getFile('non existant file')).toEqual(undefined) }) }) }) describe('meta data', () => { it('should set meta data by calling setMeta', () => { const core = new Core({ meta: { foo2: 'bar2' } }) core.setMeta({ foo: 'bar', bur: 'mur' }) core.setMeta({ boo: 'moo', bur: 'fur' }) expect(core.state.meta).toEqual({ foo: 'bar', foo2: 'bar2', boo: 'moo', bur: 'fur' }) }) it('should update meta data for a file by calling updateMeta', () => { const core = new Core() return core .addFile({ source: 'jest', name: 'foo.jpg', type: 'image/jpeg', data: utils.dataURItoFile(sampleImageDataURI, {}) }) .then(() => { const fileId = Object.keys(core.state.files)[0] core.setFileMeta(fileId, { foo: 'bar', bur: 'mur' }) core.setFileMeta(fileId, { boo: 'moo', bur: 'fur' }) expect(core.state.files[fileId].meta).toEqual({ name: 'foo.jpg', type: 'image/jpeg', foo: 'bar', bur: 'fur', boo: 'moo' }) }) }) }) describe('progress', () => { it('should calculate the progress of a file upload', () => { const core = new Core() return core .addFile({ source: 'jest', name: 'foo.jpg', type: 'image/jpeg', data: utils.dataURItoFile(sampleImageDataURI, {}) }) .then(() => { const fileId = Object.keys(core.state.files)[0] core._calculateProgress({ id: fileId, bytesUploaded: 12345, bytesTotal: 17175 }) expect(core.state.files[fileId].progress).toEqual({ percentage: 71, bytesUploaded: 12345, bytesTotal: 17175, uploadComplete: false, uploadStarted: false }) core._calculateProgress({ id: fileId, bytesUploaded: 17175, bytesTotal: 17175 }) expect(core.state.files[fileId].progress).toEqual({ percentage: 100, bytesUploaded: 17175, bytesTotal: 17175, uploadComplete: false, uploadStarted: false }) }) }) it('should calculate the total progress of all file uploads', () => { const core = new Core() return core .addFile({ source: 'jest', name: 'foo.jpg', type: 'image/jpeg', data: utils.dataURItoFile(sampleImageDataURI, {}) }) .then(() => { return core .addFile({ source: 'jest', name: 'foo2.jpg', type: 'image/jpeg', data: utils.dataURItoFile(sampleImageDataURI, {}) }) }).then(() => { const fileId1 = Object.keys(core.state.files)[0] const fileId2 = Object.keys(core.state.files)[1] core.state.files[fileId1].progress.uploadStarted = new Date() core.state.files[fileId2].progress.uploadStarted = new Date() core._calculateProgress({ id: fileId1, bytesUploaded: 12345, bytesTotal: 17175 }) core._calculateProgress({ id: fileId2, bytesUploaded: 10201, bytesTotal: 17175 }) core._calculateTotalProgress() expect(core.state.totalProgress).toEqual(65) }) }) it('should reset the progress', () => { const resetProgressEvent = jest.fn() const core = new Core() core.run() core.on('reset-progress', resetProgressEvent) return core .addFile({ source: 'jest', name: 'foo.jpg', type: 'image/jpeg', data: utils.dataURItoFile(sampleImageDataURI, {}) }) .then(() => { return core .addFile({ source: 'jest', name: 'foo2.jpg', type: 'image/jpeg', data: utils.dataURItoFile(sampleImageDataURI, {}) }) }).then(() => { const fileId1 = Object.keys(core.state.files)[0] const fileId2 = Object.keys(core.state.files)[1] core.state.files[fileId1].progress.uploadStarted = new Date() core.state.files[fileId2].progress.uploadStarted = new Date() core._calculateProgress({ id: fileId1, bytesUploaded: 12345, bytesTotal: 17175 }) core._calculateProgress({ id: fileId2, bytesUploaded: 10201, bytesTotal: 17175 }) core._calculateTotalProgress() expect(core.state.totalProgress).toEqual(65) core.resetProgress() expect(core.state.files[fileId1].progress).toEqual({ percentage: 0, bytesUploaded: 0, bytesTotal: 17175, uploadComplete: false, uploadStarted: false }) expect(core.state.files[fileId2].progress).toEqual({ percentage: 0, bytesUploaded: 0, bytesTotal: 17175, uploadComplete: false, uploadStarted: false }) expect(core.state.totalProgress).toEqual(0) expect(resetProgressEvent.mock.calls.length).toEqual(1) }) }) }) describe('checkRestrictions', () => { it('should enforce the maxNumberOfFiles rule', () => { const core = new Core({ autoProceed: false, restrictions: { maxNumberOfFiles: 1 } }) // add 2 files core.addFile({ source: 'jest', name: 'foo1.jpg', type: 'image/jpeg', data: utils.dataURItoFile(sampleImageDataURI, {}) }) return expect(core.addFile({ source: 'jest', name: 'foo2.jpg', type: 'image/jpeg', data: utils.dataURItoFile(sampleImageDataURI, {}) })).rejects.toMatchObject(new Error('File not allowed')).then(() => { expect(core.state.info.message).toEqual('You can only upload 1 file') }) }) xit('should enforce the minNumberOfFiles rule', () => {}) it('should enfore the allowedFileTypes rule', () => { const core = new Core({ autoProceed: false, restrictions: { allowedFileTypes: ['image/gif', 'image/png'] } }) return expect(core.addFile({ source: 'jest', name: 'foo2.jpg', type: 'image/jpeg', data: utils.dataURItoFile(sampleImageDataURI, {}) })).rejects.toMatchObject(new Error('File not allowed')).then(() => { expect(core.state.info.message).toEqual('You can only upload: image/gif, image/png') }) }) it('should enforce the maxFileSize rule', () => { const core = new Core({ autoProceed: false, restrictions: { maxFileSize: 1234 } }) return expect(core.addFile({ source: 'jest', name: 'foo.jpg', type: 'image/jpeg', data: utils.dataURItoFile(sampleImageDataURI, {}) })).rejects.toMatchObject(new Error('File not allowed')).then(() => { expect(core.state.info.message).toEqual('This file exceeds maximum allowed size of 1.2 KB') }) }) }) describe('actions', () => { it('should update the state when receiving the error event', () => { const core = new Core() core.run() core.emit('error', new Error('foooooo')) expect(core.state.error).toEqual('foooooo') }) it('should update the state when receiving the upload-error event', () => { const core = new Core() core.run() core.state.files['fileId'] = { name: 'filename' } core.emit('upload-error', 'fileId', new Error('this is the error')) expect(core.state.info).toEqual({'message': 'Failed to upload filename', 'details': 'this is the error', 'isHidden': false, 'type': 'error'}) }) it('should reset the error state when receiving the upload event', () => { const core = new Core() core.run() core.emit('error', { foo: 'bar' }) core.emit('upload') expect(core.state.error).toEqual(null) }) }) describe('updateOnlineStatus', () => { const RealNavigatorOnline = global.window.navigator.onLine function mockNavigatorOnline (status) { Object.defineProperty( global.window.navigator, 'onLine', { value: status, writable: true } ) } afterEach(() => { global.window.navigator.onLine = RealNavigatorOnline }) it('should emit the correct event based on whether there is a network connection', () => { const onlineEventMock = jest.fn() const offlineEventMock = jest.fn() const backOnlineEventMock = jest.fn() const core = new Core() core.on('is-offline', offlineEventMock) core.on('is-online', onlineEventMock) core.on('back-online', backOnlineEventMock) mockNavigatorOnline(true) core.updateOnlineStatus() expect(onlineEventMock.mock.calls.length).toEqual(1) expect(offlineEventMock.mock.calls.length).toEqual(0) expect(backOnlineEventMock.mock.calls.length).toEqual(0) mockNavigatorOnline(false) core.updateOnlineStatus() expect(onlineEventMock.mock.calls.length).toEqual(1) expect(offlineEventMock.mock.calls.length).toEqual(1) expect(backOnlineEventMock.mock.calls.length).toEqual(0) mockNavigatorOnline(true) core.updateOnlineStatus() expect(onlineEventMock.mock.calls.length).toEqual(2) expect(offlineEventMock.mock.calls.length).toEqual(1) expect(backOnlineEventMock.mock.calls.length).toEqual(1) }) }) describe('info', () => { it('should set a string based message to be displayed infinitely', () => { const infoVisibleEvent = jest.fn() const core = new Core() core.run() core.on('info-visible', infoVisibleEvent) core.info('This is the message', 'info', 0) expect(core.state.info).toEqual({ isHidden: false, type: 'info', message: 'This is the message', details: null }) expect(infoVisibleEvent.mock.calls.length).toEqual(1) expect(typeof core.infoTimeoutID).toEqual('undefined') }) it('should set a object based message to be displayed infinitely', () => { const infoVisibleEvent = jest.fn() const core = new Core() core.run() core.on('info-visible', infoVisibleEvent) core.info({ message: 'This is the message', details: { foo: 'bar' } }, 'warning', 0) expect(core.state.info).toEqual({ isHidden: false, type: 'warning', message: 'This is the message', details: { foo: 'bar' } }) expect(infoVisibleEvent.mock.calls.length).toEqual(1) expect(typeof core.infoTimeoutID).toEqual('undefined') }) it('should set an info message to be displayed for a period of time before hiding', (done) => { const infoVisibleEvent = jest.fn() const infoHiddenEvent = jest.fn() const core = new Core() core.run() core.on('info-visible', infoVisibleEvent) core.on('info-hidden', infoHiddenEvent) core.info('This is the message', 'info', 100) expect(typeof core.infoTimeoutID).toEqual('number') expect(infoHiddenEvent.mock.calls.length).toEqual(0) setTimeout(() => { expect(infoHiddenEvent.mock.calls.length).toEqual(1) expect(core.state.info).toEqual({ isHidden: true, type: 'info', message: 'This is the message', details: null }) done() }, 110) }) it('should hide an info message', () => { const infoVisibleEvent = jest.fn() const infoHiddenEvent = jest.fn() const core = new Core() core.run() core.on('info-visible', infoVisibleEvent) core.on('info-hidden', infoHiddenEvent) core.info('This is the message', 'info', 0) expect(typeof core.infoTimeoutID).toEqual('undefined') expect(infoHiddenEvent.mock.calls.length).toEqual(0) core.hideInfo() expect(infoHiddenEvent.mock.calls.length).toEqual(1) expect(core.state.info).toEqual({ isHidden: true, type: 'info', message: 'This is the message', details: null }) }) }) describe('createUpload', () => { it('should assign the specified files to a new upload', () => { const core = new Core() core.run() return core.addFile({ source: 'jest', name: 'foo.jpg', type: 'image/jpeg', data: utils.dataURItoFile(sampleImageDataURI, {}) }).then(() => { core._createUpload(Object.keys(core.state.files)) const uploadId = Object.keys(core.state.currentUploads)[0] const currentUploadsState = {} currentUploadsState[uploadId] = { fileIDs: Object.keys(core.state.files), step: 0, result: {} } expect(core.state.currentUploads).toEqual(currentUploadsState) }) }) }) })