client/src/app/shared/shared-moderation/abuse.service.ts
import { omit } from 'lodash-es'
import { SortMeta } from 'primeng/api'
import { Observable } from 'rxjs'
import { catchError } from 'rxjs/operators'
import { HttpClient, HttpParams } from '@angular/common/http'
import { Injectable } from '@angular/core'
import { RestExtractor, RestPagination, RestService } from '@app/core'
import {
AbuseCreate,
AbuseFilter,
AbuseMessage,
AbusePredefinedReasonsString,
AbuseState,
AbuseUpdate,
AdminAbuse,
ResultList,
UserAbuse
} from '@peertube/peertube-models'
import { environment } from '../../../environments/environment'
@Injectable()
export class AbuseService {
private static BASE_ABUSE_URL = environment.apiUrl + '/api/v1/abuses'
private static BASE_MY_ABUSE_URL = environment.apiUrl + '/api/v1/users/me/abuses'
constructor (
private authHttp: HttpClient,
private restService: RestService,
private restExtractor: RestExtractor
) { }
getAdminAbuses (options: {
pagination: RestPagination
sort: SortMeta
search?: string
}): Observable<ResultList<AdminAbuse>> {
const { pagination, sort, search } = options
const url = AbuseService.BASE_ABUSE_URL
let params = new HttpParams()
params = this.restService.addRestGetParams(params, pagination, sort)
if (search) {
params = this.buildParamsFromSearch(search, params)
}
return this.authHttp.get<ResultList<AdminAbuse>>(url, { params })
.pipe(
catchError(res => this.restExtractor.handleError(res))
)
}
getUserAbuses (options: {
pagination: RestPagination
sort: SortMeta
search?: string
}): Observable<ResultList<UserAbuse>> {
const { pagination, sort, search } = options
const url = AbuseService.BASE_MY_ABUSE_URL
let params = new HttpParams()
params = this.restService.addRestGetParams(params, pagination, sort)
if (search) {
params = this.buildParamsFromSearch(search, params)
}
return this.authHttp.get<ResultList<UserAbuse>>(url, { params })
.pipe(
catchError(res => this.restExtractor.handleError(res))
)
}
reportVideo (parameters: AbuseCreate) {
const url = AbuseService.BASE_ABUSE_URL
const body = omit(parameters, [ 'id' ])
return this.authHttp.post(url, body)
.pipe(catchError(res => this.restExtractor.handleError(res)))
}
updateAbuse (abuse: AdminAbuse, abuseUpdate: AbuseUpdate) {
const url = AbuseService.BASE_ABUSE_URL + '/' + abuse.id
return this.authHttp.put(url, abuseUpdate)
.pipe(catchError(res => this.restExtractor.handleError(res)))
}
removeAbuse (abuse: AdminAbuse) {
const url = AbuseService.BASE_ABUSE_URL + '/' + abuse.id
return this.authHttp.delete(url)
.pipe(catchError(res => this.restExtractor.handleError(res)))
}
addAbuseMessage (abuse: UserAbuse, message: string) {
const url = AbuseService.BASE_ABUSE_URL + '/' + abuse.id + '/messages'
return this.authHttp.post(url, { message })
.pipe(catchError(res => this.restExtractor.handleError(res)))
}
listAbuseMessages (abuse: UserAbuse) {
const url = AbuseService.BASE_ABUSE_URL + '/' + abuse.id + '/messages'
return this.authHttp.get<ResultList<AbuseMessage>>(url)
.pipe(
catchError(res => this.restExtractor.handleError(res))
)
}
deleteAbuseMessage (abuse: UserAbuse, abuseMessage: AbuseMessage) {
const url = AbuseService.BASE_ABUSE_URL + '/' + abuse.id + '/messages/' + abuseMessage.id
return this.authHttp.delete(url)
.pipe(catchError(res => this.restExtractor.handleError(res)))
}
getPrefefinedReasons (type: AbuseFilter) {
let reasons: { id: AbusePredefinedReasonsString, label: string, description?: string, help?: string }[] = [
{
id: 'violentOrRepulsive',
label: $localize`Violent or repulsive`,
help: $localize`Contains offensive, violent, or coarse language or iconography.`
},
{
id: 'hatefulOrAbusive',
label: $localize`Hateful or abusive`,
help: $localize`Contains abusive, racist or sexist language or iconography.`
},
{
id: 'spamOrMisleading',
label: $localize`Spam, ad or false news`,
// eslint-disable-next-line max-len
help: $localize`Contains marketing, spam, purposefully deceitful news, or otherwise misleading thumbnail/text/tags. Please provide reputable sources to report hoaxes.`
},
{
id: 'privacy',
label: $localize`Privacy breach or doxxing`,
// eslint-disable-next-line max-len
help: $localize`Contains personal information that could be used to track, identify, contact or impersonate someone (e.g. name, address, phone number, email, or credit card details).`
},
{
id: 'rights',
label: $localize`Copyright`,
help: $localize`Infringes your copyright wrt. the regional laws with which the server must comply.`
},
{
id: 'serverRules',
label: $localize`Breaks server rules`,
// eslint-disable-next-line max-len
help: $localize`Anything not included in the above that breaks the terms of service, code of conduct, or general rules in place on the server.`
}
]
if (type === 'video') {
reasons = reasons.concat([
{
id: 'thumbnails',
label: $localize`Thumbnails`,
help: $localize`The above can only be seen in thumbnails.`
},
{
id: 'captions',
label: $localize`Captions`,
help: $localize`The above can only be seen in captions (please describe which).`
}
])
}
return reasons
}
private buildParamsFromSearch (search: string, params: HttpParams) {
const filters = this.restService.parseQueryStringFilter(search, {
id: { prefix: '#' },
state: {
prefix: 'state:',
handler: v => {
if (v === 'accepted') return AbuseState.ACCEPTED
if (v === 'pending') return AbuseState.PENDING
if (v === 'rejected') return AbuseState.REJECTED
return undefined
}
},
videoIs: {
prefix: 'videoIs:',
handler: v => {
if (v === 'deleted') return v
if (v === 'blacklisted') return v
return undefined
}
},
searchReporter: { prefix: 'reporter:' },
searchReportee: { prefix: 'reportee:' },
predefinedReason: { prefix: 'tag:' }
})
return this.restService.addObjectParams(params, filters)
}
}