const { sort, createNewSortInstance } = require('fast-sort') const Logger = require('../Logger') const naturalSort = createNewSortInstance({ comparer: new Intl.Collator(undefined, { numeric: true, sensitivity: 'base' }).compare }) module.exports = { decode(text) { return Buffer.from(decodeURIComponent(text), 'base64').toString() }, getFilteredLibraryItems(libraryItems, filterBy, user) { var filtered = libraryItems var searchGroups = ['genres', 'tags', 'series', 'authors', 'progress', 'narrators', 'missing', 'languages'] var group = searchGroups.find(_group => filterBy.startsWith(_group + '.')) if (group) { var filterVal = filterBy.replace(`${group}.`, '') var filter = this.decode(filterVal) if (group === 'genres') filtered = filtered.filter(li => li.media.metadata && li.media.metadata.genres.includes(filter)) else if (group === 'tags') filtered = filtered.filter(li => li.media.tags.includes(filter)) else if (group === 'series') { if (filter === 'No Series') filtered = filtered.filter(li => li.media.metadata && !li.media.metadata.series.length) else { filtered = filtered.filter(li => li.media.metadata && li.media.metadata.hasSeries(filter)) } } else if (group === 'authors') filtered = filtered.filter(li => li.media.metadata && li.media.metadata.hasAuthor(filter)) else if (group === 'narrators') filtered = filtered.filter(li => li.media.metadata && li.media.metadata.hasNarrator(filter)) else if (group === 'progress') { filtered = filtered.filter(li => { var itemProgress = user.getMediaProgress(li.id) if (filter === 'Finished' && (itemProgress && itemProgress.isFinished)) return true if (filter === 'Not Started' && !itemProgress) return true if (filter === 'In Progress' && (itemProgress && itemProgress.inProgress)) return true return false }) } else if (group == 'missing') { filtered = filtered.filter(li => { if (filter === 'ASIN' && li.media.metadata.asin === null) return true; if (filter === 'ISBN' && li.media.metadata.isbn === null) return true; if (filter === 'Subtitle' && li.media.metadata.subtitle === null) return true; if (filter === 'Author' && li.media.metadata.authors.length === 0) return true; if (filter === 'Publish Year' && li.media.metadata.publishedYear === null) return true; if (filter === 'Series' && li.media.metadata.series.length === 0) return true; if (filter === 'Volume Number' && (li.media.metadata.series.length === 0 || li.media.metadata.series[0].sequence === null)) return true; if (filter === 'Description' && li.media.metadata.description === null) return true; if (filter === 'Genres' && li.media.metadata.genres.length === 0) return true; if (filter === 'Tags' && li.media.tags.length === 0) return true; if (filter === 'Narrator' && li.media.metadata.narrators.length === 0) return true; if (filter === 'Publisher' && li.media.metadata.publisher === null) return true; if (filter === 'Language' && li.media.metadata.language === null) return true; }) } else if (group === 'languages') { filtered = filtered.filter(li => li.media.metadata && li.media.metadata.language === filter) } } else if (filterBy === 'issues') { filtered = filtered.filter(ab => { // TODO: Update filter for issues return ab.isMissing || ab.isInvalid // return ab.numMissingParts || ab.numInvalidParts || ab.isMissing || ab.isInvalid }) } return filtered }, getDistinctFilterDataNew(libraryItems) { var data = { authors: [], genres: [], tags: [], series: [], narrators: [], languages: [] } libraryItems.forEach((li) => { var mediaMetadata = li.media.metadata if (mediaMetadata.authors && mediaMetadata.authors.length) { mediaMetadata.authors.forEach((author) => { if (author && !data.authors.find(au => au.id === author.id)) data.authors.push({ id: author.id, name: author.name }) }) } if (mediaMetadata.series && mediaMetadata.series.length) { mediaMetadata.series.forEach((series) => { if (series && !data.series.find(se => se.id === series.id)) data.series.push({ id: series.id, name: series.name }) }) } if (mediaMetadata.genres.length) { mediaMetadata.genres.forEach((genre) => { if (genre && !data.genres.includes(genre)) data.genres.push(genre) }) } if (li.media.tags.length) { li.media.tags.forEach((tag) => { if (tag && !data.tags.includes(tag)) data.tags.push(tag) }) } if (mediaMetadata.narrators && mediaMetadata.narrators.length) { mediaMetadata.narrators.forEach((narrator) => { if (narrator && !data.narrators.includes(narrator)) data.narrators.push(narrator) }) } if (mediaMetadata.language && !data.languages.includes(mediaMetadata.language)) data.languages.push(mediaMetadata.language) }) data.authors = naturalSort(data.authors).asc() data.genres = naturalSort(data.genres).asc() data.tags = naturalSort(data.tags).asc() data.series = naturalSort(data.series).asc() data.narrators = naturalSort(data.narrators).asc() data.languages = naturalSort(data.languages).asc() return data }, getSeriesFromBooks(books, minified = false) { var _series = {} books.forEach((libraryItem) => { var bookSeries = libraryItem.media.metadata.series || [] bookSeries.forEach((series) => { var abJson = minified ? libraryItem.toJSONMinified() : libraryItem.toJSONExpanded() abJson.sequence = series.sequence if (!_series[series.id]) { _series[series.id] = { id: series.id, name: series.name, type: 'series', books: [abJson] } } else { _series[series.id].books.push(abJson) } }) }) return Object.values(_series).map((series) => { series.books = naturalSort(series.books).asc(li => li.sequence) return series }) }, getSeriesWithProgressFromBooks(user, books) { return [] // var _series = {} // books.forEach((audiobook) => { // if (audiobook.book.series) { // var bookWithUserAb = { userAudiobook: user.getMediaProgress(audiobook.id), book: audiobook } // if (!_series[audiobook.book.series]) { // _series[audiobook.book.series] = { // id: audiobook.book.series, // name: audiobook.book.series, // type: 'series', // books: [bookWithUserAb] // } // } else { // _series[audiobook.book.series].books.push(bookWithUserAb) // } // } // }) // return Object.values(_series).map((series) => { // series.books = naturalSort(series.books).asc(ab => ab.book.book.volumeNumber) // return series // }).filter((series) => series.books.some((book) => book.userAudiobook && book.userAudiobook.isRead)) }, sortSeriesBooks(books, seriesId, minified = false) { return naturalSort(books).asc(li => { if (!li.media.metadata.series) return null var series = li.media.metadata.series.find(se => se.id === seriesId) if (!series) return null return series.sequence }).map(li => { if (minified) return li.toJSONMinified() return li.toJSONExpanded() }) }, getMediaProgressWithItems(user, libraryItems) { var mediaProgress = [] libraryItems.forEach(li => { var itemProgress = user.getAllMediaProgressForLibraryItem(li.id).map(mp => { var episode = null if (mp.episodeId) { episode = li.media.getEpisode(mp.episodeId) if (!episode) { // Episode not found for library item return null } } return { userProgress: mp.toJSON(), libraryItem: li, episode } }).filter(mp => !!mp) mediaProgress = mediaProgress.concat(itemProgress) }) return mediaProgress }, getItemsMostRecentlyListened(itemsWithUserProgress, limit, minified = false) { var itemsInProgress = itemsWithUserProgress.filter((data) => data.userProgress && data.userProgress.progress > 0 && !data.userProgress.isFinished) itemsInProgress.sort((a, b) => { return b.userProgress.lastUpdate - a.userProgress.lastUpdate }) return itemsInProgress.map(b => { var libjson = minified ? b.libraryItem.toJSONMinified() : b.libraryItem.toJSONExpanded() if (b.episode) { libjson.recentEpisode = b.episode } return libjson }).slice(0, limit) }, getBooksNextInSeries(seriesWithUserAb, limit, minified = false) { var incompleteSeires = seriesWithUserAb.filter((series) => series.books.some((book) => !book.userAudiobook || (!book.userAudiobook.isRead && book.userAudiobook.progress == 0))) var booksNextInSeries = [] incompleteSeires.forEach((series) => { var dateLastRead = series.books.filter((data) => data.userAudiobook && data.userAudiobook.isRead).sort((a, b) => { return b.userAudiobook.finishedAt - a.userAudiobook.finishedAt })[0].userAudiobook.finishedAt var nextUnreadBook = series.books.filter((data) => !data.userAudiobook || (!data.userAudiobook.isRead && data.userAudiobook.progress == 0))[0] nextUnreadBook.DateLastReadSeries = dateLastRead booksNextInSeries.push(nextUnreadBook) }) return booksNextInSeries.sort((a, b) => { return b.DateLastReadSeries - a.DateLastReadSeries }).map(b => minified ? b.book.toJSONMinified() : b.book.toJSONExpanded()).slice(0, limit) }, getItemsMostRecentlyFinished(itemsWithUserProgress, limit, minified = false) { var itemsFinished = itemsWithUserProgress.filter((data) => data.userProgress && data.userProgress.isFinished) itemsFinished.sort((a, b) => { return b.userProgress.finishedAt - a.userProgress.finishedAt }) return itemsFinished.map(i => { var libjson = minified ? i.libraryItem.toJSONMinified() : i.libraryItem.toJSONExpanded() if (i.episode) { libjson.recentEpisode = i.episode } return libjson }).slice(0, limit) }, getItemsMostRecentlyAdded(libraryItems, limit, minified = false) { var itemsSortedByAddedAt = sort(libraryItems).desc(li => li.addedAt) return itemsSortedByAddedAt.map(b => minified ? b.toJSONMinified() : b.toJSONExpanded()).slice(0, limit) }, getEpisodesRecentlyAdded(libraryItems, limit, minified = false) { var libraryItemsWithEpisode = [] libraryItems.forEach((li) => { if (li.mediaType !== 'podcast' || !li.media.hasMediaEntities) return var libjson = minified ? li.toJSONMinified() : li.toJSONExpanded() var episodes = sort(li.media.episodes || []).desc(ep => ep.addedAt) episodes.forEach((ep) => { var lie = { ...libjson } lie.recentEpisode = ep libraryItemsWithEpisode.push(lie) }) }) libraryItemsWithEpisode = sort(libraryItemsWithEpisode).desc(lie => lie.recentEpisode.addedAt) return libraryItemsWithEpisode.slice(0, limit) }, getSeriesMostRecentlyAdded(series, limit) { var seriesSortedByAddedAt = sort(series).desc(_series => { var booksSortedByMostRecent = sort(_series.books).desc(b => b.addedAt) return booksSortedByMostRecent[0].addedAt }) return seriesSortedByAddedAt.slice(0, limit) }, getGenresWithCount(libraryItems) { var genresMap = {} libraryItems.forEach((li) => { var genres = li.media.metadata.genres || [] genres.forEach((genre) => { if (genresMap[genre]) genresMap[genre].count++ else genresMap[genre] = { genre, count: 1 } }) }) return Object.values(genresMap).sort((a, b) => b.count - a.count) }, getAuthorsWithCount(libraryItems) { var authorsMap = {} libraryItems.forEach((li) => { var authors = li.media.metadata.authors || [] authors.forEach((author) => { if (authorsMap[author.id]) authorsMap[author.id].count++ else authorsMap[author.id] = { id: author.id, name: author.name, count: 1 } }) }) return Object.values(authorsMap).sort((a, b) => b.count - a.count) }, getItemDurationStats(libraryItems) { var sorted = sort(libraryItems).desc(li => li.media.duration) var top10 = sorted.slice(0, 10).map(li => ({ id: li.id, title: li.media.metadata.title, duration: li.media.duration })).filter(i => i.duration > 0) var totalDuration = 0 var numAudioTracks = 0 libraryItems.forEach((li) => { totalDuration += li.media.duration numAudioTracks += li.media.numTracks }) return { totalDuration, numAudioTracks, longestItems: top10 } }, getLibraryItemsTotalSize(libraryItems) { var totalSize = 0 libraryItems.forEach((li) => { totalSize += li.media.size }) return totalSize }, collapseBookSeries(libraryItems) { var seriesObjects = this.getSeriesFromBooks(libraryItems, true) var seriesToUse = {} var libraryItemIdsToHide = [] seriesObjects.forEach((series) => { series.firstBook = series.books.find(b => !seriesToUse[b.id]) // Find first book not already used if (series.firstBook) { seriesToUse[series.firstBook.id] = series libraryItemIdsToHide = libraryItemIdsToHide.concat(series.books.filter(b => !seriesToUse[b.id]).map(b => b.id)) } }) return libraryItems.map((li) => { if (li.mediaType != 'book') return var libraryItemJson = li.toJSONMinified() if (libraryItemIdsToHide.includes(li.id)) { return null } if (seriesToUse[li.id]) { libraryItemJson.collapsedSeries = { id: seriesToUse[li.id].id, name: seriesToUse[li.id].name, numBooks: seriesToUse[li.id].books.length } } return libraryItemJson }).filter(li => li) } }