import { omit } from 'lodash-es' import { SortMeta } from 'primeng/api' import { Observable } from 'rxjs' import { catchError, map } 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 '@shared/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> { 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>(url, { params }) .pipe( catchError(res => this.restExtractor.handleError(res)) ) } getUserAbuses (options: { pagination: RestPagination, sort: SortMeta, search?: string }): Observable> { 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>(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( map(this.restExtractor.extractDataBool), 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( map(this.restExtractor.extractDataBool), catchError(res => this.restExtractor.handleError(res)) ) } removeAbuse (abuse: AdminAbuse) { const url = AbuseService.BASE_ABUSE_URL + '/' + abuse.id return this.authHttp.delete(url) .pipe( map(this.restExtractor.extractDataBool), 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( map(this.restExtractor.extractDataBool), catchError(res => this.restExtractor.handleError(res)) ) } listAbuseMessages (abuse: UserAbuse) { const url = AbuseService.BASE_ABUSE_URL + '/' + abuse.id + '/messages' return this.authHttp.get>(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( map(this.restExtractor.extractDataBool), 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`, 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`, 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`, description: $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) } }