Utils.test.js 24 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406
  1. import utils from './Utils'
  2. const sampleImageDataURI =
  3. ''
  4. describe('core/utils', () => {
  5. describe('generateFileID', () => {
  6. it('should take the filename object and produce a lowercase file id made up of uppy- prefix, file name (cleaned up to be lowercase, letters and numbers only), type, size and lastModified date', () => {
  7. const fileObj = {
  8. name: 'fOo0Fi@£$.jpg',
  9. type: 'image/jpeg',
  10. data: {
  11. lastModified: 1498510508000,
  12. size: 2271173
  13. }
  14. }
  15. expect(utils.generateFileID(fileObj)).toEqual(
  16. 'uppy-foo0fijpg-image/jpeg-2271173-1498510508000'
  17. )
  18. })
  19. })
  20. describe('toArray', () => {
  21. it('should convert a array-like object into an array', () => {
  22. const obj = {
  23. '0': 'zero',
  24. '1': 'one',
  25. '2': 'two',
  26. '3': 'three',
  27. '4': 'four',
  28. length: 5
  29. }
  30. expect(utils.toArray(obj)).toEqual([
  31. 'zero',
  32. 'one',
  33. 'two',
  34. 'three',
  35. 'four'
  36. ])
  37. })
  38. })
  39. describe('every', () => {
  40. it('should return true if every array element passes predicate', () => {
  41. const arr = ['foo', 'bar', 'boo', 'moo']
  42. const predicateFn = jest.fn().mockReturnValue(true)
  43. expect(utils.every(arr, predicateFn)).toEqual(true)
  44. })
  45. it('should return false if one or more array elements fails predicate', () => {
  46. const arr = ['foo', 'bar', 'boo', 'moo']
  47. const predicateFn = jest
  48. .fn()
  49. .mockReturnValue(true)
  50. .mockReturnValueOnce(false)
  51. .mockReturnValueOnce(true)
  52. .mockReturnValueOnce(true)
  53. expect(utils.every(arr, predicateFn)).toEqual(false)
  54. })
  55. })
  56. describe('flatten', () => {
  57. it('should flatten nested arrays', () => {
  58. const nestedArrays = [['a', 'b', 'c'], ['d', 'e']]
  59. expect(utils.flatten(nestedArrays)).toEqual(['a', 'b', 'c', 'd', 'e'])
  60. })
  61. })
  62. describe('groupBy', () => {
  63. it('should partiton an array by a grouping function, returning a map', () => {
  64. const arr = [0, 10, 3, 20, 2, 4, 9]
  65. const groupingFn = val => {
  66. return val < 5
  67. }
  68. expect(utils.groupBy(arr, groupingFn)).toEqual(
  69. new Map([[true, [0, 3, 2, 4]], [false, [10, 20, 9]]])
  70. )
  71. })
  72. })
  73. describe('extend', () => {
  74. it('should merge the passed in objects', () => {
  75. const obj1 = { foo: 'bar', boo: 'moo' }
  76. const obj2 = { root: 'toot' }
  77. expect(utils.extend(obj1, obj2)).toEqual({
  78. foo: 'bar',
  79. boo: 'moo',
  80. root: 'toot'
  81. })
  82. })
  83. })
  84. describe('runPromiseSequence', () => {
  85. it('should run an array of promise-returning functions in sequence', () => {
  86. const promiseFn1 = jest.fn().mockReturnValue(Promise.resolve)
  87. const promiseFn2 = jest.fn().mockReturnValue(Promise.resolve)
  88. const promiseFn3 = jest.fn().mockReturnValue(Promise.resolve)
  89. return utils
  90. .runPromiseSequence([promiseFn1, promiseFn2, promiseFn3])
  91. .then(() => {
  92. expect(promiseFn1.mock.calls.length).toEqual(1)
  93. expect(promiseFn2.mock.calls.length).toEqual(1)
  94. expect(promiseFn3.mock.calls.length).toEqual(1)
  95. })
  96. })
  97. })
  98. describe('supportsMediaRecorder', () => {
  99. it('should return true if MediaRecorder is supported', () => {
  100. global.MediaRecorder = () => {}
  101. global.MediaRecorder.prototype.start = () => {}
  102. expect(utils.supportsMediaRecorder()).toEqual(true)
  103. })
  104. it('should return false if MediaRecorder is not supported', () => {
  105. global.MediaRecorder = undefined
  106. expect(utils.supportsMediaRecorder()).toEqual(false)
  107. global.MediaRecorder = () => {}
  108. expect(utils.supportsMediaRecorder()).toEqual(false)
  109. global.MediaRecorder.prototype.foo = () => {}
  110. expect(utils.supportsMediaRecorder()).toEqual(false)
  111. })
  112. })
  113. describe('isTouchDevice', () => {
  114. it("should return true if it's a touch device", () => {
  115. global.window.ontouchstart = () => {}
  116. expect(utils.isTouchDevice()).toEqual(true)
  117. delete global.window.ontouchstart
  118. global.navigator.maxTouchPoints = () => {}
  119. expect(utils.isTouchDevice()).toEqual(true)
  120. })
  121. })
  122. describe('getFileNameAndExtension', () => {
  123. it('should return the filename and extension as an array', () => {
  124. expect(utils.getFileNameAndExtension('fsdfjodsuf23rfw.jpg')).toEqual([
  125. 'fsdfjodsuf23rfw',
  126. 'jpg'
  127. ])
  128. })
  129. it('should handle invalid filenames', () => {
  130. expect(utils.getFileNameAndExtension('fsdfjodsuf23rfw')).toEqual([
  131. 'fsdfjodsuf23rfw',
  132. undefined
  133. ])
  134. })
  135. })
  136. describe('truncateString', () => {
  137. it('should truncate the string by the specified amount', () => {
  138. expect(utils.truncateString('abcdefghijkl', 10)).toEqual('abcde...jkl')
  139. expect(utils.truncateString('abcdefghijkl', 9)).toEqual('abcd...jkl')
  140. expect(utils.truncateString('abcdefghijkl', 8)).toEqual('abcd...kl')
  141. expect(utils.truncateString('abcdefghijkl', 7)).toEqual('abc...kl')
  142. expect(utils.truncateString('abcdefghijkl', 6)).toEqual('abc...kl')
  143. expect(utils.truncateString('abcdefghijkl', 5)).toEqual('ab...kl')
  144. expect(utils.truncateString('abcdefghijkl', 4)).toEqual('ab...l')
  145. expect(utils.truncateString('abcdefghijkl', 3)).toEqual('a...l')
  146. expect(utils.truncateString('abcdefghijkl', 2)).toEqual('a...l')
  147. expect(utils.truncateString('abcdefghijkl', 1)).toEqual('...l')
  148. })
  149. })
  150. describe('secondsToTime', () => {
  151. expect(utils.secondsToTime(60)).toEqual({
  152. hours: 0,
  153. minutes: 1,
  154. seconds: 0
  155. })
  156. expect(utils.secondsToTime(123)).toEqual({
  157. hours: 0,
  158. minutes: 2,
  159. seconds: 3
  160. })
  161. expect(utils.secondsToTime(1060)).toEqual({
  162. hours: 0,
  163. minutes: 17,
  164. seconds: 40
  165. })
  166. expect(utils.secondsToTime(123453460)).toEqual({
  167. hours: 20,
  168. minutes: 37,
  169. seconds: 40
  170. })
  171. })
  172. describe('getFileTypeExtension', () => {
  173. it('should return the filetype based on the specified mime type', () => {
  174. expect(utils.getFileTypeExtension('video/ogg')).toEqual('ogv')
  175. expect(utils.getFileTypeExtension('audio/ogg')).toEqual('ogg')
  176. expect(utils.getFileTypeExtension('video/webm')).toEqual('webm')
  177. expect(utils.getFileTypeExtension('audio/webm')).toEqual('webm')
  178. expect(utils.getFileTypeExtension('video/mp4')).toEqual('mp4')
  179. expect(utils.getFileTypeExtension('audio/mp3')).toEqual('mp3')
  180. expect(utils.getFileTypeExtension('foo/bar')).toEqual(null)
  181. })
  182. })
  183. describe('getFileType', () => {
  184. beforeEach(() => {
  185. global.FileReader = class FileReader {
  186. addEventListener (e, cb) {
  187. if (e === 'load') {
  188. this.loadCb = cb
  189. }
  190. if (e === 'error') {
  191. this.errorCb = cb
  192. }
  193. }
  194. readAsArrayBuffer (chunk) {
  195. this.loadCb({ target: { result: new ArrayBuffer(8) } })
  196. }
  197. }
  198. })
  199. afterEach(() => {
  200. global.FileReader = undefined
  201. })
  202. it('should trust the filetype if the file comes from a remote source', () => {
  203. const file = {
  204. isRemote: true,
  205. type: 'audio/webm',
  206. name: 'foo.webm'
  207. }
  208. return utils.getFileType(file).then(r => {
  209. expect(r).toEqual(['audio', 'webm'])
  210. })
  211. })
  212. it('should determine the filetype from the mimetype', () => {
  213. const file = {
  214. type: 'audio/webm',
  215. name: 'foo.webm',
  216. data: 'sdfsdfhq9efbicw'
  217. }
  218. return utils.getFileType(file).then(r => {
  219. expect(r).toEqual(['audio', 'webm'])
  220. })
  221. })
  222. it('should determine the filetype from the extension', () => {
  223. const file = {
  224. name: 'foo.mp3',
  225. data: 'sdfsfhfh329fhwihs'
  226. }
  227. return utils.getFileType(file).then(r => {
  228. expect(r).toEqual(['audio', 'mp3'])
  229. })
  230. })
  231. it('should fail gracefully if unable to detect', () => {
  232. const file = {
  233. name: 'foobar',
  234. data: 'sdfsfhfh329fhwihs'
  235. }
  236. return utils.getFileType(file).then(r => {
  237. expect(r).toEqual(['', ''])
  238. })
  239. })
  240. })
  241. describe('getArrayBuffer', () => {
  242. beforeEach(() => {
  243. global.FileReader = class FileReader {
  244. addEventListener (e, cb) {
  245. if (e === 'load') {
  246. this.loadCb = cb
  247. }
  248. if (e === 'error') {
  249. this.errorCb = cb
  250. }
  251. }
  252. readAsArrayBuffer (chunk) {
  253. this.loadCb({ target: { result: new ArrayBuffer(8) } })
  254. }
  255. }
  256. })
  257. afterEach(() => {
  258. global.FileReader = undefined
  259. })
  260. it('should return a promise that resolves with the specified chunk', () => {
  261. return utils.getArrayBuffer('abcde').then(buffer => {
  262. expect(typeof buffer).toEqual('object')
  263. expect(buffer.byteLength).toEqual(8)
  264. })
  265. })
  266. })
  267. describe('isPreviewSupported', () => {
  268. it('should return true for any filetypes that browsers can preview', () => {
  269. const supported = ['jpeg', 'gif', 'png', 'svg', 'svg+xml', 'bmp']
  270. supported.forEach(ext => {
  271. expect(utils.isPreviewSupported(ext)).toEqual(true)
  272. })
  273. expect(utils.isPreviewSupported('foo')).toEqual(false)
  274. })
  275. })
  276. describe('isObjectURL', () => {
  277. it('should return true if the specified url is an object url', () => {
  278. expect(utils.isObjectURL('blob:abc123')).toEqual(true)
  279. expect(utils.isObjectURL('kblob:abc123')).toEqual(false)
  280. expect(utils.isObjectURL('blob-abc123')).toEqual(false)
  281. expect(utils.isObjectURL('abc123')).toEqual(false)
  282. })
  283. })
  284. describe('createThumbnail', () => {
  285. xit(
  286. 'should create a thumbnail of the specified image at the specified width',
  287. () => {}
  288. )
  289. })
  290. describe('dataURItoBlob', () => {
  291. it('should convert a data uri to a blob', () => {
  292. const blob = utils.dataURItoBlob(sampleImageDataURI, {})
  293. expect(blob instanceof Blob).toEqual(true)
  294. expect(blob.size).toEqual(9348)
  295. expect(blob.type).toEqual('image/jpeg')
  296. })
  297. })
  298. describe('dataURItoFile', () => {
  299. it('should convert a data uri to a file', () => {
  300. const file = utils.dataURItoFile(sampleImageDataURI, { name: 'foo.jpg' })
  301. expect(file instanceof File).toEqual(true)
  302. expect(file.size).toEqual(9348)
  303. expect(file.type).toEqual('image/jpeg')
  304. expect(file.name).toEqual('foo.jpg')
  305. })
  306. })
  307. describe('getSpeed', () => {
  308. it('should calculate the speed given a fileProgress object', () => {
  309. const dateNow = new Date()
  310. const date5SecondsAgo = new Date(dateNow.getTime() - 5 * 1000)
  311. const fileProgress = {
  312. bytesUploaded: 1024,
  313. uploadStarted: date5SecondsAgo
  314. }
  315. expect(Math.round(utils.getSpeed(fileProgress))).toEqual(Math.round(205))
  316. })
  317. })
  318. describe('getBytesRemaining', () => {
  319. it('should calculate the bytes remaining given a fileProgress object', () => {
  320. const fileProgress = {
  321. bytesUploaded: 1024,
  322. bytesTotal: 3096
  323. }
  324. expect(utils.getBytesRemaining(fileProgress)).toEqual(2072)
  325. })
  326. })
  327. describe('getETA', () => {
  328. it('should get the ETA remaining based on a fileProgress object', () => {
  329. const dateNow = new Date()
  330. const date5SecondsAgo = new Date(dateNow.getTime() - 5 * 1000)
  331. const fileProgress = {
  332. bytesUploaded: 1024,
  333. bytesTotal: 3096,
  334. uploadStarted: date5SecondsAgo
  335. }
  336. expect(utils.getETA(fileProgress)).toEqual(10.1)
  337. })
  338. })
  339. describe('prettyETA', () => {
  340. it('should convert the specified number of seconds to a pretty ETA', () => {
  341. expect(utils.prettyETA(0)).toEqual('0s')
  342. expect(utils.prettyETA(1.2)).toEqual('1s')
  343. expect(utils.prettyETA(1)).toEqual('1s')
  344. expect(utils.prettyETA(103)).toEqual('1m 43s')
  345. expect(utils.prettyETA(1034.9)).toEqual('17m 14s')
  346. expect(utils.prettyETA(103984.1)).toEqual('4h 53m 04s')
  347. })
  348. })
  349. describe('copyToClipboard', () => {
  350. xit('should copy the specified text to the clipboard', () => {})
  351. })
  352. describe('getSocketHost', () => {
  353. it('should get the host from the specified url', () => {
  354. expect(
  355. utils.getSocketHost('https://foo.bar/a/b/cd?e=fghi&l=k&m=n')
  356. ).toEqual('ws://foo.bar/a/b/cd?e=fghi&l=k&m=n')
  357. })
  358. })
  359. })