PeerTube/server/models/account/user-notification.ts

476 lines
12 KiB
TypeScript
Raw Normal View History

2019-04-18 04:28:17 -05:00
import { AllowNull, BelongsTo, Column, CreatedAt, Default, ForeignKey, Is, Model, Scopes, Table, UpdatedAt } from 'sequelize-typescript'
2018-12-26 03:36:24 -06:00
import { UserNotification, UserNotificationType } from '../../../shared'
import { getSort, throwIfNotValid } from '../utils'
import { isBooleanValid } from '../../helpers/custom-validators/misc'
import { isUserNotificationTypeValid } from '../../helpers/custom-validators/user-notifications'
import { UserModel } from './user'
import { VideoModel } from '../video/video'
import { VideoCommentModel } from '../video/video-comment'
2019-04-23 02:50:57 -05:00
import { FindOptions, ModelIndexesOptions, Op, WhereOptions } from 'sequelize'
2018-12-26 03:36:24 -06:00
import { VideoChannelModel } from '../video/video-channel'
import { AccountModel } from './account'
import { VideoAbuseModel } from '../video/video-abuse'
import { VideoBlacklistModel } from '../video/video-blacklist'
import { VideoImportModel } from '../video/video-import'
import { ActorModel } from '../activitypub/actor'
import { ActorFollowModel } from '../activitypub/actor-follow'
import { AvatarModel } from '../avatar/avatar'
import { ServerModel } from '../server/server'
2019-08-15 04:53:26 -05:00
import { UserNotificationIncludes, UserNotificationModelForApi } from '@server/typings/models/user'
2018-12-26 03:36:24 -06:00
enum ScopeNames {
WITH_ALL = 'WITH_ALL'
}
function buildActorWithAvatarInclude () {
return {
attributes: [ 'preferredUsername' ],
2019-04-23 02:50:57 -05:00
model: ActorModel.unscoped(),
required: true,
include: [
{
attributes: [ 'filename' ],
2019-04-23 02:50:57 -05:00
model: AvatarModel.unscoped(),
required: false
},
{
attributes: [ 'host' ],
2019-04-23 02:50:57 -05:00
model: ServerModel.unscoped(),
required: false
}
]
}
}
function buildVideoInclude (required: boolean) {
return {
attributes: [ 'id', 'uuid', 'name' ],
2019-04-23 02:50:57 -05:00
model: VideoModel.unscoped(),
required
}
}
function buildChannelInclude (required: boolean, withActor = false) {
return {
required,
attributes: [ 'id', 'name' ],
2019-04-23 02:50:57 -05:00
model: VideoChannelModel.unscoped(),
include: withActor === true ? [ buildActorWithAvatarInclude() ] : []
}
}
function buildAccountInclude (required: boolean, withActor = false) {
return {
required,
attributes: [ 'id', 'name' ],
2019-04-23 02:50:57 -05:00
model: AccountModel.unscoped(),
include: withActor === true ? [ buildActorWithAvatarInclude() ] : []
}
}
2019-04-23 02:50:57 -05:00
@Scopes(() => ({
2018-12-26 03:36:24 -06:00
[ScopeNames.WITH_ALL]: {
include: [
Object.assign(buildVideoInclude(false), {
include: [ buildChannelInclude(true, true) ]
}),
2018-12-26 03:36:24 -06:00
{
attributes: [ 'id', 'originCommentId' ],
2019-04-23 02:50:57 -05:00
model: VideoCommentModel.unscoped(),
2018-12-26 03:36:24 -06:00
required: false,
include: [
buildAccountInclude(true, true),
buildVideoInclude(true)
2018-12-26 03:36:24 -06:00
]
},
2018-12-26 03:36:24 -06:00
{
attributes: [ 'id' ],
2019-04-23 02:50:57 -05:00
model: VideoAbuseModel.unscoped(),
2018-12-26 03:36:24 -06:00
required: false,
include: [ buildVideoInclude(true) ]
2018-12-26 03:36:24 -06:00
},
2018-12-26 03:36:24 -06:00
{
attributes: [ 'id' ],
2019-04-23 02:50:57 -05:00
model: VideoBlacklistModel.unscoped(),
2018-12-26 03:36:24 -06:00
required: false,
include: [ buildVideoInclude(true) ]
},
{
attributes: [ 'id', 'magnetUri', 'targetUrl', 'torrentName' ],
2019-04-23 02:50:57 -05:00
model: VideoImportModel.unscoped(),
required: false,
include: [ buildVideoInclude(false) ]
},
{
attributes: [ 'id', 'state' ],
2019-04-23 02:50:57 -05:00
model: ActorFollowModel.unscoped(),
required: false,
include: [
{
attributes: [ 'preferredUsername' ],
2019-04-23 02:50:57 -05:00
model: ActorModel.unscoped(),
required: true,
as: 'ActorFollower',
include: [
{
attributes: [ 'id', 'name' ],
2019-04-23 02:50:57 -05:00
model: AccountModel.unscoped(),
required: true
},
{
attributes: [ 'filename' ],
2019-04-23 02:50:57 -05:00
model: AvatarModel.unscoped(),
required: false
},
{
attributes: [ 'host' ],
2019-04-23 02:50:57 -05:00
model: ServerModel.unscoped(),
required: false
}
]
},
{
attributes: [ 'preferredUsername', 'type' ],
2019-04-23 02:50:57 -05:00
model: ActorModel.unscoped(),
required: true,
as: 'ActorFollowing',
include: [
buildChannelInclude(false),
buildAccountInclude(false),
{
attributes: [ 'host' ],
model: ServerModel.unscoped(),
required: false
}
]
}
]
},
buildAccountInclude(false, true)
2019-04-23 02:50:57 -05:00
]
2018-12-26 03:36:24 -06:00
}
2019-04-23 02:50:57 -05:00
}))
2018-12-26 03:36:24 -06:00
@Table({
tableName: 'userNotification',
indexes: [
{
fields: [ 'userId' ]
2018-12-26 03:36:24 -06:00
},
{
fields: [ 'videoId' ],
where: {
videoId: {
[Op.ne]: null
}
}
},
{
fields: [ 'commentId' ],
where: {
commentId: {
[Op.ne]: null
}
}
},
{
fields: [ 'videoAbuseId' ],
where: {
videoAbuseId: {
[Op.ne]: null
}
}
},
{
fields: [ 'videoBlacklistId' ],
where: {
videoBlacklistId: {
[Op.ne]: null
}
}
},
{
fields: [ 'videoImportId' ],
where: {
videoImportId: {
[Op.ne]: null
}
}
},
{
fields: [ 'accountId' ],
where: {
accountId: {
[Op.ne]: null
}
}
},
{
fields: [ 'actorFollowId' ],
where: {
actorFollowId: {
[Op.ne]: null
}
}
2018-12-26 03:36:24 -06:00
}
2019-04-23 02:50:57 -05:00
] as (ModelIndexesOptions & { where?: WhereOptions })[]
2018-12-26 03:36:24 -06:00
})
export class UserNotificationModel extends Model<UserNotificationModel> {
@AllowNull(false)
@Default(null)
@Is('UserNotificationType', value => throwIfNotValid(value, isUserNotificationTypeValid, 'type'))
@Column
type: UserNotificationType
@AllowNull(false)
@Default(false)
@Is('UserNotificationRead', value => throwIfNotValid(value, isBooleanValid, 'read'))
@Column
read: boolean
@CreatedAt
createdAt: Date
@UpdatedAt
updatedAt: Date
@ForeignKey(() => UserModel)
@Column
userId: number
@BelongsTo(() => UserModel, {
foreignKey: {
allowNull: false
},
onDelete: 'cascade'
})
User: UserModel
@ForeignKey(() => VideoModel)
@Column
videoId: number
@BelongsTo(() => VideoModel, {
foreignKey: {
allowNull: true
},
onDelete: 'cascade'
})
Video: VideoModel
@ForeignKey(() => VideoCommentModel)
@Column
commentId: number
@BelongsTo(() => VideoCommentModel, {
foreignKey: {
allowNull: true
},
onDelete: 'cascade'
})
Comment: VideoCommentModel
@ForeignKey(() => VideoAbuseModel)
@Column
videoAbuseId: number
@BelongsTo(() => VideoAbuseModel, {
foreignKey: {
allowNull: true
},
onDelete: 'cascade'
})
VideoAbuse: VideoAbuseModel
@ForeignKey(() => VideoBlacklistModel)
@Column
videoBlacklistId: number
@BelongsTo(() => VideoBlacklistModel, {
foreignKey: {
allowNull: true
},
onDelete: 'cascade'
})
VideoBlacklist: VideoBlacklistModel
@ForeignKey(() => VideoImportModel)
@Column
videoImportId: number
@BelongsTo(() => VideoImportModel, {
foreignKey: {
allowNull: true
},
onDelete: 'cascade'
})
VideoImport: VideoImportModel
@ForeignKey(() => AccountModel)
@Column
accountId: number
@BelongsTo(() => AccountModel, {
foreignKey: {
allowNull: true
},
onDelete: 'cascade'
})
Account: AccountModel
@ForeignKey(() => ActorFollowModel)
@Column
actorFollowId: number
@BelongsTo(() => ActorFollowModel, {
foreignKey: {
allowNull: true
},
onDelete: 'cascade'
})
ActorFollow: ActorFollowModel
static listForApi (userId: number, start: number, count: number, sort: string, unread?: boolean) {
2020-01-09 02:01:08 -06:00
const where = { userId }
2019-04-18 04:28:17 -05:00
const query: FindOptions = {
2018-12-26 03:36:24 -06:00
offset: start,
limit: count,
order: getSort(sort),
2020-01-09 02:01:08 -06:00
where
2018-12-26 03:36:24 -06:00
}
if (unread !== undefined) query.where['read'] = !unread
2020-01-09 02:01:08 -06:00
return Promise.all([
UserNotificationModel.count({ where })
.then(count => count || 0),
count === 0
? []
: UserNotificationModel.scope(ScopeNames.WITH_ALL).findAll(query)
]).then(([ total, data ]) => ({ total, data }))
2018-12-26 03:36:24 -06:00
}
static markAsRead (userId: number, notificationIds: number[]) {
const query = {
where: {
userId,
id: {
2019-04-23 02:50:57 -05:00
[Op.in]: notificationIds // FIXME: sequelize ANY seems broken
2018-12-26 03:36:24 -06:00
}
}
}
return UserNotificationModel.update({ read: true }, query)
}
2019-01-08 04:26:41 -06:00
static markAllAsRead (userId: number) {
const query = { where: { userId } }
return UserNotificationModel.update({ read: true }, query)
}
2019-08-15 04:53:26 -05:00
toFormattedJSON (this: UserNotificationModelForApi): UserNotification {
const video = this.Video
? Object.assign(this.formatVideo(this.Video),{ channel: this.formatActor(this.Video.VideoChannel) })
: undefined
const videoImport = this.VideoImport ? {
id: this.VideoImport.id,
video: this.VideoImport.Video ? this.formatVideo(this.VideoImport.Video) : undefined,
torrentName: this.VideoImport.torrentName,
magnetUri: this.VideoImport.magnetUri,
targetUrl: this.VideoImport.targetUrl
2018-12-26 03:36:24 -06:00
} : undefined
const comment = this.Comment ? {
id: this.Comment.id,
threadId: this.Comment.getThreadId(),
account: this.formatActor(this.Comment.Account),
video: this.formatVideo(this.Comment.Video)
2018-12-26 03:36:24 -06:00
} : undefined
const videoAbuse = this.VideoAbuse ? {
id: this.VideoAbuse.id,
video: this.formatVideo(this.VideoAbuse.Video)
2018-12-26 03:36:24 -06:00
} : undefined
const videoBlacklist = this.VideoBlacklist ? {
id: this.VideoBlacklist.id,
video: this.formatVideo(this.VideoBlacklist.Video)
2018-12-26 03:36:24 -06:00
} : undefined
const account = this.Account ? this.formatActor(this.Account) : undefined
const actorFollowingType = {
Application: 'instance' as 'instance',
Group: 'channel' as 'channel',
Person: 'account' as 'account'
}
const actorFollow = this.ActorFollow ? {
id: this.ActorFollow.id,
state: this.ActorFollow.state,
follower: {
id: this.ActorFollow.ActorFollower.Account.id,
displayName: this.ActorFollow.ActorFollower.Account.getDisplayName(),
name: this.ActorFollow.ActorFollower.preferredUsername,
2019-08-09 04:32:40 -05:00
avatar: this.ActorFollow.ActorFollower.Avatar ? { path: this.ActorFollow.ActorFollower.Avatar.getStaticPath() } : undefined,
host: this.ActorFollow.ActorFollower.getHost()
},
following: {
type: actorFollowingType[this.ActorFollow.ActorFollowing.type],
displayName: (this.ActorFollow.ActorFollowing.VideoChannel || this.ActorFollow.ActorFollowing.Account).getDisplayName(),
name: this.ActorFollow.ActorFollowing.preferredUsername,
host: this.ActorFollow.ActorFollowing.getHost()
}
} : undefined
2018-12-26 03:36:24 -06:00
return {
id: this.id,
type: this.type,
read: this.read,
video,
videoImport,
2018-12-26 03:36:24 -06:00
comment,
videoAbuse,
videoBlacklist,
account,
actorFollow,
2018-12-26 03:36:24 -06:00
createdAt: this.createdAt.toISOString(),
updatedAt: this.updatedAt.toISOString()
}
}
2019-08-15 04:53:26 -05:00
formatVideo (this: UserNotificationModelForApi, video: UserNotificationIncludes.VideoInclude) {
return {
id: video.id,
uuid: video.uuid,
name: video.name
}
}
2019-08-15 04:53:26 -05:00
formatActor (
this: UserNotificationModelForApi,
accountOrChannel: UserNotificationIncludes.AccountIncludeActor | UserNotificationIncludes.VideoChannelIncludeActor
) {
const avatar = accountOrChannel.Actor.Avatar
2019-08-09 04:32:40 -05:00
? { path: accountOrChannel.Actor.Avatar.getStaticPath() }
: undefined
return {
id: accountOrChannel.id,
displayName: accountOrChannel.getDisplayName(),
name: accountOrChannel.Actor.preferredUsername,
host: accountOrChannel.Actor.getHost(),
avatar
}
}
2018-12-26 03:36:24 -06:00
}