123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643 |
- import { FindOptions, Includeable, literal, Op, ScopeOptions } from 'sequelize'
- import {
- AllowNull,
- BeforeDestroy,
- BelongsTo,
- Column,
- CreatedAt,
- DataType,
- Default,
- DefaultScope,
- ForeignKey,
- HasMany,
- Is,
- Model,
- Scopes,
- Sequelize,
- Table,
- UpdatedAt
- } from 'sequelize-typescript'
- import { MAccountActor } from '@server/types/models'
- import { ActivityPubActor } from '../../../shared/models/activitypub'
- import { VideoChannel, VideoChannelSummary } from '../../../shared/models/videos'
- import {
- isVideoChannelDescriptionValid,
- isVideoChannelNameValid,
- isVideoChannelSupportValid
- } from '../../helpers/custom-validators/video-channels'
- import { CONSTRAINTS_FIELDS, WEBSERVER } from '../../initializers/constants'
- import { sendDeleteActor } from '../../lib/activitypub/send'
- import {
- MChannelAccountDefault,
- MChannelActor,
- MChannelActorAccountDefaultVideos,
- MChannelAP,
- MChannelFormattable,
- MChannelSummaryFormattable
- } from '../../types/models/video'
- import { AccountModel, ScopeNames as AccountModelScopeNames, SummaryOptions as AccountSummaryOptions } from '../account/account'
- import { ActorModel, unusedActorAttributesForAPI } from '../activitypub/actor'
- import { ActorFollowModel } from '../activitypub/actor-follow'
- import { AvatarModel } from '../avatar/avatar'
- import { ServerModel } from '../server/server'
- import { buildServerIdsFollowedBy, buildTrigramSearchIndex, createSimilarityAttribute, getSort, throwIfNotValid } from '../utils'
- import { VideoModel } from './video'
- import { VideoPlaylistModel } from './video-playlist'
- export enum ScopeNames {
- FOR_API = 'FOR_API',
- SUMMARY = 'SUMMARY',
- WITH_ACCOUNT = 'WITH_ACCOUNT',
- WITH_ACTOR = 'WITH_ACTOR',
- WITH_VIDEOS = 'WITH_VIDEOS',
- WITH_STATS = 'WITH_STATS'
- }
- type AvailableForListOptions = {
- actorId: number
- search?: string
- }
- type AvailableWithStatsOptions = {
- daysPrior: number
- }
- export type SummaryOptions = {
- actorRequired?: boolean // Default: true
- withAccount?: boolean // Default: false
- withAccountBlockerIds?: number[]
- }
- @DefaultScope(() => ({
- include: [
- {
- model: ActorModel,
- required: true
- }
- ]
- }))
- @Scopes(() => ({
- [ScopeNames.FOR_API]: (options: AvailableForListOptions) => {
- // Only list local channels OR channels that are on an instance followed by actorId
- const inQueryInstanceFollow = buildServerIdsFollowedBy(options.actorId)
- return {
- include: [
- {
- attributes: {
- exclude: unusedActorAttributesForAPI
- },
- model: ActorModel,
- where: {
- [Op.or]: [
- {
- serverId: null
- },
- {
- serverId: {
- [Op.in]: Sequelize.literal(inQueryInstanceFollow)
- }
- }
- ]
- }
- },
- {
- model: AccountModel,
- required: true,
- include: [
- {
- attributes: {
- exclude: unusedActorAttributesForAPI
- },
- model: ActorModel, // Default scope includes avatar and server
- required: true
- }
- ]
- }
- ]
- }
- },
- [ScopeNames.SUMMARY]: (options: SummaryOptions = {}) => {
- const include: Includeable[] = [
- {
- attributes: [ 'id', 'preferredUsername', 'url', 'serverId', 'avatarId' ],
- model: ActorModel.unscoped(),
- required: options.actorRequired ?? true,
- include: [
- {
- attributes: [ 'host' ],
- model: ServerModel.unscoped(),
- required: false
- },
- {
- model: AvatarModel.unscoped(),
- required: false
- }
- ]
- }
- ]
- const base: FindOptions = {
- attributes: [ 'id', 'name', 'description', 'actorId' ]
- }
- if (options.withAccount === true) {
- include.push({
- model: AccountModel.scope({
- method: [ AccountModelScopeNames.SUMMARY, { withAccountBlockerIds: options.withAccountBlockerIds } as AccountSummaryOptions ]
- }),
- required: true
- })
- }
- base.include = include
- return base
- },
- [ScopeNames.WITH_ACCOUNT]: {
- include: [
- {
- model: AccountModel,
- required: true
- }
- ]
- },
- [ScopeNames.WITH_ACTOR]: {
- include: [
- ActorModel
- ]
- },
- [ScopeNames.WITH_VIDEOS]: {
- include: [
- VideoModel
- ]
- },
- [ScopeNames.WITH_STATS]: (options: AvailableWithStatsOptions = { daysPrior: 30 }) => {
- const daysPrior = parseInt(options.daysPrior + '', 10)
- return {
- attributes: {
- include: [
- [
- literal('(SELECT COUNT(*) FROM "video" WHERE "channelId" = "VideoChannelModel"."id")'),
- 'videosCount'
- ],
- [
- literal(
- '(' +
- `SELECT string_agg(concat_ws('|', t.day, t.views), ',') ` +
- 'FROM ( ' +
- 'WITH ' +
- 'days AS ( ' +
- `SELECT generate_series(date_trunc('day', now()) - '${daysPrior} day'::interval, ` +
- `date_trunc('day', now()), '1 day'::interval) AS day ` +
- ') ' +
- 'SELECT days.day AS day, COALESCE(SUM("videoView".views), 0) AS views ' +
- 'FROM days ' +
- 'LEFT JOIN (' +
- '"videoView" INNER JOIN "video" ON "videoView"."videoId" = "video"."id" ' +
- 'AND "video"."channelId" = "VideoChannelModel"."id"' +
- `) ON date_trunc('day', "videoView"."startDate") = date_trunc('day', days.day) ` +
- 'GROUP BY day ' +
- 'ORDER BY day ' +
- ') t' +
- ')'
- ),
- 'viewsPerDay'
- ]
- ]
- }
- }
- }
- }))
- @Table({
- tableName: 'videoChannel',
- indexes: [
- buildTrigramSearchIndex('video_channel_name_trigram', 'name'),
- {
- fields: [ 'accountId' ]
- },
- {
- fields: [ 'actorId' ]
- }
- ]
- })
- export class VideoChannelModel extends Model {
- @AllowNull(false)
- @Is('VideoChannelName', value => throwIfNotValid(value, isVideoChannelNameValid, 'name'))
- @Column
- name: string
- @AllowNull(true)
- @Default(null)
- @Is('VideoChannelDescription', value => throwIfNotValid(value, isVideoChannelDescriptionValid, 'description', true))
- @Column(DataType.STRING(CONSTRAINTS_FIELDS.VIDEO_CHANNELS.DESCRIPTION.max))
- description: string
- @AllowNull(true)
- @Default(null)
- @Is('VideoChannelSupport', value => throwIfNotValid(value, isVideoChannelSupportValid, 'support', true))
- @Column(DataType.STRING(CONSTRAINTS_FIELDS.VIDEO_CHANNELS.SUPPORT.max))
- support: string
- @CreatedAt
- createdAt: Date
- @UpdatedAt
- updatedAt: Date
- @ForeignKey(() => ActorModel)
- @Column
- actorId: number
- @BelongsTo(() => ActorModel, {
- foreignKey: {
- allowNull: false
- },
- onDelete: 'cascade'
- })
- Actor: ActorModel
- @ForeignKey(() => AccountModel)
- @Column
- accountId: number
- @BelongsTo(() => AccountModel, {
- foreignKey: {
- allowNull: false
- },
- hooks: true
- })
- Account: AccountModel
- @HasMany(() => VideoModel, {
- foreignKey: {
- name: 'channelId',
- allowNull: false
- },
- onDelete: 'CASCADE',
- hooks: true
- })
- Videos: VideoModel[]
- @HasMany(() => VideoPlaylistModel, {
- foreignKey: {
- allowNull: true
- },
- onDelete: 'CASCADE',
- hooks: true
- })
- VideoPlaylists: VideoPlaylistModel[]
- @BeforeDestroy
- static async sendDeleteIfOwned (instance: VideoChannelModel, options) {
- if (!instance.Actor) {
- instance.Actor = await instance.$get('Actor', { transaction: options.transaction })
- }
- await ActorFollowModel.removeFollowsOf(instance.Actor.id, options.transaction)
- if (instance.Actor.isOwned()) {
- return sendDeleteActor(instance.Actor, options.transaction)
- }
- return undefined
- }
- static countByAccount (accountId: number) {
- const query = {
- where: {
- accountId
- }
- }
- return VideoChannelModel.count(query)
- }
- static listForApi (parameters: {
- actorId: number
- start: number
- count: number
- sort: string
- }) {
- const { actorId } = parameters
- const query = {
- offset: parameters.start,
- limit: parameters.count,
- order: getSort(parameters.sort)
- }
- return VideoChannelModel
- .scope({
- method: [ ScopeNames.FOR_API, { actorId } as AvailableForListOptions ]
- })
- .findAndCountAll(query)
- .then(({ rows, count }) => {
- return { total: count, data: rows }
- })
- }
- static listLocalsForSitemap (sort: string): Promise<MChannelActor[]> {
- const query = {
- attributes: [ ],
- offset: 0,
- order: getSort(sort),
- include: [
- {
- attributes: [ 'preferredUsername', 'serverId' ],
- model: ActorModel.unscoped(),
- where: {
- serverId: null
- }
- }
- ]
- }
- return VideoChannelModel
- .unscoped()
- .findAll(query)
- }
- static searchForApi (options: {
- actorId: number
- search: string
- start: number
- count: number
- sort: string
- }) {
- const attributesInclude = []
- const escapedSearch = VideoModel.sequelize.escape(options.search)
- const escapedLikeSearch = VideoModel.sequelize.escape('%' + options.search + '%')
- attributesInclude.push(createSimilarityAttribute('VideoChannelModel.name', options.search))
- const query = {
- attributes: {
- include: attributesInclude
- },
- offset: options.start,
- limit: options.count,
- order: getSort(options.sort),
- where: {
- [Op.or]: [
- Sequelize.literal(
- 'lower(immutable_unaccent("VideoChannelModel"."name")) % lower(immutable_unaccent(' + escapedSearch + '))'
- ),
- Sequelize.literal(
- 'lower(immutable_unaccent("VideoChannelModel"."name")) LIKE lower(immutable_unaccent(' + escapedLikeSearch + '))'
- )
- ]
- }
- }
- return VideoChannelModel
- .scope({
- method: [ ScopeNames.FOR_API, { actorId: options.actorId } as AvailableForListOptions ]
- })
- .findAndCountAll(query)
- .then(({ rows, count }) => {
- return { total: count, data: rows }
- })
- }
- static listByAccount (options: {
- accountId: number
- start: number
- count: number
- sort: string
- withStats?: boolean
- search?: string
- }) {
- const escapedSearch = VideoModel.sequelize.escape(options.search)
- const escapedLikeSearch = VideoModel.sequelize.escape('%' + options.search + '%')
- const where = options.search
- ? {
- [Op.or]: [
- Sequelize.literal(
- 'lower(immutable_unaccent("VideoChannelModel"."name")) % lower(immutable_unaccent(' + escapedSearch + '))'
- ),
- Sequelize.literal(
- 'lower(immutable_unaccent("VideoChannelModel"."name")) LIKE lower(immutable_unaccent(' + escapedLikeSearch + '))'
- )
- ]
- }
- : null
- const query = {
- offset: options.start,
- limit: options.count,
- order: getSort(options.sort),
- include: [
- {
- model: AccountModel,
- where: {
- id: options.accountId
- },
- required: true
- }
- ],
- where
- }
- const scopes: string | ScopeOptions | (string | ScopeOptions)[] = [ ScopeNames.WITH_ACTOR ]
- if (options.withStats === true) {
- scopes.push({
- method: [ ScopeNames.WITH_STATS, { daysPrior: 30 } as AvailableWithStatsOptions ]
- })
- }
- return VideoChannelModel
- .scope(scopes)
- .findAndCountAll(query)
- .then(({ rows, count }) => {
- return { total: count, data: rows }
- })
- }
- static loadByIdAndPopulateAccount (id: number): Promise<MChannelAccountDefault> {
- return VideoChannelModel.unscoped()
- .scope([ ScopeNames.WITH_ACTOR, ScopeNames.WITH_ACCOUNT ])
- .findByPk(id)
- }
- static loadByIdAndAccount (id: number, accountId: number): Promise<MChannelAccountDefault> {
- const query = {
- where: {
- id,
- accountId
- }
- }
- return VideoChannelModel.unscoped()
- .scope([ ScopeNames.WITH_ACTOR, ScopeNames.WITH_ACCOUNT ])
- .findOne(query)
- }
- static loadAndPopulateAccount (id: number): Promise<MChannelAccountDefault> {
- return VideoChannelModel.unscoped()
- .scope([ ScopeNames.WITH_ACTOR, ScopeNames.WITH_ACCOUNT ])
- .findByPk(id)
- }
- static loadByUrlAndPopulateAccount (url: string): Promise<MChannelAccountDefault> {
- const query = {
- include: [
- {
- model: ActorModel,
- required: true,
- where: {
- url
- }
- }
- ]
- }
- return VideoChannelModel
- .scope([ ScopeNames.WITH_ACCOUNT ])
- .findOne(query)
- }
- static loadByNameWithHostAndPopulateAccount (nameWithHost: string) {
- const [ name, host ] = nameWithHost.split('@')
- if (!host || host === WEBSERVER.HOST) return VideoChannelModel.loadLocalByNameAndPopulateAccount(name)
- return VideoChannelModel.loadByNameAndHostAndPopulateAccount(name, host)
- }
- static loadLocalByNameAndPopulateAccount (name: string): Promise<MChannelAccountDefault> {
- const query = {
- include: [
- {
- model: ActorModel,
- required: true,
- where: {
- preferredUsername: name,
- serverId: null
- }
- }
- ]
- }
- return VideoChannelModel.unscoped()
- .scope([ ScopeNames.WITH_ACTOR, ScopeNames.WITH_ACCOUNT ])
- .findOne(query)
- }
- static loadByNameAndHostAndPopulateAccount (name: string, host: string): Promise<MChannelAccountDefault> {
- const query = {
- include: [
- {
- model: ActorModel,
- required: true,
- where: {
- preferredUsername: name
- },
- include: [
- {
- model: ServerModel,
- required: true,
- where: { host }
- }
- ]
- }
- ]
- }
- return VideoChannelModel.unscoped()
- .scope([ ScopeNames.WITH_ACTOR, ScopeNames.WITH_ACCOUNT ])
- .findOne(query)
- }
- static loadAndPopulateAccountAndVideos (id: number): Promise<MChannelActorAccountDefaultVideos> {
- const options = {
- include: [
- VideoModel
- ]
- }
- return VideoChannelModel.unscoped()
- .scope([ ScopeNames.WITH_ACTOR, ScopeNames.WITH_ACCOUNT, ScopeNames.WITH_VIDEOS ])
- .findByPk(id, options)
- }
- toFormattedSummaryJSON (this: MChannelSummaryFormattable): VideoChannelSummary {
- const actor = this.Actor.toFormattedSummaryJSON()
- return {
- id: this.id,
- name: actor.name,
- displayName: this.getDisplayName(),
- url: actor.url,
- host: actor.host,
- avatar: actor.avatar
- }
- }
- toFormattedJSON (this: MChannelFormattable): VideoChannel {
- const viewsPerDayString = this.get('viewsPerDay') as string
- const videosCount = this.get('videosCount') as number
- let viewsPerDay: { date: Date, views: number }[]
- if (viewsPerDayString) {
- viewsPerDay = viewsPerDayString.split(',')
- .map(v => {
- const [ dateString, amount ] = v.split('|')
- return {
- date: new Date(dateString),
- views: +amount
- }
- })
- }
- const actor = this.Actor.toFormattedJSON()
- const videoChannel = {
- id: this.id,
- displayName: this.getDisplayName(),
- description: this.description,
- support: this.support,
- isLocal: this.Actor.isOwned(),
- createdAt: this.createdAt,
- updatedAt: this.updatedAt,
- ownerAccount: undefined,
- videosCount,
- viewsPerDay
- }
- if (this.Account) videoChannel.ownerAccount = this.Account.toFormattedJSON()
- return Object.assign(actor, videoChannel)
- }
- toActivityPubObject (this: MChannelAP): ActivityPubActor {
- const obj = this.Actor.toActivityPubObject(this.name)
- return Object.assign(obj, {
- summary: this.description,
- support: this.support,
- attributedTo: [
- {
- type: 'Person' as 'Person',
- id: this.Account.Actor.url
- }
- ]
- })
- }
- getLocalUrl (this: MAccountActor | MChannelActor) {
- return WEBSERVER.URL + `/video-channels/` + this.Actor.preferredUsername
- }
- getDisplayName () {
- return this.name
- }
- isOutdated () {
- return this.Actor.isOutdated()
- }
- }
|