Utils.test.js 24 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394
  1. const utils = require('./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('runPromiseSequence', () => {
  40. it('should run an array of promise-returning functions in sequence', () => {
  41. const promiseFn1 = jest.fn().mockReturnValue(Promise.resolve)
  42. const promiseFn2 = jest.fn().mockReturnValue(Promise.resolve)
  43. const promiseFn3 = jest.fn().mockReturnValue(Promise.resolve)
  44. return utils
  45. .runPromiseSequence([promiseFn1, promiseFn2, promiseFn3])
  46. .then(() => {
  47. expect(promiseFn1.mock.calls.length).toEqual(1)
  48. expect(promiseFn2.mock.calls.length).toEqual(1)
  49. expect(promiseFn3.mock.calls.length).toEqual(1)
  50. })
  51. })
  52. })
  53. describe('isTouchDevice', () => {
  54. const RealTouchStart = global.window.ontouchstart
  55. const RealMaxTouchPoints = global.navigator.maxTouchPoints
  56. beforeEach(() => {
  57. global.window.ontouchstart = true
  58. global.navigator.maxTouchPoints = 1
  59. })
  60. afterEach(() => {
  61. global.navigator.maxTouchPoints = RealMaxTouchPoints
  62. global.window.ontouchstart = RealTouchStart
  63. })
  64. xit("should return true if it's a touch device", () => {
  65. expect(utils.isTouchDevice()).toEqual(true)
  66. delete global.window.ontouchstart
  67. global.navigator.maxTouchPoints = false
  68. expect(utils.isTouchDevice()).toEqual(false)
  69. })
  70. })
  71. describe('getFileNameAndExtension', () => {
  72. it('should return the filename and extension as an array', () => {
  73. expect(utils.getFileNameAndExtension('fsdfjodsuf23rfw.jpg')).toEqual({
  74. name: 'fsdfjodsuf23rfw',
  75. extension: 'jpg'
  76. })
  77. })
  78. it('should handle invalid filenames', () => {
  79. expect(utils.getFileNameAndExtension('fsdfjodsuf23rfw')).toEqual({
  80. name: 'fsdfjodsuf23rfw',
  81. extension: undefined
  82. })
  83. })
  84. })
  85. describe('truncateString', () => {
  86. it('should truncate the string by the specified amount', () => {
  87. expect(utils.truncateString('abcdefghijkl', 10)).toEqual('abcde...jkl')
  88. expect(utils.truncateString('abcdefghijkl', 9)).toEqual('abcd...jkl')
  89. expect(utils.truncateString('abcdefghijkl', 8)).toEqual('abcd...kl')
  90. expect(utils.truncateString('abcdefghijkl', 7)).toEqual('abc...kl')
  91. expect(utils.truncateString('abcdefghijkl', 6)).toEqual('abc...kl')
  92. expect(utils.truncateString('abcdefghijkl', 5)).toEqual('ab...kl')
  93. expect(utils.truncateString('abcdefghijkl', 4)).toEqual('ab...l')
  94. expect(utils.truncateString('abcdefghijkl', 3)).toEqual('a...l')
  95. expect(utils.truncateString('abcdefghijkl', 2)).toEqual('a...l')
  96. expect(utils.truncateString('abcdefghijkl', 1)).toEqual('...l')
  97. })
  98. })
  99. describe('secondsToTime', () => {
  100. expect(utils.secondsToTime(60)).toEqual({
  101. hours: 0,
  102. minutes: 1,
  103. seconds: 0
  104. })
  105. expect(utils.secondsToTime(123)).toEqual({
  106. hours: 0,
  107. minutes: 2,
  108. seconds: 3
  109. })
  110. expect(utils.secondsToTime(1060)).toEqual({
  111. hours: 0,
  112. minutes: 17,
  113. seconds: 40
  114. })
  115. expect(utils.secondsToTime(123453460)).toEqual({
  116. hours: 20,
  117. minutes: 37,
  118. seconds: 40
  119. })
  120. })
  121. describe('getFileTypeExtension', () => {
  122. it('should return the filetype based on the specified mime type', () => {
  123. expect(utils.getFileTypeExtension('video/ogg')).toEqual('ogv')
  124. expect(utils.getFileTypeExtension('audio/ogg')).toEqual('ogg')
  125. expect(utils.getFileTypeExtension('video/webm')).toEqual('webm')
  126. expect(utils.getFileTypeExtension('audio/webm')).toEqual('webm')
  127. expect(utils.getFileTypeExtension('video/mp4')).toEqual('mp4')
  128. expect(utils.getFileTypeExtension('audio/mp3')).toEqual('mp3')
  129. expect(utils.getFileTypeExtension('foo/bar')).toEqual(null)
  130. })
  131. })
  132. describe('getFileType', () => {
  133. it('should trust the filetype if the file comes from a remote source', () => {
  134. const file = {
  135. isRemote: true,
  136. type: 'audio/webm',
  137. name: 'foo.webm'
  138. }
  139. expect(utils.getFileType(file)).toEqual('audio/webm')
  140. })
  141. it('should determine the filetype from the mimetype', () => {
  142. const file = {
  143. type: 'audio/webm',
  144. name: 'foo.webm',
  145. data: 'sdfsdfhq9efbicw'
  146. }
  147. expect(utils.getFileType(file)).toEqual('audio/webm')
  148. })
  149. it('should determine the filetype from the extension', () => {
  150. const fileMP3 = {
  151. name: 'foo.mp3',
  152. data: 'sdfsfhfh329fhwihs'
  153. }
  154. const fileYAML = {
  155. name: 'bar.yaml',
  156. data: 'sdfsfhfh329fhwihs'
  157. }
  158. const fileMKV = {
  159. name: 'bar.mkv',
  160. data: 'sdfsfhfh329fhwihs'
  161. }
  162. expect(utils.getFileType(fileMP3)).toEqual('audio/mp3')
  163. expect(utils.getFileType(fileYAML)).toEqual('text/yaml')
  164. expect(utils.getFileType(fileMKV)).toEqual('video/x-matroska')
  165. })
  166. it('should fail gracefully if unable to detect', () => {
  167. const file = {
  168. name: 'foobar',
  169. data: 'sdfsfhfh329fhwihs'
  170. }
  171. expect(utils.getFileType(file)).toEqual(null)
  172. })
  173. })
  174. describe('getArrayBuffer', () => {
  175. beforeEach(() => {
  176. global.FileReader = class FileReader {
  177. addEventListener (e, cb) {
  178. if (e === 'load') {
  179. this.loadCb = cb
  180. }
  181. if (e === 'error') {
  182. this.errorCb = cb
  183. }
  184. }
  185. readAsArrayBuffer (chunk) {
  186. this.loadCb({ target: { result: new ArrayBuffer(8) } })
  187. }
  188. }
  189. })
  190. afterEach(() => {
  191. global.FileReader = undefined
  192. })
  193. it('should return a promise that resolves with the specified chunk', () => {
  194. return utils.getArrayBuffer('abcde').then(buffer => {
  195. expect(typeof buffer).toEqual('object')
  196. expect(buffer.byteLength).toEqual(8)
  197. })
  198. })
  199. })
  200. describe('isPreviewSupported', () => {
  201. it('should return true for any filetypes that browsers can preview', () => {
  202. const supported = ['image/jpeg', 'image/gif', 'image/png', 'image/svg', 'image/svg+xml', 'image/bmp']
  203. supported.forEach(ext => {
  204. expect(utils.isPreviewSupported(ext)).toEqual(true)
  205. })
  206. expect(utils.isPreviewSupported('foo')).toEqual(false)
  207. })
  208. })
  209. describe('isObjectURL', () => {
  210. it('should return true if the specified url is an object url', () => {
  211. expect(utils.isObjectURL('blob:abc123')).toEqual(true)
  212. expect(utils.isObjectURL('kblob:abc123')).toEqual(false)
  213. expect(utils.isObjectURL('blob-abc123')).toEqual(false)
  214. expect(utils.isObjectURL('abc123')).toEqual(false)
  215. })
  216. })
  217. describe('dataURItoBlob', () => {
  218. it('should convert a data uri to a blob', () => {
  219. const blob = utils.dataURItoBlob(sampleImageDataURI, {})
  220. expect(blob instanceof Blob).toEqual(true)
  221. expect(blob.size).toEqual(9348)
  222. expect(blob.type).toEqual('image/jpeg')
  223. })
  224. })
  225. describe('dataURItoFile', () => {
  226. it('should convert a data uri to a file', () => {
  227. const file = utils.dataURItoFile(sampleImageDataURI, { name: 'foo.jpg' })
  228. expect(file instanceof File).toEqual(true)
  229. expect(file.size).toEqual(9348)
  230. expect(file.type).toEqual('image/jpeg')
  231. expect(file.name).toEqual('foo.jpg')
  232. })
  233. })
  234. describe('getSpeed', () => {
  235. it('should calculate the speed given a fileProgress object', () => {
  236. const dateNow = new Date()
  237. const date5SecondsAgo = new Date(dateNow.getTime() - 5 * 1000)
  238. const fileProgress = {
  239. bytesUploaded: 1024,
  240. uploadStarted: date5SecondsAgo
  241. }
  242. expect(Math.round(utils.getSpeed(fileProgress))).toEqual(Math.round(205))
  243. })
  244. })
  245. describe('getBytesRemaining', () => {
  246. it('should calculate the bytes remaining given a fileProgress object', () => {
  247. const fileProgress = {
  248. bytesUploaded: 1024,
  249. bytesTotal: 3096
  250. }
  251. expect(utils.getBytesRemaining(fileProgress)).toEqual(2072)
  252. })
  253. })
  254. describe('getETA', () => {
  255. it('should get the ETA remaining based on a fileProgress object', () => {
  256. const dateNow = new Date()
  257. const date5SecondsAgo = new Date(dateNow.getTime() - 5 * 1000)
  258. const fileProgress = {
  259. bytesUploaded: 1024,
  260. bytesTotal: 3096,
  261. uploadStarted: date5SecondsAgo
  262. }
  263. expect(utils.getETA(fileProgress)).toEqual(10.1)
  264. })
  265. })
  266. describe('prettyETA', () => {
  267. it('should convert the specified number of seconds to a pretty ETA', () => {
  268. expect(utils.prettyETA(0)).toEqual('0s')
  269. expect(utils.prettyETA(1.2)).toEqual('1s')
  270. expect(utils.prettyETA(1)).toEqual('1s')
  271. expect(utils.prettyETA(103)).toEqual('1m 43s')
  272. expect(utils.prettyETA(1034.9)).toEqual('17m 14s')
  273. expect(utils.prettyETA(103984.1)).toEqual('4h 53m 04s')
  274. })
  275. })
  276. describe('copyToClipboard', () => {
  277. xit('should copy the specified text to the clipboard', () => {})
  278. })
  279. describe('getSocketHost', () => {
  280. it('should get the host from the specified url', () => {
  281. expect(
  282. utils.getSocketHost('https://foo.bar/a/b/cd?e=fghi&l=k&m=n')
  283. ).toEqual('ws://foo.bar/a/b/cd?e=fghi&l=k&m=n')
  284. })
  285. })
  286. describe('settle', () => {
  287. it('should resolve even if all input promises reject', () => {
  288. return expect(
  289. utils.settle([
  290. Promise.reject(new Error('oops')),
  291. Promise.reject(new Error('this went wrong'))
  292. ])
  293. ).resolves.toMatchObject({
  294. successful: [],
  295. failed: [ new Error('oops'), new Error('this went wrong') ]
  296. })
  297. })
  298. it('should resolve with an object if some input promises resolve', () => {
  299. return expect(
  300. utils.settle([
  301. Promise.reject(new Error('rejected')),
  302. Promise.resolve('resolved'),
  303. Promise.resolve('also-resolved')
  304. ])
  305. ).resolves.toMatchObject({
  306. successful: ['resolved', 'also-resolved'],
  307. failed: [new Error('rejected')]
  308. })
  309. })
  310. })
  311. describe('limitPromises', () => {
  312. let pending = 0
  313. function fn () {
  314. pending++
  315. return new Promise((resolve) => setTimeout(resolve, 10))
  316. .then(() => pending--)
  317. }
  318. it('should run at most N promises at the same time', () => {
  319. const limit = utils.limitPromises(4)
  320. const fn2 = limit(fn)
  321. const result = Promise.all([
  322. fn2(), fn2(), fn2(), fn2(),
  323. fn2(), fn2(), fn2(), fn2(),
  324. fn2(), fn2()
  325. ])
  326. expect(pending).toBe(4)
  327. setTimeout(() => {
  328. expect(pending).toBe(4)
  329. }, 10)
  330. return result.then(() => {
  331. expect(pending).toBe(0)
  332. })
  333. })
  334. it('should accept Infinity as limit', () => {
  335. const limit = utils.limitPromises(Infinity)
  336. const fn2 = limit(fn)
  337. const result = Promise.all([
  338. fn2(), fn2(), fn2(), fn2(),
  339. fn2(), fn2(), fn2(), fn2(),
  340. fn2(), fn2()
  341. ])
  342. expect(pending).toBe(10)
  343. return result.then(() => {
  344. expect(pending).toBe(0)
  345. })
  346. })
  347. })
  348. })