]> git.immae.eu Git - github/Chocobozzz/PeerTube.git/blame - client/src/app/core/users/user.service.ts
Update ffmpeg min version
[github/Chocobozzz/PeerTube.git] / client / src / app / core / users / user.service.ts
CommitLineData
5c20a455
C
1import { BytesPipe } from 'ngx-pipes'
2import { SortMeta } from 'primeng/api'
3import { from, Observable, of } from 'rxjs'
67ed6552 4import { catchError, concatMap, filter, first, map, shareReplay, throttleTime, toArray } from 'rxjs/operators'
74d63469 5import { HttpClient, HttpParams } from '@angular/common/http'
63c4db6d 6import { Injectable } from '@angular/core'
5c20a455 7import { AuthService } from '@app/core/auth'
e724fa93 8import { I18n } from '@ngx-translate/i18n-polyfill'
a4ff3100 9import { UserLocalStorageKeys } from '@root-helpers/users'
67ed6552
C
10import {
11 Avatar,
12 NSFWPolicyType,
13 ResultList,
14 User as UserServerModel,
15 UserCreate,
16 UserRegister,
17 UserRole,
18 UserUpdate,
19 UserUpdateMe,
20 UserVideoQuota
21} from '@shared/models'
5c20a455 22import { environment } from '../../../environments/environment'
5c20a455 23import { RestExtractor, RestPagination, RestService } from '../rest'
67ed6552 24import { LocalStorageService, SessionStorageService } from '../wrappers/storage.service'
5c20a455 25import { User } from './user.model'
629d8d6f
C
26
27@Injectable()
e2a2d6c8 28export class UserService {
63c4db6d 29 static BASE_USERS_URL = environment.apiUrl + '/api/v1/users/'
629d8d6f 30
e724fa93
C
31 private bytesPipe = new BytesPipe()
32
d3217560 33 private userCache: { [ id: number ]: Observable<UserServerModel> } = {}
218b0874 34
df98563e 35 constructor (
d592e0a9 36 private authHttp: HttpClient,
5c20a455 37 private authService: AuthService,
e724fa93
C
38 private restExtractor: RestExtractor,
39 private restService: RestService,
d3217560
RK
40 private localStorageService: LocalStorageService,
41 private sessionStorageService: SessionStorageService,
e724fa93
C
42 private i18n: I18n
43 ) { }
629d8d6f 44
a890d1e0 45 changePassword (currentPassword: string, newPassword: string) {
8094a898
C
46 const url = UserService.BASE_USERS_URL + 'me'
47 const body: UserUpdateMe = {
a890d1e0 48 currentPassword,
629d8d6f 49 password: newPassword
df98563e 50 }
629d8d6f 51
de59c48f 52 return this.authHttp.put(url, body)
db400f44
C
53 .pipe(
54 map(this.restExtractor.extractDataBool),
e4f0e92e 55 catchError(err => this.restExtractor.handleError(err))
db400f44 56 )
629d8d6f 57 }
af5e743b 58
0ba5f5ba
C
59 changeEmail (password: string, newEmail: string) {
60 const url = UserService.BASE_USERS_URL + 'me'
61 const body: UserUpdateMe = {
62 currentPassword: password,
63 email: newEmail
64 }
65
66 return this.authHttp.put(url, body)
67 .pipe(
68 map(this.restExtractor.extractDataBool),
69 catchError(err => this.restExtractor.handleError(err))
70 )
71 }
72
ed56ad11 73 updateMyProfile (profile: UserUpdateMe) {
8094a898 74 const url = UserService.BASE_USERS_URL + 'me'
af5e743b 75
ed56ad11 76 return this.authHttp.put(url, profile)
db400f44
C
77 .pipe(
78 map(this.restExtractor.extractDataBool),
e4f0e92e 79 catchError(err => this.restExtractor.handleError(err))
db400f44 80 )
af5e743b 81 }
a184c71b 82
d3217560 83 updateMyAnonymousProfile (profile: UserUpdateMe) {
a4ff3100
C
84 try {
85 this.localStorageService.setItem(UserLocalStorageKeys.NSFW_POLICY, profile.nsfwPolicy)
86 this.localStorageService.setItem(UserLocalStorageKeys.WEBTORRENT_ENABLED, profile.webTorrentEnabled)
d3217560 87
a4ff3100
C
88 this.localStorageService.setItem(UserLocalStorageKeys.AUTO_PLAY_VIDEO, profile.autoPlayNextVideo)
89 this.localStorageService.setItem(UserLocalStorageKeys.AUTO_PLAY_VIDEO_PLAYLIST, profile.autoPlayNextVideoPlaylist)
d3217560 90
a4ff3100
C
91 this.localStorageService.setItem(UserLocalStorageKeys.THEME, profile.theme)
92 this.localStorageService.setItem(UserLocalStorageKeys.VIDEO_LANGUAGES, profile.videoLanguages)
93 } catch (err) {
94 console.error(`Cannot set item in localStorage. Likely due to a value impossible to stringify.`, err)
d3217560
RK
95 }
96 }
97
5c20a455
C
98 listenAnonymousUpdate () {
99 return this.localStorageService.watch([
a4ff3100
C
100 UserLocalStorageKeys.NSFW_POLICY,
101 UserLocalStorageKeys.WEBTORRENT_ENABLED,
102 UserLocalStorageKeys.AUTO_PLAY_VIDEO,
103 UserLocalStorageKeys.AUTO_PLAY_VIDEO_PLAYLIST,
104 UserLocalStorageKeys.THEME,
105 UserLocalStorageKeys.VIDEO_LANGUAGES
5c20a455
C
106 ]).pipe(
107 throttleTime(200),
108 filter(() => this.authService.isLoggedIn() !== true),
109 map(() => this.getAnonymousUser())
110 )
111 }
112
92b9d60c
C
113 deleteMe () {
114 const url = UserService.BASE_USERS_URL + 'me'
115
116 return this.authHttp.delete(url)
117 .pipe(
118 map(this.restExtractor.extractDataBool),
119 catchError(err => this.restExtractor.handleError(err))
120 )
121 }
122
c5911fd3
C
123 changeAvatar (avatarForm: FormData) {
124 const url = UserService.BASE_USERS_URL + 'me/avatar/pick'
125
5fcbd898 126 return this.authHttp.post<{ avatar: Avatar }>(url, avatarForm)
e4f0e92e 127 .pipe(catchError(err => this.restExtractor.handleError(err)))
c5911fd3
C
128 }
129
1d5342ab 130 signup (userCreate: UserRegister) {
d592e0a9 131 return this.authHttp.post(UserService.BASE_USERS_URL + 'register', userCreate)
db400f44
C
132 .pipe(
133 map(this.restExtractor.extractDataBool),
e4f0e92e 134 catchError(err => this.restExtractor.handleError(err))
db400f44 135 )
a184c71b 136 }
c5911fd3 137
ce5496d6 138 getMyVideoQuotaUsed () {
bf64ed41 139 const url = UserService.BASE_USERS_URL + 'me/video-quota-used'
c5911fd3 140
5fcbd898 141 return this.authHttp.get<UserVideoQuota>(url)
e4f0e92e 142 .pipe(catchError(err => this.restExtractor.handleError(err)))
c5911fd3 143 }
ecb4e35f
C
144
145 askResetPassword (email: string) {
146 const url = UserService.BASE_USERS_URL + '/ask-reset-password'
147
148 return this.authHttp.post(url, { email })
db400f44
C
149 .pipe(
150 map(this.restExtractor.extractDataBool),
e4f0e92e 151 catchError(err => this.restExtractor.handleError(err))
db400f44 152 )
ecb4e35f
C
153 }
154
155 resetPassword (userId: number, verificationString: string, password: string) {
156 const url = `${UserService.BASE_USERS_URL}/${userId}/reset-password`
157 const body = {
158 verificationString,
159 password
160 }
161
162 return this.authHttp.post(url, body)
db400f44
C
163 .pipe(
164 map(this.restExtractor.extractDataBool),
165 catchError(res => this.restExtractor.handleError(res))
166 )
ecb4e35f 167 }
d9eaee39 168
0ba5f5ba 169 verifyEmail (userId: number, verificationString: string, isPendingEmail: boolean) {
d9eaee39
JM
170 const url = `${UserService.BASE_USERS_URL}/${userId}/verify-email`
171 const body = {
0ba5f5ba
C
172 verificationString,
173 isPendingEmail
d9eaee39
JM
174 }
175
176 return this.authHttp.post(url, body)
177 .pipe(
178 map(this.restExtractor.extractDataBool),
179 catchError(res => this.restExtractor.handleError(res))
180 )
181 }
182
183 askSendVerifyEmail (email: string) {
184 const url = UserService.BASE_USERS_URL + '/ask-send-verify-email'
185
186 return this.authHttp.post(url, { email })
187 .pipe(
188 map(this.restExtractor.extractDataBool),
189 catchError(err => this.restExtractor.handleError(err))
190 )
191 }
74d63469
GR
192
193 autocomplete (search: string): Observable<string[]> {
194 const url = UserService.BASE_USERS_URL + 'autocomplete'
195 const params = new HttpParams().append('search', search)
196
197 return this.authHttp
198 .get<string[]>(url, { params })
199 .pipe(catchError(res => this.restExtractor.handleError(res)))
200 }
e724fa93 201
1f20622f
C
202 getNewUsername (oldDisplayName: string, newDisplayName: string, currentUsername: string) {
203 // Don't update display name, the user seems to have changed it
204 if (this.displayNameToUsername(oldDisplayName) !== currentUsername) return currentUsername
205
206 return this.displayNameToUsername(newDisplayName)
207 }
208
209 displayNameToUsername (displayName: string) {
210 if (!displayName) return ''
211
212 return displayName
213 .toLowerCase()
214 .replace(/\s/g, '_')
215 .replace(/[^a-z0-9_.]/g, '')
216 }
217
e724fa93
C
218 /* ###### Admin methods ###### */
219
220 addUser (userCreate: UserCreate) {
221 return this.authHttp.post(UserService.BASE_USERS_URL, userCreate)
222 .pipe(
223 map(this.restExtractor.extractDataBool),
224 catchError(err => this.restExtractor.handleError(err))
225 )
226 }
227
228 updateUser (userId: number, userUpdate: UserUpdate) {
229 return this.authHttp.put(UserService.BASE_USERS_URL + userId, userUpdate)
230 .pipe(
231 map(this.restExtractor.extractDataBool),
232 catchError(err => this.restExtractor.handleError(err))
233 )
234 }
235
d3217560 236 updateUsers (users: UserServerModel[], userUpdate: UserUpdate) {
fc2ec87a
JM
237 return from(users)
238 .pipe(
239 concatMap(u => this.authHttp.put(UserService.BASE_USERS_URL + u.id, userUpdate)),
240 toArray(),
241 catchError(err => this.restExtractor.handleError(err))
242 )
243 }
244
218b0874
C
245 getUserWithCache (userId: number) {
246 if (!this.userCache[userId]) {
247 this.userCache[ userId ] = this.getUser(userId).pipe(shareReplay())
248 }
249
250 return this.userCache[userId]
251 }
252
76314386
RK
253 getUser (userId: number, withStats = false) {
254 const params = new HttpParams().append('withStats', withStats + '')
255 return this.authHttp.get<UserServerModel>(UserService.BASE_USERS_URL + userId, { params })
e724fa93
C
256 .pipe(catchError(err => this.restExtractor.handleError(err)))
257 }
258
d3217560 259 getAnonymousUser () {
5c20a455 260 let videoLanguages: string[]
9870329f 261
d3217560 262 try {
a4ff3100 263 videoLanguages = JSON.parse(this.localStorageService.getItem(UserLocalStorageKeys.VIDEO_LANGUAGES))
d3217560
RK
264 } catch (err) {
265 videoLanguages = null
266 console.error('Cannot parse desired video languages from localStorage.', err)
267 }
268
269 return new User({
270 // local storage keys
a4ff3100
C
271 nsfwPolicy: this.localStorageService.getItem(UserLocalStorageKeys.NSFW_POLICY) as NSFWPolicyType,
272 webTorrentEnabled: this.localStorageService.getItem(UserLocalStorageKeys.WEBTORRENT_ENABLED) !== 'false',
273 theme: this.localStorageService.getItem(UserLocalStorageKeys.THEME) || 'instance-default',
d3217560
RK
274 videoLanguages,
275
a4ff3100
C
276 autoPlayNextVideoPlaylist: this.localStorageService.getItem(UserLocalStorageKeys.AUTO_PLAY_VIDEO_PLAYLIST) !== 'false',
277 autoPlayVideo: this.localStorageService.getItem(UserLocalStorageKeys.AUTO_PLAY_VIDEO) === 'true',
9870329f 278
d3217560 279 // session storage keys
a4ff3100 280 autoPlayNextVideo: this.sessionStorageService.getItem(UserLocalStorageKeys.SESSION_STORAGE_AUTO_PLAY_NEXT_VIDEO) === 'true'
d3217560
RK
281 })
282 }
283
8491293b
RK
284 getUsers (parameters: {
285 pagination: RestPagination
286 sort: SortMeta
287 search?: string
288 }): Observable<ResultList<UserServerModel>> {
289 const { pagination, sort, search } = parameters
290
e724fa93
C
291 let params = new HttpParams()
292 params = this.restService.addRestGetParams(params, pagination, sort)
293
8491293b
RK
294 if (search) {
295 const filters = this.restService.parseQueryStringFilter(search, {
296 blocked: {
297 prefix: 'banned:',
298 isBoolean: true,
299 handler: v => {
300 if (v === 'true') return v
301 if (v === 'false') return v
302
303 return undefined
304 }
305 }
306 })
307
308 params = this.restService.addObjectParams(params, filters)
309 }
24b9417c 310
d3217560 311 return this.authHttp.get<ResultList<UserServerModel>>(UserService.BASE_USERS_URL, { params })
e724fa93
C
312 .pipe(
313 map(res => this.restExtractor.convertResultListDateToHuman(res)),
314 map(res => this.restExtractor.applyToResultListData(res, this.formatUser.bind(this))),
315 catchError(err => this.restExtractor.handleError(err))
316 )
317 }
318
d3217560 319 removeUser (usersArg: UserServerModel | UserServerModel[]) {
791645e6
C
320 const users = Array.isArray(usersArg) ? usersArg : [ usersArg ]
321
322 return from(users)
323 .pipe(
324 concatMap(u => this.authHttp.delete(UserService.BASE_USERS_URL + u.id)),
325 toArray(),
326 catchError(err => this.restExtractor.handleError(err))
327 )
e724fa93
C
328 }
329
d3217560 330 banUsers (usersArg: UserServerModel | UserServerModel[], reason?: string) {
e724fa93 331 const body = reason ? { reason } : {}
791645e6 332 const users = Array.isArray(usersArg) ? usersArg : [ usersArg ]
e724fa93 333
791645e6
C
334 return from(users)
335 .pipe(
336 concatMap(u => this.authHttp.post(UserService.BASE_USERS_URL + u.id + '/block', body)),
337 toArray(),
338 catchError(err => this.restExtractor.handleError(err))
339 )
e724fa93
C
340 }
341
d3217560 342 unbanUsers (usersArg: UserServerModel | UserServerModel[]) {
791645e6
C
343 const users = Array.isArray(usersArg) ? usersArg : [ usersArg ]
344
345 return from(users)
346 .pipe(
347 concatMap(u => this.authHttp.post(UserService.BASE_USERS_URL + u.id + '/unblock', {})),
348 toArray(),
349 catchError(err => this.restExtractor.handleError(err))
350 )
e724fa93
C
351 }
352
5c20a455
C
353 getAnonymousOrLoggedUser () {
354 if (!this.authService.isLoggedIn()) {
355 return of(this.getAnonymousUser())
356 }
357
358 return this.authService.userInformationLoaded
359 .pipe(
360 first(),
361 map(() => this.authService.getUser())
362 )
363 }
364
d3217560 365 private formatUser (user: UserServerModel) {
e724fa93
C
366 let videoQuota
367 if (user.videoQuota === -1) {
bc99dfe5 368 videoQuota = '∞'
e724fa93
C
369 } else {
370 videoQuota = this.bytesPipe.transform(user.videoQuota, 0)
371 }
372
373 const videoQuotaUsed = this.bytesPipe.transform(user.videoQuotaUsed, 0)
374
4f5d0459
RK
375 let videoQuotaDaily: string
376 let videoQuotaUsedDaily: string
bc99dfe5
RK
377 if (user.videoQuotaDaily === -1) {
378 videoQuotaDaily = '∞'
4f5d0459 379 videoQuotaUsedDaily = this.bytesPipe.transform(0, 0) + ''
bc99dfe5 380 } else {
4f5d0459
RK
381 videoQuotaDaily = this.bytesPipe.transform(user.videoQuotaDaily, 0) + ''
382 videoQuotaUsedDaily = this.bytesPipe.transform(user.videoQuotaUsedDaily || 0, 0) + ''
bc99dfe5
RK
383 }
384
e724fa93
C
385 const roleLabels: { [ id in UserRole ]: string } = {
386 [UserRole.USER]: this.i18n('User'),
387 [UserRole.ADMINISTRATOR]: this.i18n('Administrator'),
388 [UserRole.MODERATOR]: this.i18n('Moderator')
389 }
390
391 return Object.assign(user, {
392 roleLabel: roleLabels[user.role],
393 videoQuota,
bc99dfe5
RK
394 videoQuotaUsed,
395 rawVideoQuota: user.videoQuota,
396 rawVideoQuotaUsed: user.videoQuotaUsed,
397 videoQuotaDaily,
398 videoQuotaUsedDaily,
399 rawVideoQuotaDaily: user.videoQuotaDaily,
400 rawVideoQuotaUsedDaily: user.videoQuotaUsedDaily
e724fa93
C
401 })
402 }
629d8d6f 403}