2021-07-28 10:32:40 +02:00
|
|
|
import { FindOptions, Includeable, literal, Op, QueryTypes, ScopeOptions, Transaction, WhereOptions } from 'sequelize'
|
2017-12-12 17:53:50 +01:00
|
|
|
import {
|
2018-08-16 15:25:20 +02:00
|
|
|
AllowNull,
|
|
|
|
BeforeDestroy,
|
|
|
|
BelongsTo,
|
|
|
|
Column,
|
|
|
|
CreatedAt,
|
|
|
|
DataType,
|
|
|
|
Default,
|
|
|
|
DefaultScope,
|
|
|
|
ForeignKey,
|
2019-04-11 11:33:44 +02:00
|
|
|
HasMany,
|
2018-08-16 15:25:20 +02:00
|
|
|
Is,
|
|
|
|
Model,
|
|
|
|
Scopes,
|
2018-08-23 17:58:39 +02:00
|
|
|
Sequelize,
|
2018-08-16 15:25:20 +02:00
|
|
|
Table,
|
|
|
|
UpdatedAt
|
2017-12-12 17:53:50 +01:00
|
|
|
} from 'sequelize-typescript'
|
2021-12-16 18:04:16 +01:00
|
|
|
import { CONFIG } from '@server/initializers/config'
|
2020-12-08 10:53:41 +01:00
|
|
|
import { MAccountActor } from '@server/types/models'
|
2021-12-16 18:04:16 +01:00
|
|
|
import { pick } from '@shared/core-utils'
|
|
|
|
import { AttributesOnly } from '@shared/typescript-utils'
|
2017-12-14 17:38:41 +01:00
|
|
|
import { ActivityPubActor } from '../../../shared/models/activitypub'
|
2019-02-26 10:55:40 +01:00
|
|
|
import { VideoChannel, VideoChannelSummary } from '../../../shared/models/videos'
|
2018-02-15 14:46:26 +01:00
|
|
|
import {
|
2018-08-16 15:25:20 +02:00
|
|
|
isVideoChannelDescriptionValid,
|
2021-08-05 13:54:35 +02:00
|
|
|
isVideoChannelDisplayNameValid,
|
2018-02-15 14:46:26 +01:00
|
|
|
isVideoChannelSupportValid
|
|
|
|
} from '../../helpers/custom-validators/video-channels'
|
2021-10-26 16:42:10 +02:00
|
|
|
import { CONSTRAINTS_FIELDS, WEBSERVER } from '../../initializers/constants'
|
2020-11-10 16:29:35 +01:00
|
|
|
import { sendDeleteActor } from '../../lib/activitypub/send'
|
2019-08-15 11:53:26 +02:00
|
|
|
import {
|
|
|
|
MChannelActor,
|
2019-08-21 14:31:57 +02:00
|
|
|
MChannelAP,
|
2021-04-06 17:01:35 +02:00
|
|
|
MChannelBannerAccountDefault,
|
2019-08-21 14:31:57 +02:00
|
|
|
MChannelFormattable,
|
|
|
|
MChannelSummaryFormattable
|
2020-06-18 10:45:25 +02:00
|
|
|
} from '../../types/models/video'
|
2020-11-10 16:29:35 +01:00
|
|
|
import { AccountModel, ScopeNames as AccountModelScopeNames, SummaryOptions as AccountSummaryOptions } from '../account/account'
|
2021-05-11 11:15:29 +02:00
|
|
|
import { ActorModel, unusedActorAttributesForAPI } from '../actor/actor'
|
|
|
|
import { ActorFollowModel } from '../actor/actor-follow'
|
|
|
|
import { ActorImageModel } from '../actor/actor-image'
|
2020-11-10 16:29:35 +01:00
|
|
|
import { ServerModel } from '../server/server'
|
2021-07-28 10:32:40 +02:00
|
|
|
import { setAsUpdated } from '../shared'
|
2020-11-10 16:29:35 +01:00
|
|
|
import { buildServerIdsFollowedBy, buildTrigramSearchIndex, createSimilarityAttribute, getSort, throwIfNotValid } from '../utils'
|
|
|
|
import { VideoModel } from './video'
|
|
|
|
import { VideoPlaylistModel } from './video-playlist'
|
2018-08-23 17:58:39 +02:00
|
|
|
|
2019-02-26 10:55:40 +01:00
|
|
|
export enum ScopeNames {
|
2019-08-15 11:53:26 +02:00
|
|
|
FOR_API = 'FOR_API',
|
2020-03-23 10:14:05 +01:00
|
|
|
SUMMARY = 'SUMMARY',
|
2017-12-14 10:07:57 +01:00
|
|
|
WITH_ACCOUNT = 'WITH_ACCOUNT',
|
2017-12-14 17:38:41 +01:00
|
|
|
WITH_ACTOR = 'WITH_ACTOR',
|
2021-04-06 17:01:35 +02:00
|
|
|
WITH_ACTOR_BANNER = 'WITH_ACTOR_BANNER',
|
2019-02-26 10:55:40 +01:00
|
|
|
WITH_VIDEOS = 'WITH_VIDEOS',
|
2020-03-23 10:14:05 +01:00
|
|
|
WITH_STATS = 'WITH_STATS'
|
2017-12-14 10:07:57 +01:00
|
|
|
}
|
|
|
|
|
2018-08-23 17:58:39 +02:00
|
|
|
type AvailableForListOptions = {
|
|
|
|
actorId: number
|
2020-07-15 11:17:03 +02:00
|
|
|
search?: string
|
2021-07-28 10:32:40 +02:00
|
|
|
host?: string
|
2021-07-29 10:27:24 +02:00
|
|
|
handles?: string[]
|
2018-08-23 17:58:39 +02:00
|
|
|
}
|
|
|
|
|
2020-03-23 10:14:05 +01:00
|
|
|
type AvailableWithStatsOptions = {
|
|
|
|
daysPrior: number
|
|
|
|
}
|
|
|
|
|
2019-07-31 15:57:32 +02:00
|
|
|
export type SummaryOptions = {
|
2020-07-07 14:34:16 +02:00
|
|
|
actorRequired?: boolean // Default: true
|
2019-07-31 15:57:32 +02:00
|
|
|
withAccount?: boolean // Default: false
|
|
|
|
withAccountBlockerIds?: number[]
|
|
|
|
}
|
|
|
|
|
2019-04-23 09:50:57 +02:00
|
|
|
@DefaultScope(() => ({
|
2017-12-14 17:38:41 +01:00
|
|
|
include: [
|
|
|
|
{
|
2019-04-23 09:50:57 +02:00
|
|
|
model: ActorModel,
|
2017-12-14 17:38:41 +01:00
|
|
|
required: true
|
|
|
|
}
|
|
|
|
]
|
2019-04-23 09:50:57 +02:00
|
|
|
}))
|
|
|
|
@Scopes(() => ({
|
2019-08-15 11:53:26 +02:00
|
|
|
[ScopeNames.FOR_API]: (options: AvailableForListOptions) => {
|
2018-08-23 17:58:39 +02:00
|
|
|
// Only list local channels OR channels that are on an instance followed by actorId
|
2019-02-26 10:55:40 +01:00
|
|
|
const inQueryInstanceFollow = buildServerIdsFollowedBy(options.actorId)
|
2018-08-23 17:58:39 +02:00
|
|
|
|
2021-07-28 16:40:21 +02:00
|
|
|
const whereActorAnd: WhereOptions[] = [
|
|
|
|
{
|
|
|
|
[Op.or]: [
|
|
|
|
{
|
|
|
|
serverId: null
|
|
|
|
},
|
|
|
|
{
|
|
|
|
serverId: {
|
|
|
|
[Op.in]: Sequelize.literal(inQueryInstanceFollow)
|
|
|
|
}
|
2021-07-28 10:32:40 +02:00
|
|
|
}
|
2021-07-28 16:40:21 +02:00
|
|
|
]
|
|
|
|
}
|
|
|
|
]
|
2021-07-28 10:32:40 +02:00
|
|
|
|
|
|
|
let serverRequired = false
|
|
|
|
let whereServer: WhereOptions
|
|
|
|
|
|
|
|
if (options.host && options.host !== WEBSERVER.HOST) {
|
|
|
|
serverRequired = true
|
|
|
|
whereServer = { host: options.host }
|
|
|
|
}
|
|
|
|
|
|
|
|
if (options.host === WEBSERVER.HOST) {
|
2021-07-28 16:40:21 +02:00
|
|
|
whereActorAnd.push({
|
|
|
|
serverId: null
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2021-07-29 10:27:24 +02:00
|
|
|
let rootWhere: WhereOptions
|
|
|
|
if (options.handles) {
|
|
|
|
const or: WhereOptions[] = []
|
|
|
|
|
|
|
|
for (const handle of options.handles || []) {
|
|
|
|
const [ preferredUsername, host ] = handle.split('@')
|
|
|
|
|
2021-10-08 11:15:06 +02:00
|
|
|
if (!host || host === WEBSERVER.HOST) {
|
2021-07-29 10:27:24 +02:00
|
|
|
or.push({
|
|
|
|
'$Actor.preferredUsername$': preferredUsername,
|
|
|
|
'$Actor.serverId$': null
|
|
|
|
})
|
|
|
|
} else {
|
|
|
|
or.push({
|
|
|
|
'$Actor.preferredUsername$': preferredUsername,
|
|
|
|
'$Actor.Server.host$': host
|
|
|
|
})
|
2021-07-28 16:40:21 +02:00
|
|
|
}
|
2021-07-29 10:27:24 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
rootWhere = {
|
|
|
|
[Op.or]: or
|
|
|
|
}
|
2021-07-28 10:32:40 +02:00
|
|
|
}
|
|
|
|
|
2018-08-23 17:58:39 +02:00
|
|
|
return {
|
2021-07-29 10:27:24 +02:00
|
|
|
where: rootWhere,
|
2018-08-23 17:58:39 +02:00
|
|
|
include: [
|
|
|
|
{
|
|
|
|
attributes: {
|
|
|
|
exclude: unusedActorAttributesForAPI
|
|
|
|
},
|
|
|
|
model: ActorModel,
|
2021-07-28 16:40:21 +02:00
|
|
|
where: {
|
|
|
|
[Op.and]: whereActorAnd
|
|
|
|
},
|
2021-04-07 10:36:13 +02:00
|
|
|
include: [
|
2021-07-28 10:32:40 +02:00
|
|
|
{
|
|
|
|
model: ServerModel,
|
|
|
|
required: serverRequired,
|
|
|
|
where: whereServer
|
|
|
|
},
|
|
|
|
{
|
|
|
|
model: ActorImageModel,
|
|
|
|
as: 'Avatar',
|
|
|
|
required: false
|
|
|
|
},
|
2021-04-07 10:36:13 +02:00
|
|
|
{
|
|
|
|
model: ActorImageModel,
|
|
|
|
as: 'Banner',
|
|
|
|
required: false
|
|
|
|
}
|
|
|
|
]
|
2018-08-23 17:58:39 +02:00
|
|
|
},
|
|
|
|
{
|
|
|
|
model: AccountModel,
|
|
|
|
required: true,
|
|
|
|
include: [
|
|
|
|
{
|
|
|
|
attributes: {
|
|
|
|
exclude: unusedActorAttributesForAPI
|
|
|
|
},
|
|
|
|
model: ActorModel, // Default scope includes avatar and server
|
|
|
|
required: true
|
|
|
|
}
|
|
|
|
]
|
|
|
|
}
|
|
|
|
]
|
|
|
|
}
|
|
|
|
},
|
2020-03-23 10:14:05 +01:00
|
|
|
[ScopeNames.SUMMARY]: (options: SummaryOptions = {}) => {
|
2020-12-08 14:30:29 +01:00
|
|
|
const include: Includeable[] = [
|
|
|
|
{
|
|
|
|
attributes: [ 'id', 'preferredUsername', 'url', 'serverId', 'avatarId' ],
|
|
|
|
model: ActorModel.unscoped(),
|
|
|
|
required: options.actorRequired ?? true,
|
|
|
|
include: [
|
|
|
|
{
|
|
|
|
attributes: [ 'host' ],
|
|
|
|
model: ServerModel.unscoped(),
|
|
|
|
required: false
|
|
|
|
},
|
|
|
|
{
|
2021-04-06 11:35:56 +02:00
|
|
|
model: ActorImageModel.unscoped(),
|
|
|
|
as: 'Avatar',
|
2020-12-08 14:30:29 +01:00
|
|
|
required: false
|
|
|
|
}
|
|
|
|
]
|
|
|
|
}
|
|
|
|
]
|
|
|
|
|
2020-03-23 10:14:05 +01:00
|
|
|
const base: FindOptions = {
|
2020-12-08 14:30:29 +01:00
|
|
|
attributes: [ 'id', 'name', 'description', 'actorId' ]
|
2020-03-23 10:14:05 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
if (options.withAccount === true) {
|
2020-12-08 14:30:29 +01:00
|
|
|
include.push({
|
2020-03-23 10:14:05 +01:00
|
|
|
model: AccountModel.scope({
|
|
|
|
method: [ AccountModelScopeNames.SUMMARY, { withAccountBlockerIds: options.withAccountBlockerIds } as AccountSummaryOptions ]
|
|
|
|
}),
|
|
|
|
required: true
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2020-12-08 14:30:29 +01:00
|
|
|
base.include = include
|
|
|
|
|
2020-03-23 10:14:05 +01:00
|
|
|
return base
|
|
|
|
},
|
2018-08-23 17:58:39 +02:00
|
|
|
[ScopeNames.WITH_ACCOUNT]: {
|
|
|
|
include: [
|
|
|
|
{
|
2019-04-23 09:50:57 +02:00
|
|
|
model: AccountModel,
|
2018-08-23 17:58:39 +02:00
|
|
|
required: true
|
2017-12-14 10:07:57 +01:00
|
|
|
}
|
|
|
|
]
|
|
|
|
},
|
2020-03-23 10:14:05 +01:00
|
|
|
[ScopeNames.WITH_ACTOR]: {
|
2017-12-14 10:07:57 +01:00
|
|
|
include: [
|
2020-03-23 10:14:05 +01:00
|
|
|
ActorModel
|
2017-12-14 10:07:57 +01:00
|
|
|
]
|
2017-12-14 17:38:41 +01:00
|
|
|
},
|
2021-04-06 17:01:35 +02:00
|
|
|
[ScopeNames.WITH_ACTOR_BANNER]: {
|
|
|
|
include: [
|
|
|
|
{
|
|
|
|
model: ActorModel,
|
|
|
|
include: [
|
|
|
|
{
|
|
|
|
model: ActorImageModel,
|
|
|
|
required: false,
|
|
|
|
as: 'Banner'
|
|
|
|
}
|
|
|
|
]
|
|
|
|
}
|
|
|
|
]
|
|
|
|
},
|
2020-03-23 10:14:05 +01:00
|
|
|
[ScopeNames.WITH_VIDEOS]: {
|
2017-12-14 17:38:41 +01:00
|
|
|
include: [
|
2020-03-23 10:14:05 +01:00
|
|
|
VideoModel
|
2017-12-14 17:38:41 +01:00
|
|
|
]
|
2020-03-23 10:14:05 +01:00
|
|
|
},
|
2020-03-30 12:06:46 +02:00
|
|
|
[ScopeNames.WITH_STATS]: (options: AvailableWithStatsOptions = { daysPrior: 30 }) => {
|
|
|
|
const daysPrior = parseInt(options.daysPrior + '', 10)
|
|
|
|
|
|
|
|
return {
|
|
|
|
attributes: {
|
|
|
|
include: [
|
2020-06-16 14:13:01 +02:00
|
|
|
[
|
|
|
|
literal('(SELECT COUNT(*) FROM "video" WHERE "channelId" = "VideoChannelModel"."id")'),
|
|
|
|
'videosCount'
|
|
|
|
],
|
2020-03-30 12:06:46 +02:00
|
|
|
[
|
|
|
|
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 ` +
|
2020-03-23 10:14:05 +01:00
|
|
|
') ' +
|
2020-06-12 16:01:42 +02:00
|
|
|
'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' +
|
2020-03-30 12:06:46 +02:00
|
|
|
')'
|
|
|
|
),
|
|
|
|
'viewsPerDay'
|
|
|
|
]
|
2020-03-23 10:14:05 +01:00
|
|
|
]
|
2020-03-30 12:06:46 +02:00
|
|
|
}
|
2020-03-23 10:14:05 +01:00
|
|
|
}
|
2020-03-30 12:06:46 +02:00
|
|
|
}
|
2019-04-23 09:50:57 +02:00
|
|
|
}))
|
2017-12-12 17:53:50 +01:00
|
|
|
@Table({
|
|
|
|
tableName: 'videoChannel',
|
2020-01-28 14:45:17 +01:00
|
|
|
indexes: [
|
|
|
|
buildTrigramSearchIndex('video_channel_name_trigram', 'name'),
|
|
|
|
|
|
|
|
{
|
|
|
|
fields: [ 'accountId' ]
|
|
|
|
},
|
|
|
|
{
|
|
|
|
fields: [ 'actorId' ]
|
|
|
|
}
|
|
|
|
]
|
2017-12-12 17:53:50 +01:00
|
|
|
})
|
2021-05-12 14:09:04 +02:00
|
|
|
export class VideoChannelModel extends Model<Partial<AttributesOnly<VideoChannelModel>>> {
|
2017-10-24 19:41:09 +02:00
|
|
|
|
2017-12-12 17:53:50 +01:00
|
|
|
@AllowNull(false)
|
2021-08-05 13:54:35 +02:00
|
|
|
@Is('VideoChannelName', value => throwIfNotValid(value, isVideoChannelDisplayNameValid, 'name'))
|
2017-12-12 17:53:50 +01:00
|
|
|
@Column
|
|
|
|
name: string
|
2017-10-24 19:41:09 +02:00
|
|
|
|
2017-12-12 17:53:50 +01:00
|
|
|
@AllowNull(true)
|
2018-02-15 14:46:26 +01:00
|
|
|
@Default(null)
|
2019-04-18 11:28:17 +02:00
|
|
|
@Is('VideoChannelDescription', value => throwIfNotValid(value, isVideoChannelDescriptionValid, 'description', true))
|
2018-05-09 13:32:44 +02:00
|
|
|
@Column(DataType.STRING(CONSTRAINTS_FIELDS.VIDEO_CHANNELS.DESCRIPTION.max))
|
2017-12-12 17:53:50 +01:00
|
|
|
description: string
|
2017-10-24 19:41:09 +02:00
|
|
|
|
2018-02-15 14:46:26 +01:00
|
|
|
@AllowNull(true)
|
|
|
|
@Default(null)
|
2019-04-18 11:28:17 +02:00
|
|
|
@Is('VideoChannelSupport', value => throwIfNotValid(value, isVideoChannelSupportValid, 'support', true))
|
2018-05-09 13:32:44 +02:00
|
|
|
@Column(DataType.STRING(CONSTRAINTS_FIELDS.VIDEO_CHANNELS.SUPPORT.max))
|
2018-02-15 14:46:26 +01:00
|
|
|
support: string
|
|
|
|
|
2017-12-12 17:53:50 +01:00
|
|
|
@CreatedAt
|
|
|
|
createdAt: Date
|
2017-10-24 19:41:09 +02:00
|
|
|
|
2017-12-12 17:53:50 +01:00
|
|
|
@UpdatedAt
|
|
|
|
updatedAt: Date
|
2017-11-27 14:44:51 +01:00
|
|
|
|
2017-12-14 11:18:49 +01:00
|
|
|
@ForeignKey(() => ActorModel)
|
|
|
|
@Column
|
|
|
|
actorId: number
|
|
|
|
|
|
|
|
@BelongsTo(() => ActorModel, {
|
|
|
|
foreignKey: {
|
|
|
|
allowNull: false
|
|
|
|
},
|
|
|
|
onDelete: 'cascade'
|
|
|
|
})
|
|
|
|
Actor: ActorModel
|
|
|
|
|
2017-12-12 17:53:50 +01:00
|
|
|
@ForeignKey(() => AccountModel)
|
|
|
|
@Column
|
|
|
|
accountId: number
|
2017-11-27 14:44:51 +01:00
|
|
|
|
2017-12-12 17:53:50 +01:00
|
|
|
@BelongsTo(() => AccountModel, {
|
|
|
|
foreignKey: {
|
|
|
|
allowNull: false
|
2021-07-01 16:47:14 +02:00
|
|
|
}
|
2017-12-12 17:53:50 +01:00
|
|
|
})
|
|
|
|
Account: AccountModel
|
2017-10-24 19:41:09 +02:00
|
|
|
|
2017-12-12 17:53:50 +01:00
|
|
|
@HasMany(() => VideoModel, {
|
2017-10-24 19:41:09 +02:00
|
|
|
foreignKey: {
|
2017-12-12 17:53:50 +01:00
|
|
|
name: 'channelId',
|
2017-10-24 19:41:09 +02:00
|
|
|
allowNull: false
|
|
|
|
},
|
2018-01-18 10:53:54 +01:00
|
|
|
onDelete: 'CASCADE',
|
|
|
|
hooks: true
|
2017-10-24 19:41:09 +02:00
|
|
|
})
|
2017-12-12 17:53:50 +01:00
|
|
|
Videos: VideoModel[]
|
2017-10-24 19:41:09 +02:00
|
|
|
|
2019-02-26 10:55:40 +01:00
|
|
|
@HasMany(() => VideoPlaylistModel, {
|
|
|
|
foreignKey: {
|
2019-02-28 11:14:26 +01:00
|
|
|
allowNull: true
|
2019-02-26 10:55:40 +01:00
|
|
|
},
|
2019-03-05 10:58:44 +01:00
|
|
|
onDelete: 'CASCADE',
|
2019-02-26 10:55:40 +01:00
|
|
|
hooks: true
|
|
|
|
})
|
|
|
|
VideoPlaylists: VideoPlaylistModel[]
|
|
|
|
|
2018-01-18 10:53:54 +01:00
|
|
|
@BeforeDestroy
|
|
|
|
static async sendDeleteIfOwned (instance: VideoChannelModel, options) {
|
|
|
|
if (!instance.Actor) {
|
2020-01-08 15:11:38 +01:00
|
|
|
instance.Actor = await instance.$get('Actor', { transaction: options.transaction })
|
2018-01-18 10:53:54 +01:00
|
|
|
}
|
|
|
|
|
2020-11-10 16:29:35 +01:00
|
|
|
await ActorFollowModel.removeFollowsOf(instance.Actor.id, options.transaction)
|
|
|
|
|
2018-07-30 13:39:20 +02:00
|
|
|
if (instance.Actor.isOwned()) {
|
|
|
|
return sendDeleteActor(instance.Actor, options.transaction)
|
|
|
|
}
|
|
|
|
|
|
|
|
return undefined
|
2017-12-12 17:53:50 +01:00
|
|
|
}
|
2017-10-24 19:41:09 +02:00
|
|
|
|
2017-12-12 17:53:50 +01:00
|
|
|
static countByAccount (accountId: number) {
|
|
|
|
const query = {
|
|
|
|
where: {
|
|
|
|
accountId
|
|
|
|
}
|
2017-10-24 19:41:09 +02:00
|
|
|
}
|
2017-12-12 17:53:50 +01:00
|
|
|
|
|
|
|
return VideoChannelModel.count(query)
|
2017-10-24 19:41:09 +02:00
|
|
|
}
|
|
|
|
|
2021-04-12 11:19:07 +02:00
|
|
|
static async getStats () {
|
|
|
|
|
|
|
|
function getActiveVideoChannels (days: number) {
|
|
|
|
const options = {
|
|
|
|
type: QueryTypes.SELECT as QueryTypes.SELECT,
|
|
|
|
raw: true
|
|
|
|
}
|
|
|
|
|
|
|
|
const query = `
|
|
|
|
SELECT COUNT(DISTINCT("VideoChannelModel"."id")) AS "count"
|
|
|
|
FROM "videoChannel" AS "VideoChannelModel"
|
|
|
|
INNER JOIN "video" AS "Videos"
|
|
|
|
ON "VideoChannelModel"."id" = "Videos"."channelId"
|
|
|
|
AND ("Videos"."publishedAt" > Now() - interval '${days}d')
|
|
|
|
INNER JOIN "account" AS "Account"
|
|
|
|
ON "VideoChannelModel"."accountId" = "Account"."id"
|
|
|
|
INNER JOIN "actor" AS "Account->Actor"
|
|
|
|
ON "Account"."actorId" = "Account->Actor"."id"
|
|
|
|
AND "Account->Actor"."serverId" IS NULL
|
|
|
|
LEFT OUTER JOIN "server" AS "Account->Actor->Server"
|
|
|
|
ON "Account->Actor"."serverId" = "Account->Actor->Server"."id"`
|
|
|
|
|
|
|
|
return VideoChannelModel.sequelize.query<{ count: string }>(query, options)
|
|
|
|
.then(r => parseInt(r[0].count, 10))
|
|
|
|
}
|
|
|
|
|
|
|
|
const totalLocalVideoChannels = await VideoChannelModel.count()
|
|
|
|
const totalLocalDailyActiveVideoChannels = await getActiveVideoChannels(1)
|
|
|
|
const totalLocalWeeklyActiveVideoChannels = await getActiveVideoChannels(7)
|
|
|
|
const totalLocalMonthlyActiveVideoChannels = await getActiveVideoChannels(30)
|
|
|
|
const totalHalfYearActiveVideoChannels = await getActiveVideoChannels(180)
|
|
|
|
|
|
|
|
return {
|
|
|
|
totalLocalVideoChannels,
|
|
|
|
totalLocalDailyActiveVideoChannels,
|
|
|
|
totalLocalWeeklyActiveVideoChannels,
|
|
|
|
totalLocalMonthlyActiveVideoChannels,
|
|
|
|
totalHalfYearActiveVideoChannels
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-12-08 14:30:29 +01:00
|
|
|
static listLocalsForSitemap (sort: string): Promise<MChannelActor[]> {
|
2018-12-05 17:27:24 +01:00
|
|
|
const query = {
|
|
|
|
attributes: [ ],
|
|
|
|
offset: 0,
|
|
|
|
order: getSort(sort),
|
|
|
|
include: [
|
|
|
|
{
|
|
|
|
attributes: [ 'preferredUsername', 'serverId' ],
|
|
|
|
model: ActorModel.unscoped(),
|
|
|
|
where: {
|
|
|
|
serverId: null
|
|
|
|
}
|
|
|
|
}
|
|
|
|
]
|
|
|
|
}
|
|
|
|
|
|
|
|
return VideoChannelModel
|
|
|
|
.unscoped()
|
|
|
|
.findAll(query)
|
|
|
|
}
|
|
|
|
|
2021-07-29 14:17:03 +02:00
|
|
|
static listForApi (parameters: Pick<AvailableForListOptions, 'actorId'> & {
|
2018-08-23 17:58:39 +02:00
|
|
|
start: number
|
|
|
|
count: number
|
|
|
|
sort: string
|
2021-07-29 14:17:03 +02:00
|
|
|
}) {
|
|
|
|
const { actorId } = parameters
|
2021-07-28 10:32:40 +02:00
|
|
|
|
2021-07-29 14:17:03 +02:00
|
|
|
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 searchForApi (options: Pick<AvailableForListOptions, 'actorId' | 'search' | 'host' | 'handles'> & {
|
|
|
|
start: number
|
|
|
|
count: number
|
|
|
|
sort: string
|
2018-08-23 17:58:39 +02:00
|
|
|
}) {
|
2021-07-28 16:40:21 +02:00
|
|
|
let attributesInclude: any[] = [ literal('0 as similarity') ]
|
|
|
|
let where: WhereOptions
|
2018-08-23 17:58:39 +02:00
|
|
|
|
2021-07-28 16:40:21 +02:00
|
|
|
if (options.search) {
|
|
|
|
const escapedSearch = VideoChannelModel.sequelize.escape(options.search)
|
|
|
|
const escapedLikeSearch = VideoChannelModel.sequelize.escape('%' + options.search + '%')
|
|
|
|
attributesInclude = [ createSimilarityAttribute('VideoChannelModel.name', options.search) ]
|
|
|
|
|
|
|
|
where = {
|
2019-04-18 11:28:17 +02:00
|
|
|
[Op.or]: [
|
2018-08-28 15:16:04 +02:00
|
|
|
Sequelize.literal(
|
|
|
|
'lower(immutable_unaccent("VideoChannelModel"."name")) % lower(immutable_unaccent(' + escapedSearch + '))'
|
|
|
|
),
|
|
|
|
Sequelize.literal(
|
|
|
|
'lower(immutable_unaccent("VideoChannelModel"."name")) LIKE lower(immutable_unaccent(' + escapedLikeSearch + '))'
|
2018-08-23 17:58:39 +02:00
|
|
|
)
|
2018-08-28 15:16:04 +02:00
|
|
|
]
|
2018-08-23 17:58:39 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-07-28 16:40:21 +02:00
|
|
|
const query = {
|
|
|
|
attributes: {
|
|
|
|
include: attributesInclude
|
|
|
|
},
|
|
|
|
offset: options.start,
|
|
|
|
limit: options.count,
|
|
|
|
order: getSort(options.sort),
|
|
|
|
where
|
|
|
|
}
|
|
|
|
|
2018-08-23 17:58:39 +02:00
|
|
|
return VideoChannelModel
|
2020-12-08 14:30:29 +01:00
|
|
|
.scope({
|
2021-07-29 10:27:24 +02:00
|
|
|
method: [ ScopeNames.FOR_API, pick(options, [ 'actorId', 'host', 'handles' ]) as AvailableForListOptions ]
|
2020-12-08 14:30:29 +01:00
|
|
|
})
|
2017-12-14 17:38:41 +01:00
|
|
|
.findAndCountAll(query)
|
2017-12-12 17:53:50 +01:00
|
|
|
.then(({ rows, count }) => {
|
|
|
|
return { total: count, data: rows }
|
|
|
|
})
|
2017-10-24 19:41:09 +02:00
|
|
|
}
|
|
|
|
|
2021-10-19 09:44:43 +02:00
|
|
|
static listByAccountForAPI (options: {
|
2020-01-31 16:56:52 +01:00
|
|
|
accountId: number
|
|
|
|
start: number
|
|
|
|
count: number
|
2019-05-29 15:09:38 +02:00
|
|
|
sort: string
|
2020-03-23 10:14:05 +01:00
|
|
|
withStats?: boolean
|
2020-07-23 21:30:04 +02:00
|
|
|
search?: string
|
2019-05-29 15:09:38 +02:00
|
|
|
}) {
|
2020-07-23 21:30:04 +02:00
|
|
|
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
|
|
|
|
|
2017-12-12 17:53:50 +01:00
|
|
|
const query = {
|
2019-05-29 15:09:38 +02:00
|
|
|
offset: options.start,
|
|
|
|
limit: options.count,
|
|
|
|
order: getSort(options.sort),
|
2017-12-12 17:53:50 +01:00
|
|
|
include: [
|
|
|
|
{
|
|
|
|
model: AccountModel,
|
|
|
|
where: {
|
2019-05-29 15:09:38 +02:00
|
|
|
id: options.accountId
|
2017-12-12 17:53:50 +01:00
|
|
|
},
|
2017-12-14 17:38:41 +01:00
|
|
|
required: true
|
2017-12-12 17:53:50 +01:00
|
|
|
}
|
2020-07-23 21:30:04 +02:00
|
|
|
],
|
|
|
|
where
|
2017-12-12 17:53:50 +01:00
|
|
|
}
|
2017-10-24 19:41:09 +02:00
|
|
|
|
2021-04-06 17:01:35 +02:00
|
|
|
const scopes: string | ScopeOptions | (string | ScopeOptions)[] = [ ScopeNames.WITH_ACTOR_BANNER ]
|
2020-03-23 10:14:05 +01:00
|
|
|
|
2020-06-12 16:01:42 +02:00
|
|
|
if (options.withStats === true) {
|
2020-03-23 10:14:05 +01:00
|
|
|
scopes.push({
|
|
|
|
method: [ ScopeNames.WITH_STATS, { daysPrior: 30 } as AvailableWithStatsOptions ]
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2017-12-14 17:38:41 +01:00
|
|
|
return VideoChannelModel
|
2020-03-23 10:14:05 +01:00
|
|
|
.scope(scopes)
|
2017-12-14 17:38:41 +01:00
|
|
|
.findAndCountAll(query)
|
2017-12-12 17:53:50 +01:00
|
|
|
.then(({ rows, count }) => {
|
|
|
|
return { total: count, data: rows }
|
|
|
|
})
|
2017-10-24 19:41:09 +02:00
|
|
|
}
|
|
|
|
|
2021-10-19 09:44:43 +02:00
|
|
|
static listAllByAccount (accountId: number) {
|
|
|
|
const query = {
|
2021-10-26 16:42:10 +02:00
|
|
|
limit: CONFIG.VIDEO_CHANNELS.MAX_PER_USER,
|
2021-10-19 09:44:43 +02:00
|
|
|
include: [
|
|
|
|
{
|
|
|
|
attributes: [],
|
|
|
|
model: AccountModel,
|
|
|
|
where: {
|
|
|
|
id: accountId
|
|
|
|
},
|
|
|
|
required: true
|
|
|
|
}
|
|
|
|
]
|
|
|
|
}
|
|
|
|
|
|
|
|
return VideoChannelModel.findAll(query)
|
|
|
|
}
|
|
|
|
|
2021-06-15 09:17:19 +02:00
|
|
|
static loadAndPopulateAccount (id: number, transaction?: Transaction): Promise<MChannelBannerAccountDefault> {
|
2018-09-04 10:22:10 +02:00
|
|
|
return VideoChannelModel.unscoped()
|
2021-04-06 17:01:35 +02:00
|
|
|
.scope([ ScopeNames.WITH_ACTOR_BANNER, ScopeNames.WITH_ACCOUNT ])
|
2021-06-15 09:17:19 +02:00
|
|
|
.findByPk(id, { transaction })
|
2017-12-12 17:53:50 +01:00
|
|
|
}
|
2017-11-10 14:34:45 +01:00
|
|
|
|
2021-04-06 17:01:35 +02:00
|
|
|
static loadByUrlAndPopulateAccount (url: string): Promise<MChannelBannerAccountDefault> {
|
2018-08-23 17:58:39 +02:00
|
|
|
const query = {
|
|
|
|
include: [
|
|
|
|
{
|
|
|
|
model: ActorModel,
|
|
|
|
required: true,
|
|
|
|
where: {
|
|
|
|
url
|
2021-04-06 17:01:35 +02:00
|
|
|
},
|
|
|
|
include: [
|
|
|
|
{
|
|
|
|
model: ActorImageModel,
|
|
|
|
required: false,
|
|
|
|
as: 'Banner'
|
|
|
|
}
|
|
|
|
]
|
2018-08-23 17:58:39 +02:00
|
|
|
}
|
|
|
|
]
|
|
|
|
}
|
|
|
|
|
|
|
|
return VideoChannelModel
|
|
|
|
.scope([ ScopeNames.WITH_ACCOUNT ])
|
2018-08-17 15:45:42 +02:00
|
|
|
.findOne(query)
|
2017-10-24 19:41:09 +02:00
|
|
|
}
|
|
|
|
|
2019-02-21 14:06:10 +01:00
|
|
|
static loadByNameWithHostAndPopulateAccount (nameWithHost: string) {
|
|
|
|
const [ name, host ] = nameWithHost.split('@')
|
|
|
|
|
2019-04-11 11:33:44 +02:00
|
|
|
if (!host || host === WEBSERVER.HOST) return VideoChannelModel.loadLocalByNameAndPopulateAccount(name)
|
2019-02-21 14:06:10 +01:00
|
|
|
|
|
|
|
return VideoChannelModel.loadByNameAndHostAndPopulateAccount(name, host)
|
|
|
|
}
|
|
|
|
|
2021-04-06 17:01:35 +02:00
|
|
|
static loadLocalByNameAndPopulateAccount (name: string): Promise<MChannelBannerAccountDefault> {
|
2018-08-17 15:45:42 +02:00
|
|
|
const query = {
|
2017-12-12 17:53:50 +01:00
|
|
|
include: [
|
2018-08-17 15:45:42 +02:00
|
|
|
{
|
|
|
|
model: ActorModel,
|
|
|
|
required: true,
|
|
|
|
where: {
|
|
|
|
preferredUsername: name,
|
|
|
|
serverId: null
|
2021-04-06 17:01:35 +02:00
|
|
|
},
|
|
|
|
include: [
|
|
|
|
{
|
|
|
|
model: ActorImageModel,
|
|
|
|
required: false,
|
|
|
|
as: 'Banner'
|
|
|
|
}
|
|
|
|
]
|
2018-08-17 15:45:42 +02:00
|
|
|
}
|
2017-12-12 17:53:50 +01:00
|
|
|
]
|
|
|
|
}
|
2017-10-24 19:41:09 +02:00
|
|
|
|
2018-09-04 10:22:10 +02:00
|
|
|
return VideoChannelModel.unscoped()
|
2021-04-06 17:01:35 +02:00
|
|
|
.scope([ ScopeNames.WITH_ACCOUNT ])
|
2018-08-17 15:45:42 +02:00
|
|
|
.findOne(query)
|
2017-10-24 19:41:09 +02:00
|
|
|
}
|
|
|
|
|
2021-04-06 17:01:35 +02:00
|
|
|
static loadByNameAndHostAndPopulateAccount (name: string, host: string): Promise<MChannelBannerAccountDefault> {
|
2018-08-16 15:25:20 +02:00
|
|
|
const query = {
|
|
|
|
include: [
|
|
|
|
{
|
|
|
|
model: ActorModel,
|
|
|
|
required: true,
|
|
|
|
where: {
|
2018-08-17 15:45:42 +02:00
|
|
|
preferredUsername: name
|
|
|
|
},
|
|
|
|
include: [
|
|
|
|
{
|
|
|
|
model: ServerModel,
|
|
|
|
required: true,
|
|
|
|
where: { host }
|
2021-04-06 17:01:35 +02:00
|
|
|
},
|
|
|
|
{
|
|
|
|
model: ActorImageModel,
|
|
|
|
required: false,
|
|
|
|
as: 'Banner'
|
2018-08-17 15:45:42 +02:00
|
|
|
}
|
|
|
|
]
|
2018-08-16 15:25:20 +02:00
|
|
|
}
|
|
|
|
]
|
|
|
|
}
|
|
|
|
|
2018-09-04 10:22:10 +02:00
|
|
|
return VideoChannelModel.unscoped()
|
2021-04-06 17:01:35 +02:00
|
|
|
.scope([ ScopeNames.WITH_ACCOUNT ])
|
2018-08-17 15:45:42 +02:00
|
|
|
.findOne(query)
|
|
|
|
}
|
|
|
|
|
2019-08-20 19:05:31 +02:00
|
|
|
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 {
|
2020-06-16 14:13:01 +02:00
|
|
|
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
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
2020-03-23 10:14:05 +01:00
|
|
|
|
2017-12-14 17:38:41 +01:00
|
|
|
const actor = this.Actor.toFormattedJSON()
|
2018-04-25 10:21:38 +02:00
|
|
|
const videoChannel = {
|
2017-12-12 17:53:50 +01:00
|
|
|
id: this.id,
|
2018-06-13 15:07:25 +02:00
|
|
|
displayName: this.getDisplayName(),
|
2017-12-12 17:53:50 +01:00
|
|
|
description: this.description,
|
2018-02-15 14:46:26 +01:00
|
|
|
support: this.support,
|
2017-12-14 17:38:41 +01:00
|
|
|
isLocal: this.Actor.isOwned(),
|
2021-05-07 17:14:39 +02:00
|
|
|
updatedAt: this.updatedAt,
|
2020-03-23 10:14:05 +01:00
|
|
|
ownerAccount: undefined,
|
2020-06-16 14:13:01 +02:00
|
|
|
videosCount,
|
|
|
|
viewsPerDay
|
2018-04-25 10:21:38 +02:00
|
|
|
}
|
|
|
|
|
2018-05-23 11:38:00 +02:00
|
|
|
if (this.Account) videoChannel.ownerAccount = this.Account.toFormattedJSON()
|
2017-10-24 19:41:09 +02:00
|
|
|
|
2018-04-25 10:21:38 +02:00
|
|
|
return Object.assign(actor, videoChannel)
|
2017-10-24 19:41:09 +02:00
|
|
|
}
|
|
|
|
|
2019-08-21 14:31:57 +02:00
|
|
|
toActivityPubObject (this: MChannelAP): ActivityPubActor {
|
2019-08-30 16:50:12 +02:00
|
|
|
const obj = this.Actor.toActivityPubObject(this.name)
|
2017-12-14 17:38:41 +01:00
|
|
|
|
|
|
|
return Object.assign(obj, {
|
|
|
|
summary: this.description,
|
2018-02-15 14:46:26 +01:00
|
|
|
support: this.support,
|
2017-12-14 17:38:41 +01:00
|
|
|
attributedTo: [
|
|
|
|
{
|
|
|
|
type: 'Person' as 'Person',
|
|
|
|
id: this.Account.Actor.url
|
|
|
|
}
|
|
|
|
]
|
|
|
|
})
|
2017-10-24 19:41:09 +02:00
|
|
|
}
|
2018-06-13 15:07:25 +02:00
|
|
|
|
2020-12-08 10:53:41 +01:00
|
|
|
getLocalUrl (this: MAccountActor | MChannelActor) {
|
|
|
|
return WEBSERVER.URL + `/video-channels/` + this.Actor.preferredUsername
|
|
|
|
}
|
|
|
|
|
2018-06-13 15:07:25 +02:00
|
|
|
getDisplayName () {
|
|
|
|
return this.name
|
|
|
|
}
|
2019-01-14 11:30:15 +01:00
|
|
|
|
|
|
|
isOutdated () {
|
|
|
|
return this.Actor.isOutdated()
|
|
|
|
}
|
2021-05-07 17:14:39 +02:00
|
|
|
|
2021-08-30 16:24:25 +02:00
|
|
|
setAsUpdated (transaction?: Transaction) {
|
2021-05-07 17:14:39 +02:00
|
|
|
return setAsUpdated('videoChannel', this.id, transaction)
|
|
|
|
}
|
2017-10-24 19:41:09 +02:00
|
|
|
}
|