mirror of
https://github.com/advplyr/audiobookshelf.git
synced 2024-12-20 19:06:06 +01:00
272 lines
7.8 KiB
JavaScript
272 lines
7.8 KiB
JavaScript
const uuidv4 = require("uuid").v4
|
|
const Path = require('path')
|
|
const Logger = require('../../Logger')
|
|
const { cleanStringForSearch, areEquivalent, copyValue } = require('../../utils/index')
|
|
const AudioFile = require('../files/AudioFile')
|
|
const AudioTrack = require('../files/AudioTrack')
|
|
|
|
class PodcastEpisode {
|
|
constructor(episode) {
|
|
this.libraryItemId = null
|
|
this.podcastId = null
|
|
this.id = null
|
|
this.oldEpisodeId = null
|
|
this.index = null
|
|
|
|
this.season = null
|
|
this.episode = null
|
|
this.episodeType = null
|
|
this.title = null
|
|
this.subtitle = null
|
|
this.description = null
|
|
this.enclosure = null
|
|
this.guid = null
|
|
this.pubDate = null
|
|
this.chapters = []
|
|
|
|
this.audioFile = null
|
|
this.publishedAt = null
|
|
this.addedAt = null
|
|
this.updatedAt = null
|
|
|
|
if (episode) {
|
|
this.construct(episode)
|
|
}
|
|
}
|
|
|
|
construct(episode) {
|
|
this.libraryItemId = episode.libraryItemId
|
|
this.podcastId = episode.podcastId
|
|
this.id = episode.id
|
|
this.oldEpisodeId = episode.oldEpisodeId
|
|
this.index = episode.index
|
|
this.season = episode.season
|
|
this.episode = episode.episode
|
|
this.episodeType = episode.episodeType
|
|
this.title = episode.title
|
|
this.subtitle = episode.subtitle
|
|
this.description = episode.description
|
|
this.enclosure = episode.enclosure ? { ...episode.enclosure } : null
|
|
this.guid = episode.guid || null
|
|
this.pubDate = episode.pubDate
|
|
this.chapters = episode.chapters?.map(ch => ({ ...ch })) || []
|
|
this.audioFile = new AudioFile(episode.audioFile)
|
|
this.publishedAt = episode.publishedAt
|
|
this.addedAt = episode.addedAt
|
|
this.updatedAt = episode.updatedAt
|
|
|
|
this.audioFile.index = 1 // Only 1 audio file per episode
|
|
}
|
|
|
|
toJSON() {
|
|
return {
|
|
libraryItemId: this.libraryItemId,
|
|
podcastId: this.podcastId,
|
|
id: this.id,
|
|
oldEpisodeId: this.oldEpisodeId,
|
|
index: this.index,
|
|
season: this.season,
|
|
episode: this.episode,
|
|
episodeType: this.episodeType,
|
|
title: this.title,
|
|
subtitle: this.subtitle,
|
|
description: this.description,
|
|
enclosure: this.enclosure ? { ...this.enclosure } : null,
|
|
guid: this.guid,
|
|
pubDate: this.pubDate,
|
|
chapters: this.chapters.map(ch => ({ ...ch })),
|
|
audioFile: this.audioFile.toJSON(),
|
|
publishedAt: this.publishedAt,
|
|
addedAt: this.addedAt,
|
|
updatedAt: this.updatedAt
|
|
}
|
|
}
|
|
|
|
toJSONExpanded() {
|
|
return {
|
|
libraryItemId: this.libraryItemId,
|
|
podcastId: this.podcastId,
|
|
id: this.id,
|
|
oldEpisodeId: this.oldEpisodeId,
|
|
index: this.index,
|
|
season: this.season,
|
|
episode: this.episode,
|
|
episodeType: this.episodeType,
|
|
title: this.title,
|
|
subtitle: this.subtitle,
|
|
description: this.description,
|
|
enclosure: this.enclosure ? { ...this.enclosure } : null,
|
|
guid: this.guid,
|
|
pubDate: this.pubDate,
|
|
chapters: this.chapters.map(ch => ({ ...ch })),
|
|
audioFile: this.audioFile.toJSON(),
|
|
audioTrack: this.audioTrack.toJSON(),
|
|
publishedAt: this.publishedAt,
|
|
addedAt: this.addedAt,
|
|
updatedAt: this.updatedAt,
|
|
duration: this.duration,
|
|
size: this.size
|
|
}
|
|
}
|
|
|
|
get audioTrack() {
|
|
const audioTrack = new AudioTrack()
|
|
audioTrack.setData(this.libraryItemId, this.audioFile, 0)
|
|
return audioTrack
|
|
}
|
|
get tracks() {
|
|
return [this.audioTrack]
|
|
}
|
|
get duration() {
|
|
return this.audioFile.duration
|
|
}
|
|
get size() { return this.audioFile.metadata.size }
|
|
get enclosureUrl() {
|
|
return this.enclosure?.url || null
|
|
}
|
|
get pubYear() {
|
|
if (!this.publishedAt) return null
|
|
return new Date(this.publishedAt).getFullYear()
|
|
}
|
|
|
|
setData(data, index = 1) {
|
|
this.id = uuidv4()
|
|
this.index = index
|
|
this.title = data.title
|
|
this.subtitle = data.subtitle || ''
|
|
this.pubDate = data.pubDate || ''
|
|
this.description = data.description || ''
|
|
this.enclosure = data.enclosure ? { ...data.enclosure } : null
|
|
this.guid = data.guid || null
|
|
this.season = data.season || ''
|
|
this.episode = data.episode || ''
|
|
this.episodeType = data.episodeType || 'full'
|
|
this.publishedAt = data.publishedAt || 0
|
|
this.addedAt = Date.now()
|
|
this.updatedAt = Date.now()
|
|
}
|
|
|
|
setDataFromAudioFile(audioFile, index) {
|
|
this.id = uuidv4()
|
|
this.audioFile = audioFile
|
|
this.title = Path.basename(audioFile.metadata.filename, Path.extname(audioFile.metadata.filename))
|
|
this.index = index
|
|
|
|
this.setDataFromAudioMetaTags(audioFile.metaTags, true)
|
|
|
|
this.chapters = audioFile.chapters?.map((c) => ({ ...c }))
|
|
this.addedAt = Date.now()
|
|
this.updatedAt = Date.now()
|
|
}
|
|
|
|
update(payload) {
|
|
let hasUpdates = false
|
|
for (const key in this.toJSON()) {
|
|
let newValue = payload[key]
|
|
if (newValue === "") newValue = null
|
|
let existingValue = this[key]
|
|
if (existingValue === "") existingValue = null
|
|
|
|
if (newValue != undefined && !areEquivalent(newValue, existingValue)) {
|
|
this[key] = copyValue(newValue)
|
|
hasUpdates = true
|
|
}
|
|
}
|
|
if (hasUpdates) {
|
|
this.updatedAt = Date.now()
|
|
}
|
|
return hasUpdates
|
|
}
|
|
|
|
// Only checks container format
|
|
checkCanDirectPlay(payload) {
|
|
const supportedMimeTypes = payload.supportedMimeTypes || []
|
|
return supportedMimeTypes.includes(this.audioFile.mimeType)
|
|
}
|
|
|
|
getDirectPlayTracklist() {
|
|
return this.tracks
|
|
}
|
|
|
|
checkEqualsEnclosureUrl(url) {
|
|
if (!this.enclosure || !this.enclosure.url) return false
|
|
return this.enclosure.url == url
|
|
}
|
|
|
|
searchQuery(query) {
|
|
return cleanStringForSearch(this.title).includes(query)
|
|
}
|
|
|
|
setDataFromAudioMetaTags(audioFileMetaTags, overrideExistingDetails = false) {
|
|
if (!audioFileMetaTags) return false
|
|
|
|
const MetadataMapArray = [
|
|
{
|
|
tag: 'tagComment',
|
|
altTag: 'tagSubtitle',
|
|
key: 'description'
|
|
},
|
|
{
|
|
tag: 'tagSubtitle',
|
|
key: 'subtitle'
|
|
},
|
|
{
|
|
tag: 'tagDate',
|
|
key: 'pubDate'
|
|
},
|
|
{
|
|
tag: 'tagDisc',
|
|
key: 'season',
|
|
},
|
|
{
|
|
tag: 'tagTrack',
|
|
altTag: 'tagSeriesPart',
|
|
key: 'episode'
|
|
},
|
|
{
|
|
tag: 'tagTitle',
|
|
key: 'title'
|
|
},
|
|
{
|
|
tag: 'tagEpisodeType',
|
|
key: 'episodeType'
|
|
}
|
|
]
|
|
|
|
MetadataMapArray.forEach((mapping) => {
|
|
let value = audioFileMetaTags[mapping.tag]
|
|
let tagToUse = mapping.tag
|
|
if (!value && mapping.altTag) {
|
|
tagToUse = mapping.altTag
|
|
value = audioFileMetaTags[mapping.altTag]
|
|
}
|
|
|
|
if (value && typeof value === 'string') {
|
|
value = value.trim() // Trim whitespace
|
|
|
|
if (mapping.key === 'pubDate' && (!this.pubDate || overrideExistingDetails)) {
|
|
const pubJsDate = new Date(value)
|
|
if (pubJsDate && !isNaN(pubJsDate)) {
|
|
this.publishedAt = pubJsDate.valueOf()
|
|
this.pubDate = value
|
|
Logger.debug(`[PodcastEpisode] Mapping metadata to key ${tagToUse} => ${mapping.key}: ${this[mapping.key]}`)
|
|
} else {
|
|
Logger.warn(`[PodcastEpisode] Mapping pubDate with tag ${tagToUse} has invalid date "${value}"`)
|
|
}
|
|
} else if (mapping.key === 'episodeType' && (!this.episodeType || overrideExistingDetails)) {
|
|
if (['full', 'trailer', 'bonus'].includes(value)) {
|
|
this.episodeType = value
|
|
Logger.debug(`[PodcastEpisode] Mapping metadata to key ${tagToUse} => ${mapping.key}: ${this[mapping.key]}`)
|
|
} else {
|
|
Logger.warn(`[PodcastEpisode] Mapping episodeType with invalid value "${value}". Must be one of [full, trailer, bonus].`)
|
|
}
|
|
} else if (!this[mapping.key] || overrideExistingDetails) {
|
|
this[mapping.key] = value
|
|
Logger.debug(`[PodcastEpisode] Mapping metadata to key ${tagToUse} => ${mapping.key}: ${this[mapping.key]}`)
|
|
}
|
|
}
|
|
})
|
|
}
|
|
}
|
|
module.exports = PodcastEpisode
|