2022-11-19 20:28:06 +01:00
|
|
|
const Logger = require('../Logger')
|
2023-04-29 21:57:51 +02:00
|
|
|
const ffmpegHelpers = require('../utils/ffmpegHelpers')
|
2022-11-19 20:28:06 +01:00
|
|
|
|
|
|
|
class ToolsController {
|
|
|
|
constructor() { }
|
|
|
|
|
|
|
|
// POST: api/tools/item/:id/encode-m4b
|
|
|
|
async encodeM4b(req, res) {
|
|
|
|
if (req.libraryItem.isMissing || req.libraryItem.isInvalid) {
|
|
|
|
Logger.error(`[MiscController] encodeM4b: library item not found or invalid ${req.params.id}`)
|
|
|
|
return res.status(404).send('Audiobook not found')
|
|
|
|
}
|
|
|
|
|
|
|
|
if (req.libraryItem.mediaType !== 'book') {
|
|
|
|
Logger.error(`[MiscController] encodeM4b: Invalid library item ${req.params.id}: not a book`)
|
|
|
|
return res.status(500).send('Invalid library item: not a book')
|
|
|
|
}
|
|
|
|
|
|
|
|
if (req.libraryItem.media.tracks.length <= 0) {
|
|
|
|
Logger.error(`[MiscController] encodeM4b: Invalid audiobook ${req.params.id}: no audio tracks`)
|
|
|
|
return res.status(500).send('Invalid audiobook: no audio tracks')
|
|
|
|
}
|
|
|
|
|
2022-12-20 00:13:04 +01:00
|
|
|
const options = req.query || {}
|
|
|
|
this.abMergeManager.startAudiobookMerge(req.user, req.libraryItem, options)
|
2022-11-19 20:28:06 +01:00
|
|
|
|
|
|
|
res.sendStatus(200)
|
|
|
|
}
|
|
|
|
|
|
|
|
// DELETE: api/tools/item/:id/encode-m4b
|
|
|
|
async cancelM4bEncode(req, res) {
|
|
|
|
const workerTask = this.abMergeManager.getPendingTaskByLibraryItemId(req.params.id)
|
|
|
|
if (!workerTask) return res.sendStatus(404)
|
|
|
|
|
|
|
|
this.abMergeManager.cancelEncode(workerTask.task)
|
|
|
|
|
|
|
|
res.sendStatus(200)
|
|
|
|
}
|
|
|
|
|
|
|
|
// POST: api/tools/item/:id/embed-metadata
|
|
|
|
async embedAudioFileMetadata(req, res) {
|
|
|
|
if (req.libraryItem.isMissing || !req.libraryItem.hasAudioFiles || !req.libraryItem.isBook) {
|
2023-04-02 23:13:18 +02:00
|
|
|
Logger.error(`[ToolsController] Invalid library item`)
|
2022-11-19 20:28:06 +01:00
|
|
|
return res.sendStatus(500)
|
|
|
|
}
|
|
|
|
|
2023-04-02 23:13:18 +02:00
|
|
|
if (this.audioMetadataManager.getIsLibraryItemQueuedOrProcessing(req.libraryItem.id)) {
|
|
|
|
Logger.error(`[ToolsController] Library item (${req.libraryItem.id}) is already in queue or processing`)
|
|
|
|
return res.status(500).send('Library item is already in queue or processing')
|
|
|
|
}
|
|
|
|
|
2023-01-07 22:16:52 +01:00
|
|
|
const options = {
|
|
|
|
forceEmbedChapters: req.query.forceEmbedChapters === '1',
|
|
|
|
backup: req.query.backup === '1'
|
|
|
|
}
|
|
|
|
this.audioMetadataManager.updateMetadataForItem(req.user, req.libraryItem, options)
|
2022-11-19 20:28:06 +01:00
|
|
|
res.sendStatus(200)
|
|
|
|
}
|
|
|
|
|
2023-04-02 23:13:18 +02:00
|
|
|
// POST: api/tools/batch/embed-metadata
|
|
|
|
async batchEmbedMetadata(req, res) {
|
|
|
|
const libraryItemIds = req.body.libraryItemIds || []
|
|
|
|
if (!libraryItemIds.length) {
|
|
|
|
return res.status(400).send('Invalid request payload')
|
|
|
|
}
|
|
|
|
|
|
|
|
const libraryItems = []
|
|
|
|
for (const libraryItemId of libraryItemIds) {
|
|
|
|
const libraryItem = this.db.getLibraryItem(libraryItemId)
|
|
|
|
if (!libraryItem) {
|
|
|
|
Logger.error(`[ToolsController] Batch embed metadata library item (${libraryItemId}) not found`)
|
|
|
|
return res.sendStatus(404)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check user can access this library item
|
|
|
|
if (!req.user.checkCanAccessLibraryItem(libraryItem)) {
|
|
|
|
Logger.error(`[ToolsController] Batch embed metadata library item (${libraryItemId}) not accessible to user`, req.user)
|
|
|
|
return res.sendStatus(403)
|
|
|
|
}
|
|
|
|
|
|
|
|
if (libraryItem.isMissing || !libraryItem.hasAudioFiles || !libraryItem.isBook) {
|
|
|
|
Logger.error(`[ToolsController] Batch embed invalid library item (${libraryItemId})`)
|
|
|
|
return res.sendStatus(500)
|
|
|
|
}
|
|
|
|
|
|
|
|
if (this.audioMetadataManager.getIsLibraryItemQueuedOrProcessing(libraryItemId)) {
|
|
|
|
Logger.error(`[ToolsController] Batch embed library item (${libraryItemId}) is already in queue or processing`)
|
|
|
|
return res.status(500).send('Library item is already in queue or processing')
|
|
|
|
}
|
|
|
|
|
|
|
|
libraryItems.push(libraryItem)
|
|
|
|
}
|
2022-11-19 20:28:06 +01:00
|
|
|
|
2023-04-02 23:13:18 +02:00
|
|
|
const options = {
|
|
|
|
forceEmbedChapters: req.query.forceEmbedChapters === '1',
|
|
|
|
backup: req.query.backup === '1'
|
|
|
|
}
|
|
|
|
this.audioMetadataManager.handleBatchEmbed(req.user, libraryItems, options)
|
|
|
|
res.sendStatus(200)
|
|
|
|
}
|
|
|
|
|
2023-04-29 21:57:51 +02:00
|
|
|
getAudioFileWaveform(req, res) {
|
|
|
|
let start = Number(req.query.start || 0)
|
|
|
|
let end = Number(req.query.end || 0)
|
|
|
|
if (isNaN(start) || isNaN(end) || start < 0 || end > req.libraryItem.media.duration || end <= start || end - start < 5) {
|
|
|
|
return res.status(400).send('Invalid start/end query params')
|
|
|
|
}
|
|
|
|
|
|
|
|
const paths = []
|
|
|
|
let currentTime = 0
|
|
|
|
let startOffset = 0
|
|
|
|
for (const track of req.libraryItem.media.tracks) {
|
|
|
|
currentTime += track.duration
|
|
|
|
if (currentTime > start) {
|
|
|
|
if (!paths.length) startOffset = track.startOffset
|
|
|
|
paths.push(track.metadata.path)
|
|
|
|
}
|
|
|
|
if (currentTime > end) {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
start -= startOffset
|
|
|
|
end -= startOffset
|
|
|
|
|
|
|
|
ffmpegHelpers.generateWaveform(paths, start, end, res)
|
|
|
|
}
|
|
|
|
|
2023-04-02 23:13:18 +02:00
|
|
|
middleware(req, res, next) {
|
|
|
|
if (!req.user.isAdminOrUp) {
|
|
|
|
Logger.error(`[LibraryItemController] Non-root user attempted to access tools route`, req.user)
|
2022-11-19 20:28:06 +01:00
|
|
|
return res.sendStatus(403)
|
|
|
|
}
|
|
|
|
|
2023-04-02 23:13:18 +02:00
|
|
|
if (req.params.id) {
|
|
|
|
const item = this.db.libraryItems.find(li => li.id === req.params.id)
|
|
|
|
if (!item || !item.media) return res.sendStatus(404)
|
|
|
|
|
|
|
|
// Check user can access this library item
|
|
|
|
if (!req.user.checkCanAccessLibraryItem(item)) {
|
|
|
|
return res.sendStatus(403)
|
|
|
|
}
|
|
|
|
|
|
|
|
req.libraryItem = item
|
|
|
|
}
|
|
|
|
|
2022-11-19 20:28:06 +01:00
|
|
|
next()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
module.exports = new ToolsController()
|