uppy
Version:
Extensible JavaScript file upload widget with support for drag&drop, resumable uploads, previews, restrictions, file processing/encoding, remote providers like Instagram, Dropbox, Google Drive, S3 and more :dog:
850 lines (771 loc) • 61.2 kB
JavaScript
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'
jest.mock('cuid', () => {
return () => 'cjd09qwxb000dlql4tp4doz8h'
})
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]
const file = core.getFile(fileId)
core.emit('preprocess-progress', file, {
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]
const file = core.state.files[fileID]
core.emit('preprocess-complete', file, {
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]
const file = core.getFile(fileId)
core.emit('postprocess-progress', file, {
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]
const file = core.state.files[fileId]
core.emit('postprocess-complete', file, {
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('You can only upload: image/gif')
})
})
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('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) => {
const file = core.getFile(fileID)
if (/bar/.test(file.name)) {
core.emit('upload-error', file, 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' }
]
})
})
})
it('should only upload files that are not already assigned to another upload id', () => {
const core = new Core().run()
core.store.state.currentUploads = {
upload1: {
fileIDs: ['uppy-file1jpg-image/jpeg', 'uppy-file2jpg-image/jpeg', 'uppy-file3jpg-image/jpeg']
},
upload2: {
fileIDs: ['uppy-file4jpg-image/jpeg', 'uppy-file5jpg-image/jpeg', 'uppy-file6jpg-image/jpeg']
}
}
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() }),
core.addFile({ source: 'file3', name: 'file3.jpg', type: 'image/jpeg', data: new Uint8Array() })
]).then(() => {
return core.upload()
}).then((result) => {
expect(result).toMatchSnapshot()
})
})
})
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
})
const file = core.getFile(fileId)
core.removeFile(fileId)
expect(Object.keys(core.state.files).length).toEqual(0)
expect(fileRemovedEventMock.mock.calls[0][0]).toEqual(file)
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]
const file = core.getFile(fileId)
core._calculateProgress(file, {
bytesUploaded: 12345,
bytesTotal: 17175
})
expect(core.state.files[fileId].progress).toEqual({
percentage: 71,
bytesUploaded: 12345,
bytesTotal: 17175,
uploadComplete: false,
uploadStarted: false
})
core._calculateProgress(file, {
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: '