-import { map, share, switchMap, tap } from 'rxjs/operators'
+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 { peertubeLocalStorage } from '@app/shared/misc/peertube-web-storage'
+import { Observable, of, Subject } from 'rxjs'
import { getCompleteLocale, ServerConfig } from '../../../../../shared'
-import { About } from '../../../../../shared/models/server/about.model'
import { environment } from '../../../environments/environment'
import { VideoConstant } from '../../../../../shared/models/videos'
-import { isDefaultLocale } from '../../../../../shared/models/i18n'
-import { getDevLocale, isOnDevLocale, peertubeTranslate } from '@app/shared/i18n/i18n-utils'
+import { isDefaultLocale, peertubeTranslate } from '../../../../../shared/models/i18n'
+import { getDevLocale, isOnDevLocale } from '@app/shared/i18n/i18n-utils'
import { sortBy } from '@app/shared/misc/utils'
+import { ServerStats } from '@shared/models/server'
@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 BASE_STATS_URL = environment.apiUrl + '/api/v1/server/stats'
+
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>
+ configReloaded = new Subject<void>()
+
+ 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 = {
instance: {
name: 'PeerTube',
shortDescription: 'PeerTube, a federated (ActivityPub) video streaming platform ' +
'using P2P (BitTorrent) directly in the web browser with WebTorrent and Angular.',
defaultClientRoute: '',
+ isNSFW: false,
defaultNSFWPolicy: 'do_not_list' as 'do_not_list',
customizations: {
javascript: '',
css: ''
}
},
+ search: {
+ remoteUri: {
+ users: true,
+ anonymous: false
+ }
+ },
+ plugin: {
+ registered: []
+ },
+ theme: {
+ registered: [],
+ default: 'default'
+ },
+ email: {
+ enabled: false
+ },
+ contactForm: {
+ enabled: false
+ },
serverVersion: 'Unknown',
signup: {
allowed: false,
- allowedForCurrentIP: false
+ allowedForCurrentIP: false,
+ requiresEmailVerification: false
},
transcoding: {
- enabledResolutions: []
+ enabledResolutions: [],
+ hls: {
+ enabled: false
+ },
+ webtorrent: {
+ enabled: true
+ }
},
avatar: {
file: {
}
},
user: {
- videoQuota: -1
+ videoQuota: -1,
+ videoQuotaDaily: -1
},
import: {
videos: {
http: {
enabled: false
+ },
+ torrent: {
+ enabled: false
+ }
+ }
+ },
+ trending: {
+ videos: {
+ intervalDays: 0
+ }
+ },
+ autoBlacklist: {
+ videos: {
+ ofUsers: {
+ enabled: false
+ }
+ }
+ },
+ tracker: {
+ enabled: true
+ },
+ followings: {
+ instance: {
+ autoFollowIndex: {
+ indexUrl: 'https://instances.joinpeertube.org'
}
}
}
}
- private videoCategories: Array<VideoConstant<string>> = []
- private videoLicences: Array<VideoConstant<string>> = []
- private videoLanguages: Array<VideoConstant<string>> = []
- private videoPrivacies: Array<VideoConstant<string>> = []
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
+ getServerVersionAndCommit () {
+ const serverVersion = this.config.serverVersion
+ const commit = this.config.serverCommit || ''
- this.configLoaded.next(true)
- })
- }
+ let result = serverVersion
+ if (commit) result += '...' + commit
- loadVideoCategories () {
- return this.loadVideoAttributeEnum('categories', this.videoCategories, this.videoCategoriesLoaded, true)
+ return result
}
- loadVideoLicences () {
- return this.loadVideoAttributeEnum('licences', this.videoLicences, this.videoLicencesLoaded)
+ resetConfig () {
+ this.configLoaded = false
+ this.configReset = true
}
- loadVideoLanguages () {
- return this.loadVideoAttributeEnum('languages', this.videoLanguages, this.videoLanguagesLoaded, true)
- }
+ getConfig () {
+ if (this.configLoaded) return of(this.config)
+
+ if (!this.configObservable) {
+ this.configObservable = this.http.get<ServerConfig>(ServerService.BASE_CONFIG_URL)
+ .pipe(
+ tap(config => this.saveConfigLocally(config)),
+ tap(config => {
+ this.config = config
+ this.configLoaded = true
+ }),
+ tap(() => {
+ if (this.configReset) {
+ this.configReloaded.next()
+ this.configReset = false
+ }
+ }),
+ share()
+ )
+ }
- loadVideoPrivacies () {
- return this.loadVideoAttributeEnum('privacies', this.videoPrivacies, this.videoPrivaciesLoaded)
+ return this.configObservable
}
- getConfig () {
+ getTmpConfig () {
return this.config
}
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')
+ }
+
+ return this.videoPrivaciesObservable.pipe(first())
+ }
+
+ getVideoPlaylistPrivacies () {
+ if (!this.videoPlaylistPrivaciesObservable) {
+ this.videoPlaylistPrivaciesObservable = this.loadAttributeEnum<number>(ServerService.BASE_VIDEO_PLAYLIST_URL, 'privacies')
+ }
+
+ return this.videoPlaylistPrivaciesObservable.pipe(first())
}
- getAbout () {
- return this.http.get<About>(ServerService.BASE_CONFIG_URL + '/about')
+ getServerLocale () {
+ if (!this.localeObservable) {
+ const completeLocale = isOnDevLocale() ? getDevLocale() : getCompleteLocale(this.localeId)
+
+ // 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())
+ }
+ }
+
+ return this.localeObservable.pipe(first())
}
- private loadVideoAttributeEnum (
+ getServerStats () {
+ return this.http.get<ServerStats>(ServerService.BASE_STATS_URL)
+ }
+
+ private loadAttributeEnum <T extends string | number> (
+ baseUrl: string,
attributeName: 'categories' | 'licences' | 'languages' | 'privacies',
- hashToPopulate: VideoConstant<string>[],
- 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: dataKey,
- label: peertubeTranslate(label, translations)
- })
- })
-
- if (sort === true) sortBy(hashToPopulate, 'label')
-
- notifier.next(true)
- })
- }
-
- private loadServerLocale () {
- const completeLocale = isOnDevLocale() ? getDevLocale() : getCompleteLocale(this.localeId)
-
- // Default locale, nothing to translate
- if (isDefaultLocale(completeLocale)) {
- this.localeObservable = of({}).pipe(share())
- return
- }
+ 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)
+ .forEach(dataKey => {
+ const label = data[ dataKey ]
+
+ hashToPopulate.push({
+ id: (attributeName === 'languages' ? dataKey : parseInt(dataKey, 10)) as T,
+ label: peertubeTranslate(label, translations)
+ })
+ })
+
+ if (sort === true) sortBy(hashToPopulate, 'label')
- this.localeObservable = this.http
- .get(ServerService.BASE_LOCALE_URL + completeLocale + '/server.json')
- .pipe(share())
+ return hashToPopulate
+ }),
+ shareReplay()
+ )
}
private saveConfigLocally (config: ServerConfig) {