Check follow constraints when getting a video
This commit is contained in:
parent
5776f78e3b
commit
8d4273463f
|
@ -114,7 +114,7 @@ export class VideoWatchComponent implements OnInit, OnDestroy {
|
||||||
)
|
)
|
||||||
.pipe(
|
.pipe(
|
||||||
// If 401, the video is private or blacklisted so redirect to 404
|
// If 401, the video is private or blacklisted so redirect to 404
|
||||||
catchError(err => this.restExtractor.redirectTo404IfNotFound(err, [ 400, 401, 404 ]))
|
catchError(err => this.restExtractor.redirectTo404IfNotFound(err, [ 400, 401, 403, 404 ]))
|
||||||
)
|
)
|
||||||
.subscribe(([ video, captionsResult ]) => {
|
.subscribe(([ video, captionsResult ]) => {
|
||||||
const startTime = this.route.snapshot.queryParams.start
|
const startTime = this.route.snapshot.queryParams.start
|
||||||
|
|
|
@ -58,7 +58,10 @@ log:
|
||||||
level: 'info' # debug/info/warning/error
|
level: 'info' # debug/info/warning/error
|
||||||
|
|
||||||
search:
|
search:
|
||||||
remote_uri: # Add ability to fetch remote videos/actors by their URI, that may not be federated with your instance
|
# Add ability to fetch remote videos/actors by their URI, that may not be federated with your instance
|
||||||
|
# If enabled, the associated group will be able to "escape" from the instance follows
|
||||||
|
# That means they will be able to follow channels, watch videos, list videos of non followed instances
|
||||||
|
remote_uri:
|
||||||
users: true
|
users: true
|
||||||
anonymous: false
|
anonymous: false
|
||||||
|
|
||||||
|
|
|
@ -59,7 +59,10 @@ log:
|
||||||
level: 'info' # debug/info/warning/error
|
level: 'info' # debug/info/warning/error
|
||||||
|
|
||||||
search:
|
search:
|
||||||
remote_uri: # Add ability to search remote videos/actors by URI, that may not be federated with your instance
|
# Add ability to fetch remote videos/actors by their URI, that may not be federated with your instance
|
||||||
|
# If enabled, the associated group will be able to "escape" from the instance follows
|
||||||
|
# That means they will be able to follow channels, watch videos, list videos of non followed instances
|
||||||
|
remote_uri:
|
||||||
users: true
|
users: true
|
||||||
anonymous: false
|
anonymous: false
|
||||||
|
|
||||||
|
|
|
@ -31,6 +31,7 @@ import {
|
||||||
asyncMiddleware,
|
asyncMiddleware,
|
||||||
asyncRetryTransactionMiddleware,
|
asyncRetryTransactionMiddleware,
|
||||||
authenticate,
|
authenticate,
|
||||||
|
checkVideoFollowConstraints,
|
||||||
commonVideosFiltersValidator,
|
commonVideosFiltersValidator,
|
||||||
optionalAuthenticate,
|
optionalAuthenticate,
|
||||||
paginationValidator,
|
paginationValidator,
|
||||||
|
@ -123,6 +124,7 @@ videosRouter.get('/:id/description',
|
||||||
videosRouter.get('/:id',
|
videosRouter.get('/:id',
|
||||||
optionalAuthenticate,
|
optionalAuthenticate,
|
||||||
asyncMiddleware(videosGetValidator),
|
asyncMiddleware(videosGetValidator),
|
||||||
|
asyncMiddleware(checkVideoFollowConstraints),
|
||||||
getVideo
|
getVideo
|
||||||
)
|
)
|
||||||
videosRouter.post('/:id/views',
|
videosRouter.post('/:id/views',
|
||||||
|
|
|
@ -28,9 +28,24 @@ function authenticate (req: express.Request, res: express.Response, next: expres
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function authenticatePromiseIfNeeded (req: express.Request, res: express.Response) {
|
||||||
|
return new Promise(resolve => {
|
||||||
|
// Already authenticated? (or tried to)
|
||||||
|
if (res.locals.oauth && res.locals.oauth.token.User) return resolve()
|
||||||
|
|
||||||
|
if (res.locals.authenticated === false) return res.sendStatus(401)
|
||||||
|
|
||||||
|
authenticate(req, res, () => {
|
||||||
|
return resolve()
|
||||||
|
})
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
function optionalAuthenticate (req: express.Request, res: express.Response, next: express.NextFunction) {
|
function optionalAuthenticate (req: express.Request, res: express.Response, next: express.NextFunction) {
|
||||||
if (req.header('authorization')) return authenticate(req, res, next)
|
if (req.header('authorization')) return authenticate(req, res, next)
|
||||||
|
|
||||||
|
res.locals.authenticated = false
|
||||||
|
|
||||||
return next()
|
return next()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -53,6 +68,7 @@ function token (req: express.Request, res: express.Response, next: express.NextF
|
||||||
|
|
||||||
export {
|
export {
|
||||||
authenticate,
|
authenticate,
|
||||||
|
authenticatePromiseIfNeeded,
|
||||||
optionalAuthenticate,
|
optionalAuthenticate,
|
||||||
token
|
token
|
||||||
}
|
}
|
||||||
|
|
|
@ -31,8 +31,8 @@ import {
|
||||||
} from '../../../helpers/custom-validators/videos'
|
} from '../../../helpers/custom-validators/videos'
|
||||||
import { getDurationFromVideoFile } from '../../../helpers/ffmpeg-utils'
|
import { getDurationFromVideoFile } from '../../../helpers/ffmpeg-utils'
|
||||||
import { logger } from '../../../helpers/logger'
|
import { logger } from '../../../helpers/logger'
|
||||||
import { CONSTRAINTS_FIELDS } from '../../../initializers'
|
import { CONFIG, CONSTRAINTS_FIELDS } from '../../../initializers'
|
||||||
import { authenticate } from '../../oauth'
|
import { authenticatePromiseIfNeeded } from '../../oauth'
|
||||||
import { areValidationErrors } from '../utils'
|
import { areValidationErrors } from '../utils'
|
||||||
import { cleanUpReqFiles } from '../../../helpers/express-utils'
|
import { cleanUpReqFiles } from '../../../helpers/express-utils'
|
||||||
import { VideoModel } from '../../../models/video/video'
|
import { VideoModel } from '../../../models/video/video'
|
||||||
|
@ -43,6 +43,7 @@ import { VideoChangeOwnershipModel } from '../../../models/video/video-change-ow
|
||||||
import { AccountModel } from '../../../models/account/account'
|
import { AccountModel } from '../../../models/account/account'
|
||||||
import { VideoFetchType } from '../../../helpers/video'
|
import { VideoFetchType } from '../../../helpers/video'
|
||||||
import { isNSFWQueryValid, isNumberArray, isStringArray } from '../../../helpers/custom-validators/search'
|
import { isNSFWQueryValid, isNumberArray, isStringArray } from '../../../helpers/custom-validators/search'
|
||||||
|
import { getServerActor } from '../../../helpers/utils'
|
||||||
|
|
||||||
const videosAddValidator = getCommonVideoAttributes().concat([
|
const videosAddValidator = getCommonVideoAttributes().concat([
|
||||||
body('videofile')
|
body('videofile')
|
||||||
|
@ -127,6 +128,31 @@ const videosUpdateValidator = getCommonVideoAttributes().concat([
|
||||||
}
|
}
|
||||||
])
|
])
|
||||||
|
|
||||||
|
async function checkVideoFollowConstraints (req: express.Request, res: express.Response, next: express.NextFunction) {
|
||||||
|
const video: VideoModel = res.locals.video
|
||||||
|
|
||||||
|
// Anybody can watch local videos
|
||||||
|
if (video.isOwned() === true) return next()
|
||||||
|
|
||||||
|
// Logged user
|
||||||
|
if (res.locals.oauth) {
|
||||||
|
// Users can search or watch remote videos
|
||||||
|
if (CONFIG.SEARCH.REMOTE_URI.USERS === true) return next()
|
||||||
|
}
|
||||||
|
|
||||||
|
// Anybody can search or watch remote videos
|
||||||
|
if (CONFIG.SEARCH.REMOTE_URI.ANONYMOUS === true) return next()
|
||||||
|
|
||||||
|
// Check our instance follows an actor that shared this video
|
||||||
|
const serverActor = await getServerActor()
|
||||||
|
if (await VideoModel.checkVideoHasInstanceFollow(video.id, serverActor.id) === true) return next()
|
||||||
|
|
||||||
|
return res.status(403)
|
||||||
|
.json({
|
||||||
|
error: 'Cannot get this video regarding follow constraints.'
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
const videosCustomGetValidator = (fetchType: VideoFetchType) => {
|
const videosCustomGetValidator = (fetchType: VideoFetchType) => {
|
||||||
return [
|
return [
|
||||||
param('id').custom(isIdOrUUIDValid).not().isEmpty().withMessage('Should have a valid id'),
|
param('id').custom(isIdOrUUIDValid).not().isEmpty().withMessage('Should have a valid id'),
|
||||||
|
@ -141,17 +167,20 @@ const videosCustomGetValidator = (fetchType: VideoFetchType) => {
|
||||||
|
|
||||||
// Video private or blacklisted
|
// Video private or blacklisted
|
||||||
if (video.privacy === VideoPrivacy.PRIVATE || video.VideoBlacklist) {
|
if (video.privacy === VideoPrivacy.PRIVATE || video.VideoBlacklist) {
|
||||||
return authenticate(req, res, () => {
|
await authenticatePromiseIfNeeded(req, res)
|
||||||
const user: UserModel = res.locals.oauth.token.User
|
|
||||||
|
|
||||||
// Only the owner or a user that have blacklist rights can see the video
|
const user: UserModel = res.locals.oauth ? res.locals.oauth.token.User : null
|
||||||
if (video.VideoChannel.Account.userId !== user.id && !user.hasRight(UserRight.MANAGE_VIDEO_BLACKLIST)) {
|
|
||||||
return res.status(403)
|
|
||||||
.json({ error: 'Cannot get this private or blacklisted video.' })
|
|
||||||
}
|
|
||||||
|
|
||||||
return next()
|
// Only the owner or a user that have blacklist rights can see the video
|
||||||
})
|
if (
|
||||||
|
!user ||
|
||||||
|
(video.VideoChannel.Account.userId !== user.id && !user.hasRight(UserRight.MANAGE_VIDEO_BLACKLIST))
|
||||||
|
) {
|
||||||
|
return res.status(403)
|
||||||
|
.json({ error: 'Cannot get this private or blacklisted video.' })
|
||||||
|
}
|
||||||
|
|
||||||
|
return next()
|
||||||
}
|
}
|
||||||
|
|
||||||
// Video is public, anyone can access it
|
// Video is public, anyone can access it
|
||||||
|
@ -376,6 +405,7 @@ export {
|
||||||
videosAddValidator,
|
videosAddValidator,
|
||||||
videosUpdateValidator,
|
videosUpdateValidator,
|
||||||
videosGetValidator,
|
videosGetValidator,
|
||||||
|
checkVideoFollowConstraints,
|
||||||
videosCustomGetValidator,
|
videosCustomGetValidator,
|
||||||
videosRemoveValidator,
|
videosRemoveValidator,
|
||||||
|
|
||||||
|
|
|
@ -1253,6 +1253,23 @@ export class VideoModel extends Model<VideoModel> {
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static checkVideoHasInstanceFollow (videoId: number, followerActorId: number) {
|
||||||
|
// Instances only share videos
|
||||||
|
const query = 'SELECT 1 FROM "videoShare" ' +
|
||||||
|
'INNER JOIN "actorFollow" ON "actorFollow"."targetActorId" = "videoShare"."actorId" ' +
|
||||||
|
'WHERE "actorFollow"."actorId" = $followerActorId AND "videoShare"."videoId" = $videoId ' +
|
||||||
|
'LIMIT 1'
|
||||||
|
|
||||||
|
const options = {
|
||||||
|
type: Sequelize.QueryTypes.SELECT,
|
||||||
|
bind: { followerActorId, videoId },
|
||||||
|
raw: true
|
||||||
|
}
|
||||||
|
|
||||||
|
return VideoModel.sequelize.query(query, options)
|
||||||
|
.then(results => results.length === 1)
|
||||||
|
}
|
||||||
|
|
||||||
// threshold corresponds to how many video the field should have to be returned
|
// threshold corresponds to how many video the field should have to be returned
|
||||||
static async getRandomFieldSamples (field: 'category' | 'channelId', threshold: number, count: number) {
|
static async getRandomFieldSamples (field: 'category' | 'channelId', threshold: number, count: number) {
|
||||||
const serverActor = await getServerActor()
|
const serverActor = await getServerActor()
|
||||||
|
|
|
@ -0,0 +1,215 @@
|
||||||
|
/* tslint:disable:no-unused-expression */
|
||||||
|
|
||||||
|
import * as chai from 'chai'
|
||||||
|
import 'mocha'
|
||||||
|
import { doubleFollow, getAccountVideos, getVideo, getVideoChannelVideos, getVideoWithToken } from '../../utils'
|
||||||
|
import { flushAndRunMultipleServers, killallServers, ServerInfo, setAccessTokensToServers, uploadVideo } from '../../utils/index'
|
||||||
|
import { unfollow } from '../../utils/server/follows'
|
||||||
|
import { userLogin } from '../../utils/users/login'
|
||||||
|
import { createUser } from '../../utils/users/users'
|
||||||
|
|
||||||
|
const expect = chai.expect
|
||||||
|
|
||||||
|
describe('Test follow constraints', function () {
|
||||||
|
let servers: ServerInfo[] = []
|
||||||
|
let video1UUID: string
|
||||||
|
let video2UUID: string
|
||||||
|
let userAccessToken: string
|
||||||
|
|
||||||
|
before(async function () {
|
||||||
|
this.timeout(30000)
|
||||||
|
|
||||||
|
servers = await flushAndRunMultipleServers(2)
|
||||||
|
|
||||||
|
// Get the access tokens
|
||||||
|
await setAccessTokensToServers(servers)
|
||||||
|
|
||||||
|
{
|
||||||
|
const res = await uploadVideo(servers[ 0 ].url, servers[ 0 ].accessToken, { name: 'video server 1' })
|
||||||
|
video1UUID = res.body.video.uuid
|
||||||
|
}
|
||||||
|
{
|
||||||
|
const res = await uploadVideo(servers[ 1 ].url, servers[ 1 ].accessToken, { name: 'video server 2' })
|
||||||
|
video2UUID = res.body.video.uuid
|
||||||
|
}
|
||||||
|
|
||||||
|
const user = {
|
||||||
|
username: 'user1',
|
||||||
|
password: 'super_password'
|
||||||
|
}
|
||||||
|
await createUser(servers[0].url, servers[0].accessToken, user.username, user.password)
|
||||||
|
userAccessToken = await userLogin(servers[0], user)
|
||||||
|
|
||||||
|
await doubleFollow(servers[0], servers[1])
|
||||||
|
})
|
||||||
|
|
||||||
|
describe('With a followed instance', function () {
|
||||||
|
|
||||||
|
describe('With an unlogged user', function () {
|
||||||
|
|
||||||
|
it('Should get the local video', async function () {
|
||||||
|
await getVideo(servers[0].url, video1UUID, 200)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('Should get the remote video', async function () {
|
||||||
|
await getVideo(servers[0].url, video2UUID, 200)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('Should list local account videos', async function () {
|
||||||
|
const res = await getAccountVideos(servers[0].url, undefined, 'root@localhost:9001', 0, 5)
|
||||||
|
|
||||||
|
expect(res.body.total).to.equal(1)
|
||||||
|
expect(res.body.data).to.have.lengthOf(1)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('Should list remote account videos', async function () {
|
||||||
|
const res = await getAccountVideos(servers[0].url, undefined, 'root@localhost:9002', 0, 5)
|
||||||
|
|
||||||
|
expect(res.body.total).to.equal(1)
|
||||||
|
expect(res.body.data).to.have.lengthOf(1)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('Should list local channel videos', async function () {
|
||||||
|
const res = await getVideoChannelVideos(servers[0].url, undefined, 'root_channel@localhost:9001', 0, 5)
|
||||||
|
|
||||||
|
expect(res.body.total).to.equal(1)
|
||||||
|
expect(res.body.data).to.have.lengthOf(1)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('Should list remote channel videos', async function () {
|
||||||
|
const res = await getVideoChannelVideos(servers[0].url, undefined, 'root_channel@localhost:9002', 0, 5)
|
||||||
|
|
||||||
|
expect(res.body.total).to.equal(1)
|
||||||
|
expect(res.body.data).to.have.lengthOf(1)
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
describe('With a logged user', function () {
|
||||||
|
it('Should get the local video', async function () {
|
||||||
|
await getVideoWithToken(servers[0].url, userAccessToken, video1UUID, 200)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('Should get the remote video', async function () {
|
||||||
|
await getVideoWithToken(servers[0].url, userAccessToken, video2UUID, 200)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('Should list local account videos', async function () {
|
||||||
|
const res = await getAccountVideos(servers[0].url, userAccessToken, 'root@localhost:9001', 0, 5)
|
||||||
|
|
||||||
|
expect(res.body.total).to.equal(1)
|
||||||
|
expect(res.body.data).to.have.lengthOf(1)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('Should list remote account videos', async function () {
|
||||||
|
const res = await getAccountVideos(servers[0].url, userAccessToken, 'root@localhost:9002', 0, 5)
|
||||||
|
|
||||||
|
expect(res.body.total).to.equal(1)
|
||||||
|
expect(res.body.data).to.have.lengthOf(1)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('Should list local channel videos', async function () {
|
||||||
|
const res = await getVideoChannelVideos(servers[0].url, userAccessToken, 'root_channel@localhost:9001', 0, 5)
|
||||||
|
|
||||||
|
expect(res.body.total).to.equal(1)
|
||||||
|
expect(res.body.data).to.have.lengthOf(1)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('Should list remote channel videos', async function () {
|
||||||
|
const res = await getVideoChannelVideos(servers[0].url, userAccessToken, 'root_channel@localhost:9002', 0, 5)
|
||||||
|
|
||||||
|
expect(res.body.total).to.equal(1)
|
||||||
|
expect(res.body.data).to.have.lengthOf(1)
|
||||||
|
})
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
describe('With a non followed instance', function () {
|
||||||
|
|
||||||
|
before(async function () {
|
||||||
|
this.timeout(30000)
|
||||||
|
|
||||||
|
await unfollow(servers[0].url, servers[0].accessToken, servers[1])
|
||||||
|
})
|
||||||
|
|
||||||
|
describe('With an unlogged user', function () {
|
||||||
|
|
||||||
|
it('Should get the local video', async function () {
|
||||||
|
await getVideo(servers[0].url, video1UUID, 200)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('Should not get the remote video', async function () {
|
||||||
|
await getVideo(servers[0].url, video2UUID, 403)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('Should list local account videos', async function () {
|
||||||
|
const res = await getAccountVideos(servers[0].url, undefined, 'root@localhost:9001', 0, 5)
|
||||||
|
|
||||||
|
expect(res.body.total).to.equal(1)
|
||||||
|
expect(res.body.data).to.have.lengthOf(1)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('Should not list remote account videos', async function () {
|
||||||
|
const res = await getAccountVideos(servers[0].url, undefined, 'root@localhost:9002', 0, 5)
|
||||||
|
|
||||||
|
expect(res.body.total).to.equal(0)
|
||||||
|
expect(res.body.data).to.have.lengthOf(0)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('Should list local channel videos', async function () {
|
||||||
|
const res = await getVideoChannelVideos(servers[0].url, undefined, 'root_channel@localhost:9001', 0, 5)
|
||||||
|
|
||||||
|
expect(res.body.total).to.equal(1)
|
||||||
|
expect(res.body.data).to.have.lengthOf(1)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('Should not list remote channel videos', async function () {
|
||||||
|
const res = await getVideoChannelVideos(servers[0].url, undefined, 'root_channel@localhost:9002', 0, 5)
|
||||||
|
|
||||||
|
expect(res.body.total).to.equal(0)
|
||||||
|
expect(res.body.data).to.have.lengthOf(0)
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
describe('With a logged user', function () {
|
||||||
|
it('Should get the local video', async function () {
|
||||||
|
await getVideoWithToken(servers[0].url, userAccessToken, video1UUID, 200)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('Should get the remote video', async function () {
|
||||||
|
await getVideoWithToken(servers[0].url, userAccessToken, video2UUID, 200)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('Should list local account videos', async function () {
|
||||||
|
const res = await getAccountVideos(servers[0].url, userAccessToken, 'root@localhost:9001', 0, 5)
|
||||||
|
|
||||||
|
expect(res.body.total).to.equal(1)
|
||||||
|
expect(res.body.data).to.have.lengthOf(1)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('Should list remote account videos', async function () {
|
||||||
|
const res = await getAccountVideos(servers[0].url, userAccessToken, 'root@localhost:9002', 0, 5)
|
||||||
|
|
||||||
|
expect(res.body.total).to.equal(1)
|
||||||
|
expect(res.body.data).to.have.lengthOf(1)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('Should list local channel videos', async function () {
|
||||||
|
const res = await getVideoChannelVideos(servers[0].url, userAccessToken, 'root_channel@localhost:9001', 0, 5)
|
||||||
|
|
||||||
|
expect(res.body.total).to.equal(1)
|
||||||
|
expect(res.body.data).to.have.lengthOf(1)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('Should list remote channel videos', async function () {
|
||||||
|
const res = await getVideoChannelVideos(servers[0].url, userAccessToken, 'root_channel@localhost:9002', 0, 5)
|
||||||
|
|
||||||
|
expect(res.body.total).to.equal(1)
|
||||||
|
expect(res.body.data).to.have.lengthOf(1)
|
||||||
|
})
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
after(async function () {
|
||||||
|
killallServers(servers)
|
||||||
|
})
|
||||||
|
})
|
|
@ -1,5 +1,6 @@
|
||||||
import './config'
|
import './config'
|
||||||
import './email'
|
import './email'
|
||||||
|
import './follow-constraints'
|
||||||
import './follows'
|
import './follows'
|
||||||
import './handle-down'
|
import './handle-down'
|
||||||
import './jobs'
|
import './jobs'
|
||||||
|
|
Loading…
Reference in New Issue