]> git.immae.eu Git - github/Chocobozzz/PeerTube.git/blame_incremental - server/lib/activitypub/send/utils.ts
Cleanup server fixme
[github/Chocobozzz/PeerTube.git] / server / lib / activitypub / send / utils.ts
... / ...
CommitLineData
1import { Transaction } from 'sequelize'
2import { Activity, ActivityAudience } from '../../../../shared/models/activitypub'
3import { logger } from '../../../helpers/logger'
4import { ActorModel } from '../../../models/activitypub/actor'
5import { ActorFollowModel } from '../../../models/activitypub/actor-follow'
6import { JobQueue } from '../../job-queue'
7import { getActorsInvolvedInVideo, getAudienceFromFollowersOf, getRemoteVideoAudience } from '../audience'
8import { getServerActor } from '../../../helpers/utils'
9import { afterCommitIfTransaction } from '../../../helpers/database-utils'
10import { MActorWithInboxes, MActor, MActorId, MActorLight, MVideo, MVideoAccountLight } from '../../../typings/models'
11
12async function sendVideoRelatedActivity (activityBuilder: (audience: ActivityAudience) => Activity, options: {
13 byActor: MActorLight,
14 video: MVideoAccountLight,
15 transaction?: Transaction
16}) {
17 const { byActor, video, transaction } = options
18
19 const actorsInvolvedInVideo = await getActorsInvolvedInVideo(video, transaction)
20
21 // Send to origin
22 if (video.isOwned() === false) {
23 const audience = getRemoteVideoAudience(video, actorsInvolvedInVideo)
24 const activity = activityBuilder(audience)
25
26 return afterCommitIfTransaction(transaction, () => {
27 return unicastTo(activity, byActor, video.VideoChannel.Account.Actor.getSharedInbox())
28 })
29 }
30
31 // Send to followers
32 const audience = getAudienceFromFollowersOf(actorsInvolvedInVideo)
33 const activity = activityBuilder(audience)
34
35 const actorsException = [ byActor ]
36
37 return broadcastToFollowers(activity, byActor, actorsInvolvedInVideo, transaction, actorsException)
38}
39
40async function forwardVideoRelatedActivity (
41 activity: Activity,
42 t: Transaction,
43 followersException: MActorWithInboxes[] = [],
44 video: MVideo
45) {
46 // Mastodon does not add our announces in audience, so we forward to them manually
47 const additionalActors = await getActorsInvolvedInVideo(video, t)
48 const additionalFollowerUrls = additionalActors.map(a => a.followersUrl)
49
50 return forwardActivity(activity, t, followersException, additionalFollowerUrls)
51}
52
53async function forwardActivity (
54 activity: Activity,
55 t: Transaction,
56 followersException: MActorWithInboxes[] = [],
57 additionalFollowerUrls: string[] = []
58) {
59 logger.info('Forwarding activity %s.', activity.id)
60
61 const to = activity.to || []
62 const cc = activity.cc || []
63
64 const followersUrls = additionalFollowerUrls
65 for (const dest of to.concat(cc)) {
66 if (dest.endsWith('/followers')) {
67 followersUrls.push(dest)
68 }
69 }
70
71 const toActorFollowers = await ActorModel.listByFollowersUrls(followersUrls, t)
72 const uris = await computeFollowerUris(toActorFollowers, followersException, t)
73
74 if (uris.length === 0) {
75 logger.info('0 followers for %s, no forwarding.', toActorFollowers.map(a => a.id).join(', '))
76 return undefined
77 }
78
79 logger.debug('Creating forwarding job.', { uris })
80
81 const payload = {
82 uris,
83 body: activity
84 }
85 return afterCommitIfTransaction(t, () => JobQueue.Instance.createJob({ type: 'activitypub-http-broadcast', payload }))
86}
87
88async function broadcastToFollowers (
89 data: any,
90 byActor: MActorId,
91 toFollowersOf: MActorId[],
92 t: Transaction,
93 actorsException: MActorWithInboxes[] = []
94) {
95 const uris = await computeFollowerUris(toFollowersOf, actorsException, t)
96
97 return afterCommitIfTransaction(t, () => broadcastTo(uris, data, byActor))
98}
99
100async function broadcastToActors (
101 data: any,
102 byActor: MActorId,
103 toActors: MActor[],
104 t?: Transaction,
105 actorsException: MActorWithInboxes[] = []
106) {
107 const uris = await computeUris(toActors, actorsException)
108 return afterCommitIfTransaction(t, () => broadcastTo(uris, data, byActor))
109}
110
111function broadcastTo (uris: string[], data: any, byActor: MActorId) {
112 if (uris.length === 0) return undefined
113
114 logger.debug('Creating broadcast job.', { uris })
115
116 const payload = {
117 uris,
118 signatureActorId: byActor.id,
119 body: data
120 }
121
122 return JobQueue.Instance.createJob({ type: 'activitypub-http-broadcast', payload })
123}
124
125function unicastTo (data: any, byActor: MActorId, toActorUrl: string) {
126 logger.debug('Creating unicast job.', { uri: toActorUrl })
127
128 const payload = {
129 uri: toActorUrl,
130 signatureActorId: byActor.id,
131 body: data
132 }
133
134 JobQueue.Instance.createJob({ type: 'activitypub-http-unicast', payload })
135}
136
137// ---------------------------------------------------------------------------
138
139export {
140 broadcastToFollowers,
141 unicastTo,
142 forwardActivity,
143 broadcastToActors,
144 forwardVideoRelatedActivity,
145 sendVideoRelatedActivity
146}
147
148// ---------------------------------------------------------------------------
149
150async function computeFollowerUris (toFollowersOf: MActorId[], actorsException: MActorWithInboxes[], t: Transaction) {
151 const toActorFollowerIds = toFollowersOf.map(a => a.id)
152
153 const result = await ActorFollowModel.listAcceptedFollowerSharedInboxUrls(toActorFollowerIds, t)
154 const sharedInboxesException = await buildSharedInboxesException(actorsException)
155
156 return result.data.filter(sharedInbox => sharedInboxesException.indexOf(sharedInbox) === -1)
157}
158
159async function computeUris (toActors: MActor[], actorsException: MActorWithInboxes[] = []) {
160 const serverActor = await getServerActor()
161 const targetUrls = toActors
162 .filter(a => a.id !== serverActor.id) // Don't send to ourselves
163 .map(a => a.getSharedInbox())
164
165 const toActorSharedInboxesSet = new Set(targetUrls)
166
167 const sharedInboxesException = await buildSharedInboxesException(actorsException)
168 return Array.from(toActorSharedInboxesSet)
169 .filter(sharedInbox => sharedInboxesException.indexOf(sharedInbox) === -1)
170}
171
172async function buildSharedInboxesException (actorsException: MActorWithInboxes[]) {
173 const serverActor = await getServerActor()
174
175 return actorsException
176 .map(f => f.getSharedInbox())
177 .concat([ serverActor.sharedInboxUrl ])
178}