X-Git-Url: https://git.immae.eu/?a=blobdiff_plain;f=server%2Fhelpers%2Factivitypub.ts;h=fe721cbac1f01e351d157dd8a7d965c45ffd8c1d;hb=743dab5517d4501f6b35cfc795de6c8b6f41ebb3;hp=a9de11fb0984ff8b76f95bb84259b9c6b5a03b71;hpb=3d52b300ea79bec21f090e2447c4808307078618;p=github%2FChocobozzz%2FPeerTube.git diff --git a/server/helpers/activitypub.ts b/server/helpers/activitypub.ts index a9de11fb0..fe721cbac 100644 --- a/server/helpers/activitypub.ts +++ b/server/helpers/activitypub.ts @@ -1,35 +1,102 @@ -import * as Bluebird from 'bluebird' -import * as validator from 'validator' +import Bluebird from 'bluebird' +import { URL } from 'url' +import validator from 'validator' +import { ContextType } from '@shared/models/activitypub/context' import { ResultList } from '../../shared/models' -import { Activity, ActivityPubActor } from '../../shared/models/activitypub' -import { ACTIVITY_PUB } from '../initializers' -import { ActorModel } from '../models/activitypub/actor' -import { signObject } from './peertube-crypto' +import { ACTIVITY_PUB, REMOTE_SCHEME } from '../initializers/constants' +import { MActor, MVideoWithHost } from '../types/models' import { pageToStartAndCount } from './core-utils' +import { signJsonLDObject } from './peertube-crypto' + +function getContextData (type: ContextType) { + const context: any[] = [ + 'https://www.w3.org/ns/activitystreams', + 'https://w3id.org/security/v1', + { + RsaSignature2017: 'https://w3id.org/security#RsaSignature2017' + } + ] + + if (type !== 'View' && type !== 'Announce') { + const additional = { + pt: 'https://joinpeertube.org/ns#', + sc: 'http://schema.org#' + } -function activityPubContextify (data: T) { - return Object.assign(data, { - '@context': [ - 'https://www.w3.org/ns/activitystreams', - 'https://w3id.org/security/v1', - { - RsaSignature2017: 'https://w3id.org/security#RsaSignature2017', + if (type === 'CacheFile') { + Object.assign(additional, { + expires: 'sc:expires', + CacheFile: 'pt:CacheFile' + }) + } else { + Object.assign(additional, { Hashtag: 'as:Hashtag', - uuid: 'http://schema.org/identifier', - category: 'http://schema.org/category', - licence: 'http://schema.org/license', - subtitleLanguage: 'http://schema.org/subtitleLanguage', + uuid: 'sc:identifier', + category: 'sc:category', + licence: 'sc:license', + subtitleLanguage: 'sc:subtitleLanguage', sensitive: 'as:sensitive', - language: 'http://schema.org/inLanguage', - views: 'http://schema.org/Number', - stats: 'http://schema.org/Number', - size: 'http://schema.org/Number', - fps: 'http://schema.org/Number', - commentsEnabled: 'http://schema.org/Boolean', - waitTranscoding: 'http://schema.org/Boolean', - support: 'http://schema.org/Text' - }, - { + language: 'sc:inLanguage', + + isLiveBroadcast: 'sc:isLiveBroadcast', + liveSaveReplay: { + '@type': 'sc:Boolean', + '@id': 'pt:liveSaveReplay' + }, + permanentLive: { + '@type': 'sc:Boolean', + '@id': 'pt:permanentLive' + }, + + Infohash: 'pt:Infohash', + Playlist: 'pt:Playlist', + PlaylistElement: 'pt:PlaylistElement', + + originallyPublishedAt: 'sc:datePublished', + views: { + '@type': 'sc:Number', + '@id': 'pt:views' + }, + state: { + '@type': 'sc:Number', + '@id': 'pt:state' + }, + size: { + '@type': 'sc:Number', + '@id': 'pt:size' + }, + fps: { + '@type': 'sc:Number', + '@id': 'pt:fps' + }, + startTimestamp: { + '@type': 'sc:Number', + '@id': 'pt:startTimestamp' + }, + stopTimestamp: { + '@type': 'sc:Number', + '@id': 'pt:stopTimestamp' + }, + position: { + '@type': 'sc:Number', + '@id': 'pt:position' + }, + commentsEnabled: { + '@type': 'sc:Boolean', + '@id': 'pt:commentsEnabled' + }, + downloadEnabled: { + '@type': 'sc:Boolean', + '@id': 'pt:downloadEnabled' + }, + waitTranscoding: { + '@type': 'sc:Boolean', + '@id': 'pt:waitTranscoding' + }, + support: { + '@type': 'sc:Text', + '@id': 'pt:support' + }, likes: { '@id': 'as:likes', '@type': '@id' @@ -38,6 +105,10 @@ function activityPubContextify (data: T) { '@id': 'as:dislikes', '@type': '@id' }, + playlists: { + '@id': 'pt:playlists', + '@type': '@id' + }, shares: { '@id': 'as:shares', '@type': '@id' @@ -46,26 +117,41 @@ function activityPubContextify (data: T) { '@id': 'as:comments', '@type': '@id' } - } - ] - }) + }) + } + + context.push(additional) + } + + return { + '@context': context + } +} + +function activityPubContextify (data: T, type: ContextType = 'All') { + return Object.assign({}, data, getContextData(type)) } type ActivityPubCollectionPaginationHandler = (start: number, count: number) => Bluebird> | Promise> -async function activityPubCollectionPagination (url: string, handler: ActivityPubCollectionPaginationHandler, page?: any) { +async function activityPubCollectionPagination ( + baseUrl: string, + handler: ActivityPubCollectionPaginationHandler, + page?: any, + size = ACTIVITY_PUB.COLLECTION_ITEMS_PER_PAGE +) { if (!page || !validator.isInt(page)) { // We just display the first page URL, we only need the total items const result = await handler(0, 1) return { - id: url, - type: 'OrderedCollection', + id: baseUrl, + type: 'OrderedCollectionPage', totalItems: result.total, - first: url + '?page=1' + first: baseUrl + '?page=1' } } - const { start, count } = pageToStartAndCount(page, ACTIVITY_PUB.COLLECTION_ITEMS_PER_PAGE) + const { start, count } = pageToStartAndCount(page, size) const result = await handler(start, count) let next: string | undefined @@ -75,43 +161,60 @@ async function activityPubCollectionPagination (url: string, handler: ActivityPu page = parseInt(page, 10) // There are more results - if (result.total > page * ACTIVITY_PUB.COLLECTION_ITEMS_PER_PAGE) { - next = url + '?page=' + (page + 1) + if (result.total > page * size) { + next = baseUrl + '?page=' + (page + 1) } if (page > 1) { - prev = url + '?page=' + (page - 1) + prev = baseUrl + '?page=' + (page - 1) } return { - id: url + '?page=' + page, + id: baseUrl + '?page=' + page, type: 'OrderedCollectionPage', prev, next, - partOf: url, + partOf: baseUrl, orderedItems: result.data, totalItems: result.total } } -function buildSignedActivity (byActor: ActorModel, data: Object) { - const activity = activityPubContextify(data) +function buildSignedActivity (byActor: MActor, data: T, contextType?: ContextType) { + const activity = activityPubContextify(data, contextType) + + return signJsonLDObject(byActor, activity) +} + +function getAPId (object: string | { id: string }) { + if (typeof object === 'string') return object - return signObject(byActor, activity) as Promise + return object.id } -function getActorUrl (activityActor: string | ActivityPubActor) { - if (typeof activityActor === 'string') return activityActor +function checkUrlsSameHost (url1: string, url2: string) { + const idHost = new URL(url1).host + const actorHost = new URL(url2).host + + return idHost && actorHost && idHost.toLowerCase() === actorHost.toLowerCase() +} + +function buildRemoteVideoBaseUrl (video: MVideoWithHost, path: string, scheme?: string) { + if (!scheme) scheme = REMOTE_SCHEME.HTTP + + const host = video.VideoChannel.Actor.Server.host - return activityActor.id + return scheme + '://' + host + path } // --------------------------------------------------------------------------- export { - getActorUrl, + checkUrlsSameHost, + getAPId, activityPubContextify, activityPubCollectionPagination, - buildSignedActivity + buildSignedActivity, + buildRemoteVideoBaseUrl }