2017-11-20 04:24:29 -05:00
|
|
|
import { Transaction } from 'sequelize'
|
2017-12-19 04:34:56 -05:00
|
|
|
import { VideoPrivacy } from '../../../shared/models/videos'
|
2017-12-28 05:16:08 -05:00
|
|
|
import { getServerActor } from '../../helpers/utils'
|
2017-12-12 11:53:50 -05:00
|
|
|
import { VideoShareModel } from '../../models/video/video-share'
|
2018-05-11 09:10:13 -04:00
|
|
|
import { sendUndoAnnounce, sendVideoAnnounce } from './send'
|
2018-11-14 09:01:28 -05:00
|
|
|
import { getVideoAnnounceActivityPubUrl } from './url'
|
2018-08-22 10:15:35 -04:00
|
|
|
import * as Bluebird from 'bluebird'
|
|
|
|
import { doRequest } from '../../helpers/requests'
|
|
|
|
import { getOrCreateActorAndServerAndModel } from './actor'
|
|
|
|
import { logger } from '../../helpers/logger'
|
2019-04-11 08:26:41 -04:00
|
|
|
import { CRAWL_REQUEST_CONCURRENCY } from '../../initializers/constants'
|
2019-01-15 05:14:12 -05:00
|
|
|
import { checkUrlsSameHost, getAPId } from '../../helpers/activitypub'
|
2019-08-15 05:53:26 -04:00
|
|
|
import { MChannelActor, MChannelActorLight, MVideo, MVideoAccountLight, MVideoId } from '../../typings/models/video'
|
2017-11-20 04:24:29 -05:00
|
|
|
|
2019-08-15 05:53:26 -04:00
|
|
|
async function shareVideoByServerAndChannel (video: MVideoAccountLight, t: Transaction) {
|
2017-12-19 08:22:38 -05:00
|
|
|
if (video.privacy === VideoPrivacy.PRIVATE) return undefined
|
2017-12-19 04:34:56 -05:00
|
|
|
|
2018-05-11 09:10:13 -04:00
|
|
|
return Promise.all([
|
|
|
|
shareByServer(video, t),
|
|
|
|
shareByVideoChannel(video, t)
|
|
|
|
])
|
|
|
|
}
|
|
|
|
|
2019-08-15 05:53:26 -04:00
|
|
|
async function changeVideoChannelShare (
|
|
|
|
video: MVideoAccountLight,
|
|
|
|
oldVideoChannel: MChannelActorLight,
|
|
|
|
t: Transaction
|
|
|
|
) {
|
2018-09-04 04:22:10 -04:00
|
|
|
logger.info('Updating video channel of video %s: %s -> %s.', video.uuid, oldVideoChannel.name, video.VideoChannel.name)
|
|
|
|
|
2018-05-11 09:10:13 -04:00
|
|
|
await undoShareByVideoChannel(video, oldVideoChannel, t)
|
|
|
|
|
|
|
|
await shareByVideoChannel(video, t)
|
|
|
|
}
|
|
|
|
|
2019-08-15 05:53:26 -04:00
|
|
|
async function addVideoShares (shareUrls: string[], video: MVideoId) {
|
2018-08-22 10:15:35 -04:00
|
|
|
await Bluebird.map(shareUrls, async shareUrl => {
|
|
|
|
try {
|
|
|
|
// Fetch url
|
|
|
|
const { body } = await doRequest({
|
|
|
|
uri: shareUrl,
|
|
|
|
json: true,
|
|
|
|
activityPub: true
|
|
|
|
})
|
2018-11-14 09:01:28 -05:00
|
|
|
if (!body || !body.actor) throw new Error('Body or body actor is invalid')
|
|
|
|
|
2019-01-15 05:14:12 -05:00
|
|
|
const actorUrl = getAPId(body.actor)
|
2018-11-14 09:01:28 -05:00
|
|
|
if (checkUrlsSameHost(shareUrl, actorUrl) !== true) {
|
|
|
|
throw new Error(`Actor url ${actorUrl} has not the same host than the share url ${shareUrl}`)
|
|
|
|
}
|
2018-08-22 10:15:35 -04:00
|
|
|
|
|
|
|
const actor = await getOrCreateActorAndServerAndModel(actorUrl)
|
|
|
|
|
|
|
|
const entry = {
|
|
|
|
actorId: actor.id,
|
2019-08-15 05:53:26 -04:00
|
|
|
videoId: video.id,
|
2018-08-22 10:15:35 -04:00
|
|
|
url: shareUrl
|
|
|
|
}
|
|
|
|
|
2019-03-19 11:23:02 -04:00
|
|
|
await VideoShareModel.upsert(entry)
|
2018-08-22 10:15:35 -04:00
|
|
|
} catch (err) {
|
|
|
|
logger.warn('Cannot add share %s.', shareUrl, { err })
|
|
|
|
}
|
|
|
|
}, { concurrency: CRAWL_REQUEST_CONCURRENCY })
|
|
|
|
}
|
|
|
|
|
2018-05-11 09:10:13 -04:00
|
|
|
export {
|
|
|
|
changeVideoChannelShare,
|
2018-08-22 10:15:35 -04:00
|
|
|
addVideoShares,
|
2018-05-11 09:10:13 -04:00
|
|
|
shareVideoByServerAndChannel
|
|
|
|
}
|
|
|
|
|
|
|
|
// ---------------------------------------------------------------------------
|
|
|
|
|
2019-08-15 05:53:26 -04:00
|
|
|
async function shareByServer (video: MVideo, t: Transaction) {
|
2017-12-14 11:38:41 -05:00
|
|
|
const serverActor = await getServerActor()
|
2017-11-20 04:24:29 -05:00
|
|
|
|
2018-11-14 09:01:28 -05:00
|
|
|
const serverShareUrl = getVideoAnnounceActivityPubUrl(serverActor, video)
|
2019-01-10 09:39:51 -05:00
|
|
|
const [ serverShare ] = await VideoShareModel.findOrCreate({
|
2018-03-19 10:02:36 -04:00
|
|
|
defaults: {
|
|
|
|
actorId: serverActor.id,
|
|
|
|
videoId: video.id,
|
|
|
|
url: serverShareUrl
|
|
|
|
},
|
|
|
|
where: {
|
|
|
|
url: serverShareUrl
|
|
|
|
},
|
|
|
|
transaction: t
|
|
|
|
})
|
2019-01-10 09:39:51 -05:00
|
|
|
|
|
|
|
return sendVideoAnnounce(serverActor, serverShare, video, t)
|
2018-05-11 09:10:13 -04:00
|
|
|
}
|
2017-11-20 04:24:29 -05:00
|
|
|
|
2019-08-15 05:53:26 -04:00
|
|
|
async function shareByVideoChannel (video: MVideoAccountLight, t: Transaction) {
|
2018-11-14 09:01:28 -05:00
|
|
|
const videoChannelShareUrl = getVideoAnnounceActivityPubUrl(video.VideoChannel.Actor, video)
|
2019-01-10 09:39:51 -05:00
|
|
|
const [ videoChannelShare ] = await VideoShareModel.findOrCreate({
|
2018-03-19 10:02:36 -04:00
|
|
|
defaults: {
|
|
|
|
actorId: video.VideoChannel.actorId,
|
|
|
|
videoId: video.id,
|
|
|
|
url: videoChannelShareUrl
|
|
|
|
},
|
|
|
|
where: {
|
|
|
|
url: videoChannelShareUrl
|
|
|
|
},
|
|
|
|
transaction: t
|
|
|
|
})
|
2019-01-10 09:39:51 -05:00
|
|
|
|
|
|
|
return sendVideoAnnounce(video.VideoChannel.Actor, videoChannelShare, video, t)
|
2017-11-20 04:24:29 -05:00
|
|
|
}
|
|
|
|
|
2019-08-15 05:53:26 -04:00
|
|
|
async function undoShareByVideoChannel (video: MVideo, oldVideoChannel: MChannelActorLight, t: Transaction) {
|
2018-05-11 09:10:13 -04:00
|
|
|
// Load old share
|
|
|
|
const oldShare = await VideoShareModel.load(oldVideoChannel.actorId, video.id, t)
|
|
|
|
if (!oldShare) return new Error('Cannot find old video channel share ' + oldVideoChannel.actorId + ' for video ' + video.id)
|
|
|
|
|
|
|
|
await sendUndoAnnounce(oldVideoChannel.Actor, oldShare, video, t)
|
|
|
|
await oldShare.destroy({ transaction: t })
|
2017-11-20 04:24:29 -05:00
|
|
|
}
|