mirror of https://github.com/Chocobozzz/PeerTube
				
				
				
			
		
			
				
	
	
		
			191 lines
		
	
	
		
			4.5 KiB
		
	
	
	
		
			TypeScript
		
	
	
			
		
		
	
	
			191 lines
		
	
	
		
			4.5 KiB
		
	
	
	
		
			TypeScript
		
	
	
| import 'multer'
 | |
| import { UploadFilesForCheck } from 'express'
 | |
| import { sep } from 'path'
 | |
| import validator from 'validator'
 | |
| import { isShortUUID, shortToUUID } from '@peertube/peertube-node-utils'
 | |
| 
 | |
| function exists (value: any) {
 | |
|   return value !== undefined && value !== null
 | |
| }
 | |
| 
 | |
| function isSafePath (p: string) {
 | |
|   return exists(p) &&
 | |
|     (p + '').split(sep).every(part => {
 | |
|       return [ '..' ].includes(part) === false
 | |
|     })
 | |
| }
 | |
| 
 | |
| function isSafeFilename (filename: string, extension?: string) {
 | |
|   const regex = extension
 | |
|     ? new RegExp(`^[a-z0-9-]+\\.${extension}$`)
 | |
|     : new RegExp(`^[a-z0-9-]+\\.[a-z0-9]{1,8}$`)
 | |
| 
 | |
|   return typeof filename === 'string' && !!filename.match(regex)
 | |
| }
 | |
| 
 | |
| function isSafePeerTubeFilenameWithoutExtension (filename: string) {
 | |
|   return filename.match(/^[a-z0-9-]+$/)
 | |
| }
 | |
| 
 | |
| function isArray (value: any): value is any[] {
 | |
|   return Array.isArray(value)
 | |
| }
 | |
| 
 | |
| function isNotEmptyIntArray (value: any) {
 | |
|   return Array.isArray(value) && value.every(v => validator.default.isInt('' + v)) && value.length !== 0
 | |
| }
 | |
| 
 | |
| function isNotEmptyStringArray (value: any) {
 | |
|   return Array.isArray(value) && value.every(v => typeof v === 'string' && v.length !== 0) && value.length !== 0
 | |
| }
 | |
| 
 | |
| function isArrayOf (value: any, validator: (value: any) => boolean) {
 | |
|   return isArray(value) && value.every(v => validator(v))
 | |
| }
 | |
| 
 | |
| function isDateValid (value: string) {
 | |
|   return exists(value) && validator.default.isISO8601(value)
 | |
| }
 | |
| 
 | |
| function isIdValid (value: string) {
 | |
|   return exists(value) && validator.default.isInt('' + value)
 | |
| }
 | |
| 
 | |
| function isUUIDValid (value: string) {
 | |
|   return exists(value) && validator.default.isUUID('' + value, 4)
 | |
| }
 | |
| 
 | |
| function areUUIDsValid (values: string[]) {
 | |
|   return isArray(values) && values.every(v => isUUIDValid(v))
 | |
| }
 | |
| 
 | |
| function isIdOrUUIDValid (value: string) {
 | |
|   return isIdValid(value) || isUUIDValid(value)
 | |
| }
 | |
| 
 | |
| function isBooleanValid (value: any) {
 | |
|   return typeof value === 'boolean' || (typeof value === 'string' && validator.default.isBoolean(value))
 | |
| }
 | |
| 
 | |
| function isIntOrNull (value: any) {
 | |
|   return value === null || validator.default.isInt('' + value)
 | |
| }
 | |
| 
 | |
| // ---------------------------------------------------------------------------
 | |
| 
 | |
| function isFileValid (options: {
 | |
|   files: UploadFilesForCheck
 | |
| 
 | |
|   maxSize: number | null
 | |
|   mimeTypeRegex: string | null
 | |
| 
 | |
|   field?: string
 | |
| 
 | |
|   optional?: boolean // Default false
 | |
| }) {
 | |
|   const { files, mimeTypeRegex, field, maxSize, optional = false } = options
 | |
| 
 | |
|   // Should have files
 | |
|   if (!files) return optional
 | |
| 
 | |
|   const fileArray = isArray(files)
 | |
|     ? files
 | |
|     : files[field]
 | |
| 
 | |
|   if (!fileArray || !isArray(fileArray) || fileArray.length === 0) {
 | |
|     return optional
 | |
|   }
 | |
| 
 | |
|   // The file exists
 | |
|   const file = fileArray[0]
 | |
|   if (!file?.originalname) return false
 | |
| 
 | |
|   // Check size
 | |
|   if ((maxSize !== null) && file.size > maxSize) return false
 | |
| 
 | |
|   if (mimeTypeRegex === null) return true
 | |
| 
 | |
|   return checkMimetypeRegex(file.mimetype, mimeTypeRegex)
 | |
| }
 | |
| 
 | |
| function checkMimetypeRegex (fileMimeType: string, mimeTypeRegex: string) {
 | |
|   return new RegExp(`^${mimeTypeRegex}$`, 'i').test(fileMimeType)
 | |
| }
 | |
| 
 | |
| // ---------------------------------------------------------------------------
 | |
| 
 | |
| function toCompleteUUID (value: string) {
 | |
|   if (isShortUUID(value)) {
 | |
|     try {
 | |
|       return shortToUUID(value)
 | |
|     } catch {
 | |
|       return ''
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   return value
 | |
| }
 | |
| 
 | |
| function toCompleteUUIDs (values: string[]) {
 | |
|   return values.map(v => toCompleteUUID(v))
 | |
| }
 | |
| 
 | |
| function toIntOrNull (value: string) {
 | |
|   const v = toValueOrNull(value)
 | |
| 
 | |
|   if (v === null || v === undefined) return v
 | |
|   if (typeof v === 'number') return v
 | |
| 
 | |
|   return validator.default.toInt('' + v)
 | |
| }
 | |
| 
 | |
| function toBooleanOrNull (value: any) {
 | |
|   const v = toValueOrNull(value)
 | |
| 
 | |
|   if (v === null || v === undefined) return v
 | |
|   if (typeof v === 'boolean') return v
 | |
| 
 | |
|   return validator.default.toBoolean('' + v)
 | |
| }
 | |
| 
 | |
| function toValueOrNull (value: string) {
 | |
|   if (value === 'null') return null
 | |
| 
 | |
|   return value
 | |
| }
 | |
| 
 | |
| function toIntArray (value: any) {
 | |
|   if (!value) return []
 | |
|   if (isArray(value) === false) return [ validator.default.toInt(value) ]
 | |
| 
 | |
|   return value.map(v => validator.default.toInt(v))
 | |
| }
 | |
| 
 | |
| // ---------------------------------------------------------------------------
 | |
| 
 | |
| export {
 | |
|   exists,
 | |
|   isArrayOf,
 | |
|   isNotEmptyIntArray,
 | |
|   isArray,
 | |
|   isIntOrNull,
 | |
|   isIdValid,
 | |
|   isSafePath,
 | |
|   isNotEmptyStringArray,
 | |
|   isUUIDValid,
 | |
|   toCompleteUUIDs,
 | |
|   toCompleteUUID,
 | |
|   isIdOrUUIDValid,
 | |
|   isDateValid,
 | |
|   toValueOrNull,
 | |
|   toBooleanOrNull,
 | |
|   isBooleanValid,
 | |
|   toIntOrNull,
 | |
|   areUUIDsValid,
 | |
|   toIntArray,
 | |
|   isFileValid,
 | |
|   isSafePeerTubeFilenameWithoutExtension,
 | |
|   isSafeFilename,
 | |
|   checkMimetypeRegex
 | |
| }
 |