PeerTube/client/src/app/core/rest/rest-extractor.service.ts

147 lines
4.2 KiB
TypeScript
Raw Normal View History

2018-05-31 11:35:01 +02:00
import { throwError as observableThrowError } from 'rxjs'
import { HttpHeaderResponse } from '@angular/common/http'
2022-08-10 10:26:20 +02:00
import { Inject, Injectable, LOCALE_ID } from '@angular/core'
2018-05-31 11:35:01 +02:00
import { Router } from '@angular/router'
2022-08-10 10:26:20 +02:00
import { DateFormat, dateToHuman } from '@app/helpers'
import { HttpStatusCode, HttpStatusCodeType, ResultList } from '@peertube/peertube-models'
import { logger } from '@root-helpers/logger'
@Injectable()
export class RestExtractor {
2022-08-10 10:26:20 +02:00
constructor (
@Inject(LOCALE_ID) private localeId: string,
private router: Router
) { }
2018-05-31 11:35:01 +02:00
2021-08-17 14:42:53 +02:00
applyToResultListData <T, A, U> (
result: ResultList<T>,
fun: (data: T, ...args: A[]) => U,
additionalArgs: A[] = []
): ResultList<U> {
const data: T[] = result.data
return {
total: result.total,
2021-08-17 14:42:53 +02:00
data: data.map(d => fun.apply(this, [ d, ...additionalArgs ]))
}
}
2022-08-10 10:26:20 +02:00
convertResultListDateToHuman <T> (
result: ResultList<T>,
fieldsToConvert: string[] = [ 'createdAt' ],
format?: DateFormat
): ResultList<T> {
2024-02-22 10:12:04 +01:00
return this.applyToResultListData(result, this.convertDateToHuman.bind(this), [ fieldsToConvert, format ])
}
2022-08-10 10:26:20 +02:00
convertDateToHuman (target: any, fieldsToConvert: string[], format?: DateFormat) {
2021-08-17 14:42:53 +02:00
fieldsToConvert.forEach(field => {
2022-10-25 16:08:11 +02:00
if (!target[field]) return
2022-08-10 10:26:20 +02:00
target[field] = dateToHuman(this.localeId, new Date(target[field]), format)
2021-08-17 14:42:53 +02:00
})
2018-01-31 10:41:44 +01:00
return target
}
redirectTo404IfNotFound (
obj: { status: HttpStatusCodeType },
type: 'video' | 'other',
status: HttpStatusCodeType[] = [ HttpStatusCode.NOT_FOUND_404 ]
) {
2022-05-24 16:29:01 +02:00
if (obj?.status && status.includes(obj.status)) {
// Do not use redirectService to avoid circular dependencies
this.router.navigate([ '/404' ], { state: { type, obj }, skipLocationChange: true })
}
2022-05-24 16:29:01 +02:00
return observableThrowError(() => obj)
}
2018-01-29 16:15:39 +01:00
2022-05-24 16:29:01 +02:00
handleError (err: any) {
const errorMessage = this.buildErrorMessage(err)
2022-10-07 11:06:28 +02:00
const errorObj: { message: string, status: string, body: string, headers: HttpHeaderResponse } = {
message: errorMessage,
2017-09-15 12:26:02 +02:00
status: undefined,
2022-10-07 11:06:28 +02:00
body: undefined,
headers: err.headers
}
if (err.status) {
errorObj.status = err.status
2017-09-15 12:26:02 +02:00
errorObj.body = err.error
}
2021-08-17 11:27:47 +02:00
return observableThrowError(() => errorObj)
}
2018-05-31 11:35:01 +02:00
2022-05-24 16:29:01 +02:00
private buildErrorMessage (err: any) {
if (err.error instanceof Error) {
// A client-side or network error occurred. Handle it accordingly.
const errorMessage = err.error.detail || err.error.title
logger.error('An error occurred:', errorMessage)
2022-05-24 16:29:01 +02:00
return errorMessage
2018-05-31 11:35:01 +02:00
}
2022-05-24 16:29:01 +02:00
if (typeof err.error === 'string') {
return err.error
}
if (err.status !== undefined) {
const errorMessage = this.buildServerErrorMessage(err)
const message = `Backend returned code ${err.status}, errorMessage is: ${errorMessage}`
if (err.status === HttpStatusCode.NOT_FOUND_404) logger.clientError(message)
else logger.error(message)
2022-05-24 16:29:01 +02:00
return errorMessage
}
logger.error(err)
2022-05-24 16:29:01 +02:00
return err
}
private buildServerErrorMessage (err: any) {
// A server-side error occurred.
if (err.error?.errors) {
const errors = err.error.errors
return Object.keys(errors)
.map(key => errors[key].msg)
.join('. ')
}
if (err.error?.error) {
return err.error.error
}
if (err.status === HttpStatusCode.PAYLOAD_TOO_LARGE_413) {
return $localize`Media is too large for the server. Please contact you administrator if you want to increase the limit size.`
}
if (err.status === HttpStatusCode.TOO_MANY_REQUESTS_429) {
const secondsLeft = err.headers.get('retry-after')
if (secondsLeft) {
const minutesLeft = Math.floor(parseInt(secondsLeft, 10) / 60)
return $localize`Too many attempts, please try again after ${minutesLeft} minutes.`
}
return $localize`Too many attempts, please try again later.`
}
if (err.status === HttpStatusCode.INTERNAL_SERVER_ERROR_500) {
return $localize`Server error. Please retry later.`
}
2023-02-17 10:11:15 +01:00
if (err.status === HttpStatusCode.BAD_GATEWAY_502) {
return $localize`Server is unavailable. Please retry later.`
}
2022-05-24 16:29:01 +02:00
return $localize`Unknown server error`
2018-05-31 11:35:01 +02:00
}
}