Remove series search api endpoint, update authors and series to load from db

This commit is contained in:
advplyr 2023-09-03 10:49:02 -05:00
parent 7567e91878
commit 9123dcb365
10 changed files with 79 additions and 53 deletions

View File

@ -19,8 +19,6 @@ class Database {
// TODO: below data should be loaded from the DB as needed // TODO: below data should be loaded from the DB as needed
this.libraryItems = [] this.libraryItems = []
this.settings = [] this.settings = []
this.authors = []
this.series = []
// Cached library filter data // Cached library filter data
this.libraryFilterData = {} this.libraryFilterData = {}
@ -274,12 +272,6 @@ class Database {
this.libraryItems = await this.models.libraryItem.loadAllLibraryItems() this.libraryItems = await this.models.libraryItem.loadAllLibraryItems()
Logger.info(`[Database] Loaded ${this.libraryItems.length} library items`) Logger.info(`[Database] Loaded ${this.libraryItems.length} library items`)
this.authors = await this.models.author.getOldAuthors()
Logger.info(`[Database] Loaded ${this.authors.length} authors`)
this.series = await this.models.series.getAllOldSeries()
Logger.info(`[Database] Loaded ${this.series.length} series`)
// Set if root user has been created // Set if root user has been created
this.hasRootUser = await this.models.user.getHasRootUser() this.hasRootUser = await this.models.user.getHasRootUser()
@ -456,31 +448,26 @@ class Database {
async createSeries(oldSeries) { async createSeries(oldSeries) {
if (!this.sequelize) return false if (!this.sequelize) return false
await this.models.series.createFromOld(oldSeries) await this.models.series.createFromOld(oldSeries)
this.series.push(oldSeries)
} }
async createBulkSeries(oldSeriesObjs) { async createBulkSeries(oldSeriesObjs) {
if (!this.sequelize) return false if (!this.sequelize) return false
await this.models.series.createBulkFromOld(oldSeriesObjs) await this.models.series.createBulkFromOld(oldSeriesObjs)
this.series.push(...oldSeriesObjs)
} }
async removeSeries(seriesId) { async removeSeries(seriesId) {
if (!this.sequelize) return false if (!this.sequelize) return false
await this.models.series.removeById(seriesId) await this.models.series.removeById(seriesId)
this.series = this.series.filter(se => se.id !== seriesId)
} }
async createAuthor(oldAuthor) { async createAuthor(oldAuthor) {
if (!this.sequelize) return false if (!this.sequelize) return false
await this.models.author.createFromOld(oldAuthor) await this.models.author.createFromOld(oldAuthor)
this.authors.push(oldAuthor)
} }
async createBulkAuthors(oldAuthors) { async createBulkAuthors(oldAuthors) {
if (!this.sequelize) return false if (!this.sequelize) return false
await this.models.author.createBulkFromOld(oldAuthors) await this.models.author.createBulkFromOld(oldAuthors)
this.authors.push(...oldAuthors)
} }
updateAuthor(oldAuthor) { updateAuthor(oldAuthor) {
@ -491,24 +478,17 @@ class Database {
async removeAuthor(authorId) { async removeAuthor(authorId) {
if (!this.sequelize) return false if (!this.sequelize) return false
await this.models.author.removeById(authorId) await this.models.author.removeById(authorId)
this.authors = this.authors.filter(au => au.id !== authorId)
} }
async createBulkBookAuthors(bookAuthors) { async createBulkBookAuthors(bookAuthors) {
if (!this.sequelize) return false if (!this.sequelize) return false
await this.models.bookAuthor.bulkCreate(bookAuthors) await this.models.bookAuthor.bulkCreate(bookAuthors)
this.authors.push(...bookAuthors)
} }
async removeBulkBookAuthors(authorId = null, bookId = null) { async removeBulkBookAuthors(authorId = null, bookId = null) {
if (!this.sequelize) return false if (!this.sequelize) return false
if (!authorId && !bookId) return if (!authorId && !bookId) return
await this.models.bookAuthor.removeByIds(authorId, bookId) await this.models.bookAuthor.removeByIds(authorId, bookId)
this.authors = this.authors.filter(au => {
if (authorId && au.authorId !== authorId) return true
if (bookId && au.bookId !== bookId) return true
return false
})
} }
getPlaybackSessions(where = null) { getPlaybackSessions(where = null) {

View File

@ -289,8 +289,15 @@ class Server {
for (const _user of users) { for (const _user of users) {
let hasUpdated = false let hasUpdated = false
if (_user.seriesHideFromContinueListening.length) { if (_user.seriesHideFromContinueListening.length) {
const seriesHiding = (await Database.seriesModel.findAll({
where: {
id: _user.seriesHideFromContinueListening
},
attributes: ['id'],
raw: true
})).map(se => se.id)
_user.seriesHideFromContinueListening = _user.seriesHideFromContinueListening.filter(seriesId => { _user.seriesHideFromContinueListening = _user.seriesHideFromContinueListening.filter(seriesId => {
if (!Database.series.some(se => se.id === seriesId)) { // Series removed if (!seriesHiding.includes(seriesId)) { // Series removed
hasUpdated = true hasUpdated = true
return false return false
} }

View File

@ -281,7 +281,7 @@ class MeController {
// GET: api/me/series/:id/remove-from-continue-listening // GET: api/me/series/:id/remove-from-continue-listening
async removeSeriesFromContinueListening(req, res) { async removeSeriesFromContinueListening(req, res) {
const series = Database.series.find(se => se.id === req.params.id) const series = await Database.seriesModel.getOldById(req.params.id)
if (!series) { if (!series) {
Logger.error(`[MeController] removeSeriesFromContinueListening: Series ${req.params.id} not found`) Logger.error(`[MeController] removeSeriesFromContinueListening: Series ${req.params.id} not found`)
return res.sendStatus(404) return res.sendStatus(404)
@ -297,7 +297,7 @@ class MeController {
// GET: api/me/series/:id/readd-to-continue-listening // GET: api/me/series/:id/readd-to-continue-listening
async readdSeriesFromContinueListening(req, res) { async readdSeriesFromContinueListening(req, res) {
const series = Database.series.find(se => se.id === req.params.id) const series = await Database.seriesModel.getOldById(req.params.id)
if (!series) { if (!series) {
Logger.error(`[MeController] readdSeriesFromContinueListening: Series ${req.params.id} not found`) Logger.error(`[MeController] readdSeriesFromContinueListening: Series ${req.params.id} not found`)
return res.sendStatus(404) return res.sendStatus(404)

View File

@ -9,7 +9,7 @@ class RSSFeedController {
const feeds = await this.rssFeedManager.getFeeds() const feeds = await this.rssFeedManager.getFeeds()
res.json({ res.json({
feeds: feeds.map(f => f.toJSON()), feeds: feeds.map(f => f.toJSON()),
minified: feeds.map(f => f.toJSONMinified()) minified: feeds.map(f => f.toJSONMinified())
}) })
} }
@ -88,7 +88,7 @@ class RSSFeedController {
async openRSSFeedForSeries(req, res) { async openRSSFeedForSeries(req, res) {
const options = req.body || {} const options = req.body || {}
const series = Database.series.find(se => se.id === req.params.seriesId) const series = await Database.seriesModel.getOldById(req.params.seriesId)
if (!series) return res.sendStatus(404) if (!series) return res.sendStatus(404)
// Check request body options exist // Check request body options exist

View File

@ -43,17 +43,6 @@ class SeriesController {
res.json(seriesJson) res.json(seriesJson)
} }
async search(req, res) {
var q = (req.query.q || '').toLowerCase()
if (!q) return res.json([])
var limit = (req.query.limit && !isNaN(req.query.limit)) ? Number(req.query.limit) : 25
var series = Database.series.filter(se => se.name.toLowerCase().includes(q))
series = series.slice(0, limit)
res.json({
results: series
})
}
async update(req, res) { async update(req, res) {
const hasUpdated = req.series.update(req.body) const hasUpdated = req.series.update(req.body)
if (hasUpdated) { if (hasUpdated) {
@ -64,7 +53,7 @@ class SeriesController {
} }
async middleware(req, res, next) { async middleware(req, res, next) {
const series = Database.series.find(se => se.id === req.params.id) const series = await Database.seriesModel.getOldById(req.params.id)
if (!series) return res.sendStatus(404) if (!series) return res.sendStatus(404)
/** /**

View File

@ -25,7 +25,7 @@ class RssFeedManager {
return false return false
} }
} else if (feedObj.entityType === 'series') { } else if (feedObj.entityType === 'series') {
const series = Database.series.find(s => s.id === feedObj.entityId) const series = await Database.seriesModel.getOldById(feedObj.entityId)
if (!series) { if (!series) {
Logger.error(`[RssFeedManager] Removing feed "${feedObj.id}". Series "${feedObj.entityId}" not found`) Logger.error(`[RssFeedManager] Removing feed "${feedObj.id}". Series "${feedObj.entityId}" not found`)
return false return false
@ -123,7 +123,7 @@ class RssFeedManager {
} }
} }
} else if (feed.entityType === 'series') { } else if (feed.entityType === 'series') {
const series = Database.series.find(s => s.id === feed.entityId) const series = await Database.seriesModel.getOldById(feed.entityId)
if (series) { if (series) {
const seriesJson = series.toJSON() const seriesJson = series.toJSON()

View File

@ -1,4 +1,4 @@
const { DataTypes, Model } = require('sequelize') const { DataTypes, Model, literal } = require('sequelize')
const oldAuthor = require('../objects/entities/Author') const oldAuthor = require('../objects/entities/Author')
@ -86,7 +86,7 @@ class Author extends Model {
/** /**
* Get oldAuthor by id * Get oldAuthor by id
* @param {string} authorId * @param {string} authorId
* @returns {oldAuthor} * @returns {Promise<oldAuthor>}
*/ */
static async getOldById(authorId) { static async getOldById(authorId) {
const author = await this.findByPk(authorId) const author = await this.findByPk(authorId)
@ -103,6 +103,26 @@ class Author extends Model {
return (await this.count({ where: { id: authorId } })) > 0 return (await this.count({ where: { id: authorId } })) > 0
} }
/**
* Get old author by name and libraryId. name case insensitive
* TODO: Look for authors ignoring punctuation
*
* @param {string} authorName
* @param {string} libraryId
* @returns {Promise<oldAuthor>}
*/
static async getOldByNameAndLibrary(authorName, libraryId) {
const author = (await this.findOne({
where: [
literal(`name = '${authorName}' COLLATE NOCASE`),
{
libraryId
}
]
}))?.getOldAuthor()
return author
}
/** /**
* Initialize model * Initialize model
* @param {import('../Database').sequelize} sequelize * @param {import('../Database').sequelize} sequelize

View File

@ -1,4 +1,4 @@
const { DataTypes, Model } = require('sequelize') const { DataTypes, Model, literal } = require('sequelize')
const oldSeries = require('../objects/entities/Series') const oldSeries = require('../objects/entities/Series')
@ -75,6 +75,17 @@ class Series extends Model {
}) })
} }
/**
* Get oldSeries by id
* @param {string} seriesId
* @returns {Promise<oldSeries>}
*/
static async getOldById(seriesId) {
const series = await this.findByPk(seriesId)
if (!series) return null
return series.getOldSeries()
}
/** /**
* Check if series exists * Check if series exists
* @param {string} seriesId * @param {string} seriesId
@ -84,6 +95,25 @@ class Series extends Model {
return (await this.count({ where: { id: seriesId } })) > 0 return (await this.count({ where: { id: seriesId } })) > 0
} }
/**
* Get old series by name and libraryId. name case insensitive
*
* @param {string} seriesName
* @param {string} libraryId
* @returns {Promise<oldSeries>}
*/
static async getOldByNameAndLibrary(seriesName, libraryId) {
const series = (await this.findOne({
where: [
literal(`name = '${seriesName}' COLLATE NOCASE`),
{
libraryId
}
]
}))?.getOldSeries()
return series
}
/** /**
* Initialize model * Initialize model
* @param {import('../Database').sequelize} sequelize * @param {import('../Database').sequelize} sequelize

View File

@ -1,3 +1,4 @@
const sequelize = require('sequelize')
const express = require('express') const express = require('express')
const Path = require('path') const Path = require('path')
@ -217,7 +218,6 @@ class ApiRouter {
// //
// Series Routes // Series Routes
// //
this.router.get('/series/search', SeriesController.search.bind(this))
this.router.get('/series/:id', SeriesController.middleware.bind(this), SeriesController.findOne.bind(this)) this.router.get('/series/:id', SeriesController.middleware.bind(this), SeriesController.findOne.bind(this))
this.router.patch('/series/:id', SeriesController.middleware.bind(this), SeriesController.update.bind(this)) this.router.patch('/series/:id', SeriesController.middleware.bind(this), SeriesController.update.bind(this))
@ -572,7 +572,7 @@ class ApiRouter {
} }
if (!mediaMetadata.authors[i].id || mediaMetadata.authors[i].id.startsWith('new')) { if (!mediaMetadata.authors[i].id || mediaMetadata.authors[i].id.startsWith('new')) {
let author = Database.authors.find(au => au.libraryId === libraryId && au.checkNameEquals(authorName)) let author = await Database.authorModel.getOldByNameAndLibrary(authorName, libraryId)
if (!author) { if (!author) {
author = new Author() author = new Author()
author.setData(mediaMetadata.authors[i], libraryId) author.setData(mediaMetadata.authors[i], libraryId)
@ -609,7 +609,7 @@ class ApiRouter {
} }
if (!mediaMetadata.series[i].id || mediaMetadata.series[i].id.startsWith('new')) { if (!mediaMetadata.series[i].id || mediaMetadata.series[i].id.startsWith('new')) {
let seriesItem = Database.series.find(se => se.libraryId === libraryId && se.checkNameEquals(seriesName)) let seriesItem = await Database.seriesModel.getOldByNameAndLibrary(seriesName, libraryId)
if (!seriesItem) { if (!seriesItem) {
seriesItem = new Series() seriesItem = new Series()
seriesItem.setData(mediaMetadata.series[i], libraryId) seriesItem.setData(mediaMetadata.series[i], libraryId)

View File

@ -479,8 +479,8 @@ class Scanner {
// Create or match all new authors and series // Create or match all new authors and series
if (libraryItem.media.metadata.authors.some(au => au.id.startsWith('new'))) { if (libraryItem.media.metadata.authors.some(au => au.id.startsWith('new'))) {
const newAuthors = [] const newAuthors = []
libraryItem.media.metadata.authors = libraryItem.media.metadata.authors.map((tempMinAuthor) => { libraryItem.media.metadata.authors = Promise.all(libraryItem.media.metadata.authors.map(async (tempMinAuthor) => {
let _author = Database.authors.find(au => au.libraryId === libraryItem.libraryId && au.checkNameEquals(tempMinAuthor.name)) let _author = await Database.authorModel.getOldByNameAndLibrary(tempMinAuthor.name, libraryItem.libraryId)
if (!_author) _author = newAuthors.find(au => au.libraryId === libraryItem.libraryId && au.checkNameEquals(tempMinAuthor.name)) // Check new unsaved authors if (!_author) _author = newAuthors.find(au => au.libraryId === libraryItem.libraryId && au.checkNameEquals(tempMinAuthor.name)) // Check new unsaved authors
if (!_author) { // Must create new author if (!_author) { // Must create new author
_author = new Author() _author = new Author()
@ -494,7 +494,7 @@ class Scanner {
id: _author.id, id: _author.id,
name: _author.name name: _author.name
} }
}) }))
if (newAuthors.length) { if (newAuthors.length) {
await Database.createBulkAuthors(newAuthors) await Database.createBulkAuthors(newAuthors)
SocketAuthority.emitter('authors_added', newAuthors.map(au => au.toJSON())) SocketAuthority.emitter('authors_added', newAuthors.map(au => au.toJSON()))
@ -502,8 +502,8 @@ class Scanner {
} }
if (libraryItem.media.metadata.series.some(se => se.id.startsWith('new'))) { if (libraryItem.media.metadata.series.some(se => se.id.startsWith('new'))) {
const newSeries = [] const newSeries = []
libraryItem.media.metadata.series = libraryItem.media.metadata.series.map((tempMinSeries) => { libraryItem.media.metadata.series = await Promise.all(libraryItem.media.metadata.series.map(async (tempMinSeries) => {
let _series = Database.series.find(se => se.libraryId === libraryItem.libraryId && se.checkNameEquals(tempMinSeries.name)) let _series = await Database.seriesModel.getOldByNameAndLibrary(tempMinSeries.name, libraryItem.libraryId)
if (!_series) { if (!_series) {
// Check new unsaved series // Check new unsaved series
_series = newSeries.find(se => se.libraryId === libraryItem.libraryId && se.checkNameEquals(tempMinSeries.name)) _series = newSeries.find(se => se.libraryId === libraryItem.libraryId && se.checkNameEquals(tempMinSeries.name))
@ -521,7 +521,7 @@ class Scanner {
name: _series.name, name: _series.name,
sequence: tempMinSeries.sequence sequence: tempMinSeries.sequence
} }
}) }))
if (newSeries.length) { if (newSeries.length) {
await Database.createBulkSeries(newSeries) await Database.createBulkSeries(newSeries)
SocketAuthority.emitter('multiple_series_added', newSeries.map(se => se.toJSON())) SocketAuthority.emitter('multiple_series_added', newSeries.map(se => se.toJSON()))
@ -931,7 +931,7 @@ class Scanner {
} }
const authorPayload = [] const authorPayload = []
for (const authorName of matchData.author) { for (const authorName of matchData.author) {
let author = Database.authors.find(au => au.libraryId === libraryItem.libraryId && au.checkNameEquals(authorName)) let author = await Database.authorModel.getOldByNameAndLibrary(authorName, libraryItem.libraryId)
if (!author) { if (!author) {
author = new Author() author = new Author()
author.setData({ name: authorName }, libraryItem.libraryId) author.setData({ name: authorName }, libraryItem.libraryId)
@ -950,7 +950,7 @@ class Scanner {
if (!Array.isArray(matchData.series)) matchData.series = [{ series: matchData.series, sequence: matchData.sequence }] if (!Array.isArray(matchData.series)) matchData.series = [{ series: matchData.series, sequence: matchData.sequence }]
const seriesPayload = [] const seriesPayload = []
for (const seriesMatchItem of matchData.series) { for (const seriesMatchItem of matchData.series) {
let seriesItem = Database.series.find(se => se.libraryId === libraryItem.libraryId && se.checkNameEquals(seriesMatchItem.series)) let seriesItem = await Database.seriesModel.getOldByNameAndLibrary(seriesMatchItem.series, libraryItem.libraryId)
if (!seriesItem) { if (!seriesItem) {
seriesItem = new Series() seriesItem = new Series()
seriesItem.setData({ name: seriesMatchItem.series }, libraryItem.libraryId) seriesItem.setData({ name: seriesMatchItem.series }, libraryItem.libraryId)