-import { map, shareReplay, switchMap, tap } from 'rxjs/operators'
+import { Observable, of, Subject } from 'rxjs'
+import { first, map, share, shareReplay, switchMap, tap } from 'rxjs/operators'
import { HttpClient } from '@angular/common/http'
import { Inject, Injectable, LOCALE_ID } from '@angular/core'
-import { peertubeLocalStorage } from '@app/shared/misc/peertube-local-storage'
-import { Observable, of, ReplaySubject } from 'rxjs'
-import { getCompleteLocale, ServerConfig } from '../../../../../shared'
-import { About } from '../../../../../shared/models/server/about.model'
+import { getDevLocale, isOnDevLocale, sortBy } from '@app/helpers'
+import { logger } from '@root-helpers/logger'
+import { getCompleteLocale, isDefaultLocale, peertubeTranslate } from '@shared/core-utils/i18n'
+import { HTMLServerConfig, ServerConfig, ServerStats, VideoConstant } from '@shared/models'
import { environment } from '../../../environments/environment'
-import { VideoConstant, VideoPrivacy } from '../../../../../shared/models/videos'
-import { isDefaultLocale, peertubeTranslate } from '../../../../../shared/models/i18n'
-import { getDevLocale, isOnDevLocale } from '@app/shared/i18n/i18n-utils'
-import { sortBy } from '@app/shared/misc/utils'
@Injectable()
export class ServerService {
private static BASE_CONFIG_URL = environment.apiUrl + '/api/v1/config/'
private static BASE_VIDEO_URL = environment.apiUrl + '/api/v1/videos/'
+ private static BASE_VIDEO_PLAYLIST_URL = environment.apiUrl + '/api/v1/video-playlists/'
private static BASE_LOCALE_URL = environment.apiUrl + '/client/locales/'
- private static CONFIG_LOCAL_STORAGE_KEY = 'server-config'
-
- configLoaded = new ReplaySubject<boolean>(1)
- videoPrivaciesLoaded = new ReplaySubject<boolean>(1)
- videoCategoriesLoaded = new ReplaySubject<boolean>(1)
- videoLicencesLoaded = new ReplaySubject<boolean>(1)
- videoLanguagesLoaded = new ReplaySubject<boolean>(1)
- localeObservable: Observable<any>
-
- private config: ServerConfig = {
- instance: {
- name: 'PeerTube',
- shortDescription: 'PeerTube, a federated (ActivityPub) video streaming platform ' +
- 'using P2P (BitTorrent) directly in the web browser with WebTorrent and Angular.',
- defaultClientRoute: '',
- defaultNSFWPolicy: 'do_not_list' as 'do_not_list',
- customizations: {
- javascript: '',
- css: ''
- }
- },
- serverVersion: 'Unknown',
- signup: {
- allowed: false,
- allowedForCurrentIP: false,
- requiresEmailVerification: false
- },
- transcoding: {
- enabledResolutions: []
- },
- avatar: {
- file: {
- size: { max: 0 },
- extensions: []
- }
- },
- video: {
- image: {
- size: { max: 0 },
- extensions: []
- },
- file: {
- extensions: []
- }
- },
- videoCaption: {
- file: {
- size: { max: 0 },
- extensions: []
- }
- },
- user: {
- videoQuota: -1,
- videoQuotaDaily: -1
- },
- import: {
- videos: {
- http: {
- enabled: false
- },
- torrent: {
- enabled: false
- }
- }
- }
- }
- private videoCategories: Array<VideoConstant<number>> = []
- private videoLicences: Array<VideoConstant<number>> = []
- private videoLanguages: Array<VideoConstant<string>> = []
- private videoPrivacies: Array<VideoConstant<VideoPrivacy>> = []
+ private static BASE_STATS_URL = environment.apiUrl + '/api/v1/server/stats'
+
+ configReloaded = new Subject<ServerConfig>()
+
+ private localeObservable: Observable<any>
+ private videoLicensesObservable: Observable<VideoConstant<number>[]>
+ private videoCategoriesObservable: Observable<VideoConstant<number>[]>
+ private videoPrivaciesObservable: Observable<VideoConstant<number>[]>
+ private videoPlaylistPrivaciesObservable: Observable<VideoConstant<number>[]>
+ private videoLanguagesObservable: Observable<VideoConstant<string>[]>
+ private configObservable: Observable<ServerConfig>
+
+ private configReset = false
+
+ private configLoaded = false
+ private config: ServerConfig
+ private htmlConfig: HTMLServerConfig
constructor (
private http: HttpClient,
@Inject(LOCALE_ID) private localeId: string
) {
- this.loadServerLocale()
- this.loadConfigLocally()
}
- loadConfig () {
- this.http.get<ServerConfig>(ServerService.BASE_CONFIG_URL)
- .pipe(tap(this.saveConfigLocally))
- .subscribe(data => {
- this.config = data
+ loadHTMLConfig () {
+ try {
+ this.loadHTMLConfigLocally()
+ } catch (err) {
+ // Expected in dev mode since we can't inject the config in the HTML
+ if (environment.production !== false) {
+ logger.error('Cannot load config locally. Fallback to API.')
+ }
- this.configLoaded.next(true)
- })
+ return this.getConfig()
+ }
}
- loadVideoCategories () {
- return this.loadVideoAttributeEnum('categories', this.videoCategories, this.videoCategoriesLoaded, true)
- }
+ getServerVersionAndCommit () {
+ const serverVersion = this.config.serverVersion
+ const commit = this.config.serverCommit || ''
- loadVideoLicences () {
- return this.loadVideoAttributeEnum('licences', this.videoLicences, this.videoLicencesLoaded)
- }
+ let result = serverVersion
+ if (commit) result += '...' + commit
- loadVideoLanguages () {
- return this.loadVideoAttributeEnum('languages', this.videoLanguages, this.videoLanguagesLoaded, true)
+ return result
}
- loadVideoPrivacies () {
- return this.loadVideoAttributeEnum('privacies', this.videoPrivacies, this.videoPrivaciesLoaded)
+ resetConfig () {
+ this.configLoaded = false
+ this.configReset = true
+
+ // Notify config update
+ return this.getConfig()
}
getConfig () {
- return this.config
+ if (this.configLoaded) return of(this.config)
+
+ if (!this.configObservable) {
+ this.configObservable = this.http.get<ServerConfig>(ServerService.BASE_CONFIG_URL)
+ .pipe(
+ tap(config => {
+ this.config = config
+ this.htmlConfig = config
+ this.configLoaded = true
+ }),
+ tap(config => {
+ if (this.configReset) {
+ this.configReloaded.next(config)
+ this.configReset = false
+ }
+ }),
+ share()
+ )
+ }
+
+ return this.configObservable
+ }
+
+ getHTMLConfig () {
+ return this.htmlConfig
}
getVideoCategories () {
- return this.videoCategories
+ if (!this.videoCategoriesObservable) {
+ this.videoCategoriesObservable = this.loadAttributeEnum<number>(ServerService.BASE_VIDEO_URL, 'categories', true)
+ }
+
+ return this.videoCategoriesObservable.pipe(first())
}
getVideoLicences () {
- return this.videoLicences
+ if (!this.videoLicensesObservable) {
+ this.videoLicensesObservable = this.loadAttributeEnum<number>(ServerService.BASE_VIDEO_URL, 'licences')
+ }
+
+ return this.videoLicensesObservable.pipe(first())
}
getVideoLanguages () {
- return this.videoLanguages
+ if (!this.videoLanguagesObservable) {
+ this.videoLanguagesObservable = this.loadAttributeEnum<string>(ServerService.BASE_VIDEO_URL, 'languages', true)
+ }
+
+ return this.videoLanguagesObservable.pipe(first())
}
getVideoPrivacies () {
- return this.videoPrivacies
- }
+ if (!this.videoPrivaciesObservable) {
+ this.videoPrivaciesObservable = this.loadAttributeEnum<number>(ServerService.BASE_VIDEO_URL, 'privacies')
+ }
- getAbout () {
- return this.http.get<About>(ServerService.BASE_CONFIG_URL + '/about')
+ return this.videoPrivaciesObservable.pipe(first())
}
- private loadVideoAttributeEnum (
- attributeName: 'categories' | 'licences' | 'languages' | 'privacies',
- hashToPopulate: VideoConstant<string | number>[],
- notifier: ReplaySubject<boolean>,
- sort = false
- ) {
- this.localeObservable
- .pipe(
- switchMap(translations => {
- return this.http.get(ServerService.BASE_VIDEO_URL + attributeName)
- .pipe(map(data => ({ data, translations })))
- })
- )
- .subscribe(({ data, translations }) => {
- Object.keys(data)
- .forEach(dataKey => {
- const label = data[ dataKey ]
-
- hashToPopulate.push({
- id: attributeName === 'languages' ? dataKey : parseInt(dataKey, 10),
- label: peertubeTranslate(label, translations)
- })
- })
-
- if (sort === true) sortBy(hashToPopulate, 'label')
-
- notifier.next(true)
- })
+ getVideoPlaylistPrivacies () {
+ if (!this.videoPlaylistPrivaciesObservable) {
+ this.videoPlaylistPrivaciesObservable = this.loadAttributeEnum<number>(ServerService.BASE_VIDEO_PLAYLIST_URL, 'privacies')
+ }
+
+ return this.videoPlaylistPrivaciesObservable.pipe(first())
}
- private loadServerLocale () {
- const completeLocale = isOnDevLocale() ? getDevLocale() : getCompleteLocale(this.localeId)
+ getServerLocale (): Observable<{ [ id: string ]: string }> {
+ if (!this.localeObservable) {
+ const completeLocale = isOnDevLocale() ? getDevLocale() : getCompleteLocale(this.localeId)
- // Default locale, nothing to translate
- if (isDefaultLocale(completeLocale)) {
- this.localeObservable = of({}).pipe(shareReplay())
- return
+ // Default locale, nothing to translate
+ if (isDefaultLocale(completeLocale)) {
+ this.localeObservable = of({}).pipe(shareReplay())
+ } else {
+ this.localeObservable = this.http
+ .get(ServerService.BASE_LOCALE_URL + completeLocale + '/server.json')
+ .pipe(shareReplay())
+ }
}
- this.localeObservable = this.http
- .get(ServerService.BASE_LOCALE_URL + completeLocale + '/server.json')
- .pipe(shareReplay())
+ return this.localeObservable.pipe(first())
}
- private saveConfigLocally (config: ServerConfig) {
- peertubeLocalStorage.setItem(ServerService.CONFIG_LOCAL_STORAGE_KEY, JSON.stringify(config))
+ getServerStats () {
+ return this.http.get<ServerStats>(ServerService.BASE_STATS_URL)
}
- private loadConfigLocally () {
- const configString = peertubeLocalStorage.getItem(ServerService.CONFIG_LOCAL_STORAGE_KEY)
-
- if (configString) {
- try {
- const parsed = JSON.parse(configString)
- Object.assign(this.config, parsed)
- } catch (err) {
- console.error('Cannot parse config saved in local storage.', err)
- }
+ private loadAttributeEnum <T extends string | number> (
+ baseUrl: string,
+ attributeName: 'categories' | 'licences' | 'languages' | 'privacies',
+ sort = false
+ ) {
+ return this.getServerLocale()
+ .pipe(
+ switchMap(translations => {
+ return this.http.get<{ [ id: string ]: string }>(baseUrl + attributeName)
+ .pipe(map(data => ({ data, translations })))
+ }),
+ map(({ data, translations }) => {
+ const hashToPopulate: VideoConstant<T>[] = Object.keys(data)
+ .map(dataKey => {
+ const label = data[dataKey]
+
+ const id = attributeName === 'languages'
+ ? dataKey as T
+ : parseInt(dataKey, 10) as T
+
+ return {
+ id,
+ label: peertubeTranslate(label, translations)
+ }
+ })
+
+ if (sort === true) sortBy(hashToPopulate, 'label')
+
+ return hashToPopulate
+ }),
+ shareReplay()
+ )
+ }
+
+ private loadHTMLConfigLocally () {
+ // FIXME: typings
+ const configString = (window as any)['PeerTubeServerConfig']
+ if (!configString) {
+ throw new Error('Could not find PeerTubeServerConfig in HTML')
}
+
+ this.htmlConfig = JSON.parse(configString)
}
}