123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525 |
- import {
- AllowNull,
- BeforeDestroy,
- BelongsTo,
- Column,
- CreatedAt,
- DataType,
- Default,
- DefaultScope,
- ForeignKey,
- HasMany,
- Is,
- Model,
- Scopes,
- Sequelize,
- Table,
- UpdatedAt
- } from 'sequelize-typescript'
- import { ActivityPubActor } from '../../../shared/models/activitypub'
- import { VideoChannel, VideoChannelSummary } from '../../../shared/models/videos'
- import {
- isVideoChannelDescriptionValid,
- isVideoChannelNameValid,
- isVideoChannelSupportValid
- } from '../../helpers/custom-validators/video-channels'
- import { sendDeleteActor } from '../../lib/activitypub/send'
- import { AccountModel, ScopeNames as AccountModelScopeNames } from '../account/account'
- import { ActorModel, unusedActorAttributesForAPI } from '../activitypub/actor'
- import { buildServerIdsFollowedBy, buildTrigramSearchIndex, createSimilarityAttribute, getSort, throwIfNotValid } from '../utils'
- import { VideoModel } from './video'
- import { CONSTRAINTS_FIELDS, WEBSERVER } from '../../initializers/constants'
- import { ServerModel } from '../server/server'
- import { FindOptions, ModelIndexesOptions, Op } from 'sequelize'
- import { AvatarModel } from '../avatar/avatar'
- import { VideoPlaylistModel } from './video-playlist'
- // FIXME: Define indexes here because there is an issue with TS and Sequelize.literal when called directly in the annotation
- const indexes: ModelIndexesOptions[] = [
- buildTrigramSearchIndex('video_channel_name_trigram', 'name'),
- {
- fields: [ 'accountId' ]
- },
- {
- fields: [ 'actorId' ]
- }
- ]
- export enum ScopeNames {
- AVAILABLE_FOR_LIST = 'AVAILABLE_FOR_LIST',
- WITH_ACCOUNT = 'WITH_ACCOUNT',
- WITH_ACTOR = 'WITH_ACTOR',
- WITH_VIDEOS = 'WITH_VIDEOS',
- SUMMARY = 'SUMMARY'
- }
- type AvailableForListOptions = {
- actorId: number
- }
- @DefaultScope(() => ({
- include: [
- {
- model: ActorModel,
- required: true
- }
- ]
- }))
- @Scopes(() => ({
- [ScopeNames.SUMMARY]: (withAccount = false) => {
- const base: FindOptions = {
- attributes: [ 'name', 'description', 'id', 'actorId' ],
- include: [
- {
- attributes: [ 'preferredUsername', 'url', 'serverId', 'avatarId' ],
- model: ActorModel.unscoped(),
- required: true,
- include: [
- {
- attributes: [ 'host' ],
- model: ServerModel.unscoped(),
- required: false
- },
- {
- model: AvatarModel.unscoped(),
- required: false
- }
- ]
- }
- ]
- }
- if (withAccount === true) {
- base.include.push({
- model: AccountModel.scope(AccountModelScopeNames.SUMMARY),
- required: true
- })
- }
- return base
- },
- [ScopeNames.AVAILABLE_FOR_LIST]: (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.WITH_ACCOUNT]: {
- include: [
- {
- model: AccountModel,
- required: true
- }
- ]
- },
- [ScopeNames.WITH_VIDEOS]: {
- include: [
- VideoModel
- ]
- },
- [ScopeNames.WITH_ACTOR]: {
- include: [
- ActorModel
- ]
- }
- }))
- @Table({
- tableName: 'videoChannel',
- indexes
- })
- export class VideoChannelModel extends Model<VideoChannelModel> {
- @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 }) as ActorModel
- }
- 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 (actorId: number, start: number, count: number, sort: string) {
- const query = {
- offset: start,
- limit: count,
- order: getSort(sort)
- }
- const scopes = {
- method: [ ScopeNames.AVAILABLE_FOR_LIST, { actorId } as AvailableForListOptions ]
- }
- return VideoChannelModel
- .scope(scopes)
- .findAndCountAll(query)
- .then(({ rows, count }) => {
- return { total: count, data: rows }
- })
- }
- static listLocalsForSitemap (sort: string) {
- 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 + '))'
- )
- ]
- }
- }
- const scopes = {
- method: [ ScopeNames.AVAILABLE_FOR_LIST, { actorId: options.actorId } as AvailableForListOptions ]
- }
- return VideoChannelModel
- .scope(scopes)
- .findAndCountAll(query)
- .then(({ rows, count }) => {
- return { total: count, data: rows }
- })
- }
- static listByAccount (options: {
- accountId: number,
- start: number,
- count: number,
- sort: string
- }) {
- const query = {
- offset: options.start,
- limit: options.count,
- order: getSort(options.sort),
- include: [
- {
- model: AccountModel,
- where: {
- id: options.accountId
- },
- required: true
- }
- ]
- }
- return VideoChannelModel
- .findAndCountAll(query)
- .then(({ rows, count }) => {
- return { total: count, data: rows }
- })
- }
- static loadByIdAndPopulateAccount (id: number) {
- return VideoChannelModel.unscoped()
- .scope([ ScopeNames.WITH_ACTOR, ScopeNames.WITH_ACCOUNT ])
- .findByPk(id)
- }
- static loadByIdAndAccount (id: number, accountId: number) {
- const query = {
- where: {
- id,
- accountId
- }
- }
- return VideoChannelModel.unscoped()
- .scope([ ScopeNames.WITH_ACTOR, ScopeNames.WITH_ACCOUNT ])
- .findOne(query)
- }
- static loadAndPopulateAccount (id: number) {
- return VideoChannelModel.unscoped()
- .scope([ ScopeNames.WITH_ACTOR, ScopeNames.WITH_ACCOUNT ])
- .findByPk(id)
- }
- static loadByUrlAndPopulateAccount (url: string) {
- 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) {
- 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) {
- 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) {
- const options = {
- include: [
- VideoModel
- ]
- }
- return VideoChannelModel.unscoped()
- .scope([ ScopeNames.WITH_ACTOR, ScopeNames.WITH_ACCOUNT, ScopeNames.WITH_VIDEOS ])
- .findByPk(id, options)
- }
- toFormattedJSON (): VideoChannel {
- 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
- }
- if (this.Account) videoChannel.ownerAccount = this.Account.toFormattedJSON()
- return Object.assign(actor, videoChannel)
- }
- toFormattedSummaryJSON (): VideoChannelSummary {
- const actor = this.Actor.toFormattedJSON()
- return {
- id: this.id,
- name: actor.name,
- displayName: this.getDisplayName(),
- url: actor.url,
- host: actor.host,
- avatar: actor.avatar
- }
- }
- toActivityPubObject (): ActivityPubActor {
- const obj = this.Actor.toActivityPubObject(this.name, 'VideoChannel')
- return Object.assign(obj, {
- summary: this.description,
- support: this.support,
- attributedTo: [
- {
- type: 'Person' as 'Person',
- id: this.Account.Actor.url
- }
- ]
- })
- }
- getDisplayName () {
- return this.name
- }
- isOutdated () {
- return this.Actor.isOutdated()
- }
- }
|