import { isArray } from '@server/helpers/custom-validators/misc' import { generateMagnetUri } from '@server/helpers/webtorrent' import { getActivityStreamDuration } from '@server/lib/activitypub/activity' import { getLocalVideoFileMetadataUrl } from '@server/lib/video-urls' import { ActivityIconObject, ActivityPlaylistUrlObject, ActivityPubStoryboard, ActivityTagObject, ActivityTrackerUrlObject, ActivityUrlObject, VideoObject } from '@shared/models' import { MIMETYPES, WEBSERVER } from '../../../initializers/constants' import { getLocalVideoCommentsActivityPubUrl, getLocalVideoDislikesActivityPubUrl, getLocalVideoLikesActivityPubUrl, getLocalVideoSharesActivityPubUrl } from '../../../lib/activitypub/url' import { MStreamingPlaylistFiles, MUserId, MVideo, MVideoAP, MVideoFile } from '../../../types/models' import { VideoCaptionModel } from '../video-caption' import { sortByResolutionDesc } from './shared' import { getCategoryLabel, getLanguageLabel, getLicenceLabel } from './video-api-format' export function videoModelToActivityPubObject (video: MVideoAP): VideoObject { const language = video.language ? { identifier: video.language, name: getLanguageLabel(video.language) } : undefined const category = video.category ? { identifier: video.category + '', name: getCategoryLabel(video.category) } : undefined const licence = video.licence ? { identifier: video.licence + '', name: getLicenceLabel(video.licence) } : undefined const url: ActivityUrlObject[] = [ // HTML url should be the first element in the array so Mastodon correctly displays the embed { type: 'Link', mediaType: 'text/html', href: WEBSERVER.URL + '/videos/watch/' + video.uuid } as ActivityUrlObject, ...buildVideoFileUrls({ video, files: video.VideoFiles }), ...buildStreamingPlaylistUrls(video), ...buildTrackerUrls(video) ] return { type: 'Video' as 'Video', id: video.url, name: video.name, duration: getActivityStreamDuration(video.duration), uuid: video.uuid, category, licence, language, views: video.views, sensitive: video.nsfw, waitTranscoding: video.waitTranscoding, state: video.state, commentsEnabled: video.commentsEnabled, downloadEnabled: video.downloadEnabled, published: video.publishedAt.toISOString(), originallyPublishedAt: video.originallyPublishedAt ? video.originallyPublishedAt.toISOString() : null, updated: video.updatedAt.toISOString(), uploadDate: video.inputFileUpdatedAt?.toISOString(), tag: buildTags(video), mediaType: 'text/markdown', content: video.description, support: video.support, subtitleLanguage: buildSubtitleLanguage(video), icon: buildIcon(video), preview: buildPreviewAPAttribute(video), url, likes: getLocalVideoLikesActivityPubUrl(video), dislikes: getLocalVideoDislikesActivityPubUrl(video), shares: getLocalVideoSharesActivityPubUrl(video), comments: getLocalVideoCommentsActivityPubUrl(video), attributedTo: [ { type: 'Person', id: video.VideoChannel.Account.Actor.url }, { type: 'Group', id: video.VideoChannel.Actor.url } ], ...buildLiveAPAttributes(video) } } // --------------------------------------------------------------------------- // Private // --------------------------------------------------------------------------- function buildLiveAPAttributes (video: MVideoAP) { if (!video.isLive) { return { isLiveBroadcast: false, liveSaveReplay: null, permanentLive: null, latencyMode: null } } return { isLiveBroadcast: true, liveSaveReplay: video.VideoLive.saveReplay, permanentLive: video.VideoLive.permanentLive, latencyMode: video.VideoLive.latencyMode } } function buildPreviewAPAttribute (video: MVideoAP): ActivityPubStoryboard[] { if (!video.Storyboard) return undefined const storyboard = video.Storyboard return [ { type: 'Image', rel: [ 'storyboard' ], url: [ { mediaType: 'image/jpeg', href: storyboard.getOriginFileUrl(video), width: storyboard.totalWidth, height: storyboard.totalHeight, tileWidth: storyboard.spriteWidth, tileHeight: storyboard.spriteHeight, tileDuration: getActivityStreamDuration(storyboard.spriteDuration) } ] } ] } function buildVideoFileUrls (options: { video: MVideo files: MVideoFile[] user?: MUserId }): ActivityUrlObject[] { const { video, files } = options if (!isArray(files)) return [] const urls: ActivityUrlObject[] = [] const trackerUrls = video.getTrackerUrls() const sortedFiles = files .filter(f => !f.isLive()) .sort(sortByResolutionDesc) for (const file of sortedFiles) { urls.push({ type: 'Link', mediaType: MIMETYPES.VIDEO.EXT_MIMETYPE[file.extname] as any, href: file.getFileUrl(video), height: file.resolution, size: file.size, fps: file.fps }) urls.push({ type: 'Link', rel: [ 'metadata', MIMETYPES.VIDEO.EXT_MIMETYPE[file.extname] ], mediaType: 'application/json' as 'application/json', href: getLocalVideoFileMetadataUrl(video, file), height: file.resolution, fps: file.fps }) if (file.hasTorrent()) { urls.push({ type: 'Link', mediaType: 'application/x-bittorrent' as 'application/x-bittorrent', href: file.getTorrentUrl(), height: file.resolution }) urls.push({ type: 'Link', mediaType: 'application/x-bittorrent;x-scheme-handler/magnet' as 'application/x-bittorrent;x-scheme-handler/magnet', href: generateMagnetUri(video, file, trackerUrls), height: file.resolution }) } } return urls } // --------------------------------------------------------------------------- function buildStreamingPlaylistUrls (video: MVideoAP): ActivityPlaylistUrlObject[] { if (!isArray(video.VideoStreamingPlaylists)) return [] return video.VideoStreamingPlaylists .map(playlist => ({ type: 'Link', mediaType: 'application/x-mpegURL' as 'application/x-mpegURL', href: playlist.getMasterPlaylistUrl(video), tag: buildStreamingPlaylistTags(video, playlist) })) } function buildStreamingPlaylistTags (video: MVideoAP, playlist: MStreamingPlaylistFiles) { return [ ...playlist.p2pMediaLoaderInfohashes.map(i => ({ type: 'Infohash' as 'Infohash', name: i })), { type: 'Link', name: 'sha256', mediaType: 'application/json' as 'application/json', href: playlist.getSha256SegmentsUrl(video) }, ...buildVideoFileUrls({ video, files: playlist.VideoFiles }) ] as ActivityTagObject[] } // --------------------------------------------------------------------------- function buildTrackerUrls (video: MVideoAP): ActivityTrackerUrlObject[] { return video.getTrackerUrls() .map(trackerUrl => { const rel2 = trackerUrl.startsWith('http') ? 'http' : 'websocket' return { type: 'Link', name: `tracker-${rel2}`, rel: [ 'tracker', rel2 ], href: trackerUrl } }) } // --------------------------------------------------------------------------- function buildTags (video: MVideoAP) { if (!isArray(video.Tags)) return [] return video.Tags.map(t => ({ type: 'Hashtag' as 'Hashtag', name: t.name })) } function buildIcon (video: MVideoAP): ActivityIconObject[] { return [ video.getMiniature(), video.getPreview() ] .map(i => ({ type: 'Image', url: i.getOriginFileUrl(video), mediaType: 'image/jpeg', width: i.width, height: i.height })) } function buildSubtitleLanguage (video: MVideoAP) { if (!isArray(video.VideoCaptions)) return [] return video.VideoCaptions .map(caption => ({ identifier: caption.language, name: VideoCaptionModel.getLanguageLabel(caption.language), url: caption.getFileUrl(video) })) }