2018-02-13 18:17:05 +01:00
|
|
|
import 'multer'
|
2021-05-10 11:13:41 +02:00
|
|
|
import { UploadFilesForCheck } from 'express'
|
2019-07-05 13:54:32 +02:00
|
|
|
import { sep } from 'path'
|
2021-05-10 11:13:41 +02:00
|
|
|
import validator from 'validator'
|
2023-07-31 14:34:36 +02:00
|
|
|
import { isShortUUID, shortToUUID } from '@peertube/peertube-node-utils'
|
2017-09-07 15:27:35 +02:00
|
|
|
|
2024-07-23 16:38:51 +02:00
|
|
|
export function exists (value: any) {
|
2016-07-31 20:58:43 +02:00
|
|
|
return value !== undefined && value !== null
|
|
|
|
}
|
|
|
|
|
2024-07-23 16:38:51 +02:00
|
|
|
export function isSafePath (p: string) {
|
2019-07-05 13:54:32 +02:00
|
|
|
return exists(p) &&
|
|
|
|
(p + '').split(sep).every(part => {
|
2019-07-05 15:28:49 +02:00
|
|
|
return [ '..' ].includes(part) === false
|
2019-07-05 13:54:32 +02:00
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2024-07-23 16:38:51 +02:00
|
|
|
export function isSafeFilename (filename: string, extension?: string) {
|
2023-05-04 15:29:34 +02:00
|
|
|
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)
|
2023-04-21 14:55:10 +02:00
|
|
|
}
|
|
|
|
|
2024-07-23 16:38:51 +02:00
|
|
|
export function isSafePeerTubeFilenameWithoutExtension (filename: string) {
|
2023-02-27 09:44:03 +01:00
|
|
|
return filename.match(/^[a-z0-9-]+$/)
|
|
|
|
}
|
|
|
|
|
2024-07-23 16:38:51 +02:00
|
|
|
// ---------------------------------------------------------------------------
|
|
|
|
|
|
|
|
export function isArray (value: any): value is any[] {
|
2016-07-31 20:58:43 +02:00
|
|
|
return Array.isArray(value)
|
|
|
|
}
|
|
|
|
|
2024-07-23 16:38:51 +02:00
|
|
|
export function isNotEmptyIntArray (value: any) {
|
2023-07-31 14:34:36 +02:00
|
|
|
return Array.isArray(value) && value.every(v => validator.default.isInt('' + v)) && value.length !== 0
|
2018-12-26 10:36:24 +01:00
|
|
|
}
|
|
|
|
|
2024-07-23 16:38:51 +02:00
|
|
|
export function isNotEmptyStringArray (value: any) {
|
2021-07-29 10:27:24 +02:00
|
|
|
return Array.isArray(value) && value.every(v => typeof v === 'string' && v.length !== 0) && value.length !== 0
|
|
|
|
}
|
|
|
|
|
2024-07-23 16:38:51 +02:00
|
|
|
export function hasArrayLength (value: unknown[], options: { min?: number, max?: number }) {
|
|
|
|
if (options.min !== undefined && value.length < options.min) return false
|
|
|
|
if (options.max !== undefined && value.length > options.max) return false
|
|
|
|
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
|
|
|
|
export function isArrayOf (value: any, validator: (value: any) => boolean) {
|
2019-01-29 08:37:25 +01:00
|
|
|
return isArray(value) && value.every(v => validator(v))
|
|
|
|
}
|
|
|
|
|
2024-07-23 16:38:51 +02:00
|
|
|
// ---------------------------------------------------------------------------
|
|
|
|
|
|
|
|
export function isDateValid (value: string) {
|
2023-07-31 14:34:36 +02:00
|
|
|
return exists(value) && validator.default.isISO8601(value)
|
2017-10-24 19:41:09 +02:00
|
|
|
}
|
|
|
|
|
2024-07-23 16:38:51 +02:00
|
|
|
export function isIdValid (value: string) {
|
2023-07-31 14:34:36 +02:00
|
|
|
return exists(value) && validator.default.isInt('' + value)
|
2017-10-24 19:41:09 +02:00
|
|
|
}
|
|
|
|
|
2024-07-23 16:38:51 +02:00
|
|
|
export function isUUIDValid (value: string) {
|
2023-07-31 14:34:36 +02:00
|
|
|
return exists(value) && validator.default.isUUID('' + value, 4)
|
2017-10-24 19:41:09 +02:00
|
|
|
}
|
|
|
|
|
2024-07-23 16:38:51 +02:00
|
|
|
export function areUUIDsValid (values: string[]) {
|
2021-07-28 16:40:21 +02:00
|
|
|
return isArray(values) && values.every(v => isUUIDValid(v))
|
|
|
|
}
|
|
|
|
|
2024-07-23 16:38:51 +02:00
|
|
|
export function isIdOrUUIDValid (value: string) {
|
2017-10-24 19:41:09 +02:00
|
|
|
return isIdValid(value) || isUUIDValid(value)
|
|
|
|
}
|
|
|
|
|
2024-07-23 16:38:51 +02:00
|
|
|
export function isBooleanValid (value: any) {
|
2023-07-31 14:34:36 +02:00
|
|
|
return typeof value === 'boolean' || (typeof value === 'string' && validator.default.isBoolean(value))
|
2018-01-03 10:12:36 +01:00
|
|
|
}
|
|
|
|
|
2024-07-23 16:38:51 +02:00
|
|
|
export function isIntOrNull (value: any) {
|
2023-07-31 14:34:36 +02:00
|
|
|
return value === null || validator.default.isInt('' + value)
|
2020-09-25 16:19:35 +02:00
|
|
|
}
|
|
|
|
|
2021-06-28 17:30:59 +02:00
|
|
|
// ---------------------------------------------------------------------------
|
2019-03-07 17:06:00 +01:00
|
|
|
|
2024-07-23 16:38:51 +02:00
|
|
|
export function isFileValid (options: {
|
2022-02-11 10:51:33 +01:00
|
|
|
files: UploadFilesForCheck
|
2020-12-08 21:16:10 +01:00
|
|
|
|
2022-02-11 10:51:33 +01:00
|
|
|
maxSize: number | null
|
|
|
|
mimeTypeRegex: string | null
|
2020-12-08 21:16:10 +01:00
|
|
|
|
2022-02-11 10:51:33 +01:00
|
|
|
field?: string
|
2020-12-08 21:16:10 +01:00
|
|
|
|
2022-02-11 10:51:33 +01:00
|
|
|
optional?: boolean // Default false
|
|
|
|
}) {
|
|
|
|
const { files, mimeTypeRegex, field, maxSize, optional = false } = options
|
2020-12-08 21:16:10 +01:00
|
|
|
|
2018-02-13 18:17:05 +01:00
|
|
|
// Should have files
|
|
|
|
if (!files) return optional
|
|
|
|
|
2022-02-11 10:51:33 +01:00
|
|
|
const fileArray = isArray(files)
|
|
|
|
? files
|
|
|
|
: files[field]
|
|
|
|
|
|
|
|
if (!fileArray || !isArray(fileArray) || fileArray.length === 0) {
|
2018-02-13 18:17:05 +01:00
|
|
|
return optional
|
|
|
|
}
|
|
|
|
|
2022-02-11 10:51:33 +01:00
|
|
|
// The file exists
|
2020-01-31 16:56:52 +01:00
|
|
|
const file = fileArray[0]
|
2022-11-15 15:00:19 +01:00
|
|
|
if (!file?.originalname) return false
|
2018-02-13 18:17:05 +01:00
|
|
|
|
2018-06-22 15:42:55 +02:00
|
|
|
// Check size
|
2018-07-25 22:01:25 +02:00
|
|
|
if ((maxSize !== null) && file.size > maxSize) return false
|
2018-06-22 15:42:55 +02:00
|
|
|
|
2022-02-11 10:51:33 +01:00
|
|
|
if (mimeTypeRegex === null) return true
|
|
|
|
|
|
|
|
return checkMimetypeRegex(file.mimetype, mimeTypeRegex)
|
|
|
|
}
|
|
|
|
|
2024-07-23 16:38:51 +02:00
|
|
|
export function checkMimetypeRegex (fileMimeType: string, mimeTypeRegex: string) {
|
2022-02-11 10:51:33 +01:00
|
|
|
return new RegExp(`^${mimeTypeRegex}$`, 'i').test(fileMimeType)
|
2018-02-13 18:17:05 +01:00
|
|
|
}
|
|
|
|
|
2016-07-31 20:58:43 +02:00
|
|
|
// ---------------------------------------------------------------------------
|
|
|
|
|
2024-07-23 16:38:51 +02:00
|
|
|
export function toCompleteUUID (value: string) {
|
2022-12-28 14:42:48 +01:00
|
|
|
if (isShortUUID(value)) {
|
|
|
|
try {
|
|
|
|
return shortToUUID(value)
|
|
|
|
} catch {
|
2023-07-26 10:48:30 +02:00
|
|
|
return ''
|
2022-12-28 14:42:48 +01:00
|
|
|
}
|
|
|
|
}
|
2021-06-28 17:30:59 +02:00
|
|
|
|
|
|
|
return value
|
|
|
|
}
|
|
|
|
|
2024-07-23 16:38:51 +02:00
|
|
|
export function toCompleteUUIDs (values: string[]) {
|
2021-07-28 16:40:21 +02:00
|
|
|
return values.map(v => toCompleteUUID(v))
|
|
|
|
}
|
|
|
|
|
2024-07-23 16:38:51 +02:00
|
|
|
export function toIntOrNull (value: string) {
|
2021-06-28 17:30:59 +02:00
|
|
|
const v = toValueOrNull(value)
|
|
|
|
|
|
|
|
if (v === null || v === undefined) return v
|
|
|
|
if (typeof v === 'number') return v
|
|
|
|
|
2023-07-31 14:34:36 +02:00
|
|
|
return validator.default.toInt('' + v)
|
2021-06-28 17:30:59 +02:00
|
|
|
}
|
|
|
|
|
2024-07-23 16:38:51 +02:00
|
|
|
export function toBooleanOrNull (value: any) {
|
2021-06-28 17:30:59 +02:00
|
|
|
const v = toValueOrNull(value)
|
|
|
|
|
|
|
|
if (v === null || v === undefined) return v
|
|
|
|
if (typeof v === 'boolean') return v
|
|
|
|
|
2023-07-31 14:34:36 +02:00
|
|
|
return validator.default.toBoolean('' + v)
|
2021-06-28 17:30:59 +02:00
|
|
|
}
|
|
|
|
|
2024-07-23 16:38:51 +02:00
|
|
|
export function toValueOrNull (value: string) {
|
2021-06-28 17:30:59 +02:00
|
|
|
if (value === 'null') return null
|
|
|
|
|
|
|
|
return value
|
|
|
|
}
|
|
|
|
|
2024-07-23 16:38:51 +02:00
|
|
|
export function toIntArray (value: any) {
|
2021-06-28 17:30:59 +02:00
|
|
|
if (!value) return []
|
2023-07-31 14:34:36 +02:00
|
|
|
if (isArray(value) === false) return [ validator.default.toInt(value) ]
|
2021-06-28 17:30:59 +02:00
|
|
|
|
2023-07-31 14:34:36 +02:00
|
|
|
return value.map(v => validator.default.toInt(v))
|
2021-06-28 17:30:59 +02:00
|
|
|
}
|