Chocobozzz/PeerTube

View on GitHub
client/src/app/shared/shared-moderation/abuse.service.ts

Summary

Maintainability
D
2 days
Test Coverage
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)
  }
}