]> git.immae.eu Git - github/Chocobozzz/PeerTube.git/blame - server/models/video/video-abuse.ts
Add nth abuse count for a given video, add reporter/reportee reports stats
[github/Chocobozzz/PeerTube.git] / server / models / video / video-abuse.ts
CommitLineData
86521a67 1import {
844db39e 2 AllowNull, BelongsTo, Column, CreatedAt, DataType, Default, ForeignKey, Is, Model, Table, UpdatedAt, Scopes
86521a67 3} from 'sequelize-typescript'
3fd3ab2d 4import { VideoAbuseObject } from '../../../shared/models/activitypub/objects'
19a3b914 5import { VideoAbuse } from '../../../shared/models/videos'
268eebed
C
6import {
7 isVideoAbuseModerationCommentValid,
8 isVideoAbuseReasonValid,
9 isVideoAbuseStateValid
10} from '../../helpers/custom-validators/video-abuses'
3fd3ab2d 11import { AccountModel } from '../account/account'
f0a47bc9 12import { buildBlockedAccountSQL, getSort, throwIfNotValid } from '../utils'
3fd3ab2d 13import { VideoModel } from './video'
5fd4ca00 14import { VideoAbuseState, VideoDetails } from '../../../shared'
74dc3bca 15import { CONSTRAINTS_FIELDS, VIDEO_ABUSE_STATES } from '../../initializers/constants'
f0a47bc9 16import { MUserAccountId, MVideoAbuse, MVideoAbuseFormattable, MVideoAbuseVideo } from '../../typings/models'
453e83ea 17import * as Bluebird from 'bluebird'
f0a47bc9 18import { literal, Op } from 'sequelize'
86521a67
RK
19import { ThumbnailModel } from './thumbnail'
20import { VideoChannelModel } from './video-channel'
86521a67 21import { VideoBlacklistModel } from './video-blacklist'
3fd3ab2d 22
844db39e
RK
23export enum ScopeNames {
24 FOR_API = 'FOR_API'
25}
26
27@Scopes(() => ({
28 [ScopeNames.FOR_API]: (options: {
29 search?: string
30 searchReporter?: string
31 searchVideo?: string
32 searchVideoChannel?: string
33 serverAccountId: number
34 userAccountId: any
35 }) => {
36 const search = (sourceField, targetField) => sourceField ? ({
37 [targetField]: {
38 [Op.iLike]: `%${sourceField}%`
39 }
40 }) : {}
41
42 let where = {
43 reporterAccountId: {
44 [Op.notIn]: literal('(' + buildBlockedAccountSQL(options.serverAccountId, options.userAccountId) + ')')
45 }
46 }
47
48 if (options.search) {
49 where = Object.assign(where, {
50 [Op.or]: [
51 {
52 [Op.and]: [
53 { videoId: { [Op.not]: null } },
54 { '$Video.name$': { [Op.iLike]: `%${options.search}%` } }
55 ]
56 },
57 {
58 [Op.and]: [
59 { videoId: { [Op.not]: null } },
60 { '$Video.VideoChannel.name$': { [Op.iLike]: `%${options.search}%` } }
61 ]
62 },
63 {
64 [Op.and]: [
65 { deletedVideo: { [Op.not]: null } },
66 { deletedVideo: { name: { [Op.iLike]: `%${options.search}%` } } }
67 ]
68 },
69 {
70 [Op.and]: [
71 { deletedVideo: { [Op.not]: null } },
72 { deletedVideo: { channel: { displayName: { [Op.iLike]: `%${options.search}%` } } } }
73 ]
74 },
75 { '$Account.name$': { [Op.iLike]: `%${options.search}%` } }
76 ]
77 })
78 }
79
844db39e 80 return {
5fd4ca00
RK
81 attributes: {
82 include: [
83 [
84 literal(
85 '(' +
86 'SELECT t.count ' +
87 'FROM ( ' +
88 'SELECT id, ' +
89 'count(id) OVER (PARTITION BY "videoId") ' +
90 'FROM "videoAbuse" ' +
91 ') t ' +
92 'WHERE t.id = "VideoAbuseModel".id ' +
93 ')'
94 ),
95 'countReportsForVideo'
96 ],
97 [
98 literal(
99 '(' +
100 'SELECT t.nth ' +
101 'FROM ( ' +
102 'SELECT id, ' +
103 'row_number() OVER (PARTITION BY "videoId" ORDER BY "createdAt") AS nth ' +
104 'FROM "videoAbuse" ' +
105 ') t ' +
106 'WHERE t.id = "VideoAbuseModel".id ' +
107 ')'
108 ),
109 'nthReportForVideo'
110 ],
111 [
112 literal(
113 '(' +
114 'SELECT count("videoAbuse"."id") ' +
115 'FROM "videoAbuse" ' +
116 'INNER JOIN "video" ON "video"."id" = "videoAbuse"."videoId" ' +
117 'INNER JOIN "videoChannel" ON "videoChannel"."id" = "video"."channelId" ' +
118 'INNER JOIN "account" ON "videoChannel"."accountId" = "account"."id" ' +
119 'WHERE "account"."id" = "VideoAbuseModel"."reporterAccountId" ' +
120 ')'
121 ),
122 'countReportsForReporter'
123 ],
124 [
125 literal(
126 '(' +
127 'WITH ' +
128 'ids AS ( ' +
129 'SELECT "account"."id" ' +
130 'FROM "account" ' +
131 'INNER JOIN "videoChannel" ON "videoChannel"."accountId" = "account"."id" ' +
132 'INNER JOIN "video" ON "video"."channelId" = "videoChannel"."id" ' +
133 'WHERE "video"."id" = "VideoAbuseModel"."videoId" ' +
134 ') ' +
135 'SELECT count("videoAbuse"."id") ' +
136 'FROM "videoAbuse" ' +
137 'INNER JOIN "video" ON "video"."id" = "videoAbuse"."videoId" ' +
138 'INNER JOIN "videoChannel" ON "videoChannel"."id" = "video"."channelId" ' +
139 'INNER JOIN "account" ON "videoChannel"."accountId" = "account"."id" ' +
140 'INNER JOIN ids ON "account"."id" = ids.id ' +
141 ')'
142 ),
143 'countReportsForReportee'
144 ]
145 ]
146 },
86521a67
RK
147 include: [
148 {
844db39e
RK
149 model: AccountModel,
150 required: true,
151 where: { ...search(options.searchReporter, 'name') }
86521a67
RK
152 },
153 {
844db39e
RK
154 model: VideoModel,
155 required: false,
156 where: { ...search(options.searchVideo, 'name') },
86521a67
RK
157 include: [
158 {
844db39e
RK
159 model: ThumbnailModel
160 },
161 {
5fd4ca00
RK
162 model: VideoChannelModel.scope([ 'WITH_ACTOR', 'WITH_ACCOUNT' ]),
163 where: { ...search(options.searchVideoChannel, 'name') }
844db39e
RK
164 },
165 {
166 attributes: [ 'id', 'reason', 'unfederated' ],
167 model: VideoBlacklistModel
86521a67
RK
168 }
169 ]
86521a67 170 }
844db39e
RK
171 ],
172 where
86521a67 173 }
844db39e 174 }
86521a67 175}))
3fd3ab2d
C
176@Table({
177 tableName: 'videoAbuse',
178 indexes: [
55fa55a9 179 {
3fd3ab2d 180 fields: [ 'videoId' ]
55fa55a9
C
181 },
182 {
3fd3ab2d 183 fields: [ 'reporterAccountId' ]
55fa55a9 184 }
e02643f3 185 ]
3fd3ab2d
C
186})
187export class VideoAbuseModel extends Model<VideoAbuseModel> {
e02643f3 188
3fd3ab2d 189 @AllowNull(false)
1506307f 190 @Default(null)
3fd3ab2d 191 @Is('VideoAbuseReason', value => throwIfNotValid(value, isVideoAbuseReasonValid, 'reason'))
1506307f 192 @Column(DataType.STRING(CONSTRAINTS_FIELDS.VIDEO_ABUSES.REASON.max))
3fd3ab2d 193 reason: string
21e0727a 194
268eebed
C
195 @AllowNull(false)
196 @Default(null)
197 @Is('VideoAbuseState', value => throwIfNotValid(value, isVideoAbuseStateValid, 'state'))
198 @Column
199 state: VideoAbuseState
200
201 @AllowNull(true)
202 @Default(null)
1735c825 203 @Is('VideoAbuseModerationComment', value => throwIfNotValid(value, isVideoAbuseModerationCommentValid, 'moderationComment', true))
268eebed
C
204 @Column(DataType.STRING(CONSTRAINTS_FIELDS.VIDEO_ABUSES.MODERATION_COMMENT.max))
205 moderationComment: string
206
68d19a0a
RK
207 @AllowNull(true)
208 @Default(null)
209 @Column(DataType.JSONB)
5fd4ca00 210 deletedVideo: VideoDetails
68d19a0a 211
3fd3ab2d
C
212 @CreatedAt
213 createdAt: Date
21e0727a 214
3fd3ab2d
C
215 @UpdatedAt
216 updatedAt: Date
e02643f3 217
3fd3ab2d
C
218 @ForeignKey(() => AccountModel)
219 @Column
220 reporterAccountId: number
55fa55a9 221
3fd3ab2d 222 @BelongsTo(() => AccountModel, {
55fa55a9 223 foreignKey: {
68d19a0a 224 allowNull: true
55fa55a9 225 },
68d19a0a 226 onDelete: 'set null'
55fa55a9 227 })
3fd3ab2d
C
228 Account: AccountModel
229
230 @ForeignKey(() => VideoModel)
231 @Column
232 videoId: number
55fa55a9 233
3fd3ab2d 234 @BelongsTo(() => VideoModel, {
55fa55a9 235 foreignKey: {
68d19a0a 236 allowNull: true
55fa55a9 237 },
68d19a0a 238 onDelete: 'set null'
55fa55a9 239 })
3fd3ab2d
C
240 Video: VideoModel
241
68d19a0a
RK
242 static loadByIdAndVideoId (id: number, videoId?: number, uuid?: string): Bluebird<MVideoAbuse> {
243 const videoAttributes = {}
244 if (videoId) videoAttributes['videoId'] = videoId
245 if (uuid) videoAttributes['deletedVideo'] = { uuid }
246
268eebed
C
247 const query = {
248 where: {
249 id,
68d19a0a 250 ...videoAttributes
268eebed
C
251 }
252 }
253 return VideoAbuseModel.findOne(query)
254 }
255
f0a47bc9 256 static listForApi (parameters: {
a1587156
C
257 start: number
258 count: number
259 sort: string
844db39e 260 search?: string
f0a47bc9
C
261 serverAccountId: number
262 user?: MUserAccountId
263 }) {
844db39e 264 const { start, count, sort, search, user, serverAccountId } = parameters
f0a47bc9
C
265 const userAccountId = user ? user.Account.id : undefined
266
3fd3ab2d
C
267 const query = {
268 offset: start,
269 limit: count,
3bb6c526 270 order: getSort(sort),
86521a67
RK
271 col: 'VideoAbuseModel.id',
272 distinct: true
3fd3ab2d 273 }
55fa55a9 274
844db39e
RK
275 const filters = {
276 search,
277 serverAccountId,
278 userAccountId
279 }
280
281 return VideoAbuseModel
282 .scope({ method: [ ScopeNames.FOR_API, filters ] })
283 .findAndCountAll(query)
3fd3ab2d
C
284 .then(({ rows, count }) => {
285 return { total: count, data: rows }
286 })
55fa55a9
C
287 }
288
1ca9f7c3 289 toFormattedJSON (this: MVideoAbuseFormattable): VideoAbuse {
5fd4ca00
RK
290 const countReportsForVideo = this.get('countReportsForVideo') as number
291 const nthReportForVideo = this.get('nthReportForVideo') as number
292 const countReportsForReporter = this.get('countReportsForReporter') as number
293 const countReportsForReportee = this.get('countReportsForReportee') as number
294
68d19a0a
RK
295 const video = this.Video
296 ? this.Video
297 : this.deletedVideo
298
3fd3ab2d
C
299 return {
300 id: this.id,
301 reason: this.reason,
19a3b914 302 reporterAccount: this.Account.toFormattedJSON(),
268eebed
C
303 state: {
304 id: this.state,
305 label: VideoAbuseModel.getStateLabel(this.state)
306 },
307 moderationComment: this.moderationComment,
19a3b914 308 video: {
68d19a0a
RK
309 id: video.id,
310 uuid: video.uuid,
311 name: video.name,
312 nsfw: video.nsfw,
86521a67
RK
313 deleted: !this.Video,
314 blacklisted: this.Video && this.Video.isBlacklisted(),
315 thumbnailPath: this.Video?.getMiniatureStaticPath(),
5fd4ca00 316 channel: this.Video?.VideoChannel.toFormattedJSON() || this.deletedVideo?.channel
19a3b914 317 },
5fd4ca00
RK
318 createdAt: this.createdAt,
319 updatedAt: this.updatedAt,
320 count: countReportsForVideo || 0,
321 nth: nthReportForVideo || 0,
322 countReportsForReporter: countReportsForReporter || 0,
323 countReportsForReportee: countReportsForReportee || 0
3fd3ab2d
C
324 }
325 }
326
453e83ea 327 toActivityPubObject (this: MVideoAbuseVideo): VideoAbuseObject {
3fd3ab2d
C
328 return {
329 type: 'Flag' as 'Flag',
330 content: this.reason,
331 object: this.Video.url
332 }
333 }
268eebed
C
334
335 private static getStateLabel (id: number) {
336 return VIDEO_ABUSE_STATES[id] || 'Unknown'
337 }
55fa55a9 338}