PeerTube/server/core/models/video/video-playlist-element.ts

396 lines
9.9 KiB
TypeScript
Raw Normal View History

2020-12-08 14:30:29 +01:00
import { AggregateOptions, Op, ScopeOptions, Sequelize, Transaction } from 'sequelize'
2019-02-26 10:55:40 +01:00
import {
AllowNull,
BelongsTo,
Column,
CreatedAt,
DataType,
Default,
ForeignKey,
Is,
IsInt,
2024-02-22 10:12:04 +01:00
Min, Table,
2019-02-26 10:55:40 +01:00
UpdatedAt
} from 'sequelize-typescript'
2020-01-07 14:56:07 +01:00
import validator from 'validator'
import { forceNumber } from '@peertube/peertube-core-utils'
import {
PlaylistElementObject,
VideoPlaylistElement,
VideoPlaylistElementType,
VideoPrivacy,
VideoPrivacyType
} from '@peertube/peertube-models'
import { MUserAccountId } from '@server/types/models/index.js'
2019-08-20 19:05:31 +02:00
import {
MVideoPlaylistElement,
MVideoPlaylistElementAP,
MVideoPlaylistElementFormattable,
2019-08-21 14:31:57 +02:00
MVideoPlaylistElementVideoUrlPlaylistPrivacy,
2024-02-12 10:47:52 +01:00
MVideoPlaylistElementVideoThumbnail,
MVideoPlaylistElementVideoUrl
} from '@server/types/models/video/video-playlist-element.js'
import { isActivityPubUrlValid } from '../../helpers/custom-validators/activitypub/misc.js'
import { CONSTRAINTS_FIELDS } from '../../initializers/constants.js'
import { AccountModel } from '../account/account.js'
2024-02-22 10:12:04 +01:00
import { SequelizeModel, getSort, throwIfNotValid } from '../shared/index.js'
import { VideoPlaylistModel } from './video-playlist.js'
import { ForAPIOptions, ScopeNames as VideoScopeNames, VideoModel } from './video.js'
2019-02-26 10:55:40 +01:00
@Table({
tableName: 'videoPlaylistElement',
indexes: [
{
fields: [ 'videoPlaylistId' ]
},
{
fields: [ 'videoId' ]
},
{
fields: [ 'url' ],
unique: true
}
]
})
2024-02-22 10:12:04 +01:00
export class VideoPlaylistElementModel extends SequelizeModel<VideoPlaylistElementModel> {
2019-02-26 10:55:40 +01:00
@CreatedAt
createdAt: Date
@UpdatedAt
updatedAt: Date
@AllowNull(true)
@Is('VideoPlaylistUrl', value => throwIfNotValid(value, isActivityPubUrlValid, 'url', true))
2019-02-26 10:55:40 +01:00
@Column(DataType.STRING(CONSTRAINTS_FIELDS.VIDEO_PLAYLISTS.URL.max))
url: string
@AllowNull(false)
@Default(1)
@IsInt
@Min(1)
@Column
position: number
@AllowNull(true)
@IsInt
@Min(0)
@Column
startTimestamp: number
@AllowNull(true)
@IsInt
@Min(0)
@Column
stopTimestamp: number
@ForeignKey(() => VideoPlaylistModel)
@Column
videoPlaylistId: number
@BelongsTo(() => VideoPlaylistModel, {
foreignKey: {
allowNull: false
},
onDelete: 'CASCADE'
})
VideoPlaylist: Awaited<VideoPlaylistModel>
2019-02-26 10:55:40 +01:00
@ForeignKey(() => VideoModel)
@Column
videoId: number
@BelongsTo(() => VideoModel, {
foreignKey: {
2019-07-31 15:57:32 +02:00
allowNull: true
2019-02-26 10:55:40 +01:00
},
2019-07-31 15:57:32 +02:00
onDelete: 'set null'
2019-02-26 10:55:40 +01:00
})
Video: Awaited<VideoModel>
2019-02-26 10:55:40 +01:00
2019-04-18 11:28:17 +02:00
static deleteAllOf (videoPlaylistId: number, transaction?: Transaction) {
2019-02-26 10:55:40 +01:00
const query = {
where: {
videoPlaylistId
},
transaction
}
return VideoPlaylistElementModel.destroy(query)
}
2019-07-31 15:57:32 +02:00
static listForApi (options: {
2020-01-31 16:56:52 +01:00
start: number
count: number
videoPlaylistId: number
serverAccount: AccountModel
2019-08-15 11:53:26 +02:00
user?: MUserAccountId
2019-07-31 15:57:32 +02:00
}) {
const accountIds = [ options.serverAccount.id ]
const videoScope: (ScopeOptions | string)[] = [
VideoScopeNames.WITH_BLACKLISTED
]
if (options.user) {
accountIds.push(options.user.Account.id)
videoScope.push({ method: [ VideoScopeNames.WITH_USER_HISTORY, options.user.id ] })
}
const forApiOptions: ForAPIOptions = { withAccountBlockerIds: accountIds }
videoScope.push({
method: [
VideoScopeNames.FOR_API, forApiOptions
]
})
const findQuery = {
offset: options.start,
limit: options.count,
order: getSort('position'),
where: {
videoPlaylistId: options.videoPlaylistId
},
include: [
{
model: VideoModel.scope(videoScope),
required: false
}
]
}
const countQuery = {
where: {
videoPlaylistId: options.videoPlaylistId
}
}
return Promise.all([
VideoPlaylistElementModel.count(countQuery),
VideoPlaylistElementModel.findAll(findQuery)
]).then(([ total, data ]) => ({ total, data }))
}
2020-12-08 14:30:29 +01:00
static loadByPlaylistAndVideo (videoPlaylistId: number, videoId: number): Promise<MVideoPlaylistElement> {
2019-02-26 10:55:40 +01:00
const query = {
where: {
videoPlaylistId,
videoId
}
}
return VideoPlaylistElementModel.findOne(query)
}
2020-12-08 14:30:29 +01:00
static loadById (playlistElementId: number | string): Promise<MVideoPlaylistElement> {
2019-07-31 15:57:32 +02:00
return VideoPlaylistElementModel.findByPk(playlistElementId)
}
static loadByPlaylistAndElementIdForAP (
2019-08-21 14:31:57 +02:00
playlistId: number | string,
playlistElementId: number
2020-12-08 14:30:29 +01:00
): Promise<MVideoPlaylistElementVideoUrlPlaylistPrivacy> {
const playlistWhere = validator.default.isUUID('' + playlistId)
? { uuid: playlistId }
: { id: playlistId }
2019-02-26 10:55:40 +01:00
const query = {
include: [
{
attributes: [ 'privacy' ],
model: VideoPlaylistModel.unscoped(),
where: playlistWhere
},
{
attributes: [ 'url' ],
model: VideoModel.unscoped()
2019-02-26 10:55:40 +01:00
}
],
where: {
id: playlistElementId
}
2019-02-26 10:55:40 +01:00
}
return VideoPlaylistElementModel.findOne(query)
}
2024-02-12 10:47:52 +01:00
static loadFirstElementWithVideoThumbnail (videoPlaylistId: number): Promise<MVideoPlaylistElementVideoThumbnail> {
const query = {
order: getSort('position'),
where: {
videoPlaylistId
},
include: [
{
model: VideoModel.scope(VideoScopeNames.WITH_THUMBNAILS),
required: true
}
]
}
return VideoPlaylistElementModel
.findOne(query)
}
// ---------------------------------------------------------------------------
2019-04-18 11:28:17 +02:00
static listUrlsOfForAP (videoPlaylistId: number, start: number, count: number, t?: Transaction) {
const getQuery = (forCount: boolean) => {
return {
attributes: forCount
? []
: [ 'url' ],
offset: start,
limit: count,
order: getSort('position'),
where: {
videoPlaylistId
},
transaction: t
}
2019-02-26 10:55:40 +01:00
}
return Promise.all([
VideoPlaylistElementModel.count(getQuery(true)),
VideoPlaylistElementModel.findAll(getQuery(false))
]).then(([ total, rows ]) => ({
total,
data: rows.map(e => e.url)
}))
2019-02-26 10:55:40 +01:00
}
2024-02-12 10:47:52 +01:00
static listElementsForExport (videoPlaylistId: number): Promise<MVideoPlaylistElementVideoUrl[]> {
const query = {
where: {
videoPlaylistId
},
include: [
{
2024-02-12 10:47:52 +01:00
attributes: [ 'url' ],
model: VideoModel.unscoped(),
required: true
}
2024-02-12 11:12:05 +01:00
],
order: getSort('position')
}
2024-02-12 10:47:52 +01:00
return VideoPlaylistElementModel.findAll(query)
}
2024-02-12 10:47:52 +01:00
// ---------------------------------------------------------------------------
2019-04-18 11:28:17 +02:00
static getNextPositionOf (videoPlaylistId: number, transaction?: Transaction) {
const query: AggregateOptions<number> = {
2019-02-26 10:55:40 +01:00
where: {
videoPlaylistId
},
transaction
}
return VideoPlaylistElementModel.max('position', query)
.then(position => position ? position + 1 : 1)
}
static reassignPositionOf (options: {
videoPlaylistId: number
firstPosition: number
endPosition: number
newPosition: number
2019-04-18 11:28:17 +02:00
transaction?: Transaction
}) {
const { videoPlaylistId, firstPosition, endPosition, newPosition, transaction } = options
2019-02-26 10:55:40 +01:00
const query = {
where: {
videoPlaylistId,
position: {
2019-04-18 11:28:17 +02:00
[Op.gte]: firstPosition,
[Op.lte]: endPosition
2019-02-26 10:55:40 +01:00
}
},
2019-02-28 11:14:26 +01:00
transaction,
validate: false // We use a literal to update the position
2019-02-26 10:55:40 +01:00
}
const positionQuery = Sequelize.literal(`${forceNumber(newPosition)} + "position" - ${forceNumber(firstPosition)}`)
return VideoPlaylistElementModel.update({ position: positionQuery }, query)
2019-02-26 10:55:40 +01:00
}
static increasePositionOf (
videoPlaylistId: number,
fromPosition: number,
by = 1,
2019-04-18 11:28:17 +02:00
transaction?: Transaction
2019-02-26 10:55:40 +01:00
) {
const query = {
where: {
videoPlaylistId,
position: {
2019-04-18 11:28:17 +02:00
[Op.gte]: fromPosition
2019-02-26 10:55:40 +01:00
}
},
transaction
}
return VideoPlaylistElementModel.increment({ position: by }, query)
}
toFormattedJSON (
this: MVideoPlaylistElementFormattable,
options: { accountId?: number } = {}
): VideoPlaylistElement {
return {
id: this.id,
position: this.position,
startTimestamp: this.startTimestamp,
stopTimestamp: this.stopTimestamp,
type: this.getType(options.accountId),
video: this.getVideoElement(options.accountId)
}
}
getType (this: MVideoPlaylistElementFormattable, accountId?: number) {
2019-07-31 15:57:32 +02:00
const video = this.Video
if (!video) return VideoPlaylistElementType.DELETED
// Owned video, don't filter it
if (accountId && video.VideoChannel.Account.id === accountId) return VideoPlaylistElementType.REGULAR
2020-03-20 09:55:57 +01:00
// Internal video?
if (video.privacy === VideoPrivacy.INTERNAL && accountId) return VideoPlaylistElementType.REGULAR
// Private, internal and password protected videos cannot be read without appropriate access (ownership, internal)
const protectedPrivacy = new Set<VideoPrivacyType>([ VideoPrivacy.PRIVATE, VideoPrivacy.INTERNAL, VideoPrivacy.PASSWORD_PROTECTED ])
if (protectedPrivacy.has(video.privacy)) {
return VideoPlaylistElementType.PRIVATE
}
2019-07-31 15:57:32 +02:00
if (video.isBlacklisted() || video.isBlocked()) return VideoPlaylistElementType.UNAVAILABLE
return VideoPlaylistElementType.REGULAR
}
getVideoElement (this: MVideoPlaylistElementFormattable, accountId?: number) {
2019-07-31 15:57:32 +02:00
if (!this.Video) return null
if (this.getType(accountId) !== VideoPlaylistElementType.REGULAR) return null
2019-07-31 15:57:32 +02:00
return this.Video.toFormattedJSON()
}
2019-08-21 14:31:57 +02:00
toActivityPubObject (this: MVideoPlaylistElementAP): PlaylistElementObject {
2019-02-26 10:55:40 +01:00
const base: PlaylistElementObject = {
id: this.url,
type: 'PlaylistElement',
url: this.Video?.url || null,
2019-02-26 10:55:40 +01:00
position: this.position
}
if (this.startTimestamp) base.startTimestamp = this.startTimestamp
if (this.stopTimestamp) base.stopTimestamp = this.stopTimestamp
return base
}
}