mirror of
https://github.com/advplyr/audiobookshelf.git
synced 2025-01-18 20:08:41 +01:00
Add back in m4b merge downloader in experimental #478
This commit is contained in:
parent
f781fa9e6b
commit
ad3fbe7abf
@ -52,7 +52,7 @@
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div v-if="userCanUpdate" v-show="!isSelectionMode" class="absolute cursor-pointer hover:text-yellow-300 hover:scale-125 transform duration-50" :style="{ top: 0.375 * sizeMultiplier + 'rem', right: 0.375 * sizeMultiplier + 'rem' }" @click.stop.prevent="editClick">
|
<div v-if="userCanUpdate" v-show="!isSelectionMode" class="absolute cursor-pointer hover:text-yellow-300 hover:scale-125 transform duration-50 top-0 right-0" :style="{ padding: 0.375 * sizeMultiplier + 'rem' }" @click.stop.prevent="editClick">
|
||||||
<span class="material-icons" :style="{ fontSize: sizeMultiplier + 'rem' }">edit</span>
|
<span class="material-icons" :style="{ fontSize: sizeMultiplier + 'rem' }">edit</span>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
@ -337,12 +337,6 @@ export default {
|
|||||||
text: 'Match'
|
text: 'Match'
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
if (this.userCanDownload && !this.isPodcast) {
|
|
||||||
items.push({
|
|
||||||
func: 'showEditModalDownload',
|
|
||||||
text: 'Download'
|
|
||||||
})
|
|
||||||
}
|
|
||||||
if (this.userIsRoot) {
|
if (this.userIsRoot) {
|
||||||
items.push({
|
items.push({
|
||||||
func: 'rescan',
|
func: 'rescan',
|
||||||
|
@ -30,7 +30,6 @@ export default {
|
|||||||
return {
|
return {
|
||||||
processing: false,
|
processing: false,
|
||||||
libraryItem: null,
|
libraryItem: null,
|
||||||
|
|
||||||
tabs: [
|
tabs: [
|
||||||
{
|
{
|
||||||
id: 'details',
|
id: 'details',
|
||||||
@ -57,15 +56,16 @@ export default {
|
|||||||
title: 'Files',
|
title: 'Files',
|
||||||
component: 'modals-item-tabs-files'
|
component: 'modals-item-tabs-files'
|
||||||
},
|
},
|
||||||
// {
|
|
||||||
// id: 'download',
|
|
||||||
// title: 'Download',
|
|
||||||
// component: 'modals-item-tabs-download'
|
|
||||||
// },
|
|
||||||
{
|
{
|
||||||
id: 'match',
|
id: 'match',
|
||||||
title: 'Match',
|
title: 'Match',
|
||||||
component: 'modals-item-tabs-match'
|
component: 'modals-item-tabs-match'
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: 'merge',
|
||||||
|
title: 'Merge',
|
||||||
|
component: 'modals-item-tabs-merge',
|
||||||
|
experimental: true
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
}
|
}
|
||||||
@ -110,6 +110,9 @@ export default {
|
|||||||
this.$store.commit('setEditModalTab', val)
|
this.$store.commit('setEditModalTab', val)
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
showExperimentalFeatures() {
|
||||||
|
return this.$store.state.showExperimentalFeatures
|
||||||
|
},
|
||||||
userCanUpdate() {
|
userCanUpdate() {
|
||||||
return this.$store.getters['user/getUserCanUpdate']
|
return this.$store.getters['user/getUserCanUpdate']
|
||||||
},
|
},
|
||||||
@ -119,12 +122,13 @@ export default {
|
|||||||
availableTabs() {
|
availableTabs() {
|
||||||
if (!this.userCanUpdate && !this.userCanDownload) return []
|
if (!this.userCanUpdate && !this.userCanDownload) return []
|
||||||
return this.tabs.filter((tab) => {
|
return this.tabs.filter((tab) => {
|
||||||
if (tab.id === 'download' && this.isMissing) return false
|
if (tab.experimental && !this.showExperimentalFeatures) return false
|
||||||
|
if (tab.id === 'merge' && (this.isMissing || this.mediaType !== 'book')) return false
|
||||||
if (this.mediaType == 'podcast' && tab.id == 'chapters') return false
|
if (this.mediaType == 'podcast' && tab.id == 'chapters') return false
|
||||||
if (this.mediaType == 'book' && tab.id == 'episodes') return false
|
if (this.mediaType == 'book' && tab.id == 'episodes') return false
|
||||||
|
|
||||||
if ((tab.id === 'download' || tab.id === 'files') && this.userCanDownload) return true
|
if ((tab.id === 'merge' || tab.id === 'files') && this.userCanDownload) return true
|
||||||
if (tab.id !== 'download' && tab.id !== 'files' && this.userCanUpdate) return true
|
if (tab.id !== 'merge' && tab.id !== 'files' && this.userCanUpdate) return true
|
||||||
if (tab.id === 'match' && this.userCanUpdate) return true
|
if (tab.id === 'match' && this.userCanUpdate) return true
|
||||||
return false
|
return false
|
||||||
})
|
})
|
||||||
|
@ -1,212 +0,0 @@
|
|||||||
<template>
|
|
||||||
<div class="w-full h-full overflow-hidden overflow-y-auto px-4 py-6">
|
|
||||||
<p class="text-center text-lg mb-4 py-8">Preparing downloads can take several minutes and will be stored in <span class="bg-primary bg-opacity-75 font-mono p-1 text-base">/metadata/downloads</span>. After the download is ready, it will remain available for 60 minutes, then be deleted.<br />Download will timeout after 15 minutes.</p>
|
|
||||||
<div v-if="showM4bDownload" class="w-full border border-black-200 p-4 my-4">
|
|
||||||
<div class="flex items-center">
|
|
||||||
<div>
|
|
||||||
<p class="text-lg">M4B Audiobook File <span class="text-error">*</span></p>
|
|
||||||
<p class="max-w-xs text-sm pt-2 text-gray-300">Generate a .M4B audiobook file with embedded cover image and chapters.</p>
|
|
||||||
</div>
|
|
||||||
<div class="flex-grow" />
|
|
||||||
<div>
|
|
||||||
<p v-if="singleDownloadStatus === $constants.DownloadStatus.FAILED" class="text-error mb-2">Download Failed</p>
|
|
||||||
<p v-if="singleDownloadStatus === $constants.DownloadStatus.READY" class="text-success mb-2">Download Ready!</p>
|
|
||||||
<p v-if="singleDownloadStatus === $constants.DownloadStatus.EXPIRED" class="text-error mb-2">Download Expired</p>
|
|
||||||
|
|
||||||
<ui-btn v-if="singleDownloadStatus !== $constants.DownloadStatus.READY" :loading="singleDownloadStatus === $constants.DownloadStatus.PENDING" :disabled="tempDisable" @click="startSingleAudioDownload">Start Download</ui-btn>
|
|
||||||
<div v-else>
|
|
||||||
<ui-btn @click="downloadWithProgress(singleAudioDownload)">Download</ui-btn>
|
|
||||||
<p class="px-0.5 py-1 text-sm font-mono text-center">Size: {{ $bytesPretty(singleAudioDownload.size) }}</p>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<div class="w-full border border-black-200 p-4 my-4">
|
|
||||||
<div class="flex items-center">
|
|
||||||
<div>
|
|
||||||
<p v-if="totalFiles > 1" class="text-lg">Zip {{ totalFiles }} Files</p>
|
|
||||||
<p v-else>Zip 1 File</p>
|
|
||||||
<p class="max-w-xs text-sm pt-2 text-gray-300">Generate a .ZIP file from the contents of the audiobook directory.</p>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="flex-grow" />
|
|
||||||
<div>
|
|
||||||
<p v-if="zipDownloadStatus === $constants.DownloadStatus.FAILED" class="text-error mb-2">Download Failed</p>
|
|
||||||
<p v-if="zipDownloadStatus === $constants.DownloadStatus.READY" class="text-success mb-2">Download Ready!</p>
|
|
||||||
<p v-if="zipDownloadStatus === $constants.DownloadStatus.EXPIRED" class="text-error mb-2">Download Expired</p>
|
|
||||||
|
|
||||||
<ui-btn v-if="zipDownloadStatus !== $constants.DownloadStatus.READY" :loading="zipDownloadStatus === $constants.DownloadStatus.PENDING" :disabled="tempDisable" @click="startZipDownload">Start Download</ui-btn>
|
|
||||||
<div v-else>
|
|
||||||
<ui-btn @click="downloadWithProgress(zipDownload)">Download</ui-btn>
|
|
||||||
<p class="px-0.5 py-1 text-sm font-mono text-center">Size: {{ $bytesPretty(zipDownload.size) }}</p>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div v-if="showM4bDownload" class="w-full flex items-center justify-center absolute bottom-4 left-0 right-0 text-center">
|
|
||||||
<p class="text-error text-lg">* <strong>Experimental:</strong> Merging multiple .m4b files may have issues. <a href="https://github.com/advplyr/audiobookshelf/issues" class="underline text-blue-600" target="_blank">Report issues here.</a></p>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div v-if="isDownloading" class="absolute top-0 left-0 w-full h-full bg-black bg-opacity-50 z-50 flex items-center justify-center">
|
|
||||||
<div class="w-80 border border-black-400 bg-bg rounded-xl h-20">
|
|
||||||
<div class="w-full h-full flex items-center justify-center">
|
|
||||||
<p class="text-lg">Download.... {{ downloadPercent }}%</p>
|
|
||||||
<p class="w-24 font-mono pl-8 text-right">
|
|
||||||
{{ downloadAmount }}
|
|
||||||
</p>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</template>
|
|
||||||
|
|
||||||
<script>
|
|
||||||
export default {
|
|
||||||
props: {
|
|
||||||
processing: Boolean,
|
|
||||||
libraryItem: {
|
|
||||||
type: Object,
|
|
||||||
default: () => {}
|
|
||||||
}
|
|
||||||
},
|
|
||||||
data() {
|
|
||||||
return {
|
|
||||||
tempDisable: false,
|
|
||||||
isDownloading: false,
|
|
||||||
downloadPercent: '0',
|
|
||||||
downloadAmount: '0 KB'
|
|
||||||
}
|
|
||||||
},
|
|
||||||
watch: {
|
|
||||||
singleDownloadStatus(newVal) {
|
|
||||||
if (newVal) {
|
|
||||||
this.tempDisable = false
|
|
||||||
}
|
|
||||||
}
|
|
||||||
},
|
|
||||||
computed: {
|
|
||||||
libraryItemId() {
|
|
||||||
return this.libraryItem ? this.libraryItem.id : null
|
|
||||||
},
|
|
||||||
media() {
|
|
||||||
return this.libraryItem ? this.libraryItem.media || {} : {}
|
|
||||||
},
|
|
||||||
downloads() {
|
|
||||||
return this.$store.getters['downloads/getDownloads'](this.libraryItemId)
|
|
||||||
},
|
|
||||||
singleAudioDownload() {
|
|
||||||
return this.downloads.find((d) => d.type === 'singleAudio')
|
|
||||||
},
|
|
||||||
singleDownloadStatus() {
|
|
||||||
return this.singleAudioDownload ? this.singleAudioDownload.status : false
|
|
||||||
},
|
|
||||||
zipDownload() {
|
|
||||||
return this.downloads.find((d) => d.type === 'zip')
|
|
||||||
},
|
|
||||||
zipDownloadStatus() {
|
|
||||||
return this.zipDownload ? this.zipDownload.status : false
|
|
||||||
},
|
|
||||||
isSingleTrack() {
|
|
||||||
if (!this.libraryItem.tracks) return false
|
|
||||||
return this.libraryItem.tracks.length === 1
|
|
||||||
},
|
|
||||||
singleTrackPath() {
|
|
||||||
if (!this.isSingleTrack) return null
|
|
||||||
return this.audiobook.tracks[0].path
|
|
||||||
},
|
|
||||||
libraryFiles() {
|
|
||||||
return this.libraryItem.libraryFiles
|
|
||||||
},
|
|
||||||
totalFiles() {
|
|
||||||
return this.libraryFiles.length
|
|
||||||
},
|
|
||||||
showM4bDownload() {
|
|
||||||
return !this.libraryItem.isMissing && this.media.tracks.length
|
|
||||||
}
|
|
||||||
},
|
|
||||||
methods: {
|
|
||||||
startZipDownload() {
|
|
||||||
// console.log('Download request received', this.audiobook)
|
|
||||||
|
|
||||||
this.tempDisable = true
|
|
||||||
setTimeout(() => {
|
|
||||||
this.tempDisable = false
|
|
||||||
}, 1000)
|
|
||||||
|
|
||||||
var downloadPayload = {
|
|
||||||
audiobookId: this.audiobook.id,
|
|
||||||
type: 'zip'
|
|
||||||
}
|
|
||||||
this.$root.socket.emit('download', downloadPayload)
|
|
||||||
},
|
|
||||||
startSingleAudioDownload() {
|
|
||||||
// console.log('Download request received', this.audiobook)
|
|
||||||
|
|
||||||
this.tempDisable = true
|
|
||||||
setTimeout(() => {
|
|
||||||
this.tempDisable = false
|
|
||||||
}, 1000)
|
|
||||||
|
|
||||||
var downloadPayload = {
|
|
||||||
audiobookId: this.audiobook.id,
|
|
||||||
type: 'singleAudio',
|
|
||||||
includeMetadata: true,
|
|
||||||
includeCover: true
|
|
||||||
}
|
|
||||||
this.$root.socket.emit('download', downloadPayload)
|
|
||||||
},
|
|
||||||
downloadWithProgress(download) {
|
|
||||||
var downloadId = download.id
|
|
||||||
var downloadUrl = `${process.env.serverUrl}/api/download/${downloadId}`
|
|
||||||
var filename = download.filename
|
|
||||||
|
|
||||||
this.isDownloading = true
|
|
||||||
|
|
||||||
var request = new XMLHttpRequest()
|
|
||||||
request.responseType = 'blob'
|
|
||||||
request.open('get', downloadUrl, true)
|
|
||||||
request.setRequestHeader('Authorization', `Bearer ${this.$store.getters['user/getToken']}`)
|
|
||||||
request.send()
|
|
||||||
|
|
||||||
request.onreadystatechange = () => {
|
|
||||||
if (request.readyState === 4) {
|
|
||||||
this.isDownloading = false
|
|
||||||
}
|
|
||||||
if (request.readyState == 4 && request.status == 200) {
|
|
||||||
const url = window.URL.createObjectURL(request.response)
|
|
||||||
|
|
||||||
const anchor = document.createElement('a')
|
|
||||||
anchor.href = url
|
|
||||||
anchor.download = filename
|
|
||||||
document.body.appendChild(anchor)
|
|
||||||
anchor.click()
|
|
||||||
setTimeout(() => {
|
|
||||||
if (anchor) anchor.remove()
|
|
||||||
}, 1000)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
request.onerror = (err) => {
|
|
||||||
console.error('Download error', err)
|
|
||||||
this.isDownloading = false
|
|
||||||
}
|
|
||||||
|
|
||||||
request.onprogress = (e) => {
|
|
||||||
const percent_complete = Math.floor((e.loaded / e.total) * 100)
|
|
||||||
this.downloadAmount = this.$bytesPretty(e.loaded)
|
|
||||||
this.downloadPercent = percent_complete
|
|
||||||
|
|
||||||
// const duration = (new Date().getTime() - startTime) / 1000
|
|
||||||
// const bps = e.loaded / duration
|
|
||||||
// const kbps = Math.floor(bps / 1024)
|
|
||||||
// const time = (e.total - e.loaded) / bps
|
|
||||||
// const seconds = Math.floor(time % 60)
|
|
||||||
// const minutes = Math.floor(time / 60)
|
|
||||||
// console.log(`${percent_complete}% - ${kbps} Kbps - ${minutes} min ${seconds} sec remaining`)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
},
|
|
||||||
mounted() {}
|
|
||||||
}
|
|
||||||
</script>
|
|
214
client/components/modals/item/tabs/Merge.vue
Normal file
214
client/components/modals/item/tabs/Merge.vue
Normal file
@ -0,0 +1,214 @@
|
|||||||
|
<template>
|
||||||
|
<div class="w-full h-full overflow-hidden overflow-y-auto px-4 py-6">
|
||||||
|
<div v-if="showM4bDownload" class="w-full border border-black-200 p-4 my-8">
|
||||||
|
<div class="flex items-center">
|
||||||
|
<div>
|
||||||
|
<p class="text-lg">M4B Audiobook File <span class="text-error">*</span></p>
|
||||||
|
<p class="max-w-xs text-sm pt-2 text-gray-300">Generate a .M4B audiobook file with embedded cover image and chapters.</p>
|
||||||
|
</div>
|
||||||
|
<div class="flex-grow" />
|
||||||
|
<div>
|
||||||
|
<p v-if="abmergeStatus === $constants.DownloadStatus.FAILED" class="text-error mb-2">Download Failed</p>
|
||||||
|
<p v-if="abmergeStatus === $constants.DownloadStatus.READY" class="text-success mb-2">Download Ready!</p>
|
||||||
|
<p v-if="abmergeStatus === $constants.DownloadStatus.EXPIRED" class="text-error mb-2">Download Expired</p>
|
||||||
|
|
||||||
|
<ui-btn v-if="abmergeStatus !== $constants.DownloadStatus.READY" :loading="abmergeStatus === $constants.DownloadStatus.PENDING" :disabled="tempDisable" @click="startAudiobookMerge">Start Merge</ui-btn>
|
||||||
|
<div v-else>
|
||||||
|
<div class="flex">
|
||||||
|
<ui-btn @click="downloadWithProgress(abmergeDownload)">Download</ui-btn>
|
||||||
|
<ui-icon-btn small icon="delete" bg-color="error" class="ml-2" @click="removeDownload" />
|
||||||
|
</div>
|
||||||
|
<p class="px-0.5 py-1 text-sm font-mono text-center">Size: {{ $bytesPretty(abmergeDownload.size) }}</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<p class="text-left text-base mb-4 py-4">
|
||||||
|
<span class="text-error">* <strong>Experimental</strong></span
|
||||||
|
> - M4b merge can take several minutes and will be stored in <span class="bg-primary bg-opacity-75 font-mono p-1 text-base">/metadata/downloads</span>. After the download is ready, it will remain available for 60 minutes, then be deleted. Download will timeout after 20 minutes.
|
||||||
|
</p>
|
||||||
|
|
||||||
|
<p v-if="isSingleM4b" class="text-lg text-center my-8">Audiobook is already a single m4b!</p>
|
||||||
|
<p v-else-if="!mediaTracks.length" class="text-lg text-center my-8">No audio tracks to merge</p>
|
||||||
|
|
||||||
|
<div v-if="isDownloading" class="absolute top-0 left-0 w-full h-full bg-black bg-opacity-50 z-50 flex items-center justify-center">
|
||||||
|
<div class="w-80 border border-black-400 bg-bg rounded-xl h-20">
|
||||||
|
<div class="w-full h-full flex items-center justify-center">
|
||||||
|
<p class="text-lg">Download.... {{ downloadPercent }}%</p>
|
||||||
|
<p class="w-24 font-mono pl-8 text-right">
|
||||||
|
{{ downloadAmount }}
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
export default {
|
||||||
|
props: {
|
||||||
|
processing: Boolean,
|
||||||
|
libraryItem: {
|
||||||
|
type: Object,
|
||||||
|
default: () => {}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
data() {
|
||||||
|
return {
|
||||||
|
tempDisable: false,
|
||||||
|
isDownloading: false,
|
||||||
|
downloadPercent: '0',
|
||||||
|
downloadAmount: '0 KB'
|
||||||
|
}
|
||||||
|
},
|
||||||
|
watch: {
|
||||||
|
abmergeStatus(newVal) {
|
||||||
|
if (newVal) {
|
||||||
|
this.tempDisable = false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
computed: {
|
||||||
|
libraryItemId() {
|
||||||
|
return this.libraryItem ? this.libraryItem.id : null
|
||||||
|
},
|
||||||
|
media() {
|
||||||
|
return this.libraryItem ? this.libraryItem.media || {} : {}
|
||||||
|
},
|
||||||
|
downloads() {
|
||||||
|
return this.$store.getters['downloads/getDownloads'](this.libraryItemId)
|
||||||
|
},
|
||||||
|
abmergeDownload() {
|
||||||
|
return this.downloads.find((d) => d.type === 'abmerge')
|
||||||
|
},
|
||||||
|
abmergeStatus() {
|
||||||
|
return this.abmergeDownload ? this.abmergeDownload.status : false
|
||||||
|
},
|
||||||
|
libraryFiles() {
|
||||||
|
return this.libraryItem.libraryFiles
|
||||||
|
},
|
||||||
|
totalFiles() {
|
||||||
|
return this.libraryFiles.length
|
||||||
|
},
|
||||||
|
mediaTracks() {
|
||||||
|
return this.media.tracks || []
|
||||||
|
},
|
||||||
|
isSingleM4b() {
|
||||||
|
return this.mediaTracks.length === 1 && this.mediaTracks[0].metadata.ext.toLowerCase() === '.m4b'
|
||||||
|
},
|
||||||
|
showM4bDownload() {
|
||||||
|
if (this.libraryItem.isMissing || !this.mediaTracks.length) return false
|
||||||
|
return !this.isSingleM4b && this.mediaTracks.length > 0
|
||||||
|
}
|
||||||
|
},
|
||||||
|
methods: {
|
||||||
|
removeDownload() {
|
||||||
|
if (!this.abmergeDownload) return
|
||||||
|
if (!confirm(`Are you sure you want to remove this merge download?`)) return
|
||||||
|
|
||||||
|
var downloadId = this.abmergeDownload.id
|
||||||
|
|
||||||
|
this.tempDisable = true
|
||||||
|
this.$axios
|
||||||
|
.$delete(`/api/download/${downloadId}`)
|
||||||
|
.then(() => {
|
||||||
|
this.tempDisable = false
|
||||||
|
this.$toast.success('Merge download deleted')
|
||||||
|
this.$store.commit('downloads/removeDownload', { id: downloadId })
|
||||||
|
})
|
||||||
|
.catch((error) => {
|
||||||
|
var errorMsg = error.response ? error.response.data || 'Unknown Error' : 'Unknown Error'
|
||||||
|
this.$toast.error(errorMsg)
|
||||||
|
this.tempDisable = false
|
||||||
|
})
|
||||||
|
},
|
||||||
|
startAudiobookMerge() {
|
||||||
|
this.tempDisable = true
|
||||||
|
|
||||||
|
this.$axios
|
||||||
|
.$get(`/api/audiobook-merge/${this.libraryItemId}`)
|
||||||
|
.then(() => {
|
||||||
|
this.tempDisable = false
|
||||||
|
})
|
||||||
|
.catch((error) => {
|
||||||
|
var errorMsg = error.response ? error.response.data || 'Unknown Error' : 'Unknown Error'
|
||||||
|
this.$toast.error(errorMsg)
|
||||||
|
this.tempDisable = false
|
||||||
|
})
|
||||||
|
},
|
||||||
|
downloadWithProgress(download) {
|
||||||
|
var downloadId = download.id
|
||||||
|
var downloadUrl = `${process.env.serverUrl}/api/download/${downloadId}`
|
||||||
|
var filename = download.filename
|
||||||
|
|
||||||
|
this.isDownloading = true
|
||||||
|
|
||||||
|
var request = new XMLHttpRequest()
|
||||||
|
request.responseType = 'blob'
|
||||||
|
request.open('get', downloadUrl, true)
|
||||||
|
request.setRequestHeader('Authorization', `Bearer ${this.$store.getters['user/getToken']}`)
|
||||||
|
request.send()
|
||||||
|
|
||||||
|
request.onreadystatechange = () => {
|
||||||
|
if (request.readyState === 4) {
|
||||||
|
this.isDownloading = false
|
||||||
|
}
|
||||||
|
if (request.readyState == 4 && request.status == 200) {
|
||||||
|
const url = window.URL.createObjectURL(request.response)
|
||||||
|
|
||||||
|
const anchor = document.createElement('a')
|
||||||
|
anchor.href = url
|
||||||
|
anchor.download = filename
|
||||||
|
document.body.appendChild(anchor)
|
||||||
|
anchor.click()
|
||||||
|
setTimeout(() => {
|
||||||
|
if (anchor) anchor.remove()
|
||||||
|
}, 1000)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
request.onerror = (err) => {
|
||||||
|
console.error('Download error', err)
|
||||||
|
this.isDownloading = false
|
||||||
|
}
|
||||||
|
|
||||||
|
request.onprogress = (e) => {
|
||||||
|
const percent_complete = Math.floor((e.loaded / e.total) * 100)
|
||||||
|
this.downloadAmount = this.$bytesPretty(e.loaded)
|
||||||
|
this.downloadPercent = percent_complete
|
||||||
|
|
||||||
|
// const duration = (new Date().getTime() - startTime) / 1000
|
||||||
|
// const bps = e.loaded / duration
|
||||||
|
// const kbps = Math.floor(bps / 1024)
|
||||||
|
// const time = (e.total - e.loaded) / bps
|
||||||
|
// const seconds = Math.floor(time % 60)
|
||||||
|
// const minutes = Math.floor(time / 60)
|
||||||
|
// console.log(`${percent_complete}% - ${kbps} Kbps - ${minutes} min ${seconds} sec remaining`)
|
||||||
|
}
|
||||||
|
},
|
||||||
|
loadDownloads() {
|
||||||
|
this.$axios
|
||||||
|
.$get(`/api/downloads`)
|
||||||
|
.then((data) => {
|
||||||
|
var pendingDownloads = data.pendingDownloads.map((pd) => {
|
||||||
|
pd.download.status = this.$constants.DownloadStatus.PENDING
|
||||||
|
return pd.download
|
||||||
|
})
|
||||||
|
var downloads = data.downloads.map((d) => {
|
||||||
|
d.status = this.$constants.DownloadStatus.READY
|
||||||
|
return d
|
||||||
|
})
|
||||||
|
var allDownloads = downloads.concat(pendingDownloads)
|
||||||
|
this.$store.commit('downloads/setDownloads', allDownloads)
|
||||||
|
})
|
||||||
|
.catch((error) => {
|
||||||
|
console.error('Failed to load downloads', error)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
},
|
||||||
|
mounted() {
|
||||||
|
this.loadDownloads()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
</script>
|
@ -271,29 +271,24 @@ export default {
|
|||||||
}
|
}
|
||||||
this.$store.commit('user/removeCollection', collection)
|
this.$store.commit('user/removeCollection', collection)
|
||||||
},
|
},
|
||||||
downloadToastClick(download) {
|
abmergeStarted(download) {
|
||||||
if (!download || !download.audiobookId) {
|
|
||||||
return console.error('Invalid download object', download)
|
|
||||||
}
|
|
||||||
},
|
|
||||||
downloadStarted(download) {
|
|
||||||
download.status = this.$constants.DownloadStatus.PENDING
|
download.status = this.$constants.DownloadStatus.PENDING
|
||||||
download.toastId = this.$toast(`Preparing download "${download.filename}"`, { timeout: false, draggable: false, closeOnClick: false, onClick: () => this.downloadToastClick(download) })
|
download.toastId = this.$toast(`Preparing download "${download.filename}"`, { timeout: false, draggable: false, closeOnClick: false })
|
||||||
this.$store.commit('downloads/addUpdateDownload', download)
|
this.$store.commit('downloads/addUpdateDownload', download)
|
||||||
},
|
},
|
||||||
downloadReady(download) {
|
abmergeReady(download) {
|
||||||
download.status = this.$constants.DownloadStatus.READY
|
download.status = this.$constants.DownloadStatus.READY
|
||||||
var existingDownload = this.$store.getters['downloads/getDownload'](download.id)
|
var existingDownload = this.$store.getters['downloads/getDownload'](download.id)
|
||||||
|
|
||||||
if (existingDownload && existingDownload.toastId !== undefined) {
|
if (existingDownload && existingDownload.toastId !== undefined) {
|
||||||
download.toastId = existingDownload.toastId
|
download.toastId = existingDownload.toastId
|
||||||
this.$toast.update(existingDownload.toastId, { content: `Download "${download.filename}" is ready!`, options: { timeout: 5000, type: 'success', onClick: () => this.downloadToastClick(download) } }, true)
|
this.$toast.update(existingDownload.toastId, { content: `Download "${download.filename}" is ready!`, options: { timeout: 5000, type: 'success' } }, true)
|
||||||
} else {
|
} else {
|
||||||
this.$toast.success(`Download "${download.filename}" is ready!`)
|
this.$toast.success(`Download "${download.filename}" is ready!`)
|
||||||
}
|
}
|
||||||
this.$store.commit('downloads/addUpdateDownload', download)
|
this.$store.commit('downloads/addUpdateDownload', download)
|
||||||
},
|
},
|
||||||
downloadFailed(download) {
|
abmergeFailed(download) {
|
||||||
download.status = this.$constants.DownloadStatus.FAILED
|
download.status = this.$constants.DownloadStatus.FAILED
|
||||||
var existingDownload = this.$store.getters['downloads/getDownload'](download.id)
|
var existingDownload = this.$store.getters['downloads/getDownload'](download.id)
|
||||||
|
|
||||||
@ -301,25 +296,25 @@ export default {
|
|||||||
|
|
||||||
if (existingDownload && existingDownload.toastId !== undefined) {
|
if (existingDownload && existingDownload.toastId !== undefined) {
|
||||||
download.toastId = existingDownload.toastId
|
download.toastId = existingDownload.toastId
|
||||||
this.$toast.update(existingDownload.toastId, { content: `Download "${download.filename}" ${failedMsg}`, options: { timeout: 5000, type: 'error', onClick: () => this.downloadToastClick(download) } }, true)
|
this.$toast.update(existingDownload.toastId, { content: `Download "${download.filename}" ${failedMsg}`, options: { timeout: 5000, type: 'error' } }, true)
|
||||||
} else {
|
} else {
|
||||||
console.warn('Download failed no existing download', existingDownload)
|
console.warn('Download failed no existing download', existingDownload)
|
||||||
this.$toast.error(`Download "${download.filename}" ${failedMsg}`)
|
this.$toast.error(`Download "${download.filename}" ${failedMsg}`)
|
||||||
}
|
}
|
||||||
this.$store.commit('downloads/addUpdateDownload', download)
|
this.$store.commit('downloads/addUpdateDownload', download)
|
||||||
},
|
},
|
||||||
downloadKilled(download) {
|
abmergeKilled(download) {
|
||||||
var existingDownload = this.$store.getters['downloads/getDownload'](download.id)
|
var existingDownload = this.$store.getters['downloads/getDownload'](download.id)
|
||||||
if (existingDownload && existingDownload.toastId !== undefined) {
|
if (existingDownload && existingDownload.toastId !== undefined) {
|
||||||
download.toastId = existingDownload.toastId
|
download.toastId = existingDownload.toastId
|
||||||
this.$toast.update(existingDownload.toastId, { content: `Download "${download.filename}" was terminated`, options: { timeout: 5000, type: 'error', onClick: () => this.downloadToastClick(download) } }, true)
|
this.$toast.update(existingDownload.toastId, { content: `Download "${download.filename}" was terminated`, options: { timeout: 5000, type: 'error' } }, true)
|
||||||
} else {
|
} else {
|
||||||
console.warn('Download killed no existing download found', existingDownload)
|
console.warn('Download killed no existing download found', existingDownload)
|
||||||
this.$toast.error(`Download "${download.filename}" was terminated`)
|
this.$toast.error(`Download "${download.filename}" was terminated`)
|
||||||
}
|
}
|
||||||
this.$store.commit('downloads/removeDownload', download)
|
this.$store.commit('downloads/removeDownload', download)
|
||||||
},
|
},
|
||||||
downloadExpired(download) {
|
abmergeExpired(download) {
|
||||||
download.status = this.$constants.DownloadStatus.EXPIRED
|
download.status = this.$constants.DownloadStatus.EXPIRED
|
||||||
this.$store.commit('downloads/addUpdateDownload', download)
|
this.$store.commit('downloads/addUpdateDownload', download)
|
||||||
},
|
},
|
||||||
@ -393,11 +388,11 @@ export default {
|
|||||||
this.socket.on('scan_progress', this.scanProgress)
|
this.socket.on('scan_progress', this.scanProgress)
|
||||||
|
|
||||||
// Download Listeners
|
// Download Listeners
|
||||||
this.socket.on('download_started', this.downloadStarted)
|
this.socket.on('abmerge_started', this.abmergeStarted)
|
||||||
this.socket.on('download_ready', this.downloadReady)
|
this.socket.on('abmerge_ready', this.abmergeReady)
|
||||||
this.socket.on('download_failed', this.downloadFailed)
|
this.socket.on('abmerge_failed', this.abmergeFailed)
|
||||||
this.socket.on('download_killed', this.downloadKilled)
|
this.socket.on('abmerge_killed', this.abmergeKilled)
|
||||||
this.socket.on('download_expired', this.downloadExpired)
|
this.socket.on('abmerge_expired', this.abmergeExpired)
|
||||||
|
|
||||||
// Toast Listeners
|
// Toast Listeners
|
||||||
this.socket.on('show_error_toast', this.showErrorToast)
|
this.socket.on('show_error_toast', this.showErrorToast)
|
||||||
|
@ -4,8 +4,8 @@ export const state = () => ({
|
|||||||
})
|
})
|
||||||
|
|
||||||
export const getters = {
|
export const getters = {
|
||||||
getDownloads: (state) => (audiobookId) => {
|
getDownloads: (state) => (libraryItemId) => {
|
||||||
return state.downloads.filter(d => d.audiobookId === audiobookId)
|
return state.downloads.filter(d => d.libraryItemId === libraryItemId)
|
||||||
},
|
},
|
||||||
getDownload: (state) => (id) => {
|
getDownload: (state) => (id) => {
|
||||||
return state.downloads.find(d => d.id === id)
|
return state.downloads.find(d => d.id === id)
|
||||||
@ -17,15 +17,10 @@ export const actions = {
|
|||||||
}
|
}
|
||||||
|
|
||||||
export const mutations = {
|
export const mutations = {
|
||||||
|
setDownloads(state, downloads) {
|
||||||
|
state.downloads = downloads
|
||||||
|
},
|
||||||
addUpdateDownload(state, download) {
|
addUpdateDownload(state, download) {
|
||||||
// Remove older downloads of matching type
|
|
||||||
state.downloads = state.downloads.filter(d => {
|
|
||||||
if (d.id !== download.id && d.type === download.type) {
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
return true
|
|
||||||
})
|
|
||||||
|
|
||||||
var index = state.downloads.findIndex(d => d.id === download.id)
|
var index = state.downloads.findIndex(d => d.id === download.id)
|
||||||
if (index >= 0) {
|
if (index >= 0) {
|
||||||
state.downloads.splice(index, 1, download)
|
state.downloads.splice(index, 1, download)
|
||||||
|
@ -23,7 +23,7 @@ const HlsRouter = require('./routers/HlsRouter')
|
|||||||
const StaticRouter = require('./routers/StaticRouter')
|
const StaticRouter = require('./routers/StaticRouter')
|
||||||
|
|
||||||
const CoverManager = require('./managers/CoverManager')
|
const CoverManager = require('./managers/CoverManager')
|
||||||
const DownloadManager = require('./managers/DownloadManager')
|
const AbMergeManager = require('./managers/AbMergeManager')
|
||||||
const CacheManager = require('./managers/CacheManager')
|
const CacheManager = require('./managers/CacheManager')
|
||||||
const LogManager = require('./managers/LogManager')
|
const LogManager = require('./managers/LogManager')
|
||||||
const BackupManager = require('./managers/BackupManager')
|
const BackupManager = require('./managers/BackupManager')
|
||||||
@ -58,7 +58,7 @@ class Server {
|
|||||||
this.backupManager = new BackupManager(this.db, this.emitter.bind(this))
|
this.backupManager = new BackupManager(this.db, this.emitter.bind(this))
|
||||||
this.logManager = new LogManager(this.db)
|
this.logManager = new LogManager(this.db)
|
||||||
this.cacheManager = new CacheManager()
|
this.cacheManager = new CacheManager()
|
||||||
this.downloadManager = new DownloadManager(this.db)
|
this.abMergeManager = new AbMergeManager(this.db, this.clientEmitter.bind(this))
|
||||||
this.playbackSessionManager = new PlaybackSessionManager(this.db, this.emitter.bind(this), this.clientEmitter.bind(this))
|
this.playbackSessionManager = new PlaybackSessionManager(this.db, this.emitter.bind(this), this.clientEmitter.bind(this))
|
||||||
this.coverManager = new CoverManager(this.db, this.cacheManager)
|
this.coverManager = new CoverManager(this.db, this.cacheManager)
|
||||||
this.podcastManager = new PodcastManager(this.db, this.watcher, this.emitter.bind(this))
|
this.podcastManager = new PodcastManager(this.db, this.watcher, this.emitter.bind(this))
|
||||||
@ -66,7 +66,7 @@ class Server {
|
|||||||
this.scanner = new Scanner(this.db, this.coverManager, this.emitter.bind(this))
|
this.scanner = new Scanner(this.db, this.coverManager, this.emitter.bind(this))
|
||||||
|
|
||||||
// Routers
|
// Routers
|
||||||
this.apiRouter = new ApiRouter(this.db, this.auth, this.scanner, this.playbackSessionManager, this.downloadManager, this.coverManager, this.backupManager, this.watcher, this.cacheManager, this.podcastManager, this.emitter.bind(this), this.clientEmitter.bind(this))
|
this.apiRouter = new ApiRouter(this.db, this.auth, this.scanner, this.playbackSessionManager, this.abMergeManager, this.coverManager, this.backupManager, this.watcher, this.cacheManager, this.podcastManager, this.emitter.bind(this), this.clientEmitter.bind(this))
|
||||||
this.hlsRouter = new HlsRouter(this.db, this.auth, this.playbackSessionManager, this.emitter.bind(this))
|
this.hlsRouter = new HlsRouter(this.db, this.auth, this.playbackSessionManager, this.emitter.bind(this))
|
||||||
this.staticRouter = new StaticRouter(this.db)
|
this.staticRouter = new StaticRouter(this.db)
|
||||||
|
|
||||||
@ -112,8 +112,8 @@ class Server {
|
|||||||
|
|
||||||
async init() {
|
async init() {
|
||||||
Logger.info('[Server] Init v' + version)
|
Logger.info('[Server] Init v' + version)
|
||||||
|
await this.abMergeManager.removeOrphanDownloads()
|
||||||
await this.playbackSessionManager.removeOrphanStreams()
|
await this.playbackSessionManager.removeOrphanStreams()
|
||||||
await this.downloadManager.removeOrphanDownloads()
|
|
||||||
|
|
||||||
var previousVersion = await this.db.checkPreviousVersion() // Returns null if same server version
|
var previousVersion = await this.db.checkPreviousVersion() // Returns null if same server version
|
||||||
if (previousVersion) {
|
if (previousVersion) {
|
||||||
|
@ -82,15 +82,43 @@ class MiscController {
|
|||||||
res.sendStatus(200)
|
res.sendStatus(200)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// GET: api/audiobook-merge/:id
|
||||||
|
async mergeAudiobook(req, res) {
|
||||||
|
if (!req.user.canDownload) {
|
||||||
|
Logger.error('User attempting to download without permission', req.user)
|
||||||
|
return res.sendStatus(403)
|
||||||
|
}
|
||||||
|
|
||||||
|
var libraryItem = this.db.getLibraryItem(req.params.id)
|
||||||
|
if (!libraryItem || libraryItem.isMissing || libraryItem.isInvalid) {
|
||||||
|
Logger.error(`[MiscController] mergeAudiboook: library item not found or invalid ${req.params.id}`)
|
||||||
|
return res.status(404).send('Audiobook not found')
|
||||||
|
}
|
||||||
|
|
||||||
|
if (libraryItem.mediaType !== 'book') {
|
||||||
|
Logger.error(`[MiscController] mergeAudiboook: Invalid library item ${req.params.id}: not a book`)
|
||||||
|
return res.status(500).send('Invalid library item: not a book')
|
||||||
|
}
|
||||||
|
|
||||||
|
if (libraryItem.media.tracks.length <= 0) {
|
||||||
|
Logger.error(`[MiscController] mergeAudiboook: Invalid audiobook ${req.params.id}: no audio tracks`)
|
||||||
|
return res.status(500).send('Invalid audiobook: no audio tracks')
|
||||||
|
}
|
||||||
|
|
||||||
|
this.abMergeManager.startAudiobookMerge(req.user, libraryItem)
|
||||||
|
|
||||||
|
res.sendStatus(200)
|
||||||
|
}
|
||||||
|
|
||||||
// GET: api/download/:id
|
// GET: api/download/:id
|
||||||
async download(req, res) {
|
async getDownload(req, res) {
|
||||||
if (!req.user.canDownload) {
|
if (!req.user.canDownload) {
|
||||||
Logger.error('User attempting to download without permission', req.user)
|
Logger.error('User attempting to download without permission', req.user)
|
||||||
return res.sendStatus(403)
|
return res.sendStatus(403)
|
||||||
}
|
}
|
||||||
var downloadId = req.params.id
|
var downloadId = req.params.id
|
||||||
Logger.info('Download Request', downloadId)
|
Logger.info('Download Request', downloadId)
|
||||||
var download = this.downloadManager.getDownload(downloadId)
|
var download = this.abMergeManager.getDownload(downloadId)
|
||||||
if (!download) {
|
if (!download) {
|
||||||
Logger.error('Download request not found', downloadId)
|
Logger.error('Download request not found', downloadId)
|
||||||
return res.sendStatus(404)
|
return res.sendStatus(404)
|
||||||
@ -101,13 +129,36 @@ class MiscController {
|
|||||||
'Content-Type': download.mimeType
|
'Content-Type': download.mimeType
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
res.download(download.fullPath, download.filename, options, (err) => {
|
res.download(download.path, download.filename, options, (err) => {
|
||||||
if (err) {
|
if (err) {
|
||||||
Logger.error('Download Error', err)
|
Logger.error('Download Error', err)
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// DELETE: api/download/:id
|
||||||
|
async removeDownload(req, res) {
|
||||||
|
if (!req.user.canDownload || !req.user.canDelete) {
|
||||||
|
Logger.error('User attempting to remove download without permission', req.user.username)
|
||||||
|
return res.sendStatus(403)
|
||||||
|
}
|
||||||
|
this.abMergeManager.removeDownloadById(req.params.id)
|
||||||
|
res.sendStatus(200)
|
||||||
|
}
|
||||||
|
|
||||||
|
// GET: api/downloads
|
||||||
|
async getDownloads(req, res) {
|
||||||
|
if (!req.user.canDownload) {
|
||||||
|
Logger.error('User attempting to get downloads without permission', req.user.username)
|
||||||
|
return res.sendStatus(403)
|
||||||
|
}
|
||||||
|
var downloads = {
|
||||||
|
downloads: this.abMergeManager.downloads,
|
||||||
|
pendingDownloads: this.abMergeManager.pendingDownloads
|
||||||
|
}
|
||||||
|
res.json(downloads)
|
||||||
|
}
|
||||||
|
|
||||||
// PATCH: api/settings (Root)
|
// PATCH: api/settings (Root)
|
||||||
async updateServerSettings(req, res) {
|
async updateServerSettings(req, res) {
|
||||||
if (!req.user.isRoot) {
|
if (!req.user.isRoot) {
|
||||||
|
284
server/managers/AbMergeManager.js
Normal file
284
server/managers/AbMergeManager.js
Normal file
@ -0,0 +1,284 @@
|
|||||||
|
|
||||||
|
const Path = require('path')
|
||||||
|
const fs = require('fs-extra')
|
||||||
|
|
||||||
|
const workerThreads = require('worker_threads')
|
||||||
|
const Logger = require('../Logger')
|
||||||
|
const Download = require('../objects/Download')
|
||||||
|
const filePerms = require('../utils/filePerms')
|
||||||
|
const { getId } = require('../utils/index')
|
||||||
|
const { writeConcatFile, writeMetadataFile } = require('../utils/ffmpegHelpers')
|
||||||
|
const { getFileSize } = require('../utils/fileUtils')
|
||||||
|
|
||||||
|
class AbMergeManager {
|
||||||
|
constructor(db, clientEmitter) {
|
||||||
|
this.db = db
|
||||||
|
this.clientEmitter = clientEmitter
|
||||||
|
|
||||||
|
this.downloadDirPath = Path.join(global.MetadataPath, 'downloads')
|
||||||
|
|
||||||
|
this.pendingDownloads = []
|
||||||
|
this.downloads = []
|
||||||
|
}
|
||||||
|
|
||||||
|
getDownload(downloadId) {
|
||||||
|
return this.downloads.find(d => d.id === downloadId)
|
||||||
|
}
|
||||||
|
|
||||||
|
removeDownloadById(downloadId) {
|
||||||
|
var download = this.getDownload(downloadId)
|
||||||
|
if (download) {
|
||||||
|
this.removeDownload(download)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async removeOrphanDownloads() {
|
||||||
|
try {
|
||||||
|
var dirs = await fs.readdir(this.downloadDirPath)
|
||||||
|
if (!dirs || !dirs.length) return true
|
||||||
|
|
||||||
|
dirs = dirs.filter(d => d.startsWith('abmerge'))
|
||||||
|
|
||||||
|
await Promise.all(dirs.map(async (dirname) => {
|
||||||
|
var fullPath = Path.join(this.downloadDirPath, dirname)
|
||||||
|
Logger.info(`Removing Orphan Download ${dirname}`)
|
||||||
|
return fs.remove(fullPath)
|
||||||
|
}))
|
||||||
|
return true
|
||||||
|
} catch (error) {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async startAudiobookMerge(user, libraryItem) {
|
||||||
|
var downloadId = getId('abmerge')
|
||||||
|
var dlpath = Path.join(this.downloadDirPath, downloadId)
|
||||||
|
Logger.info(`Start audiobook merge for ${libraryItem.id} - DownloadId: ${downloadId} - ${dlpath}`)
|
||||||
|
|
||||||
|
var audiobookDirname = Path.basename(libraryItem.path)
|
||||||
|
var filename = audiobookDirname + '.m4b'
|
||||||
|
var downloadData = {
|
||||||
|
id: downloadId,
|
||||||
|
libraryItemId: libraryItem.id,
|
||||||
|
type: 'abmerge',
|
||||||
|
dirpath: dlpath,
|
||||||
|
path: Path.join(dlpath, filename),
|
||||||
|
filename,
|
||||||
|
ext: '.m4b',
|
||||||
|
userId: user.id
|
||||||
|
}
|
||||||
|
var download = new Download()
|
||||||
|
download.setData(downloadData)
|
||||||
|
download.setTimeoutTimer(this.downloadTimedOut.bind(this))
|
||||||
|
|
||||||
|
|
||||||
|
try {
|
||||||
|
await fs.mkdir(download.dirpath)
|
||||||
|
} catch (error) {
|
||||||
|
Logger.error(`[AbMergeManager] Failed to make directory ${download.dirpath}`)
|
||||||
|
var downloadJson = download.toJSON()
|
||||||
|
this.clientEmitter(user.id, 'abmerge_failed', downloadJson)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
this.clientEmitter(user.id, 'abmerge_started', download.toJSON())
|
||||||
|
this.runAudiobookMerge(libraryItem, download)
|
||||||
|
}
|
||||||
|
|
||||||
|
async runAudiobookMerge(libraryItem, download) {
|
||||||
|
|
||||||
|
// If changing audio file type then encoding is needed
|
||||||
|
var audioTracks = libraryItem.media.tracks
|
||||||
|
var audioRequiresEncode = audioTracks[0].metadata.ext !== download.ext
|
||||||
|
var shouldIncludeCover = libraryItem.media.coverPath
|
||||||
|
var firstTrackIsM4b = audioTracks[0].metadata.ext.toLowerCase() === '.m4b'
|
||||||
|
var isOneTrack = audioTracks.length === 1
|
||||||
|
|
||||||
|
const ffmpegInputs = []
|
||||||
|
|
||||||
|
if (!isOneTrack) {
|
||||||
|
var concatFilePath = Path.join(download.dirpath, 'files.txt')
|
||||||
|
console.log('Write files.txt', concatFilePath)
|
||||||
|
await writeConcatFile(audioTracks, concatFilePath)
|
||||||
|
ffmpegInputs.push({
|
||||||
|
input: concatFilePath,
|
||||||
|
options: ['-safe 0', '-f concat']
|
||||||
|
})
|
||||||
|
} else {
|
||||||
|
ffmpegInputs.push({
|
||||||
|
input: audioTracks[0].metadata.path,
|
||||||
|
options: firstTrackIsM4b ? ['-f mp4'] : []
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
const logLevel = process.env.NODE_ENV === 'production' ? 'error' : 'warning'
|
||||||
|
var ffmpegOptions = [`-loglevel ${logLevel}`]
|
||||||
|
var ffmpegOutputOptions = []
|
||||||
|
|
||||||
|
if (audioRequiresEncode) {
|
||||||
|
ffmpegOptions = ffmpegOptions.concat([
|
||||||
|
'-map 0:a',
|
||||||
|
'-acodec aac',
|
||||||
|
'-ac 2',
|
||||||
|
'-b:a 64k',
|
||||||
|
'-id3v2_version 3'
|
||||||
|
])
|
||||||
|
} else {
|
||||||
|
ffmpegOptions.push('-max_muxing_queue_size 1000')
|
||||||
|
|
||||||
|
if (isOneTrack && firstTrackIsM4b && !shouldIncludeCover) {
|
||||||
|
ffmpegOptions.push('-c copy')
|
||||||
|
} else {
|
||||||
|
ffmpegOptions.push('-c:a copy')
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (download.ext === '.m4b') {
|
||||||
|
ffmpegOutputOptions.push('-f mp4')
|
||||||
|
}
|
||||||
|
|
||||||
|
// Create ffmetadata file
|
||||||
|
var metadataFilePath = Path.join(download.dirpath, 'metadata.txt')
|
||||||
|
await writeMetadataFile(libraryItem, metadataFilePath)
|
||||||
|
ffmpegInputs.push({
|
||||||
|
input: metadataFilePath
|
||||||
|
})
|
||||||
|
ffmpegOptions.push('-map_metadata 1')
|
||||||
|
|
||||||
|
// Embed cover art
|
||||||
|
if (shouldIncludeCover) {
|
||||||
|
var coverPath = libraryItem.media.coverPath.replace(/\\/g, '/')
|
||||||
|
ffmpegInputs.push({
|
||||||
|
input: coverPath,
|
||||||
|
options: ['-f image2pipe']
|
||||||
|
})
|
||||||
|
ffmpegOptions.push('-vf [2:v]crop=trunc(iw/2)*2:trunc(ih/2)*2')
|
||||||
|
ffmpegOptions.push('-map 2:v')
|
||||||
|
}
|
||||||
|
|
||||||
|
var workerData = {
|
||||||
|
inputs: ffmpegInputs,
|
||||||
|
options: ffmpegOptions,
|
||||||
|
outputOptions: ffmpegOutputOptions,
|
||||||
|
output: download.path,
|
||||||
|
}
|
||||||
|
|
||||||
|
var worker = null
|
||||||
|
try {
|
||||||
|
var workerPath = Path.join(global.appRoot, 'server/utils/downloadWorker.js')
|
||||||
|
worker = new workerThreads.Worker(workerPath, { workerData })
|
||||||
|
} catch (error) {
|
||||||
|
Logger.error(`[AbMergeManager] Start worker thread failed`, error)
|
||||||
|
if (download.userId) {
|
||||||
|
var downloadJson = download.toJSON()
|
||||||
|
this.clientEmitter(download.userId, 'abmerge_failed', downloadJson)
|
||||||
|
}
|
||||||
|
this.removeDownload(download)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
worker.on('message', (message) => {
|
||||||
|
if (message != null && typeof message === 'object') {
|
||||||
|
if (message.type === 'RESULT') {
|
||||||
|
if (!download.isTimedOut) {
|
||||||
|
this.sendResult(download, message)
|
||||||
|
}
|
||||||
|
} else if (message.type === 'FFMPEG') {
|
||||||
|
if (Logger[message.level]) {
|
||||||
|
Logger[message.level](message.log)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
Logger.error('Invalid worker message', message)
|
||||||
|
}
|
||||||
|
})
|
||||||
|
this.pendingDownloads.push({
|
||||||
|
id: download.id,
|
||||||
|
download,
|
||||||
|
worker
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
async sendResult(download, result) {
|
||||||
|
download.clearTimeoutTimer()
|
||||||
|
|
||||||
|
// Remove pending download
|
||||||
|
this.pendingDownloads = this.pendingDownloads.filter(d => d.id !== download.id)
|
||||||
|
|
||||||
|
if (result.isKilled) {
|
||||||
|
if (download.userId) {
|
||||||
|
this.clientEmitter(download.userId, 'abmerge_killed', download.toJSON())
|
||||||
|
}
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!result.success) {
|
||||||
|
if (download.userId) {
|
||||||
|
this.clientEmitter(download.userId, 'abmerge_failed', download.toJSON())
|
||||||
|
}
|
||||||
|
this.removeDownload(download)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
// Set file permissions and ownership
|
||||||
|
await filePerms.setDefault(download.path)
|
||||||
|
|
||||||
|
var filesize = await getFileSize(download.path)
|
||||||
|
download.setComplete(filesize)
|
||||||
|
if (download.userId) {
|
||||||
|
this.clientEmitter(download.userId, 'abmerge_ready', download.toJSON())
|
||||||
|
}
|
||||||
|
download.setExpirationTimer(this.downloadExpired.bind(this))
|
||||||
|
|
||||||
|
this.downloads.push(download)
|
||||||
|
Logger.info(`[AbMergeManager] Download Ready ${download.id}`)
|
||||||
|
}
|
||||||
|
|
||||||
|
async downloadExpired(download) {
|
||||||
|
Logger.info(`[AbMergeManager] Download ${download.id} expired`)
|
||||||
|
|
||||||
|
if (download.userId) {
|
||||||
|
this.clientEmitter(download.userId, 'abmerge_expired', download.toJSON())
|
||||||
|
}
|
||||||
|
this.removeDownload(download)
|
||||||
|
}
|
||||||
|
|
||||||
|
async downloadTimedOut(download) {
|
||||||
|
Logger.info(`[AbMergeManager] Download ${download.id} timed out (${download.timeoutTimeMs}ms)`)
|
||||||
|
|
||||||
|
if (download.userId) {
|
||||||
|
var downloadJson = download.toJSON()
|
||||||
|
downloadJson.isTimedOut = true
|
||||||
|
this.clientEmitter(download.userId, 'abmerge_failed', downloadJson)
|
||||||
|
}
|
||||||
|
this.removeDownload(download)
|
||||||
|
}
|
||||||
|
|
||||||
|
async removeDownload(download) {
|
||||||
|
Logger.info('[AbMergeManager] Removing download ' + download.id)
|
||||||
|
|
||||||
|
download.clearTimeoutTimer()
|
||||||
|
download.clearExpirationTimer()
|
||||||
|
|
||||||
|
var pendingDl = this.pendingDownloads.find(d => d.id === download.id)
|
||||||
|
|
||||||
|
if (pendingDl) {
|
||||||
|
this.pendingDownloads = this.pendingDownloads.filter(d => d.id !== download.id)
|
||||||
|
Logger.warn(`[AbMergeManager] Removing download in progress - stopping worker`)
|
||||||
|
if (pendingDl.worker) {
|
||||||
|
try {
|
||||||
|
pendingDl.worker.postMessage('STOP')
|
||||||
|
} catch (error) {
|
||||||
|
Logger.error('[AbMergeManager] Error posting stop message to worker', error)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
await fs.remove(download.dirpath).then(() => {
|
||||||
|
Logger.info('[AbMergeManager] Deleted download', download.dirpath)
|
||||||
|
}).catch((err) => {
|
||||||
|
Logger.error('[AbMergeManager] Failed to delete download', err)
|
||||||
|
})
|
||||||
|
this.downloads = this.downloads.filter(d => d.id !== download.id)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
module.exports = AbMergeManager
|
@ -1,20 +1,18 @@
|
|||||||
const DEFAULT_EXPIRATION = 1000 * 60 * 60 // 60 minutes
|
const DEFAULT_EXPIRATION = 1000 * 60 * 60 // 60 minutes
|
||||||
const DEFAULT_TIMEOUT = 1000 * 60 * 15 // 15 minutes
|
const DEFAULT_TIMEOUT = 1000 * 60 * 20 // 20 minutes
|
||||||
class Download {
|
class Download {
|
||||||
constructor(download) {
|
constructor(download) {
|
||||||
this.id = null
|
this.id = null
|
||||||
this.audiobookId = null
|
this.libraryItemId = null
|
||||||
this.type = null
|
this.type = null
|
||||||
this.options = {}
|
|
||||||
|
|
||||||
this.dirpath = null
|
this.dirpath = null
|
||||||
this.fullPath = null
|
this.path = null
|
||||||
this.ext = null
|
this.ext = null
|
||||||
this.filename = null
|
this.filename = null
|
||||||
this.size = 0
|
this.size = 0
|
||||||
|
|
||||||
this.userId = null
|
this.userId = null
|
||||||
this.socket = null // Socket to notify when complete
|
|
||||||
this.isReady = false
|
this.isReady = false
|
||||||
this.isTimedOut = false
|
this.isTimedOut = false
|
||||||
|
|
||||||
@ -33,14 +31,6 @@ class Download {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
get includeMetadata() {
|
|
||||||
return !!this.options.includeMetadata
|
|
||||||
}
|
|
||||||
|
|
||||||
get includeCover() {
|
|
||||||
return !!this.options.includeCover
|
|
||||||
}
|
|
||||||
|
|
||||||
get mimeType() {
|
get mimeType() {
|
||||||
if (this.ext === '.mp3' || this.ext === '.m4b' || this.ext === '.m4a') {
|
if (this.ext === '.mp3' || this.ext === '.m4b' || this.ext === '.m4a') {
|
||||||
return 'audio/mpeg'
|
return 'audio/mpeg'
|
||||||
@ -57,11 +47,10 @@ class Download {
|
|||||||
toJSON() {
|
toJSON() {
|
||||||
return {
|
return {
|
||||||
id: this.id,
|
id: this.id,
|
||||||
audiobookId: this.audiobookId,
|
libraryItemId: this.libraryItemId,
|
||||||
type: this.type,
|
type: this.type,
|
||||||
options: this.options,
|
|
||||||
dirpath: this.dirpath,
|
dirpath: this.dirpath,
|
||||||
fullPath: this.fullPath,
|
path: this.path,
|
||||||
ext: this.ext,
|
ext: this.ext,
|
||||||
filename: this.filename,
|
filename: this.filename,
|
||||||
size: this.size,
|
size: this.size,
|
||||||
@ -75,18 +64,16 @@ class Download {
|
|||||||
|
|
||||||
construct(download) {
|
construct(download) {
|
||||||
this.id = download.id
|
this.id = download.id
|
||||||
this.audiobookId = download.audiobookId
|
this.libraryItemId = download.libraryItemId
|
||||||
this.type = download.type
|
this.type = download.type
|
||||||
this.options = { ...download.options }
|
|
||||||
|
|
||||||
this.dirpath = download.dirpath
|
this.dirpath = download.dirpath
|
||||||
this.fullPath = download.fullPath
|
this.path = download.path
|
||||||
this.ext = download.ext
|
this.ext = download.ext
|
||||||
this.filename = download.filename
|
this.filename = download.filename
|
||||||
this.size = download.size || 0
|
this.size = download.size || 0
|
||||||
|
|
||||||
this.userId = download.userId
|
this.userId = download.userId
|
||||||
this.socket = download.socket || null
|
|
||||||
this.isReady = !!download.isReady
|
this.isReady = !!download.isReady
|
||||||
|
|
||||||
this.startedAt = download.startedAt
|
this.startedAt = download.startedAt
|
||||||
|
@ -25,12 +25,12 @@ const Series = require('../objects/entities/Series')
|
|||||||
const FileSystemController = require('../controllers/FileSystemController')
|
const FileSystemController = require('../controllers/FileSystemController')
|
||||||
|
|
||||||
class ApiRouter {
|
class ApiRouter {
|
||||||
constructor(db, auth, scanner, playbackSessionManager, downloadManager, coverManager, backupManager, watcher, cacheManager, podcastManager, emitter, clientEmitter) {
|
constructor(db, auth, scanner, playbackSessionManager, abMergeManager, coverManager, backupManager, watcher, cacheManager, podcastManager, emitter, clientEmitter) {
|
||||||
this.db = db
|
this.db = db
|
||||||
this.auth = auth
|
this.auth = auth
|
||||||
this.scanner = scanner
|
this.scanner = scanner
|
||||||
this.playbackSessionManager = playbackSessionManager
|
this.playbackSessionManager = playbackSessionManager
|
||||||
this.downloadManager = downloadManager
|
this.abMergeManager = abMergeManager
|
||||||
this.backupManager = backupManager
|
this.backupManager = backupManager
|
||||||
this.coverManager = coverManager
|
this.coverManager = coverManager
|
||||||
this.watcher = watcher
|
this.watcher = watcher
|
||||||
@ -185,7 +185,10 @@ class ApiRouter {
|
|||||||
// Misc Routes
|
// Misc Routes
|
||||||
//
|
//
|
||||||
this.router.post('/upload', MiscController.handleUpload.bind(this))
|
this.router.post('/upload', MiscController.handleUpload.bind(this))
|
||||||
this.router.get('/download/:id', MiscController.download.bind(this))
|
this.router.get('/audiobook-merge/:id', MiscController.mergeAudiobook.bind(this))
|
||||||
|
this.router.get('/download/:id', MiscController.getDownload.bind(this))
|
||||||
|
this.router.delete('/download/:id', MiscController.removeDownload.bind(this))
|
||||||
|
this.router.get('/downloads', MiscController.getDownloads.bind(this))
|
||||||
this.router.patch('/settings', MiscController.updateServerSettings.bind(this)) // Root only
|
this.router.patch('/settings', MiscController.updateServerSettings.bind(this)) // Root only
|
||||||
this.router.post('/purgecache', MiscController.purgeCache.bind(this)) // Root only
|
this.router.post('/purgecache', MiscController.purgeCache.bind(this)) // Root only
|
||||||
this.router.post('/authorize', MiscController.authorize.bind(this))
|
this.router.post('/authorize', MiscController.authorize.bind(this))
|
||||||
|
@ -41,20 +41,19 @@ async function writeConcatFile(tracks, outputPath, startTime = 0) {
|
|||||||
module.exports.writeConcatFile = writeConcatFile
|
module.exports.writeConcatFile = writeConcatFile
|
||||||
|
|
||||||
|
|
||||||
async function writeMetadataFile(audiobook, outputPath) {
|
async function writeMetadataFile(libraryItem, outputPath) {
|
||||||
var inputstrs = [
|
var inputstrs = [
|
||||||
';FFMETADATA1',
|
';FFMETADATA1',
|
||||||
`title=${audiobook.title}`,
|
`title=${libraryItem.media.metadata.title}`,
|
||||||
`artist=${audiobook.authorFL}`,
|
`artist=${libraryItem.media.metadata.authorName}`,
|
||||||
`album_artist=${audiobook.authorFL}`,
|
`album_artist=${libraryItem.media.metadata.authorName}`,
|
||||||
`date=${audiobook.book.publishedYear || ''}`,
|
`date=${libraryItem.media.metadata.publishedYear || ''}`,
|
||||||
`description=${audiobook.book.description}`,
|
`description=${libraryItem.media.metadata.description}`,
|
||||||
`genre=${audiobook.book._genres.join(';')}`,
|
`genre=${libraryItem.media.metadata.genres.join(';')}`
|
||||||
`comment=Audiobookshelf v${package.version}`
|
|
||||||
]
|
]
|
||||||
|
|
||||||
if (audiobook.chapters) {
|
if (libraryItem.media.chapters) {
|
||||||
audiobook.chapters.forEach((chap) => {
|
libraryItem.media.chapters.forEach((chap) => {
|
||||||
const chapterstrs = [
|
const chapterstrs = [
|
||||||
'[CHAPTER]',
|
'[CHAPTER]',
|
||||||
'TIMEBASE=1/1000',
|
'TIMEBASE=1/1000',
|
||||||
|
Loading…
Reference in New Issue
Block a user