mirror of
https://github.com/advplyr/audiobookshelf.git
synced 2025-01-03 00:06:46 +01:00
Add:Podcast auto-download option to delete an episode if it exceeds X max episodes to keep #903
This commit is contained in:
parent
2c0c53bbf1
commit
7a69afdcd9
@ -1,8 +1,10 @@
|
|||||||
<template>
|
<template>
|
||||||
<div class="w-full">
|
<div class="w-full">
|
||||||
|
<slot>
|
||||||
<p class="px-1 text-sm font-semibold" :class="{ 'text-gray-400': disabled }">
|
<p class="px-1 text-sm font-semibold" :class="{ 'text-gray-400': disabled }">
|
||||||
{{ label }}<em v-if="note" class="font-normal text-xs pl-2">{{ note }}</em>
|
{{ label }}<em v-if="note" class="font-normal text-xs pl-2">{{ note }}</em>
|
||||||
</p>
|
</p>
|
||||||
|
</slot>
|
||||||
<ui-text-input ref="input" v-model="inputValue" :disabled="disabled" :readonly="readonly" :type="type" class="w-full" @blur="inputBlurred" />
|
<ui-text-input ref="input" v-model="inputValue" :disabled="disabled" :readonly="readonly" :type="type" class="w-full" @blur="inputBlurred" />
|
||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
|
@ -40,9 +40,21 @@
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div class="flex-grow px-1 pt-6">
|
<div class="flex items-center px-1 pt-6">
|
||||||
|
<div class="w-1/2 px-1 py-5">
|
||||||
<ui-checkbox v-model="autoDownloadEpisodes" label="Auto Download New Episodes" checkbox-bg="primary" border-color="gray-600" label-class="pl-2 text-base font-semibold" />
|
<ui-checkbox v-model="autoDownloadEpisodes" label="Auto Download New Episodes" checkbox-bg="primary" border-color="gray-600" label-class="pl-2 text-base font-semibold" />
|
||||||
</div>
|
</div>
|
||||||
|
<div v-if="autoDownloadEpisodes" class="w-1/2 px-1">
|
||||||
|
<ui-text-input-with-label ref="maxEpisodesToKeep" v-model="maxEpisodesToKeep" type="number" class="max-w-48">
|
||||||
|
<ui-tooltip direction="bottom" text="Value of 0 sets no max limit. After a new episode is auto-downloaded this will delete the oldest episode if you have more than X episodes.">
|
||||||
|
<p class="text-sm">
|
||||||
|
Max episodes to keep
|
||||||
|
<span class="material-icons icon-text text-sm">info_outlined</span>
|
||||||
|
</p>
|
||||||
|
</ui-tooltip>
|
||||||
|
</ui-text-input-with-label>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
</form>
|
</form>
|
||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
@ -72,6 +84,7 @@ export default {
|
|||||||
language: null
|
language: null
|
||||||
},
|
},
|
||||||
autoDownloadEpisodes: false,
|
autoDownloadEpisodes: false,
|
||||||
|
maxEpisodesToKeep: 0,
|
||||||
newTags: []
|
newTags: []
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
@ -199,6 +212,9 @@ export default {
|
|||||||
if (this.media.autoDownloadEpisodes !== this.autoDownloadEpisodes) {
|
if (this.media.autoDownloadEpisodes !== this.autoDownloadEpisodes) {
|
||||||
updatePayload.autoDownloadEpisodes = !!this.autoDownloadEpisodes
|
updatePayload.autoDownloadEpisodes = !!this.autoDownloadEpisodes
|
||||||
}
|
}
|
||||||
|
if (this.autoDownloadEpisodes && !isNaN(this.maxEpisodesToKeep) && Number(this.maxEpisodesToKeep) != this.media.maxEpisodesToKeep) {
|
||||||
|
updatePayload.maxEpisodesToKeep = Number(this.maxEpisodesToKeep)
|
||||||
|
}
|
||||||
|
|
||||||
return {
|
return {
|
||||||
updatePayload,
|
updatePayload,
|
||||||
@ -220,6 +236,7 @@ export default {
|
|||||||
this.details.explicit = !!this.mediaMetadata.explicit
|
this.details.explicit = !!this.mediaMetadata.explicit
|
||||||
|
|
||||||
this.autoDownloadEpisodes = !!this.media.autoDownloadEpisodes
|
this.autoDownloadEpisodes = !!this.media.autoDownloadEpisodes
|
||||||
|
this.maxEpisodesToKeep = this.media.maxEpisodesToKeep || 0
|
||||||
this.newTags = [...(this.media.tags || [])]
|
this.newTags = [...(this.media.tags || [])]
|
||||||
},
|
},
|
||||||
submitForm() {
|
submitForm() {
|
||||||
|
@ -5,7 +5,7 @@ const axios = require('axios')
|
|||||||
const { parsePodcastRssFeedXml } = require('../utils/podcastUtils')
|
const { parsePodcastRssFeedXml } = require('../utils/podcastUtils')
|
||||||
const Logger = require('../Logger')
|
const Logger = require('../Logger')
|
||||||
|
|
||||||
const { downloadFile } = require('../utils/fileUtils')
|
const { downloadFile, removeFile } = require('../utils/fileUtils')
|
||||||
const { levenshteinDistance } = require('../utils/index')
|
const { levenshteinDistance } = require('../utils/index')
|
||||||
const opmlParser = require('../utils/parsers/parseOPML')
|
const opmlParser = require('../utils/parsers/parseOPML')
|
||||||
const prober = require('../utils/prober')
|
const prober = require('../utils/prober')
|
||||||
@ -56,14 +56,14 @@ class PodcastManager {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
async downloadPodcastEpisodes(libraryItem, episodesToDownload) {
|
async downloadPodcastEpisodes(libraryItem, episodesToDownload, isAutoDownload) {
|
||||||
var index = libraryItem.media.episodes.length + 1
|
var index = libraryItem.media.episodes.length + 1
|
||||||
episodesToDownload.forEach((ep) => {
|
episodesToDownload.forEach((ep) => {
|
||||||
var newPe = new PodcastEpisode()
|
var newPe = new PodcastEpisode()
|
||||||
newPe.setData(ep, index++)
|
newPe.setData(ep, index++)
|
||||||
newPe.libraryItemId = libraryItem.id
|
newPe.libraryItemId = libraryItem.id
|
||||||
var newPeDl = new PodcastEpisodeDownload()
|
var newPeDl = new PodcastEpisodeDownload()
|
||||||
newPeDl.setData(newPe, libraryItem)
|
newPeDl.setData(newPe, libraryItem, isAutoDownload)
|
||||||
this.startPodcastEpisodeDownload(newPeDl)
|
this.startPodcastEpisodeDownload(newPeDl)
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
@ -131,12 +131,46 @@ class PodcastManager {
|
|||||||
libraryItem.isInvalid = false
|
libraryItem.isInvalid = false
|
||||||
}
|
}
|
||||||
libraryItem.libraryFiles.push(libraryFile)
|
libraryItem.libraryFiles.push(libraryFile)
|
||||||
|
|
||||||
|
// Check setting maxEpisodesToKeep and remove episode if necessary
|
||||||
|
if (this.currentDownload.isAutoDownload) { // only applies for auto-downloaded episodes
|
||||||
|
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)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
libraryItem.updatedAt = Date.now()
|
libraryItem.updatedAt = Date.now()
|
||||||
await this.db.updateLibraryItem(libraryItem)
|
await this.db.updateLibraryItem(libraryItem)
|
||||||
this.emitter('item_updated', libraryItem.toJSONExpanded())
|
this.emitter('item_updated', libraryItem.toJSONExpanded())
|
||||||
return true
|
return true
|
||||||
}
|
}
|
||||||
|
|
||||||
|
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
|
||||||
|
}
|
||||||
|
|
||||||
async getLibraryFile(path, relPath) {
|
async getLibraryFile(path, relPath) {
|
||||||
var newLibFile = new LibraryFile()
|
var newLibFile = new LibraryFile()
|
||||||
await newLibFile.setDataFromPath(path, relPath)
|
await newLibFile.setDataFromPath(path, relPath)
|
||||||
@ -211,7 +245,7 @@ class PodcastManager {
|
|||||||
} else if (newEpisodes.length) {
|
} else if (newEpisodes.length) {
|
||||||
delete this.failedCheckMap[libraryItem.id]
|
delete this.failedCheckMap[libraryItem.id]
|
||||||
Logger.info(`[PodcastManager] Found ${newEpisodes.length} new episodes for podcast "${libraryItem.media.metadata.title}" - starting download`)
|
Logger.info(`[PodcastManager] Found ${newEpisodes.length} new episodes for podcast "${libraryItem.media.metadata.title}" - starting download`)
|
||||||
this.downloadPodcastEpisodes(libraryItem, newEpisodes)
|
this.downloadPodcastEpisodes(libraryItem, newEpisodes, true)
|
||||||
} else {
|
} else {
|
||||||
delete this.failedCheckMap[libraryItem.id]
|
delete this.failedCheckMap[libraryItem.id]
|
||||||
Logger.debug(`[PodcastManager] No new episodes for "${libraryItem.media.metadata.title}"`)
|
Logger.debug(`[PodcastManager] No new episodes for "${libraryItem.media.metadata.title}"`)
|
||||||
@ -248,7 +282,7 @@ class PodcastManager {
|
|||||||
var newEpisodes = await this.checkPodcastForNewEpisodes(libraryItem, libraryItem.media.lastEpisodeCheck)
|
var newEpisodes = await this.checkPodcastForNewEpisodes(libraryItem, libraryItem.media.lastEpisodeCheck)
|
||||||
if (newEpisodes.length) {
|
if (newEpisodes.length) {
|
||||||
Logger.info(`[PodcastManager] Found ${newEpisodes.length} new episodes for podcast "${libraryItem.media.metadata.title}" - starting download`)
|
Logger.info(`[PodcastManager] Found ${newEpisodes.length} new episodes for podcast "${libraryItem.media.metadata.title}" - starting download`)
|
||||||
this.downloadPodcastEpisodes(libraryItem, newEpisodes)
|
this.downloadPodcastEpisodes(libraryItem, newEpisodes, false)
|
||||||
} else {
|
} else {
|
||||||
Logger.info(`[PodcastManager] No new episodes found for podcast "${libraryItem.media.metadata.title}"`)
|
Logger.info(`[PodcastManager] No new episodes found for podcast "${libraryItem.media.metadata.title}"`)
|
||||||
}
|
}
|
||||||
|
@ -9,6 +9,7 @@ class PodcastEpisodeDownload {
|
|||||||
this.url = null
|
this.url = null
|
||||||
this.libraryItem = null
|
this.libraryItem = null
|
||||||
|
|
||||||
|
this.isAutoDownload = false
|
||||||
this.isDownloading = false
|
this.isDownloading = false
|
||||||
this.isFinished = false
|
this.isFinished = false
|
||||||
this.failed = false
|
this.failed = false
|
||||||
@ -46,11 +47,12 @@ class PodcastEpisodeDownload {
|
|||||||
return this.libraryItem ? this.libraryItem.id : null
|
return this.libraryItem ? this.libraryItem.id : null
|
||||||
}
|
}
|
||||||
|
|
||||||
setData(podcastEpisode, libraryItem) {
|
setData(podcastEpisode, libraryItem, isAutoDownload) {
|
||||||
this.id = getId('epdl')
|
this.id = getId('epdl')
|
||||||
this.podcastEpisode = podcastEpisode
|
this.podcastEpisode = podcastEpisode
|
||||||
this.url = podcastEpisode.enclosure.url
|
this.url = podcastEpisode.enclosure.url
|
||||||
this.libraryItem = libraryItem
|
this.libraryItem = libraryItem
|
||||||
|
this.isAutoDownload = isAutoDownload
|
||||||
this.createdAt = Date.now()
|
this.createdAt = Date.now()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -19,6 +19,7 @@ class Podcast {
|
|||||||
|
|
||||||
this.autoDownloadEpisodes = false
|
this.autoDownloadEpisodes = false
|
||||||
this.lastEpisodeCheck = 0
|
this.lastEpisodeCheck = 0
|
||||||
|
this.maxEpisodesToKeep = 0
|
||||||
|
|
||||||
this.lastCoverSearch = null
|
this.lastCoverSearch = null
|
||||||
this.lastCoverSearchQuery = null
|
this.lastCoverSearchQuery = null
|
||||||
@ -40,6 +41,7 @@ class Podcast {
|
|||||||
})
|
})
|
||||||
this.autoDownloadEpisodes = !!podcast.autoDownloadEpisodes
|
this.autoDownloadEpisodes = !!podcast.autoDownloadEpisodes
|
||||||
this.lastEpisodeCheck = podcast.lastEpisodeCheck || 0
|
this.lastEpisodeCheck = podcast.lastEpisodeCheck || 0
|
||||||
|
this.maxEpisodesToKeep = podcast.maxEpisodesToKeep || 0
|
||||||
}
|
}
|
||||||
|
|
||||||
toJSON() {
|
toJSON() {
|
||||||
@ -50,7 +52,8 @@ class Podcast {
|
|||||||
tags: [...this.tags],
|
tags: [...this.tags],
|
||||||
episodes: this.episodes.map(e => e.toJSON()),
|
episodes: this.episodes.map(e => e.toJSON()),
|
||||||
autoDownloadEpisodes: this.autoDownloadEpisodes,
|
autoDownloadEpisodes: this.autoDownloadEpisodes,
|
||||||
lastEpisodeCheck: this.lastEpisodeCheck
|
lastEpisodeCheck: this.lastEpisodeCheck,
|
||||||
|
maxEpisodesToKeep: this.maxEpisodesToKeep
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -62,6 +65,7 @@ class Podcast {
|
|||||||
numEpisodes: this.episodes.length,
|
numEpisodes: this.episodes.length,
|
||||||
autoDownloadEpisodes: this.autoDownloadEpisodes,
|
autoDownloadEpisodes: this.autoDownloadEpisodes,
|
||||||
lastEpisodeCheck: this.lastEpisodeCheck,
|
lastEpisodeCheck: this.lastEpisodeCheck,
|
||||||
|
maxEpisodesToKeep: this.maxEpisodesToKeep,
|
||||||
size: this.size
|
size: this.size
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -75,6 +79,7 @@ class Podcast {
|
|||||||
episodes: this.episodes.map(e => e.toJSONExpanded()),
|
episodes: this.episodes.map(e => e.toJSONExpanded()),
|
||||||
autoDownloadEpisodes: this.autoDownloadEpisodes,
|
autoDownloadEpisodes: this.autoDownloadEpisodes,
|
||||||
lastEpisodeCheck: this.lastEpisodeCheck,
|
lastEpisodeCheck: this.lastEpisodeCheck,
|
||||||
|
maxEpisodesToKeep: this.maxEpisodesToKeep,
|
||||||
size: this.size
|
size: this.size
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -113,6 +118,9 @@ class Podcast {
|
|||||||
})
|
})
|
||||||
return largestPublishedAt
|
return largestPublishedAt
|
||||||
}
|
}
|
||||||
|
get episodesWithPubDate() {
|
||||||
|
return this.episodes.filter(ep => !!ep.publishedAt)
|
||||||
|
}
|
||||||
|
|
||||||
update(payload) {
|
update(payload) {
|
||||||
var json = this.toJSON()
|
var json = this.toJSON()
|
||||||
|
@ -16,7 +16,7 @@ async function getFileStat(path) {
|
|||||||
birthtime: stat.birthtime
|
birthtime: stat.birthtime
|
||||||
}
|
}
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
console.error('Failed to stat', err)
|
Logger.error('[fileUtils] Failed to stat', err)
|
||||||
return false
|
return false
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -33,7 +33,7 @@ async function getFileTimestampsWithIno(path) {
|
|||||||
ino: String(stat.ino)
|
ino: String(stat.ino)
|
||||||
}
|
}
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
console.error('Failed to getFileTimestampsWithIno', err)
|
Logger.error('[fileUtils] Failed to getFileTimestampsWithIno', err)
|
||||||
return false
|
return false
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -220,3 +220,11 @@ module.exports.getAudioMimeTypeFromExtname = (extname) => {
|
|||||||
if (AudioMimeType[formatUpper]) return AudioMimeType[formatUpper]
|
if (AudioMimeType[formatUpper]) return AudioMimeType[formatUpper]
|
||||||
return null
|
return null
|
||||||
}
|
}
|
||||||
|
|
||||||
|
module.exports.removeFile = (path) => {
|
||||||
|
if (!path) return false
|
||||||
|
return fs.remove(path).then(() => true).catch((error) => {
|
||||||
|
Logger.error(`[fileUtils] Failed remove file "${path}"`, error)
|
||||||
|
return false
|
||||||
|
})
|
||||||
|
}
|
Loading…
Reference in New Issue
Block a user