cb0eda5602
* Initial test implementation of Podcast RSS This is a pretty simple implementation to add support for The Podcast Namespace in RSS -- instead of affecting the existing RSS implementation, this adds a new UI option. I attempted to retain compatibility with the rest of the RSS feed implementation as much as possible and have created a temporary fork of the "pfeed" library to support this effort. * Update to pfeed-podcast 1.2.2 * Initial test implementation of Podcast RSS This is a pretty simple implementation to add support for The Podcast Namespace in RSS -- instead of affecting the existing RSS implementation, this adds a new UI option. I attempted to retain compatibility with the rest of the RSS feed implementation as much as possible and have created a temporary fork of the "pfeed" library to support this effort. * Update to pfeed-podcast 1.2.2 * Initial test implementation of Podcast RSS This is a pretty simple implementation to add support for The Podcast Namespace in RSS -- instead of affecting the existing RSS implementation, this adds a new UI option. I attempted to retain compatibility with the rest of the RSS feed implementation as much as possible and have created a temporary fork of the "pfeed" library to support this effort. * Update to pfeed-podcast 1.2.2 * Add correct feed image to RSS channel * Prefer HLS videos for podcast RSS Remove video/stream titles, add optional height attribute to podcast RSS * Prefix podcast RSS images with root server URL * Add optional video query support to include captions * Add transcripts & person images to podcast RSS feed * Prefer webseed/webtorrent files over HLS fragmented mp4s * Experimentally adding podcast fields to basic config page * Add validation for new basic config fields * Don't include "content" in podcast feed, use full description for "description" * Initial test implementation of Podcast RSS This is a pretty simple implementation to add support for The Podcast Namespace in RSS -- instead of affecting the existing RSS implementation, this adds a new UI option. I attempted to retain compatibility with the rest of the RSS feed implementation as much as possible and have created a temporary fork of the "pfeed" library to support this effort. * Update to pfeed-podcast 1.2.2 * Add correct feed image to RSS channel * Prefer HLS videos for podcast RSS Remove video/stream titles, add optional height attribute to podcast RSS * Prefix podcast RSS images with root server URL * Add optional video query support to include captions * Add transcripts & person images to podcast RSS feed * Prefer webseed/webtorrent files over HLS fragmented mp4s * Experimentally adding podcast fields to basic config page * Add validation for new basic config fields * Don't include "content" in podcast feed, use full description for "description" * Add medium/socialInteract to podcast RSS feeds. Use HTML for description * Change base production image to bullseye, install prosody in image * Add liveItem and trackers to Podcast RSS feeds Remove height from alternateEnclosure, replaced with title. * Clear Podcast RSS feed cache when live streams start/end * Upgrade to Node 16 * Refactor clearCacheRoute to use ApiCache * Remove unnecessary type hint * Update dockerfile to node 16, install python-is-python2 * Use new file paths for captions/playlists * Fix legacy videos in RSS after migration to object storage * Improve method of identifying non-fragmented mp4s in podcast RSS feeds * Don't include fragmented MP4s in podcast RSS feeds * Add experimental support for podcast:categories on the podcast RSS item * Fix undefined category when no videos exist Allows for empty feeds to exist (important for feeds that might only go live) * Add support for podcast:locked -- user has to opt in to show their email * Use comma for podcast:categories delimiter * Make cache clearing async * Fix merge, temporarily test with pfeed-podcast * Syntax changes * Add EXT_MIMETYPE constants for captions * Update & fix tests, fix enclosure mimetypes, remove admin email * Add test for podacst:socialInteract * Add filters hooks for podcast customTags * Remove showdown, updated to pfeed-podcast 6.1.2 * Add 'action:api.live-video.state.updated' hook * Avoid assigning undefined category to podcast feeds * Remove nvmrc * Remove comment * Remove unused podcast config * Remove more unused podcast config * Fix MChannelAccountDefault type hint missed in merge * Remove extra line * Re-add newline in config * Fix lint errors for isEmailPublic * Fix thumbnails in podcast feeds * Requested changes based on review * Provide podcast rss 2.0 only on video channels * Misc cleanup for a less messy PR * Lint fixes * Remove pfeed-podcast * Add peertube version to new hooks * Don't use query include, remove TODO * Remove film medium hack * Clear podcast rss cache before video/channel update hooks * Clear podcast rss cache before video uploaded/deleted hooks * Refactor podcast feed cache clearing * Set correct person name from video channel * Styling * Fix tests --------- Co-authored-by: Chocobozzz <me@florianbigard.com>
123 lines
3.6 KiB
TypeScript
123 lines
3.6 KiB
TypeScript
import validator from 'validator'
|
|
import { UserRole } from '@shared/models'
|
|
import { isEmailEnabled } from '../../initializers/config'
|
|
import { CONSTRAINTS_FIELDS, NSFW_POLICY_TYPES } from '../../initializers/constants'
|
|
import { exists, isArray, isBooleanValid } from './misc'
|
|
|
|
const USERS_CONSTRAINTS_FIELDS = CONSTRAINTS_FIELDS.USERS
|
|
|
|
function isUserPasswordValid (value: string) {
|
|
return validator.isLength(value, USERS_CONSTRAINTS_FIELDS.PASSWORD)
|
|
}
|
|
|
|
function isUserPasswordValidOrEmpty (value: string) {
|
|
// Empty password is only possible if emailing is enabled.
|
|
if (value === '') return isEmailEnabled()
|
|
|
|
return isUserPasswordValid(value)
|
|
}
|
|
|
|
function isUserVideoQuotaValid (value: string) {
|
|
return exists(value) && validator.isInt(value + '', USERS_CONSTRAINTS_FIELDS.VIDEO_QUOTA)
|
|
}
|
|
|
|
function isUserVideoQuotaDailyValid (value: string) {
|
|
return exists(value) && validator.isInt(value + '', USERS_CONSTRAINTS_FIELDS.VIDEO_QUOTA_DAILY)
|
|
}
|
|
|
|
function isUserUsernameValid (value: string) {
|
|
return exists(value) &&
|
|
validator.matches(value, new RegExp(`^[a-z0-9_]+([a-z0-9_.-]+[a-z0-9_]+)?$`)) &&
|
|
validator.isLength(value, USERS_CONSTRAINTS_FIELDS.USERNAME)
|
|
}
|
|
|
|
function isUserDisplayNameValid (value: string) {
|
|
return value === null || (exists(value) && validator.isLength(value, CONSTRAINTS_FIELDS.USERS.NAME))
|
|
}
|
|
|
|
function isUserDescriptionValid (value: string) {
|
|
return value === null || (exists(value) && validator.isLength(value, CONSTRAINTS_FIELDS.USERS.DESCRIPTION))
|
|
}
|
|
|
|
function isUserEmailVerifiedValid (value: any) {
|
|
return isBooleanValid(value)
|
|
}
|
|
|
|
const nsfwPolicies = new Set(Object.values(NSFW_POLICY_TYPES))
|
|
function isUserNSFWPolicyValid (value: any) {
|
|
return exists(value) && nsfwPolicies.has(value)
|
|
}
|
|
|
|
function isUserP2PEnabledValid (value: any) {
|
|
return isBooleanValid(value)
|
|
}
|
|
|
|
function isUserVideosHistoryEnabledValid (value: any) {
|
|
return isBooleanValid(value)
|
|
}
|
|
|
|
function isUserAutoPlayVideoValid (value: any) {
|
|
return isBooleanValid(value)
|
|
}
|
|
|
|
function isUserVideoLanguages (value: any) {
|
|
return value === null || (isArray(value) && value.length < CONSTRAINTS_FIELDS.USERS.VIDEO_LANGUAGES.max)
|
|
}
|
|
|
|
function isUserAdminFlagsValid (value: any) {
|
|
return exists(value) && validator.isInt('' + value)
|
|
}
|
|
|
|
function isUserBlockedValid (value: any) {
|
|
return isBooleanValid(value)
|
|
}
|
|
|
|
function isUserAutoPlayNextVideoValid (value: any) {
|
|
return isBooleanValid(value)
|
|
}
|
|
|
|
function isUserAutoPlayNextVideoPlaylistValid (value: any) {
|
|
return isBooleanValid(value)
|
|
}
|
|
|
|
function isUserEmailPublicValid (value: any) {
|
|
return isBooleanValid(value)
|
|
}
|
|
|
|
function isUserNoModal (value: any) {
|
|
return isBooleanValid(value)
|
|
}
|
|
|
|
function isUserBlockedReasonValid (value: any) {
|
|
return value === null || (exists(value) && validator.isLength(value, CONSTRAINTS_FIELDS.USERS.BLOCKED_REASON))
|
|
}
|
|
|
|
function isUserRoleValid (value: any) {
|
|
return exists(value) && validator.isInt('' + value) && [ UserRole.ADMINISTRATOR, UserRole.MODERATOR, UserRole.USER ].includes(value)
|
|
}
|
|
|
|
// ---------------------------------------------------------------------------
|
|
|
|
export {
|
|
isUserVideosHistoryEnabledValid,
|
|
isUserBlockedValid,
|
|
isUserPasswordValid,
|
|
isUserPasswordValidOrEmpty,
|
|
isUserVideoLanguages,
|
|
isUserBlockedReasonValid,
|
|
isUserRoleValid,
|
|
isUserVideoQuotaValid,
|
|
isUserVideoQuotaDailyValid,
|
|
isUserUsernameValid,
|
|
isUserAdminFlagsValid,
|
|
isUserEmailVerifiedValid,
|
|
isUserNSFWPolicyValid,
|
|
isUserP2PEnabledValid,
|
|
isUserAutoPlayVideoValid,
|
|
isUserAutoPlayNextVideoValid,
|
|
isUserAutoPlayNextVideoPlaylistValid,
|
|
isUserDisplayNameValid,
|
|
isUserDescriptionValid,
|
|
isUserEmailPublicValid,
|
|
isUserNoModal
|
|
}
|