import { Observable } from 'rxjs' import { catchError, map, switchMap } from 'rxjs/operators' import { HttpClient, HttpParams, HttpRequest } from '@angular/common/http' import { Injectable } from '@angular/core' import { ComponentPaginationLight, RestExtractor, RestService, ServerService, UserService } from '@app/core' import { objectToFormData } from '@app/helpers' import { FeedFormat, NSFWPolicyType, ResultList, UserVideoRate, UserVideoRateType, UserVideoRateUpdate, Video as VideoServerModel, VideoConstant, VideoDetails as VideoDetailsServerModel, VideoFileMetadata, VideoFilter, VideoPrivacy, VideoSortField, VideoUpdate } from '@shared/models' import { environment } from '../../../../environments/environment' import { Account } from '../account/account.model' import { AccountService } from '../account/account.service' import { VideoChannel, VideoChannelService } from '../video-channel' import { VideoDetails } from './video-details.model' import { VideoEdit } from './video-edit.model' import { Video } from './video.model' export interface VideosProvider { getVideos (parameters: { videoPagination: ComponentPaginationLight, sort: VideoSortField, filter?: VideoFilter, categoryOneOf?: number[], languageOneOf?: string[] nsfwPolicy: NSFWPolicyType }): Observable> } @Injectable() export class VideoService implements VideosProvider { static BASE_VIDEO_URL = environment.apiUrl + '/api/v1/videos/' static BASE_FEEDS_URL = environment.apiUrl + '/feeds/videos.' static BASE_SUBSCRIPTION_FEEDS_URL = environment.apiUrl + '/feeds/subscriptions.' constructor ( private authHttp: HttpClient, private restExtractor: RestExtractor, private restService: RestService, private serverService: ServerService ) {} getVideoViewUrl (uuid: string) { return VideoService.BASE_VIDEO_URL + uuid + '/views' } getUserWatchingVideoUrl (uuid: string) { return VideoService.BASE_VIDEO_URL + uuid + '/watching' } getVideo (options: { videoId: string }): Observable { return this.serverService.getServerLocale() .pipe( switchMap(translations => { return this.authHttp.get(VideoService.BASE_VIDEO_URL + options.videoId) .pipe(map(videoHash => ({ videoHash, translations }))) }), map(({ videoHash, translations }) => new VideoDetails(videoHash, translations)), catchError(err => this.restExtractor.handleError(err)) ) } updateVideo (video: VideoEdit) { const language = video.language || null const licence = video.licence || null const category = video.category || null const description = video.description || null const support = video.support || null const scheduleUpdate = video.scheduleUpdate || null const originallyPublishedAt = video.originallyPublishedAt || null const body: VideoUpdate = { name: video.name, category, licence, language, support, description, channelId: video.channelId, privacy: video.privacy, tags: video.tags, nsfw: video.nsfw, waitTranscoding: video.waitTranscoding, commentsEnabled: video.commentsEnabled, downloadEnabled: video.downloadEnabled, thumbnailfile: video.thumbnailfile, previewfile: video.previewfile, pluginData: video.pluginData, scheduleUpdate, originallyPublishedAt } const data = objectToFormData(body) return this.authHttp.put(VideoService.BASE_VIDEO_URL + video.id, data) .pipe( map(this.restExtractor.extractDataBool), catchError(err => this.restExtractor.handleError(err)) ) } uploadVideo (video: FormData) { const req = new HttpRequest('POST', VideoService.BASE_VIDEO_URL + 'upload', video, { reportProgress: true }) return this.authHttp .request<{ video: { id: number, uuid: string } }>(req) .pipe(catchError(err => this.restExtractor.handleError(err))) } getMyVideos (videoPagination: ComponentPaginationLight, sort: VideoSortField, search?: string): Observable> { const pagination = this.restService.componentPaginationToRestPagination(videoPagination) let params = new HttpParams() params = this.restService.addRestGetParams(params, pagination, sort) if (search) { const filters = this.restService.parseQueryStringFilter(search, { isLive: { prefix: 'isLive:', isBoolean: true } }) params = this.restService.addObjectParams(params, filters) } return this.authHttp .get>(UserService.BASE_USERS_URL + 'me/videos', { params }) .pipe( switchMap(res => this.extractVideos(res)), catchError(err => this.restExtractor.handleError(err)) ) } getAccountVideos (parameters: { account: Pick, videoPagination: ComponentPaginationLight, sort: VideoSortField nsfwPolicy?: NSFWPolicyType videoFilter?: VideoFilter search?: string }): Observable> { const { account, videoPagination, sort, videoFilter, nsfwPolicy, search } = parameters const pagination = this.restService.componentPaginationToRestPagination(videoPagination) let params = new HttpParams() params = this.restService.addRestGetParams(params, pagination, sort) if (nsfwPolicy) { params = params.set('nsfw', this.nsfwPolicyToParam(nsfwPolicy)) } if (videoFilter) { params = params.set('filter', videoFilter) } if (search) { params = params.set('search', search) } return this.authHttp .get>(AccountService.BASE_ACCOUNT_URL + account.nameWithHost + '/videos', { params }) .pipe( switchMap(res => this.extractVideos(res)), catchError(err => this.restExtractor.handleError(err)) ) } getVideoChannelVideos (parameters: { videoChannel: Pick, videoPagination: ComponentPaginationLight, sort: VideoSortField, nsfwPolicy?: NSFWPolicyType videoFilter?: VideoFilter }): Observable> { const { videoChannel, videoPagination, sort, nsfwPolicy, videoFilter } = parameters const pagination = this.restService.componentPaginationToRestPagination(videoPagination) let params = new HttpParams() params = this.restService.addRestGetParams(params, pagination, sort) if (nsfwPolicy) { params = params.set('nsfw', this.nsfwPolicyToParam(nsfwPolicy)) } if (videoFilter) { params = params.set('filter', videoFilter) } return this.authHttp .get>(VideoChannelService.BASE_VIDEO_CHANNEL_URL + videoChannel.nameWithHost + '/videos', { params }) .pipe( switchMap(res => this.extractVideos(res)), catchError(err => this.restExtractor.handleError(err)) ) } getVideos (parameters: { videoPagination: ComponentPaginationLight, sort: VideoSortField, filter?: VideoFilter, categoryOneOf?: number[], languageOneOf?: string[], skipCount?: boolean, nsfwPolicy?: NSFWPolicyType }): Observable> { const { videoPagination, sort, filter, categoryOneOf, languageOneOf, skipCount, nsfwPolicy } = parameters const pagination = this.restService.componentPaginationToRestPagination(videoPagination) let params = new HttpParams() params = this.restService.addRestGetParams(params, pagination, sort) if (filter) params = params.set('filter', filter) if (skipCount) params = params.set('skipCount', skipCount + '') if (nsfwPolicy) { params = params.set('nsfw', this.nsfwPolicyToParam(nsfwPolicy)) } if (languageOneOf) { for (const l of languageOneOf) { params = params.append('languageOneOf[]', l) } } if (categoryOneOf) { for (const c of categoryOneOf) { params = params.append('categoryOneOf[]', c + '') } } return this.authHttp .get>(VideoService.BASE_VIDEO_URL, { params }) .pipe( switchMap(res => this.extractVideos(res)), catchError(err => this.restExtractor.handleError(err)) ) } buildBaseFeedUrls (params: HttpParams, base = VideoService.BASE_FEEDS_URL) { const feeds = [ { format: FeedFormat.RSS, label: 'media rss 2.0', url: base + FeedFormat.RSS.toLowerCase() }, { format: FeedFormat.ATOM, label: 'atom 1.0', url: base + FeedFormat.ATOM.toLowerCase() }, { format: FeedFormat.JSON, label: 'json 1.0', url: base + FeedFormat.JSON.toLowerCase() } ] if (params && params.keys().length !== 0) { for (const feed of feeds) { feed.url += '?' + params.toString() } } return feeds } getVideoFeedUrls (sort: VideoSortField, filter?: VideoFilter, categoryOneOf?: number[]) { let params = this.restService.addRestGetParams(new HttpParams(), undefined, sort) if (filter) params = params.set('filter', filter) if (categoryOneOf) { for (const c of categoryOneOf) { params = params.append('categoryOneOf[]', c + '') } } return this.buildBaseFeedUrls(params) } getAccountFeedUrls (accountId: number) { let params = this.restService.addRestGetParams(new HttpParams()) params = params.set('accountId', accountId.toString()) return this.buildBaseFeedUrls(params) } getVideoChannelFeedUrls (videoChannelId: number) { let params = this.restService.addRestGetParams(new HttpParams()) params = params.set('videoChannelId', videoChannelId.toString()) return this.buildBaseFeedUrls(params) } getVideoSubscriptionFeedUrls (accountId: number, feedToken: string) { let params = this.restService.addRestGetParams(new HttpParams()) params = params.set('accountId', accountId.toString()) params = params.set('token', feedToken) return this.buildBaseFeedUrls(params, VideoService.BASE_SUBSCRIPTION_FEEDS_URL) } getVideoFileMetadata (metadataUrl: string) { return this.authHttp .get(metadataUrl) .pipe( catchError(err => this.restExtractor.handleError(err)) ) } removeVideo (id: number) { return this.authHttp .delete(VideoService.BASE_VIDEO_URL + id) .pipe( map(this.restExtractor.extractDataBool), catchError(err => this.restExtractor.handleError(err)) ) } loadCompleteDescription (descriptionPath: string) { return this.authHttp .get<{ description: string }>(environment.apiUrl + descriptionPath) .pipe( map(res => res.description), catchError(err => this.restExtractor.handleError(err)) ) } setVideoLike (id: number) { return this.setVideoRate(id, 'like') } setVideoDislike (id: number) { return this.setVideoRate(id, 'dislike') } unsetVideoLike (id: number) { return this.setVideoRate(id, 'none') } getUserVideoRating (id: number) { const url = UserService.BASE_USERS_URL + 'me/videos/' + id + '/rating' return this.authHttp.get(url) .pipe(catchError(err => this.restExtractor.handleError(err))) } extractVideos (result: ResultList) { return this.serverService.getServerLocale() .pipe( map(translations => { const videosJson = result.data const totalVideos = result.total const videos: Video[] = [] for (const videoJson of videosJson) { videos.push(new Video(videoJson, translations)) } return { total: totalVideos, data: videos } }) ) } explainedPrivacyLabels (serverPrivacies: VideoConstant[], defaultPrivacyId = VideoPrivacy.PUBLIC) { const descriptions = [ { id: VideoPrivacy.PRIVATE, description: $localize`Only I can see this video` }, { id: VideoPrivacy.UNLISTED, description: $localize`Only shareable via a private link` }, { id: VideoPrivacy.PUBLIC, description: $localize`Anyone can see this video` }, { id: VideoPrivacy.INTERNAL, description: $localize`Only users of this instance can see this video` } ] return { defaultPrivacyId: serverPrivacies.find(p => p.id === defaultPrivacyId)?.id || serverPrivacies[0].id, videoPrivacies: serverPrivacies.map(p => ({ ...p, description: descriptions.find(p => p.id).description })) } } getHighestAvailablePrivacy (serverPrivacies: VideoConstant[]) { const order = [ VideoPrivacy.PRIVATE, VideoPrivacy.INTERNAL, VideoPrivacy.UNLISTED, VideoPrivacy.PUBLIC ] for (const privacy of order) { if (serverPrivacies.find(p => p.id === privacy)) { return privacy } } throw new Error('No highest privacy available') } nsfwPolicyToParam (nsfwPolicy: NSFWPolicyType) { return nsfwPolicy === 'do_not_list' ? 'false' : 'both' } private setVideoRate (id: number, rateType: UserVideoRateType) { const url = VideoService.BASE_VIDEO_URL + id + '/rate' const body: UserVideoRateUpdate = { rating: rateType } return this.authHttp .put(url, body) .pipe( map(this.restExtractor.extractDataBool), catchError(err => this.restExtractor.handleError(err)) ) } }