mirror of
https://github.com/advplyr/audiobookshelf.git
synced 2025-01-18 03:48:55 +01:00
440 lines
13 KiB
JavaScript
440 lines
13 KiB
JavaScript
const Logger = require('../../Logger')
|
|
const { areEquivalent, copyValue, cleanStringForSearch, getTitleIgnorePrefix, getTitlePrefixAtEnd } = require('../../utils/index')
|
|
const parseNameString = require('../../utils/parsers/parseNameString')
|
|
class BookMetadata {
|
|
constructor(metadata) {
|
|
this.title = null
|
|
this.subtitle = null
|
|
this.authors = []
|
|
this.narrators = [] // Array of strings
|
|
this.series = []
|
|
this.genres = [] // Array of strings
|
|
this.publishedYear = null
|
|
this.publishedDate = null
|
|
this.publisher = null
|
|
this.description = null
|
|
this.isbn = null
|
|
this.asin = null
|
|
this.language = null
|
|
this.explicit = false
|
|
this.abridged = false
|
|
|
|
if (metadata) {
|
|
this.construct(metadata)
|
|
}
|
|
}
|
|
|
|
construct(metadata) {
|
|
this.title = metadata.title
|
|
this.subtitle = metadata.subtitle
|
|
this.authors = (metadata.authors?.map) ? metadata.authors.map(a => ({ ...a })) : []
|
|
this.narrators = metadata.narrators ? [...metadata.narrators].filter(n => n) : []
|
|
this.series = (metadata.series?.map) ? metadata.series.map(s => ({ ...s })) : []
|
|
this.genres = metadata.genres ? [...metadata.genres] : []
|
|
this.publishedYear = metadata.publishedYear || null
|
|
this.publishedDate = metadata.publishedDate || null
|
|
this.publisher = metadata.publisher
|
|
this.description = metadata.description
|
|
this.isbn = metadata.isbn
|
|
this.asin = metadata.asin
|
|
this.language = metadata.language
|
|
this.explicit = !!metadata.explicit
|
|
this.abridged = !!metadata.abridged
|
|
}
|
|
|
|
toJSON() {
|
|
return {
|
|
title: this.title,
|
|
subtitle: this.subtitle,
|
|
authors: this.authors.map(a => ({ ...a })), // Author JSONMinimal with name and id
|
|
narrators: [...this.narrators],
|
|
series: this.series.map(s => ({ ...s })), // Series JSONMinimal with name, id and sequence
|
|
genres: [...this.genres],
|
|
publishedYear: this.publishedYear,
|
|
publishedDate: this.publishedDate,
|
|
publisher: this.publisher,
|
|
description: this.description,
|
|
isbn: this.isbn,
|
|
asin: this.asin,
|
|
language: this.language,
|
|
explicit: this.explicit,
|
|
abridged: this.abridged
|
|
}
|
|
}
|
|
|
|
toJSONMinified() {
|
|
return {
|
|
title: this.title,
|
|
titleIgnorePrefix: this.titlePrefixAtEnd,
|
|
subtitle: this.subtitle,
|
|
authorName: this.authorName,
|
|
authorNameLF: this.authorNameLF,
|
|
narratorName: this.narratorName,
|
|
seriesName: this.seriesName,
|
|
genres: [...this.genres],
|
|
publishedYear: this.publishedYear,
|
|
publishedDate: this.publishedDate,
|
|
publisher: this.publisher,
|
|
description: this.description,
|
|
isbn: this.isbn,
|
|
asin: this.asin,
|
|
language: this.language,
|
|
explicit: this.explicit,
|
|
abridged: this.abridged
|
|
}
|
|
}
|
|
|
|
toJSONExpanded() {
|
|
return {
|
|
title: this.title,
|
|
titleIgnorePrefix: this.titlePrefixAtEnd,
|
|
subtitle: this.subtitle,
|
|
authors: this.authors.map(a => ({ ...a })), // Author JSONMinimal with name and id
|
|
narrators: [...this.narrators],
|
|
series: this.series.map(s => ({ ...s })),
|
|
genres: [...this.genres],
|
|
publishedYear: this.publishedYear,
|
|
publishedDate: this.publishedDate,
|
|
publisher: this.publisher,
|
|
description: this.description,
|
|
isbn: this.isbn,
|
|
asin: this.asin,
|
|
language: this.language,
|
|
explicit: this.explicit,
|
|
authorName: this.authorName,
|
|
authorNameLF: this.authorNameLF,
|
|
narratorName: this.narratorName,
|
|
seriesName: this.seriesName,
|
|
abridged: this.abridged
|
|
}
|
|
}
|
|
|
|
toJSONForMetadataFile() {
|
|
const json = this.toJSON()
|
|
json.authors = json.authors.map(au => au.name)
|
|
json.series = json.series.map(se => {
|
|
if (!se.sequence) return se.name
|
|
return `${se.name} #${se.sequence}`
|
|
})
|
|
return json
|
|
}
|
|
|
|
clone() {
|
|
return new BookMetadata(this.toJSON())
|
|
}
|
|
|
|
get titleIgnorePrefix() {
|
|
return getTitleIgnorePrefix(this.title)
|
|
}
|
|
get titlePrefixAtEnd() {
|
|
return getTitlePrefixAtEnd(this.title)
|
|
}
|
|
get authorName() {
|
|
if (!this.authors.length) return ''
|
|
return this.authors.map(au => au.name).join(', ')
|
|
}
|
|
get authorNameLF() { // Last, First
|
|
if (!this.authors.length) return ''
|
|
return this.authors.map(au => parseNameString.nameToLastFirst(au.name)).join(', ')
|
|
}
|
|
get seriesName() {
|
|
if (!this.series.length) return ''
|
|
return this.series.map(se => {
|
|
if (!se.sequence) return se.name
|
|
return `${se.name} #${se.sequence}`
|
|
}).join(', ')
|
|
}
|
|
get seriesNameIgnorePrefix() {
|
|
if (!this.series.length) return ''
|
|
return this.series.map(se => {
|
|
if (!se.sequence) return getTitleIgnorePrefix(se.name)
|
|
return `${getTitleIgnorePrefix(se.name)} #${se.sequence}`
|
|
}).join(', ')
|
|
}
|
|
get seriesNamePrefixAtEnd() {
|
|
if (!this.series.length) return ''
|
|
return this.series.map(se => {
|
|
if (!se.sequence) return getTitlePrefixAtEnd(se.name)
|
|
return `${getTitlePrefixAtEnd(se.name)} #${se.sequence}`
|
|
}).join(', ')
|
|
}
|
|
get firstSeriesName() {
|
|
if (!this.series.length) return ''
|
|
return this.series[0].name
|
|
}
|
|
get firstSeriesSequence() {
|
|
if (!this.series.length) return ''
|
|
return this.series[0].sequence
|
|
}
|
|
get narratorName() {
|
|
return this.narrators.join(', ')
|
|
}
|
|
get coverSearchQuery() {
|
|
if (!this.authorName) return this.title
|
|
return this.title + '&' + this.authorName
|
|
}
|
|
|
|
hasAuthor(id) {
|
|
return !!this.authors.find(au => au.id == id)
|
|
}
|
|
hasSeries(seriesId) {
|
|
return !!this.series.find(se => se.id == seriesId)
|
|
}
|
|
hasNarrator(narratorName) {
|
|
return this.narrators.includes(narratorName)
|
|
}
|
|
getSeries(seriesId) {
|
|
return this.series.find(se => se.id == seriesId)
|
|
}
|
|
getFirstSeries() {
|
|
return this.series.length ? this.series[0] : null
|
|
}
|
|
getSeriesSequence(seriesId) {
|
|
const series = this.series.find(se => se.id == seriesId)
|
|
if (!series) return null
|
|
return series.sequence || ''
|
|
}
|
|
getSeriesSortTitle(series) {
|
|
if (!series) return ''
|
|
if (!series.sequence) return series.name
|
|
return `${series.name} #${series.sequence}`
|
|
}
|
|
|
|
update(payload) {
|
|
const json = this.toJSON()
|
|
let hasUpdates = false
|
|
|
|
for (const key in json) {
|
|
if (payload[key] !== undefined) {
|
|
if (!areEquivalent(payload[key], json[key])) {
|
|
this[key] = copyValue(payload[key])
|
|
Logger.debug('[BookMetadata] Key updated', key, this[key])
|
|
hasUpdates = true
|
|
}
|
|
}
|
|
}
|
|
return hasUpdates
|
|
}
|
|
|
|
// Updates author name
|
|
updateAuthor(updatedAuthor) {
|
|
const author = this.authors.find(au => au.id === updatedAuthor.id)
|
|
if (!author || author.name == updatedAuthor.name) return false
|
|
author.name = updatedAuthor.name
|
|
return true
|
|
}
|
|
|
|
replaceAuthor(oldAuthor, newAuthor) {
|
|
this.authors = this.authors.filter(au => au.id !== oldAuthor.id) // Remove old author
|
|
this.authors.push({
|
|
id: newAuthor.id,
|
|
name: newAuthor.name
|
|
})
|
|
}
|
|
|
|
/**
|
|
* Update narrator name if narrator is in book
|
|
* @param {String} oldNarratorName - Narrator name to get updated
|
|
* @param {String} newNarratorName - Updated narrator name
|
|
* @return {Boolean} True if narrator was updated
|
|
*/
|
|
updateNarrator(oldNarratorName, newNarratorName) {
|
|
if (!this.hasNarrator(oldNarratorName)) return false
|
|
this.narrators = this.narrators.filter(n => n !== oldNarratorName)
|
|
if (newNarratorName && !this.hasNarrator(newNarratorName)) {
|
|
this.narrators.push(newNarratorName)
|
|
}
|
|
return true
|
|
}
|
|
|
|
/**
|
|
* Remove narrator name if narrator is in book
|
|
* @param {String} narratorName - Narrator name to remove
|
|
* @return {Boolean} True if narrator was updated
|
|
*/
|
|
removeNarrator(narratorName) {
|
|
if (!this.hasNarrator(narratorName)) return false
|
|
this.narrators = this.narrators.filter(n => n !== narratorName)
|
|
return true
|
|
}
|
|
|
|
setData(scanMediaData = {}) {
|
|
this.title = scanMediaData.title || null
|
|
this.subtitle = scanMediaData.subtitle || null
|
|
this.narrators = this.parseNarratorsTag(scanMediaData.narrators)
|
|
this.publishedYear = scanMediaData.publishedYear || null
|
|
this.description = scanMediaData.description || null
|
|
this.isbn = scanMediaData.isbn || null
|
|
this.asin = scanMediaData.asin || null
|
|
this.language = scanMediaData.language || null
|
|
this.genres = []
|
|
this.explicit = !!scanMediaData.explicit
|
|
|
|
if (scanMediaData.author) {
|
|
this.authors = this.parseAuthorsTag(scanMediaData.author)
|
|
}
|
|
if (scanMediaData.series) {
|
|
this.series = this.parseSeriesTag(scanMediaData.series, scanMediaData.sequence)
|
|
}
|
|
}
|
|
|
|
setDataFromAudioMetaTags(audioFileMetaTags, overrideExistingDetails = false) {
|
|
const MetadataMapArray = [
|
|
{
|
|
tag: 'tagComposer',
|
|
key: 'narrators'
|
|
},
|
|
{
|
|
tag: 'tagDescription',
|
|
altTag: 'tagComment',
|
|
key: 'description'
|
|
},
|
|
{
|
|
tag: 'tagPublisher',
|
|
key: 'publisher'
|
|
},
|
|
{
|
|
tag: 'tagDate',
|
|
key: 'publishedYear'
|
|
},
|
|
{
|
|
tag: 'tagSubtitle',
|
|
key: 'subtitle'
|
|
},
|
|
{
|
|
tag: 'tagAlbum',
|
|
altTag: 'tagTitle',
|
|
key: 'title',
|
|
},
|
|
{
|
|
tag: 'tagArtist',
|
|
altTag: 'tagAlbumArtist',
|
|
key: 'authors'
|
|
},
|
|
{
|
|
tag: 'tagGenre',
|
|
key: 'genres'
|
|
},
|
|
{
|
|
tag: 'tagSeries',
|
|
key: 'series'
|
|
},
|
|
{
|
|
tag: 'tagIsbn',
|
|
key: 'isbn'
|
|
},
|
|
{
|
|
tag: 'tagLanguage',
|
|
key: 'language'
|
|
},
|
|
{
|
|
tag: 'tagASIN',
|
|
key: 'asin'
|
|
},
|
|
{
|
|
tag: 'tagOverdriveMediaMarker',
|
|
key: 'overdriveMediaMarker'
|
|
}
|
|
]
|
|
|
|
const updatePayload = {}
|
|
|
|
// Metadata is only mapped to the book if it is empty
|
|
MetadataMapArray.forEach((mapping) => {
|
|
let value = audioFileMetaTags[mapping.tag]
|
|
// let tagToUse = mapping.tag
|
|
if (!value && mapping.altTag) {
|
|
value = audioFileMetaTags[mapping.altTag]
|
|
// tagToUse = mapping.altTag
|
|
}
|
|
|
|
if (value && typeof value === 'string') {
|
|
value = value.trim() // Trim whitespace
|
|
|
|
if (mapping.key === 'narrators' && (!this.narrators.length || overrideExistingDetails)) {
|
|
updatePayload.narrators = this.parseNarratorsTag(value)
|
|
} else if (mapping.key === 'authors' && (!this.authors.length || overrideExistingDetails)) {
|
|
updatePayload.authors = this.parseAuthorsTag(value)
|
|
} else if (mapping.key === 'genres' && (!this.genres.length || overrideExistingDetails)) {
|
|
updatePayload.genres = this.parseGenresTag(value)
|
|
} else if (mapping.key === 'series' && (!this.series.length || overrideExistingDetails)) {
|
|
const sequenceTag = audioFileMetaTags.tagSeriesPart || null
|
|
updatePayload.series = this.parseSeriesTag(value, sequenceTag)
|
|
} else if (!this[mapping.key] || overrideExistingDetails) {
|
|
updatePayload[mapping.key] = value
|
|
// Logger.debug(`[Book] Mapping metadata to key ${tagToUse} => ${mapping.key}: ${updatePayload[mapping.key]}`)
|
|
}
|
|
}
|
|
})
|
|
|
|
if (Object.keys(updatePayload).length) {
|
|
return this.update(updatePayload)
|
|
}
|
|
return false
|
|
}
|
|
|
|
// Returns array of names in First Last format
|
|
parseNarratorsTag(narratorsTag) {
|
|
const parsed = parseNameString.parse(narratorsTag)
|
|
return parsed ? parsed.names : []
|
|
}
|
|
|
|
// Return array of authors minified with placeholder id
|
|
parseAuthorsTag(authorsTag) {
|
|
const parsed = parseNameString.parse(authorsTag)
|
|
if (!parsed) return []
|
|
return (parsed.names || []).map((au) => {
|
|
const findAuthor = this.authors.find(_au => _au.name == au)
|
|
|
|
return {
|
|
id: findAuthor?.id || `new-${Math.floor(Math.random() * 1000000)}`,
|
|
name: au
|
|
}
|
|
})
|
|
}
|
|
|
|
parseGenresTag(genreTag) {
|
|
if (!genreTag || !genreTag.length) return []
|
|
const separators = ['/', '//', ';']
|
|
for (let i = 0; i < separators.length; i++) {
|
|
if (genreTag.includes(separators[i])) {
|
|
return genreTag.split(separators[i]).map(genre => genre.trim()).filter(g => !!g)
|
|
}
|
|
}
|
|
return [genreTag]
|
|
}
|
|
|
|
// Return array with series with placeholder id
|
|
parseSeriesTag(seriesTag, sequenceTag) {
|
|
if (!seriesTag) return []
|
|
return [{
|
|
id: `new-${Math.floor(Math.random() * 1000000)}`,
|
|
name: seriesTag,
|
|
sequence: sequenceTag || ''
|
|
}]
|
|
}
|
|
|
|
searchSeries(query) {
|
|
return this.series.filter(se => cleanStringForSearch(se.name).includes(query))
|
|
}
|
|
searchAuthors(query) {
|
|
return this.authors.filter(au => cleanStringForSearch(au.name).includes(query))
|
|
}
|
|
searchNarrators(query) {
|
|
return this.narrators.filter(n => cleanStringForSearch(n).includes(query))
|
|
}
|
|
searchQuery(query) { // Returns key if match is found
|
|
const keysToCheck = ['title', 'asin', 'isbn', 'subtitle']
|
|
for (const key of keysToCheck) {
|
|
if (this[key] && cleanStringForSearch(String(this[key])).includes(query)) {
|
|
return {
|
|
matchKey: key,
|
|
matchText: this[key]
|
|
}
|
|
}
|
|
}
|
|
return null
|
|
}
|
|
}
|
|
module.exports = BookMetadata
|