X-Git-Url: https://git.immae.eu/?a=blobdiff_plain;f=server%2Fhelpers%2Factivitypub.ts;h=e850efe138eb53a4097e4a1a4cd742f6a1f60c57;hb=7519127b5cb44095f78f6bf4c51d4ebf2b7d5e88;hp=ecb509b66c81dd9edea35d7466717493ff77b422;hpb=e4f97babf701481b55cc10fb3448feab5f97c867;p=github%2FChocobozzz%2FPeerTube.git diff --git a/server/helpers/activitypub.ts b/server/helpers/activitypub.ts index ecb509b66..e850efe13 100644 --- a/server/helpers/activitypub.ts +++ b/server/helpers/activitypub.ts @@ -1,123 +1,133 @@ -import * as url from 'url' - -import { database as db } from '../initializers' -import { logger } from './logger' -import { doRequest } from './requests' -import { isRemoteAccountValid } from './custom-validators' -import { ActivityPubActor } from '../../shared/models/activitypub/activitypub-actor' -import { ResultList } from '../../shared/models/result-list.model' - -async function fetchRemoteAccountAndCreatePod (accountUrl: string) { - const options = { - uri: accountUrl, - method: 'GET' - } - - let requestResult - try { - requestResult = await doRequest(options) - } catch (err) { - logger.warning('Cannot fetch remote account %s.', accountUrl, err) - return undefined - } - - const accountJSON: ActivityPubActor = requestResult.body - if (isRemoteAccountValid(accountJSON) === false) return undefined - - const followersCount = await fetchAccountCount(accountJSON.followers) - const followingCount = await fetchAccountCount(accountJSON.following) - - const account = db.Account.build({ - uuid: accountJSON.uuid, - name: accountJSON.preferredUsername, - url: accountJSON.url, - publicKey: accountJSON.publicKey.publicKeyPem, - privateKey: null, - followersCount: followersCount, - followingCount: followingCount, - inboxUrl: accountJSON.inbox, - outboxUrl: accountJSON.outbox, - sharedInboxUrl: accountJSON.endpoints.sharedInbox, - followersUrl: accountJSON.followers, - followingUrl: accountJSON.following - }) - - const accountHost = url.parse(account.url).host - const podOptions = { - where: { - host: accountHost - }, - defaults: { - host: accountHost - } - } - const pod = await db.Pod.findOrCreate(podOptions) - - return { account, pod } -} - -function activityPubContextify (data: object) { - return Object.assign(data,{ +import * as Bluebird from 'bluebird' +import * as validator from 'validator' +import { ResultList } from '../../shared/models' +import { Activity } from '../../shared/models/activitypub' +import { ACTIVITY_PUB } from '../initializers' +import { ActorModel } from '../models/activitypub/actor' +import { signJsonLDObject } from './peertube-crypto' +import { pageToStartAndCount } from './core-utils' +import { parse } from 'url' + +function activityPubContextify (data: T) { + return Object.assign(data, { '@context': [ 'https://www.w3.org/ns/activitystreams', 'https://w3id.org/security/v1', { - 'Hashtag': 'as:Hashtag', - 'uuid': 'http://schema.org/identifier', - 'category': 'http://schema.org/category', - 'licence': 'http://schema.org/license', - 'nsfw': 'as:sensitive', - 'language': 'http://schema.org/inLanguage', - 'views': 'http://schema.org/Number', - 'size': 'http://schema.org/Number' + RsaSignature2017: 'https://w3id.org/security#RsaSignature2017', + pt: 'https://joinpeertube.org/ns#', + sc: 'http://schema.org#', + Hashtag: 'as:Hashtag', + uuid: 'sc:identifier', + category: 'sc:category', + licence: 'sc:license', + subtitleLanguage: 'sc:subtitleLanguage', + sensitive: 'as:sensitive', + language: 'sc:inLanguage', + views: 'sc:Number', + state: 'sc:Number', + size: 'sc:Number', + fps: 'sc:Number', + commentsEnabled: 'sc:Boolean', + downloadEnabled: 'sc:Boolean', + waitTranscoding: 'sc:Boolean', + expires: 'sc:expires', + support: 'sc:Text', + CacheFile: 'pt:CacheFile', + Infohash: 'pt:Infohash', + originallyPublishedAt: 'sc:DateTime' + }, + { + likes: { + '@id': 'as:likes', + '@type': '@id' + }, + dislikes: { + '@id': 'as:dislikes', + '@type': '@id' + }, + shares: { + '@id': 'as:shares', + '@type': '@id' + }, + comments: { + '@id': 'as:comments', + '@type': '@id' + } } ] }) } -function activityPubCollectionPagination (url: string, page: number, result: ResultList) { - const baseUrl = url.split('?').shift +type ActivityPubCollectionPaginationHandler = (start: number, count: number) => Bluebird> | Promise> +async function activityPubCollectionPagination (baseUrl: string, handler: ActivityPubCollectionPaginationHandler, page?: any) { + if (!page || !validator.isInt(page)) { + // We just display the first page URL, we only need the total items + const result = await handler(0, 1) - const obj = { - id: baseUrl, - type: 'Collection', - totalItems: result.total, - first: { - id: baseUrl + '?page=' + page, - type: 'CollectionPage', + return { + id: baseUrl, + type: 'OrderedCollection', totalItems: result.total, - next: baseUrl + '?page=' + (page + 1), - partOf: baseUrl, - items: result.data + first: baseUrl + '?page=1' } } - return activityPubContextify(obj) -} + const { start, count } = pageToStartAndCount(page, ACTIVITY_PUB.COLLECTION_ITEMS_PER_PAGE) + const result = await handler(start, count) -// --------------------------------------------------------------------------- + let next: string | undefined + let prev: string | undefined -export { - fetchRemoteAccountAndCreatePod, - activityPubContextify, - activityPubCollectionPagination -} + // Assert page is a number + page = parseInt(page, 10) -// --------------------------------------------------------------------------- + // There are more results + if (result.total > page * ACTIVITY_PUB.COLLECTION_ITEMS_PER_PAGE) { + next = baseUrl + '?page=' + (page + 1) + } -async function fetchAccountCount (url: string) { - const options = { - uri: url, - method: 'GET' + if (page > 1) { + prev = baseUrl + '?page=' + (page - 1) } - let requestResult - try { - requestResult = await doRequest(options) - } catch (err) { - logger.warning('Cannot fetch remote account count %s.', url, err) - return undefined + return { + id: baseUrl + '?page=' + page, + type: 'OrderedCollectionPage', + prev, + next, + partOf: baseUrl, + orderedItems: result.data, + totalItems: result.total } - return requestResult.totalItems ? requestResult.totalItems : 0 +} + +function buildSignedActivity (byActor: ActorModel, data: Object) { + const activity = activityPubContextify(data) + + return signJsonLDObject(byActor, activity) as Promise +} + +function getAPId (activity: string | { id: string }) { + if (typeof activity === 'string') return activity + + return activity.id +} + +function checkUrlsSameHost (url1: string, url2: string) { + const idHost = parse(url1).host + const actorHost = parse(url2).host + + return idHost && actorHost && idHost.toLowerCase() === actorHost.toLowerCase() +} + +// --------------------------------------------------------------------------- + +export { + checkUrlsSameHost, + getAPId, + activityPubContextify, + activityPubCollectionPagination, + buildSignedActivity }