]> git.immae.eu Git - github/Chocobozzz/PeerTube.git/blame_incremental - server/controllers/api/users/my-subscriptions.ts
Use bullmq job dependency
[github/Chocobozzz/PeerTube.git] / server / controllers / api / users / my-subscriptions.ts
... / ...
CommitLineData
1import 'multer'
2import express from 'express'
3import { handlesToNameAndHost } from '@server/helpers/actors'
4import { pickCommonVideoQuery } from '@server/helpers/query'
5import { sendUndoFollow } from '@server/lib/activitypub/send'
6import { guessAdditionalAttributesFromQuery } from '@server/models/video/formatter/video-format-utils'
7import { VideoChannelModel } from '@server/models/video/video-channel'
8import { HttpStatusCode } from '../../../../shared/models/http/http-error-codes'
9import { buildNSFWFilter, getCountVideos } from '../../../helpers/express-utils'
10import { getFormattedObjects } from '../../../helpers/utils'
11import { sequelizeTypescript } from '../../../initializers/database'
12import { JobQueue } from '../../../lib/job-queue'
13import {
14 asyncMiddleware,
15 asyncRetryTransactionMiddleware,
16 authenticate,
17 commonVideosFiltersValidator,
18 paginationValidator,
19 setDefaultPagination,
20 setDefaultSort,
21 setDefaultVideosSort,
22 userSubscriptionAddValidator,
23 userSubscriptionGetValidator
24} from '../../../middlewares'
25import {
26 areSubscriptionsExistValidator,
27 userSubscriptionListValidator,
28 userSubscriptionsSortValidator,
29 videosSortValidator
30} from '../../../middlewares/validators'
31import { ActorFollowModel } from '../../../models/actor/actor-follow'
32import { VideoModel } from '../../../models/video/video'
33
34const mySubscriptionsRouter = express.Router()
35
36mySubscriptionsRouter.get('/me/subscriptions/videos',
37 authenticate,
38 paginationValidator,
39 videosSortValidator,
40 setDefaultVideosSort,
41 setDefaultPagination,
42 commonVideosFiltersValidator,
43 asyncMiddleware(getUserSubscriptionVideos)
44)
45
46mySubscriptionsRouter.get('/me/subscriptions/exist',
47 authenticate,
48 areSubscriptionsExistValidator,
49 asyncMiddleware(areSubscriptionsExist)
50)
51
52mySubscriptionsRouter.get('/me/subscriptions',
53 authenticate,
54 paginationValidator,
55 userSubscriptionsSortValidator,
56 setDefaultSort,
57 setDefaultPagination,
58 userSubscriptionListValidator,
59 asyncMiddleware(getUserSubscriptions)
60)
61
62mySubscriptionsRouter.post('/me/subscriptions',
63 authenticate,
64 userSubscriptionAddValidator,
65 addUserSubscription
66)
67
68mySubscriptionsRouter.get('/me/subscriptions/:uri',
69 authenticate,
70 userSubscriptionGetValidator,
71 asyncMiddleware(getUserSubscription)
72)
73
74mySubscriptionsRouter.delete('/me/subscriptions/:uri',
75 authenticate,
76 userSubscriptionGetValidator,
77 asyncRetryTransactionMiddleware(deleteUserSubscription)
78)
79
80// ---------------------------------------------------------------------------
81
82export {
83 mySubscriptionsRouter
84}
85
86// ---------------------------------------------------------------------------
87
88async function areSubscriptionsExist (req: express.Request, res: express.Response) {
89 const uris = req.query.uris as string[]
90 const user = res.locals.oauth.token.User
91
92 const sanitizedHandles = handlesToNameAndHost(uris)
93
94 const results = await ActorFollowModel.listSubscriptionsOf(user.Account.Actor.id, sanitizedHandles)
95
96 const existObject: { [id: string ]: boolean } = {}
97 for (const sanitizedHandle of sanitizedHandles) {
98 const obj = results.find(r => {
99 const server = r.ActorFollowing.Server
100
101 return r.ActorFollowing.preferredUsername === sanitizedHandle.name &&
102 (
103 (!server && !sanitizedHandle.host) ||
104 (server.host === sanitizedHandle.host)
105 )
106 })
107
108 existObject[sanitizedHandle.handle] = obj !== undefined
109 }
110
111 return res.json(existObject)
112}
113
114function addUserSubscription (req: express.Request, res: express.Response) {
115 const user = res.locals.oauth.token.User
116 const [ name, host ] = req.body.uri.split('@')
117
118 const payload = {
119 name,
120 host,
121 assertIsChannel: true,
122 followerActorId: user.Account.Actor.id
123 }
124
125 JobQueue.Instance.createJobAsync({ type: 'activitypub-follow', payload })
126
127 return res.status(HttpStatusCode.NO_CONTENT_204).end()
128}
129
130async function getUserSubscription (req: express.Request, res: express.Response) {
131 const subscription = res.locals.subscription
132 const videoChannel = await VideoChannelModel.loadAndPopulateAccount(subscription.ActorFollowing.VideoChannel.id)
133
134 return res.json(videoChannel.toFormattedJSON())
135}
136
137async function deleteUserSubscription (req: express.Request, res: express.Response) {
138 const subscription = res.locals.subscription
139
140 await sequelizeTypescript.transaction(async t => {
141 if (subscription.state === 'accepted') await sendUndoFollow(subscription, t)
142
143 return subscription.destroy({ transaction: t })
144 })
145
146 return res.type('json')
147 .status(HttpStatusCode.NO_CONTENT_204)
148 .end()
149}
150
151async function getUserSubscriptions (req: express.Request, res: express.Response) {
152 const user = res.locals.oauth.token.User
153 const actorId = user.Account.Actor.id
154
155 const resultList = await ActorFollowModel.listSubscriptionsForApi({
156 actorId,
157 start: req.query.start,
158 count: req.query.count,
159 sort: req.query.sort,
160 search: req.query.search
161 })
162
163 return res.json(getFormattedObjects(resultList.data, resultList.total))
164}
165
166async function getUserSubscriptionVideos (req: express.Request, res: express.Response) {
167 const user = res.locals.oauth.token.User
168 const countVideos = getCountVideos(req)
169 const query = pickCommonVideoQuery(req.query)
170
171 const resultList = await VideoModel.listForApi({
172 ...query,
173
174 displayOnlyForFollower: {
175 actorId: user.Account.Actor.id,
176 orLocalVideos: false
177 },
178 nsfw: buildNSFWFilter(res, query.nsfw),
179 user,
180 countVideos
181 })
182
183 return res.json(getFormattedObjects(resultList.data, resultList.total, guessAdditionalAttributesFromQuery(query)))
184}