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
|
||
|
}
|