]> git.immae.eu Git - github/Chocobozzz/PeerTube.git/blame - server/lib/activitypub/process/process-update.ts
Fix lint
[github/Chocobozzz/PeerTube.git] / server / lib / activitypub / process / process-update.ts
CommitLineData
25ed141c 1import * as Bluebird from 'bluebird'
3fd3ab2d 2import { ActivityUpdate } from '../../../../shared/models/activitypub'
265ba139
C
3import { ActivityPubActor } from '../../../../shared/models/activitypub/activitypub-actor'
4import { VideoTorrentObject } from '../../../../shared/models/activitypub/objects'
da854ddd
C
5import { retryTransactionWrapper } from '../../../helpers/database-utils'
6import { logger } from '../../../helpers/logger'
7import { resetSequelizeInstance } from '../../../helpers/utils'
3fd3ab2d 8import { sequelizeTypescript } from '../../../initializers'
265ba139 9import { AccountModel } from '../../../models/account/account'
50d6de9c 10import { ActorModel } from '../../../models/activitypub/actor'
3fd3ab2d 11import { TagModel } from '../../../models/video/tag'
3fd3ab2d 12import { VideoFileModel } from '../../../models/video/video-file'
a5625b41 13import { fetchAvatarIfExists, getOrCreateActorAndServerAndModel, updateActorAvatarInstance, updateActorInstance } from '../actor'
2ccaeeb3 14import { getOrCreateAccountAndVideoAndChannel, videoActivityObjectToDBAttributes, videoFileActivityUrlToDBAttributes } from '../videos'
0d0e8dd0
C
15
16async function processUpdateActivity (activity: ActivityUpdate) {
50d6de9c 17 const actor = await getOrCreateActorAndServerAndModel(activity.actor)
e4f97bab 18
e4f97bab 19 if (activity.object.type === 'Video') {
50d6de9c 20 return processUpdateVideo(actor, activity)
265ba139
C
21 } else if (activity.object.type === 'Person') {
22 return processUpdateAccount(actor, activity)
e4f97bab 23 }
0d0e8dd0 24
b1cbc0dd 25 return
e4f97bab
C
26}
27
28// ---------------------------------------------------------------------------
29
30export {
31 processUpdateActivity
32}
33
34// ---------------------------------------------------------------------------
35
50d6de9c 36function processUpdateVideo (actor: ActorModel, activity: ActivityUpdate) {
0d0e8dd0 37 const options = {
50d6de9c 38 arguments: [ actor, activity ],
0d0e8dd0
C
39 errorMessage: 'Cannot update the remote video with many retries'
40 }
e4f97bab 41
0d0e8dd0 42 return retryTransactionWrapper(updateRemoteVideo, options)
e4f97bab
C
43}
44
50d6de9c 45async function updateRemoteVideo (actor: ActorModel, activity: ActivityUpdate) {
265ba139 46 const videoAttributesToUpdate = activity.object as VideoTorrentObject
50d6de9c 47
2ccaeeb3
C
48 const res = await getOrCreateAccountAndVideoAndChannel(videoAttributesToUpdate.id)
49
0d0e8dd0 50 logger.debug('Updating remote video "%s".', videoAttributesToUpdate.uuid)
2ccaeeb3 51 let videoInstance = res.video
265ba139 52 let videoFieldsSave: any
0d0e8dd0
C
53
54 try {
3fd3ab2d 55 await sequelizeTypescript.transaction(async t => {
0d0e8dd0
C
56 const sequelizeOptions = {
57 transaction: t
58 }
59
265ba139
C
60 videoFieldsSave = videoInstance.toJSON()
61
50d6de9c
C
62 const videoChannel = videoInstance.VideoChannel
63 if (videoChannel.Account.Actor.id !== actor.id) {
64 throw new Error('Account ' + actor.url + ' does not own video channel ' + videoChannel.Actor.url)
0d0e8dd0
C
65 }
66
50d6de9c 67 const videoData = await videoActivityObjectToDBAttributes(videoChannel, videoAttributesToUpdate, activity.to, activity.cc)
0d0e8dd0 68 videoInstance.set('name', videoData.name)
9a8cbd82
C
69 videoInstance.set('uuid', videoData.uuid)
70 videoInstance.set('url', videoData.url)
0d0e8dd0
C
71 videoInstance.set('category', videoData.category)
72 videoInstance.set('licence', videoData.licence)
73 videoInstance.set('language', videoData.language)
9a8cbd82 74 videoInstance.set('description', videoData.description)
0d0e8dd0 75 videoInstance.set('nsfw', videoData.nsfw)
4e8c8728 76 videoInstance.set('commentsEnabled', videoData.commentsEnabled)
0d0e8dd0
C
77 videoInstance.set('duration', videoData.duration)
78 videoInstance.set('createdAt', videoData.createdAt)
79 videoInstance.set('updatedAt', videoData.updatedAt)
80 videoInstance.set('views', videoData.views)
9a8cbd82 81 videoInstance.set('privacy', videoData.privacy)
0d0e8dd0
C
82
83 await videoInstance.save(sequelizeOptions)
84
85 // Remove old video files
86 const videoFileDestroyTasks: Bluebird<void>[] = []
87 for (const videoFile of videoInstance.VideoFiles) {
88 videoFileDestroyTasks.push(videoFile.destroy(sequelizeOptions))
89 }
90 await Promise.all(videoFileDestroyTasks)
91
79d5caf9 92 const videoFileAttributes = videoFileActivityUrlToDBAttributes(videoInstance, videoAttributesToUpdate)
1f7ab4f3 93 const tasks = videoFileAttributes.map(f => VideoFileModel.create(f))
0d0e8dd0
C
94 await Promise.all(tasks)
95
96 const tags = videoAttributesToUpdate.tag.map(t => t.name)
3fd3ab2d
C
97 const tagInstances = await TagModel.findOrCreateTags(tags, t)
98 await videoInstance.$set('Tags', tagInstances, sequelizeOptions)
0d0e8dd0
C
99 })
100
101 logger.info('Remote video with uuid %s updated', videoAttributesToUpdate.uuid)
102 } catch (err) {
103 if (videoInstance !== undefined && videoFieldsSave !== undefined) {
104 resetSequelizeInstance(videoInstance, videoFieldsSave)
105 }
106
107 // This is just a debug because we will retry the insert
108 logger.debug('Cannot update the remote video.', err)
109 throw err
110 }
111}
265ba139
C
112
113function processUpdateAccount (actor: ActorModel, activity: ActivityUpdate) {
114 const options = {
115 arguments: [ actor, activity ],
116 errorMessage: 'Cannot update the remote account with many retries'
117 }
118
119 return retryTransactionWrapper(updateRemoteAccount, options)
120}
121
122async function updateRemoteAccount (actor: ActorModel, activity: ActivityUpdate) {
123 const accountAttributesToUpdate = activity.object as ActivityPubActor
124
125 logger.debug('Updating remote account "%s".', accountAttributesToUpdate.uuid)
265ba139
C
126 let accountInstance: AccountModel
127 let actorFieldsSave: object
128 let accountFieldsSave: object
129
130 // Fetch icon?
131 const avatarName = await fetchAvatarIfExists(accountAttributesToUpdate)
132
133 try {
134 await sequelizeTypescript.transaction(async t => {
a5625b41
C
135 actorFieldsSave = actor.toJSON()
136 accountInstance = actor.Account
137 accountFieldsSave = actor.Account.toJSON()
265ba139 138
a5625b41 139 await updateActorInstance(actor, accountAttributesToUpdate)
265ba139 140
a5625b41
C
141 if (avatarName !== undefined) {
142 await updateActorAvatarInstance(actor, avatarName, t)
265ba139
C
143 }
144
145 await actor.save({ transaction: t })
146
147 actor.Account.set('name', accountAttributesToUpdate.name || accountAttributesToUpdate.preferredUsername)
148 await actor.Account.save({ transaction: t })
149 })
150
151 logger.info('Remote account with uuid %s updated', accountAttributesToUpdate.uuid)
152 } catch (err) {
a5625b41
C
153 if (actor !== undefined && actorFieldsSave !== undefined) {
154 resetSequelizeInstance(actor, actorFieldsSave)
265ba139
C
155 }
156
157 if (accountInstance !== undefined && accountFieldsSave !== undefined) {
158 resetSequelizeInstance(accountInstance, accountFieldsSave)
159 }
160
161 // This is just a debug because we will retry the insert
162 logger.debug('Cannot update the remote account.', err)
163 throw err
164 }
165}