2022-11-24 15:53:58 -06:00
const Logger = require ( '../Logger' )
const SocketAuthority = require ( '../SocketAuthority' )
2022-07-05 19:53:01 -05:00
const fs = require ( '../libs/fsExtra' )
2022-03-26 11:59:34 -05:00
2022-09-15 18:35:56 -05:00
const { getPodcastFeed } = require ( '../utils/podcastUtils' )
2022-08-15 17:35:13 -05:00
const { downloadFile , removeFile } = require ( '../utils/fileUtils' )
2022-09-30 16:55:31 -05:00
const filePerms = require ( '../utils/filePerms' )
2022-07-31 13:12:37 -05:00
const { levenshteinDistance } = require ( '../utils/index' )
2022-05-29 11:46:45 -05:00
const opmlParser = require ( '../utils/parsers/parseOPML' )
2022-03-21 19:24:38 -05:00
const prober = require ( '../utils/prober' )
2022-11-24 15:53:58 -06:00
2022-03-21 19:24:38 -05:00
const LibraryFile = require ( '../objects/files/LibraryFile' )
const PodcastEpisodeDownload = require ( '../objects/PodcastEpisodeDownload' )
const PodcastEpisode = require ( '../objects/entities/PodcastEpisode' )
const AudioFile = require ( '../objects/files/AudioFile' )
2022-03-20 16:41:06 -05:00
class PodcastManager {
2022-11-24 15:53:58 -06:00
constructor ( db , watcher , notificationManager ) {
2022-03-20 16:41:06 -05:00
this . db = db
2022-03-21 19:24:38 -05:00
this . watcher = watcher
2022-09-20 18:08:41 -05:00
this . notificationManager = notificationManager
2022-03-20 16:41:06 -05:00
this . downloadQueue = [ ]
2022-03-21 19:24:38 -05:00
this . currentDownload = null
2022-03-26 11:59:34 -05:00
2022-08-19 18:41:58 -05:00
this . failedCheckMap = { }
this . MaxFailedEpisodeChecks = 24
2022-03-26 11:59:34 -05:00
}
2022-03-26 19:58:59 -05:00
get serverSettings ( ) {
return this . db . serverSettings || { }
}
2022-04-23 19:41:06 -05:00
getEpisodeDownloadsInQueue ( libraryItemId ) {
return this . downloadQueue . filter ( d => d . libraryItemId === libraryItemId )
}
clearDownloadQueue ( libraryItemId = null ) {
if ( ! this . downloadQueue . length ) return
if ( ! libraryItemId ) {
Logger . info ( ` [PodcastManager] Clearing all downloads in queue ( ${ this . downloadQueue . length } ) ` )
this . downloadQueue = [ ]
} else {
var itemDownloads = this . getEpisodeDownloadsInQueue ( libraryItemId )
Logger . info ( ` [PodcastManager] Clearing downloads in queue for item " ${ libraryItemId } " ( ${ itemDownloads . length } ) ` )
this . downloadQueue = this . downloadQueue . filter ( d => d . libraryItemId !== libraryItemId )
}
}
2022-08-15 17:35:13 -05:00
async downloadPodcastEpisodes ( libraryItem , episodesToDownload , isAutoDownload ) {
2022-03-26 19:58:59 -05:00
var index = libraryItem . media . episodes . length + 1
2022-03-21 19:24:38 -05:00
episodesToDownload . forEach ( ( ep ) => {
var newPe = new PodcastEpisode ( )
newPe . setData ( ep , index ++ )
2022-04-05 19:40:40 -05:00
newPe . libraryItemId = libraryItem . id
2022-03-21 19:24:38 -05:00
var newPeDl = new PodcastEpisodeDownload ( )
2023-02-27 02:56:07 +00:00
newPeDl . setData ( newPe , libraryItem , isAutoDownload , libraryItem . libraryId )
2022-03-21 19:24:38 -05:00
this . startPodcastEpisodeDownload ( newPeDl )
} )
2022-03-20 16:41:06 -05:00
}
2022-03-21 19:24:38 -05:00
async startPodcastEpisodeDownload ( podcastEpisodeDownload ) {
2023-02-27 02:56:07 +00:00
SocketAuthority . emitter ( 'download_queue_updated' , this . getDownloadQueueDetails ( ) )
2022-03-21 19:24:38 -05:00
if ( this . currentDownload ) {
this . downloadQueue . push ( podcastEpisodeDownload )
2022-11-24 15:53:58 -06:00
SocketAuthority . emitter ( 'episode_download_queued' , podcastEpisodeDownload . toJSONForClient ( ) )
2022-03-21 19:24:38 -05:00
return
}
2022-04-23 19:41:06 -05:00
2022-11-24 15:53:58 -06:00
SocketAuthority . emitter ( 'episode_download_started' , podcastEpisodeDownload . toJSONForClient ( ) )
2022-03-21 19:24:38 -05:00
this . currentDownload = podcastEpisodeDownload
// Ignores all added files to this dir
this . watcher . addIgnoreDir ( this . currentDownload . libraryItem . path )
2022-09-30 16:55:31 -05:00
// Make sure podcast library item folder exists
if ( ! ( await fs . pathExists ( this . currentDownload . libraryItem . path ) ) ) {
Logger . warn ( ` [PodcastManager] Podcast episode download: Podcast folder no longer exists at " ${ this . currentDownload . libraryItem . path } " - Creating it ` )
await fs . mkdir ( this . currentDownload . libraryItem . path )
await filePerms . setDefault ( this . currentDownload . libraryItem . path )
}
2022-03-21 19:24:38 -05:00
var success = await downloadFile ( this . currentDownload . url , this . currentDownload . targetPath ) . then ( ( ) => true ) . catch ( ( error ) => {
Logger . error ( ` [PodcastManager] Podcast Episode download failed ` , error )
return false
} )
if ( success ) {
success = await this . scanAddPodcastEpisodeAudioFile ( )
if ( ! success ) {
await fs . remove ( this . currentDownload . targetPath )
2022-04-23 19:41:06 -05:00
this . currentDownload . setFinished ( false )
2022-03-21 19:24:38 -05:00
} else {
Logger . info ( ` [PodcastManager] Successfully downloaded podcast episode " ${ this . currentDownload . podcastEpisode . title } " ` )
2022-04-23 19:41:06 -05:00
this . currentDownload . setFinished ( true )
2022-03-21 19:24:38 -05:00
}
2022-04-23 19:41:06 -05:00
} else {
this . currentDownload . setFinished ( false )
2022-03-21 19:24:38 -05:00
}
2022-11-24 15:53:58 -06:00
SocketAuthority . emitter ( 'episode_download_finished' , this . currentDownload . toJSONForClient ( ) )
2023-02-27 02:56:07 +00:00
SocketAuthority . emitter ( 'download_queue_updated' , this . getDownloadQueueDetails ( ) )
2022-04-23 19:41:06 -05:00
2022-03-21 19:24:38 -05:00
this . watcher . removeIgnoreDir ( this . currentDownload . libraryItem . path )
this . currentDownload = null
if ( this . downloadQueue . length ) {
this . startPodcastEpisodeDownload ( this . downloadQueue . shift ( ) )
}
}
async scanAddPodcastEpisodeAudioFile ( ) {
var libraryFile = await this . getLibraryFile ( this . currentDownload . targetPath , this . currentDownload . targetRelPath )
2022-04-12 17:32:27 -05:00
// TODO: Set meta tags on new audio file
2022-03-21 19:24:38 -05:00
var audioFile = await this . probeAudioFile ( libraryFile )
if ( ! audioFile ) {
return false
}
var libraryItem = this . db . libraryItems . find ( li => li . id === this . currentDownload . libraryItem . id )
if ( ! libraryItem ) {
Logger . error ( ` [PodcastManager] Podcast Episode finished but library item was not found ${ this . currentDownload . libraryItem . id } ` )
return false
}
2022-03-26 18:23:33 -05:00
2022-03-21 19:24:38 -05:00
var podcastEpisode = this . currentDownload . podcastEpisode
podcastEpisode . audioFile = audioFile
libraryItem . media . addPodcastEpisode ( podcastEpisode )
2022-04-24 17:03:43 -05:00
if ( libraryItem . isInvalid ) {
// First episode added to an empty podcast
libraryItem . isInvalid = false
}
2022-03-26 18:23:33 -05:00
libraryItem . libraryFiles . push ( libraryFile )
2022-08-15 17:35:13 -05:00
2022-09-20 18:08:41 -05:00
if ( this . currentDownload . isAutoDownload ) {
// Check setting maxEpisodesToKeep and remove episode if necessary
2022-08-15 17:35:13 -05:00
if ( libraryItem . media . maxEpisodesToKeep && libraryItem . media . episodesWithPubDate . length > libraryItem . media . maxEpisodesToKeep ) {
Logger . info ( ` [PodcastManager] # of episodes ( ${ libraryItem . media . episodesWithPubDate . length } ) exceeds max episodes to keep ( ${ libraryItem . media . maxEpisodesToKeep } ) ` )
await this . removeOldestEpisode ( libraryItem , podcastEpisode . id )
}
}
2022-03-21 19:24:38 -05:00
libraryItem . updatedAt = Date . now ( )
await this . db . updateLibraryItem ( libraryItem )
2022-11-24 15:53:58 -06:00
SocketAuthority . emitter ( 'item_updated' , libraryItem . toJSONExpanded ( ) )
2022-09-20 18:08:41 -05:00
if ( this . currentDownload . isAutoDownload ) { // Notifications only for auto downloaded episodes
2022-09-21 18:01:10 -05:00
this . notificationManager . onPodcastEpisodeDownloaded ( libraryItem , podcastEpisode )
2022-09-20 18:08:41 -05:00
}
2022-03-21 19:24:38 -05:00
return true
}
2022-08-15 17:35:13 -05:00
async removeOldestEpisode ( libraryItem , episodeIdJustDownloaded ) {
var smallestPublishedAt = 0
var oldestEpisode = null
libraryItem . media . episodesWithPubDate . filter ( ep => ep . id !== episodeIdJustDownloaded ) . forEach ( ( ep ) => {
if ( ! smallestPublishedAt || ep . publishedAt < smallestPublishedAt ) {
smallestPublishedAt = ep . publishedAt
oldestEpisode = ep
}
} )
// TODO: Should we check for open playback sessions for this episode?
// TODO: remove all user progress for this episode
if ( oldestEpisode && oldestEpisode . audioFile ) {
Logger . info ( ` [PodcastManager] Deleting oldest episode " ${ oldestEpisode . title } " ` )
const successfullyDeleted = await removeFile ( oldestEpisode . audioFile . metadata . path )
if ( successfullyDeleted ) {
libraryItem . media . removeEpisode ( oldestEpisode . id )
libraryItem . removeLibraryFile ( oldestEpisode . audioFile . ino )
return true
} else {
Logger . warn ( ` [PodcastManager] Failed to remove oldest episode " ${ oldestEpisode . title } " ` )
}
}
return false
}
2022-03-21 19:24:38 -05:00
async getLibraryFile ( path , relPath ) {
var newLibFile = new LibraryFile ( )
await newLibFile . setDataFromPath ( path , relPath )
return newLibFile
}
2022-03-20 16:41:06 -05:00
2022-03-21 19:24:38 -05:00
async probeAudioFile ( libraryFile ) {
var path = libraryFile . metadata . path
2022-05-30 19:26:53 -05:00
var mediaProbeData = await prober . probe ( path )
if ( mediaProbeData . error ) {
Logger . error ( ` [PodcastManager] Podcast Episode downloaded but failed to probe " ${ path } " ` , mediaProbeData . error )
2022-03-21 19:24:38 -05:00
return false
}
var newAudioFile = new AudioFile ( )
2022-05-30 19:26:53 -05:00
newAudioFile . setDataFromProbe ( libraryFile , mediaProbeData )
2022-03-21 19:24:38 -05:00
return newAudioFile
2022-03-20 16:41:06 -05:00
}
2022-03-26 11:59:34 -05:00
2022-08-19 18:41:58 -05:00
// Returns false if auto download episodes was disabled (disabled if reaches max failed checks)
async runEpisodeCheck ( libraryItem ) {
const lastEpisodeCheckDate = new Date ( libraryItem . media . lastEpisodeCheck || 0 )
const latestEpisodePublishedAt = libraryItem . media . latestEpisodePublished
Logger . info ( ` [PodcastManager] runEpisodeCheck: " ${ libraryItem . media . metadata . title } " | Last check: ${ lastEpisodeCheckDate } | ${ latestEpisodePublishedAt ? ` Latest episode pubDate: ${ new Date ( latestEpisodePublishedAt ) } ` : 'No latest episode' } ` )
// Use latest episode pubDate if exists OR fallback to using lastEpisodeCheckDate
// lastEpisodeCheckDate will be the current time when adding a new podcast
const dateToCheckForEpisodesAfter = latestEpisodePublishedAt || lastEpisodeCheckDate
Logger . debug ( ` [PodcastManager] runEpisodeCheck: " ${ libraryItem . media . metadata . title } " checking for episodes after ${ new Date ( dateToCheckForEpisodesAfter ) } ` )
2022-10-26 16:55:16 -05:00
var newEpisodes = await this . checkPodcastForNewEpisodes ( libraryItem , dateToCheckForEpisodesAfter , libraryItem . media . maxNewEpisodesToDownload )
2022-08-19 18:41:58 -05:00
Logger . debug ( ` [PodcastManager] runEpisodeCheck: ${ newEpisodes ? newEpisodes . length : 'N/A' } episodes found ` )
if ( ! newEpisodes ) { // Failed
// Allow up to MaxFailedEpisodeChecks failed attempts before disabling auto download
if ( ! this . failedCheckMap [ libraryItem . id ] ) this . failedCheckMap [ libraryItem . id ] = 0
this . failedCheckMap [ libraryItem . id ] ++
if ( this . failedCheckMap [ libraryItem . id ] >= this . MaxFailedEpisodeChecks ) {
Logger . error ( ` [PodcastManager] runEpisodeCheck ${ this . failedCheckMap [ libraryItem . id ] } failed attempts at checking episodes for " ${ libraryItem . media . metadata . title } " - disabling auto download ` )
libraryItem . media . autoDownloadEpisodes = false
2022-05-01 19:54:33 -05:00
delete this . failedCheckMap [ libraryItem . id ]
2022-04-14 10:15:42 -05:00
} else {
2022-08-19 18:41:58 -05:00
Logger . warn ( ` [PodcastManager] runEpisodeCheck ${ this . failedCheckMap [ libraryItem . id ] } failed attempts at checking episodes for " ${ libraryItem . media . metadata . title } " ` )
2022-03-26 19:58:59 -05:00
}
2022-08-19 18:41:58 -05:00
} else if ( newEpisodes . length ) {
delete this . failedCheckMap [ libraryItem . id ]
Logger . info ( ` [PodcastManager] Found ${ newEpisodes . length } new episodes for podcast " ${ libraryItem . media . metadata . title } " - starting download ` )
this . downloadPodcastEpisodes ( libraryItem , newEpisodes , true )
} else {
delete this . failedCheckMap [ libraryItem . id ]
Logger . debug ( ` [PodcastManager] No new episodes for " ${ libraryItem . media . metadata . title } " ` )
2022-03-26 11:59:34 -05:00
}
2022-08-19 18:41:58 -05:00
libraryItem . media . lastEpisodeCheck = Date . now ( )
libraryItem . updatedAt = Date . now ( )
await this . db . updateLibraryItem ( libraryItem )
2022-11-24 15:53:58 -06:00
SocketAuthority . emitter ( 'item_updated' , libraryItem . toJSONExpanded ( ) )
2022-08-19 18:41:58 -05:00
return libraryItem . media . autoDownloadEpisodes
2022-03-26 11:59:34 -05:00
}
2022-09-03 08:06:52 -05:00
async checkPodcastForNewEpisodes ( podcastLibraryItem , dateToCheckForEpisodesAfter , maxNewEpisodes = 3 ) {
2022-03-26 19:58:59 -05:00
if ( ! podcastLibraryItem . media . metadata . feedUrl ) {
2022-05-01 19:54:33 -05:00
Logger . error ( ` [PodcastManager] checkPodcastForNewEpisodes no feed url for ${ podcastLibraryItem . media . metadata . title } (ID: ${ podcastLibraryItem . id } ) ` )
2022-03-26 19:58:59 -05:00
return false
}
2022-09-15 18:35:56 -05:00
var feed = await getPodcastFeed ( podcastLibraryItem . media . metadata . feedUrl )
2022-03-26 19:58:59 -05:00
if ( ! feed || ! feed . episodes ) {
2022-05-01 19:54:33 -05:00
Logger . error ( ` [PodcastManager] checkPodcastForNewEpisodes invalid feed payload for ${ podcastLibraryItem . media . metadata . title } (ID: ${ podcastLibraryItem . id } ) ` , feed )
2022-03-26 19:58:59 -05:00
return false
}
2022-05-01 19:54:33 -05:00
2022-03-26 19:58:59 -05:00
// Filter new and not already has
2022-05-11 18:55:19 -05:00
var newEpisodes = feed . episodes . filter ( ep => ep . publishedAt > dateToCheckForEpisodesAfter && ! podcastLibraryItem . media . checkHasEpisodeByFeedUrl ( ep . enclosure . url ) )
2022-09-03 08:06:52 -05:00
if ( maxNewEpisodes > 0 ) {
newEpisodes = newEpisodes . slice ( 0 , maxNewEpisodes )
}
2022-04-29 16:42:40 -05:00
return newEpisodes
}
2022-09-03 08:06:52 -05:00
async checkAndDownloadNewEpisodes ( libraryItem , maxEpisodesToDownload ) {
2022-04-29 16:42:40 -05:00
const lastEpisodeCheckDate = new Date ( libraryItem . media . lastEpisodeCheck || 0 )
Logger . info ( ` [PodcastManager] checkAndDownloadNewEpisodes for " ${ libraryItem . media . metadata . title } " - Last episode check: ${ lastEpisodeCheckDate } ` )
2022-09-03 08:06:52 -05:00
var newEpisodes = await this . checkPodcastForNewEpisodes ( libraryItem , libraryItem . media . lastEpisodeCheck , maxEpisodesToDownload )
2022-04-29 16:42:40 -05:00
if ( newEpisodes . length ) {
Logger . info ( ` [PodcastManager] Found ${ newEpisodes . length } new episodes for podcast " ${ libraryItem . media . metadata . title } " - starting download ` )
2022-08-15 17:35:13 -05:00
this . downloadPodcastEpisodes ( libraryItem , newEpisodes , false )
2022-04-29 16:42:40 -05:00
} else {
Logger . info ( ` [PodcastManager] No new episodes found for podcast " ${ libraryItem . media . metadata . title } " ` )
}
libraryItem . media . lastEpisodeCheck = Date . now ( )
libraryItem . updatedAt = Date . now ( )
await this . db . updateLibraryItem ( libraryItem )
2022-11-24 15:53:58 -06:00
SocketAuthority . emitter ( 'item_updated' , libraryItem . toJSONExpanded ( ) )
2022-04-29 16:42:40 -05:00
2022-03-26 19:58:59 -05:00
return newEpisodes
}
2022-07-31 13:12:37 -05:00
async findEpisode ( rssFeedUrl , searchTitle ) {
2022-09-15 18:35:56 -05:00
const feed = await getPodcastFeed ( rssFeedUrl ) . catch ( ( ) => {
2022-07-31 13:12:37 -05:00
return null
} )
if ( ! feed || ! feed . episodes ) {
return null
}
const matches = [ ]
feed . episodes . forEach ( ep => {
if ( ! ep . title ) return
const epTitle = ep . title . toLowerCase ( ) . trim ( )
if ( epTitle === searchTitle ) {
matches . push ( {
episode : ep ,
levenshtein : 0
} )
} else {
const levenshtein = levenshteinDistance ( searchTitle , epTitle , true )
if ( levenshtein <= 6 && epTitle . length > levenshtein ) {
matches . push ( {
episode : ep ,
levenshtein
} )
}
}
} )
return matches . sort ( ( a , b ) => a . levenshtein - b . levenshtein )
}
2022-05-29 11:46:45 -05:00
async getOPMLFeeds ( opmlText ) {
var extractedFeeds = opmlParser . parse ( opmlText )
if ( ! extractedFeeds || ! extractedFeeds . length ) {
Logger . error ( '[PodcastManager] getOPMLFeeds: No RSS feeds found in OPML' )
return {
error : 'No RSS feeds found in OPML'
}
}
var rssFeedData = [ ]
for ( let feed of extractedFeeds ) {
2022-09-15 18:35:56 -05:00
var feedData = await getPodcastFeed ( feed . feedUrl , true )
2022-05-29 11:46:45 -05:00
if ( feedData ) {
feedData . metadata . feedUrl = feed . feedUrl
rssFeedData . push ( feedData )
}
}
return {
feeds : rssFeedData
}
}
2023-02-27 02:56:07 +00:00
getDownloadQueueDetails ( ) {
return this . downloadQueue . map ( item => {
return {
id : item . id ,
libraryId : item . libraryId || null ,
libraryItemId : item . libraryItemId || null ,
podcastTitle : item . libraryItem . media . metadata . title || null ,
podcastExplicit : item . libraryItem . media . metadata . explicit || false ,
episodeDisplayTitle : item . podcastEpisode . title || null ,
season : item . podcastEpisode . season || null ,
episode : item . podcastEpisode . episode || null ,
episodeType : item . podcastEpisode . episodeType || 'full' ,
publishedAt : item . podcastEpisode . publishedAt || null
}
} )
}
2022-03-20 16:41:06 -05:00
}
2023-02-27 02:56:07 +00:00
module . exports = PodcastManager