| 
									
										
										
										
											2022-08-12 19:44:10 -04:00
										 |  |  | // Copyright 2022 Signal Messenger, LLC
 | 
					
						
							|  |  |  | // SPDX-License-Identifier: AGPL-3.0-only
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | import MP4Box from 'mp4box'; | 
					
						
							|  |  |  | import { VIDEO_MP4, isVideo } from '../types/MIME'; | 
					
						
							| 
									
										
										
										
											2023-02-28 14:17:22 -08:00
										 |  |  | import { KIBIBYTE, getRenderDetailsForLimit } from '../types/AttachmentSize'; | 
					
						
							|  |  |  | import { explodePromise } from './explodePromise'; | 
					
						
							| 
									
										
										
										
											2022-08-12 19:44:10 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-28 14:17:22 -08:00
										 |  |  | const MAX_VIDEO_DURATION_IN_SEC = 30; | 
					
						
							| 
									
										
										
										
											2022-08-12 19:44:10 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | type MP4ArrayBuffer = ArrayBuffer & { fileStart: number }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | export enum ReasonVideoNotGood { | 
					
						
							|  |  |  |   AllGoodNevermind = 'AllGoodNevermind', | 
					
						
							|  |  |  |   CouldNotReadFile = 'CouldNotReadFile', | 
					
						
							|  |  |  |   TooLong = 'TooLong', | 
					
						
							| 
									
										
										
										
											2023-02-28 14:17:22 -08:00
										 |  |  |   TooBig = 'TooBig', | 
					
						
							| 
									
										
										
										
											2022-08-12 19:44:10 -04:00
										 |  |  |   UnsupportedCodec = 'UnsupportedCodec', | 
					
						
							|  |  |  |   UnsupportedContainer = 'UnsupportedContainer', | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | function createMp4ArrayBuffer(src: ArrayBuffer): MP4ArrayBuffer { | 
					
						
							|  |  |  |   const arrayBuffer = new ArrayBuffer(src.byteLength); | 
					
						
							|  |  |  |   new Uint8Array(arrayBuffer).set(new Uint8Array(src)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   (arrayBuffer as MP4ArrayBuffer).fileStart = 0; | 
					
						
							|  |  |  |   return arrayBuffer as MP4ArrayBuffer; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-28 14:17:22 -08:00
										 |  |  | export type IsVideoGoodForStoriesResultType = Readonly< | 
					
						
							|  |  |  |   | { | 
					
						
							|  |  |  |       reason: Exclude< | 
					
						
							|  |  |  |         ReasonVideoNotGood, | 
					
						
							|  |  |  |         ReasonVideoNotGood.TooLong | ReasonVideoNotGood.TooBig | 
					
						
							|  |  |  |       >; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   | { | 
					
						
							|  |  |  |       reason: ReasonVideoNotGood.TooLong; | 
					
						
							|  |  |  |       maxDurationInSec: number; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   | { | 
					
						
							|  |  |  |       reason: ReasonVideoNotGood.TooBig; | 
					
						
							|  |  |  |       renderDetails: ReturnType<typeof getRenderDetailsForLimit>; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | >; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | export type IsVideoGoodForStoriesOptionsType = Readonly<{ | 
					
						
							|  |  |  |   maxAttachmentSizeInKb: number; | 
					
						
							|  |  |  | }>; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-12 19:44:10 -04:00
										 |  |  | export async function isVideoGoodForStories( | 
					
						
							| 
									
										
										
										
											2023-02-28 14:17:22 -08:00
										 |  |  |   file: File, | 
					
						
							|  |  |  |   { maxAttachmentSizeInKb }: IsVideoGoodForStoriesOptionsType | 
					
						
							|  |  |  | ): Promise<IsVideoGoodForStoriesResultType> { | 
					
						
							| 
									
										
										
										
											2022-08-12 19:44:10 -04:00
										 |  |  |   if (!isVideo(file.type)) { | 
					
						
							| 
									
										
										
										
											2023-02-28 14:17:22 -08:00
										 |  |  |     return { reason: ReasonVideoNotGood.AllGoodNevermind }; | 
					
						
							| 
									
										
										
										
											2022-08-12 19:44:10 -04:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (file.type !== VIDEO_MP4) { | 
					
						
							| 
									
										
										
										
											2023-02-28 14:17:22 -08:00
										 |  |  |     return { reason: ReasonVideoNotGood.UnsupportedContainer }; | 
					
						
							| 
									
										
										
										
											2022-08-12 19:44:10 -04:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-28 14:17:22 -08:00
										 |  |  |   let src: ArrayBuffer; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     const { promise, resolve } = explodePromise<ArrayBuffer | undefined>(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const reader = new FileReader(); | 
					
						
							|  |  |  |     reader.onload = () => { | 
					
						
							|  |  |  |       if (reader.result) { | 
					
						
							|  |  |  |         resolve(reader.result as ArrayBuffer); | 
					
						
							|  |  |  |       } else { | 
					
						
							|  |  |  |         resolve(undefined); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  |     reader.readAsArrayBuffer(file); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const maybeSrc = await promise; | 
					
						
							|  |  |  |     if (maybeSrc === undefined) { | 
					
						
							|  |  |  |       return { reason: ReasonVideoNotGood.CouldNotReadFile }; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     src = maybeSrc; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2022-08-12 19:44:10 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-28 14:17:22 -08:00
										 |  |  |   if (src.byteLength / KIBIBYTE > maxAttachmentSizeInKb) { | 
					
						
							|  |  |  |     return { | 
					
						
							|  |  |  |       reason: ReasonVideoNotGood.TooBig, | 
					
						
							|  |  |  |       renderDetails: getRenderDetailsForLimit(maxAttachmentSizeInKb), | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   const arrayBuffer = createMp4ArrayBuffer(src); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   const { promise, resolve } = | 
					
						
							|  |  |  |     explodePromise<IsVideoGoodForStoriesResultType>(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   const mp4 = MP4Box.createFile(); | 
					
						
							|  |  |  |   mp4.onReady = info => { | 
					
						
							|  |  |  |     // mp4box returns a `duration` in `timescale` units
 | 
					
						
							|  |  |  |     const seconds = info.duration / info.timescale; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (seconds > MAX_VIDEO_DURATION_IN_SEC) { | 
					
						
							|  |  |  |       resolve({ | 
					
						
							|  |  |  |         reason: ReasonVideoNotGood.TooLong, | 
					
						
							|  |  |  |         maxDurationInSec: MAX_VIDEO_DURATION_IN_SEC, | 
					
						
							|  |  |  |       }); | 
					
						
							|  |  |  |       return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const codecs = /codecs="([\w,.]+)"/.exec(info.mime); | 
					
						
							|  |  |  |     if (!codecs || !codecs[1]) { | 
					
						
							|  |  |  |       resolve({ reason: ReasonVideoNotGood.UnsupportedCodec }); | 
					
						
							|  |  |  |       return; | 
					
						
							| 
									
										
										
										
											2022-08-12 19:44:10 -04:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2023-02-28 14:17:22 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     const isH264 = codecs[1].split(',').some(codec => codec.startsWith('avc1')); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (!isH264) { | 
					
						
							|  |  |  |       resolve({ reason: ReasonVideoNotGood.UnsupportedCodec }); | 
					
						
							|  |  |  |       return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     resolve({ reason: ReasonVideoNotGood.AllGoodNevermind }); | 
					
						
							|  |  |  |   }; | 
					
						
							|  |  |  |   mp4.appendBuffer(arrayBuffer); | 
					
						
							|  |  |  |   try { | 
					
						
							|  |  |  |     return await promise; | 
					
						
							|  |  |  |   } finally { | 
					
						
							|  |  |  |     mp4.flush(); | 
					
						
							| 
									
										
										
										
											2022-08-12 19:44:10 -04:00
										 |  |  |   } | 
					
						
							|  |  |  | } |