From e6abf95e9fc9fde52d583850cf7faafdf7d050d5 Mon Sep 17 00:00:00 2001 From: Chocobozzz Date: Wed, 26 Aug 2020 09:14:14 +0200 Subject: Add redirection on unavailable video due to follow constraints --- .../+videos/+video-watch/video-watch.component.ts | 21 +++++++++++-- server/middlewares/validators/videos/videos.ts | 34 ++++++++++++---------- shared/models/server/index.ts | 1 + shared/models/server/server-error-code.enum.ts | 3 ++ 4 files changed, 41 insertions(+), 18 deletions(-) create mode 100644 shared/models/server/server-error-code.enum.ts diff --git a/client/src/app/+videos/+video-watch/video-watch.component.ts b/client/src/app/+videos/+video-watch/video-watch.component.ts index 1b2820810..f048edec4 100644 --- a/client/src/app/+videos/+video-watch/video-watch.component.ts +++ b/client/src/app/+videos/+video-watch/video-watch.component.ts @@ -15,7 +15,7 @@ import { VideoDownloadComponent } from '@app/shared/shared-video-miniature' import { VideoPlaylist, VideoPlaylistService } from '@app/shared/shared-video-playlist' import { MetaService } from '@ngx-meta/core' import { peertubeLocalStorage } from '@root-helpers/peertube-web-storage' -import { ServerConfig, UserVideoRateType, VideoCaption, VideoPrivacy, VideoState } from '@shared/models' +import { ServerConfig, ServerErrorCode, UserVideoRateType, VideoCaption, VideoPrivacy, VideoState } from '@shared/models' import { getStoredP2PEnabled, getStoredTheater } from '../../../assets/player/peertube-player-local-storage' import { CustomizationOptions, @@ -361,7 +361,24 @@ export class VideoWatchComponent implements OnInit, OnDestroy { ]) .pipe( // If 401, the video is private or blocked so redirect to 404 - catchError(err => this.restExtractor.redirectTo404IfNotFound(err, [ 400, 401, 403, 404 ])) + catchError(err => { + if (err.body.errorCode === ServerErrorCode.DOES_NOT_RESPECT_FOLLOW_CONSTRAINTS && err.body.originUrl) { + const search = window.location.search + let originUrl = err.body.originUrl + if (search) originUrl += search + + this.confirmService.confirm( + $localize`This video is not available on this instance. Do you want to be redirected on the origin instance: ${originUrl}?`, + $localize`Redirection` + ).then(res => { + if (res === false) return this.restExtractor.redirectTo404IfNotFound(err, [ 400, 401, 403, 404 ]) + + return window.location.href = originUrl + }) + } + + return this.restExtractor.redirectTo404IfNotFound(err, [ 400, 401, 403, 404 ]) + }) ) .subscribe(([ video, captionsResult ]) => { const queryParams = this.route.snapshot.queryParams diff --git a/server/middlewares/validators/videos/videos.ts b/server/middlewares/validators/videos/videos.ts index 40a34d3b8..b022b2c23 100644 --- a/server/middlewares/validators/videos/videos.ts +++ b/server/middlewares/validators/videos/videos.ts @@ -1,6 +1,9 @@ import * as express from 'express' import { body, param, query, ValidationChain } from 'express-validator' -import { UserRight, VideoChangeOwnershipStatus, VideoPrivacy } from '../../../../shared' +import { getServerActor } from '@server/models/application/application' +import { MVideoFullLight } from '@server/types/models' +import { ServerErrorCode, UserRight, VideoChangeOwnershipStatus, VideoPrivacy } from '../../../../shared' +import { VideoChangeOwnershipAccept } from '../../../../shared/models/videos/video-change-ownership-accept.model' import { isBooleanValid, isDateValid, @@ -12,6 +15,8 @@ import { toIntOrNull, toValueOrNull } from '../../../helpers/custom-validators/misc' +import { isNSFWQueryValid, isNumberArray, isStringArray } from '../../../helpers/custom-validators/search' +import { checkUserCanTerminateOwnershipChange, doesChangeVideoOwnershipExist } from '../../../helpers/custom-validators/video-ownership' import { isScheduleVideoUpdatePrivacyValid, isVideoCategoryValid, @@ -27,29 +32,24 @@ import { isVideoSupportValid, isVideoTagsValid } from '../../../helpers/custom-validators/videos' +import { cleanUpReqFiles } from '../../../helpers/express-utils' import { getDurationFromVideoFile } from '../../../helpers/ffmpeg-utils' import { logger } from '../../../helpers/logger' -import { CONSTRAINTS_FIELDS, OVERVIEWS } from '../../../initializers/constants' -import { authenticatePromiseIfNeeded } from '../../oauth' -import { areValidationErrors } from '../utils' -import { cleanUpReqFiles } from '../../../helpers/express-utils' -import { VideoModel } from '../../../models/video/video' -import { checkUserCanTerminateOwnershipChange, doesChangeVideoOwnershipExist } from '../../../helpers/custom-validators/video-ownership' -import { VideoChangeOwnershipAccept } from '../../../../shared/models/videos/video-change-ownership-accept.model' -import { AccountModel } from '../../../models/account/account' -import { isNSFWQueryValid, isNumberArray, isStringArray } from '../../../helpers/custom-validators/search' -import { CONFIG } from '../../../initializers/config' -import { isLocalVideoAccepted } from '../../../lib/moderation' -import { Hooks } from '../../../lib/plugins/hooks' import { checkUserCanManageVideo, doesVideoChannelOfAccountExist, doesVideoExist, doesVideoFileOfVideoExist } from '../../../helpers/middlewares' -import { MVideoFullLight } from '@server/types/models' import { getVideoWithAttributes } from '../../../helpers/video' -import { getServerActor } from '@server/models/application/application' +import { CONFIG } from '../../../initializers/config' +import { CONSTRAINTS_FIELDS, OVERVIEWS } from '../../../initializers/constants' +import { isLocalVideoAccepted } from '../../../lib/moderation' +import { Hooks } from '../../../lib/plugins/hooks' +import { AccountModel } from '../../../models/account/account' +import { VideoModel } from '../../../models/video/video' +import { authenticatePromiseIfNeeded } from '../../oauth' +import { areValidationErrors } from '../utils' const videosAddValidator = getCommonVideoEditAttributes().concat([ body('videofile') @@ -148,7 +148,9 @@ async function checkVideoFollowConstraints (req: express.Request, res: express.R return res.status(403) .json({ - error: 'Cannot get this video regarding follow constraints.' + errorCode: ServerErrorCode.DOES_NOT_RESPECT_FOLLOW_CONSTRAINTS, + error: 'Cannot get this video regarding follow constraints.', + originUrl: video.url }) } diff --git a/shared/models/server/index.ts b/shared/models/server/index.ts index b079bc9a0..f253a3a79 100644 --- a/shared/models/server/index.ts +++ b/shared/models/server/index.ts @@ -7,4 +7,5 @@ export * from './emailer.model' export * from './job.model' export * from './log-level.type' export * from './server-config.model' +export * from './server-error-code.enum' export * from './server-stats.model' diff --git a/shared/models/server/server-error-code.enum.ts b/shared/models/server/server-error-code.enum.ts new file mode 100644 index 000000000..0bfb2c470 --- /dev/null +++ b/shared/models/server/server-error-code.enum.ts @@ -0,0 +1,3 @@ +export const enum ServerErrorCode { + DOES_NOT_RESPECT_FOLLOW_CONSTRAINTS = 1 +} -- cgit v1.2.3