2022-03-16 01:28:54 +01:00
|
|
|
const Path = require('path')
|
2022-03-18 01:10:47 +01:00
|
|
|
const { PlayMethod } = require('./utils/constants')
|
2022-03-16 01:28:54 +01:00
|
|
|
const PlaybackSession = require('./objects/PlaybackSession')
|
2022-03-18 01:10:47 +01:00
|
|
|
const Stream = require('./objects/Stream')
|
|
|
|
const Logger = require('./Logger')
|
2022-03-16 00:57:15 +01:00
|
|
|
|
|
|
|
class PlaybackSessionManager {
|
2022-03-16 01:28:54 +01:00
|
|
|
constructor(db, emitter, clientEmitter) {
|
|
|
|
this.db = db
|
|
|
|
this.StreamsPath = Path.join(global.MetadataPath, 'streams')
|
|
|
|
this.emitter = emitter
|
|
|
|
this.clientEmitter = clientEmitter
|
|
|
|
|
|
|
|
this.sessions = []
|
|
|
|
}
|
|
|
|
|
2022-03-18 01:10:47 +01:00
|
|
|
getSession(sessionId) {
|
|
|
|
return this.sessions.find(s => s.id === sessionId)
|
|
|
|
}
|
|
|
|
getUserSession(userId) {
|
|
|
|
return this.sessions.find(s => s.userId === userId)
|
|
|
|
}
|
|
|
|
getStream(sessionId) {
|
|
|
|
var session = this.getSession(sessionId)
|
|
|
|
return session ? session.stream : null
|
2022-03-16 01:28:54 +01:00
|
|
|
}
|
2022-03-16 00:57:15 +01:00
|
|
|
|
2022-03-18 01:10:47 +01:00
|
|
|
async startSessionRequest(user, libraryItem, mediaEntity, options, res) {
|
|
|
|
const session = await this.startSession(user, libraryItem, mediaEntity, options)
|
|
|
|
res.json(session.toJSONForClient())
|
|
|
|
}
|
|
|
|
|
|
|
|
async syncSessionRequest(user, session, payload, res) {
|
|
|
|
await this.syncSession(user, session, payload)
|
|
|
|
res.json(session.toJSONForClient())
|
|
|
|
}
|
|
|
|
|
|
|
|
async closeSessionRequest(user, session, syncData, res) {
|
|
|
|
await this.closeSession(user, session, syncData)
|
|
|
|
res.sendStatus(200)
|
|
|
|
}
|
2022-03-17 01:15:25 +01:00
|
|
|
|
2022-03-18 01:10:47 +01:00
|
|
|
async startSession(user, libraryItem, mediaEntity, options) {
|
|
|
|
var shouldDirectPlay = options.forceDirectPlay || (!options.forceTranscode && mediaEntity.checkCanDirectPlay(options))
|
|
|
|
|
|
|
|
const userProgress = user.getLibraryItemProgress(libraryItem.id)
|
|
|
|
var userStartTime = 0
|
|
|
|
if (userProgress) userStartTime = userProgress.currentTime || 0
|
2022-03-16 01:28:54 +01:00
|
|
|
const newPlaybackSession = new PlaybackSession()
|
2022-03-18 01:10:47 +01:00
|
|
|
newPlaybackSession.setData(libraryItem, mediaEntity, user)
|
|
|
|
|
|
|
|
var audioTracks = []
|
|
|
|
if (shouldDirectPlay) {
|
|
|
|
Logger.debug(`[PlaybackSessionManager] "${user.username}" starting direct play session for media entity "${mediaEntity.id}"`)
|
|
|
|
audioTracks = mediaEntity.getDirectPlayTracklist(libraryItem.id)
|
|
|
|
newPlaybackSession.playMethod = PlayMethod.DIRECTPLAY
|
|
|
|
} else {
|
|
|
|
Logger.debug(`[PlaybackSessionManager] "${user.username}" starting stream session for media entity "${mediaEntity.id}"`)
|
|
|
|
var stream = new Stream(newPlaybackSession.id, this.StreamsPath, user, libraryItem, mediaEntity, userStartTime, this.clientEmitter.bind(this))
|
|
|
|
await stream.generatePlaylist()
|
|
|
|
audioTracks = [stream.getAudioTrack()]
|
|
|
|
newPlaybackSession.stream = stream
|
|
|
|
newPlaybackSession.playMethod = PlayMethod.TRANSCODE
|
|
|
|
stream.on('closed', () => {
|
|
|
|
Logger.debug(`[PlaybackSessionManager] Stream closed for session "${newPlaybackSession.id}"`)
|
|
|
|
newPlaybackSession.stream = null
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
newPlaybackSession.currentTime = userStartTime
|
|
|
|
newPlaybackSession.audioTracks = audioTracks
|
|
|
|
|
|
|
|
// Will save on the first sync
|
|
|
|
user.currentSessionId = newPlaybackSession.id
|
|
|
|
|
2022-03-16 01:28:54 +01:00
|
|
|
this.sessions.push(newPlaybackSession)
|
2022-03-18 01:10:47 +01:00
|
|
|
this.emitter('user_stream_update', user.toJSONForPublic(this.sessions, this.db.libraryItems))
|
|
|
|
|
2022-03-16 01:28:54 +01:00
|
|
|
return newPlaybackSession
|
2022-03-16 00:57:15 +01:00
|
|
|
}
|
2022-03-18 01:10:47 +01:00
|
|
|
|
|
|
|
async syncSession(user, session, syncData) {
|
2022-03-18 21:31:46 +01:00
|
|
|
var libraryItem = this.db.libraryItems.find(li => li.id === session.libraryItemId)
|
|
|
|
if (!libraryItem) {
|
|
|
|
Logger.error(`[PlaybackSessionManager] syncSession Library Item not found "${sessino.libraryItemId}"`)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2022-03-18 01:10:47 +01:00
|
|
|
session.currentTime = syncData.currentTime
|
|
|
|
session.addListeningTime(syncData.timeListened)
|
|
|
|
Logger.debug(`[PlaybackSessionManager] syncSession "${session.id}" | Total Time Listened: ${session.timeListening}`)
|
|
|
|
|
|
|
|
const itemProgressUpdate = {
|
2022-03-18 21:31:46 +01:00
|
|
|
mediaEntityId: syncData.mediaEntityId || null,
|
|
|
|
duration: syncData.duration,
|
2022-03-18 01:10:47 +01:00
|
|
|
currentTime: syncData.currentTime,
|
|
|
|
progress: session.progress
|
|
|
|
}
|
2022-03-18 21:31:46 +01:00
|
|
|
var wasUpdated = user.createUpdateLibraryItemProgress(libraryItem, itemProgressUpdate)
|
2022-03-18 01:10:47 +01:00
|
|
|
if (wasUpdated) {
|
|
|
|
await this.db.updateEntity('user', user)
|
|
|
|
var itemProgress = user.getLibraryItemProgress(session.libraryItemId)
|
|
|
|
this.clientEmitter(user.id, 'user_item_progress_updated', {
|
|
|
|
id: itemProgress.id,
|
|
|
|
data: itemProgress.toJSON()
|
|
|
|
})
|
|
|
|
}
|
|
|
|
this.saveSession(session)
|
|
|
|
}
|
|
|
|
|
|
|
|
async closeSession(user, session, syncData = null) {
|
|
|
|
if (syncData) {
|
|
|
|
await this.syncSession(user, session, syncData)
|
|
|
|
} else {
|
|
|
|
await this.saveSession(session)
|
|
|
|
}
|
|
|
|
Logger.debug(`[PlaybackSessionManager] closeSession "${session.id}"`)
|
|
|
|
this.emitter('user_stream_update', user.toJSONForPublic(this.sessions, this.db.libraryItems))
|
|
|
|
return this.removeSession(session.id)
|
|
|
|
}
|
|
|
|
|
|
|
|
saveSession(session) {
|
2022-03-18 21:31:46 +01:00
|
|
|
if (!session.timeListening) return // Do not save a session with no listening time
|
|
|
|
|
2022-03-18 01:10:47 +01:00
|
|
|
if (session.lastSave) {
|
|
|
|
return this.db.updateEntity('session', session)
|
|
|
|
} else {
|
|
|
|
session.lastSave = Date.now()
|
|
|
|
return this.db.insertEntity('session', session)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
async removeSession(sessionId) {
|
|
|
|
var session = this.sessions.find(s => s.id === sessionId)
|
|
|
|
if (!session) return
|
|
|
|
if (session.stream) {
|
|
|
|
await session.stream.close()
|
|
|
|
}
|
|
|
|
this.sessions = this.sessions.filter(s => s.id !== sessionId)
|
|
|
|
Logger.debug(`[PlaybackSessionManager] Removed session "${sessionId}"`)
|
|
|
|
}
|
2022-03-16 00:57:15 +01:00
|
|
|
}
|
|
|
|
module.exports = PlaybackSessionManager
|