]> git.immae.eu Git - github/Chocobozzz/PeerTube.git/blame - server/controllers/api/users/me.ts
Avoid layout shift in account setup modal
[github/Chocobozzz/PeerTube.git] / server / controllers / api / users / me.ts
CommitLineData
d03cd8bb 1import 'multer'
67ed6552 2import * as express from 'express'
2dbc170d 3import { auditLoggerFactory, getAuditIdFromRes, UserAuditView } from '@server/helpers/audit-logger'
a4d2ca07 4import { Hooks } from '@server/lib/plugins/hooks'
4c7e60bc 5import { AttributesOnly } from '@shared/core-utils'
2cb03dc1 6import { ActorImageType, UserUpdateMe, UserVideoRate as FormattedUserVideoRate } from '../../../../shared'
c0e8b12e 7import { HttpStatusCode } from '../../../../shared/models/http/http-error-codes'
67ed6552
C
8import { UserVideoQuota } from '../../../../shared/models/users/user-video-quota.model'
9import { createReqFiles } from '../../../helpers/express-utils'
d03cd8bb 10import { getFormattedObjects } from '../../../helpers/utils'
67ed6552 11import { CONFIG } from '../../../initializers/config'
74dc3bca 12import { MIMETYPES } from '../../../initializers/constants'
67ed6552 13import { sequelizeTypescript } from '../../../initializers/database'
d03cd8bb 14import { sendUpdateActor } from '../../../lib/activitypub/send'
136d7efd 15import { deleteLocalActorImageFile, updateLocalActorImageFile } from '../../../lib/local-actor'
fb719404 16import { getOriginalVideoFileTotalDailyFromUser, getOriginalVideoFileTotalFromUser, sendVerifyUserEmail } from '../../../lib/user'
d03cd8bb 17import {
993cef4b
C
18 asyncMiddleware,
19 asyncRetryTransactionMiddleware,
d03cd8bb
C
20 authenticate,
21 paginationValidator,
22 setDefaultPagination,
23 setDefaultSort,
8054669f 24 setDefaultVideosSort,
d03cd8bb
C
25 usersUpdateMeValidator,
26 usersVideoRatingValidator
27} from '../../../middlewares'
cf405589 28import { deleteMeValidator, videoImportsSortValidator, videosSortValidator } from '../../../middlewares/validators'
213e30ef 29import { updateAvatarValidator } from '../../../middlewares/validators/actor-image'
67ed6552 30import { AccountModel } from '../../../models/account/account'
d03cd8bb 31import { AccountVideoRateModel } from '../../../models/account/account-video-rate'
7d9ba5c0 32import { UserModel } from '../../../models/user/user'
d03cd8bb 33import { VideoModel } from '../../../models/video/video'
d03cd8bb 34import { VideoImportModel } from '../../../models/video/video-import'
2dbc170d
C
35
36const auditLogger = auditLoggerFactory('users')
d03cd8bb 37
14e2014a 38const reqAvatarFile = createReqFiles([ 'avatarfile' ], MIMETYPES.IMAGE.MIMETYPE_EXT, { avatarfile: CONFIG.STORAGE.TMP_DIR })
d03cd8bb
C
39
40const meRouter = express.Router()
41
42meRouter.get('/me',
43 authenticate,
44 asyncMiddleware(getUserInformation)
45)
46meRouter.delete('/me',
47 authenticate,
a1587156 48 deleteMeValidator,
d03cd8bb
C
49 asyncMiddleware(deleteMe)
50)
51
52meRouter.get('/me/video-quota-used',
53 authenticate,
54 asyncMiddleware(getUserVideoQuotaUsed)
55)
56
57meRouter.get('/me/videos/imports',
58 authenticate,
59 paginationValidator,
60 videoImportsSortValidator,
61 setDefaultSort,
62 setDefaultPagination,
63 asyncMiddleware(getUserVideoImports)
64)
65
66meRouter.get('/me/videos',
67 authenticate,
68 paginationValidator,
69 videosSortValidator,
8054669f 70 setDefaultVideosSort,
d03cd8bb
C
71 setDefaultPagination,
72 asyncMiddleware(getUserVideos)
73)
74
75meRouter.get('/me/videos/:videoId/rating',
76 authenticate,
77 asyncMiddleware(usersVideoRatingValidator),
78 asyncMiddleware(getUserVideoRating)
79)
80
81meRouter.put('/me',
82 authenticate,
a890d1e0 83 asyncMiddleware(usersUpdateMeValidator),
176e2114 84 asyncRetryTransactionMiddleware(updateMe)
d03cd8bb
C
85)
86
87meRouter.post('/me/avatar/pick',
88 authenticate,
89 reqAvatarFile,
90 updateAvatarValidator,
176e2114 91 asyncRetryTransactionMiddleware(updateMyAvatar)
d03cd8bb
C
92)
93
1ea7da81
RK
94meRouter.delete('/me/avatar',
95 authenticate,
96 asyncRetryTransactionMiddleware(deleteMyAvatar)
97)
98
d03cd8bb
C
99// ---------------------------------------------------------------------------
100
101export {
102 meRouter
103}
104
105// ---------------------------------------------------------------------------
106
dae86118
C
107async function getUserVideos (req: express.Request, res: express.Response) {
108 const user = res.locals.oauth.token.User
a4d2ca07
C
109
110 const apiOptions = await Hooks.wrapObject({
111 accountId: user.Account.id,
112 start: req.query.start,
113 count: req.query.count,
114 sort: req.query.sort,
1fd61899
C
115 search: req.query.search,
116 isLive: req.query.isLive
a4d2ca07
C
117 }, 'filter:api.user.me.videos.list.params')
118
119 const resultList = await Hooks.wrapPromiseFun(
120 VideoModel.listUserVideosForApi,
121 apiOptions,
122 'filter:api.user.me.videos.list.result'
d03cd8bb
C
123 )
124
125 const additionalAttributes = {
126 waitTranscoding: true,
127 state: true,
128 scheduledUpdate: true,
129 blacklistInfo: true
130 }
131 return res.json(getFormattedObjects(resultList.data, resultList.total, { additionalAttributes }))
132}
133
dae86118
C
134async function getUserVideoImports (req: express.Request, res: express.Response) {
135 const user = res.locals.oauth.token.User
d03cd8bb
C
136 const resultList = await VideoImportModel.listUserVideoImportsForApi(
137 user.id,
138 req.query.start as number,
139 req.query.count as number,
140 req.query.sort
141 )
142
143 return res.json(getFormattedObjects(resultList.data, resultList.total))
144}
145
dae86118 146async function getUserInformation (req: express.Request, res: express.Response) {
d03cd8bb 147 // We did not load channels in res.locals.user
1acb9475 148 const user = await UserModel.loadForMeAPI(res.locals.oauth.token.user.id)
d03cd8bb 149
ac0868bc 150 return res.json(user.toMeFormattedJSON())
d03cd8bb
C
151}
152
dae86118 153async function getUserVideoQuotaUsed (req: express.Request, res: express.Response) {
ac0868bc 154 const user = res.locals.oauth.token.user
fb719404
C
155 const videoQuotaUsed = await getOriginalVideoFileTotalFromUser(user)
156 const videoQuotaUsedDaily = await getOriginalVideoFileTotalDailyFromUser(user)
d03cd8bb
C
157
158 const data: UserVideoQuota = {
bee0abff
FA
159 videoQuotaUsed,
160 videoQuotaUsedDaily
d03cd8bb
C
161 }
162 return res.json(data)
163}
164
dae86118 165async function getUserVideoRating (req: express.Request, res: express.Response) {
453e83ea 166 const videoId = res.locals.videoId.id
d03cd8bb
C
167 const accountId = +res.locals.oauth.token.User.Account.id
168
169 const ratingObj = await AccountVideoRateModel.load(accountId, videoId, null)
170 const rating = ratingObj ? ratingObj.type : 'none'
171
172 const json: FormattedUserVideoRate = {
173 videoId,
174 rating
175 }
06a05d5f 176 return res.json(json)
d03cd8bb
C
177}
178
179async function deleteMe (req: express.Request, res: express.Response) {
2dbc170d
C
180 const user = await UserModel.loadByIdWithChannels(res.locals.oauth.token.User.id)
181
182 auditLogger.delete(getAuditIdFromRes(res), new UserAuditView(user.toFormattedJSON()))
d03cd8bb
C
183
184 await user.destroy()
185
76148b27 186 return res.status(HttpStatusCode.NO_CONTENT_204).end()
d03cd8bb
C
187}
188
b426edd4 189async function updateMe (req: express.Request, res: express.Response) {
d03cd8bb 190 const body: UserUpdateMe = req.body
d1ab89de 191 let sendVerificationEmail = false
d03cd8bb 192
dae86118 193 const user = res.locals.oauth.token.user
d03cd8bb 194
c158a5fa
C
195 const keysToUpdate: (keyof UserUpdateMe & keyof AttributesOnly<UserModel>)[] = [
196 'password',
197 'nsfwPolicy',
198 'webTorrentEnabled',
199 'autoPlayVideo',
200 'autoPlayNextVideo',
201 'autoPlayNextVideoPlaylist',
202 'videosHistoryEnabled',
203 'videoLanguages',
204 'theme',
205 'noInstanceConfigWarningModal',
206 'noWelcomeModal'
207 ]
208
209 for (const key of keysToUpdate) {
210 if (body[key] !== undefined) user.set(key, body[key])
211 }
d03cd8bb 212
d1ab89de
C
213 if (body.email !== undefined) {
214 if (CONFIG.SIGNUP.REQUIRES_EMAIL_VERIFICATION) {
215 user.pendingEmail = body.email
216 sendVerificationEmail = true
217 } else {
218 user.email = body.email
219 }
220 }
221
589d9f55
C
222 await sequelizeTypescript.transaction(async t => {
223 await user.save({ transaction: t })
91411dba 224
c158a5fa 225 if (body.displayName === undefined && body.description === undefined) return
d03cd8bb 226
c158a5fa 227 const userAccount = await AccountModel.load(user.Account.id, t)
d03cd8bb 228
c158a5fa
C
229 if (body.displayName !== undefined) userAccount.name = body.displayName
230 if (body.description !== undefined) userAccount.description = body.description
231 await userAccount.save({ transaction: t })
232
233 await sendUpdateActor(userAccount, t)
589d9f55 234 })
d03cd8bb 235
d1ab89de
C
236 if (sendVerificationEmail === true) {
237 await sendVerifyUserEmail(user, true)
238 }
239
76148b27 240 return res.status(HttpStatusCode.NO_CONTENT_204).end()
d03cd8bb
C
241}
242
6040f87d 243async function updateMyAvatar (req: express.Request, res: express.Response) {
a1587156 244 const avatarPhysicalFile = req.files['avatarfile'][0]
dae86118 245 const user = res.locals.oauth.token.user
d03cd8bb 246
91411dba 247 const userAccount = await AccountModel.load(user.Account.id)
d03cd8bb 248
2cb03dc1 249 const avatar = await updateLocalActorImageFile(userAccount, avatarPhysicalFile, ActorImageType.AVATAR)
91411dba 250
06a05d5f 251 return res.json({ avatar: avatar.toFormattedJSON() })
d03cd8bb 252}
1ea7da81
RK
253
254async function deleteMyAvatar (req: express.Request, res: express.Response) {
255 const user = res.locals.oauth.token.user
256
257 const userAccount = await AccountModel.load(user.Account.id)
2cb03dc1 258 await deleteLocalActorImageFile(userAccount, ActorImageType.AVATAR)
1ea7da81 259
76148b27 260 return res.status(HttpStatusCode.NO_CONTENT_204).end()
1ea7da81 261}